| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200 |
- import { Type } from '@vendure/common/lib/shared-types';
- import { Job } from '../../job-queue/job';
- import { BackoffStrategy } from '../../job-queue/polling-job-queue-strategy';
- import { PluginCommonModule } from '../plugin-common.module';
- import { VendurePlugin } from '../vendure-plugin';
- import { JobRecordBuffer } from './job-record-buffer.entity';
- import { JobRecord } from './job-record.entity';
- import { SqlJobBufferStorageStrategy } from './sql-job-buffer-storage-strategy';
- import { SqlJobQueueStrategy } from './sql-job-queue-strategy';
- /**
- * @description
- * Configuration options for the DefaultJobQueuePlugin. These values get passed into the
- * {@link SqlJobQueueStrategy}.
- *
- * @docsCategory JobQueue
- * @docsPage DefaultJobQueuePlugin
- */
- export interface DefaultJobQueueOptions {
- /**
- * @description
- * The interval in ms between polling the database for new jobs. If many job queues
- * are active, the polling may cause undue load on the database, in which case this value
- * should be increased to e.g. 1000.
- *
- * @default 200
- */
- pollInterval?: number | ((queueName: string) => number);
- /**
- * @description
- * How many jobs from a given queue to process concurrently.
- *
- * @default 1
- */
- concurrency?: number;
- /**
- * @description
- * The strategy used to decide how long to wait before retrying a failed job.
- *
- * @default () => 1000
- */
- backoffStrategy?: BackoffStrategy;
- /**
- * @description
- * When a job is added to the JobQueue using `JobQueue.add()`, the calling
- * code may specify the number of retries in case of failure. This option allows
- * you to override that number and specify your own number of retries based on
- * the job being added.
- *
- * @example
- * ```TypeScript
- * setRetries: (queueName, job) => {
- * if (queueName === 'send-email') {
- * // Override the default number of retries
- * // for the 'send-email' job because we have
- * // a very unreliable email service.
- * return 10;
- * }
- * return job.retries;
- * }
- * ```
- * @param queueName
- * @param job
- */
- setRetries?: (queueName: string, job: Job) => number;
- /**
- * @description
- * If set to `true`, the database will be used to store buffered jobs. This is
- * recommended for production.
- *
- * When enabled, a new `JobRecordBuffer` database entity will be defined which will
- * require a migration when first enabling this option.
- *
- * @since 1.3.0
- */
- useDatabaseForBuffer?: boolean;
- }
- /**
- * @description
- * A plugin which configures Vendure to use the SQL database to persist the JobQueue jobs using the {@link SqlJobQueueStrategy}. If you add this
- * plugin to an existing Vendure installation, you'll need to run a [database migration](/docs/developer-guide/migrations), since this
- * plugin will add a new "job_record" table to the database.
- *
- * @example
- * ```TypeScript
- * import { DefaultJobQueuePlugin, VendureConfig } from '\@vendure/core';
- *
- * export const config: VendureConfig = {
- * // Add an instance of the plugin to the plugins array
- * plugins: [
- * DefaultJobQueuePlugin,
- * ],
- * };
- * ```
- *
- * ## Configuration
- *
- * It is possible to configure the behaviour of the {@link SqlJobQueueStrategy} by passing options to the static `init()` function:
- *
- * ### pollInterval
- * The interval in ms between polling for new jobs. The default is 200ms.
- * Using a longer interval reduces load on the database but results in a slight
- * delay in processing jobs. For more control, it is possible to supply a function which can specify
- * a pollInterval based on the queue name:
- *
- * @example
- * ```TypeScript
- * export const config: VendureConfig = {
- * plugins: [
- * DefaultJobQueuePlugin.init({
- * pollInterval: queueName => {
- * if (queueName === 'cart-recovery-email') {
- * // This queue does not need to be polled so frequently,
- * // so we set a longer interval in order to reduce load
- * // on the database.
- * return 10000;
- * }
- * return 200;
- * },
- * }),
- * ],
- * };
- * ```
- * ### concurrency
- * The number of jobs to process concurrently per worker. Defaults to 1.
- *
- * ### backoffStrategy
- * Defines the backoff strategy used when retrying failed jobs. In other words, if a job fails
- * and is configured to be re-tried, how long should we wait before the next attempt?
- *
- * By default, a job will be retried as soon as possible, but in some cases this is not desirable. For example,
- * a job may interact with an unreliable 3rd-party API which is sensitive to too many requests. In this case, an
- * exponential backoff may be used which progressively increases the delay between each subsequent retry.
- *
- * @example
- * ```TypeScript
- * export const config: VendureConfig = {
- * plugins: [
- * DefaultJobQueuePlugin.init({
- * pollInterval: 5000,
- * concurrency: 2
- * backoffStrategy: (queueName, attemptsMade, job) => {
- * if (queueName === 'transcode-video') {
- * // exponential backoff example
- * return (attemptsMade ** 2) * 1000;
- * }
- *
- * // A default delay for all other queues
- * return 1000;
- * },
- * setRetries: (queueName, job) => {
- * if (queueName === 'send-email') {
- * // Override the default number of retries
- * // for the 'send-email' job because we have
- * // a very unreliable email service.
- * return 10;
- * }
- * return job.retries;
- * }
- * }),
- * ],
- * };
- * ```
- *
- * @docsCategory JobQueue
- * @docsWeight 0
- */
- @VendurePlugin({
- imports: [PluginCommonModule],
- entities: () =>
- DefaultJobQueuePlugin.options.useDatabaseForBuffer === true
- ? [JobRecord, JobRecordBuffer]
- : [JobRecord],
- configuration: config => {
- const { pollInterval, concurrency, backoffStrategy, setRetries } =
- DefaultJobQueuePlugin.options ?? {};
- config.jobQueueOptions.jobQueueStrategy = new SqlJobQueueStrategy({
- concurrency,
- pollInterval,
- backoffStrategy,
- setRetries,
- });
- if (DefaultJobQueuePlugin.options.useDatabaseForBuffer === true) {
- config.jobQueueOptions.jobBufferStorageStrategy = new SqlJobBufferStorageStrategy();
- }
- return config;
- },
- })
- export class DefaultJobQueuePlugin {
- /** @internal */
- static options: DefaultJobQueueOptions = {};
- static init(options: DefaultJobQueueOptions): Type<DefaultJobQueuePlugin> {
- DefaultJobQueuePlugin.options = options;
- return DefaultJobQueuePlugin;
- }
- }
|