Fx
Versioned database functions and triggers for Rails
Install / Use
/learn @teoljungberg/FxREADME
F(x)
F(x) adds methods to ActiveRecord::Migration to create and manage database
functions and triggers in Rails.
Using F(x), you can bring the power of SQL functions and triggers to your Rails application without having to switch your schema format to SQL. F(x) provides a convention for versioning functions and triggers that keeps your migration history consistent and reversible and avoids having to duplicate SQL strings across migrations. As an added bonus, you define the structure of your function in a SQL file, meaning you get full SQL syntax highlighting in the editor of your choice and can easily test your SQL in the database console during development.
F(x) ships with support for PostgreSQL. The adapter is configurable (see
Fx::Configuration) and has a minimal interface (see
Fx::Adapters::Postgres) that other gems can provide.
Great, how do I create a trigger and a function?
You've got this great idea for a function you'd like to call
uppercase_users_name. You can create the migration and the corresponding
definition file with the following command:
% rails generate fx:function uppercase_users_name
create db/functions/uppercase_users_name_v01.sql
create db/migrate/[TIMESTAMP]_create_function_uppercase_users_name.rb
Edit the db/functions/uppercase_users_name_v01.sql file with the SQL statement
that defines your function.
Next, let's add a trigger called uppercase_users_name to call our new
function each time we INSERT on the users table.
% rails generate fx:trigger uppercase_users_name table_name:users
create db/triggers/uppercase_users_name_v01.sql
create db/migrate/[TIMESTAMP]_create_trigger_uppercase_users_name.rb
In our example, this might look something like this:
CREATE TRIGGER uppercase_users_name
BEFORE INSERT ON users
FOR EACH ROW
EXECUTE FUNCTION uppercase_users_name();
The generated migrations contains create_function and create_trigger
statements. The migration is reversible and the schema will be dumped into your
schema.rb file.
% rake db:migrate
Cool, but what if I need to change a trigger or function?
Here's where F(x) really shines. Run that same function generator once more:
% rails generate fx:function uppercase_users_name
create db/functions/uppercase_users_name_v02.sql
create db/migrate/[TIMESTAMP]_update_function_uppercase_users_name_to_version_2.rb
F(x) detected that we already had an existing uppercase_users_name function at
version 1, created a copy of that definition as version 2, and created a
migration to update to the version 2 schema. All that's left for you to do is
tweak the schema in the new definition and run the update_function migration.
I don't need this trigger or function anymore. Make it go away.
F(x) gives you drop_trigger and drop_function too:
def change
drop_function :uppercase_users_name, revert_to_version: 2
end
What if I need to use a function as the default value of a column?
You need to set F(x) to dump the functions in the beginning of db/schema.rb in a initializer:
# config/initializers/fx.rb
Fx.configure do |config|
config.dump_functions_at_beginning_of_schema = true
end
And then you can use a lambda in your migration file:
create_table :my_table do |t|
t.string :my_column, default: -> { "my_function()" }
end
That's how you tell Rails to use the default as a literal SQL for the default column value instead of a plain string.
Customizing Schema Dump Order
By default, functions and triggers are dumped to schema.rb in the order
returned by the database. If you need a specific ordering (e.g., alphabetical
for deterministic diffs), subclass the adapter and override #functions or
#triggers. These methods are part of the adapter's public API and will remain
stable across releases:
# config/initializers/fx.rb
class SortedPostgresAdapter < Fx::Adapters::Postgres
def functions
super.sort_by(&:name)
end
def triggers
super.sort_by(&:name)
end
end
Fx.configure do |config|
config.database = SortedPostgresAdapter.new
end
The same approach works for more advanced ordering. For example, if your
functions depend on each other and need to be dumped in dependency order, you
could use Ruby's built-in TSort to topologically sort them.
Plugins/Adapters
Version Support
F(x) follows the maintenance policies of Ruby, Rails, and PostgreSQL, supporting versions within their official maintenance windows.
Ruby: 3.2+ (maintenance branches)
Rails: 7.2, 8.0, 8.1 (maintenance policy)
PostgreSQL: 14, 15, 16, 17, 18 (versioning policy)
When a version reaches end-of-life, support will be dropped in the next minor release of F(x). Older versions may continue to work but are not tested or guaranteed.
Contributing
See contributing for more details.
