From 564047dbb9813db7b77101bf45cace021b15ac01 Mon Sep 17 00:00:00 2001 From: David Heinemeier Hansson Date: Thu, 5 Sep 2024 08:38:43 -0700 Subject: [PATCH] Slim down installer and have it use a final schema instead of migrations (#321) Following the same pattern we took with Solid Cache. This entails collapsing all the migrations into a premade `queue_schema.rb` and switching the default configuration to use a database configuration under the queue key. Drop all mentions and focus on running SQ off the same database. --- README.md | 159 +++++------------- ...0231211200639_create_solid_queue_tables.rb | 100 ----------- ...add_missing_index_to_blocked_executions.rb | 5 - ...40218110712_create_recurring_executions.rb | 14 -- .../20240719134516_create_recurring_tasks.rb | 20 --- .../20240811173327_add_name_to_processes.rb | 5 - .../20240813160053_make_name_not_null.rb | 16 -- ...ueue_recurring_tasks_static_to_not_null.rb | 5 - lib/generators/solid_queue/install/USAGE | 2 +- .../solid_queue/install/install_generator.rb | 34 +--- .../install/templates/{ => bin}/jobs | 0 .../solid_queue/install/templates/config.yml | 18 -- .../install/templates/config/solid_queue.yml | 18 ++ .../install/templates/db/queue_schema.rb | 129 ++++++++++++++ lib/solid_queue/tasks.rb | 5 + 15 files changed, 208 insertions(+), 322 deletions(-) delete mode 100644 db/migrate/20231211200639_create_solid_queue_tables.rb delete mode 100644 db/migrate/20240110143450_add_missing_index_to_blocked_executions.rb delete mode 100644 db/migrate/20240218110712_create_recurring_executions.rb delete mode 100644 db/migrate/20240719134516_create_recurring_tasks.rb delete mode 100644 db/migrate/20240811173327_add_name_to_processes.rb delete mode 100644 db/migrate/20240813160053_make_name_not_null.rb delete mode 100644 db/migrate/20240819165045_change_solid_queue_recurring_tasks_static_to_not_null.rb rename lib/generators/solid_queue/install/templates/{ => bin}/jobs (100%) mode change 100644 => 100755 delete mode 100644 lib/generators/solid_queue/install/templates/config.yml create mode 100644 lib/generators/solid_queue/install/templates/config/solid_queue.yml create mode 100644 lib/generators/solid_queue/install/templates/db/queue_schema.rb diff --git a/README.md b/README.md index 9175733a..db242038 100644 --- a/README.md +++ b/README.md @@ -6,43 +6,52 @@ Besides regular job enqueuing and processing, Solid Queue supports delayed jobs, Solid Queue can be used with SQL databases such as MySQL, PostgreSQL or SQLite, and it leverages the `FOR UPDATE SKIP LOCKED` clause, if available, to avoid blocking and waiting on locks when polling jobs. It relies on Active Job for retries, discarding, error handling, serialization, or delays, and it's compatible with Ruby on Rails multi-threading. -## Installation and usage -Add this line to your application's Gemfile: +## Installation -```ruby -gem "solid_queue" -``` +Solid Queue is configured by default in new Rails 8 applications. But if you're running an earlier version, you can add it manually following these steps: -And then execute: -```bash -$ bundle -``` +1. `bundle add solid_queue` +2. `bin/rails solid_queue:install` -Or install it yourself as: -```bash -$ gem install solid_queue -``` +This will configure Solid Queue as the production Active Job backend, create `config/solid_queue.yml`, and create the `db/queue_schema.rb`. -Now, you need to install the necessary migrations and configure the Active Job's adapter. You can do both at once using the provided generator: +You will then have to add the configuration for the queue database in `config/database.yml`. If you're using sqlite, it'll look like this: -```bash -$ bin/rails generate solid_queue:install +```yaml +production: + primary: + <<: *default + database: storage/production.sqlite3 + queue: + <<: *default + database: storage/production_queue.sqlite3 + migrations_paths: db/queue_migrate ``` -This will set `solid_queue` as the Active Job's adapter in production, and will copy the required migration over to your app. +...or if you're using MySQL/PostgreSQL/Trilogy: -Alternatively, you can skip setting the Active Job's adapter with: -```bash -$ bin/rails generate solid_queue:install --skip_adapter +```yaml +production: + primary: &primary_production + <<: *default + database: app_production + username: app + password: <%= ENV["APP_DATABASE_PASSWORD"] %> + queue: + <<: *primary_production + database: app_production_queue + migrations_paths: db/queue_migrate ``` -And set Solid Queue as your Active Job's queue backend manually, in your environment config: -```ruby -# config/environments/production.rb -config.active_job.queue_adapter = :solid_queue -``` +Then run `db:prepare` in production to ensure the database is created and the schema is loaded. + +Now you're ready to start processing jobs by running `bin/jobs` on the server that's doing the work. This will start processing jobs in all queues using the default configuration. See [below](#configuration) to learn more about configuring Solid Queue. + +For small projects, you can run Solid Queue on the same machine as your webserver. When you're ready to scale, Solid Queue supports horizontal scaling out-of-the-box. You can run Solid Queue on a separate server from your webserver, or even run `bin/jobs` on multiple machines at the same time. Depending on the configuration, you can designate some machines to run only dispatchers or only workers. See the [configuration](#configuration) section for more details on this. + +## Incremental adoption -Or you can set only specific jobs to use Solid Queue as their backend if you're migrating from another adapter and want to move jobs progressively: +If you're planning to adopt Solid Queue incrementally by switching one job at the time, you can do so by leaving the `config.active_job.queue_adapter` set to your old backend, and then set the `queue_adapter` directly in the jobs you're moving: ```ruby # app/jobs/my_job.rb @@ -52,24 +61,9 @@ class MyJob < ApplicationJob # ... end ``` +## High performance requirements -Finally, you need to run the migrations: - -```bash -$ bin/rails db:migrate -``` - -After this, you'll be ready to enqueue jobs using Solid Queue, but you need to start Solid Queue's supervisor to run them. You can use the provided binstub:` -```bash -$ bin/jobs -``` - -This will start processing jobs in all queues using the default configuration. See [below](#configuration) to learn more about configuring Solid Queue. - -For small projects, you can run Solid Queue on the same machine as your webserver. When you're ready to scale, Solid Queue supports horizontal scaling out-of-the-box. You can run Solid Queue on a separate server from your webserver, or even run `bin/jobs` on multiple machines at the same time. Depending on the configuration, you can designate some machines to run only dispatchers or only workers. See the [configuration](#configuration) section for more details on this. - -## Requirements -Besides Rails 7.1, Solid Queue works best with MySQL 8+ or PostgreSQL 9.5+, as they support `FOR UPDATE SKIP LOCKED`. You can use it with older versions, but in that case, you might run into lock waits if you run multiple workers for the same queue. +Solid Queue was designed for the highest throughput when used with MySQL 8+ or PostgreSQL 9.5+, as they support `FOR UPDATE SKIP LOCKED`. You can use it with older versions, but in that case, you might run into lock waits if you run multiple workers for the same queue. You can also use it with SQLite on smaller applications. ## Configuration @@ -135,8 +129,8 @@ Here's an overview of the different options: - `concurrency_maintenance`: whether the dispatcher will perform the concurrency maintenance work. This is `true` by default, and it's useful if you don't use any [concurrency controls](#concurrency-controls) and want to disable it or if you run multiple dispatchers and want some of them to just dispatch jobs without doing anything else. - `recurring_tasks`: a list of recurring tasks the dispatcher will manage. Read more details about this one in the [Recurring tasks](#recurring-tasks) section. - ### Queue order and priorities + As mentioned above, if you specify a list of queues for a worker, these will be polled in the order given, such as for the list `real_time,background`, no jobs will be taken from `background` unless there aren't any more jobs waiting in `real_time`. Active Job also supports positive integer priorities when enqueuing jobs. In Solid Queue, the smaller the value, the higher the priority. The default is `0`. @@ -159,54 +153,11 @@ When receiving a `QUIT` signal, if workers still have jobs in-flight, these will If processes have no chance of cleaning up before exiting (e.g. if someone pulls a cable somewhere), in-flight jobs might remain claimed by the processes executing them. Processes send heartbeats, and the supervisor checks and prunes processes with expired heartbeats, which will release any claimed jobs back to their queues. You can configure both the frequency of heartbeats and the threshold to consider a process dead. See the section below for this. -### Dedicated database configuration - -Solid Queue can be configured to run on a different database than the main application. - -Configure the `connects_to` option in `config/application.rb` or your environment config, with the custom database configuration that will be used in the abstract `SolidQueue::Record` Active Record model. - -```ruby - # Use a separate DB for Solid Queue - config.solid_queue.connects_to = { database: { writing: :solid_queue_primary, reading: :solid_queue_replica } } - ``` - -Add the dedicated database configuration to `config/database.yml`, differentiating between the main app's database and the dedicated `solid_queue` database. Make sure to include the `migrations_paths` for the solid queue database. This is where migration files for Solid Queue tables will reside. - -```yml -default: &default - adapter: sqlite3 - pool: <%= ENV.fetch("RAILS_MAX_THREADS") { 5 } %> - timeout: 5000 - -solid_queue: &solid_queue - <<: *default - migrations_paths: db/solid_queue_migrate - -development: - primary: - <<: *default - # ... - solid_queue_primary: - <<: *solid_queue - # ... - solid_queue_replica: - <<: *solid_queue - # ... -``` - -Install migrations and specify the dedicated database name with the `--database` option. This will create the Solid Queue migration files in a separate directory, matching the value provided in `migrations_paths` in `config/database.yml`. - -```bash -$ bin/rails g solid_queue:install --database solid_queue -``` - -Note: If you've already run the solid queue install command (`bin/rails generate solid_queue:install`) without a `--database` option, the migration files will have already been generated under the primary database's `db/migrate/` directory. You can remove these files and keep the ones generated by the database-specific migration installation above. +### Database configuration -Finally, run the migrations: +You can configure the database used by Solid Queue via the `config.solid_queue.connects_to` option in the `config/application.rb` or `config/environments/production.rb` config files. By default, a single database is used for both writing and reading called `queue` to match the database configuration you set up during the install. -```bash -$ bin/rails db:migrate -``` +All the options available to Active Record for multiple databases can be used here. ## Lifecycle hooks @@ -235,8 +186,8 @@ SolidQueue.on_stop { stop_metrics_server } These can be called several times to add multiple hooks, but it needs to happen before Solid Queue is started. An initializer would be a good place to do this. - ### Other configuration settings + _Note_: The settings in this section should be set in your `config/application.rb` or your environment config like this: `config.solid_queue.silence_polling = true` There are several settings that control how Solid Queue works that you can set as well: @@ -261,11 +212,13 @@ There are several settings that control how Solid Queue works that you can set a - `enqueue_after_transaction_commit`: whether the job queuing is deferred to after the current Active Record transaction is committed. The default is `false`. [Read more](https://github.com/rails/rails/pull/51426). ## Errors when enqueuing + Solid Queue will raise a `SolidQueue::Job::EnqueueError` for any Active Record errors that happen when enqueuing a job. The reason for not raising `ActiveJob::EnqueueError` is that this one gets handled by Active Job, causing `perform_later` to return `false` and set `job.enqueue_error`, yielding the job to a block that you need to pass to `perform_later`. This works very well for your own jobs, but makes failure very hard to handle for jobs enqueued by Rails or other gems, such as `Turbo::Streams::BroadcastJob` or `ActiveStorage::AnalyzeJob`, because you don't control the call to `perform_later` in that cases. In the case of recurring tasks, if such error is raised when enqueuing the job corresponding to the task, it'll be handled and logged but it won't bubble up. ## Concurrency controls + Solid Queue extends Active Job with concurrency controls, that allows you to limit how many jobs of a certain type or with certain arguments can run at the same time. When limited in this way, jobs will be blocked from running, and they'll stay blocked until another job finishes and unblocks them, or after the set expiry time (concurrency limit's _duration_) elapses. Jobs are never discarded or lost, only blocked. ```ruby @@ -331,35 +284,15 @@ failed_execution.discard # This will delete the job from the system However, we recommend taking a look at [mission_control-jobs](https://github.com/rails/mission_control-jobs), a dashboard where, among other things, you can examine and retry/discard failed jobs. ## Puma plugin + We provide a Puma plugin if you want to run the Solid Queue's supervisor together with Puma and have Puma monitor and manage it. You just need to add ```ruby plugin :solid_queue ``` to your `puma.rb` configuration. - -## Jobs and transactional integrity -:warning: Having your jobs in the same ACID-compliant database as your application data enables a powerful yet sharp tool: taking advantage of transactional integrity to ensure some action in your app is not committed unless your job is also committed. This can be very powerful and useful, but it can also backfire if you base some of your logic on this behaviour, and in the future, you move to another active job backend, or if you simply move Solid Queue to its own database, and suddenly the behaviour changes under you. - -By default, Solid Queue runs in the same DB as your app, and job enqueuing is _not_ deferred until any ongoing transaction is committed, which means that by default, you'll be taking advantage of this transactional integrity. - -If you prefer not to rely on this, or avoid relying on it unintentionally, you should make sure that: -- You set [`config.active_job.enqueue_after_transaction_commit`](https://edgeguides.rubyonrails.org/configuring.html#config-active-job-enqueue-after-transaction-commit) to `always`, if you're using Rails 7.2+. -- Or, your jobs relying on specific records are always enqueued on [`after_commit` callbacks](https://guides.rubyonrails.org/active_record_callbacks.html#after-commit-and-after-rollback) or otherwise from a place where you're certain that whatever data the job will use has been committed to the database before the job is enqueued. -- Or, you configure a different database for Solid Queue, even if it's the same as your app, ensuring that a different connection on the thread handling requests or running jobs for your app will be used to enqueue jobs. For example: - - ```ruby - class ApplicationRecord < ActiveRecord::Base - self.abstract_class = true - - connects_to database: { writing: :primary, reading: :replica } - ``` - - ```ruby - config.solid_queue.connects_to = { database: { writing: :primary, reading: :replica } } - ``` - ## Recurring tasks + Solid Queue supports defining recurring tasks that run at specific times in the future, on a regular basis like cron jobs. These are managed by dispatcher processes and as such, they can be defined in the dispatcher's configuration like this: ```yml dispatchers: diff --git a/db/migrate/20231211200639_create_solid_queue_tables.rb b/db/migrate/20231211200639_create_solid_queue_tables.rb deleted file mode 100644 index 4e1cf95c..00000000 --- a/db/migrate/20231211200639_create_solid_queue_tables.rb +++ /dev/null @@ -1,100 +0,0 @@ -class CreateSolidQueueTables < ActiveRecord::Migration[7.0] - def change - create_table :solid_queue_jobs do |t| - t.string :queue_name, null: false - t.string :class_name, null: false, index: true - t.text :arguments - t.integer :priority, default: 0, null: false - t.string :active_job_id, index: true - t.datetime :scheduled_at - t.datetime :finished_at, index: true - t.string :concurrency_key - - t.timestamps - - t.index [ :queue_name, :finished_at ], name: "index_solid_queue_jobs_for_filtering" - t.index [ :scheduled_at, :finished_at ], name: "index_solid_queue_jobs_for_alerting" - end - - create_table :solid_queue_scheduled_executions do |t| - t.references :job, index: { unique: true }, null: false - t.string :queue_name, null: false - t.integer :priority, default: 0, null: false - t.datetime :scheduled_at, null: false - - t.datetime :created_at, null: false - - t.index [ :scheduled_at, :priority, :job_id ], name: "index_solid_queue_dispatch_all" - end - - create_table :solid_queue_ready_executions do |t| - t.references :job, index: { unique: true }, null: false - t.string :queue_name, null: false - t.integer :priority, default: 0, null: false - - t.datetime :created_at, null: false - - t.index [ :priority, :job_id ], name: "index_solid_queue_poll_all" - t.index [ :queue_name, :priority, :job_id ], name: "index_solid_queue_poll_by_queue" - end - - create_table :solid_queue_claimed_executions do |t| - t.references :job, index: { unique: true }, null: false - t.bigint :process_id - t.datetime :created_at, null: false - - t.index [ :process_id, :job_id ] - end - - create_table :solid_queue_blocked_executions do |t| - t.references :job, index: { unique: true }, null: false - t.string :queue_name, null: false - t.integer :priority, default: 0, null: false - t.string :concurrency_key, null: false - t.datetime :expires_at, null: false - - t.datetime :created_at, null: false - - t.index [ :expires_at, :concurrency_key ], name: "index_solid_queue_blocked_executions_for_maintenance" - end - - create_table :solid_queue_failed_executions do |t| - t.references :job, index: { unique: true }, null: false - t.text :error - t.datetime :created_at, null: false - end - - create_table :solid_queue_pauses do |t| - t.string :queue_name, null: false, index: { unique: true } - t.datetime :created_at, null: false - end - - create_table :solid_queue_processes do |t| - t.string :kind, null: false - t.datetime :last_heartbeat_at, null: false, index: true - t.bigint :supervisor_id, index: true - - t.integer :pid, null: false - t.string :hostname - t.text :metadata - - t.datetime :created_at, null: false - end - - create_table :solid_queue_semaphores do |t| - t.string :key, null: false, index: { unique: true } - t.integer :value, default: 1, null: false - t.datetime :expires_at, null: false, index: true - - t.timestamps - - t.index [ :key, :value ], name: "index_solid_queue_semaphores_on_key_and_value" - end - - add_foreign_key :solid_queue_blocked_executions, :solid_queue_jobs, column: :job_id, on_delete: :cascade - add_foreign_key :solid_queue_claimed_executions, :solid_queue_jobs, column: :job_id, on_delete: :cascade - add_foreign_key :solid_queue_failed_executions, :solid_queue_jobs, column: :job_id, on_delete: :cascade - add_foreign_key :solid_queue_ready_executions, :solid_queue_jobs, column: :job_id, on_delete: :cascade - add_foreign_key :solid_queue_scheduled_executions, :solid_queue_jobs, column: :job_id, on_delete: :cascade - end -end diff --git a/db/migrate/20240110143450_add_missing_index_to_blocked_executions.rb b/db/migrate/20240110143450_add_missing_index_to_blocked_executions.rb deleted file mode 100644 index c6417000..00000000 --- a/db/migrate/20240110143450_add_missing_index_to_blocked_executions.rb +++ /dev/null @@ -1,5 +0,0 @@ -class AddMissingIndexToBlockedExecutions < ActiveRecord::Migration[7.1] - def change - add_index :solid_queue_blocked_executions, [ :concurrency_key, :priority, :job_id ], name: "index_solid_queue_blocked_executions_for_release" - end -end diff --git a/db/migrate/20240218110712_create_recurring_executions.rb b/db/migrate/20240218110712_create_recurring_executions.rb deleted file mode 100644 index 87c2ac64..00000000 --- a/db/migrate/20240218110712_create_recurring_executions.rb +++ /dev/null @@ -1,14 +0,0 @@ -class CreateRecurringExecutions < ActiveRecord::Migration[7.1] - def change - create_table :solid_queue_recurring_executions do |t| - t.references :job, index: { unique: true }, null: false - t.string :task_key, null: false - t.datetime :run_at, null: false - t.datetime :created_at, null: false - - t.index [ :task_key, :run_at ], unique: true - end - - add_foreign_key :solid_queue_recurring_executions, :solid_queue_jobs, column: :job_id, on_delete: :cascade - end -end diff --git a/db/migrate/20240719134516_create_recurring_tasks.rb b/db/migrate/20240719134516_create_recurring_tasks.rb deleted file mode 100644 index 6b2ff000..00000000 --- a/db/migrate/20240719134516_create_recurring_tasks.rb +++ /dev/null @@ -1,20 +0,0 @@ -class CreateRecurringTasks < ActiveRecord::Migration[7.1] - def change - create_table :solid_queue_recurring_tasks do |t| - t.string :key, null: false, index: { unique: true } - t.string :schedule, null: false - t.string :command, limit: 2048 - t.string :class_name - t.text :arguments - - t.string :queue_name - t.integer :priority, default: 0 - - t.boolean :static, default: true, index: true - - t.text :description - - t.timestamps - end - end -end diff --git a/db/migrate/20240811173327_add_name_to_processes.rb b/db/migrate/20240811173327_add_name_to_processes.rb deleted file mode 100644 index 1309e2cc..00000000 --- a/db/migrate/20240811173327_add_name_to_processes.rb +++ /dev/null @@ -1,5 +0,0 @@ -class AddNameToProcesses < ActiveRecord::Migration[7.1] - def change - add_column :solid_queue_processes, :name, :string - end -end diff --git a/db/migrate/20240813160053_make_name_not_null.rb b/db/migrate/20240813160053_make_name_not_null.rb deleted file mode 100644 index 4a49c226..00000000 --- a/db/migrate/20240813160053_make_name_not_null.rb +++ /dev/null @@ -1,16 +0,0 @@ -class MakeNameNotNull < ActiveRecord::Migration[7.1] - def up - SolidQueue::Process.where(name: nil).find_each do |process| - process.name ||= [ process.kind.downcase, SecureRandom.hex(10) ].join("-") - process.save! - end - - change_column :solid_queue_processes, :name, :string, null: false - add_index :solid_queue_processes, [ :name, :supervisor_id ], unique: true - end - - def down - remove_index :solid_queue_processes, [ :name, :supervisor_id ] - change_column :solid_queue_processes, :name, :string, null: true - end -end diff --git a/db/migrate/20240819165045_change_solid_queue_recurring_tasks_static_to_not_null.rb b/db/migrate/20240819165045_change_solid_queue_recurring_tasks_static_to_not_null.rb deleted file mode 100644 index 89866beb..00000000 --- a/db/migrate/20240819165045_change_solid_queue_recurring_tasks_static_to_not_null.rb +++ /dev/null @@ -1,5 +0,0 @@ -class ChangeSolidQueueRecurringTasksStaticToNotNull < ActiveRecord::Migration[7.1] - def change - change_column_null :solid_queue_recurring_tasks, :static, false, true - end -end diff --git a/lib/generators/solid_queue/install/USAGE b/lib/generators/solid_queue/install/USAGE index b3e1248b..2450ad83 100644 --- a/lib/generators/solid_queue/install/USAGE +++ b/lib/generators/solid_queue/install/USAGE @@ -5,6 +5,6 @@ Example: bin/rails generate solid_queue:install This will perform the following: - Installs solid_queue migrations + Adds solid_queue db schema Replaces Active Job's adapter in environment configuration Installs bin/jobs binstub to start the supervisor diff --git a/lib/generators/solid_queue/install/install_generator.rb b/lib/generators/solid_queue/install/install_generator.rb index f77366e1..c7e8330d 100644 --- a/lib/generators/solid_queue/install/install_generator.rb +++ b/lib/generators/solid_queue/install/install_generator.rb @@ -3,33 +3,17 @@ class SolidQueue::InstallGenerator < Rails::Generators::Base source_root File.expand_path("templates", __dir__) - class_option :skip_adapter, type: :boolean, default: nil, desc: "Skip setting Solid Queue as the Active Job's adapter" - class_option :database, type: :string, default: nil, desc: "The database to use for migrations, if different from the primary one." - - def add_solid_queue - unless options[:skip_adapter] - if (env_config = Pathname(destination_root).join("config/environments/production.rb")).exist? - say "Setting solid_queue as Active Job's queue adapter" - gsub_file env_config, /(# )?config\.active_job\.queue_adapter\s+=.*/, "config.active_job.queue_adapter = :solid_queue" - end - end - - if File.exist?("config/solid_queue.yml") - say "Skipping sample configuration as config/solid_queue.yml exists" - else - say "Copying sample configuration" - copy_file "config.yml", "config/solid_queue.yml" - end - - say "Copying binstub" - copy_file "jobs", "bin/jobs" + def copy_files + template "config/solid_queue.yml" + template "db/queue_schema.rb" + template "bin/jobs" chmod "bin/jobs", 0755 & ~File.umask, verbose: false end - def create_migrations - say "Installing database migrations" - arguments = [ "FROM=solid_queue" ] - arguments << "DATABASE=#{options[:database]}" if options[:database].present? - rails_command "railties:install:migrations #{arguments.join(" ")}", inline: true + def configure_active_job_adapter + gsub_file Pathname(destination_root).join("config/environments/production.rb"), + /(# )?config\.active_job\.queue_adapter\s+=.*/, + "config.active_job.queue_adapter = :solid_queue\n" + + " config.solid_queue.connects_to = { database: { writing: :queue } }\n" end end diff --git a/lib/generators/solid_queue/install/templates/jobs b/lib/generators/solid_queue/install/templates/bin/jobs old mode 100644 new mode 100755 similarity index 100% rename from lib/generators/solid_queue/install/templates/jobs rename to lib/generators/solid_queue/install/templates/bin/jobs diff --git a/lib/generators/solid_queue/install/templates/config.yml b/lib/generators/solid_queue/install/templates/config.yml deleted file mode 100644 index 2d8d922f..00000000 --- a/lib/generators/solid_queue/install/templates/config.yml +++ /dev/null @@ -1,18 +0,0 @@ -# default: &default -# dispatchers: -# - polling_interval: 1 -# batch_size: 500 -# workers: -# - queues: "*" -# threads: 3 -# processes: 1 -# polling_interval: 0.1 -# -# development: -# <<: *default -# -# test: -# <<: *default -# -# production: -# <<: *default diff --git a/lib/generators/solid_queue/install/templates/config/solid_queue.yml b/lib/generators/solid_queue/install/templates/config/solid_queue.yml new file mode 100644 index 00000000..15691e9d --- /dev/null +++ b/lib/generators/solid_queue/install/templates/config/solid_queue.yml @@ -0,0 +1,18 @@ +default: &default + dispatchers: + - polling_interval: 1 + batch_size: 500 + workers: + - queues: "*" + threads: 3 + processes: <%%= ENV.fetch("JOB_CONCURRENCY", 1) %> + polling_interval: 0.1 + +development: + <<: *default + +test: + <<: *default + +production: + <<: *default diff --git a/lib/generators/solid_queue/install/templates/db/queue_schema.rb b/lib/generators/solid_queue/install/templates/db/queue_schema.rb new file mode 100644 index 00000000..420bd3a7 --- /dev/null +++ b/lib/generators/solid_queue/install/templates/db/queue_schema.rb @@ -0,0 +1,129 @@ +ActiveRecord::Schema[8.0].define(version: 2024_09_04_193154) do + create_table "solid_queue_blocked_executions", force: :cascade do |t| + t.integer "job_id", null: false + t.string "queue_name", null: false + t.integer "priority", default: 0, null: false + t.string "concurrency_key", null: false + t.datetime "expires_at", null: false + t.datetime "created_at", null: false + t.index [ "concurrency_key", "priority", "job_id" ], name: "index_solid_queue_blocked_executions_for_release" + t.index [ "expires_at", "concurrency_key" ], name: "index_solid_queue_blocked_executions_for_maintenance" + t.index [ "job_id" ], name: "index_solid_queue_blocked_executions_on_job_id", unique: true + end + + create_table "solid_queue_claimed_executions", force: :cascade do |t| + t.integer "job_id", null: false + t.bigint "process_id" + t.datetime "created_at", null: false + t.index [ "job_id" ], name: "index_solid_queue_claimed_executions_on_job_id", unique: true + t.index [ "process_id", "job_id" ], name: "index_solid_queue_claimed_executions_on_process_id_and_job_id" + end + + create_table "solid_queue_failed_executions", force: :cascade do |t| + t.integer "job_id", null: false + t.text "error" + t.datetime "created_at", null: false + t.index [ "job_id" ], name: "index_solid_queue_failed_executions_on_job_id", unique: true + end + + create_table "solid_queue_jobs", force: :cascade do |t| + t.string "queue_name", null: false + t.string "class_name", null: false + t.text "arguments" + t.integer "priority", default: 0, null: false + t.string "active_job_id" + t.datetime "scheduled_at" + t.datetime "finished_at" + t.string "concurrency_key" + t.datetime "created_at", null: false + t.datetime "updated_at", null: false + t.index [ "active_job_id" ], name: "index_solid_queue_jobs_on_active_job_id" + t.index [ "class_name" ], name: "index_solid_queue_jobs_on_class_name" + t.index [ "finished_at" ], name: "index_solid_queue_jobs_on_finished_at" + t.index [ "queue_name", "finished_at" ], name: "index_solid_queue_jobs_for_filtering" + t.index [ "scheduled_at", "finished_at" ], name: "index_solid_queue_jobs_for_alerting" + end + + create_table "solid_queue_pauses", force: :cascade do |t| + t.string "queue_name", null: false + t.datetime "created_at", null: false + t.index [ "queue_name" ], name: "index_solid_queue_pauses_on_queue_name", unique: true + end + + create_table "solid_queue_processes", force: :cascade do |t| + t.string "kind", null: false + t.datetime "last_heartbeat_at", null: false + t.bigint "supervisor_id" + t.integer "pid", null: false + t.string "hostname" + t.text "metadata" + t.datetime "created_at", null: false + t.string "name", null: false + t.index [ "last_heartbeat_at" ], name: "index_solid_queue_processes_on_last_heartbeat_at" + t.index [ "name", "supervisor_id" ], name: "index_solid_queue_processes_on_name_and_supervisor_id", unique: true + t.index [ "supervisor_id" ], name: "index_solid_queue_processes_on_supervisor_id" + end + + create_table "solid_queue_ready_executions", force: :cascade do |t| + t.integer "job_id", null: false + t.string "queue_name", null: false + t.integer "priority", default: 0, null: false + t.datetime "created_at", null: false + t.index [ "job_id" ], name: "index_solid_queue_ready_executions_on_job_id", unique: true + t.index [ "priority", "job_id" ], name: "index_solid_queue_poll_all" + t.index [ "queue_name", "priority", "job_id" ], name: "index_solid_queue_poll_by_queue" + end + + create_table "solid_queue_recurring_executions", force: :cascade do |t| + t.integer "job_id", null: false + t.string "task_key", null: false + t.datetime "run_at", null: false + t.datetime "created_at", null: false + t.index [ "job_id" ], name: "index_solid_queue_recurring_executions_on_job_id", unique: true + t.index [ "task_key", "run_at" ], name: "index_solid_queue_recurring_executions_on_task_key_and_run_at", unique: true + end + + create_table "solid_queue_recurring_tasks", force: :cascade do |t| + t.string "key", null: false + t.string "schedule", null: false + t.string "command", limit: 2048 + t.string "class_name" + t.text "arguments" + t.string "queue_name" + t.integer "priority", default: 0 + t.boolean "static", default: true, null: false + t.text "description" + t.datetime "created_at", null: false + t.datetime "updated_at", null: false + t.index [ "key" ], name: "index_solid_queue_recurring_tasks_on_key", unique: true + t.index [ "static" ], name: "index_solid_queue_recurring_tasks_on_static" + end + + create_table "solid_queue_scheduled_executions", force: :cascade do |t| + t.integer "job_id", null: false + t.string "queue_name", null: false + t.integer "priority", default: 0, null: false + t.datetime "scheduled_at", null: false + t.datetime "created_at", null: false + t.index [ "job_id" ], name: "index_solid_queue_scheduled_executions_on_job_id", unique: true + t.index [ "scheduled_at", "priority", "job_id" ], name: "index_solid_queue_dispatch_all" + end + + create_table "solid_queue_semaphores", force: :cascade do |t| + t.string "key", null: false + t.integer "value", default: 1, null: false + t.datetime "expires_at", null: false + t.datetime "created_at", null: false + t.datetime "updated_at", null: false + t.index [ "expires_at" ], name: "index_solid_queue_semaphores_on_expires_at" + t.index [ "key", "value" ], name: "index_solid_queue_semaphores_on_key_and_value" + t.index [ "key" ], name: "index_solid_queue_semaphores_on_key", unique: true + end + + add_foreign_key "solid_queue_blocked_executions", "solid_queue_jobs", column: "job_id", on_delete: :cascade + add_foreign_key "solid_queue_claimed_executions", "solid_queue_jobs", column: "job_id", on_delete: :cascade + add_foreign_key "solid_queue_failed_executions", "solid_queue_jobs", column: "job_id", on_delete: :cascade + add_foreign_key "solid_queue_ready_executions", "solid_queue_jobs", column: "job_id", on_delete: :cascade + add_foreign_key "solid_queue_recurring_executions", "solid_queue_jobs", column: "job_id", on_delete: :cascade + add_foreign_key "solid_queue_scheduled_executions", "solid_queue_jobs", column: "job_id", on_delete: :cascade +end diff --git a/lib/solid_queue/tasks.rb b/lib/solid_queue/tasks.rb index 9b387b85..91cd778b 100644 --- a/lib/solid_queue/tasks.rb +++ b/lib/solid_queue/tasks.rb @@ -1,4 +1,9 @@ namespace :solid_queue do + desc "Install Solid Queue" + task :install do + Rails::Command.invoke :generate, [ "solid_queue:install" ] + end + desc "start solid_queue supervisor to dispatch and process jobs" task start: :environment do SolidQueue::Supervisor.start