railties 6.1.6 → 7.1.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (245) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +626 -259
  3. data/MIT-LICENSE +1 -1
  4. data/RDOC_MAIN.md +99 -0
  5. data/README.rdoc +5 -6
  6. data/lib/minitest/rails_plugin.rb +64 -1
  7. data/lib/rails/all.rb +0 -1
  8. data/lib/rails/api/task.rb +35 -4
  9. data/lib/rails/app_updater.rb +9 -6
  10. data/lib/rails/application/bootstrap.rb +41 -7
  11. data/lib/rails/application/configuration.rb +262 -88
  12. data/lib/rails/application/default_middleware_stack.rb +29 -4
  13. data/lib/rails/application/dummy_config.rb +19 -0
  14. data/lib/rails/application/finisher.rb +89 -124
  15. data/lib/rails/application/routes_reloader.rb +8 -0
  16. data/lib/rails/application.rb +200 -125
  17. data/lib/rails/application_controller.rb +3 -3
  18. data/lib/rails/autoloaders/inflector.rb +21 -0
  19. data/lib/rails/autoloaders.rb +42 -42
  20. data/lib/rails/backtrace_cleaner.rb +1 -1
  21. data/lib/rails/cli.rb +5 -2
  22. data/lib/rails/code_statistics.rb +2 -2
  23. data/lib/rails/code_statistics_calculator.rb +10 -1
  24. data/lib/rails/command/actions.rb +10 -12
  25. data/lib/rails/command/base.rb +61 -45
  26. data/lib/rails/command/behavior.rb +1 -1
  27. data/lib/rails/command/environment_argument.rb +33 -17
  28. data/lib/rails/command/helpers/editor.rb +17 -12
  29. data/lib/rails/command.rb +90 -36
  30. data/lib/rails/commands/about/about_command.rb +14 -0
  31. data/lib/rails/commands/application/application_command.rb +2 -0
  32. data/lib/rails/commands/console/console_command.rb +17 -13
  33. data/lib/rails/commands/credentials/USAGE +53 -53
  34. data/lib/rails/commands/credentials/credentials_command/diffing.rb +31 -19
  35. data/lib/rails/commands/credentials/credentials_command.rb +67 -69
  36. data/lib/rails/commands/db/system/change/change_command.rb +2 -1
  37. data/lib/rails/commands/dbconsole/dbconsole_command.rb +26 -117
  38. data/lib/rails/commands/destroy/destroy_command.rb +3 -2
  39. data/lib/rails/commands/dev/dev_command.rb +1 -6
  40. data/lib/rails/commands/encrypted/USAGE +15 -20
  41. data/lib/rails/commands/encrypted/encrypted_command.rb +46 -35
  42. data/lib/rails/commands/gem_help/USAGE +16 -0
  43. data/lib/rails/commands/gem_help/gem_help_command.rb +13 -0
  44. data/lib/rails/commands/generate/generate_command.rb +2 -2
  45. data/lib/rails/commands/help/USAGE +15 -14
  46. data/lib/rails/commands/help/help_command.rb +21 -2
  47. data/lib/rails/commands/initializers/initializers_command.rb +1 -4
  48. data/lib/rails/commands/middleware/middleware_command.rb +17 -0
  49. data/lib/rails/commands/new/new_command.rb +2 -0
  50. data/lib/rails/commands/notes/notes_command.rb +2 -1
  51. data/lib/rails/commands/plugin/plugin_command.rb +2 -0
  52. data/lib/rails/commands/rake/rake_command.rb +25 -22
  53. data/lib/rails/commands/restart/restart_command.rb +14 -0
  54. data/lib/rails/commands/routes/routes_command.rb +13 -1
  55. data/lib/rails/commands/runner/USAGE +14 -12
  56. data/lib/rails/commands/runner/runner_command.rb +34 -21
  57. data/lib/rails/commands/secret/secret_command.rb +13 -0
  58. data/lib/rails/commands/secrets/USAGE +44 -49
  59. data/lib/rails/commands/secrets/secrets_command.rb +20 -38
  60. data/lib/rails/commands/server/server_command.rb +33 -30
  61. data/lib/rails/commands/test/USAGE +14 -0
  62. data/lib/rails/commands/test/test_command.rb +56 -14
  63. data/lib/rails/commands/unused_routes/unused_routes_command.rb +75 -0
  64. data/lib/rails/commands/version/version_command.rb +1 -0
  65. data/lib/rails/configuration.rb +23 -28
  66. data/lib/rails/console/app.rb +1 -4
  67. data/lib/rails/console/helpers.rb +2 -2
  68. data/lib/rails/deprecator.rb +7 -0
  69. data/lib/rails/engine/configuration.rb +53 -9
  70. data/lib/rails/engine.rb +63 -48
  71. data/lib/rails/gem_version.rb +3 -3
  72. data/lib/rails/generators/actions/create_migration.rb +2 -4
  73. data/lib/rails/generators/actions.rb +231 -73
  74. data/lib/rails/generators/active_model.rb +28 -14
  75. data/lib/rails/generators/app_base.rb +476 -176
  76. data/lib/rails/generators/app_name.rb +3 -14
  77. data/lib/rails/generators/base.rb +24 -20
  78. data/lib/rails/generators/database.rb +39 -1
  79. data/lib/rails/generators/erb/mailer/templates/layout.html.erb.tt +1 -1
  80. data/lib/rails/generators/erb/scaffold/scaffold_generator.rb +2 -0
  81. data/lib/rails/generators/erb/scaffold/templates/_form.html.erb.tt +8 -8
  82. data/lib/rails/generators/erb/scaffold/templates/edit.html.erb.tt +8 -4
  83. data/lib/rails/generators/erb/scaffold/templates/index.html.erb.tt +11 -28
  84. data/lib/rails/generators/erb/scaffold/templates/new.html.erb.tt +7 -3
  85. data/lib/rails/generators/erb/scaffold/templates/partial.html.erb.tt +17 -0
  86. data/lib/rails/generators/erb/scaffold/templates/show.html.erb.tt +8 -17
  87. data/lib/rails/generators/erb.rb +1 -1
  88. data/lib/rails/generators/generated_attribute.rb +57 -9
  89. data/lib/rails/generators/migration.rb +2 -7
  90. data/lib/rails/generators/model_helpers.rb +3 -2
  91. data/lib/rails/generators/named_base.rb +13 -13
  92. data/lib/rails/generators/rails/app/USAGE +22 -6
  93. data/lib/rails/generators/rails/app/app_generator.rb +113 -114
  94. data/lib/rails/generators/rails/app/templates/Dockerfile.tt +103 -0
  95. data/lib/rails/generators/rails/app/templates/Gemfile.tt +43 -53
  96. data/lib/rails/generators/rails/app/templates/app/assets/stylesheets/application.css.tt +2 -2
  97. data/lib/rails/generators/rails/app/templates/app/mailers/application_mailer.rb.tt +2 -2
  98. data/lib/rails/generators/rails/app/templates/app/models/application_record.rb.tt +1 -1
  99. data/lib/rails/generators/rails/app/templates/app/views/layouts/application.html.erb.tt +3 -9
  100. data/lib/rails/generators/rails/app/templates/app/views/layouts/mailer.html.erb.tt +1 -1
  101. data/lib/rails/generators/rails/app/templates/bin/rails.tt +1 -4
  102. data/lib/rails/generators/rails/app/templates/bin/rake.tt +0 -3
  103. data/lib/rails/generators/rails/app/templates/bin/setup.tt +16 -12
  104. data/lib/rails/generators/rails/app/templates/config/application.rb.tt +6 -18
  105. data/lib/rails/generators/rails/app/templates/config/boot.rb.tt +1 -1
  106. data/lib/rails/generators/rails/app/templates/config/databases/jdbc.yml.tt +2 -2
  107. data/lib/rails/generators/rails/app/templates/config/databases/jdbcmysql.yml.tt +3 -3
  108. data/lib/rails/generators/rails/app/templates/config/databases/jdbcpostgresql.yml.tt +7 -7
  109. data/lib/rails/generators/rails/app/templates/config/databases/jdbcsqlite3.yml.tt +5 -5
  110. data/lib/rails/generators/rails/app/templates/config/databases/mysql.yml.tt +3 -3
  111. data/lib/rails/generators/rails/app/templates/config/databases/oracle.yml.tt +2 -2
  112. data/lib/rails/generators/rails/app/templates/config/databases/postgresql.yml.tt +7 -9
  113. data/lib/rails/generators/rails/app/templates/config/databases/sqlite3.yml.tt +4 -4
  114. data/lib/rails/generators/rails/app/templates/config/databases/sqlserver.yml.tt +5 -5
  115. data/lib/rails/generators/rails/app/templates/config/databases/trilogy.yml.tt +59 -0
  116. data/lib/rails/generators/rails/app/templates/config/environments/development.rb.tt +16 -12
  117. data/lib/rails/generators/rails/app/templates/config/environments/production.rb.tt +38 -61
  118. data/lib/rails/generators/rails/app/templates/config/environments/test.rb.tt +14 -15
  119. data/lib/rails/generators/rails/app/templates/config/initializers/assets.rb.tt +3 -5
  120. data/lib/rails/generators/rails/app/templates/config/initializers/content_security_policy.rb.tt +21 -28
  121. data/lib/rails/generators/rails/app/templates/config/initializers/cors.rb.tt +3 -3
  122. data/lib/rails/generators/rails/app/templates/config/initializers/filter_parameter_logging.rb.tt +3 -1
  123. data/lib/rails/generators/rails/app/templates/config/initializers/inflections.rb.tt +4 -4
  124. data/lib/rails/generators/rails/app/templates/config/initializers/new_framework_defaults_7_1.rb.tt +283 -0
  125. data/lib/rails/generators/rails/app/templates/config/initializers/permissions_policy.rb.tt +11 -9
  126. data/lib/rails/generators/rails/app/templates/config/locales/en.yml +13 -15
  127. data/lib/rails/generators/rails/app/templates/config/puma.rb.tt +12 -20
  128. data/lib/rails/generators/rails/app/templates/config/routes.rb.tt +8 -1
  129. data/lib/rails/generators/rails/app/templates/config/storage.yml.tt +5 -5
  130. data/lib/rails/generators/rails/app/templates/db/seeds.rb.tt +6 -4
  131. data/lib/rails/generators/rails/app/templates/docker-entrypoint.tt +10 -0
  132. data/lib/rails/generators/rails/app/templates/dockerignore.tt +43 -0
  133. data/lib/rails/generators/rails/app/templates/gitattributes.tt +0 -5
  134. data/lib/rails/generators/rails/app/templates/gitignore.tt +7 -9
  135. data/lib/rails/generators/rails/app/templates/node-version.tt +1 -0
  136. data/lib/rails/generators/rails/app/templates/test/channels/application_cable/connection_test.rb.tt +10 -8
  137. data/lib/rails/generators/rails/app/templates/test/test_helper.rb.tt +10 -8
  138. data/lib/rails/generators/rails/application_record/application_record_generator.rb +4 -0
  139. data/lib/rails/generators/rails/benchmark/benchmark_generator.rb +2 -1
  140. data/lib/rails/generators/rails/controller/USAGE +12 -4
  141. data/lib/rails/generators/rails/controller/controller_generator.rb +6 -2
  142. data/lib/rails/generators/rails/controller/templates/controller.rb.tt +1 -5
  143. data/lib/rails/generators/rails/credentials/credentials_generator.rb +29 -24
  144. data/lib/rails/generators/rails/credentials/templates/credentials.yml.tt +8 -0
  145. data/lib/rails/generators/rails/db/system/change/change_generator.rb +31 -1
  146. data/lib/rails/generators/rails/encryption_key_file/encryption_key_file_generator.rb +1 -2
  147. data/lib/rails/generators/rails/generator/templates/%file_name%_generator.rb.tt +1 -1
  148. data/lib/rails/generators/rails/migration/USAGE +21 -11
  149. data/lib/rails/generators/rails/model/model_generator.rb +4 -0
  150. data/lib/rails/generators/rails/plugin/USAGE +17 -6
  151. data/lib/rails/generators/rails/plugin/plugin_generator.rb +69 -30
  152. data/lib/rails/generators/rails/plugin/templates/%name%.gemspec.tt +7 -3
  153. data/lib/rails/generators/rails/plugin/templates/Gemfile.tt +9 -31
  154. data/lib/rails/generators/rails/plugin/templates/MIT-LICENSE.tt +1 -1
  155. data/lib/rails/generators/rails/plugin/templates/README.md.tt +1 -1
  156. data/lib/rails/generators/rails/plugin/templates/app/mailers/%namespaced_name%/application_mailer.rb.tt +2 -2
  157. data/lib/rails/generators/rails/plugin/templates/app/views/layouts/%namespaced_name%/application.html.erb.tt +0 -3
  158. data/lib/rails/generators/rails/plugin/templates/bin/rails.tt +5 -20
  159. data/lib/rails/generators/rails/plugin/templates/gitignore.tt +0 -7
  160. data/lib/rails/generators/rails/plugin/templates/lib/%namespaced_name%/engine.rb.tt +2 -2
  161. data/lib/rails/generators/rails/plugin/templates/lib/%namespaced_name%/version.rb.tt +1 -1
  162. data/lib/rails/generators/rails/plugin/templates/rails/boot.rb.tt +2 -2
  163. data/lib/rails/generators/rails/plugin/templates/test/test_helper.rb.tt +6 -11
  164. data/lib/rails/generators/rails/resource/resource_generator.rb +6 -0
  165. data/lib/rails/generators/rails/scaffold/scaffold_generator.rb +2 -20
  166. data/lib/rails/generators/rails/scaffold_controller/scaffold_controller_generator.rb +1 -1
  167. data/lib/rails/generators/rails/scaffold_controller/templates/api_controller.rb.tt +1 -5
  168. data/lib/rails/generators/rails/scaffold_controller/templates/controller.rb.tt +6 -10
  169. data/lib/rails/generators/resource_helpers.rb +2 -2
  170. data/lib/rails/generators/test_case.rb +2 -2
  171. data/lib/rails/generators/test_unit/generator/templates/generator_test.rb.tt +1 -1
  172. data/lib/rails/generators/test_unit/model/templates/fixtures.yml.tt +2 -2
  173. data/lib/rails/generators/test_unit/scaffold/scaffold_generator.rb +4 -4
  174. data/lib/rails/generators/test_unit/scaffold/templates/api_functional_test.rb.tt +5 -5
  175. data/lib/rails/generators/test_unit/scaffold/templates/functional_test.rb.tt +7 -7
  176. data/lib/rails/generators/test_unit/scaffold/templates/system_test.rb.tt +9 -11
  177. data/lib/rails/generators/testing/{behaviour.rb → behavior.rb} +7 -5
  178. data/lib/rails/generators.rb +14 -29
  179. data/lib/rails/health_controller.rb +55 -0
  180. data/lib/rails/info.rb +2 -2
  181. data/lib/rails/info_controller.rb +31 -13
  182. data/lib/rails/initializable.rb +1 -1
  183. data/lib/rails/mailers_controller.rb +17 -9
  184. data/lib/rails/paths.rb +14 -11
  185. data/lib/rails/rack/logger.rb +16 -15
  186. data/lib/rails/rackup/server.rb +15 -0
  187. data/lib/rails/railtie/configuration.rb +15 -3
  188. data/lib/rails/railtie.rb +54 -31
  189. data/lib/rails/ruby_version_check.rb +5 -3
  190. data/lib/rails/secrets.rb +10 -8
  191. data/lib/rails/source_annotation_extractor.rb +68 -19
  192. data/lib/rails/tasks/engine.rake +8 -8
  193. data/lib/rails/tasks/framework.rake +4 -12
  194. data/lib/rails/tasks/log.rake +1 -1
  195. data/lib/rails/tasks/misc.rake +3 -14
  196. data/lib/rails/tasks/statistics.rake +8 -5
  197. data/lib/rails/tasks/tmp.rake +13 -6
  198. data/lib/rails/tasks/yarn.rake +8 -9
  199. data/lib/rails/tasks/zeitwerk.rake +14 -42
  200. data/lib/rails/tasks.rb +0 -2
  201. data/lib/rails/templates/layouts/application.html.erb +15 -0
  202. data/lib/rails/templates/rails/mailers/email.html.erb +45 -11
  203. data/lib/rails/templates/rails/mailers/index.html.erb +14 -7
  204. data/lib/rails/templates/rails/mailers/mailer.html.erb +11 -5
  205. data/lib/rails/templates/rails/welcome/index.html.erb +65 -48
  206. data/lib/rails/test_help.rb +13 -14
  207. data/lib/rails/test_unit/line_filtering.rb +1 -1
  208. data/lib/rails/test_unit/railtie.rb +0 -4
  209. data/lib/rails/test_unit/reporter.rb +6 -2
  210. data/lib/rails/test_unit/runner.rb +45 -20
  211. data/lib/rails/test_unit/test_parser.rb +88 -0
  212. data/lib/rails/test_unit/testing.rake +18 -43
  213. data/lib/rails/testing/maintain_test_schema.rb +16 -0
  214. data/lib/rails/version.rb +1 -1
  215. data/lib/rails/welcome_controller.rb +1 -0
  216. data/lib/rails/zeitwerk_checker.rb +15 -0
  217. data/lib/rails.rb +26 -15
  218. metadata +74 -42
  219. data/RDOC_MAIN.rdoc +0 -97
  220. data/lib/rails/.DS_Store +0 -0
  221. data/lib/rails/application/dummy_erb_compiler.rb +0 -18
  222. data/lib/rails/command/spellchecker.rb +0 -57
  223. data/lib/rails/generators/css/assets/assets_generator.rb +0 -15
  224. data/lib/rails/generators/css/assets/templates/stylesheet.css +0 -4
  225. data/lib/rails/generators/css/scaffold/scaffold_generator.rb +0 -18
  226. data/lib/rails/generators/rails/app/templates/app/javascript/channels/consumer.js +0 -6
  227. data/lib/rails/generators/rails/app/templates/app/javascript/channels/index.js +0 -5
  228. data/lib/rails/generators/rails/app/templates/app/javascript/packs/application.js.tt +0 -23
  229. data/lib/rails/generators/rails/app/templates/bin/spring.tt +0 -13
  230. data/lib/rails/generators/rails/app/templates/bin/yarn.tt +0 -16
  231. data/lib/rails/generators/rails/app/templates/config/initializers/application_controller_renderer.rb.tt +0 -8
  232. data/lib/rails/generators/rails/app/templates/config/initializers/backtrace_silencers.rb.tt +0 -8
  233. data/lib/rails/generators/rails/app/templates/config/initializers/cookies_serializer.rb.tt +0 -5
  234. data/lib/rails/generators/rails/app/templates/config/initializers/mime_types.rb.tt +0 -4
  235. data/lib/rails/generators/rails/app/templates/config/initializers/new_framework_defaults_6_1.rb.tt +0 -67
  236. data/lib/rails/generators/rails/app/templates/config/initializers/wrap_parameters.rb.tt +0 -16
  237. data/lib/rails/generators/rails/app/templates/config/spring.rb.tt +0 -6
  238. data/lib/rails/generators/rails/app/templates/package.json.tt +0 -11
  239. data/lib/rails/generators/rails/assets/USAGE +0 -16
  240. data/lib/rails/generators/rails/assets/assets_generator.rb +0 -26
  241. data/lib/rails/generators/rails/assets/templates/stylesheet.css +0 -4
  242. data/lib/rails/generators/rails/model/USAGE +0 -113
  243. data/lib/rails/generators/rails/scaffold/templates/scaffold.css +0 -80
  244. data/lib/rails/tasks/middleware.rake +0 -9
  245. data/lib/rails/tasks/restart.rake +0 -9
