actual_db_schema 0.8.1 → 0.8.3

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.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 9197558d7b71582d339535933b2186c9ce3db7d0c4dd492759db5a226931ddf8
4
- data.tar.gz: ccd05c4164478a54130c6bee32a234abbb2f193292438b8d35b6d7edd313802c
3
+ metadata.gz: d78699aee32920e20a6dd084512949d9cc88e7878531de087295d43833463193
4
+ data.tar.gz: 1e55c7f7b15e0a2fbace58cc80c32deb2061bbe6cd8d8da69f8f35682e9d5b75
5
5
  SHA512:
6
- metadata.gz: b12528875d4b7d5b12739b1a37877d979bccc6d66f7a0006454ae260b1215eb69bb5baa2aae332e68bc4f1fa29859e4b00a427591cf78acf2aebc57264889a46
7
- data.tar.gz: ae2baeaae67451740cc6773abbc79e4d0896baf96b99e7c2baebe027c1b7b05bbbbb75da5717be47c43d240835810a1681408e79c3d44449f7f44c3d4e8830a6
6
+ metadata.gz: df1251d4d45888438f9e5ab3585eafa2f2eec2ace597ec0b6985479dc835a67460990e5b6e67534bfeceb779f088b64c65fcff0c9a7b7e16007a90c365f7d93e
7
+ data.tar.gz: 94a00404bb421f3b3a1124c456358003da71762a652cca2d6e89763bdbc872ddf3f86c2c68ca9b5cc26e12a514869f9c11aba972dfb705f816dc3d43200e63a1
data/.rubocop.yml CHANGED
@@ -22,9 +22,12 @@ Metrics/BlockLength:
22
22
  - actual_db_schema.gemspec
23
23
 
24
24
  Metrics/MethodLength:
25
+ Max: 15
25
26
  Exclude:
26
27
  - test/**/*
27
28
 
28
29
  Metrics/ClassLength:
29
- Exclude:
30
- - test/**/*
30
+ Enabled: false
31
+
32
+ Metrics/ModuleLength:
33
+ Enabled: false
data/CHANGELOG.md CHANGED
@@ -1,3 +1,15 @@
1
+ ## [0.8.3] - 2025-03-03
2
+
3
+ - View Schema with Migration Annotations in the UI
4
+ - Clean Up Broken Migrations
5
+ - Filter Migrations in the UI
6
+ - Customize Your Migrated Folder Location
7
+
8
+ ## [0.8.2] - 2025-02-06
9
+
10
+ - Show migration name in the schema.rb diff that caused the change
11
+ - Easy way to run DDL migration methods in Rails console
12
+
1
13
  ## [0.8.1] - 2025-01-15
2
14
 
3
15
  - Support for multiple database schemas, ensuring compatibility with multi-tenant applications using the apartment gem or similar solutions
data/Gemfile.lock CHANGED
@@ -1,10 +1,12 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- actual_db_schema (0.8.1)
4
+ actual_db_schema (0.8.3)
5
5
  activerecord
6
6
  activesupport
7
+ ast
7
8
  csv
9
+ parser
8
10
 
9
11
  GEM
10
12
  remote: https://rubygems.org/
data/README.md CHANGED
@@ -72,6 +72,25 @@ The gem offers the following rake tasks that can be manually run according to yo
72
72
  - `rails db:rollback_branches:manual` - run it to manually rolls back phantom migrations one by one.
73
73
  - `rails db:phantom_migrations` - displays a list of phantom migrations.
74
74
 
75
+ ## Migrated Folder Configuration
76
+
77
+ By default, `actual_db_schema` stores all run migrations in the `tmp/migrated` folder. However, if you want to change this location, you can configure it in two ways:
78
+
79
+ ### 1. Using Environment Variable
80
+
81
+ Set the environment variable `ACTUAL_DB_SCHEMA_MIGRATED_FOLDER` to your desired folder path:
82
+
83
+ ```sh
84
+ export ACTUAL_DB_SCHEMA_MIGRATED_FOLDER="custom/migrated"
85
+ ```
86
+
87
+ ### 2. Using Initializer
88
+ Add the following line to your initializer file (`config/initializers/actual_db_schema.rb`):
89
+
90
+ ```ruby
91
+ config.migrated_folder = Rails.root.join("custom", "migrated")
92
+ ```
93
+
75
94
  ## Accessing the UI
76
95
 
77
96
  The UI for managing migrations is enabled automatically. To access the UI, simply navigate to the following URL in your web browser:
