init redis lazy

This commit is contained in:
Carl-Gerhard Lindesvärd
2024-07-20 20:17:32 +02:00
parent 492141547d
commit f2298a1b05
19 changed files with 134 additions and 86 deletions

View File

@@ -1,11 +1,10 @@
import { logger } from '@/utils/logger';
import { getClientIp, parseIp } from '@/utils/parseIp'; import { getClientIp, parseIp } from '@/utils/parseIp';
import type { FastifyReply, FastifyRequest } from 'fastify'; import type { FastifyReply, FastifyRequest } from 'fastify';
import { generateDeviceId } from '@openpanel/common'; import { generateDeviceId } from '@openpanel/common';
import { getSalts } from '@openpanel/db'; import { getSalts } from '@openpanel/db';
import { eventsQueue } from '@openpanel/queue'; import { eventsQueue } from '@openpanel/queue';
import { redis } from '@openpanel/redis'; import { getRedisCache } from '@openpanel/redis';
import type { PostEventPayload } from '@openpanel/sdk'; import type { PostEventPayload } from '@openpanel/sdk';
export async function postEvent( export async function postEvent(
@@ -38,7 +37,7 @@ export async function postEvent(
}); });
// this will ensure that we don't have multiple events creating sessions // this will ensure that we don't have multiple events creating sessions
const locked = await redis.set( const locked = await getRedisCache().set(
`request:priority:${currentDeviceId}-${previousDeviceId}`, `request:priority:${currentDeviceId}-${previousDeviceId}`,
'locked', 'locked',
'EX', 'EX',

View File

@@ -1,6 +1,5 @@
import { validateClerkJwt } from '@/utils/auth'; import { validateClerkJwt } from '@/utils/auth';
import type { FastifyReply, FastifyRequest } from 'fastify'; import type { FastifyReply, FastifyRequest } from 'fastify';
import { escape } from 'sqlstring';
import superjson from 'superjson'; import superjson from 'superjson';
import type * as WebSocket from 'ws'; import type * as WebSocket from 'ws';
@@ -13,7 +12,7 @@ import {
TABLE_NAMES, TABLE_NAMES,
transformMinimalEvent, transformMinimalEvent,
} from '@openpanel/db'; } from '@openpanel/db';
import { redis, redisPub, redisSub } from '@openpanel/redis'; import { getRedisCache, getRedisPub, getRedisSub } from '@openpanel/redis';
import { getProjectAccess } from '@openpanel/trpc'; import { getProjectAccess } from '@openpanel/trpc';
export function getLiveEventInfo(key: string) { export function getLiveEventInfo(key: string) {
@@ -36,8 +35,8 @@ export async function testVisitors(
return reply.status(404).send('No event found'); return reply.status(404).send('No event found');
} }
event.projectId = req.params.projectId; event.projectId = req.params.projectId;
redisPub.publish('event:received', superjson.stringify(event)); getRedisPub().publish('event:received', superjson.stringify(event));
redis.set( getRedisCache().set(
`live:event:${event.projectId}:${Math.random() * 1000}`, `live:event:${event.projectId}:${Math.random() * 1000}`,
'', '',
'EX', 'EX',
@@ -61,7 +60,7 @@ export async function testEvents(
if (!event) { if (!event) {
return reply.status(404).send('No event found'); return reply.status(404).send('No event found');
} }
redisPub.publish('event:saved', superjson.stringify(event)); getRedisPub().publish('event:saved', superjson.stringify(event));
reply.status(202).send(event); reply.status(202).send(event);
} }
@@ -77,8 +76,8 @@ export function wsVisitors(
) { ) {
const { params } = req; const { params } = req;
redisSub.subscribe('event:received'); getRedisSub().subscribe('event:received');
redisSub.psubscribe('__key*:expired'); getRedisSub().psubscribe('__key*:expired');
const message = (channel: string, message: string) => { const message = (channel: string, message: string) => {
if (channel === 'event:received') { if (channel === 'event:received') {
@@ -99,14 +98,14 @@ export function wsVisitors(
} }
}; };
redisSub.on('message', message); getRedisSub().on('message', message);
redisSub.on('pmessage', pmessage); getRedisSub().on('pmessage', pmessage);
connection.socket.on('close', () => { connection.socket.on('close', () => {
redisSub.unsubscribe('event:saved'); getRedisSub().unsubscribe('event:saved');
redisSub.punsubscribe('__key*:expired'); getRedisSub().punsubscribe('__key*:expired');
redisSub.off('message', message); getRedisSub().off('message', message);
redisSub.off('pmessage', pmessage); getRedisSub().off('pmessage', pmessage);
}); });
} }
@@ -140,7 +139,7 @@ export async function wsProjectEvents(
projectId: params.projectId, projectId: params.projectId,
}); });
redisSub.subscribe(subscribeToEvent); getRedisSub().subscribe(subscribeToEvent);
const message = async (channel: string, message: string) => { const message = async (channel: string, message: string) => {
const event = getSuperJson<IServiceCreateEventPayload>(message); const event = getSuperJson<IServiceCreateEventPayload>(message);
@@ -159,10 +158,10 @@ export async function wsProjectEvents(
} }
}; };
redisSub.on('message', message as any); getRedisSub().on('message', message as any);
connection.socket.on('close', () => { connection.socket.on('close', () => {
redisSub.unsubscribe(subscribeToEvent); getRedisSub().unsubscribe(subscribeToEvent);
redisSub.off('message', message as any); getRedisSub().off('message', message as any);
}); });
} }