@@ -1,7 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  require "pathname"
4
- require "shellwords"
5
4
  require "active_support"
6
5
  require "rails/command/helpers/editor"
7
6
  require "rails/command/environment_argument"
@@ -15,53 +14,43 @@ module Rails
15
14
  require_relative "credentials_command/diffing"
16
15
  include Diffing
17
16
 
18
- self.environment_desc = "Uses credentials from config/credentials/:environment.yml.enc encrypted by config/credentials/:environment.key key"
19
-
20
- no_commands do
21
- def help
22
- say "Usage:\n #{self.class.banner}"
23
- say ""
24
- say self.class.desc
25
- end
26
- end
27
-
17
+ desc "edit", "Open the decrypted credentials in `$VISUAL` or `$EDITOR` for editing"
28
18
  def edit
29
- extract_environment_option_from_argument(default_environment: nil)
30
- require_application!
19
+ load_environment_config!
20
+ load_generators
31
21
 
32
- ensure_editor_available(command: "bin/rails credentials:edit") || (return)
22
+ if environment_specified?
23
+ @content_path = "config/credentials/#{environment}.yml.enc" unless config.key?(:content_path)
24
+ @key_path = "config/credentials/#{environment}.key" unless config.key?(:key_path)
25
+ end
33
26
 
34
- ensure_encryption_key_has_been_added if credentials.key.nil?
27
+ ensure_encryption_key_has_been_added
35
28
  ensure_credentials_have_been_added
