erails 2.1.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (193) hide show
  1. data/CHANGELOG +3 -0
  2. data/MIT-LICENSE +20 -0
  3. data/README +309 -0
  4. data/Rakefile +339 -0
  5. data/bin/about +4 -0
  6. data/bin/console +3 -0
  7. data/bin/dbconsole +3 -0
  8. data/bin/destroy +3 -0
  9. data/bin/erails +19 -0
  10. data/bin/generate +3 -0
  11. data/bin/performance/benchmarker +3 -0
  12. data/bin/performance/profiler +3 -0
  13. data/bin/performance/request +3 -0
  14. data/bin/plugin +3 -0
  15. data/bin/process/inspector +3 -0
  16. data/bin/process/reaper +3 -0
  17. data/bin/process/spawner +3 -0
  18. data/bin/runner +3 -0
  19. data/bin/server +3 -0
  20. data/builtin/rails_info/rails/info.rb +125 -0
  21. data/builtin/rails_info/rails/info_controller.rb +9 -0
  22. data/builtin/rails_info/rails/info_helper.rb +2 -0
  23. data/builtin/rails_info/rails_info_controller.rb +2 -0
  24. data/configs/apache.conf +40 -0
  25. data/configs/databases/frontbase.yml +28 -0
  26. data/configs/databases/mysql.yml +54 -0
  27. data/configs/databases/oracle.yml +39 -0
  28. data/configs/databases/postgresql.yml +48 -0
  29. data/configs/databases/sqlite2.yml +16 -0
  30. data/configs/databases/sqlite3.yml +19 -0
  31. data/configs/empty.log +0 -0
  32. data/configs/initializers/inflections.rb +10 -0
  33. data/configs/initializers/mime_types.rb +5 -0
  34. data/configs/initializers/new_rails_defaults.rb +17 -0
  35. data/configs/lighttpd.conf +54 -0
  36. data/configs/routes.rb +43 -0
  37. data/dispatches/dispatch.fcgi +24 -0
  38. data/dispatches/dispatch.rb +10 -0
  39. data/dispatches/gateway.cgi +97 -0
  40. data/doc/README_FOR_APP +2 -0
  41. data/environments/boot.rb +109 -0
  42. data/environments/development.rb +16 -0
  43. data/environments/environment.rb +71 -0
  44. data/environments/production.rb +22 -0
  45. data/environments/test.rb +22 -0
  46. data/fresh_rakefile +10 -0
  47. data/helpers/application.rb +15 -0
  48. data/helpers/application_helper.rb +3 -0
  49. data/helpers/test_helper.rb +38 -0
  50. data/html/404.html +30 -0
  51. data/html/422.html +30 -0
  52. data/html/500.html +30 -0
  53. data/html/favicon.ico +0 -0
  54. data/html/images/rails.png +0 -0
  55. data/html/index.html +274 -0
  56. data/html/javascripts/application.js +2 -0
  57. data/html/robots.txt +5 -0
  58. data/lib/code_statistics.rb +107 -0
  59. data/lib/commands/about.rb +3 -0
  60. data/lib/commands/console.rb +32 -0
  61. data/lib/commands/dbconsole.rb +67 -0
  62. data/lib/commands/destroy.rb +6 -0
  63. data/lib/commands/generate.rb +6 -0
  64. data/lib/commands/ncgi/listener +86 -0
  65. data/lib/commands/ncgi/tracker +69 -0
  66. data/lib/commands/performance/benchmarker.rb +24 -0
  67. data/lib/commands/performance/profiler.rb +50 -0
  68. data/lib/commands/performance/request.rb +6 -0
  69. data/lib/commands/plugin.rb +950 -0
  70. data/lib/commands/process/inspector.rb +68 -0
  71. data/lib/commands/process/reaper.rb +149 -0
  72. data/lib/commands/process/spawner.rb +219 -0
  73. data/lib/commands/process/spinner.rb +57 -0
  74. data/lib/commands/runner.rb +48 -0
  75. data/lib/commands/server.rb +39 -0
  76. data/lib/commands/servers/base.rb +31 -0
  77. data/lib/commands/servers/lighttpd.rb +94 -0
  78. data/lib/commands/servers/mongrel.rb +69 -0
  79. data/lib/commands/servers/new_mongrel.rb +16 -0
  80. data/lib/commands/servers/webrick.rb +66 -0
  81. data/lib/commands/update.rb +4 -0
  82. data/lib/commands.rb +17 -0
  83. data/lib/console_app.rb +30 -0
  84. data/lib/console_sandbox.rb +6 -0
  85. data/lib/console_with_helpers.rb +26 -0
  86. data/lib/dispatcher.rb +24 -0
  87. data/lib/fcgi_handler.rb +239 -0
  88. data/lib/initializer.rb +926 -0
  89. data/lib/rails/gem_builder.rb +21 -0
  90. data/lib/rails/gem_dependency.rb +129 -0
  91. data/lib/rails/mongrel_server/commands.rb +342 -0
  92. data/lib/rails/mongrel_server/handler.rb +55 -0
  93. data/lib/rails/plugin/loader.rb +152 -0
  94. data/lib/rails/plugin/locator.rb +100 -0
  95. data/lib/rails/plugin.rb +116 -0
  96. data/lib/rails/version.rb +9 -0
  97. data/lib/rails_generator/base.rb +263 -0
  98. data/lib/rails_generator/commands.rb +622 -0
  99. data/lib/rails_generator/generated_attribute.rb +42 -0
  100. data/lib/rails_generator/generators/applications/app/USAGE +9 -0
  101. data/lib/rails_generator/generators/applications/app/app_generator.rb +174 -0
  102. data/lib/rails_generator/generators/components/controller/USAGE +29 -0
  103. data/lib/rails_generator/generators/components/controller/controller_generator.rb +37 -0
  104. data/lib/rails_generator/generators/components/controller/templates/controller.rb +7 -0
  105. data/lib/rails_generator/generators/components/controller/templates/functional_test.rb +8 -0
  106. data/lib/rails_generator/generators/components/controller/templates/helper.rb +2 -0
  107. data/lib/rails_generator/generators/components/controller/templates/view.html.erb +2 -0
  108. data/lib/rails_generator/generators/components/integration_test/USAGE +8 -0
  109. data/lib/rails_generator/generators/components/integration_test/integration_test_generator.rb +16 -0
  110. data/lib/rails_generator/generators/components/integration_test/templates/integration_test.rb +10 -0
  111. data/lib/rails_generator/generators/components/mailer/USAGE +16 -0
  112. data/lib/rails_generator/generators/components/mailer/mailer_generator.rb +30 -0
  113. data/lib/rails_generator/generators/components/mailer/templates/fixture.erb +3 -0
  114. data/lib/rails_generator/generators/components/mailer/templates/fixture.rhtml +0 -0
  115. data/lib/rails_generator/generators/components/mailer/templates/mailer.rb +15 -0
  116. data/lib/rails_generator/generators/components/mailer/templates/unit_test.rb +21 -0
  117. data/lib/rails_generator/generators/components/mailer/templates/view.erb +3 -0
  118. data/lib/rails_generator/generators/components/mailer/templates/view.rhtml +0 -0
  119. data/lib/rails_generator/generators/components/migration/USAGE +29 -0
  120. data/lib/rails_generator/generators/components/migration/migration_generator.rb +20 -0
  121. data/lib/rails_generator/generators/components/migration/templates/migration.rb +11 -0
  122. data/lib/rails_generator/generators/components/model/USAGE +27 -0
  123. data/lib/rails_generator/generators/components/model/model_generator.rb +45 -0
  124. data/lib/rails_generator/generators/components/model/templates/fixtures.yml +19 -0
  125. data/lib/rails_generator/generators/components/model/templates/migration.rb +16 -0
  126. data/lib/rails_generator/generators/components/model/templates/model.rb +2 -0
  127. data/lib/rails_generator/generators/components/model/templates/unit_test.rb +8 -0
  128. data/lib/rails_generator/generators/components/observer/USAGE +13 -0
  129. data/lib/rails_generator/generators/components/observer/observer_generator.rb +16 -0
  130. data/lib/rails_generator/generators/components/observer/templates/observer.rb +2 -0
  131. data/lib/rails_generator/generators/components/observer/templates/unit_test.rb +8 -0
  132. data/lib/rails_generator/generators/components/plugin/USAGE +25 -0
  133. data/lib/rails_generator/generators/components/plugin/plugin_generator.rb +39 -0
  134. data/lib/rails_generator/generators/components/plugin/templates/MIT-LICENSE +20 -0
  135. data/lib/rails_generator/generators/components/plugin/templates/README +13 -0
  136. data/lib/rails_generator/generators/components/plugin/templates/Rakefile +22 -0
  137. data/lib/rails_generator/generators/components/plugin/templates/USAGE +8 -0
  138. data/lib/rails_generator/generators/components/plugin/templates/generator.rb +8 -0
  139. data/lib/rails_generator/generators/components/plugin/templates/init.rb +1 -0
  140. data/lib/rails_generator/generators/components/plugin/templates/install.rb +1 -0
  141. data/lib/rails_generator/generators/components/plugin/templates/plugin.rb +1 -0
  142. data/lib/rails_generator/generators/components/plugin/templates/tasks.rake +4 -0
  143. data/lib/rails_generator/generators/components/plugin/templates/uninstall.rb +1 -0
  144. data/lib/rails_generator/generators/components/plugin/templates/unit_test.rb +8 -0
  145. data/lib/rails_generator/generators/components/resource/USAGE +23 -0
  146. data/lib/rails_generator/generators/components/resource/resource_generator.rb +74 -0
  147. data/lib/rails_generator/generators/components/resource/templates/controller.rb +2 -0
  148. data/lib/rails_generator/generators/components/resource/templates/functional_test.rb +8 -0
  149. data/lib/rails_generator/generators/components/resource/templates/helper.rb +2 -0
  150. data/lib/rails_generator/generators/components/scaffold/USAGE +25 -0
  151. data/lib/rails_generator/generators/components/scaffold/scaffold_generator.rb +93 -0
  152. data/lib/rails_generator/generators/components/scaffold/templates/controller.rb +85 -0
  153. data/lib/rails_generator/generators/components/scaffold/templates/functional_test.rb +45 -0
  154. data/lib/rails_generator/generators/components/scaffold/templates/helper.rb +2 -0
  155. data/lib/rails_generator/generators/components/scaffold/templates/layout.html.erb +17 -0
  156. data/lib/rails_generator/generators/components/scaffold/templates/style.css +54 -0
  157. data/lib/rails_generator/generators/components/scaffold/templates/view_edit.html.erb +18 -0
  158. data/lib/rails_generator/generators/components/scaffold/templates/view_index.html.erb +24 -0
  159. data/lib/rails_generator/generators/components/scaffold/templates/view_new.html.erb +17 -0
  160. data/lib/rails_generator/generators/components/scaffold/templates/view_show.html.erb +10 -0
  161. data/lib/rails_generator/generators/components/session_migration/USAGE +10 -0
  162. data/lib/rails_generator/generators/components/session_migration/session_migration_generator.rb +18 -0
  163. data/lib/rails_generator/generators/components/session_migration/templates/migration.rb +16 -0
  164. data/lib/rails_generator/lookup.rb +249 -0
  165. data/lib/rails_generator/manifest.rb +53 -0
  166. data/lib/rails_generator/options.rb +150 -0
  167. data/lib/rails_generator/scripts/destroy.rb +30 -0
  168. data/lib/rails_generator/scripts/generate.rb +7 -0
  169. data/lib/rails_generator/scripts/update.rb +12 -0
  170. data/lib/rails_generator/scripts.rb +89 -0
  171. data/lib/rails_generator/secret_key_generator.rb +164 -0
  172. data/lib/rails_generator/simple_logger.rb +46 -0
  173. data/lib/rails_generator/spec.rb +44 -0
  174. data/lib/rails_generator.rb +43 -0
  175. data/lib/railties_path.rb +1 -0
  176. data/lib/ruby_version_check.rb +17 -0
  177. data/lib/rubyprof_ext.rb +35 -0
  178. data/lib/source_annotation_extractor.rb +102 -0
  179. data/lib/tasks/annotations.rake +23 -0
  180. data/lib/tasks/databases.rake +389 -0
  181. data/lib/tasks/documentation.rake +80 -0
  182. data/lib/tasks/framework.rake +105 -0
  183. data/lib/tasks/gems.rake +64 -0
  184. data/lib/tasks/log.rake +9 -0
  185. data/lib/tasks/misc.rake +57 -0
  186. data/lib/tasks/rails.rb +8 -0
  187. data/lib/tasks/routes.rake +17 -0
  188. data/lib/tasks/statistics.rake +18 -0
  189. data/lib/tasks/testing.rake +118 -0
  190. data/lib/tasks/tmp.rake +37 -0
  191. data/lib/test_help.rb +28 -0
  192. data/lib/webrick_server.rb +165 -0
  193. metadata +356 -0
