Skip to content

Commit

Permalink
apollo-server-lambda: Explicitly support payloadFormatVersion 2.0
Browse files Browse the repository at this point in the history
Fixes #5084.

The sooner we get out of the business of understanding Lambda event formats, the
better. But this fix should be good for now.
  • Loading branch information
glasser committed Apr 9, 2021
1 parent f772ed6 commit 0434839
Show file tree
Hide file tree
Showing 2 changed files with 40 additions and 20 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ The version headers in this history reflect the versions of Apollo Server itself
- `apollo-server-core`: Fix type for `formatResponse` function. It never is called with a `null` argument, and is allowed to return `null`. [Issue #5009](https://github.com/apollographql/apollo-server/issues/5009) [PR #5089](https://github.com/apollographql/apollo-server/pull/5089)
- `apollo-server-lambda`: Fix regression in v2.21.2 where thrown errors were replaced by throwing the JS Error class itself. [PR #5085](https://github.com/apollographql/apollo-server/pull/5085)
- `apollo-server-core`: If a client sends a variable of the wrong type, this is now reported as an error with an `extensions.code` of `BAD_USER_INPUT` rather than `INTERNAL_SERVER_ERROR`. [PR #5091](https://github.com/apollographql/apollo-server/pull/5091) [Issue #3498](https://github.com/apollographql/apollo-server/issues/3498)
- `apollo-server-lambda`: Explicitly support API Gateway `payloadFormatVersion` 2.0. Previously some codepaths did appropriate checks to partially support 2.0 and other codepaths could lead to errors like `event.path.endsWith is not a function` (especially since v2.21.1). Note that this changes the TypeScript typing of the `onHealthCheck` callback passed to `createHandler` to indicate that it can receive either type of event. In your app, you should understand which payload format you're expecting and cast to `APIGatewayProxyEvent` or `APIGatewayProxyEventV2` from `aws-lambda`, or differentiate between them by checking to see if `'path' in event`. [Issue #5084](https://github.com/apollographql/apollo-server/issues/5084) [Issue #5016](https://github.com/apollographql/apollo-server/issues/5016)

## v2.22.2

Expand Down
59 changes: 39 additions & 20 deletions packages/apollo-server-lambda/src/ApolloServer.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import {
APIGatewayProxyCallback,
APIGatewayProxyEvent,
APIGatewayProxyEventV2,
APIGatewayProxyResult,
Context as LambdaContext,
} from 'aws-lambda';
Expand All @@ -21,7 +22,6 @@ import { ServerResponse, IncomingHttpHeaders, IncomingMessage } from 'http';

import { Headers } from 'apollo-server-env';
import { Readable, Writable } from 'stream';

export interface CreateHandlerOptions {
cors?: {
origin?: boolean | string | string[];
Expand All @@ -32,13 +32,34 @@ export interface CreateHandlerOptions {
maxAge?: number;
};
uploadsConfig?: FileUploadOptions;
onHealthCheck?: (req: APIGatewayProxyEvent) => Promise<any>;
onHealthCheck?: (req: APIGatewayProxyEventV1OrV2) => Promise<any>;
}

export class FileUploadRequest extends Readable {
headers!: IncomingHttpHeaders;
}

// We try to support payloadFormatEvent 1.0 and 2.0. See
// https://docs.aws.amazon.com/apigateway/latest/developerguide/http-api-develop-integrations-lambda.html
// for a bit of documentation as to what is in these objects. You can determine
// which one you have by checking `'path' in event` (V1 has path, V2 doesn't).
type APIGatewayProxyEventV1OrV2 = APIGatewayProxyEvent | APIGatewayProxyEventV2;

function eventHttpMethod(event: APIGatewayProxyEventV1OrV2): string {
return 'httpMethod' in event
? event.httpMethod
: event.requestContext.http.method;
}

function eventPath(event: APIGatewayProxyEventV1OrV2): string {
// Note: it's unclear if the V2 version should use `event.rawPath` or
// `event.requestContext.http.path`; I can't find any documentation about the
// distinction between the two. I'm choosing rawPath because that's what
// @vendia/serverless-express does (though it also looks at a `requestPath`
// field that doesn't exist in the docs or typings).
return 'path' in event ? event.path : event.rawPath;
}

// Lambda has two ways of defining a handler: as an async Promise-returning
// function, and as a callback-invoking function.
// https://docs.aws.amazon.com/lambda/latest/dg/nodejs-handler.html The async
Expand All @@ -56,16 +77,16 @@ export class FileUploadRequest extends Readable {
// this package always return an async handler.)
function maybeCallbackify(
asyncHandler: (
event: APIGatewayProxyEvent,
event: APIGatewayProxyEventV1OrV2,
context: LambdaContext,
) => Promise<APIGatewayProxyResult>,
): (
event: APIGatewayProxyEvent,
event: APIGatewayProxyEventV1OrV2,
context: LambdaContext,
callback: APIGatewayProxyCallback | undefined,
) => void | Promise<APIGatewayProxyResult> {
return (
event: APIGatewayProxyEvent,
event: APIGatewayProxyEventV1OrV2,
context: LambdaContext,
callback: APIGatewayProxyCallback | undefined,
) => {
Expand Down Expand Up @@ -96,7 +117,7 @@ export class ApolloServer extends ApolloServerBase {
// provides typings for the integration specific behavior, ideally this would
// be propagated with a generic to the super class
createGraphQLServerOptions(
event: APIGatewayProxyEvent,
event: APIGatewayProxyEventV1OrV2,
context: LambdaContext,
): Promise<GraphQLOptions> {
return super.graphQLServerOptions({ event, context });
Expand Down Expand Up @@ -154,7 +175,7 @@ export class ApolloServer extends ApolloServerBase {

return maybeCallbackify(
async (
event: APIGatewayProxyEvent,
event: APIGatewayProxyEventV1OrV2,
context: LambdaContext,
): Promise<APIGatewayProxyResult> => {
const eventHeaders = new Headers(event.headers);
Expand Down Expand Up @@ -202,7 +223,7 @@ export class ApolloServer extends ApolloServerBase {
return headersObject;
}, {});

if (event.httpMethod === 'OPTIONS') {
if (eventHttpMethod(event) === 'OPTIONS') {
return {
body: '',
statusCode: 204,
Expand All @@ -212,7 +233,7 @@ export class ApolloServer extends ApolloServerBase {
};
}

if (event.path.endsWith('/.well-known/apollo/server-health')) {
if (eventPath(event).endsWith('/.well-known/apollo/server-health')) {
if (onHealthCheck) {
try {
await onHealthCheck(event);
Expand All @@ -237,14 +258,11 @@ export class ApolloServer extends ApolloServerBase {
};
}

if (this.playgroundOptions && event.httpMethod === 'GET') {
if (this.playgroundOptions && eventHttpMethod(event) === 'GET') {
const acceptHeader =
event.headers['Accept'] || event.headers['accept'];
if (acceptHeader && acceptHeader.includes('text/html')) {
const path =
event.path ||
(event.requestContext && event.requestContext.path) ||
'/';
const path = eventPath(event) || '/';

const playgroundRenderPageOptions: PlaygroundRenderPageOptions = {
endpoint: path,
Expand Down Expand Up @@ -310,7 +328,7 @@ export class ApolloServer extends ApolloServerBase {
body = Buffer.from(body, 'base64').toString();
}

if (event.httpMethod === 'POST' && !body) {
if (eventHttpMethod(event) === 'POST' && !body) {
return {
body: 'POST body missing.',
statusCode: 500,
Expand All @@ -319,28 +337,29 @@ export class ApolloServer extends ApolloServerBase {

if (bodyFromFileUploads) {
query = bodyFromFileUploads;
} else if (body && event.httpMethod === 'POST' && isMultipart) {
} else if (body && eventHttpMethod(event) === 'POST' && isMultipart) {
// XXX Not clear if this was only intended to handle the uploads
// case or if it had more general applicability
query = body as any;
} else if (body && event.httpMethod === 'POST') {
} else if (body && eventHttpMethod(event) === 'POST') {
query = JSON.parse(body);
} else {
// XXX Note that
query = event.queryStringParameters || {};
}

try {
const { graphqlResponse, responseInit } = await runHttpQuery(
[event, context],
{
method: event.httpMethod,
method: eventHttpMethod(event),
options: async () => {
return this.createGraphQLServerOptions(event, context);
},
query,
request: {
url: event.path,
method: event.httpMethod,
url: eventPath(event),
method: eventHttpMethod(event),
headers: eventHeaders,
},
},
Expand Down

0 comments on commit 0434839

Please sign in to comment.