36
- ensure_rails_credentials_driver_is_set
29
+ ensure_diffing_driver_is_configured
37
30
 
38
- catch_editing_exceptions do
39
- change_credentials_in_system_editor
40
- end
41
-
42
- say "File encrypted and saved."
43
- rescue ActiveSupport::MessageEncryptor::InvalidMessage
44
- say "Couldn't decrypt #{content_path}. Perhaps you passed the wrong key?"
31
+ change_credentials_in_system_editor
45
32
  end
46
33
 
34
+ desc "show", "Show the decrypted credentials"
47
35
  def show
48
- extract_environment_option_from_argument(default_environment: nil)
49
- require_application!
36
+ load_environment_config!
50
37
 
51
38
  say credentials.read.presence || missing_credentials_message
52
39
  end
53
40
 
41
+ desc "diff", "Enroll/disenroll in decrypted diffs of credentials using git"
54
42
  option :enroll, type: :boolean, default: false,
55
- desc: "Enrolls project in credential file diffing with `git diff`"
56
-
43
+ desc: "Enroll project in credentials file diffing with `git diff`"
44
+ option :disenroll, type: :boolean, default: false,
45
+ desc: "Disenroll project from credentials file diffing"
57
46
  def diff(content_path = nil)
58
47
  if @content_path = content_path
