Odoo GraphQL Subscription using Node, Express JS for Sample
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

310 lines
18 KiB

4 months ago
"use strict";
/**
*
* server
*
*/
var __asyncValues = (this && this.__asyncValues) || function (o) {
if (!Symbol.asyncIterator) throw new TypeError("Symbol.asyncIterator is not defined.");
var m = o[Symbol.asyncIterator], i;
return m ? m.call(o) : (o = typeof __values === "function" ? __values(o) : o[Symbol.iterator](), i = {}, verb("next"), verb("throw"), verb("return"), i[Symbol.asyncIterator] = function () { return this; }, i);
function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }
function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.handleProtocols = exports.makeServer = void 0;
const graphql_1 = require("graphql");
const common_1 = require("./common");
const utils_1 = require("./utils");
/**
* Makes a Protocol compliant WebSocket GraphQL server. The server
* is actually an API which is to be used with your favourite WebSocket
* server library!
*
* Read more about the [GraphQL over WebSocket Protocol](https://github.com/graphql/graphql-over-http/blob/main/rfcs/GraphQLOverWebSocket.md).
*
* @category Server
*/
function makeServer(options) {
const { schema, context, roots, validate, execute, subscribe, connectionInitWaitTimeout = 3000, // 3 seconds
onConnect, onDisconnect, onClose, onSubscribe, onOperation, onNext, onError, onComplete, jsonMessageReviver: reviver, jsonMessageReplacer: replacer, } = options;
return {
opened(socket, extra) {
const ctx = {
connectionInitReceived: false,
acknowledged: false,
subscriptions: {},
extra,
};
if (socket.protocol !== common_1.GRAPHQL_TRANSPORT_WS_PROTOCOL) {
socket.close(common_1.CloseCode.SubprotocolNotAcceptable, 'Subprotocol not acceptable');
return async (code, reason) => {
/* nothing was set up, just notify the closure */
await (onClose === null || onClose === void 0 ? void 0 : onClose(ctx, code, reason));
};
}
// kick the client off (close socket) if the connection has
// not been initialised after the specified wait timeout
const connectionInitWait = connectionInitWaitTimeout > 0 && isFinite(connectionInitWaitTimeout)
? setTimeout(() => {
if (!ctx.connectionInitReceived)
socket.close(common_1.CloseCode.ConnectionInitialisationTimeout, 'Connection initialisation timeout');
}, connectionInitWaitTimeout)
: null;
socket.onMessage(async function onMessage(data) {
var _a, e_1, _b, _c;
var _d;
let message;
try {
message = (0, common_1.parseMessage)(data, reviver);
}
catch (err) {
return socket.close(common_1.CloseCode.BadRequest, 'Invalid message received');
}
switch (message.type) {
case common_1.MessageType.ConnectionInit: {
if (ctx.connectionInitReceived)
return socket.close(common_1.CloseCode.TooManyInitialisationRequests, 'Too many initialisation requests');
// @ts-expect-error: I can write
ctx.connectionInitReceived = true;
if ((0, utils_1.isObject)(message.payload))
// @ts-expect-error: I can write
ctx.connectionParams = message.payload;
const permittedOrPayload = await (onConnect === null || onConnect === void 0 ? void 0 : onConnect(ctx));
if (permittedOrPayload === false)
return socket.close(common_1.CloseCode.Forbidden, 'Forbidden');
// we should acknowledge before send to avoid race conditions (like those exampled in https://github.com/enisdenjo/graphql-ws/issues/501)
// even if the send fails/throws, the connection should be closed because its malfunctioning
// @ts-expect-error: I can write
ctx.acknowledged = true;
await socket.send((0, common_1.stringifyMessage)((0, utils_1.isObject)(permittedOrPayload)
? {
type: common_1.MessageType.ConnectionAck,
payload: permittedOrPayload,
}
: {
type: common_1.MessageType.ConnectionAck,
// payload is completely absent if not provided
}, replacer));
return;
}
case common_1.MessageType.Ping: {
if (socket.onPing)
// if the onPing listener is registered, automatic pong is disabled
return await socket.onPing(message.payload);
await socket.send((0, common_1.stringifyMessage)(message.payload
? { type: common_1.MessageType.Pong, payload: message.payload }
: {
type: common_1.MessageType.Pong,
// payload is completely absent if not provided
}));
return;
}
case common_1.MessageType.Pong:
return await ((_d = socket.onPong) === null || _d === void 0 ? void 0 : _d.call(socket, message.payload));
case common_1.MessageType.Subscribe: {
if (!ctx.acknowledged)
return socket.close(common_1.CloseCode.Unauthorized, 'Unauthorized');
const { id, payload } = message;
if (id in ctx.subscriptions)
return socket.close(common_1.CloseCode.SubscriberAlreadyExists, `Subscriber for ${id} already exists`);
// if this turns out to be a streaming operation, the subscription value
// will change to an `AsyncIterable`, otherwise it will stay as is
ctx.subscriptions[id] = null;
const emit = {
next: async (result, args) => {
let nextMessage = {
id,
type: common_1.MessageType.Next,
payload: result,
};
const maybeResult = await (onNext === null || onNext === void 0 ? void 0 : onNext(ctx, nextMessage, args, result));
if (maybeResult)
nextMessage = Object.assign(Object.assign({}, nextMessage), { payload: maybeResult });
await socket.send((0, common_1.stringifyMessage)(nextMessage, replacer));
},
error: async (errors) => {
let errorMessage = {
id,
type: common_1.MessageType.Error,
payload: errors,
};
const maybeErrors = await (onError === null || onError === void 0 ? void 0 : onError(ctx, errorMessage, errors));
if (maybeErrors)
errorMessage = Object.assign(Object.assign({}, errorMessage), { payload: maybeErrors });
await socket.send((0, common_1.stringifyMessage)(errorMessage, replacer));
},
complete: async (notifyClient) => {
const completeMessage = {
id,
type: common_1.MessageType.Complete,
};
await (onComplete === null || onComplete === void 0 ? void 0 : onComplete(ctx, completeMessage));
if (notifyClient)
await socket.send((0, common_1.stringifyMessage)(completeMessage, replacer));
},
};
try {
let execArgs;
const maybeExecArgsOrErrors = await (onSubscribe === null || onSubscribe === void 0 ? void 0 : onSubscribe(ctx, message));
if (maybeExecArgsOrErrors) {
if ((0, utils_1.areGraphQLErrors)(maybeExecArgsOrErrors))
return id in ctx.subscriptions
? await emit.error(maybeExecArgsOrErrors)
: void 0;
else if (Array.isArray(maybeExecArgsOrErrors))
throw new Error('Invalid return value from onSubscribe hook, expected an array of GraphQLError objects');
// not errors, is exec args
execArgs = maybeExecArgsOrErrors;
}
else {
// you either provide a schema dynamically through
// `onSubscribe` or you set one up during the server setup
if (!schema)
throw new Error('The GraphQL schema is not provided');
const args = {
operationName: payload.operationName,
document: (0, graphql_1.parse)(payload.query),
variableValues: payload.variables,
};
execArgs = Object.assign(Object.assign({}, args), { schema: typeof schema === 'function'
? await schema(ctx, message, args)
: schema });
const validationErrors = (validate !== null && validate !== void 0 ? validate : graphql_1.validate)(execArgs.schema, execArgs.document);
if (validationErrors.length > 0)
return id in ctx.subscriptions
? await emit.error(validationErrors)
: void 0;
}
const operationAST = (0, graphql_1.getOperationAST)(execArgs.document, execArgs.operationName);
if (!operationAST)
return id in ctx.subscriptions
? await emit.error([
new graphql_1.GraphQLError('Unable to identify operation'),
])
: void 0;
// if `onSubscribe` didn't specify a rootValue, inject one
if (!('rootValue' in execArgs))
execArgs.rootValue = roots === null || roots === void 0 ? void 0 : roots[operationAST.operation];
// if `onSubscribe` didn't specify a context, inject one
if (!('contextValue' in execArgs))
execArgs.contextValue =
typeof context === 'function'
? await context(ctx, message, execArgs)
: context;
// the execution arguments have been prepared
// perform the operation and act accordingly
let operationResult;
if (operationAST.operation === 'subscription')
operationResult = await (subscribe !== null && subscribe !== void 0 ? subscribe : graphql_1.subscribe)(execArgs);
// operation === 'query' || 'mutation'
else
operationResult = await (execute !== null && execute !== void 0 ? execute : graphql_1.execute)(execArgs);
const maybeResult = await (onOperation === null || onOperation === void 0 ? void 0 : onOperation(ctx, message, execArgs, operationResult));
if (maybeResult)
operationResult = maybeResult;
if ((0, utils_1.isAsyncIterable)(operationResult)) {
/** multiple emitted results */
if (!(id in ctx.subscriptions)) {
// subscription was completed/canceled before the operation settled
if ((0, utils_1.isAsyncGenerator)(operationResult))
operationResult.return(undefined);
}
else {
ctx.subscriptions[id] = operationResult;
try {
for (var _e = true, operationResult_1 = __asyncValues(operationResult), operationResult_1_1; operationResult_1_1 = await operationResult_1.next(), _a = operationResult_1_1.done, !_a; _e = true) {
_c = operationResult_1_1.value;
_e = false;
const result = _c;
await emit.next(result, execArgs);
}
}
catch (e_1_1) { e_1 = { error: e_1_1 }; }
finally {
try {
if (!_e && !_a && (_b = operationResult_1.return)) await _b.call(operationResult_1);
}
finally { if (e_1) throw e_1.error; }
}
}
}
else {
/** single emitted result */
// if the client completed the subscription before the single result
// became available, he effectively canceled it and no data should be sent
if (id in ctx.subscriptions)
await emit.next(operationResult, execArgs);
}
// lack of subscription at this point indicates that the client
// completed the subscription, he doesn't need to be reminded
await emit.complete(id in ctx.subscriptions);
}
finally {
// whatever happens to the subscription, we finally want to get rid of the reservation
delete ctx.subscriptions[id];
}
return;
}
case common_1.MessageType.Complete: {
const subscription = ctx.subscriptions[message.id];
delete ctx.subscriptions[message.id]; // deleting the subscription means no further activity should take place
if ((0, utils_1.isAsyncGenerator)(subscription))
await subscription.return(undefined);
return;
}
default:
throw new Error(`Unexpected message of type ${message.type} received`);
}
});
// wait for close, cleanup and the disconnect callback
return async (code, reason) => {
if (connectionInitWait)
clearTimeout(connectionInitWait);
const subs = Object.assign({}, ctx.subscriptions);
// @ts-expect-error: I can write
ctx.subscriptions = {}; // deleting the subscription means no further activity should take place
// we return all iterable subscriptions immediatelly, independant of the order
await Promise.all(Object.values(subs)
.filter(utils_1.isAsyncGenerator)
.map((sub) => sub.return(undefined)));
if (ctx.acknowledged)
await (onDisconnect === null || onDisconnect === void 0 ? void 0 : onDisconnect(ctx, code, reason));
await (onClose === null || onClose === void 0 ? void 0 : onClose(ctx, code, reason));
};
},
};
}
exports.makeServer = makeServer;
/**
* Helper utility for choosing the "graphql-transport-ws" subprotocol from
* a set of WebSocket subprotocols.
*
* Accepts a set of already extracted WebSocket subprotocols or the raw
* Sec-WebSocket-Protocol header value. In either case, if the right
* protocol appears, it will be returned.
*
* By specification, the server should not provide a value with Sec-WebSocket-Protocol
* if it does not agree with client's subprotocols. The client has a responsibility
* to handle the connection afterwards.
*
* @category Server
*/
function handleProtocols(protocols) {
switch (true) {
case protocols instanceof Set &&
protocols.has(common_1.GRAPHQL_TRANSPORT_WS_PROTOCOL):
case Array.isArray(protocols) &&
protocols.includes(common_1.GRAPHQL_TRANSPORT_WS_PROTOCOL):
case typeof protocols === 'string' &&
protocols
.split(',')
.map((p) => p.trim())
.includes(common_1.GRAPHQL_TRANSPORT_WS_PROTOCOL):
return common_1.GRAPHQL_TRANSPORT_WS_PROTOCOL;
default:
return false;
}
}
exports.handleProtocols = handleProtocols;