@@ -168,6 +187,109 @@ config.multi_tenant_schemas = -> { # list of all active schemas }
168
187
  config.multi_tenant_schemas = -> { ["public", "tenant1", "tenant2"] }
169
188
  ```
170
189
 
190
+ ## Schema Diff with Migration Annotations
191
+
192
+ If `schema.rb` generates a diff, it can be helpful to find out which migrations caused the changes. This helps you decide whether to resolve the diff on your own or discuss it with your teammates to determine the next steps. The `diff_schema_with_migrations` Rake task generates a diff of the `schema.rb` file, annotated with the migrations responsible for each change. This makes it easier to trace which migration introduced a specific schema modification, enabling faster and more informed decision-making regarding how to handle the diff.
193
+
194
+ By default, the task uses `db/schema.rb` and `db/migrate` as the schema and migrations paths. You can also provide custom paths as arguments.
195
+
196
+ ### Usage
197
+
198
+ Run the task with default paths:
199
+ ```sh
200
+ rake actual_db_schema:diff_schema_with_migrations
201
+ ```
202
+
203
+ Run the task with custom paths:
204
+ ```sh
205
+ rake actual_db_schema:diff_schema_with_migrations[path/to/custom_schema.rb, path/to/custom_migrations]
206
+ ```
207
+
208
+ ## Console Migrations
209
+
210
+ Sometimes, it's necessary to modify the database without creating migration files. This can be useful for fixing a corrupted schema, conducting experiments (such as adding and removing indexes), or quickly adjusting the schema in development. This gem allows you to run the same commands used in migrations directly in the Rails console.
211
+
212
+ By default, Console Migrations is disabled. You can enable it in two ways:
213
+
214
+ ### 1. Using Environment Variable
215
+
216
+ Set the environment variable `ACTUAL_DB_SCHEMA_CONSOLE_MIGRATIONS_ENABLED` to `true`:
217
+
218
+ ```sh
219
+ export ACTUAL_DB_SCHEMA_CONSOLE_MIGRATIONS_ENABLED=true
220
+ ```
221
+
222
+ ### 2. Using Initializer
223
+
224
+ Add the following line to your initializer file (`config/initializers/actual_db_schema.rb`):
225
+
226
+ ```ruby
227
+ config.console_migrations_enabled = true
228
+ ```
229
+
230
+ ### Usage
231
+
232
+ Once enabled, you can run migration commands directly in the Rails console:
233
+
234
+ ```ruby
235
+ # Create a new table
236
+ create_table :posts do |t|
237
+ t.string :title
238
+ end
239
+
240
+ # Add a column
241
+ add_column :users, :age, :integer
242
+
243
+ # Remove an index
244
+ remove_index :users, :email
245
+
246
+ # Rename a column
247
+ rename_column :users, :username, :handle
248
+ ```
249
+
250
+ ## Delete Broken Migrations
251
+
252
+ A migration is considered broken if it has been migrated in the database but the corresponding migration file is missing. This functionality allows you to safely delete these broken versions from the database to keep it clean.
253
+
254
+ You can delete broken migrations using either of the following methods:
255
+
256
+ ### 1. Using the UI
257
+
258
+ Navigate to the following URL in your web browser:
259
+ ```
260
+ http://localhost:3000/rails/broken_versions
261
+ ```
262
+
263
+ This page lists all broken versions and provides an option to delete them.
264
+
265
+ ### 2. Using a Rake Task
266
+
267
+ To delete all broken migrations, run:
268
+ ```sh
269
+ rake actual_db_schema:delete_broken_versions
270
+ ```
271
+
272
+ To delete specific migrations, pass the migration version(s) and optionally a database:
273
+ ```sh
274
+ rake actual_db_schema:delete_broken_versions[<version>, <version>]
275
+ ```
276
+
277
+ - `<version>` – The migration version(s) to delete (space-separated if multiple).
278
+ - `<database>` (optional) – Specify a database if using multiple databases.
279
+
280
+ #### Examples:
281
+
282
+ ```sh
283
+ # Delete all broken migrations
284
+ rake actual_db_schema:delete_broken_versions
285
+
286
+ # Delete specific migrations
287
+ rake actual_db_schema:delete_broken_versions["20250224103352 20250224103358"]
288
+
289
+ # Delete specific migrations from a specific database
290
+ rake actual_db_schema:delete_broken_versions["20250224103352 20250224103358", "primary"]
291
+ ```
292
+
171
293
  ## Development
172
294
 
173
295
  After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake test` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
@@ -181,7 +303,8 @@ To release a new version do the following in the order:
181
303
  - `bundle install` to update `Gemfile.lock`;
182
304
  - make the commit and push;
183
305
  - run `bundle exec rake release`. This will create a git tag for the version, push git commits and the created tag, and push the `.gem` file to [rubygems.org](https://rubygems.org);
184
- - [announce the new release on GitHub](https://github.com/widefix/actual_db_schema/releases).
306
+ - [announce the new release on GitHub](https://github.com/widefix/actual_db_schema/releases);
307
+ - close the milestone on GitHub.
185
308
 
186
309
  ### Running Tests with Specific Rails Versions
187
310
 
@@ -37,7 +37,9 @@ Gem::Specification.new do |spec|
37
37
  # Uncomment to register a new dependency of your gem
38
38
  spec.add_runtime_dependency "activerecord"
39
39
  spec.add_runtime_dependency "activesupport"
40
+ spec.add_runtime_dependency "ast"
40
41
  spec.add_runtime_dependency "csv"
42
+ spec.add_runtime_dependency "parser"
41
43
 
42
44
  spec.add_development_dependency "appraisal"
43
45
  spec.add_development_dependency "debug"
@@ -0,0 +1,41 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActualDbSchema
4
+ # Controller for managing broken migration versions.
5
+ class BrokenVersionsController < ActionController::Base
6
+ protect_from_forgery with: :exception
7
+ skip_before_action :verify_authenticity_token
8
+
9
+ def index; end
10
+
11
+ def delete
12
+ handle_delete(params[:id], params[:database])
13
+ redirect_to broken_versions_path
14
+ end
15
+
16
+ def delete_all
17
+ handle_delete_all
18
+ redirect_to broken_versions_path
19
+ end
20
+
21
+ private
22
+
23
+ def handle_delete(id, database)
24
+ ActualDbSchema::Migration.instance.delete(id, database)
25
+ flash[:notice] = "Migration #{id} was successfully deleted."
26
+ rescue StandardError => e
27
+ flash[:alert] = e.message
28
+ end
29
+
30
+ def handle_delete_all
31
+ ActualDbSchema::Migration.instance.delete_all
32
+ flash[:notice] = "All broken versions were successfully deleted."
33
+ rescue StandardError => e
34
+ flash[:alert] = e.message
35
+ end
36
+
37
+ helper_method def broken_versions
38
+ @broken_versions ||= ActualDbSchema::Migration.instance.broken_versions
39
+ end
40
+ end
41
+ end
@@ -40,6 +40,20 @@ module ActualDbSchema
40
40
 
41
41
  helper_method def migrations
42
42
  @migrations ||= ActualDbSchema::Migration.instance.all
43
+ query = params[:query].to_s.strip.downcase
44
+
45
+ return @migrations if query.blank?
46
+
47
+ @migrations.select do |migration|
48
+ file_name_matches = migration[:filename].include?(query)
49
+ content_matches = begin
50
+ File.read(migration[:filename]).downcase.include?(query)
51
+ rescue StandardError
52
+ false
53
+ end
54
+
55
+ file_name_matches || content_matches
56
+ end
43
57
  end
44
58
 
45
59
  helper_method def migration
@@ -0,0 +1,18 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActualDbSchema
4
+ # Controller to display the database schema diff.
5
+ class SchemaController < ActionController::Base
6
+ protect_from_forgery with: :exception
7
+ skip_before_action :verify_authenticity_token
8
+
9
+ def index; end
10
+
11
+ private
12
+
13
+ helper_method def schema_diff_html
14
+ schema_diff = ActualDbSchema::SchemaDiffHtml.new("./db/schema.rb", "db/migrate")
15
+ schema_diff.render_html(params[:table])
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,64 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+ <head>
4
+ <title>Broken Versions</title>
5
+ <%= render partial: 'actual_db_schema/shared/js' %>
6
+ <%= render partial: 'actual_db_schema/shared/style' %>
7
+ </head>
8
+ <body>
9
+ <div>
10
+ <% flash.each do |key, message| %>
11
+ <div class="flash <%= key %>"><%= message %></div>
12
+ <% end %>
13
+ <h2>Broken Versions</h2>
14
+ <p>
15
+ These are versions that were migrated in the database, but the corresponding migration file is missing.
16
+ You can safely delete them from the database to clean up.
17
+ </p>
18
+ <div class="top-buttons">
19
+ <%= link_to 'All Migrations', migrations_path, class: "top-button" %>
20
+ <% if broken_versions.present? %>
21
+ <%= button_to '✖ Delete all',
22
+ delete_all_broken_versions_path,
23
+ method: :post,
24
+ data: { confirm: 'These migrations do not have corresponding migration files. Proceeding will remove these entries from the `schema_migrations` table. Are you sure you want to continue?' },
25
+ class: 'button migration-action' %>
26
+ <% end %>
27
+ </div>
28
+ <% if broken_versions.present? %>
29
+ <table>
30
+ <thead>
31
+ <tr>
32
+ <th>Status</th>
33
+ <th>Migration ID</th>
34
+ <th>Branch</th>
35
+ <th>Database</th>
36
+ <th>Actions</th>
37
+ </tr>
38
+ </thead>
39
+ <tbody>
40
+ <% broken_versions.each do |version| %>
41
+ <tr class="migration-row phantom">
42
+ <td><%= version[:status] %></td>
43
+ <td><%= version[:version] %></td>
44
+ <td><%= version[:branch] %></td>
45
+ <td><%= version[:database] %></td>
46
+ <td>
47
+ <div class='button-container'>
48
+ <%= button_to '✖ Delete',
49
+ delete_broken_version_path(id: version[:version], database: version[:database]),
50
+ method: :post,
51
+ data: { confirm: 'This migration does not have a corresponding migration file. Proceeding will remove its entry from the `schema_migrations` table. Are you sure you want to continue?' },
52
+ class: 'button migration-action' %>
53
+ </div>
54
+ </td>
55
+ </tr>
56
+ <% end %>
57
+ </tbody>
58
+ </table>
59
+ <% else %>
60
+ <p>No broken versions found.</p>
61
+ <% end %>
62
+ </div>
63
+ </body>
64
+ </html>
@@ -14,57 +14,69 @@
14
14
  <p>
15
15
  <span style="background-color: #ffe6e6; padding: 0 5px;">Red rows</span> represent phantom migrations.
16
16
  </p>
17
- <div class="top-buttons">
18
- <%= link_to 'Phantom Migrations', phantom_migrations_path, class: "top-button" %>
19
- </div>
20
- <% if migrations.present? %>
21
- <table>
22
- <thead>
23
- <tr>
24
- <th>Status</th>
25
- <th>Migration ID</th>
26
- <th>Name</th>
27
- <th>Branch</th>
28
- <th>Database</th>
29
- <th>Actions</th>
30
- </tr>
31
- </thead>
32
- <tbody>
33
- <% migrations.each do |migration| %>
34
- <tr class="migration-row <%= migration[:phantom] ? 'phantom' : 'normal' %>">
35
- <td><%= migration[:status] %></td>
36
- <td><%= migration[:version] %></td>
37
- <td>
38
- <div class="truncate-text" title="<%= migration[:name] %>">
39
- <%= migration[:name] %>
40
- </div>
41
- </td>
42
- <td><%= migration[:branch] %></td>
43
- <td><%= migration[:database] %></td>
44
- <td>
45
- <div class='button-container'>
46
- <%= link_to '👁 Show',
47
- migration_path(id: migration[:version], database: migration[:database]),
48
- class: 'button' %>
49
- <%= button_to '⎌ Rollback',
50
- rollback_migration_path(id: migration[:version], database: migration[:database]),
51
- method: :post,
52
- class: 'button migration-action',
53
- style: ('display: none;' if migration[:status] == "down") %>
54
- <%= button_to '⬆ Migrate',
55
- migrate_migration_path(id: migration[:version], database: migration[:database]),
56
- method: :post,
57
- class: 'button migration-action',
58
- style: ('display: none;' if migration[:status] == "up" || migration[:phantom]) %>
59
- </div>
60
- </td>
61
- </tr>
17
+ <div class="container">
18
+ <div class="top-controls">
19
+ <div class="top-buttons">
20
+ <%= link_to 'Phantom Migrations', phantom_migrations_path, class: "top-button" %>
21
+ <%= link_to 'Broken Versions', broken_versions_path, class: "top-button" %>
22
+ <%= link_to 'View Schema', schema_path, class: "top-button" %>
23
+ </div>
24
+ <div class="top-search">
25
+ <%= form_tag migrations_path, method: :get, class: "search-form" do %>
26
+ <span class="search-icon">🔍</span>
27
+ <%= text_field_tag :query, params[:query], placeholder: "Search migrations by name or content", class: "search-input" %>
62
28
  <% end %>
63
- </tbody>
64
- </table>
65
- <% else %>
66
- <p>No migrations found.</p>
67
- <% end %>
29
+ </div>
30
+ </div>
31
+ <% if migrations.present? %>
32
+ <table>
33
+ <thead>
34
+ <tr>
35
+ <th>Status</th>
36
+ <th>Migration ID</th>
37
+ <th>Name</th>
38
+ <th>Branch</th>
39
+ <th>Database</th>
40
+ <th>Actions</th>
41
+ </tr>
42
+ </thead>
43
+ <tbody>
44
+ <% migrations.each do |migration| %>
45
+ <tr class="migration-row <%= migration[:phantom] ? 'phantom' : 'normal' %>">
46
+ <td><%= migration[:status] %></td>
47
+ <td><%= migration[:version] %></td>
48
+ <td>
49
+ <div class="truncate-text" title="<%= migration[:name] %>">
50
+ <%= migration[:name] %>
51
+ </div>
52
+ </td>
53
+ <td><%= migration[:branch] %></td>
54
+ <td><%= migration[:database] %></td>
55
+ <td>
56
+ <div class='button-container'>
57
+ <%= link_to '👁 Show',
58
+ migration_path(id: migration[:version], database: migration[:database]),
59
+ class: 'button' %>
60
+ <%= button_to '⎌ Rollback',
61
+ rollback_migration_path(id: migration[:version], database: migration[:database]),
62
+ method: :post,
63
+ class: 'button migration-action',
64
+ style: ('display: none;' if migration[:status] == "down") %>
65
+ <%= button_to '⬆ Migrate',
66
+ migrate_migration_path(id: migration[:version], database: migration[:database]),
67
+ method: :post,
68
+ class: 'button migration-action',
69
+ style: ('display: none;' if migration[:status] == "up" || migration[:phantom]) %>
70
+ </div>
71
+ </td>
72
+ </tr>
73
+ <% end %>
74
+ </tbody>
75
+ </table>
76
+ <% else %>
77
+ <p>No migrations found.</p>
78
+ <% end %>
79
+ </div>
68
80
  </div>
69
81
  </body>
70
82
  </html>
@@ -0,0 +1,31 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+ <head>
4
+ <title>Database Schema</title>
5
+ <%= render partial: 'actual_db_schema/shared/js' %>
6
+ <%= render partial: 'actual_db_schema/shared/style' %>
7
+ </head>
8
+ <body>
9
+ <div>
10
+ <% flash.each do |key, message| %>
11
+ <div class="flash <%= key %>"><%= message %></div>
12
+ <% end %>
13
+ <h2>Database Schema</h2>
14
+ <div class="top-controls">
15
+ <div class="top-buttons">
16
+ <%= link_to 'All Migrations', migrations_path, class: "top-button" %>
17
+ </div>
18
+ <div class="top-search">
19
+ <%= form_tag schema_path, method: :get, class: "search-form" do %>
20
+ <span class="search-icon">🔍</span>
21
+ <%= text_field_tag :table, params[:table], placeholder: "Filter by table name", class: "search-input" %>
22
+ <% end %>
23
+ </div>
24
+ </div>
25
+
26
+ <div class="schema-diff">
27
+ <pre><%= raw schema_diff_html %></pre>
28
+ </div>
29
+ </div>
30
+ </body>
31
+ </html>
@@ -4,6 +4,12 @@
4
4
 
5
5
  migrationActions.forEach(button => {
6
6
  button.addEventListener('click', function(event) {
7
+ const confirmMessage = button.dataset.confirm;
8
+ if (confirmMessage && !confirm(confirmMessage)) {
9
+ event.preventDefault();
10
+ return;
11
+ }
12
+
7
13
  const originalText = button.value;
8
14
  button.value = 'Loading...';
9
15
  disableButtons();
@@ -15,6 +15,10 @@
15
15
  padding-left: 10px;
16
16
  }
17
17
 
18
+ p {
19
+ padding-left: 10px;
20
+ }
21
+
18
22
  table {
19
23
  margin: 0;
20
24
  border-collapse: collapse;
@@ -65,6 +69,7 @@
65
69
  text-decoration: none;
66
70
  display: inline-block;
67
71
  margin: 0 2px;
72
+ margin-right: 8px;
68
73
  cursor: pointer;
69
74
  border-radius: 4px;
70
75
  transition: background-color 0.3s;
@@ -113,4 +118,44 @@
113
118
  background-color: #f8d7da;
114
119
  color: #721c24;
115
120
  }
121
+
122
+ .container {
123
+ display: inline-block;
124
+ max-width: 100%;
125
+ }
126
+
127
+ .top-controls {
128
+ display: flex;
129
+ justify-content: space-between;
130
+ align-items: center;
131
+ width: 100%;
132
+ }
133
+
134
+ .top-search {
135
+ display: flex;
136
+ align-items: center;
137
+ justify-content: flex-end;
138
+ }
139
+
140
+ .search-form {
141
+ display: flex;
142
+ align-items: center;
143
+ }
144
+
145
+ .search-form .search-icon {
146
+ margin-right: 5px;
147
+ font-size: 16px;
148
+ }
149
+
150
+ .search-form .search-input {
151
+ padding: 5px;
152
+ border: 1px solid #ccc;
153
+ border-radius: 4px;
154
+ font-size: 13px;
155
+ width: 250px;
156
+ }
157
+
158
+ .schema-diff {
159
+ margin-left: 8px;
160
+ }
116
161
  </style>
data/config/routes.rb CHANGED
@@ -15,4 +15,14 @@ ActualDbSchema::Engine.routes.draw do
15
15
  post :rollback_all
16
16
  end
17
17
  end
18
+ resources :broken_versions, only: %i[index] do
19
+ member do
20
+ post :delete
21
+ end
22
+ collection do
23
+ post :delete_all
24
+ end
25
+ end
26
+
27
+ get "schema", to: "schema#index", as: :schema
18
28
  end
@@ -3,7 +3,8 @@
3
3
  module ActualDbSchema
4
4
  # Manages the configuration settings for the gem.
5
5
  class Configuration
6
- attr_accessor :enabled, :auto_rollback_disabled, :ui_enabled, :git_hooks_enabled, :multi_tenant_schemas
6
+ attr_accessor :enabled, :auto_rollback_disabled, :ui_enabled, :git_hooks_enabled, :multi_tenant_schemas,
7
+ :console_migrations_enabled, :migrated_folder
7
8
 
8
9
  def initialize
9
10
  @enabled = Rails.env.development?
@@ -11,6 +12,8 @@ module ActualDbSchema
11
12
  @ui_enabled = Rails.env.development? || ENV["ACTUAL_DB_SCHEMA_UI_ENABLED"].present?
12
13
  @git_hooks_enabled = ENV["ACTUAL_DB_SCHEMA_GIT_HOOKS_ENABLED"].present?
13
14
  @multi_tenant_schemas = nil
15
+ @console_migrations_enabled = ENV["ACTUAL_DB_SCHEMA_CONSOLE_MIGRATIONS_ENABLED"].present?
16
+ @migrated_folder = ENV["ACTUAL_DB_SCHEMA_MIGRATED_FOLDER"].present?
14
17
  end
15
18
 
16
19
  def [](key)
@@ -0,0 +1,47 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActualDbSchema
4
+ # Provides methods for executing schema modification commands directly in the Rails console.
5
+ module ConsoleMigrations
6
+ extend self
7
+
8
+ SCHEMA_METHODS = %i[
9
+ create_table
10
+ create_join_table
11
+ drop_table
12
+ change_table
13
+ add_column
14
+ remove_column
15
+ change_column
16
+ change_column_null
17
+ change_column_default
18
+ rename_column
19
+ add_index
20
+ remove_index
21
+ rename_index
22
+ add_timestamps
23
+ remove_timestamps
24
+ reversible
25
+ add_reference
26
+ remove_reference
27
+ add_foreign_key
28
+ remove_foreign_key
29
+ ].freeze
30
+
31
+ SCHEMA_METHODS.each do |method_name|
32
+ define_method(method_name) do |*args, **kwargs, &block|
33
+ if kwargs.any?
34
+ migration_instance.public_send(method_name, *args, **kwargs, &block)
35
+ else
36
+ migration_instance.public_send(method_name, *args, &block)
37
+ end
38
+ end
39
+ end
40
+
41
+ private
42
+
43
+ def migration_instance
44
+ @migration_instance ||= Class.new(ActiveRecord::Migration[ActiveRecord::Migration.current_version]) {}.new
45
+ end
46
+ end
47
+ end