59
- extract_environment_option_from_argument(default_environment: extract_environment_from_path(content_path))
60
- require_application!
48
+ self.environment = extract_environment_from_path(content_path)
49
+ load_environment_config!
61
50
 
62
51
  say credentials.read.presence || credentials.content_path.read
63
52
  else
64
- require_application!
53
+ disenroll_project_from_credentials_diffing if options[:disenroll]
65
54
  enroll_project_in_credentials_diffing if options[:enroll]
66
55
  end
67
56
  rescue ActiveSupport::MessageEncryptor::InvalidMessage
@@ -69,68 +58,77 @@ module Rails
69
58
  end
70
59
 
71
60
  private
61
+ def config
62
+ Rails.application.config.credentials
63
+ end
64
+
65
+ def content_path
66
+ @content_path ||= relative_path(config.content_path)
67
+ end
68
+
69
+ def key_path
70
+ @key_path ||= relative_path(config.key_path)
71
+ end
72
+
72
73
  def credentials
73
- Rails.application.encrypted(content_path, key_path: key_path)
74
+ @credentials ||= Rails.application.encrypted(content_path, key_path: key_path)
74
75
  end
75
76
 
76
77
  def ensure_encryption_key_has_been_added
78
+ return if credentials.key?
79
+
80
+ require "rails/generators/rails/encryption_key_file/encryption_key_file_generator"
81
+
82
+ encryption_key_file_generator = Rails::Generators::EncryptionKeyFileGenerator.new
77
83
  encryption_key_file_generator.add_key_file(key_path)
