blob: 277e85c8bf26c37aba08c0fdd360729c446d01d0 [file] [log] [blame]
/*!
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*/
import { URL } from 'url';
import ContextManager from '../trace/context/ContextManager';
import { Component } from '../trace/Component';
import Tag from '../Tag';
import { SpanLayer } from '../proto/language-agent/Tracing_pb';
import { ContextCarrier } from '../trace/context/ContextCarrier';
import Span from '../trace/span/Span';
import DummySpan from '../trace/span/DummySpan';
import { ignoreHttpMethodCheck } from '../config/AgentConfig';
import { AWSLambdaTriggerPlugin } from './AWSLambdaTriggerPlugin';
class AWSLambdaGatewayAPIREST extends AWSLambdaTriggerPlugin {
start(event: any, context: any): [Span, any] {
const headers = event.headers;
const reqCtx = event.requestContext;
const method = reqCtx?.httpMethod ?? event.httpMethod;
const proto = reqCtx?.protocol ? reqCtx.protocol.split('/')[0].toLowerCase() : headers?.['X-Forwarded-Proto'];
const port = headers?.['X-Forwarded-Port'] || '';
const host = headers?.Host ?? (reqCtx?.domainName || '');
const hostport = host ? (port ? `${host}:${port}` : host) : port;
const operation = reqCtx?.path ?? event.path ?? (context.functionName ? `/${context.functionName}` : '/');
const query = event.multiValueQueryStringParameters
? '?' +
Object.entries(event.multiValueQueryStringParameters)
.map(([k, vs]: any[]) => vs.map((v: String) => `${k}=${v}`).join('&'))
.join('&')
: event.queryStringParameters
? '?' +
Object.entries(event.queryStringParameters)
.map(([k, v]) => `${k}=${v}`)
.join('&')
: '';
const carrier = headers && ContextCarrier.from(headers);
const span =
method && ignoreHttpMethodCheck(method)
? DummySpan.create()
: ContextManager.current.newEntrySpan(operation, carrier);
span.component = Component.AWSLAMBDA_GATEWAYAPIREST;
span.peer = reqCtx?.identity?.sourceIp ?? headers?.['X-Forwarded-For'] ?? 'Unknown';
if (method) span.tag(Tag.httpMethod(method));
if (hostport && proto) span.tag(Tag.httpURL(new URL(`${proto}://${hostport}${operation}${query}`).toString()));
span.start();
return [span, event];
}
stop(span: Span, err: Error | null, res: any): void {
const statusCode = res?.statusCode || (typeof res === 'number' ? res : err ? 500 : null);
if (statusCode) {
if (statusCode >= 400) span.errored = true;
span.tag(Tag.httpStatusCode(statusCode));
}
span.stop();
}
}
export default new AWSLambdaGatewayAPIREST();