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.
- data/CHANGELOG +3 -0
- data/MIT-LICENSE +20 -0
- data/README +309 -0
- data/Rakefile +339 -0
- data/bin/about +4 -0
- data/bin/console +3 -0
- data/bin/dbconsole +3 -0
- data/bin/destroy +3 -0
- data/bin/erails +19 -0
- data/bin/generate +3 -0
- data/bin/performance/benchmarker +3 -0
- data/bin/performance/profiler +3 -0
- data/bin/performance/request +3 -0
- data/bin/plugin +3 -0
- data/bin/process/inspector +3 -0
- data/bin/process/reaper +3 -0
- data/bin/process/spawner +3 -0
- data/bin/runner +3 -0
- data/bin/server +3 -0
- data/builtin/rails_info/rails/info.rb +125 -0
- data/builtin/rails_info/rails/info_controller.rb +9 -0
- data/builtin/rails_info/rails/info_helper.rb +2 -0
- data/builtin/rails_info/rails_info_controller.rb +2 -0
- data/configs/apache.conf +40 -0
- data/configs/databases/frontbase.yml +28 -0
- data/configs/databases/mysql.yml +54 -0
- data/configs/databases/oracle.yml +39 -0
- data/configs/databases/postgresql.yml +48 -0
- data/configs/databases/sqlite2.yml +16 -0
- data/configs/databases/sqlite3.yml +19 -0
- data/configs/empty.log +0 -0
- data/configs/initializers/inflections.rb +10 -0
- data/configs/initializers/mime_types.rb +5 -0
- data/configs/initializers/new_rails_defaults.rb +17 -0
- data/configs/lighttpd.conf +54 -0
- data/configs/routes.rb +43 -0
- data/dispatches/dispatch.fcgi +24 -0
- data/dispatches/dispatch.rb +10 -0
- data/dispatches/gateway.cgi +97 -0
- data/doc/README_FOR_APP +2 -0
- data/environments/boot.rb +109 -0
- data/environments/development.rb +16 -0
- data/environments/environment.rb +71 -0
- data/environments/production.rb +22 -0
- data/environments/test.rb +22 -0
- data/fresh_rakefile +10 -0
- data/helpers/application.rb +15 -0
- data/helpers/application_helper.rb +3 -0
- data/helpers/test_helper.rb +38 -0
- data/html/404.html +30 -0
- data/html/422.html +30 -0
- data/html/500.html +30 -0
- data/html/favicon.ico +0 -0
- data/html/images/rails.png +0 -0
- data/html/index.html +274 -0
- data/html/javascripts/application.js +2 -0
- data/html/robots.txt +5 -0
- data/lib/code_statistics.rb +107 -0
- data/lib/commands/about.rb +3 -0
- data/lib/commands/console.rb +32 -0
- data/lib/commands/dbconsole.rb +67 -0
- data/lib/commands/destroy.rb +6 -0
- data/lib/commands/generate.rb +6 -0
- data/lib/commands/ncgi/listener +86 -0
- data/lib/commands/ncgi/tracker +69 -0
- data/lib/commands/performance/benchmarker.rb +24 -0
- data/lib/commands/performance/profiler.rb +50 -0
- data/lib/commands/performance/request.rb +6 -0
- data/lib/commands/plugin.rb +950 -0
- data/lib/commands/process/inspector.rb +68 -0
- data/lib/commands/process/reaper.rb +149 -0
- data/lib/commands/process/spawner.rb +219 -0
- data/lib/commands/process/spinner.rb +57 -0
- data/lib/commands/runner.rb +48 -0
- data/lib/commands/server.rb +39 -0
- data/lib/commands/servers/base.rb +31 -0
- data/lib/commands/servers/lighttpd.rb +94 -0
- data/lib/commands/servers/mongrel.rb +69 -0
- data/lib/commands/servers/new_mongrel.rb +16 -0
- data/lib/commands/servers/webrick.rb +66 -0
- data/lib/commands/update.rb +4 -0
- data/lib/commands.rb +17 -0
- data/lib/console_app.rb +30 -0
- data/lib/console_sandbox.rb +6 -0
- data/lib/console_with_helpers.rb +26 -0
- data/lib/dispatcher.rb +24 -0
- data/lib/fcgi_handler.rb +239 -0
- data/lib/initializer.rb +926 -0
- data/lib/rails/gem_builder.rb +21 -0
- data/lib/rails/gem_dependency.rb +129 -0
- data/lib/rails/mongrel_server/commands.rb +342 -0
- data/lib/rails/mongrel_server/handler.rb +55 -0
- data/lib/rails/plugin/loader.rb +152 -0
- data/lib/rails/plugin/locator.rb +100 -0
- data/lib/rails/plugin.rb +116 -0
- data/lib/rails/version.rb +9 -0
- data/lib/rails_generator/base.rb +263 -0
- data/lib/rails_generator/commands.rb +622 -0
- data/lib/rails_generator/generated_attribute.rb +42 -0
- data/lib/rails_generator/generators/applications/app/USAGE +9 -0
- data/lib/rails_generator/generators/applications/app/app_generator.rb +174 -0
- data/lib/rails_generator/generators/components/controller/USAGE +29 -0
- data/lib/rails_generator/generators/components/controller/controller_generator.rb +37 -0
- data/lib/rails_generator/generators/components/controller/templates/controller.rb +7 -0
- data/lib/rails_generator/generators/components/controller/templates/functional_test.rb +8 -0
- data/lib/rails_generator/generators/components/controller/templates/helper.rb +2 -0
- data/lib/rails_generator/generators/components/controller/templates/view.html.erb +2 -0
- data/lib/rails_generator/generators/components/integration_test/USAGE +8 -0
- data/lib/rails_generator/generators/components/integration_test/integration_test_generator.rb +16 -0
- data/lib/rails_generator/generators/components/integration_test/templates/integration_test.rb +10 -0
- data/lib/rails_generator/generators/components/mailer/USAGE +16 -0
- data/lib/rails_generator/generators/components/mailer/mailer_generator.rb +30 -0
- data/lib/rails_generator/generators/components/mailer/templates/fixture.erb +3 -0
- data/lib/rails_generator/generators/components/mailer/templates/fixture.rhtml +0 -0
- data/lib/rails_generator/generators/components/mailer/templates/mailer.rb +15 -0
- data/lib/rails_generator/generators/components/mailer/templates/unit_test.rb +21 -0
- data/lib/rails_generator/generators/components/mailer/templates/view.erb +3 -0
- data/lib/rails_generator/generators/components/mailer/templates/view.rhtml +0 -0
- data/lib/rails_generator/generators/components/migration/USAGE +29 -0
- data/lib/rails_generator/generators/components/migration/migration_generator.rb +20 -0
- data/lib/rails_generator/generators/components/migration/templates/migration.rb +11 -0
- data/lib/rails_generator/generators/components/model/USAGE +27 -0
- data/lib/rails_generator/generators/components/model/model_generator.rb +45 -0
- data/lib/rails_generator/generators/components/model/templates/fixtures.yml +19 -0
- data/lib/rails_generator/generators/components/model/templates/migration.rb +16 -0
- data/lib/rails_generator/generators/components/model/templates/model.rb +2 -0
- data/lib/rails_generator/generators/components/model/templates/unit_test.rb +8 -0
- data/lib/rails_generator/generators/components/observer/USAGE +13 -0
- data/lib/rails_generator/generators/components/observer/observer_generator.rb +16 -0
- data/lib/rails_generator/generators/components/observer/templates/observer.rb +2 -0
- data/lib/rails_generator/generators/components/observer/templates/unit_test.rb +8 -0
- data/lib/rails_generator/generators/components/plugin/USAGE +25 -0
- data/lib/rails_generator/generators/components/plugin/plugin_generator.rb +39 -0
- data/lib/rails_generator/generators/components/plugin/templates/MIT-LICENSE +20 -0
- data/lib/rails_generator/generators/components/plugin/templates/README +13 -0
- data/lib/rails_generator/generators/components/plugin/templates/Rakefile +22 -0
- data/lib/rails_generator/generators/components/plugin/templates/USAGE +8 -0
- data/lib/rails_generator/generators/components/plugin/templates/generator.rb +8 -0
- data/lib/rails_generator/generators/components/plugin/templates/init.rb +1 -0
- data/lib/rails_generator/generators/components/plugin/templates/install.rb +1 -0
- data/lib/rails_generator/generators/components/plugin/templates/plugin.rb +1 -0
- data/lib/rails_generator/generators/components/plugin/templates/tasks.rake +4 -0
- data/lib/rails_generator/generators/components/plugin/templates/uninstall.rb +1 -0
- data/lib/rails_generator/generators/components/plugin/templates/unit_test.rb +8 -0
- data/lib/rails_generator/generators/components/resource/USAGE +23 -0
- data/lib/rails_generator/generators/components/resource/resource_generator.rb +74 -0
- data/lib/rails_generator/generators/components/resource/templates/controller.rb +2 -0
- data/lib/rails_generator/generators/components/resource/templates/functional_test.rb +8 -0
- data/lib/rails_generator/generators/components/resource/templates/helper.rb +2 -0
- data/lib/rails_generator/generators/components/scaffold/USAGE +25 -0
- data/lib/rails_generator/generators/components/scaffold/scaffold_generator.rb +93 -0
- data/lib/rails_generator/generators/components/scaffold/templates/controller.rb +85 -0
- data/lib/rails_generator/generators/components/scaffold/templates/functional_test.rb +45 -0
- data/lib/rails_generator/generators/components/scaffold/templates/helper.rb +2 -0
- data/lib/rails_generator/generators/components/scaffold/templates/layout.html.erb +17 -0
- data/lib/rails_generator/generators/components/scaffold/templates/style.css +54 -0
- data/lib/rails_generator/generators/components/scaffold/templates/view_edit.html.erb +18 -0
- data/lib/rails_generator/generators/components/scaffold/templates/view_index.html.erb +24 -0
- data/lib/rails_generator/generators/components/scaffold/templates/view_new.html.erb +17 -0
- data/lib/rails_generator/generators/components/scaffold/templates/view_show.html.erb +10 -0
- data/lib/rails_generator/generators/components/session_migration/USAGE +10 -0
- data/lib/rails_generator/generators/components/session_migration/session_migration_generator.rb +18 -0
- data/lib/rails_generator/generators/components/session_migration/templates/migration.rb +16 -0
- data/lib/rails_generator/lookup.rb +249 -0
- data/lib/rails_generator/manifest.rb +53 -0
- data/lib/rails_generator/options.rb +150 -0
- data/lib/rails_generator/scripts/destroy.rb +30 -0
- data/lib/rails_generator/scripts/generate.rb +7 -0
- data/lib/rails_generator/scripts/update.rb +12 -0
- data/lib/rails_generator/scripts.rb +89 -0
- data/lib/rails_generator/secret_key_generator.rb +164 -0
- data/lib/rails_generator/simple_logger.rb +46 -0
- data/lib/rails_generator/spec.rb +44 -0
- data/lib/rails_generator.rb +43 -0
- data/lib/railties_path.rb +1 -0
- data/lib/ruby_version_check.rb +17 -0
- data/lib/rubyprof_ext.rb +35 -0
- data/lib/source_annotation_extractor.rb +102 -0
- data/lib/tasks/annotations.rake +23 -0
- data/lib/tasks/databases.rake +389 -0
- data/lib/tasks/documentation.rake +80 -0
- data/lib/tasks/framework.rake +105 -0
- data/lib/tasks/gems.rake +64 -0
- data/lib/tasks/log.rake +9 -0
- data/lib/tasks/misc.rake +57 -0
- data/lib/tasks/rails.rb +8 -0
- data/lib/tasks/routes.rake +17 -0
- data/lib/tasks/statistics.rake +18 -0
- data/lib/tasks/testing.rake +118 -0
- data/lib/tasks/tmp.rake +37 -0
- data/lib/test_help.rb +28 -0
- data/lib/webrick_server.rb +165 -0
- metadata +356 -0
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
module Rails
|
|
2
|
+
# A class for creating random secret keys. This class will do its best to create a
|
|
3
|
+
# random secret key that's as secure as possible, using whatever methods are
|
|
4
|
+
# available on the current platform. For example:
|
|
5
|
+
#
|
|
6
|
+
# generator = Rails::SecretKeyGenerator("some unique identifier, such as the application name")
|
|
7
|
+
# generator.generate_secret # => "f3f1be90053fa851... (some long string)"
|
|
8
|
+
class SecretKeyGenerator
|
|
9
|
+
GENERATORS = [ :secure_random, :win32_api, :urandom, :openssl, :prng ].freeze
|
|
10
|
+
|
|
11
|
+
def initialize(identifier)
|
|
12
|
+
@identifier = identifier
|
|
13
|
+
end
|
|
14
|
+
|
|
15
|
+
# Generate a random secret key with the best possible method available on
|
|
16
|
+
# the current platform.
|
|
17
|
+
def generate_secret
|
|
18
|
+
generator = GENERATORS.find do |g|
|
|
19
|
+
self.class.send("supports_#{g}?")
|
|
20
|
+
end
|
|
21
|
+
send("generate_secret_with_#{generator}")
|
|
22
|
+
end
|
|
23
|
+
|
|
24
|
+
# Generate a random secret key by using the Win32 API. Raises LoadError
|
|
25
|
+
# if the current platform cannot make use of the Win32 API. Raises
|
|
26
|
+
# SystemCallError if some other error occured.
|
|
27
|
+
def generate_secret_with_win32_api
|
|
28
|
+
# Following code is based on David Garamond's GUID library for Ruby.
|
|
29
|
+
require 'Win32API'
|
|
30
|
+
|
|
31
|
+
crypt_acquire_context = Win32API.new("advapi32", "CryptAcquireContext",
|
|
32
|
+
'PPPII', 'L')
|
|
33
|
+
crypt_gen_random = Win32API.new("advapi32", "CryptGenRandom",
|
|
34
|
+
'LIP', 'L')
|
|
35
|
+
crypt_release_context = Win32API.new("advapi32", "CryptReleaseContext",
|
|
36
|
+
'LI', 'L')
|
|
37
|
+
prov_rsa_full = 1
|
|
38
|
+
crypt_verifycontext = 0xF0000000
|
|
39
|
+
|
|
40
|
+
hProvStr = " " * 4
|
|
41
|
+
if crypt_acquire_context.call(hProvStr, nil, nil, prov_rsa_full,
|
|
42
|
+
crypt_verifycontext) == 0
|
|
43
|
+
raise SystemCallError, "CryptAcquireContext failed: #{lastWin32ErrorMessage}"
|
|
44
|
+
end
|
|
45
|
+
hProv, = hProvStr.unpack('L')
|
|
46
|
+
bytes = " " * 64
|
|
47
|
+
if crypt_gen_random.call(hProv, bytes.size, bytes) == 0
|
|
48
|
+
raise SystemCallError, "CryptGenRandom failed: #{lastWin32ErrorMessage}"
|
|
49
|
+
end
|
|
50
|
+
if crypt_release_context.call(hProv, 0) == 0
|
|
51
|
+
raise SystemCallError, "CryptReleaseContext failed: #{lastWin32ErrorMessage}"
|
|
52
|
+
end
|
|
53
|
+
bytes.unpack("H*")[0]
|
|
54
|
+
end
|
|
55
|
+
|
|
56
|
+
# Generate a random secret key with Ruby 1.9's SecureRandom module.
|
|
57
|
+
# Raises LoadError if the current Ruby version does not support
|
|
58
|
+
# SecureRandom.
|
|
59
|
+
def generate_secret_with_secure_random
|
|
60
|
+
require 'securerandom'
|
|
61
|
+
return SecureRandom.hex(64)
|
|
62
|
+
end
|
|
63
|
+
|
|
64
|
+
# Generate a random secret key with OpenSSL. If OpenSSL is not
|
|
65
|
+
# already loaded, then this method will attempt to load it.
|
|
66
|
+
# LoadError will be raised if that fails.
|
|
67
|
+
def generate_secret_with_openssl
|
|
68
|
+
require 'openssl'
|
|
69
|
+
if !File.exist?("/dev/urandom")
|
|
70
|
+
# OpenSSL transparently seeds the random number generator with
|
|
71
|
+
# data from /dev/urandom. On platforms where that is not
|
|
72
|
+
# available, such as Windows, we have to provide OpenSSL with
|
|
73
|
+
# our own seed. Unfortunately there's no way to provide a
|
|
74
|
+
# secure seed without OS support, so we'll have to do with
|
|
75
|
+
# rand() and Time.now.usec().
|
|
76
|
+
OpenSSL::Random.seed(rand(0).to_s + Time.now.usec.to_s)
|
|
77
|
+
end
|
|
78
|
+
data = OpenSSL::BN.rand(2048, -1, false).to_s
|
|
79
|
+
|
|
80
|
+
if OpenSSL::OPENSSL_VERSION_NUMBER > 0x00908000
|
|
81
|
+
OpenSSL::Digest::SHA512.new(data).hexdigest
|
|
82
|
+
else
|
|
83
|
+
generate_secret_with_prng
|
|
84
|
+
end
|
|
85
|
+
end
|
|
86
|
+
|
|
87
|
+
# Generate a random secret key with /dev/urandom.
|
|
88
|
+
# Raises SystemCallError on failure.
|
|
89
|
+
def generate_secret_with_urandom
|
|
90
|
+
return File.read("/dev/urandom", 64).unpack("H*")[0]
|
|
91
|
+
end
|
|
92
|
+
|
|
93
|
+
# Generate a random secret key with Ruby's pseudo random number generator,
|
|
94
|
+
# as well as some environment information.
|
|
95
|
+
#
|
|
96
|
+
# This is the least cryptographically secure way to generate a secret key,
|
|
97
|
+
# and should be avoided whenever possible.
|
|
98
|
+
def generate_secret_with_prng
|
|
99
|
+
require 'digest/sha2'
|
|
100
|
+
sha = Digest::SHA2.new(512)
|
|
101
|
+
now = Time.now
|
|
102
|
+
sha << now.to_s
|
|
103
|
+
sha << String(now.usec)
|
|
104
|
+
sha << String(rand(0))
|
|
105
|
+
sha << String($$)
|
|
106
|
+
sha << @identifier
|
|
107
|
+
return sha.hexdigest
|
|
108
|
+
end
|
|
109
|
+
|
|
110
|
+
private
|
|
111
|
+
def lastWin32ErrorMessage
|
|
112
|
+
# Following code is based on David Garamond's GUID library for Ruby.
|
|
113
|
+
get_last_error = Win32API.new("kernel32", "GetLastError", '', 'L')
|
|
114
|
+
format_message = Win32API.new("kernel32", "FormatMessageA",
|
|
115
|
+
'LPLLPLPPPPPPPP', 'L')
|
|
116
|
+
format_message_ignore_inserts = 0x00000200
|
|
117
|
+
format_message_from_system = 0x00001000
|
|
118
|
+
|
|
119
|
+
code = get_last_error.call
|
|
120
|
+
msg = "\0" * 1024
|
|
121
|
+
len = format_message.call(format_message_ignore_inserts +
|
|
122
|
+
format_message_from_system, 0,
|
|
123
|
+
code, 0, msg, 1024, nil, nil,
|
|
124
|
+
nil, nil, nil, nil, nil, nil)
|
|
125
|
+
msg[0, len].tr("\r", '').chomp
|
|
126
|
+
end
|
|
127
|
+
|
|
128
|
+
def self.supports_secure_random?
|
|
129
|
+
begin
|
|
130
|
+
require 'securerandom'
|
|
131
|
+
true
|
|
132
|
+
rescue LoadError
|
|
133
|
+
false
|
|
134
|
+
end
|
|
135
|
+
end
|
|
136
|
+
|
|
137
|
+
def self.supports_win32_api?
|
|
138
|
+
return false unless RUBY_PLATFORM =~ /(:?mswin|mingw)/
|
|
139
|
+
begin
|
|
140
|
+
require 'Win32API'
|
|
141
|
+
true
|
|
142
|
+
rescue LoadError
|
|
143
|
+
false
|
|
144
|
+
end
|
|
145
|
+
end
|
|
146
|
+
|
|
147
|
+
def self.supports_urandom?
|
|
148
|
+
File.exist?('/dev/urandom')
|
|
149
|
+
end
|
|
150
|
+
|
|
151
|
+
def self.supports_openssl?
|
|
152
|
+
begin
|
|
153
|
+
require 'openssl'
|
|
154
|
+
true
|
|
155
|
+
rescue LoadError
|
|
156
|
+
false
|
|
157
|
+
end
|
|
158
|
+
end
|
|
159
|
+
|
|
160
|
+
def self.supports_prng?
|
|
161
|
+
true
|
|
162
|
+
end
|
|
163
|
+
end
|
|
164
|
+
end
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
module Rails
|
|
2
|
+
module Generator
|
|
3
|
+
class SimpleLogger # :nodoc:
|
|
4
|
+
attr_reader :out
|
|
5
|
+
attr_accessor :quiet
|
|
6
|
+
|
|
7
|
+
def initialize(out = $stdout)
|
|
8
|
+
@out = out
|
|
9
|
+
@quiet = false
|
|
10
|
+
@level = 0
|
|
11
|
+
end
|
|
12
|
+
|
|
13
|
+
def log(status, message, &block)
|
|
14
|
+
@out.print("%12s %s%s\n" % [status, ' ' * @level, message]) unless quiet
|
|
15
|
+
indent(&block) if block_given?
|
|
16
|
+
end
|
|
17
|
+
|
|
18
|
+
def indent(&block)
|
|
19
|
+
@level += 1
|
|
20
|
+
if block_given?
|
|
21
|
+
begin
|
|
22
|
+
block.call
|
|
23
|
+
ensure
|
|
24
|
+
outdent
|
|
25
|
+
end
|
|
26
|
+
end
|
|
27
|
+
end
|
|
28
|
+
|
|
29
|
+
def outdent
|
|
30
|
+
@level -= 1
|
|
31
|
+
if block_given?
|
|
32
|
+
begin
|
|
33
|
+
block.call
|
|
34
|
+
ensure
|
|
35
|
+
indent
|
|
36
|
+
end
|
|
37
|
+
end
|
|
38
|
+
end
|
|
39
|
+
|
|
40
|
+
private
|
|
41
|
+
def method_missing(method, *args, &block)
|
|
42
|
+
log(method.to_s, args.first, &block)
|
|
43
|
+
end
|
|
44
|
+
end
|
|
45
|
+
end
|
|
46
|
+
end
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
module Rails
|
|
2
|
+
module Generator
|
|
3
|
+
# A spec knows where a generator was found and how to instantiate it.
|
|
4
|
+
# Metadata include the generator's name, its base path, and the source
|
|
5
|
+
# which yielded it (PathSource, GemPathSource, etc.)
|
|
6
|
+
class Spec
|
|
7
|
+
attr_reader :name, :path, :source
|
|
8
|
+
|
|
9
|
+
def initialize(name, path, source)
|
|
10
|
+
@name, @path, @source = name, path, source
|
|
11
|
+
end
|
|
12
|
+
|
|
13
|
+
# Look up the generator class. Require its class file, find the class
|
|
14
|
+
# in ObjectSpace, tag it with this spec, and return.
|
|
15
|
+
def klass
|
|
16
|
+
unless @klass
|
|
17
|
+
require class_file
|
|
18
|
+
@klass = lookup_class
|
|
19
|
+
@klass.spec = self
|
|
20
|
+
end
|
|
21
|
+
@klass
|
|
22
|
+
end
|
|
23
|
+
|
|
24
|
+
def class_file
|
|
25
|
+
"#{path}/#{name}_generator.rb"
|
|
26
|
+
end
|
|
27
|
+
|
|
28
|
+
def class_name
|
|
29
|
+
"#{name.camelize}Generator"
|
|
30
|
+
end
|
|
31
|
+
|
|
32
|
+
private
|
|
33
|
+
# Search for the first Class descending from Rails::Generator::Base
|
|
34
|
+
# whose name matches the requested class name.
|
|
35
|
+
def lookup_class
|
|
36
|
+
ObjectSpace.each_object(Class) do |obj|
|
|
37
|
+
return obj if obj.ancestors.include?(Rails::Generator::Base) and
|
|
38
|
+
obj.name.split('::').last == class_name
|
|
39
|
+
end
|
|
40
|
+
raise NameError, "Missing #{class_name} class in #{class_file}"
|
|
41
|
+
end
|
|
42
|
+
end
|
|
43
|
+
end
|
|
44
|
+
end
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
#--
|
|
2
|
+
# Copyright (c) 2004 Jeremy Kemper
|
|
3
|
+
#
|
|
4
|
+
# Permission is hereby granted, free of charge, to any person obtaining
|
|
5
|
+
# a copy of this software and associated documentation files (the
|
|
6
|
+
# "Software"), to deal in the Software without restriction, including
|
|
7
|
+
# without limitation the rights to use, copy, modify, merge, publish,
|
|
8
|
+
# distribute, sublicense, and/or sell copies of the Software, and to
|
|
9
|
+
# permit persons to whom the Software is furnished to do so, subject to
|
|
10
|
+
# the following conditions:
|
|
11
|
+
#
|
|
12
|
+
# The above copyright notice and this permission notice shall be
|
|
13
|
+
# included in all copies or substantial portions of the Software.
|
|
14
|
+
#
|
|
15
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
16
|
+
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
17
|
+
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
18
|
+
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
19
|
+
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
20
|
+
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
21
|
+
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
22
|
+
#++
|
|
23
|
+
|
|
24
|
+
$:.unshift(File.dirname(__FILE__))
|
|
25
|
+
$:.unshift(File.dirname(__FILE__) + "/../../activesupport/lib")
|
|
26
|
+
|
|
27
|
+
begin
|
|
28
|
+
require 'active_support'
|
|
29
|
+
rescue LoadError
|
|
30
|
+
require 'rubygems'
|
|
31
|
+
gem 'activesupport'
|
|
32
|
+
end
|
|
33
|
+
|
|
34
|
+
require 'rails_generator/base'
|
|
35
|
+
require 'rails_generator/lookup'
|
|
36
|
+
require 'rails_generator/commands'
|
|
37
|
+
|
|
38
|
+
Rails::Generator::Base.send(:include, Rails::Generator::Lookup)
|
|
39
|
+
Rails::Generator::Base.send(:include, Rails::Generator::Commands)
|
|
40
|
+
|
|
41
|
+
# Set up a default logger for convenience.
|
|
42
|
+
require 'rails_generator/simple_logger'
|
|
43
|
+
Rails::Generator::Base.logger = Rails::Generator::SimpleLogger.new(STDOUT)
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
RAILTIES_PATH = File.join(File.dirname(__FILE__), '..')
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
min_release = "1.8.2 (2004-12-25)"
|
|
2
|
+
ruby_release = "#{RUBY_VERSION} (#{RUBY_RELEASE_DATE})"
|
|
3
|
+
if ruby_release =~ /1\.8\.3/
|
|
4
|
+
abort <<-end_message
|
|
5
|
+
|
|
6
|
+
Rails does not work with Ruby version 1.8.3.
|
|
7
|
+
Please upgrade to version 1.8.4 or downgrade to 1.8.2.
|
|
8
|
+
|
|
9
|
+
end_message
|
|
10
|
+
elsif ruby_release < min_release
|
|
11
|
+
abort <<-end_message
|
|
12
|
+
|
|
13
|
+
Rails requires Ruby version #{min_release} or later.
|
|
14
|
+
You're running #{ruby_release}; please upgrade to continue.
|
|
15
|
+
|
|
16
|
+
end_message
|
|
17
|
+
end
|
data/lib/rubyprof_ext.rb
ADDED
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
require 'prof'
|
|
2
|
+
|
|
3
|
+
module Prof #:nodoc:
|
|
4
|
+
# Adapted from Shugo Maeda's unprof.rb
|
|
5
|
+
def self.print_profile(results, io = $stderr)
|
|
6
|
+
total = results.detect { |i|
|
|
7
|
+
i.method_class.nil? && i.method_id == :"#toplevel"
|
|
8
|
+
}.total_time
|
|
9
|
+
total = 0.001 if total < 0.001
|
|
10
|
+
|
|
11
|
+
io.puts " %% cumulative self self total"
|
|
12
|
+
io.puts " time seconds seconds calls ms/call ms/call name"
|
|
13
|
+
|
|
14
|
+
sum = 0.0
|
|
15
|
+
for r in results
|
|
16
|
+
sum += r.self_time
|
|
17
|
+
|
|
18
|
+
name = if r.method_class.nil?
|
|
19
|
+
r.method_id.to_s
|
|
20
|
+
elsif r.method_class.is_a?(Class)
|
|
21
|
+
"#{r.method_class}##{r.method_id}"
|
|
22
|
+
else
|
|
23
|
+
"#{r.method_class}.#{r.method_id}"
|
|
24
|
+
end
|
|
25
|
+
io.printf "%6.2f %8.3f %8.3f %8d %8.2f %8.2f %s\n",
|
|
26
|
+
r.self_time / total * 100,
|
|
27
|
+
sum,
|
|
28
|
+
r.self_time,
|
|
29
|
+
r.count,
|
|
30
|
+
r.self_time * 1000 / r.count,
|
|
31
|
+
r.total_time * 1000 / r.count,
|
|
32
|
+
name
|
|
33
|
+
end
|
|
34
|
+
end
|
|
35
|
+
end
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
# Implements the logic behind the rake tasks for annotations like
|
|
2
|
+
#
|
|
3
|
+
# rake notes
|
|
4
|
+
# rake notes:optimize
|
|
5
|
+
#
|
|
6
|
+
# and friends. See <tt>rake -T notes</tt> and <tt>erailties/lib/tasks/annotations.rake</tt>.
|
|
7
|
+
#
|
|
8
|
+
# Annotation objects are triplets <tt>:line</tt>, <tt>:tag</tt>, <tt>:text</tt> that
|
|
9
|
+
# represent the line where the annotation lives, its tag, and its text. Note
|
|
10
|
+
# the filename is not stored.
|
|
11
|
+
#
|
|
12
|
+
# Annotations are looked for in comments and modulus whitespace they have to
|
|
13
|
+
# start with the tag optionally followed by a colon. Everything up to the end
|
|
14
|
+
# of the line (or closing ERb comment tag) is considered to be their text.
|
|
15
|
+
class SourceAnnotationExtractor
|
|
16
|
+
class Annotation < Struct.new(:line, :tag, :text)
|
|
17
|
+
|
|
18
|
+
# Returns a representation of the annotation that looks like this:
|
|
19
|
+
#
|
|
20
|
+
# [126] [TODO] This algorithm is simple and clearly correct, make it faster.
|
|
21
|
+
#
|
|
22
|
+
# If +options+ has a flag <tt>:tag</tt> the tag is shown as in the example above.
|
|
23
|
+
# Otherwise the string contains just line and text.
|
|
24
|
+
def to_s(options={})
|
|
25
|
+
s = "[%3d] " % line
|
|
26
|
+
s << "[#{tag}] " if options[:tag]
|
|
27
|
+
s << text
|
|
28
|
+
end
|
|
29
|
+
end
|
|
30
|
+
|
|
31
|
+
# Prints all annotations with tag +tag+ under the root directories +app+, +lib+,
|
|
32
|
+
# and +test+ (recursively). Only filenames with extension +.builder+, +.rb+,
|
|
33
|
+
# +.rxml+, +.rhtml+, or +.erb+ are taken into account. The +options+
|
|
34
|
+
# hash is passed to each annotation's +to_s+.
|
|
35
|
+
#
|
|
36
|
+
# This class method is the single entry point for the rake tasks.
|
|
37
|
+
def self.enumerate(tag, options={})
|
|
38
|
+
extractor = new(tag)
|
|
39
|
+
extractor.display(extractor.find, options)
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
attr_reader :tag
|
|
43
|
+
|
|
44
|
+
def initialize(tag)
|
|
45
|
+
@tag = tag
|
|
46
|
+
end
|
|
47
|
+
|
|
48
|
+
# Returns a hash that maps filenames under +dirs+ (recursively) to arrays
|
|
49
|
+
# with their annotations. Only files with annotations are included, and only
|
|
50
|
+
# those with extension +.builder+, +.rb+, +.rxml+, +.rhtml+, and +.erb+
|
|
51
|
+
# are taken into account.
|
|
52
|
+
def find(dirs=%w(app lib test))
|
|
53
|
+
dirs.inject({}) { |h, dir| h.update(find_in(dir)) }
|
|
54
|
+
end
|
|
55
|
+
|
|
56
|
+
# Returns a hash that maps filenames under +dir+ (recursively) to arrays
|
|
57
|
+
# with their annotations. Only files with annotations are included, and only
|
|
58
|
+
# those with extension +.builder+, +.rb+, +.rxml+, +.rhtml+, and +.erb+
|
|
59
|
+
# are taken into account.
|
|
60
|
+
def find_in(dir)
|
|
61
|
+
results = {}
|
|
62
|
+
|
|
63
|
+
Dir.glob("#{dir}/*") do |item|
|
|
64
|
+
next if File.basename(item)[0] == ?.
|
|
65
|
+
|
|
66
|
+
if File.directory?(item)
|
|
67
|
+
results.update(find_in(item))
|
|
68
|
+
elsif item =~ /\.(builder|(r(?:b|xml|js)))$/
|
|
69
|
+
results.update(extract_annotations_from(item, /#\s*(#{tag}):?\s*(.*)$/))
|
|
70
|
+
elsif item =~ /\.(rhtml|erb)$/
|
|
71
|
+
results.update(extract_annotations_from(item, /<%\s*#\s*(#{tag}):?\s*(.*?)\s*%>/))
|
|
72
|
+
end
|
|
73
|
+
end
|
|
74
|
+
|
|
75
|
+
results
|
|
76
|
+
end
|
|
77
|
+
|
|
78
|
+
# If +file+ is the filename of a file that contains annotations this method returns
|
|
79
|
+
# a hash with a single entry that maps +file+ to an array of its annotations.
|
|
80
|
+
# Otherwise it returns an empty hash.
|
|
81
|
+
def extract_annotations_from(file, pattern)
|
|
82
|
+
lineno = 0
|
|
83
|
+
result = File.readlines(file).inject([]) do |list, line|
|
|
84
|
+
lineno += 1
|
|
85
|
+
next list unless line =~ pattern
|
|
86
|
+
list << Annotation.new(lineno, $1, $2)
|
|
87
|
+
end
|
|
88
|
+
result.empty? ? {} : { file => result }
|
|
89
|
+
end
|
|
90
|
+
|
|
91
|
+
# Prints the mapping from filenames to annotations in +results+ ordered by filename.
|
|
92
|
+
# The +options+ hash is passed to each annotation's +to_s+.
|
|
93
|
+
def display(results, options={})
|
|
94
|
+
results.keys.sort.each do |file|
|
|
95
|
+
puts "#{file}:"
|
|
96
|
+
results[file].each do |note|
|
|
97
|
+
puts " * #{note.to_s(options)}"
|
|
98
|
+
end
|
|
99
|
+
puts
|
|
100
|
+
end
|
|
101
|
+
end
|
|
102
|
+
end
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
require 'source_annotation_extractor'
|
|
2
|
+
|
|
3
|
+
desc "Enumerate all annotations"
|
|
4
|
+
task :notes do
|
|
5
|
+
SourceAnnotationExtractor.enumerate "OPTIMIZE|FIXME|TODO", :tag => true
|
|
6
|
+
end
|
|
7
|
+
|
|
8
|
+
namespace :notes do
|
|
9
|
+
desc "Enumerate all OPTIMIZE annotations"
|
|
10
|
+
task :optimize do
|
|
11
|
+
SourceAnnotationExtractor.enumerate "OPTIMIZE"
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
desc "Enumerate all FIXME annotations"
|
|
15
|
+
task :fixme do
|
|
16
|
+
SourceAnnotationExtractor.enumerate "FIXME"
|
|
17
|
+
end
|
|
18
|
+
|
|
19
|
+
desc "Enumerate all TODO annotations"
|
|
20
|
+
task :todo do
|
|
21
|
+
SourceAnnotationExtractor.enumerate "TODO"
|
|
22
|
+
end
|
|
23
|
+
end
|