78
84
  encryption_key_file_generator.ignore_key_file(key_path)
79
85
  end
80
86
 
81
87
  def ensure_credentials_have_been_added
82
- if options[:environment]
83
- encrypted_file_generator.add_encrypted_file_silently(content_path, key_path)
84
- else
85
- credentials_generator.add_credentials_file_silently
86
- end
88
+ require "rails/generators/rails/credentials/credentials_generator"
89
+
90
+ Rails::Generators::CredentialsGenerator.new(
91
+ [content_path, key_path],
92
+ skip_secret_key_base: environment_specified? && %w[development test].include?(environment),
93
+ quiet: true
94
+ ).invoke_all
87
95
  end
88
96
 
89
97
  def change_credentials_in_system_editor
90
- credentials.change do |tmp_path|
91
- system("#{ENV["EDITOR"]} #{Shellwords.escape(tmp_path)}")
98
+ using_system_editor do
99
+ say "Editing #{content_path}..."
100
+ credentials.change { |tmp_path| system_editor(tmp_path) }
101
+ say "File encrypted and saved."
102
+ warn_if_credentials_are_invalid
92
103
  end
104
+ rescue ActiveSupport::EncryptedFile::MissingKeyError => error
105
+ say error.message
106
+ rescue ActiveSupport::MessageEncryptor::InvalidMessage
107
+ say "Couldn't decrypt #{content_path}. Perhaps you passed the wrong key?"
108
+ end
109
+
110
+ def warn_if_credentials_are_invalid
111
+ credentials.validate!
112
+ rescue ActiveSupport::EncryptedConfiguration::InvalidContentError => error
113
+ say "WARNING: #{error.message}", :red
114
+ say ""
115
+ say "Your application will not be able to load '#{content_path}' until the error has been fixed.", :red
93
116
  end
94
117
 
95
118
  def missing_credentials_message
96
- if credentials.key.nil?
97
- "Missing '#{key_path}' to decrypt credentials. See `bin/rails credentials:help`"
119
+ if !credentials.key?
120
+ "Missing '#{key_path}' to decrypt credentials. See `#{executable(:help)}`."
98
121
  else
99
- "File '#{content_path}' does not exist. Use `bin/rails credentials:edit` to change that."
122
+ "File '#{content_path}' does not exist. Use `#{executable(:edit)}` to change that."
100
123
  end
101
124
  end
102
125
 
103
- def content_path
104
- @content_path ||= options[:environment] ? "config/credentials/#{options[:environment]}.yml.enc" : "config/credentials.yml.enc"
105
- end
106
-
107
- def key_path
108
- options[:environment] ? "config/credentials/#{options[:environment]}.key" : "config/master.key"
126
+ def relative_path(path)
127
+ Rails.root.join(path).relative_path_from(Rails.root).to_s
109
128
  end
110
129
 
111
130
  def extract_environment_from_path(path)
112
- available_environments.find { |env| path.include? env } if path.end_with?(".yml.enc")
113
- end
114
-
115
- def encryption_key_file_generator
116
- require "rails/generators"
117
- require "rails/generators/rails/encryption_key_file/encryption_key_file_generator"
118
-
119
- Rails::Generators::EncryptionKeyFileGenerator.new
120
- end
121
-
122
- def encrypted_file_generator
123
- require "rails/generators"
124
- require "rails/generators/rails/encrypted_file/encrypted_file_generator"
125
-
126
- Rails::Generators::EncryptedFileGenerator.new
127
- end
128
-
129
- def credentials_generator
130
- require "rails/generators"
131
- require "rails/generators/rails/credentials/credentials_generator"
132
-
133
- Rails::Generators::CredentialsGenerator.new
131
+ available_environments.find { |env| path.end_with?("#{env}.yml.enc") }
134
132
  end
135
133
  end
136
134
  end
@@ -15,7 +15,8 @@ module Rails
15
15
  super
16
16
  end
17
17
 
18
- def perform
18
+ desc "change", "Change `config/database.yml` and your database gem to the target database"
19
+ def perform(*)
19
20
  Rails::Generators::Db::System::ChangeGenerator.start(@argv)
20
21
  end
21
22
  end
@@ -1,7 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  require "active_support/core_ext/string/filters"
4
- require "active_support/deprecation"
5
4
  require "rails/command/environment_argument"
6
5
 
7
6
  module Rails