View File

@@ -5,7 +5,7 @@ import icoToPng from 'ico-to-png';
import sharp from 'sharp'; import sharp from 'sharp';
import { createHash } from '@openpanel/common'; import { createHash } from '@openpanel/common';
import { redis } from '@openpanel/redis'; import { getRedisCache } from '@openpanel/redis';
interface GetFaviconParams { interface GetFaviconParams {
url: string; url: string;
@@ -51,7 +51,7 @@ export async function getFavicon(
) { ) {
function sendBuffer(buffer: Buffer, cacheKey?: string) { function sendBuffer(buffer: Buffer, cacheKey?: string) {
if (cacheKey) { if (cacheKey) {
redis.set(`favicon:${cacheKey}`, buffer.toString('base64')); getRedisCache().set(`favicon:${cacheKey}`, buffer.toString('base64'));
} }
reply.type('image/png'); reply.type('image/png');
return reply.send(buffer); return reply.send(buffer);
@@ -65,7 +65,7 @@ export async function getFavicon(
if (imageExtensions.find((ext) => url.endsWith(ext))) { if (imageExtensions.find((ext) => url.endsWith(ext))) {
const cacheKey = createHash(url, 32); const cacheKey = createHash(url, 32);
const cache = await redis.get(`favicon:${cacheKey}`); const cache = await getRedisCache().get(`favicon:${cacheKey}`);
if (cache) { if (cache) {
return sendBuffer(Buffer.from(cache, 'base64')); return sendBuffer(Buffer.from(cache, 'base64'));
} }
@@ -76,7 +76,7 @@ export async function getFavicon(
} }
const { hostname } = new URL(url); const { hostname } = new URL(url);
const cache = await redis.get(`favicon:${hostname}`); const cache = await getRedisCache().get(`favicon:${hostname}`);
if (cache) { if (cache) {
return sendBuffer(Buffer.from(cache, 'base64')); return sendBuffer(Buffer.from(cache, 'base64'));
@@ -104,9 +104,9 @@ export async function clearFavicons(
request: FastifyRequest, request: FastifyRequest,
reply: FastifyReply reply: FastifyReply
) { ) {
const keys = await redis.keys('favicon:*'); const keys = await getRedisCache().keys('favicon:*');
for (const key of keys) { for (const key of keys) {
await redis.del(key); await getRedisCache().del(key);
} }
return reply.status(404).send('OK'); return reply.status(404).send('OK');
} }

View File

@@ -10,7 +10,7 @@ import { round } from '@openpanel/common';
import { chQuery, db, TABLE_NAMES } from '@openpanel/db'; import { chQuery, db, TABLE_NAMES } from '@openpanel/db';
import type { IServiceClient } from '@openpanel/db'; import type { IServiceClient } from '@openpanel/db';
import { eventsQueue } from '@openpanel/queue'; import { eventsQueue } from '@openpanel/queue';
import { redis, redisPub } from '@openpanel/redis'; import { getRedisCache, getRedisPub } from '@openpanel/redis';
import type { AppRouter } from '@openpanel/trpc'; import type { AppRouter } from '@openpanel/trpc';
import { appRouter, createContext } from '@openpanel/trpc'; import { appRouter, createContext } from '@openpanel/trpc';
@@ -98,7 +98,7 @@ const startServer = async () => {
reply.send({ name: 'openpanel sdk api' }); reply.send({ name: 'openpanel sdk api' });
}); });
fastify.get('/healthcheck', async (request, reply) => { fastify.get('/healthcheck', async (request, reply) => {
const redisRes = await withTimings(redis.keys('*')); const redisRes = await withTimings(getRedisCache().keys('*'));
const dbRes = await withTimings(db.project.findFirst()); const dbRes = await withTimings(db.project.findFirst());
const queueRes = await withTimings(eventsQueue.getCompleted()); const queueRes = await withTimings(eventsQueue.getCompleted());
const chRes = await withTimings( const chRes = await withTimings(
@@ -150,7 +150,7 @@ const startServer = async () => {
}); });
// Notify when keys expires // Notify when keys expires
redisPub.config('SET', 'notify-keyspace-events', 'Ex'); getRedisPub().config('SET', 'notify-keyspace-events', 'Ex');
} catch (e) { } catch (e) {
logger.error(e, 'Failed to start server'); logger.error(e, 'Failed to start server');
} }

View File

@@ -3,11 +3,11 @@ import { escape } from 'sqlstring';
import type { IClickhouseEvent } from '@openpanel/db'; import type { IClickhouseEvent } from '@openpanel/db';
import { chQuery, eventBuffer, TABLE_NAMES } from '@openpanel/db'; import { chQuery, eventBuffer, TABLE_NAMES } from '@openpanel/db';
import { sessionsQueue } from '@openpanel/queue/src/queues'; import { sessionsQueue } from '@openpanel/queue/src/queues';
import { redis } from '@openpanel/redis'; import { getRedisCache, getRedisQueue } from '@openpanel/redis';
async function debugStalledEvents() { async function debugStalledEvents() {
const keys = await redis.keys('bull:sessions:sessionEnd*'); const keys = await getRedisQueue().keys('bull:sessions:sessionEnd*');
const delayedZRange = await redis.zrange( const delayedZRange = await getRedisQueue().zrange(
'bull:sessions:delayed', 'bull:sessions:delayed',
0, 0,
-1, -1,
@@ -20,10 +20,14 @@ async function debugStalledEvents() {
} }
return acc; return acc;
}, },
[] as Record<string, number> {} as Record<string, number>
);
const opKeys = await getRedisCache().keys('op:*');
const stalledEvents = await getRedisCache().lrange(
'op:buffer:events:stalled',
0,
-1
); );
const opKeys = await redis.keys('op:*');
const stalledEvents = await redis.lrange('op:buffer:events:stalled', 0, -1);
// keys.forEach((key) => { // keys.forEach((key) => {
// console.log(key); // console.log(key);
// }); // });
@@ -112,7 +116,14 @@ async function debugStalledEvents() {
delayedJobs.sort((a, b) => a.timestamp + a.delay - (b.timestamp + b.delay)); delayedJobs.sort((a, b) => a.timestamp + a.delay - (b.timestamp + b.delay));
let delayedJobsCount = 0; let delayedJobsCount = 0;
delayedJobs.forEach((job) => { delayedJobs.forEach((job) => {
const date = new Date(delayedValues[job.id]); if (!job.id) {
return;
}
const timestamp = delayedValues[job.id];
if (!timestamp) {
return;
}
const date = new Date(timestamp);
// if date is in the past // if date is in the past
// if (date.getTime() - 1000 * 60 * 5 < Date.now()) { // if (date.getTime() - 1000 * 60 * 5 < Date.now()) {
if (date.getTime() < Date.now()) { if (date.getTime() < Date.now()) {

View File

@@ -5,12 +5,8 @@ import type { WorkerOptions } from 'bullmq';
import { Worker } from 'bullmq'; import { Worker } from 'bullmq';
import express from 'express'; import express from 'express';
import { import { cronQueue, eventsQueue, sessionsQueue } from '@openpanel/queue';
connection, import { getRedisQueue } from '@openpanel/redis';
cronQueue,
eventsQueue,
sessionsQueue,
} from '@openpanel/queue';
import { cronJob } from './jobs/cron'; import { cronJob } from './jobs/cron';
import { eventsJob } from './jobs/events'; import { eventsJob } from './jobs/events';
@@ -23,12 +19,7 @@ serverAdapter.setBasePath(process.env.SELF_HOSTED ? '/worker' : '/');
const app = express(); const app = express();
const workerOptions: WorkerOptions = { const workerOptions: WorkerOptions = {
connection: { connection: getRedisQueue(),
...connection,
enableReadyCheck: false,
maxRetriesPerRequest: null,
enableOfflineQueue: true,
},
concurrency: parseInt(process.env.CONCURRENCY || '1', 10), concurrency: parseInt(process.env.CONCURRENCY || '1', 10),
}; };

View File

@@ -14,7 +14,7 @@ import type {
EventsQueuePayloadCreateSessionEnd, EventsQueuePayloadCreateSessionEnd,
EventsQueuePayloadIncomingEvent, EventsQueuePayloadIncomingEvent,
} from '@openpanel/queue'; } from '@openpanel/queue';
import { redis } from '@openpanel/redis'; import { getRedisQueue } from '@openpanel/redis';
function noDateInFuture(eventDate: Date): Date { function noDateInFuture(eventDate: Date): Date {
if (eventDate > new Date()) { if (eventDate > new Date()) {
@@ -217,7 +217,9 @@ async function getSessionEnd({
currentDeviceId: string; currentDeviceId: string;
previousDeviceId: string; previousDeviceId: string;
}) { }) {
const sessionEndKeys = await redis.keys(`*:sessionEnd:${projectId}:*`); const sessionEndKeys = await getRedisQueue().keys(
`*:sessionEnd:${projectId}:*`
);
const sessionEndJobCurrentDeviceId = await findJobByPrefix( const sessionEndJobCurrentDeviceId = await findJobByPrefix(
sessionsQueue, sessionsQueue,

View File

@@ -7,7 +7,6 @@ import type {
EventsQueuePayloadCreateSessionEnd, EventsQueuePayloadCreateSessionEnd,
EventsQueuePayloadIncomingEvent, EventsQueuePayloadIncomingEvent,
} from '@openpanel/queue'; } from '@openpanel/queue';
import { redis } from '@openpanel/redis';
import { createSessionEnd } from './events.create-session-end'; import { createSessionEnd } from './events.create-session-end';
import { incomingEvent } from './events.incoming-event'; import { incomingEvent } from './events.incoming-event';

View File

@@ -1,7 +1,7 @@
import client from 'prom-client'; import client from 'prom-client';
import { cronQueue, eventsQueue, sessionsQueue } from '@openpanel/queue'; import { cronQueue, eventsQueue, sessionsQueue } from '@openpanel/queue';
import { redis } from '@openpanel/redis'; import { getRedisCache } from '@openpanel/redis';
const Registry = client.Registry; const Registry = client.Registry;
@@ -75,7 +75,7 @@ buffers.forEach((buffer) => {
name: `buffer_${buffer}_count`, name: `buffer_${buffer}_count`,
help: 'Number of users in the users array', help: 'Number of users in the users array',
async collect() { async collect() {
const metric = await redis.llen(`op:buffer:${buffer}`); const metric = await getRedisCache().llen(`op:buffer:${buffer}`);
this.set(metric); this.set(metric);
}, },
}) })
@@ -86,7 +86,9 @@ buffers.forEach((buffer) => {
name: `buffer_${buffer}_stalled_count`, name: `buffer_${buffer}_stalled_count`,
help: 'Number of users in the users array', help: 'Number of users in the users array',
async collect() { async collect() {
const metric = await redis.llen(`op:buffer:${buffer}:stalled`); const metric = await getRedisCache().llen(
`op:buffer:${buffer}:stalled`
);
this.set(metric); this.set(metric);
}, },
}) })

View File

@@ -2,7 +2,7 @@ import { groupBy } from 'ramda';
import SuperJSON from 'superjson'; import SuperJSON from 'superjson';
import { deepMergeObjects } from '@openpanel/common'; import { deepMergeObjects } from '@openpanel/common';
import { redis, redisPub } from '@openpanel/redis'; import { getRedisCache, getRedisPub } from '@openpanel/redis';
import { ch, TABLE_NAMES } from '../clickhouse-client'; import { ch, TABLE_NAMES } from '../clickhouse-client';
import { transformEvent } from '../services/event.service'; import { transformEvent } from '../services/event.service';
@@ -31,12 +31,12 @@ export class EventBuffer extends RedisBuffer<IClickhouseEvent> {
constructor() { constructor() {
super({ super({
table: TABLE_NAMES.events, table: TABLE_NAMES.events,
redis, redis: getRedisCache(),
}); });
} }
public onInsert?: OnInsert<IClickhouseEvent> | undefined = (event) => { public onInsert?: OnInsert<IClickhouseEvent> | undefined = (event) => {
redisPub.publish( getRedisPub().publish(
'event:received', 'event:received',
SuperJSON.stringify(transformEvent(event)) SuperJSON.stringify(transformEvent(event))
); );
@@ -54,7 +54,7 @@ export class EventBuffer extends RedisBuffer<IClickhouseEvent> {
savedEvents savedEvents
) => { ) => {
for (const event of savedEvents) { for (const event of savedEvents) {
redisPub.publish( getRedisPub().publish(
'event:saved', 'event:saved',
SuperJSON.stringify(transformEvent(event)) SuperJSON.stringify(transformEvent(event))
); );

View File

@@ -1,7 +1,7 @@
import { mergeDeepRight } from 'ramda'; import { mergeDeepRight } from 'ramda';
import { toDots } from '@openpanel/common'; import { toDots } from '@openpanel/common';
import { redis } from '@openpanel/redis'; import { getRedisCache } from '@openpanel/redis';
import { ch, chQuery } from '../clickhouse-client'; import { ch, chQuery } from '../clickhouse-client';
import type { import type {
@@ -22,7 +22,7 @@ import { RedisBuffer } from './buffer';
export class ProfileBuffer extends RedisBuffer<IClickhouseProfile> { export class ProfileBuffer extends RedisBuffer<IClickhouseProfile> {
constructor() { constructor() {
super({ super({
redis, redis: getRedisCache(),
table: 'profiles', table: 'profiles',
batchSize: 100, batchSize: 100,
}); });

View File

@@ -3,7 +3,7 @@ import { escape } from 'sqlstring';
import { v4 as uuid } from 'uuid'; import { v4 as uuid } from 'uuid';
import { toDots } from '@openpanel/common'; import { toDots } from '@openpanel/common';
import { redis } from '@openpanel/redis'; import { getRedisCache } from '@openpanel/redis';
import type { IChartEventFilter } from '@openpanel/validation'; import type { IChartEventFilter } from '@openpanel/validation';
import { eventBuffer } from '../buffers'; import { eventBuffer } from '../buffers';
@@ -184,7 +184,7 @@ export function transformMinimalEvent(
} }
export async function getLiveVisitors(projectId: string) { export async function getLiveVisitors(projectId: string) {
const keys = await redis.keys(`live:event:${projectId}:*`); const keys = await getRedisCache().keys(`live:event:${projectId}:*`);
return keys.length; return keys.length;
} }

View File

@@ -1,5 +1,4 @@
export { eventsQueue, cronQueue, sessionsQueue } from './src/queues'; export { eventsQueue, cronQueue, sessionsQueue } from './src/queues';
export type * from './src/queues'; export type * from './src/queues';
export { connection } from './src/connection';
export { findJobByPrefix } from './src/utils'; export { findJobByPrefix } from './src/utils';
export type { JobsOptions } from 'bullmq'; export type { JobsOptions } from 'bullmq';

View File

@@ -9,6 +9,7 @@
}, },
"dependencies": { "dependencies": {
"@openpanel/db": "workspace:*", "@openpanel/db": "workspace:*",
"@openpanel/redis": "workspace:*",
"bullmq": "^5.8.7" "bullmq": "^5.8.7"
}, },
"devDependencies": { "devDependencies": {

View File

@@ -1,10 +0,0 @@
const parse = (connectionString: string) => {
const url = new URL(connectionString);
return {
host: url.hostname,
port: Number(url.port),
password: url.password,
} as const;
};
export const connection = parse(String(process.env.REDIS_URL));

View File

@@ -1,10 +1,9 @@
import { Queue } from 'bullmq'; import { Queue } from 'bullmq';
import type { IServiceCreateEventPayload } from '@openpanel/db'; import type { IServiceCreateEventPayload } from '@openpanel/db';
import { getRedisQueue } from '@openpanel/redis';
import type { PostEventPayload } from '@openpanel/sdk'; import type { PostEventPayload } from '@openpanel/sdk';
import { connection } from './connection';
export interface EventsQueuePayloadIncomingEvent { export interface EventsQueuePayloadIncomingEvent {
type: 'incomingEvent'; type: 'incomingEvent';
payload: { payload: {
@@ -63,21 +62,21 @@ export type CronQueuePayload =
| CronQueuePayloadFlushProfiles; | CronQueuePayloadFlushProfiles;
export const eventsQueue = new Queue<EventsQueuePayload>('events', { export const eventsQueue = new Queue<EventsQueuePayload>('events', {
connection, connection: getRedisQueue(),
defaultJobOptions: { defaultJobOptions: {
removeOnComplete: 10, removeOnComplete: 10,
}, },
}); });
export const sessionsQueue = new Queue<SessionsQueuePayload>('sessions', { export const sessionsQueue = new Queue<SessionsQueuePayload>('sessions', {
connection, connection: getRedisQueue(),
defaultJobOptions: { defaultJobOptions: {
removeOnComplete: 10, removeOnComplete: 10,
}, },
}); });
export const cronQueue = new Queue<CronQueuePayload>('cron', { export const cronQueue = new Queue<CronQueuePayload>('cron', {
connection, connection: getRedisQueue(),
defaultJobOptions: { defaultJobOptions: {
removeOnComplete: 10, removeOnComplete: 10,
}, },

View File

@@ -1,4 +1,4 @@
import { redis } from './redis'; import { getRedisCache } from './redis';
export function cacheable<T extends (...args: any) => any>( export function cacheable<T extends (...args: any) => any>(
fn: T, fn: T,
@@ -9,7 +9,7 @@ export function cacheable<T extends (...args: any) => any>(
): Promise<Awaited<ReturnType<T>>> { ): Promise<Awaited<ReturnType<T>>> {
// JSON.stringify here is not bullet proof since ordering of object keys matters etc // JSON.stringify here is not bullet proof since ordering of object keys matters etc
const key = `cachable:${fn.name}:${JSON.stringify(args)}`; const key = `cachable:${fn.name}:${JSON.stringify(args)}`;
const cached = await redis.get(key); const cached = await getRedisCache().get(key);
if (cached) { if (cached) {
try { try {
return JSON.parse(cached); return JSON.parse(cached);
@@ -20,7 +20,7 @@ export function cacheable<T extends (...args: any) => any>(
const result = await fn(...(args as any)); const result = await fn(...(args as any));
if (result !== undefined || result !== null) { if (result !== undefined || result !== null) {
redis.setex(key, expire, JSON.stringify(result)); getRedisCache().setex(key, expire, JSON.stringify(result));
} }
return result; return result;

View File

@@ -2,12 +2,65 @@ import type { RedisOptions } from 'ioredis';
import Redis from 'ioredis'; import Redis from 'ioredis';
const options: RedisOptions = { const options: RedisOptions = {
connectTimeout: 30000, connectTimeout: 10000,
maxRetriesPerRequest: null,
}; };
export { Redis }; export { Redis };
export const redis = new Redis(process.env.REDIS_URL!, options); const createRedisClient = (
export const redisSub = new Redis(process.env.REDIS_URL!, options); url: string,
export const redisPub = new Redis(process.env.REDIS_URL!, options); overrides: RedisOptions = {}
): Redis => {
const client = new Redis(url, { ...options, ...overrides });
client.on('error', (error) => {
console.error('Redis Client Error:', error);
});
return client;
};
let redisCache: Redis;
export function getRedisCache() {
if (!redisCache) {
redisCache = createRedisClient(process.env.REDIS_URL!, options);
}
return redisCache;
}
let redisSub: Redis;
export function getRedisSub() {
if (!redisSub) {
redisSub = createRedisClient(process.env.REDIS_URL!, options);
}
return redisSub;
}
let redisPub: Redis;
export function getRedisPub() {
if (!redisPub) {
redisPub = createRedisClient(process.env.REDIS_URL!, options);
}
return redisPub;
}
let redisQueue: Redis;
export function getRedisQueue() {
if (!redisQueue) {
// Use different redis for queues (self-hosting will re-use the same redis instance)
redisQueue = createRedisClient(
(process.env.QUEUE_REDIS_URL || process.env.REDIS_URL)!,
{
...options,
enableReadyCheck: false,
maxRetriesPerRequest: null,
enableOfflineQueue: true,
}
);
}
return redisQueue;
}

3
pnpm-lock.yaml generated
View File

@@ -1008,6 +1008,9 @@ importers:
'@openpanel/db': '@openpanel/db':
specifier: workspace:* specifier: workspace:*
version: link:../db version: link:../db
'@openpanel/redis':
specifier: workspace:*
version: link:../redis
bullmq: bullmq:
specifier: ^5.8.7 specifier: ^5.8.7
version: 5.8.7 version: 5.8.7