jpskill.com
📄 ドキュメント コミュニティ

job-queue

BullMQ、Celery、Sidekiqなどを活用し、メール送信やPDF生成など時間のかかる処理をバックグラウンドで実行することで、Webサイトの応答速度を向上させ、ジョブの優先順位付けやスケジュール管理、進捗状況の追跡などを効率的に行うシステムを構築するSkill。

📜 元の英語説明(参考)

Build job queues and background worker systems using BullMQ, Celery, or Sidekiq. Use when you need to offload slow tasks from request handlers — email sending, PDF generation, image processing, data exports, or any work that takes more than a few hundred milliseconds. Covers job priorities, concurrency control, scheduled jobs, progress tracking, and graceful shutdown. Trigger words: background job, worker, queue, async task, BullMQ, Celery, cron job, scheduled task, job retry.

🇯🇵 日本人クリエイター向け解説

一言でいうと

BullMQ、Celery、Sidekiqなどを活用し、メール送信やPDF生成など時間のかかる処理をバックグラウンドで実行することで、Webサイトの応答速度を向上させ、ジョブの優先順位付けやスケジュール管理、進捗状況の追跡などを効率的に行うシステムを構築するSkill。

※ jpskill.com 編集部が日本のビジネス現場向けに補足した解説です。Skill本体の挙動とは独立した参考情報です。

⚡ おすすめ: コマンド1行でインストール(60秒)

下記のコマンドをコピーしてターミナル(Mac/Linux)または PowerShell(Windows)に貼り付けてください。 ダウンロード → 解凍 → 配置まで全自動。

🍎 Mac / 🐧 Linux
mkdir -p ~/.claude/skills && cd ~/.claude/skills && curl -L -o job-queue.zip https://jpskill.com/download/15025.zip && unzip -o job-queue.zip && rm job-queue.zip
🪟 Windows (PowerShell)
$d = "$env:USERPROFILE\.claude\skills"; ni -Force -ItemType Directory $d | Out-Null; iwr https://jpskill.com/download/15025.zip -OutFile "$d\job-queue.zip"; Expand-Archive "$d\job-queue.zip" -DestinationPath $d -Force; ri "$d\job-queue.zip"

完了後、Claude Code を再起動 → 普通に「動画プロンプト作って」のように話しかけるだけで自動発動します。

💾 手動でダウンロードしたい(コマンドが難しい人向け)
  1. 1. 下の青いボタンを押して job-queue.zip をダウンロード
  2. 2. ZIPファイルをダブルクリックで解凍 → job-queue フォルダができる
  3. 3. そのフォルダを C:\Users\あなたの名前\.claude\skills\(Win)または ~/.claude/skills/(Mac)へ移動
  4. 4. Claude Code を再起動

⚠️ ダウンロード・利用は自己責任でお願いします。当サイトは内容・動作・安全性について責任を負いません。

🎯 このSkillでできること

下記の説明文を読むと、このSkillがあなたに何をしてくれるかが分かります。Claudeにこの分野の依頼をすると、自動で発動します。

📦 インストール方法 (3ステップ)

  1. 1. 上の「ダウンロード」ボタンを押して .skill ファイルを取得
  2. 2. ファイル名の拡張子を .skill から .zip に変えて展開(macは自動展開可)
  3. 3. 展開してできたフォルダを、ホームフォルダの .claude/skills/ に置く
    • · macOS / Linux: ~/.claude/skills/
    • · Windows: %USERPROFILE%\.claude\skills\

Claude Code を再起動すれば完了。「このSkillを使って…」と話しかけなくても、関連する依頼で自動的に呼び出されます。

詳しい使い方ガイドを見る →
最終更新
2026-05-18
取得日時
2026-05-18
同梱ファイル
1

📖 Skill本文(日本語訳)

※ 原文(英語/中国語)を Gemini で日本語化したものです。Claude 自身は原文を読みます。誤訳がある場合は原文をご確認ください。

ジョブキュー

概要