@@ -12,88 +11,14 @@ module Rails
12
11
 
13
12
  def initialize(options = {})
14
13
  @options = options
14
+ @options[:environment] ||= Rails::Command.environment
15
15
  end
16
16
 
17
17
  def start
18
- ENV["RAILS_ENV"] ||= @options[:environment] || environment
19
- config = db_config.configuration_hash
20
-
21
- case db_config.adapter
22
- when /^(jdbc)?mysql/
23
- args = {
24
- host: "--host",
25
- port: "--port",
26
- socket: "--socket",
27
- username: "--user",
28
- encoding: "--default-character-set",
29
- sslca: "--ssl-ca",
30
- sslcert: "--ssl-cert",
31
- sslcapath: "--ssl-capath",
32
- sslcipher: "--ssl-cipher",
33
- sslkey: "--ssl-key"
34
- }.map { |opt, arg| "#{arg}=#{config[opt]}" if config[opt] }.compact
35
-
36
- if config[:password] && @options[:include_password]
37
- args << "--password=#{config[:password]}"
38
- elsif config[:password] && !config[:password].to_s.empty?
39
- args << "-p"
40
- end
41
-
42
- args << db_config.database
43
-
44
- find_cmd_and_exec(["mysql", "mysql5"], *args)
45
-
46
- when /^postgres|^postgis/
47
- ENV["PGUSER"] = config[:username] if config[:username]
48
- ENV["PGHOST"] = config[:host] if config[:host]
49
- ENV["PGPORT"] = config[:port].to_s if config[:port]
50
- ENV["PGPASSWORD"] = config[:password].to_s if config[:password] && @options[:include_password]
51
- find_cmd_and_exec("psql", db_config.database)
52
-
53
- when "sqlite3"
54
- args = []
55
-
56
- args << "-#{@options[:mode]}" if @options[:mode]
57
- args << "-header" if @options[:header]
58
- args << File.expand_path(db_config.database, Rails.respond_to?(:root) ? Rails.root : nil)
59
-
60
- find_cmd_and_exec("sqlite3", *args)
61
-
62
- when "oracle", "oracle_enhanced"
63
- logon = ""
64
-
65
- if config[:username]
66
- logon = config[:username].dup
67
- logon << "/#{config[:password]}" if config[:password] && @options[:include_password]
68
- logon << "@#{db_config.database}" if db_config.database
69
- end
70
-
71
- find_cmd_and_exec("sqlplus", logon)
72
-
73
- when "sqlserver"
74
- args = []
75
-
76
- args += ["-d", "#{db_config.database}"] if db_config.database
77
- args += ["-U", "#{config[:username]}"] if config[:username]
78
- args += ["-P", "#{config[:password]}"] if config[:password]
79
-
80
- if config[:host]
81
- host_arg = +"tcp:#{config[:host]}"
82
- host_arg << ",#{config[:port]}" if config[:port]
83
- args += ["-S", host_arg]
84
- end
85
-
86
- find_cmd_and_exec("sqlcmd", *args)
87
-
88
- else
89
- abort "Unknown command-line client for #{db_config.database}."
90
- end
91
- end
92
-
93
- def config
94
- db_config.configuration_hash
18
+ adapter_class.dbconsole(db_config, @options)
19
+ rescue NotImplementedError
20
+ abort "Unknown command-line client for #{db_config.database}."
95
21
  end
96
- deprecate config: "please use db_config.configuration_hash"
97
22
 
98
23
  def db_config
99
24
  return @db_config if defined?(@db_config)
@@ -101,7 +26,7 @@ module Rails
101
26
  # If the user provided a database, use that. Otherwise find
102
27
  # the first config in the database.yml
103
28
  if database
104
- @db_config = configurations.configs_for(env_name: environment, name: database, include_replicas: true)
29
+ @db_config = configurations.configs_for(env_name: environment, name: database, include_hidden: true)
105
30
  else
106
31
  @db_config = configurations.find_db_config(environment)
107
32
  end
@@ -115,15 +40,23 @@ module Rails
115
40
  @db_config
116
41
  end
117
42
 
118
- def environment
119
- Rails.respond_to?(:env) ? Rails.env : Rails::Command.environment
120
- end
121
-
122
43
  def database
123
44
  @options[:database]
124
45
  end
125
46
 
47
+ def environment
48
+ @options[:environment]
49
+ end
50
+
126
51
  private
52
+ def adapter_class
53
+ if ActiveRecord::Base.respond_to?(db_config.adapter_class_method)
54
+ ActiveRecord::Base.public_send(db_config.adapter_class_method)
55
+ else
56
+ ActiveRecord::ConnectionAdapters::AbstractAdapter
57
+ end
58
+ end
59
+
127
60
  def configurations # :doc:
128
61
  require APP_PATH
129
62
  ActiveRecord::Base.configurations = Rails.application.config.database_configuration
@@ -131,31 +64,11 @@ module Rails
131
64
  end
132
65
 
133
66
  def find_cmd_and_exec(commands, *args) # :doc:
134
- commands = Array(commands)
135
-
136
- dirs_on_path = ENV["PATH"].to_s.split(File::PATH_SEPARATOR)
137
- unless (ext = RbConfig::CONFIG["EXEEXT"]).empty?
138
- commands = commands.map { |cmd| "#{cmd}#{ext}" }
139
- end
140
-
141
- full_path_command = nil
142
- found = commands.detect do |cmd|
143
- dirs_on_path.detect do |path|
144
- full_path_command = File.join(path, cmd)
145
- begin
146
- stat = File.stat(full_path_command)
147
- rescue SystemCallError
148
- else
149
- stat.file? && stat.executable?
150
- end
151
- end
152
- end
153
-
154
- if found
155
- exec full_path_command, *args
156
- else
157
- abort("Couldn't find database client: #{commands.join(', ')}. Check your $PATH and try again.")
158
- end
67
+ Rails.deprecator.warn(<<~MSG.squish)
68
+ Rails::DBConsole#find_cmd_and_exec is deprecated and will be removed in Rails 7.2.
69
+ Please use find_cmd_and_exec on the connection adapter class instead.
70
+ MSG
71
+ ActiveRecord::Base.connection.find_cmd_and_exec(commands, *args)
159
72
  end
