mirror of
https://github.com/vale981/apollo-server
synced 2025-03-20 01:36:39 -04:00

We recommend the use of apollo-engine-reporting and the other built-in features of AS 2.0 rather than engineProxy. You can still use the apollo-engine npm module manually with AS 2.0, or stay on AS 1.0.
214 lines
5.6 KiB
TypeScript
214 lines
5.6 KiB
TypeScript
import hapi from 'hapi';
|
|
import { ApolloServerBase } from 'apollo-server-core';
|
|
import { parseAll } from 'accept';
|
|
import { renderPlaygroundPage } from 'graphql-playground-html';
|
|
import {
|
|
processRequest as processFileUploads,
|
|
GraphQLUpload,
|
|
} from 'apollo-upload-server';
|
|
|
|
import { graphqlHapi } from './hapiApollo';
|
|
|
|
export { GraphQLOptions, GraphQLExtension } from 'apollo-server-core';
|
|
import { GraphQLOptions } from 'apollo-server-core';
|
|
|
|
const gql = String.raw;
|
|
|
|
export class ApolloServer extends ApolloServerBase {
|
|
//This translates the arguments from the middleware into graphQL options It
|
|
//provides typings for the integration specific behavior, ideally this would
|
|
//be propagated with a generic to the super class
|
|
async createGraphQLServerOptions(
|
|
request: hapi.Request,
|
|
h: hapi.ResponseToolkit,
|
|
): Promise<GraphQLOptions> {
|
|
return super.graphQLServerOptions({ request, h });
|
|
}
|
|
}
|
|
|
|
export interface ServerRegistration {
|
|
app?: hapi.Server;
|
|
//The options type should exclude port
|
|
options?: hapi.ServerOptions;
|
|
server: ApolloServer;
|
|
path?: string;
|
|
cors?: boolean;
|
|
onHealthCheck?: (request: hapi.Request) => Promise<any>;
|
|
disableHealthCheck?: boolean;
|
|
enableGUI?: boolean;
|
|
uploads?: boolean | Record<string, any>;
|
|
}
|
|
|
|
export interface HapiListenOptions {
|
|
port?: number | string;
|
|
host?: string; // default: ''
|
|
pipePath?: string;
|
|
graphqlPaths?: string[]; // default: ['/graphql']
|
|
innerHost?: string; // default: '127.0.0.1'. This is where Node listens.
|
|
}
|
|
|
|
const handleFileUploads = (uploadsConfig: Record<string, any>) => async (
|
|
request: hapi.Request,
|
|
) => {
|
|
if (request.mime === 'multipart/form-data') {
|
|
Object.defineProperty(request, 'payload', {
|
|
value: await processFileUploads(request, uploadsConfig),
|
|
writable: false,
|
|
});
|
|
}
|
|
};
|
|
|
|
export const registerServer = async ({
|
|
app,
|
|
options,
|
|
server,
|
|
cors,
|
|
path,
|
|
disableHealthCheck,
|
|
enableGUI,
|
|
onHealthCheck,
|
|
uploads,
|
|
}: ServerRegistration) => {
|
|
if (!path) path = '/graphql';
|
|
|
|
let hapiApp: hapi.Server;
|
|
if (app) {
|
|
hapiApp = app;
|
|
if (options) {
|
|
console.warn(`A Hapi Server was passed in, so the options are ignored`);
|
|
}
|
|
} else if (options) {
|
|
if ((options as any).port) {
|
|
throw new Error(`
|
|
The options for registerServer should not include a port, since autoListen is set to false. Please set the port under the http options in listen:
|
|
|
|
const server = new ApolloServer({ typeDefs, resolvers });
|
|
|
|
registerServer({
|
|
server,
|
|
options,
|
|
});
|
|
|
|
server.listen({ http: { port: YOUR_PORT_HERE } });
|
|
`);
|
|
}
|
|
hapiApp = new hapi.Server({ ...options, autoListen: false });
|
|
} else {
|
|
hapiApp = new hapi.Server({ autoListen: false });
|
|
}
|
|
|
|
if (uploads !== false) {
|
|
server.enhanceSchema({
|
|
typeDefs: gql`
|
|
scalar Upload
|
|
`,
|
|
resolvers: { Upload: GraphQLUpload },
|
|
});
|
|
}
|
|
|
|
await hapiApp.ext({
|
|
type: 'onRequest',
|
|
method: async function(request, h) {
|
|
if (request.path !== path) {
|
|
return h.continue;
|
|
}
|
|
|
|
if (uploads !== false) {
|
|
await handleFileUploads(typeof uploads !== 'boolean' ? uploads : {})(
|
|
request,
|
|
);
|
|
}
|
|
|
|
// enableGUI takes precedence over the server tools setting
|
|
if (
|
|
(enableGUI || (enableGUI === undefined && !server.disableTools)) &&
|
|
request.method === 'get'
|
|
) {
|
|
//perform more expensive content-type check only if necessary
|
|
const accept = parseAll(request.headers);
|
|
const types = accept.mediaTypes as string[];
|
|
const prefersHTML =
|
|
types.find(
|
|
(x: string) => x === 'text/html' || x === 'application/json',
|
|
) === 'text/html';
|
|
|
|
if (prefersHTML) {
|
|
return h
|
|
.response(
|
|
renderPlaygroundPage({
|
|
subscriptionEndpoint: server.subscriptionsPath,
|
|
endpoint: path,
|
|
version: '1.4.0',
|
|
}),
|
|
)
|
|
.type('text/html')
|
|
.takeover();
|
|
}
|
|
}
|
|
return h.continue;
|
|
},
|
|
});
|
|
|
|
if (!disableHealthCheck) {
|
|
await hapiApp.route({
|
|
method: '*',
|
|
path: '/.well-known/apollo/server-health',
|
|
options: {
|
|
cors: typeof cors === 'boolean' ? cors : true,
|
|
},
|
|
handler: async function(request, h) {
|
|
if (onHealthCheck) {
|
|
try {
|
|
await onHealthCheck(request);
|
|
} catch {
|
|
const response = h.response({ status: 'fail' });
|
|
response.code(503);
|
|
response.type('application/health+json');
|
|
return response;
|
|
}
|
|
}
|
|
const response = h.response({ status: 'pass' });
|
|
response.type('application/health+json');
|
|
return response;
|
|
},
|
|
});
|
|
}
|
|
|
|
await hapiApp.register({
|
|
plugin: graphqlHapi,
|
|
options: {
|
|
path: path,
|
|
graphqlOptions: server.createGraphQLServerOptions.bind(server),
|
|
route: {
|
|
cors: typeof cors === 'boolean' ? cors : true,
|
|
},
|
|
},
|
|
});
|
|
|
|
server.use({ path, getHttp: () => hapiApp.listener });
|
|
|
|
const listen = server.listen.bind(server);
|
|
server.listen = async options => {
|
|
//requires that autoListen is false, so that
|
|
//hapi sets up app.listener without start
|
|
await hapiApp.start();
|
|
|
|
//While this is not strictly necessary, it ensures that apollo server calls
|
|
//listen first, setting the port. Otherwise the hapi server constructor
|
|
//sets the port
|
|
if (hapiApp.listener.listening) {
|
|
throw Error(
|
|
`
|
|
Ensure that constructor of Hapi server sets autoListen to false, as follows:
|
|
|
|
const app = Hapi.server({
|
|
autoListen: false,
|
|
//other parameters
|
|
});
|
|
`,
|
|
);
|
|
}
|
|
|
|
return listen({ ...options });
|
|
};
|
|
};
|