このスキルは、本番環境で使用できるバックグラウンドジョブ処理システムを構築するのに役立ちます。キューのアーキテクチャ、ワーカーの並行性、ジョブの優先度、リトライ戦略、スケジュールされたジョブ/定期的なジョブ、進捗状況の報告、および正常なシャットダウンについて説明します。これらのパターンは、BullMQ (Node.js)、Celery (Python)、および Sidekiq (Ruby) で動作します。

手順

1. キューを設定し、ジョブの種類を定義する

型付きのジョブ定義とキューインスタンスを作成します。

// src/jobs/types.ts
export interface JobMap {
  "email:send": { to: string; template: string; data: Record<string, string> };
  "pdf:generate": { reportId: string; format: "a4" | "letter" };
  "export:csv": { userId: string; query: string; columns: string[] };
  "image:resize": { sourceUrl: string; widths: number[] };
}

// src/jobs/queues.ts
import { Queue } from "bullmq";
import { JobMap } from "./types";

const connection = { host: "localhost", port: 6379 };

export const emailQueue = new Queue<JobMap["email:send"]>("email", { connection });
export const pdfQueue = new Queue<JobMap["pdf:generate"]>("pdf", { connection });
export const exportQueue = new Queue<JobMap["export:csv"]>("export", { connection });
export const imageQueue = new Queue<JobMap["image:resize"]>("image", { connection });

2. 並行性制御を使用してワーカーを実装する

// src/workers/email-worker.ts
import { Worker, Job } from "bullmq";
import { JobMap } from "../jobs/types";

const emailWorker = new Worker<JobMap["email:send"]>(
  "email",
  async (job: Job) => {
    const { to, template, data } = job.data;
    await job.updateProgress(10);
    const html = await renderTemplate(template, data);
    await job.updateProgress(50);
    await sendEmail(to, html);
    await job.updateProgress(100);
    return { sentAt: new Date().toISOString() };
  },
  {
    connection: { host: "localhost", port: 6379 },
    concurrency: 10,        // 10個のメールを並行して処理する
    limiter: { max: 100, duration: 60000 }, // レート制限: 100/分
  }
);

emailWorker.on("completed", (job) => {
  console.log(`Email sent: job ${job.id} → ${job.data.to}`);
});

emailWorker.on("failed", (job, err) => {
  console.error(`Email failed: job ${job?.id} — ${err.message}`);
});

3. ジョブのスケジュールと優先度を追加する

// Delayed job — サインアップから30分後にウェルカムメールを送信する
await emailQueue.add("email:send", {
  to: "newuser@example.com",
  template: "welcome",
  data: { name: "Alex" },
}, { delay: 30 * 60 * 1000 });

// Priority jobs — パスワードリセットをキューの先頭に移動する
await emailQueue.add("email:send", {
  to: "user@example.com",
  template: "password-reset",
  data: { resetLink: "https://app.example.com/reset/abc123" },
}, { priority: 1 }); // 数値が小さいほど優先度が高い

// Recurring job — 毎日午前8時(UTC)にダイジェストを送信する
await emailQueue.add("email:send", {
  to: "digest",
  template: "daily-digest",
  data: {},
}, {
  repeat: { pattern: "0 8 * * *" },
  jobId: "daily-digest", // 重複を防ぐ
});

4. 正常なシャットダウンを実装する

// src/workers/shutdown.ts
const workers = [emailWorker, pdfWorker, exportWorker, imageWorker];

async function gracefulShutdown(signal: string): Promise<void> {
  console.log(`Received ${signal}. Closing workers gracefully...`);
  await Promise.all(workers.map((w) => w.close()));
  console.log("All workers closed. Exiting.");
  process.exit(0);
}

process.on("SIGTERM", () => gracefulShutdown("SIGTERM"));
process.on("SIGINT", () => gracefulShutdown("SIGINT"));

例 1: PDFレポート生成キュー