160
73
  end
161
74
 
@@ -167,20 +80,16 @@ module Rails
167
80
  desc: "Automatically provide the password from database.yml"
168
81
 
169
82
  class_option :mode, enum: %w( html list line column ), type: :string,
170
- desc: "Automatically put the sqlite3 database in the specified mode (html, list, line, column)."
83
+ desc: "Automatically put the sqlite3 database in the specified mode"
171
84
 
172
85
  class_option :header, type: :boolean
173
86
 
174
87
  class_option :database, aliases: "--db", type: :string,
175
- desc: "Specifies the database to use."
88
+ desc: "Specify the database to use."
176
89
 
90
+ desc "dbconsole", "Start a console for the database specified in config/database.yml"
177
91
  def perform
178
- extract_environment_option_from_argument
179
-
180
- # RAILS_ENV needs to be set before config/application is required.
181
- ENV["RAILS_ENV"] = options[:environment]
182
-
183
- require_application_and_environment!
92
+ boot_application!
184
93
  Rails::DBConsole.start(options)
185
94
  end
186
95
  end
@@ -7,18 +7,19 @@ module Rails
7
7
  class DestroyCommand < Base # :nodoc:
8
8
  no_commands do
9
9
  def help
10
- require_application_and_environment!
10
+ boot_application!
11
11
  load_generators
12
12
 
13
13
  Rails::Generators.help self.class.command_name
14
14
  end
15
15
  end
16
16
 
17
+ desc "destroy GENERATOR", "Remove code generated by `bin/rails generate`"
17
18
  def perform(*)
18
19
  generator = args.shift
19
20
  return help unless generator
20
21
 
21
- require_application_and_environment!
22
+ boot_application!
22
23
  load_generators
23
24
 
24
25
  Rails::Generators.invoke generator, args, behavior: :revoke, destination_root: Rails::Command.root
@@ -5,12 +5,7 @@ require "rails/dev_caching"
5
5
  module Rails
6
6
  module Command
7
7
  class DevCommand < Base # :nodoc:
8
- no_commands do
9
- def help
10
- say "rails dev:cache # Toggle development mode caching on/off."
11
- end
12
- end
13
-
8
+ desc "cache", "Toggle development mode caching on/off"
14
9
  def cache
15
10
  Rails::DevCaching.enable_by_file
16
11
  end
@@ -1,28 +1,23 @@
1
- === Storing Encrypted Files in Source Control
1
+ Description:
2
+ The Rails `encrypted` commands provide access to encrypted files or configurations.
3
+ See the `Rails.application.encrypted` documentation for using them in your app.
2
4
 
3
- The Rails `encrypted` commands provide access to encrypted files or configurations.
4
- See the `Rails.application.encrypted` documentation for using them in your app.
5
+ Encryption Keys:
6
+ By default, Rails looks for the encryption key in `ENV["RAILS_MASTER_KEY"]` or
7
+ `config/master.key`, but that lookup can be overridden with `--key`:
5
8
 
6
- === Encryption Keys
9
+ <%= executable(:edit) %> config/encrypted_file.yml.enc --key config/encrypted_file.key
7
10
 
8
- By default, Rails looks for the encryption key in `config/master.key` or
9
- `ENV["RAILS_MASTER_KEY"]`, but that lookup can be overridden with `--key`:
11
+ Don't commit the key! Add it to your source control's ignore file. If you use
12
+ Git, Rails handles this for you.
10
13
 
11
- rails encrypted:edit config/encrypted_file.yml.enc --key config/encrypted_file.key
14
+ Examples:
15
+ To edit or create an encrypted file use:
12
16
 
13
- Don't commit the key! Add it to your source control's ignore file. If you use
14
- Git, Rails handles this for you.
17
+ <%= executable(:edit) %> config/encrypted_file.yml.enc
15
18
 
16
- === Editing Files
19
+ This opens a temporary file in `$VISUAL` or `$EDITOR` with the decrypted contents for editing.
17
20
 
18
- To edit or create an encrypted file use:
21
+ To print the decrypted contents of an encrypted file use:
19
22
 
20
- rails encrypted:edit config/encrypted_file.yml.enc
21
-
22
- This opens a temporary file in `$EDITOR` with the decrypted contents for editing.
23
-
24
- === Viewing Files
25
-
26
- To print the decrypted contents of an encrypted file use:
27
-
28
- rails encrypted:show config/encrypted_file.yml.enc
23
+ <%= executable(:show) %> config/encrypted_file.yml.enc
@@ -12,54 +12,65 @@ module Rails
12
12
  class_option :key, aliases: "-k", type: :string,
13
13
  default: "config/master.key", desc: "The Rails.root relative path to the encryption key"
14
14
 
15
- no_commands do
16
- def help
17
- say "Usage:\n #{self.class.banner}"
18
- say ""
19
- say self.class.desc
20
- end
21
- end
22
-
23
- def edit(file_path)
24
- require_application!
25
- encrypted = Rails.application.encrypted(file_path, key_path: options[:key])
15
+ desc "edit", "Open the decrypted file in `$VISUAL` or `$EDITOR` for editing"
16
+ def edit(*)
17
+ load_environment_config!
26
18
 
