mirror of
https://github.com/n8n-io/n8n.git
synced 2025-03-05 20:50:17 -08:00
154 lines
4.7 KiB
TypeScript
154 lines
4.7 KiB
TypeScript
import type { PushPayload, PushType } from '@n8n/api-types';
|
|
import type { Application } from 'express';
|
|
import { ServerResponse } from 'http';
|
|
import type { Server } from 'http';
|
|
import type { Socket } from 'net';
|
|
import { Container, Service } from 'typedi';
|
|
import { parse as parseUrl } from 'url';
|
|
import { Server as WSServer } from 'ws';
|
|
|
|
import { AuthService } from '@/auth/auth.service';
|
|
import config from '@/config';
|
|
import type { User } from '@/databases/entities/user';
|
|
import { OnShutdown } from '@/decorators/on-shutdown';
|
|
import { BadRequestError } from '@/errors/response-errors/bad-request.error';
|
|
import { Publisher } from '@/scaling/pubsub/publisher.service';
|
|
import { OrchestrationService } from '@/services/orchestration.service';
|
|
import { TypedEmitter } from '@/typed-emitter';
|
|
|
|
import { SSEPush } from './sse.push';
|
|
import type { OnPushMessage, PushResponse, SSEPushRequest, WebSocketPushRequest } from './types';
|
|
import { WebSocketPush } from './websocket.push';
|
|
|
|
type PushEvents = {
|
|
editorUiConnected: string;
|
|
message: OnPushMessage;
|
|
};
|
|
|
|
const useWebSockets = config.getEnv('push.backend') === 'websocket';
|
|
|
|
/**
|
|
* Push service for uni- or bi-directional communication with frontend clients.
|
|
* Uses either server-sent events (SSE, unidirectional from backend --> frontend)
|
|
* or WebSocket (bidirectional backend <--> frontend) depending on the configuration.
|
|
*
|
|
* @emits message when a message is received from a client
|
|
*/
|
|
@Service()
|
|
export class Push extends TypedEmitter<PushEvents> {
|
|
public isBidirectional = useWebSockets;
|
|
|
|
private backend = useWebSockets ? Container.get(WebSocketPush) : Container.get(SSEPush);
|
|
|
|
constructor(
|
|
private readonly orchestrationService: OrchestrationService,
|
|
private readonly publisher: Publisher,
|
|
) {
|
|
super();
|
|
|
|
if (useWebSockets) this.backend.on('message', (msg) => this.emit('message', msg));
|
|
}
|
|
|
|
getBackend() {
|
|
return this.backend;
|
|
}
|
|
|
|
handleRequest(req: SSEPushRequest | WebSocketPushRequest, res: PushResponse) {
|
|
const {
|
|
ws,
|
|
query: { pushRef },
|
|
user,
|
|
} = req;
|
|
|
|
if (!pushRef) {
|
|
if (ws) {
|
|
ws.send('The query parameter "pushRef" is missing!');
|
|
ws.close(1008);
|
|
return;
|
|
}
|
|
throw new BadRequestError('The query parameter "pushRef" is missing!');
|
|
}
|
|
|
|
if (req.ws) {
|
|
(this.backend as WebSocketPush).add(pushRef, user.id, req.ws);
|
|
} else if (!useWebSockets) {
|
|
(this.backend as SSEPush).add(pushRef, user.id, { req, res });
|
|
} else {
|
|
res.status(401).send('Unauthorized');
|
|
return;
|
|
}
|
|
|
|
this.emit('editorUiConnected', pushRef);
|
|
}
|
|
|
|
broadcast<Type extends PushType>(type: Type, data: PushPayload<Type>) {
|
|
this.backend.sendToAll(type, data);
|
|
}
|
|
|
|
send<Type extends PushType>(type: Type, data: PushPayload<Type>, pushRef: string) {
|
|
/**
|
|
* Multi-main setup: In a manual webhook execution, the main process that
|
|
* handles a webhook might not be the same as the main process that created
|
|
* the webhook. If so, the handler process commands the creator process to
|
|
* relay the former's execution lifecycle events to the creator's frontend.
|
|
*/
|
|
if (this.orchestrationService.isMultiMainSetupEnabled && !this.backend.hasPushRef(pushRef)) {
|
|
void this.publisher.publishCommand({
|
|
command: 'relay-execution-lifecycle-event',
|
|
payload: { type, args: data, pushRef },
|
|
});
|
|
return;
|
|
}
|
|
|
|
this.backend.sendToOne(type, data, pushRef);
|
|
}
|
|
|
|
sendToUsers<Type extends PushType>(
|
|
type: Type,
|
|
data: PushPayload<Type>,
|
|
userIds: Array<User['id']>,
|
|
) {
|
|
this.backend.sendToUsers(type, data, userIds);
|
|
}
|
|
|
|
@OnShutdown()
|
|
onShutdown() {
|
|
this.backend.closeAllConnections();
|
|
}
|
|
}
|
|
|
|
export const setupPushServer = (restEndpoint: string, server: Server, app: Application) => {
|
|
if (useWebSockets) {
|
|
const wsServer = new WSServer({ noServer: true });
|
|
server.on('upgrade', (request: WebSocketPushRequest, socket: Socket, head) => {
|
|
if (parseUrl(request.url).pathname === `/${restEndpoint}/push`) {
|
|
wsServer.handleUpgrade(request, socket, head, (ws) => {
|
|
request.ws = ws;
|
|
|
|
const response = new ServerResponse(request);
|
|
response.writeHead = (statusCode) => {
|
|
if (statusCode > 200) ws.close();
|
|
return response;
|
|
};
|
|
|
|
// @ts-ignore
|
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-call
|
|
app.handle(request, response);
|
|
});
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
export const setupPushHandler = (restEndpoint: string, app: Application) => {
|
|
const endpoint = `/${restEndpoint}/push`;
|
|
const push = Container.get(Push);
|
|
const authService = Container.get(AuthService);
|
|
app.use(
|
|
endpoint,
|
|
// eslint-disable-next-line @typescript-eslint/unbound-method
|
|
authService.authMiddleware,
|
|
(req: SSEPushRequest | WebSocketPushRequest, res: PushResponse) => push.handleRequest(req, res),
|
|
);
|
|
};
|