プロンプト: 「PDFレポートを生成するためのバックグラウンドジョブシステムを構築します。ユーザーはレポートをリクエストし、すぐにジョブIDを取得し、進捗状況をポーリングできます。レポートの生成には10〜30秒かかります。」

エージェントの出力:

  • 型付きジョブ定義を含む src/jobs/pdf-queue.ts を作成します
  • 各段階(データのクエリ → フォーマット → レンダリング → アップロード)で進捗状況を更新する src/workers/pdf-worker.ts を作成します
  • POST /reports (エンキュー、ジョブIDを返す) および GET /reports/:jobId/status (完了時に進捗率とダウンロードURLを返す) を含む src/routes/reports.ts を作成します
  • リトライロジックを追加します: 10秒のバックオフで3回試行

例 2: 画像処理パイプライン

プロンプト: 「アップロードされた画像を処理する必要があります: 3つの幅 (200、800、1600px) にリサイズし、WebP に変換して、クラウドストレージにアップロードします。1時間あたり最大500枚の画像を処理します。」

エージェントの出力:

  • sharpベースのリサイズおよび変換パイプラインを含む src/workers/image-worker.ts を作成します
  • 並行性を4に設定します(CPUバウンドの作業、コア数と一致)
  • 画像ごとの進捗状況の追跡を追加します(バッチアップロードに役立ちます)
  • チェーンされたジョブとして、リサイズ → 変換 → アップロードのフローを含む src/jobs/image-pipeline.ts を作成します

ガイドライン

  • ジョブをシリアライズ可能に保つ — ジョブデータはJSONのラウンドトリップに耐える必要があります。バッファやストリームではなく、IDとURLを渡します。
  • 適切な並行性を設定する — CPUバウンドの作業(画像処理):コア数と一致。I/Oバウンド(メール、API呼び出し):10〜50の並行処理。
  • 常に正常なシャットダウンを実装するSIGTERM は、プロセスが終了する前に実行中のジョブを完了させる必要があります。
  • ジョブIDを冪等性に使用する — 決定論的な jobId を設定して、同じジョブが2回エンキューされるのを防ぎます。
  • キューの深さを監視する — キューの増加は、ワーカーが追いついていないことを意味します。バックログが5分間の処理時間を超えた場合に警告します。
  • ワークロードの種類ごとにキューを分離する — 低速なPDF生成が高速なメール送信をブロックしないようにします。
  • 結果を外部に保存する — BullMQジョブの結果はデフォルトでクリーンアップされます。重要な結果はデータベースに永続化します。
📜 原文 SKILL.md(Claudeが読む英語/中国語)を展開

Job Queue

Overview

This skill helps you build production-grade background job processing systems. It covers queue architecture, worker concurrency, job priorities, retry strategies, scheduled/recurring jobs, progress reporting, and graceful shutdown. The patterns work across BullMQ (Node.js), Celery (Python), and Sidekiq (Ruby).

Instructions

1. Set up the queue and define job types

Create typed job definitions and a queue instance:

// src/jobs/types.ts
export interface JobMap {
  "email:send": { to: string; template: string; data: Record<string, string> };
  "pdf:generate": { reportId: string; format: "a4" | "letter" };
  "export:csv": { userId: string; query: string; columns: string[] };
  "image:resize": { sourceUrl: string; widths: number[] };
}

// src/jobs/queues.ts
import { Queue } from "bullmq";
import { JobMap } from "./types";

const connection = { host: "localhost", port: 6379 };

export const emailQueue = new Queue<JobMap["email:send"]>("email", { connection });
export const pdfQueue = new Queue<JobMap["pdf:generate"]>("pdf", { connection });
export const exportQueue = new Queue<JobMap["export:csv"]>("export", { connection });
export const imageQueue = new Queue<JobMap["image:resize"]>("image", { connection });

2. Implement workers with concurrency control

// src/workers/email-worker.ts
import { Worker, Job } from "bullmq";
import { JobMap } from "../jobs/types";