@@ -0,0 +1,68 @@
1
+ require 'optparse'
2
+
3
+ if RUBY_PLATFORM =~ /(:?mswin|mingw)/ then abort("Inspector is only for Unix") end
4
+
5
+ OPTIONS = {
6
+ :pid_path => File.expand_path(RAILS_ROOT + '/tmp/pids'),
7
+ :pattern => "dispatch.*.pid",
8
+ :ps => "ps -o pid,state,user,start,time,pcpu,vsz,majflt,command -p %s"
9
+ }
10
+
11
+ class Inspector
12
+ def self.inspect(pid_path, pattern)
13
+ new(pid_path, pattern).inspect
14
+ end
15
+
16
+ def initialize(pid_path, pattern)
17
+ @pid_path, @pattern = pid_path, pattern
18
+ end
19
+
20
+ def inspect
21
+ header = `#{OPTIONS[:ps] % 1}`.split("\n")[0] + "\n"
22
+ lines = pids.collect { |pid| `#{OPTIONS[:ps] % pid}`.split("\n")[1] }
23
+
24
+ puts(header + lines.join("\n"))
25
+ end
26
+
27
+ private
28
+ def pids
29
+ pid_files.collect do |pid_file|
30
+ File.read(pid_file).to_i
31
+ end
32
+ end
33
+
34
+ def pid_files
35
+ Dir.glob(@pid_path + "/" + @pattern)
36
+ end
37
+ end
38
+
39
+
40
+ ARGV.options do |opts|
41
+ opts.banner = "Usage: inspector [options]"
42
+
43
+ opts.separator ""
44
+
45
+ opts.on <<-EOF
46
+ Description:
47
+ Displays system information about Rails dispatchers (or other processes that use pid files) through
48
+ the ps command.
49
+
50
+ Examples:
51
+ inspector # default ps on all tmp/pids/dispatch.*.pid files
52
+ inspector -s 'ps -o user,start,majflt,pcpu,vsz -p %s' # custom ps, %s is where the pid is interleaved
53
+ EOF
54
+
55
+ opts.on(" Options:")
56
+
57
+ opts.on("-s", "--ps=command", "default: #{OPTIONS[:ps]}", String) { |v| OPTIONS[:ps] = v }
58
+ opts.on("-p", "--pidpath=path", "default: #{OPTIONS[:pid_path]}", String) { |v| OPTIONS[:pid_path] = v }
59
+ opts.on("-r", "--pattern=pattern", "default: #{OPTIONS[:pattern]}", String) { |v| OPTIONS[:pattern] = v }
60
+
61
+ opts.separator ""
62
+
63
+ opts.on("-h", "--help", "Show this help message.") { puts opts; exit }
64
+
65
+ opts.parse!
66
+ end
67
+
68
+ Inspector.inspect(OPTIONS[:pid_path], OPTIONS[:pattern])
@@ -0,0 +1,149 @@
1
+ require 'optparse'
2
+ require 'net/http'
3
+ require 'uri'
4
+
5
+ if RUBY_PLATFORM =~ /(:?mswin|mingw)/ then abort("Reaper is only for Unix") end
6
+
7
+ class Killer
8
+ class << self
9
+ # Searches for all processes matching the given keywords, and then invokes
10
+ # a specific action on each of them. This is useful for (e.g.) reloading a
11
+ # set of processes:
12
+ #
13
+ # Killer.process(:reload, "/tmp/pids", "dispatcher.*.pid")
14
+ def process(action, pid_path, pattern, keyword)
15
+ new(pid_path, pattern, keyword).process(action)
16
+ end
17
+
18
+ # Forces the (rails) application to reload by sending a +HUP+ signal to the
19
+ # process.
20
+ def reload(pid)
21
+ `kill -s HUP #{pid}`
22
+ end
23
+
24
+ # Force the (rails) application to restart by sending a +USR2+ signal to the
25
+ # process.
26
+ def restart(pid)
27
+ `kill -s USR2 #{pid}`
28
+ end
29
+
30
+ # Forces the (rails) application to gracefully terminate by sending a
31
+ # +TERM+ signal to the process.
32
+ def graceful(pid)
33
+ `kill -s TERM #{pid}`
34
+ end
35
+
36
+ # Forces the (rails) application to terminate immediately by sending a -9
37
+ # signal to the process.
38
+ def kill(pid)
39
+ `kill -9 #{pid}`
40
+ end
41
+
42
+ # Send a +USR1+ signal to the process.
43
+ def usr1(pid)
44
+ `kill -s USR1 #{pid}`
45
+ end
46
+ end
47
+
48
+ def initialize(pid_path, pattern, keyword=nil)
49
+ @pid_path, @pattern, @keyword = pid_path, pattern, keyword
50
+ end
51
+
52
+ def process(action)
53
+ pids = find_processes
54
+
55
+ if pids.empty?
56
+ warn "Couldn't find any pid file in '#{@pid_path}' matching '#{@pattern}'"
57
+ warn "(also looked for processes matching #{@keyword.inspect})" if @keyword
58
+ else
59
+ pids.each do |pid|
60
+ puts "#{action.capitalize}ing #{pid}"
61
+ self.class.send(action, pid)
62
+ end
63
+
64
+ delete_pid_files if terminating?(action)
65
+ end
66
+ end
67
+
68
+ private
69
+ def terminating?(action)
70
+ [ "kill", "graceful" ].include?(action)
71
+ end
72
+
73
+ def find_processes
74
+ files = pid_files
75
+ if files.empty?
76
+ find_processes_via_grep
77
+ else
78
+ files.collect { |pid_file| File.read(pid_file).to_i }
79
+ end
80
+ end
81
+
82
+ def find_processes_via_grep
83
+ lines = `ps axww -o 'pid command' | grep #{@keyword}`.split(/\n/).
84
+ reject { |line| line =~ /inq|ps axww|grep|spawn-fcgi|spawner|reaper/ }
85
+ lines.map { |line| line[/^\s*(\d+)/, 1].to_i }
86
+ end
87
+
88
+ def delete_pid_files
89
+ pid_files.each { |pid_file| File.delete(pid_file) }
90
+ end
91
+
92
+ def pid_files
93
+ Dir.glob(@pid_path + "/" + @pattern)
94
+ end
95
+ end
96
+
97
+
98
+ OPTIONS = {
99
+ :action => "restart",
100
+ :pid_path => File.expand_path(RAILS_ROOT + '/tmp/pids'),
101
+ :pattern => "dispatch.[0-9]*.pid",
102
+ :dispatcher => File.expand_path("#{RAILS_ROOT}/public/dispatch.fcgi")
103
+ }
104
+
105
+ ARGV.options do |opts|
106
+ opts.banner = "Usage: reaper [options]"
107
+
108
+ opts.separator ""
109
+
110
+ opts.on <<-EOF
111
+ Description:
112
+ The reaper is used to restart, reload, gracefully exit, and forcefully exit processes
113
+ running a Rails Dispatcher (or any other process responding to the same signals). This
114
+ is commonly done when a new version of the application is available, so the existing
115
+ processes can be updated to use the latest code.
116
+
117
+ It uses pid files to work on the processes and by default assume them to be located
118
+ in RAILS_ROOT/tmp/pids.
119
+
120
+ The reaper actions are:
121
+
122
+ * restart : Restarts the application by reloading both application and framework code
123
+ * reload : Only reloads the application, but not the framework (like the development environment)
124
+ * graceful: Marks all of the processes for exit after the next request
125
+ * kill : Forcefully exists all processes regardless of whether they're currently serving a request
126
+
127
+ Restart is the most common and default action.
128
+
129
+ Examples:
130
+ reaper # restarts the default dispatchers
131
+ reaper -a reload # reload the default dispatchers
132
+ reaper -a kill -r *.pid # kill all processes that keep pids in tmp/pids
133
+ EOF
134
+
135
+ opts.on(" Options:")
136
+
137
+ opts.on("-a", "--action=name", "reload|graceful|kill (default: #{OPTIONS[:action]})", String) { |v| OPTIONS[:action] = v }
138
+ opts.on("-p", "--pidpath=path", "default: #{OPTIONS[:pid_path]}", String) { |v| OPTIONS[:pid_path] = v }
139
+ opts.on("-r", "--pattern=pattern", "default: #{OPTIONS[:pattern]}", String) { |v| OPTIONS[:pattern] = v }
140
+ opts.on("-d", "--dispatcher=path", "DEPRECATED. default: #{OPTIONS[:dispatcher]}", String) { |v| OPTIONS[:dispatcher] = v }
141
+
142
+ opts.separator ""
143
+
144
+ opts.on("-h", "--help", "Show this help message.") { puts opts; exit }
145
+
146
+ opts.parse!
147
+ end
148
+
149
+ Killer.process(OPTIONS[:action], OPTIONS[:pid_path], OPTIONS[:pattern], OPTIONS[:dispatcher])
@@ -0,0 +1,219 @@
1
+ require 'active_support'
2
+ require 'optparse'
3
+ require 'socket'
4
+ require 'fileutils'
5
+
6
+ def daemonize #:nodoc:
7
+ exit if fork # Parent exits, child continues.
8
+ Process.setsid # Become session leader.
9
+ exit if fork # Zap session leader. See [1].
10
+ Dir.chdir "/" # Release old working directory.
11
+ File.umask 0000 # Ensure sensible umask. Adjust as needed.
12
+ STDIN.reopen "/dev/null" # Free file descriptors and
13
+ STDOUT.reopen "/dev/null", "a" # point them somewhere sensible.
14
+ STDERR.reopen STDOUT # STDOUT/ERR should better go to a logfile.
15
+ end
16
+
17
+ class Spawner
18
+ def self.record_pid(name = "#{OPTIONS[:process]}.spawner", id = Process.pid)
19
+ FileUtils.mkdir_p(OPTIONS[:pids])
20
+ File.open(File.expand_path(OPTIONS[:pids] + "/#{name}.pid"), "w+") { |f| f.write(id) }
21
+ end
22
+
23
+ def self.spawn_all
24
+ OPTIONS[:instances].times do |i|
25
+ port = OPTIONS[:port] + i
26
+ print "Checking if something is already running on #{OPTIONS[:address]}:#{port}..."
27
+
28
+ begin
29
+ srv = TCPServer.new(OPTIONS[:address], port)
30
+ srv.close
31
+ srv = nil
32
+
33
+ puts "NO"
34
+ puts "Starting dispatcher on port: #{OPTIONS[:address]}:#{port}"
35
+
36
+ FileUtils.mkdir_p(OPTIONS[:pids])
37
+ spawn(port)
38
+ rescue
39
+ puts "YES"
40
+ end
41
+ end
42
+ end
43
+ end
44
+
45
+ class FcgiSpawner < Spawner
46
+ def self.spawn(port)
47
+ cmd = "#{OPTIONS[:spawner]} -f #{OPTIONS[:dispatcher]} -p #{port} -P #{OPTIONS[:pids]}/#{OPTIONS[:process]}.#{port}.pid"
48
+ cmd << " -a #{OPTIONS[:address]}" if can_bind_to_custom_address?
49
+ system(cmd)
50
+ end
51
+
52
+ def self.can_bind_to_custom_address?
53
+ @@can_bind_to_custom_address ||= /^\s-a\s/.match `#{OPTIONS[:spawner]} -h`
54
+ end
55
+ end
56
+
57
+ class MongrelSpawner < Spawner
58
+ def self.spawn(port)
59
+ cmd =
60
+ "mongrel_rails start -d " +
61
+ "-a #{OPTIONS[:address]} " +
62
+ "-p #{port} " +
63
+ "-P #{OPTIONS[:pids]}/#{OPTIONS[:process]}.#{port}.pid " +
64
+ "-e #{OPTIONS[:environment]} " +
65
+ "-c #{OPTIONS[:rails_root]} " +
66
+ "-l #{OPTIONS[:rails_root]}/log/mongrel.log"
67
+
68
+ # Add prefix functionality to spawner's call to mongrel_rails
69
+ # Digging through monrel's project subversion server, the earliest
70
+ # Tag that has prefix implemented in the bin/mongrel_rails file
71
+ # is 0.3.15 which also happens to be the earilest tag listed.
72
+ # References: http://mongrel.rubyforge.org/svn/tags
73
+ if Mongrel::Const::MONGREL_VERSION.to_f >=0.3 && !OPTIONS[:prefix].nil?
74
+ cmd = cmd + " --prefix #{OPTIONS[:prefix]}"
75
+ end
76
+ system(cmd)
77
+ end
78
+
79
+ def self.can_bind_to_custom_address?
80
+ true
81
+ end
82
+ end
83
+
84
+
85
+ begin
86
+ require_library_or_gem 'fcgi'
87
+ rescue Exception
88
+ # FCGI not available
89
+ end
90
+
91
+ begin
92
+ require_library_or_gem 'mongrel'
93
+ rescue Exception
94
+ # Mongrel not available
95
+ end
96
+
97
+ server = case ARGV.first
98
+ when "fcgi", "mongrel"
99
+ ARGV.shift
100
+ else
101
+ if defined?(Mongrel)
102
+ "mongrel"
103
+ elsif RUBY_PLATFORM !~ /(:?mswin|mingw)/ && !silence_stderr { `spawn-fcgi -version` }.blank? && defined?(FCGI)
104
+ "fcgi"
105
+ end
106
+ end
107
+
108
+ case server
109
+ when "fcgi"
110
+ puts "=> Starting FCGI dispatchers"
111
+ spawner_class = FcgiSpawner
112
+ when "mongrel"
113
+ puts "=> Starting mongrel dispatchers"
114
+ spawner_class = MongrelSpawner
115
+ else
116
+ puts "Neither FCGI (spawn-fcgi) nor Mongrel was installed and available!"
117
+ exit(0)
118
+ end
119
+
120
+
121
+
122
+ OPTIONS = {
123
+ :environment => "production",
124
+ :spawner => '/usr/bin/env spawn-fcgi',
125
+ :dispatcher => File.expand_path(RELATIVE_RAILS_ROOT + '/public/dispatch.fcgi'),
126
+ :pids => File.expand_path(RELATIVE_RAILS_ROOT + "/tmp/pids"),
127
+ :rails_root => File.expand_path(RELATIVE_RAILS_ROOT),
128
+ :process => "dispatch",
129
+ :port => 8000,
130
+ :address => '0.0.0.0',
131
+ :instances => 3,
132
+ :repeat => nil,
133
+ :prefix => nil
134
+ }
135
+
136
+ ARGV.options do |opts|
137
+ opts.banner = "Usage: spawner [platform] [options]"
138
+
139
+ opts.separator ""
140
+
141
+ opts.on <<-EOF
142
+ Description:
143
+ The spawner is a wrapper for spawn-fcgi and mongrel that makes it
144
+ easier to start multiple processes running the Rails dispatcher. The
145
+ spawn-fcgi command is included with the lighttpd web server, but can
146
+ be used with both Apache and lighttpd (and any other web server
147
+ supporting externally managed FCGI processes). Mongrel automatically
148
+ ships with with mongrel_rails for starting dispatchers.
149
+
150
+ The first choice you need to make is whether to spawn the Rails
151
+ dispatchers as FCGI or Mongrel. By default, this spawner will prefer
152
+ Mongrel, so if that's installed, and no platform choice is made,
153
+ Mongrel is used.
154
+
155
+ Then decide a starting port (default is 8000) and the number of FCGI
156
+ process instances you'd like to run. So if you pick 9100 and 3
157
+ instances, you'll start processes on 9100, 9101, and 9102.
158
+
159
+ By setting the repeat option, you get a protection loop, which will
160
+ attempt to restart any FCGI processes that might have been exited or
161
+ outright crashed.
162
+
163
+ You can select bind address for started processes. By default these
164
+ listen on every interface. For single machine installations you would
165
+ probably want to use 127.0.0.1, hiding them form the outside world.
166
+
167
+ Examples:
168
+ spawner # starts instances on 8000, 8001, and 8002
169
+ # using Mongrel if available.
170
+ spawner fcgi # starts instances on 8000, 8001, and 8002
171
+ # using FCGI.
172
+ spawner mongrel -i 5 # starts instances on 8000, 8001, 8002,
173
+ # 8003, and 8004 using Mongrel.
174
+ spawner -p 9100 -i 10 # starts 10 instances counting from 9100 to
175
+ # 9109 using Mongrel if available.
176
+ spawner -p 9100 -r 5 # starts 3 instances counting from 9100 to
177
+ # 9102 and attempts start them every 5
178
+ # seconds.
179
+ spawner -a 127.0.0.1 # starts 3 instances binding to localhost
180
+ EOF
181
+
182
+ opts.on(" Options:")
183
+
184
+ opts.on("-p", "--port=number", Integer, "Starting port number (default: #{OPTIONS[:port]})") { |OPTIONS[:port]| }
185
+
186
+ if spawner_class.can_bind_to_custom_address?
187
+ opts.on("-a", "--address=ip", String, "Bind to IP address (default: #{OPTIONS[:address]})") { |OPTIONS[:address]| }
188
+ end
189
+
190
+ opts.on("-p", "--port=number", Integer, "Starting port number (default: #{OPTIONS[:port]})") { |v| OPTIONS[:port] = v }
191
+ opts.on("-i", "--instances=number", Integer, "Number of instances (default: #{OPTIONS[:instances]})") { |v| OPTIONS[:instances] = v }
192
+ opts.on("-r", "--repeat=seconds", Integer, "Repeat spawn attempts every n seconds (default: off)") { |v| OPTIONS[:repeat] = v }
193
+ opts.on("-e", "--environment=name", String, "test|development|production (default: #{OPTIONS[:environment]})") { |v| OPTIONS[:environment] = v }
194
+ opts.on("-P", "--prefix=path", String, "URL prefix for Rails app. [Used only with Mongrel > v0.3.15]: (default: #{OPTIONS[:prefix]})") { |v| OPTIONS[:prefix] = v }
195
+ opts.on("-n", "--process=name", String, "default: #{OPTIONS[:process]}") { |v| OPTIONS[:process] = v }
196
+ opts.on("-s", "--spawner=path", String, "default: #{OPTIONS[:spawner]}") { |v| OPTIONS[:spawner] = v }
197
+ opts.on("-d", "--dispatcher=path", String, "default: #{OPTIONS[:dispatcher]}") { |dispatcher| OPTIONS[:dispatcher] = File.expand_path(dispatcher) }
198
+
199
+ opts.separator ""
200
+
201
+ opts.on("-h", "--help", "Show this help message.") { puts opts; exit }
202
+
203
+ opts.parse!
204
+ end
205
+
206
+ ENV["RAILS_ENV"] = OPTIONS[:environment]
207
+
208
+ if OPTIONS[:repeat]
209
+ daemonize
210
+ trap("TERM") { exit }
211
+ spawner_class.record_pid
212
+
213
+ loop do
214
+ spawner_class.spawn_all
215
+ sleep(OPTIONS[:repeat])
216
+ end
217
+ else
218
+ spawner_class.spawn_all
219
+ end
@@ -0,0 +1,57 @@
1
+ require 'optparse'
2
+
3
+ def daemonize #:nodoc:
4
+ exit if fork # Parent exits, child continues.
5
+ Process.setsid # Become session leader.
6
+ exit if fork # Zap session leader. See [1].
7
+ Dir.chdir "/" # Release old working directory.
8
+ File.umask 0000 # Ensure sensible umask. Adjust as needed.
9
+ STDIN.reopen "/dev/null" # Free file descriptors and
10
+ STDOUT.reopen "/dev/null", "a" # point them somewhere sensible.
11
+ STDERR.reopen STDOUT # STDOUT/ERR should better go to a logfile.
12
+ end
13
+
14
+ OPTIONS = {
15
+ :interval => 5.0,
16
+ :command => File.expand_path(RAILS_ROOT + '/script/process/spawner'),
17
+ :daemon => false
18
+ }
19
+
20
+ ARGV.options do |opts|
21
+ opts.banner = "Usage: spinner [options]"
22
+
23
+ opts.separator ""
24
+
25
+ opts.on <<-EOF
26
+ Description:
27
+ The spinner is a protection loop for the spawner, which will attempt to restart any FCGI processes
28
+ that might have been exited or outright crashed. It's a brute-force attempt that'll just try
29
+ to run the spawner every X number of seconds, so it does pose a light load on the server.
30
+
31
+ Examples:
32
+ spinner # attempts to run the spawner with default settings every second with output on the terminal
33
+ spinner -i 3 -d # only run the spawner every 3 seconds and detach from the terminal to become a daemon
34
+ spinner -c '/path/to/app/script/process/spawner -p 9000 -i 10' -d # using custom spawner
35
+ EOF
36
+
37
+ opts.on(" Options:")
38
+
39
+ opts.on("-c", "--command=path", String) { |v| OPTIONS[:command] = v }
40
+ opts.on("-i", "--interval=seconds", Float) { |v| OPTIONS[:interval] = v }
41
+ opts.on("-d", "--daemon") { |v| OPTIONS[:daemon] = v }
42
+
43
+ opts.separator ""
44
+
45
+ opts.on("-h", "--help", "Show this help message.") { puts opts; exit }
46
+
47
+ opts.parse!
48
+ end
49
+
50
+ daemonize if OPTIONS[:daemon]
51
+
52
+ trap(OPTIONS[:daemon] ? "TERM" : "INT") { exit }
53
+
54
+ loop do
55
+ system(OPTIONS[:command])
56
+ sleep(OPTIONS[:interval])
57
+ end
@@ -0,0 +1,48 @@
1
+ require 'optparse'
2
+
3
+ options = { :environment => (ENV['RAILS_ENV'] || "development").dup }
4
+ code_or_file = nil
5
+
6
+ ARGV.clone.options do |opts|
7
+ script_name = File.basename($0)
8
+ opts.banner = "Usage: #{$0} [options] ('Some.ruby(code)' or a filename)"
9
+
10
+ opts.separator ""
11
+
12
+ opts.on("-e", "--environment=name", String,
13
+ "Specifies the environment for the runner to operate under (test/development/production).",
14
+ "Default: development") { |v| options[:environment] = v }
15
+
16
+ opts.separator ""
17
+
18
+ opts.on("-h", "--help",
19
+ "Show this help message.") { $stderr.puts opts; exit }
20
+
21
+ if RUBY_PLATFORM !~ /mswin/
22
+ opts.separator ""
23
+ opts.separator "You can also use runner as a shebang line for your scripts like this:"
24
+ opts.separator "-------------------------------------------------------------"
25
+ opts.separator "#!/usr/bin/env #{File.expand_path($0)}"
26
+ opts.separator ""
27
+ opts.separator "Product.find(:all).each { |p| p.price *= 2 ; p.save! }"
28
+ opts.separator "-------------------------------------------------------------"
29
+ end
30
+
31
+ opts.order! { |o| code_or_file ||= o } rescue retry
32
+ end
33
+
34
+ ARGV.delete(code_or_file)
35
+
36
+ ENV["RAILS_ENV"] = options[:environment]
37
+ RAILS_ENV.replace(options[:environment]) if defined?(RAILS_ENV)
38
+
39
+ require RAILS_ROOT + '/config/environment'
40
+
41
+ if code_or_file.nil?
42
+ $stderr.puts "Run '#{$0} -h' for help."
43
+ exit 1
44
+ elsif File.exist?(code_or_file)
45
+ eval(File.read(code_or_file))
46
+ else
47
+ eval(code_or_file)
48
+ end
@@ -0,0 +1,39 @@
1
+ require 'active_support'
2
+ require 'fileutils'
3
+
4
+ begin
5
+ require_library_or_gem 'fcgi'
6
+ rescue Exception
7
+ # FCGI not available
8
+ end
9
+
10
+ begin
11
+ require_library_or_gem 'mongrel'
12
+ rescue Exception
13
+ # Mongrel not available
14
+ end
15
+
16
+ server = case ARGV.first
17
+ when "lighttpd", "mongrel", "new_mongrel", "webrick"
18
+ ARGV.shift
19
+ else
20
+ if defined?(Mongrel)
21
+ "mongrel"
22
+ elsif RUBY_PLATFORM !~ /(:?mswin|mingw)/ && !silence_stderr { `lighttpd -version` }.blank? && defined?(FCGI)
23
+ "lighttpd"
24
+ else
25
+ "webrick"
26
+ end
27
+ end
28
+
29
+ case server
30
+ when "webrick"
31
+ puts "=> Booting WEBrick..."
32
+ when "lighttpd"
33
+ puts "=> Booting lighttpd (use 'script/server webrick' to force WEBrick)"
34
+ when "mongrel", "new_mongrel"
35
+ puts "=> Booting Mongrel (use 'script/server webrick' to force WEBrick)"
36
+ end
37
+
38
+ %w(cache pids sessions sockets).each { |dir_to_make| FileUtils.mkdir_p(File.join(RAILS_ROOT, 'tmp', dir_to_make)) }
39
+ require "commands/servers/#{server}"
@@ -0,0 +1,31 @@
1
+ def tail(log_file)
2
+ cursor = File.size(log_file)
3
+ last_checked = Time.now
4
+ tail_thread = Thread.new do
5
+ File.open(log_file, 'r') do |f|
6
+ loop do
7
+ f.seek cursor
8
+ if f.mtime > last_checked
9
+ last_checked = f.mtime
10
+ contents = f.read
11
+ cursor += contents.length
12
+ print contents
13
+ end
14
+ sleep 1
15
+ end
16
+ end
17
+ end
18
+ tail_thread
19
+ end
20
+
21
+ def start_debugger
22
+ begin
23
+ require_library_or_gem 'ruby-debug'
24
+ Debugger.start
25
+ Debugger.settings[:autoeval] = true if Debugger.respond_to?(:settings)
26
+ puts "=> Debugger enabled"
27
+ rescue Exception
28
+ puts "You need to install ruby-debug to run the server in debugging mode. With gems, use 'gem install ruby-debug'"
29
+ exit
30
+ end
31
+ end