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 +4 -4
- data/.rubocop.yml +5 -2
- data/CHANGELOG.md +12 -0
- data/Gemfile.lock +3 -1
- data/README.md +124 -1
- data/actual_db_schema.gemspec +2 -0
- data/app/controllers/actual_db_schema/broken_versions_controller.rb +41 -0
- data/app/controllers/actual_db_schema/migrations_controller.rb +14 -0
- data/app/controllers/actual_db_schema/schema_controller.rb +18 -0
- data/app/views/actual_db_schema/broken_versions/index.html.erb +64 -0
- data/app/views/actual_db_schema/migrations/index.html.erb +62 -50
- data/app/views/actual_db_schema/schema/index.html.erb +31 -0
- data/app/views/actual_db_schema/shared/_js.html +6 -0
- data/app/views/actual_db_schema/shared/_style.html +45 -0
- data/config/routes.rb +10 -0
- data/lib/actual_db_schema/configuration.rb +4 -1
- data/lib/actual_db_schema/console_migrations.rb +47 -0
- data/lib/actual_db_schema/git_hooks.rb +1 -1
- data/lib/actual_db_schema/migration.rb +49 -1
- data/lib/actual_db_schema/migration_parser.rb +264 -0
- data/lib/actual_db_schema/patches/migration_context.rb +1 -1
- data/lib/actual_db_schema/railtie.rb +15 -0
- data/lib/actual_db_schema/schema_diff.rb +229 -0
- data/lib/actual_db_schema/schema_diff_html.rb +122 -0
- data/lib/actual_db_schema/schema_parser.rb +153 -0
- data/lib/actual_db_schema/version.rb +1 -1
- data/lib/actual_db_schema.rb +6 -1
- data/lib/generators/actual_db_schema/templates/actual_db_schema.rb +8 -0
- data/lib/tasks/actual_db_schema.rake +33 -0
- metadata +40 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: d78699aee32920e20a6dd084512949d9cc88e7878531de087295d43833463193
|
4
|
+
data.tar.gz: 1e55c7f7b15e0a2fbace58cc80c32deb2061bbe6cd8d8da69f8f35682e9d5b75
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: df1251d4d45888438f9e5ab3585eafa2f2eec2ace597ec0b6985479dc835a67460990e5b6e67534bfeceb779f088b64c65fcff0c9a7b7e16007a90c365f7d93e
|
7
|
+
data.tar.gz: 94a00404bb421f3b3a1124c456358003da71762a652cca2d6e89763bdbc872ddf3f86c2c68ca9b5cc26e12a514869f9c11aba972dfb705f816dc3d43200e63a1
|
data/.rubocop.yml
CHANGED
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
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
|
|
data/actual_db_schema.gemspec
CHANGED
@@ -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="
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
<
|
27
|
-
|
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
|
-
</
|
64
|
-
</
|
65
|
-
|
66
|
-
|
67
|
-
|
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
|