const emailWorker = new Worker<JobMap["email:send"]>(
  "email",
  async (job: Job) => {
    const { to, template, data } = job.data;
    await job.updateProgress(10);
    const html = await renderTemplate(template, data);
    await job.updateProgress(50);
    await sendEmail(to, html);
    await job.updateProgress(100);
    return { sentAt: new Date().toISOString() };
  },
  {
    connection: { host: "localhost", port: 6379 },
    concurrency: 10,        // Process 10 emails in parallel
    limiter: { max: 100, duration: 60000 }, // Rate limit: 100/minute
  }
);

emailWorker.on("completed", (job) => {
  console.log(`Email sent: job ${job.id} → ${job.data.to}`);
});

emailWorker.on("failed", (job, err) => {
  console.error(`Email failed: job ${job?.id} — ${err.message}`);
});

3. Add job scheduling and priorities

// Delayed job — send welcome email 30 minutes after signup
await emailQueue.add("email:send", {
  to: "newuser@example.com",
  template: "welcome",
  data: { name: "Alex" },
}, { delay: 30 * 60 * 1000 });

// Priority jobs — password resets jump the queue
await emailQueue.add("email:send", {
  to: "user@example.com",
  template: "password-reset",
  data: { resetLink: "https://app.example.com/reset/abc123" },
}, { priority: 1 }); // Lower number = higher priority

// Recurring job — daily digest at 8:00 AM UTC
await emailQueue.add("email:send", {
  to: "digest",
  template: "daily-digest",
  data: {},
}, {
  repeat: { pattern: "0 8 * * *" },
  jobId: "daily-digest", // Prevent duplicates
});

4. Implement graceful shutdown

// src/workers/shutdown.ts
const workers = [emailWorker, pdfWorker, exportWorker, imageWorker];

async function gracefulShutdown(signal: string): Promise<void> {
  console.log(`Received ${signal}. Closing workers gracefully...`);
  await Promise.all(workers.map((w) => w.close()));
  console.log("All workers closed. Exiting.");
  process.exit(0);
}

process.on("SIGTERM", () => gracefulShutdown("SIGTERM"));
process.on("SIGINT", () => gracefulShutdown("SIGINT"));

Examples

Example 1: PDF report generation queue

Prompt: "Build a background job system for generating PDF reports. Users request a report, get a job ID back immediately, and can poll for progress. Reports take 10-30 seconds to generate."

Agent output:

  • Creates src/jobs/pdf-queue.ts with typed job definitions
  • Creates src/workers/pdf-worker.ts with progress updates at each stage (query data → format → render → upload)
  • Creates src/routes/reports.ts with POST /reports (enqueue, return job ID) and GET /reports/:jobId/status (return progress percentage and download URL when complete)
  • Adds retry logic: 3 attempts with 10-second backoff

Example 2: Image processing pipeline

Prompt: "I need to process uploaded images: resize to 3 widths (200, 800, 1600px), convert to WebP, and upload to cloud storage. Handle up to 500 images per hour."

Agent output:

  • Creates src/workers/image-worker.ts with sharp-based resize and conversion pipeline
  • Sets concurrency to 4 (CPU-bound work, matches core count)
  • Adds per-image progress tracking (useful for batch uploads)
  • Creates src/jobs/image-pipeline.ts with a flow: resize → convert → upload as chained jobs

Guidelines

  • Keep jobs serializable — job data must survive JSON round-trips. Pass IDs and URLs, not buffers or streams.
  • Set appropriate concurrency — CPU-bound work (image processing): match core count. I/O-bound (email, API calls): 10-50 concurrent.
  • Always implement graceful shutdownSIGTERM should let running jobs finish before the process exits.
  • Use job IDs for idempotency — set a deterministic jobId to prevent the same job from being enqueued twice.
  • Monitor queue depth — a growing queue means workers can't keep up. Alert when backlog exceeds 5 minutes of processing time.
  • Separate queues by workload type — don't let a slow PDF generation block fast email sends.
  • Store results externally — BullMQ job results are cleaned up by default. Persist important results in your database.