27
- ensure_editor_available(command: "bin/rails encrypted:edit") || (return)
28
- ensure_encryption_key_has_been_added(options[:key]) if encrypted.key.nil?
29
- ensure_encrypted_file_has_been_added(file_path, options[:key])
19
+ ensure_encryption_key_has_been_added
20
+ ensure_encrypted_configuration_has_been_added
30
21
 
31
- catch_editing_exceptions do
32
- change_encrypted_file_in_system_editor(file_path, options[:key])
33
- end
34
-
35
- say "File encrypted and saved."
36
- rescue ActiveSupport::MessageEncryptor::InvalidMessage
37
- say "Couldn't decrypt #{file_path}. Perhaps you passed the wrong key?"
22
+ change_encrypted_configuration_in_system_editor
38
23
  end
39
24
 
40
- def show(file_path)
41
- require_application!
42
- encrypted = Rails.application.encrypted(file_path, key_path: options[:key])
25
+ desc "show", "Show the decrypted contents of the file"
26
+ def show(*)
27
+ load_environment_config!
43
28
 
44
- say encrypted.read.presence || missing_encrypted_message(key: encrypted.key, key_path: options[:key], file_path: file_path)
29
+ say encrypted_configuration.read.presence || missing_encrypted_configuration_message
45
30
  end
46
31
 
47
32
  private
48
- def ensure_encryption_key_has_been_added(key_path)
33
+ def content_path
34
+ @content_path ||= args[0]
35
+ end
36
+
37
+ def key_path
38
+ options[:key]
39
+ end
40
+
41
+ def encrypted_configuration
42
+ @encrypted_configuration ||= Rails.application.encrypted(content_path, key_path: key_path)
43
+ end
44
+
45
+ def ensure_encryption_key_has_been_added
46
+ return if encrypted_configuration.key?
49
47
  encryption_key_file_generator.add_key_file(key_path)
50
48
  encryption_key_file_generator.ignore_key_file(key_path)
51
49
  end
52
50
 
53
- def ensure_encrypted_file_has_been_added(file_path, key_path)
54
- encrypted_file_generator.add_encrypted_file_silently(file_path, key_path)
51
+ def ensure_encrypted_configuration_has_been_added
52
+ encrypted_file_generator.add_encrypted_file_silently(content_path, key_path)
55
53
  end
56
54
 
57
- def change_encrypted_file_in_system_editor(file_path, key_path)
58
- Rails.application.encrypted(file_path, key_path: key_path).change do |tmp_path|
59
- system("#{ENV["EDITOR"]} #{tmp_path}")
55
+ def change_encrypted_configuration_in_system_editor
56
+ using_system_editor do
57
+ encrypted_configuration.change { |tmp_path| system_editor(tmp_path) }
58
+ say "File encrypted and saved."
59
+ warn_if_encrypted_configuration_is_invalid
60
60
  end
61
+ rescue ActiveSupport::EncryptedFile::MissingKeyError => error
62
+ say error.message
63
+ rescue ActiveSupport::MessageEncryptor::InvalidMessage
64
+ say "Couldn't decrypt #{content_path}. Perhaps you passed the wrong key?"
61
65
  end
62
66
 
67
+ def warn_if_encrypted_configuration_is_invalid
68
+ encrypted_configuration.validate!
69
+ rescue ActiveSupport::EncryptedConfiguration::InvalidContentError => error
70
+ say "WARNING: #{error.message}", :red
71
+ say ""
72
+ say "Your application will not be able to load '#{content_path}' until the error has been fixed.", :red
73
+ end
63
74
 
64
75
  def encryption_key_file_generator
65
76
  require "rails/generators"
@@ -75,11 +86,11 @@ module Rails
75
86
  Rails::Generators::EncryptedFileGenerator.new
76
87
  end
77
88
 
78
- def missing_encrypted_message(key:, key_path:, file_path:)
79
- if key.nil?
80
- "Missing '#{key_path}' to decrypt data. See `bin/rails encrypted:help`"
89
+ def missing_encrypted_configuration_message
90
+ if !encrypted_configuration.key?
91
+ "Missing '#{key_path}' to decrypt data. See `#{executable(:help)}`"
81
92
  else
82
- "File '#{file_path}' does not exist. Use `bin/rails encrypted:edit #{file_path}` to change that."
93
+ "File '#{content_path}' does not exist. Use `#{executable(:edit)} #{content_path}` to change that."
83
94
  end
84
95
  end
85
96
  end
@@ -0,0 +1,16 @@
1
+ Usage:
2
+ rails COMMAND [options]
3
+
4
+ You must specify a command:
5
+
6
+ new Create a new Rails application. "rails new my_app" creates a
7
+ new application called MyApp in "./my_app"
8
+ plugin new Create a new Rails railtie or engine
9
+
10
+ All commands can be run with -h (or --help) for more information.
11
+
12
+ Inside a Rails application directory, some common commands are:
13
+
14
+ console Start the Rails console
15
+ server Start the Rails server
16
+ test Run tests except system tests
@@ -0,0 +1,13 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails
4
+ module Command
5
+ class GemHelpCommand < Base # :nodoc:
6
+ hide_command!
7
+
8
+ def perform
9
+ say self.class.class_usage
10
+ end
11
+ end
12
+ end
13
+ end
@@ -7,7 +7,7 @@ module Rails
7
7
  class GenerateCommand < Base # :nodoc:
8
8
  no_commands do
9
9
  def help
10
- require_application_and_environment!
10
+ boot_application!
11
11
  load_generators
12
12
 
13
13
  Rails::Generators.help self.class.command_name
@@ -18,7 +18,7 @@ module Rails
18
18
  generator = args.shift
19
19
  return help unless generator
20
20
 
21
- require_application_and_environment!
21
+ boot_application!
22
22
  load_generators
23
23
 
24
24
  ARGV.replace(args) # set up ARGV for third-party libraries