123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338 |
- "use strict";
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- exports.graphqlHTTP = graphqlHTTP;
- exports.getGraphQLParams = getGraphQLParams;
- var _accepts = _interopRequireDefault(require("accepts"));
- var _httpErrors = _interopRequireDefault(require("http-errors"));
- var _graphql = require("graphql");
- var _parseBody = require("./parseBody");
- var _renderGraphiQL = require("./renderGraphiQL");
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- function graphqlHTTP(options) {
- if (!options) {
- throw new Error('GraphQL middleware requires options.');
- }
- return async function graphqlMiddleware(request, response) {
- var _result$errors;
-
- let params;
- let showGraphiQL = false;
- let graphiqlOptions;
- let formatErrorFn = _graphql.formatError;
- let pretty = false;
- let result;
- try {
- var _optionsData$validati, _optionsData$graphiql, _optionsData$context, _optionsData$customPa, _optionsData$customEx, _optionsData$customVa, _optionsData$pretty2, _ref2, _optionsData$customFo2;
-
- try {
- params = await getGraphQLParams(request);
- } catch (error) {
- var _optionsData$pretty, _ref, _optionsData$customFo;
-
-
- const optionsData = await resolveOptions();
- pretty = (_optionsData$pretty = optionsData.pretty) !== null && _optionsData$pretty !== void 0 ? _optionsData$pretty : false;
- formatErrorFn = (_ref = (_optionsData$customFo = optionsData.customFormatErrorFn) !== null && _optionsData$customFo !== void 0 ? _optionsData$customFo : optionsData.formatError) !== null && _ref !== void 0 ? _ref : formatErrorFn;
- throw error;
- }
- const optionsData = await resolveOptions(params);
- const schema = optionsData.schema;
- const rootValue = optionsData.rootValue;
- const validationRules = (_optionsData$validati = optionsData.validationRules) !== null && _optionsData$validati !== void 0 ? _optionsData$validati : [];
- const fieldResolver = optionsData.fieldResolver;
- const typeResolver = optionsData.typeResolver;
- const graphiql = (_optionsData$graphiql = optionsData.graphiql) !== null && _optionsData$graphiql !== void 0 ? _optionsData$graphiql : false;
- const extensionsFn = optionsData.extensions;
- const context = (_optionsData$context = optionsData.context) !== null && _optionsData$context !== void 0 ? _optionsData$context : request;
- const parseFn = (_optionsData$customPa = optionsData.customParseFn) !== null && _optionsData$customPa !== void 0 ? _optionsData$customPa : _graphql.parse;
- const executeFn = (_optionsData$customEx = optionsData.customExecuteFn) !== null && _optionsData$customEx !== void 0 ? _optionsData$customEx : _graphql.execute;
- const validateFn = (_optionsData$customVa = optionsData.customValidateFn) !== null && _optionsData$customVa !== void 0 ? _optionsData$customVa : _graphql.validate;
- pretty = (_optionsData$pretty2 = optionsData.pretty) !== null && _optionsData$pretty2 !== void 0 ? _optionsData$pretty2 : false;
- formatErrorFn = (_ref2 = (_optionsData$customFo2 = optionsData.customFormatErrorFn) !== null && _optionsData$customFo2 !== void 0 ? _optionsData$customFo2 : optionsData.formatError) !== null && _ref2 !== void 0 ? _ref2 : formatErrorFn;
- if (schema == null) {
- throw (0, _httpErrors.default)(500, 'GraphQL middleware options must contain a schema.');
- }
- if (request.method !== 'GET' && request.method !== 'POST') {
- throw (0, _httpErrors.default)(405, 'GraphQL only supports GET and POST requests.', {
- headers: {
- Allow: 'GET, POST'
- }
- });
- }
- const {
- query,
- variables,
- operationName
- } = params;
- showGraphiQL = canDisplayGraphiQL(request, params) && graphiql !== false;
- if (typeof graphiql !== 'boolean') {
- graphiqlOptions = graphiql;
- }
-
- if (query == null) {
- if (showGraphiQL) {
- return respondWithGraphiQL(response, graphiqlOptions);
- }
- throw (0, _httpErrors.default)(400, 'Must provide query string.');
- }
- const schemaValidationErrors = (0, _graphql.validateSchema)(schema);
- if (schemaValidationErrors.length > 0) {
-
- throw (0, _httpErrors.default)(500, 'GraphQL schema validation error.', {
- graphqlErrors: schemaValidationErrors
- });
- }
- let documentAST;
- try {
- documentAST = parseFn(new _graphql.Source(query, 'GraphQL request'));
- } catch (syntaxError) {
-
- throw (0, _httpErrors.default)(400, 'GraphQL syntax error.', {
- graphqlErrors: [syntaxError]
- });
- }
- const validationErrors = validateFn(schema, documentAST, [..._graphql.specifiedRules, ...validationRules]);
- if (validationErrors.length > 0) {
-
- throw (0, _httpErrors.default)(400, 'GraphQL validation error.', {
- graphqlErrors: validationErrors
- });
- }
- if (request.method === 'GET') {
-
- const operationAST = (0, _graphql.getOperationAST)(documentAST, operationName);
- if (operationAST && operationAST.operation !== 'query') {
-
-
-
- if (showGraphiQL) {
- return respondWithGraphiQL(response, graphiqlOptions, params);
- }
- throw (0, _httpErrors.default)(405, `Can only perform a ${operationAST.operation} operation from a POST request.`, {
- headers: {
- Allow: 'POST'
- }
- });
- }
- }
- try {
- result = await executeFn({
- schema,
- document: documentAST,
- rootValue,
- contextValue: context,
- variableValues: variables,
- operationName,
- fieldResolver,
- typeResolver
- });
- } catch (contextError) {
-
- throw (0, _httpErrors.default)(400, 'GraphQL execution context error.', {
- graphqlErrors: [contextError]
- });
- }
-
- if (extensionsFn) {
- const extensions = await extensionsFn({
- document: documentAST,
- variables,
- operationName,
- result,
- context
- });
- if (extensions != null) {
- result = { ...result,
- extensions
- };
- }
- }
- } catch (error) {
- var _error$status, _error$graphqlErrors;
-
- response.statusCode = (_error$status = error.status) !== null && _error$status !== void 0 ? _error$status : 500;
- if (error.headers != null) {
- for (const [key, value] of Object.entries(error.headers)) {
- response.setHeader(key, value);
- }
- }
- result = {
- data: undefined,
- errors: (_error$graphqlErrors = error.graphqlErrors) !== null && _error$graphqlErrors !== void 0 ? _error$graphqlErrors : [error]
- };
- }
-
-
-
-
- if (response.statusCode === 200 && result.data == null) {
- response.statusCode = 500;
- }
- const formattedResult = { ...result,
- errors: (_result$errors = result.errors) === null || _result$errors === void 0 ? void 0 : _result$errors.map(formatErrorFn)
- };
- if (showGraphiQL) {
- return respondWithGraphiQL(response, graphiqlOptions, params, formattedResult);
- }
-
-
- if (!pretty && typeof response.json === 'function') {
- response.json(formattedResult);
- } else {
- const payload = JSON.stringify(formattedResult, null, pretty ? 2 : 0);
- sendResponse(response, 'application/json', payload);
- }
- async function resolveOptions(requestParams) {
- const optionsResult = await Promise.resolve(typeof options === 'function' ? options(request, response, requestParams) : options);
- if (optionsResult == null || typeof optionsResult !== 'object') {
- throw new Error('GraphQL middleware option function must return an options object or a promise which will be resolved to an options object.');
- }
- if (optionsResult.formatError) {
-
- console.warn('`formatError` is deprecated and replaced by `customFormatErrorFn`. It will be removed in version 1.0.0.');
- }
- return optionsResult;
- }
- };
- }
- function respondWithGraphiQL(response, options, params, result) {
- const data = {
- query: params === null || params === void 0 ? void 0 : params.query,
- variables: params === null || params === void 0 ? void 0 : params.variables,
- operationName: params === null || params === void 0 ? void 0 : params.operationName,
- result
- };
- const payload = (0, _renderGraphiQL.renderGraphiQL)(data, options);
- return sendResponse(response, 'text/html', payload);
- }
- async function getGraphQLParams(request) {
- var _urlData$get, _urlData$get2;
- const urlData = new URLSearchParams(request.url.split('?')[1]);
- const bodyData = await (0, _parseBody.parseBody)(request);
- let query = (_urlData$get = urlData.get('query')) !== null && _urlData$get !== void 0 ? _urlData$get : bodyData.query;
- if (typeof query !== 'string') {
- query = null;
- }
- let variables = (_urlData$get2 = urlData.get('variables')) !== null && _urlData$get2 !== void 0 ? _urlData$get2 : bodyData.variables;
- if (typeof variables === 'string') {
- try {
- variables = JSON.parse(variables);
- } catch (error) {
- throw (0, _httpErrors.default)(400, 'Variables are invalid JSON.');
- }
- } else if (typeof variables !== 'object') {
- variables = null;
- }
- let operationName = urlData.get('operationName') || bodyData.operationName;
- if (typeof operationName !== 'string') {
- operationName = null;
- }
- const raw = urlData.get('raw') != null || bodyData.raw !== undefined;
- return {
- query,
- variables,
- operationName,
- raw
- };
- }
- function canDisplayGraphiQL(request, params) {
-
-
- return !params.raw && (0, _accepts.default)(request).types(['json', 'html']) === 'html';
- }
- function sendResponse(response, type, data) {
- const chunk = Buffer.from(data, 'utf8');
- response.setHeader('Content-Type', type + '; charset=utf-8');
- response.setHeader('Content-Length', String(chunk.length));
- response.end(chunk);
- }
|