Class: Spring::ApplicationManager

Inherits:
Object
  • Object
show all
Defined in:
lib/spring/application_manager.rb

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(app_env, spring_env) ⇒ ApplicationManager

Returns a new instance of ApplicationManager.



5
6
7
8
9
10
11
# File 'lib/spring/application_manager.rb', line 5

def initialize(app_env, spring_env)
  @app_env    = app_env
  @spring_env = spring_env
  @mutex      = Mutex.new
  @state      = :running
  @pid        = nil
end

Instance Attribute Details

#app_envObject (readonly)

Returns the value of attribute app_env.



3
4
5
# File 'lib/spring/application_manager.rb', line 3

def app_env
  @app_env
end

#childObject (readonly)

Returns the value of attribute child.



3
4
5
# File 'lib/spring/application_manager.rb', line 3

def child
  @child
end

#pidObject (readonly)

Returns the value of attribute pid.



3
4
5
# File 'lib/spring/application_manager.rb', line 3

def pid
  @pid
end

#spring_envObject (readonly)

Returns the value of attribute spring_env.



3
4
5
# File 'lib/spring/application_manager.rb', line 3

def spring_env
  @spring_env
end

#statusObject (readonly)

Returns the value of attribute status.



3
4
5
# File 'lib/spring/application_manager.rb', line 3

def status
  @status
end

Instance Method Details

#alive?Boolean

Returns:

  • (Boolean)


35
36
37
# File 'lib/spring/application_manager.rb', line 35

def alive?
  @pid
end

#log(message) ⇒ Object



13
14
15
# File 'lib/spring/application_manager.rb', line 13

def log(message)
  spring_env.log "[application_manager:#{app_env}] #{message}"
end

#restartObject



30
31
32
33
# File 'lib/spring/application_manager.rb', line 30

def restart
  return if @state == :stopping
  start_child(true)
end

#run(client) ⇒ Object

Returns the pid of the process running the command, or nil if the application process died.



60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
# File 'lib/spring/application_manager.rb', line 60

def run(client)
  with_child do
    child.send_io client
    child.gets or raise Errno::EPIPE
  end

  pid = child.gets.to_i

  unless pid.zero?
    log "got worker pid #{pid}"
    pid
  end
rescue Errno::ECONNRESET, Errno::EPIPE => e
  log "#{e} while reading from child; returning no pid"
  nil
ensure
  client.close
end

#startObject



26
27
28
# File 'lib/spring/application_manager.rb', line 26

def start
  start_child
end

#stopObject



79
80
81
82
83
84
85
86
87
88
89
# File 'lib/spring/application_manager.rb', line 79

def stop
  log "stopping"
  @state = :stopping

  if pid
    Process.kill('TERM', pid)
    Process.wait(pid)
  end
rescue Errno::ESRCH, Errno::ECHILD
  # Don't care
end

#synchronizeObject

We’re not using @mutex.synchronize to avoid the weird “<internal:prelude>:10” line which messes with backtraces in e.g. rspec



19
20
21
22
23
24
# File 'lib/spring/application_manager.rb', line 19

def synchronize
  @mutex.lock
  yield
ensure
  @mutex.unlock
end

#with_childObject



39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
# File 'lib/spring/application_manager.rb', line 39

def with_child
  synchronize do
    if alive?
      begin
        yield
      rescue Errno::ECONNRESET, Errno::EPIPE
        # The child has died but has not been collected by the wait thread yet,
        # so start a new child and try again.
        log "child dead; starting"
        start
        yield
      end
    else
      log "child not running; starting"
      start
      yield
    end
  end
end