I've seen the question "how can I get the SQL for a migration without running it?" come up a few times. The low-tech answer is to ignore the "without running" part of the question and just run the migration, grep the log file for the SQL output, and then do a db:rollback
. But that seems like cheating and is kind of a pain, too. So, how can we make it better?
My initial approach was to intercept a method call pretty far down in the Active Record stack and, if it was the particular migration I was looking for, print the SQL rather than executing it. I'm using PostgreSQL, and I'd like to intercept the ActiveRecord::ConnectionAdapters::PostgreSQLAdapter#execute
method, so here's a proxy:
Actually using this was clumsy though. I had to get into a console, apply the code modification, and then explicitly invoke the migration:
There's some sample output here, but you get the idea. There's not much to it and using it is kind of painful.
This StackOverflow thread has some nicer approaches. The top answer uses alias_method
to redefine ActiveRecord::ConnectionAdapters::AbstractMysqlAdapter#execute
so it does more or less what I did in the MyTweak
module above; it short-circuits certain methods and prints out the SQL. The real improvement though is that it defines a fake_db_migrate
Rake task that does the monkeypatch and then runs db:migrate
, so it works for any migration without having to hardcode anything.
Another answer in that thread is more work but also seems less fragile. It uses a transaction and a rollback to get the SQL. You still have to get into a console and require
the specific migration, but then instead of patching the code you run the migration in a transaction with an explicit raise of a rollback exception:
That approach also ignores the "without running the SQL" part of the question. But that's probably fine; the intent is to figure out the SQL and leave the database in the original state, and that's achieved by running the SQL and rolling it back.
Over on the Rails core project there's a recently opened PR for this feature. The PR uses an approach that piggybacks on the "dry run" flag and uses a rollback strategy. It'll be interesting to see how that implementation evolves and if it makes it into Active Record core.
You might be wondering why we don't just grab the Arel AST for the migration and call `to_sql` on it. But that's not how migrations work. They don't go through an intermediate tree generation and traversal process like Active Record queries do. Instead, they're built from SQL as needed. For example, here's an excerpt from the PostgreSQL adapter's version of the very handy migration method `change_column_default`; you can see how it's concatenating strings:
Arel doesn't support DDL ASTs, but DDL grammars exist and you could imagine putting together an intermediate layer for migrations. It's hard for me to imagine that being worth the hassle, though, since the current approach has done the job for the past decade.
To wrap up, all in all I'd say the rollback strategy seems cleanest since it doesn't involve monkeypatching the database adapter. But one of those monkeypatches might be a good escape hatch if you just want a one-time quick hack. And keep an eye on the Rails changelog to see if this makes it into core.