1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.StepFunctionsIntegration = void 0;
|
5 | const jsiiDeprecationWarnings = require("../../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const fs = require("fs");
|
8 | const path = require("path");
|
9 | const iam = require("@aws-cdk/aws-iam");
|
10 | const sfn = require("@aws-cdk/aws-stepfunctions");
|
11 | const core_1 = require("@aws-cdk/core");
|
12 | const integration_1 = require("../integration");
|
13 | const model_1 = require("../model");
|
14 | const aws_1 = require("./aws");
|
15 | /**
|
16 | * Options to integrate with various StepFunction API
|
17 | */
|
18 | class StepFunctionsIntegration {
|
19 | /**
|
20 | * Integrates a Synchronous Express State Machine from AWS Step Functions to an API Gateway method.
|
21 | *
|
22 | * @example
|
23 | *
|
24 | * const stateMachine = new stepfunctions.StateMachine(this, 'MyStateMachine', {
|
25 | * stateMachineType: stepfunctions.StateMachineType.EXPRESS,
|
26 | * definition: stepfunctions.Chain.start(new stepfunctions.Pass(this, 'Pass')),
|
27 | * });
|
28 | *
|
29 | * const api = new apigateway.RestApi(this, 'Api', {
|
30 | * restApiName: 'MyApi',
|
31 | * });
|
32 | * api.root.addMethod('GET', apigateway.StepFunctionsIntegration.startExecution(stateMachine));
|
33 | */
|
34 | static startExecution(stateMachine, options) {
|
35 | try {
|
36 | jsiiDeprecationWarnings._aws_cdk_aws_apigateway_StepFunctionsExecutionIntegrationOptions(options);
|
37 | }
|
38 | catch (error) {
|
39 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
40 | Error.captureStackTrace(error, this.startExecution);
|
41 | }
|
42 | throw error;
|
43 | }
|
44 | return new StepFunctionsExecutionIntegration(stateMachine, options);
|
45 | }
|
46 | }
|
47 | exports.StepFunctionsIntegration = StepFunctionsIntegration;
|
48 | _a = JSII_RTTI_SYMBOL_1;
|
49 | StepFunctionsIntegration[_a] = { fqn: "@aws-cdk/aws-apigateway.StepFunctionsIntegration", version: "1.156.0" };
|
50 | class StepFunctionsExecutionIntegration extends aws_1.AwsIntegration {
|
51 | constructor(stateMachine, options = {}) {
|
52 | super({
|
53 | service: 'states',
|
54 | action: 'StartSyncExecution',
|
55 | options: {
|
56 | credentialsRole: options.credentialsRole,
|
57 | integrationResponses: integrationResponse(),
|
58 | passthroughBehavior: integration_1.PassthroughBehavior.NEVER,
|
59 | requestTemplates: requestTemplates(stateMachine, options),
|
60 | ...options,
|
61 | },
|
62 | });
|
63 | this.stateMachine = stateMachine;
|
64 | }
|
65 | bind(method) {
|
66 | var _b, _c;
|
67 | const bindResult = super.bind(method);
|
68 | const credentialsRole = (_c = (_b = bindResult.options) === null || _b === void 0 ? void 0 : _b.credentialsRole) !== null && _c !== void 0 ? _c : new iam.Role(method, 'StartSyncExecutionRole', {
|
69 | assumedBy: new iam.ServicePrincipal('apigateway.amazonaws.com'),
|
70 | });
|
71 | this.stateMachine.grantStartSyncExecution(credentialsRole);
|
72 | let stateMachineName;
|
73 | if (this.stateMachine instanceof sfn.StateMachine) {
|
74 | const stateMachineType = this.stateMachine.stateMachineType;
|
75 | if (stateMachineType !== sfn.StateMachineType.EXPRESS) {
|
76 | throw new Error('State Machine must be of type "EXPRESS". Please use StateMachineType.EXPRESS as the stateMachineType');
|
77 | }
|
78 | //if not imported, extract the name from the CFN layer to reach the
|
79 | //literal value if it is given (rather than a token)
|
80 | stateMachineName = this.stateMachine.node.defaultChild.stateMachineName;
|
81 | }
|
82 | else {
|
83 | //imported state machine
|
84 | stateMachineName = `StateMachine-${this.stateMachine.stack.node.addr}`;
|
85 | }
|
86 | let deploymentToken;
|
87 | if (stateMachineName !== undefined && !core_1.Token.isUnresolved(stateMachineName)) {
|
88 | deploymentToken = JSON.stringify({ stateMachineName });
|
89 | }
|
90 | for (const methodResponse of METHOD_RESPONSES) {
|
91 | method.addMethodResponse(methodResponse);
|
92 | }
|
93 | return {
|
94 | ...bindResult,
|
95 | options: {
|
96 | ...bindResult.options,
|
97 | credentialsRole,
|
98 | },
|
99 | deploymentToken,
|
100 | };
|
101 | }
|
102 | }
|
103 | /**
|
104 | * Defines the integration response that passes the result on success,
|
105 | * or the error on failure, from the synchronous execution to the caller.
|
106 | *
|
107 | * @returns integrationResponse mapping
|
108 | */
|
109 | function integrationResponse() {
|
110 | const errorResponse = [
|
111 | {
|
112 | /**
|
113 | * Specifies the regular expression (regex) pattern used to choose
|
114 | * an integration response based on the response from the back end.
|
115 | * In this case it will match all '4XX' HTTP Errors
|
116 | */
|
117 | selectionPattern: '4\\d{2}',
|
118 | statusCode: '400',
|
119 | responseTemplates: {
|
120 | 'application/json': `{
|
121 | "error": "Bad request!"
|
122 | }`,
|
123 | },
|
124 | },
|
125 | {
|
126 | /**
|
127 | * Match all '5XX' HTTP Errors
|
128 | */
|
129 | selectionPattern: '5\\d{2}',
|
130 | statusCode: '500',
|
131 | responseTemplates: {
|
132 | 'application/json': '"error": $input.path(\'$.error\')',
|
133 | },
|
134 | },
|
135 | ];
|
136 | const integResponse = [
|
137 | {
|
138 | statusCode: '200',
|
139 | responseTemplates: {
|
140 | /* eslint-disable */
|
141 | 'application/json': [
|
142 | '#set($inputRoot = $input.path(\'$\'))',
|
143 | '#if($input.path(\'$.status\').toString().equals("FAILED"))',
|
144 | '#set($context.responseOverride.status = 500)',
|
145 | '{',
|
146 | '"error": "$input.path(\'$.error\')",',
|
147 | '"cause": "$input.path(\'$.cause\')"',
|
148 | '}',
|
149 | '#else',
|
150 | '$input.path(\'$.output\')',
|
151 | '#end',
|
152 | ].join('\n'),
|
153 | },
|
154 | },
|
155 | ...errorResponse,
|
156 | ];
|
157 | return integResponse;
|
158 | }
|
159 | /**
|
160 | * Defines the request template that will be used for the integration
|
161 | * @param stateMachine
|
162 | * @param options
|
163 | * @returns requestTemplate
|
164 | */
|
165 | function requestTemplates(stateMachine, options) {
|
166 | const templateStr = templateString(stateMachine, options);
|
167 | const requestTemplate = {
|
168 | 'application/json': templateStr,
|
169 | };
|
170 | return requestTemplate;
|
171 | }
|
172 | /**
|
173 | * Reads the VTL template and returns the template string to be used
|
174 | * for the request template.
|
175 | *
|
176 | * @param stateMachine
|
177 | * @param includeRequestContext
|
178 | * @param options
|
179 | * @reutrns templateString
|
180 | */
|
181 | function templateString(stateMachine, options) {
|
182 | var _b, _c, _d, _e;
|
183 | let templateStr;
|
184 | let requestContextStr = '';
|
185 | const includeHeader = (_b = options.headers) !== null && _b !== void 0 ? _b : false;
|
186 | const includeQueryString = (_c = options.querystring) !== null && _c !== void 0 ? _c : true;
|
187 | const includePath = (_d = options.path) !== null && _d !== void 0 ? _d : true;
|
188 | const includeAuthorizer = (_e = options.authorizer) !== null && _e !== void 0 ? _e : false;
|
189 | if (options.requestContext && Object.keys(options.requestContext).length > 0) {
|
190 | requestContextStr = requestContext(options.requestContext);
|
191 | }
|
192 | templateStr = fs.readFileSync(path.join(__dirname, 'stepfunctions.vtl'), { encoding: 'utf-8' });
|
193 | templateStr = templateStr.replace('%STATEMACHINE%', stateMachine.stateMachineArn);
|
194 | templateStr = templateStr.replace('%INCLUDE_HEADERS%', String(includeHeader));
|
195 | templateStr = templateStr.replace('%INCLUDE_QUERYSTRING%', String(includeQueryString));
|
196 | templateStr = templateStr.replace('%INCLUDE_PATH%', String(includePath));
|
197 | templateStr = templateStr.replace('%INCLUDE_AUTHORIZER%', String(includeAuthorizer));
|
198 | templateStr = templateStr.replace('%REQUESTCONTEXT%', requestContextStr);
|
199 | return templateStr;
|
200 | }
|
201 | function requestContext(requestContextObj) {
|
202 | const context = {
|
203 | accountId: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.accountId) ? '$context.identity.accountId' : undefined,
|
204 | apiId: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.apiId) ? '$context.apiId' : undefined,
|
205 | apiKey: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.apiKey) ? '$context.identity.apiKey' : undefined,
|
206 | authorizerPrincipalId: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.authorizerPrincipalId) ? '$context.authorizer.principalId' : undefined,
|
207 | caller: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.caller) ? '$context.identity.caller' : undefined,
|
208 | cognitoAuthenticationProvider: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.cognitoAuthenticationProvider) ? '$context.identity.cognitoAuthenticationProvider' : undefined,
|
209 | cognitoAuthenticationType: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.cognitoAuthenticationType) ? '$context.identity.cognitoAuthenticationType' : undefined,
|
210 | cognitoIdentityId: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.cognitoIdentityId) ? '$context.identity.cognitoIdentityId' : undefined,
|
211 | cognitoIdentityPoolId: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.cognitoIdentityPoolId) ? '$context.identity.cognitoIdentityPoolId' : undefined,
|
212 | httpMethod: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.httpMethod) ? '$context.httpMethod' : undefined,
|
213 | stage: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.stage) ? '$context.stage' : undefined,
|
214 | sourceIp: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.sourceIp) ? '$context.identity.sourceIp' : undefined,
|
215 | user: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.user) ? '$context.identity.user' : undefined,
|
216 | userAgent: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.userAgent) ? '$context.identity.userAgent' : undefined,
|
217 | userArn: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.userArn) ? '$context.identity.userArn' : undefined,
|
218 | requestId: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.requestId) ? '$context.requestId' : undefined,
|
219 | resourceId: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.resourceId) ? '$context.resourceId' : undefined,
|
220 | resourcePath: (requestContextObj === null || requestContextObj === void 0 ? void 0 : requestContextObj.resourcePath) ? '$context.resourcePath' : undefined,
|
221 | };
|
222 | const contextAsString = JSON.stringify(context);
|
223 | // The VTL Template conflicts with double-quotes (") for strings.
|
224 | // Before sending to the template, we replace double-quotes (") with @@ and replace it back inside the .vtl file
|
225 | const doublequotes = '"';
|
226 | const replaceWith = '@@';
|
227 | return contextAsString.split(doublequotes).join(replaceWith);
|
228 | }
|
229 | /**
|
230 | * Method response model for each HTTP code response
|
231 | */
|
232 | const METHOD_RESPONSES = [
|
233 | {
|
234 | statusCode: '200',
|
235 | responseModels: {
|
236 | 'application/json': model_1.Model.EMPTY_MODEL,
|
237 | },
|
238 | },
|
239 | {
|
240 | statusCode: '400',
|
241 | responseModels: {
|
242 | 'application/json': model_1.Model.ERROR_MODEL,
|
243 | },
|
244 | },
|
245 | {
|
246 | statusCode: '500',
|
247 | responseModels: {
|
248 | 'application/json': model_1.Model.ERROR_MODEL,
|
249 | },
|
250 | },
|
251 | ];
|
252 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stepfunctions.js","sourceRoot":"","sources":["stepfunctions.ts"],"names":[],"mappings":";;;;;;AAAA,yBAAyB;AACzB,6BAA6B;AAC7B,wCAAwC;AACxC,kDAAkD;AAClD,wCAAsC;AAEtC,gDAA4F;AAE5F,oCAAiC;AACjC,+BAAuC;AA8EvC;;GAEG;AACH,MAAa,wBAAwB;IACnC;;;;;;;;;;;;;;OAcG;IACI,MAAM,CAAC,cAAc,CAAC,YAA+B,EAAE,OAAkD;;;;;;;;;;QAC9G,OAAO,IAAI,iCAAiC,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;KACrE;;AAlBH,4DAmBC;;;AAED,MAAM,iCAAkC,SAAQ,oBAAc;IAE5D,YAAY,YAA+B,EAAE,UAAoD,EAAE;QACjG,KAAK,CAAC;YACJ,OAAO,EAAE,QAAQ;YACjB,MAAM,EAAE,oBAAoB;YAC5B,OAAO,EAAE;gBACP,eAAe,EAAE,OAAO,CAAC,eAAe;gBACxC,oBAAoB,EAAE,mBAAmB,EAAE;gBAC3C,mBAAmB,EAAE,iCAAmB,CAAC,KAAK;gBAC9C,gBAAgB,EAAE,gBAAgB,CAAC,YAAY,EAAE,OAAO,CAAC;gBACzD,GAAG,OAAO;aACX;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC;KAClC;IAEM,IAAI,CAAC,MAAc;;QACxB,MAAM,UAAU,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAEtC,MAAM,eAAe,eAAG,UAAU,CAAC,OAAO,0CAAE,eAAe,mCAAI,IAAI,GAAG,CAAC,IAAI,CAAC,MAAM,EAAE,wBAAwB,EAAE;YAC5G,SAAS,EAAE,IAAI,GAAG,CAAC,gBAAgB,CAAC,0BAA0B,CAAC;SAChE,CAAC,CAAC;QACH,IAAI,CAAC,YAAY,CAAC,uBAAuB,CAAC,eAAe,CAAC,CAAC;QAE3D,IAAI,gBAAgB,CAAC;QAErB,IAAI,IAAI,CAAC,YAAY,YAAY,GAAG,CAAC,YAAY,EAAE;YACjD,MAAM,gBAAgB,GAAI,IAAI,CAAC,YAAiC,CAAC,gBAAgB,CAAC;YAClF,IAAI,gBAAgB,KAAK,GAAG,CAAC,gBAAgB,CAAC,OAAO,EAAE;gBACrD,MAAM,IAAI,KAAK,CAAC,sGAAsG,CAAC,CAAC;aACzH;YAED,mEAAmE;YACnE,oDAAoD;YACpD,gBAAgB,GAAI,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,YAAoC,CAAC,gBAAgB,CAAC;SAClG;aAAM;YACL,wBAAwB;YACxB,gBAAgB,GAAG,gBAAgB,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;SACxE;QAED,IAAI,eAAe,CAAC;QAEpB,IAAI,gBAAgB,KAAK,SAAS,IAAI,CAAC,YAAK,CAAC,YAAY,CAAC,gBAAgB,CAAC,EAAE;YAC3E,eAAe,GAAG,IAAI,CAAC,SAAS,CAAC,EAAE,gBAAgB,EAAE,CAAC,CAAC;SACxD;QAED,KAAK,MAAM,cAAc,IAAI,gBAAgB,EAAE;YAC7C,MAAM,CAAC,iBAAiB,CAAC,cAAc,CAAC,CAAC;SAC1C;QAED,OAAO;YACL,GAAG,UAAU;YACb,OAAO,EAAE;gBACP,GAAG,UAAU,CAAC,OAAO;gBACrB,eAAe;aAChB;YACD,eAAe;SAChB,CAAC;KACH;CACF;AAED;;;;;GAKG;AACH,SAAS,mBAAmB;IAC1B,MAAM,aAAa,GAAG;QACpB;YACE;;;;eAIG;YACH,gBAAgB,EAAE,SAAS;YAC3B,UAAU,EAAE,KAAK;YACjB,iBAAiB,EAAE;gBACjB,kBAAkB,EAAE;;YAEhB;aACL;SACF;QACD;YACE;;eAEG;YACH,gBAAgB,EAAE,SAAS;YAC3B,UAAU,EAAE,KAAK;YACjB,iBAAiB,EAAE;gBACjB,kBAAkB,EAAE,mCAAmC;aACxD;SACF;KACF,CAAC;IAEF,MAAM,aAAa,GAAG;QACpB;YACE,UAAU,EAAE,KAAK;YACjB,iBAAiB,EAAE;gBACjB,oBAAoB;gBACpB,kBAAkB,EAAE;oBAClB,uCAAuC;oBACvC,4DAA4D;oBAC1D,8CAA8C;oBAC9C,GAAG;oBACD,sCAAsC;oBACtC,qCAAqC;oBACvC,GAAG;oBACL,OAAO;oBACL,2BAA2B;oBAC7B,MAAM;iBAEP,CAAC,IAAI,CAAC,IAAI,CAAC;aACb;SACF;QACD,GAAG,aAAa;KACjB,CAAC;IAEF,OAAO,aAAa,CAAC;AACvB,CAAC;AAED;;;;;GAKG;AACH,SAAS,gBAAgB,CAAC,YAA+B,EAAE,OAAiD;IAC1G,MAAM,WAAW,GAAG,cAAc,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;IAE1D,MAAM,eAAe,GACnB;QACE,kBAAkB,EAAE,WAAW;KAChC,CAAC;IAEJ,OAAO,eAAe,CAAC;AACzB,CAAC;AAED;;;;;;;;GAQG;AACH,SAAS,cAAc,CACrB,YAA+B,EAC/B,OAAiD;;IACjD,IAAI,WAAmB,CAAC;IAExB,IAAI,iBAAiB,GAAG,EAAE,CAAC;IAE3B,MAAM,aAAa,SAAG,OAAO,CAAC,OAAO,mCAAG,KAAK,CAAC;IAC9C,MAAM,kBAAkB,SAAG,OAAO,CAAC,WAAW,mCAAG,IAAI,CAAC;IACtD,MAAM,WAAW,SAAG,OAAO,CAAC,IAAI,mCAAG,IAAI,CAAC;IACxC,MAAM,iBAAiB,SAAG,OAAO,CAAC,UAAU,mCAAI,KAAK,CAAC;IAEtD,IAAI,OAAO,CAAC,cAAc,IAAI,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE;QAC5E,iBAAiB,GAAG,cAAc,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;KAC5D;IAED,WAAW,GAAG,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,mBAAmB,CAAC,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAC;IAChG,WAAW,GAAG,WAAW,CAAC,OAAO,CAAC,gBAAgB,EAAE,YAAY,CAAC,eAAe,CAAC,CAAC;IAClF,WAAW,GAAG,WAAW,CAAC,OAAO,CAAC,mBAAmB,EAAE,MAAM,CAAC,aAAa,CAAC,CAAC,CAAC;IAC9E,WAAW,GAAG,WAAW,CAAC,OAAO,CAAC,uBAAuB,EAAE,MAAM,CAAC,kBAAkB,CAAC,CAAC,CAAC;IACvF,WAAW,GAAG,WAAW,CAAC,OAAO,CAAC,gBAAgB,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC;IACzE,WAAW,GAAG,WAAW,CAAC,OAAO,CAAC,sBAAsB,EAAE,MAAM,CAAC,iBAAiB,CAAC,CAAC,CAAC;IACrF,WAAW,GAAG,WAAW,CAAC,OAAO,CAAC,kBAAkB,EAAE,iBAAiB,CAAC,CAAC;IAEzE,OAAO,WAAW,CAAC;AACrB,CAAC;AAED,SAAS,cAAc,CAAC,iBAA6C;IACnE,MAAM,OAAO,GAAG;QACd,SAAS,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,SAAS,EAAA,CAAC,CAAC,6BAA6B,CAAA,CAAC,CAAC,SAAS;QACjF,KAAK,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,KAAK,EAAA,CAAC,CAAC,gBAAgB,CAAA,CAAC,CAAC,SAAS;QAC5D,MAAM,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,MAAM,EAAA,CAAC,CAAC,0BAA0B,CAAA,CAAC,CAAC,SAAS;QACxE,qBAAqB,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,qBAAqB,EAAA,CAAC,CAAC,iCAAiC,CAAA,CAAC,CAAC,SAAS;QAC7G,MAAM,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,MAAM,EAAA,CAAC,CAAC,0BAA0B,CAAA,CAAC,CAAC,SAAS;QACxE,6BAA6B,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,6BAA6B,EAAA,CAAC,CAAC,iDAAiD,CAAA,CAAC,CAAC,SAAS;QAC7I,yBAAyB,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,yBAAyB,EAAA,CAAC,CAAC,6CAA6C,CAAA,CAAC,CAAC,SAAS;QACjI,iBAAiB,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,iBAAiB,EAAA,CAAC,CAAC,qCAAqC,CAAA,CAAC,CAAC,SAAS;QACzG,qBAAqB,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,qBAAqB,EAAA,CAAC,CAAC,yCAAyC,CAAA,CAAC,CAAC,SAAS;QACrH,UAAU,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,UAAU,EAAA,CAAC,CAAC,qBAAqB,CAAA,CAAC,CAAC,SAAS;QAC3E,KAAK,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,KAAK,EAAA,CAAC,CAAC,gBAAgB,CAAA,CAAC,CAAC,SAAS;QAC5D,QAAQ,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,QAAQ,EAAA,CAAC,CAAC,4BAA4B,CAAA,CAAC,CAAC,SAAS;QAC9E,IAAI,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,IAAI,EAAA,CAAC,CAAC,wBAAwB,CAAA,CAAC,CAAC,SAAS;QAClE,SAAS,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,SAAS,EAAA,CAAC,CAAC,6BAA6B,CAAA,CAAC,CAAC,SAAS;QACjF,OAAO,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,OAAO,EAAA,CAAC,CAAC,2BAA2B,CAAA,CAAC,CAAC,SAAS;QAC3E,SAAS,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,SAAS,EAAA,CAAC,CAAC,oBAAoB,CAAA,CAAC,CAAC,SAAS;QACxE,UAAU,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,UAAU,EAAA,CAAC,CAAC,qBAAqB,CAAA,CAAC,CAAC,SAAS;QAC3E,YAAY,EAAE,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,YAAY,EAAA,CAAC,CAAC,uBAAuB,CAAA,CAAC,CAAC,SAAS;KAClF,CAAC;IAEF,MAAM,eAAe,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;IAEhD,iEAAiE;IACjE,gHAAgH;IAChH,MAAM,YAAY,GAAG,GAAG,CAAC;IACzB,MAAM,WAAW,GAAG,IAAI,CAAC;IACzB,OAAO,eAAe,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;AAC/D,CAAC;AAED;;GAEG;AACH,MAAM,gBAAgB,GAAG;IACvB;QACE,UAAU,EAAE,KAAK;QACjB,cAAc,EAAE;YACd,kBAAkB,EAAE,aAAK,CAAC,WAAW;SACtC;KACF;IACD;QACE,UAAU,EAAE,KAAK;QACjB,cAAc,EAAE;YACd,kBAAkB,EAAE,aAAK,CAAC,WAAW;SACtC;KACF;IACD;QACE,UAAU,EAAE,KAAK;QACjB,cAAc,EAAE;YACd,kBAAkB,EAAE,aAAK,CAAC,WAAW;SACtC;KACF;CACF,CAAC","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport * as iam from '@aws-cdk/aws-iam';\nimport * as sfn from '@aws-cdk/aws-stepfunctions';\nimport { Token } from '@aws-cdk/core';\nimport { RequestContext } from '.';\nimport { IntegrationConfig, IntegrationOptions, PassthroughBehavior } from '../integration';\nimport { Method } from '../method';\nimport { Model } from '../model';\nimport { AwsIntegration } from './aws';\n/**\n * Options when configuring Step Functions synchronous integration with Rest API\n */\nexport interface StepFunctionsExecutionIntegrationOptions extends IntegrationOptions {\n\n  /**\n   * Which details of the incoming request must be passed onto the underlying state machine,\n   * such as, account id, user identity, request id, etc. The execution input will include a new key `requestContext`:\n   *\n   * {\n   *   \"body\": {},\n   *   \"requestContext\": {\n   *       \"key\": \"value\"\n   *   }\n   * }\n   *\n   * @default - all parameters within request context will be set as false\n   */\n  readonly requestContext?: RequestContext;\n\n  /**\n   * Check if querystring is to be included inside the execution input. The execution input will include a new key `queryString`:\n   *\n   * {\n   *   \"body\": {},\n   *   \"querystring\": {\n   *     \"key\": \"value\"\n   *   }\n   * }\n   *\n   * @default true\n   */\n  readonly querystring?: boolean;\n\n  /**\n   * Check if path is to be included inside the execution input. The execution input will include a new key `path`:\n   *\n   * {\n   *   \"body\": {},\n   *   \"path\": {\n   *     \"resourceName\": \"resourceValue\"\n   *   }\n   * }\n   *\n   * @default true\n   */\n  readonly path?: boolean;\n\n  /**\n   * Check if header is to be included inside the execution input. The execution input will include a new key `headers`:\n   *\n   * {\n   *   \"body\": {},\n   *   \"headers\": {\n   *      \"header1\": \"value\",\n   *      \"header2\": \"value\"\n   *   }\n   * }\n   * @default false\n   */\n  readonly headers?: boolean;\n\n  /**\n   * If the whole authorizer object, including custom context values should be in the execution input. The execution input will include a new key `authorizer`:\n   *\n   * {\n   *   \"body\": {},\n   *   \"authorizer\": {\n   *     \"key\": \"value\"\n   *   }\n   * }\n   *\n   * @default false\n   */\n  readonly authorizer?: boolean;\n}\n\n/**\n * Options to integrate with various StepFunction API\n */\nexport class StepFunctionsIntegration {\n  /**\n   * Integrates a Synchronous Express State Machine from AWS Step Functions to an API Gateway method.\n   *\n   * @example\n   *\n   *    const stateMachine = new stepfunctions.StateMachine(this, 'MyStateMachine', {\n   *       stateMachineType: stepfunctions.StateMachineType.EXPRESS,\n   *       definition: stepfunctions.Chain.start(new stepfunctions.Pass(this, 'Pass')),\n   *    });\n   *\n   *    const api = new apigateway.RestApi(this, 'Api', {\n   *       restApiName: 'MyApi',\n   *    });\n   *    api.root.addMethod('GET', apigateway.StepFunctionsIntegration.startExecution(stateMachine));\n   */\n  public static startExecution(stateMachine: sfn.IStateMachine, options?: StepFunctionsExecutionIntegrationOptions): AwsIntegration {\n    return new StepFunctionsExecutionIntegration(stateMachine, options);\n  }\n}\n\nclass StepFunctionsExecutionIntegration extends AwsIntegration {\n  private readonly stateMachine: sfn.IStateMachine;\n  constructor(stateMachine: sfn.IStateMachine, options: StepFunctionsExecutionIntegrationOptions = {}) {\n    super({\n      service: 'states',\n      action: 'StartSyncExecution',\n      options: {\n        credentialsRole: options.credentialsRole,\n        integrationResponses: integrationResponse(),\n        passthroughBehavior: PassthroughBehavior.NEVER,\n        requestTemplates: requestTemplates(stateMachine, options),\n        ...options,\n      },\n    });\n\n    this.stateMachine = stateMachine;\n  }\n\n  public bind(method: Method): IntegrationConfig {\n    const bindResult = super.bind(method);\n\n    const credentialsRole = bindResult.options?.credentialsRole ?? new iam.Role(method, 'StartSyncExecutionRole', {\n      assumedBy: new iam.ServicePrincipal('apigateway.amazonaws.com'),\n    });\n    this.stateMachine.grantStartSyncExecution(credentialsRole);\n\n    let stateMachineName;\n\n    if (this.stateMachine instanceof sfn.StateMachine) {\n      const stateMachineType = (this.stateMachine as sfn.StateMachine).stateMachineType;\n      if (stateMachineType !== sfn.StateMachineType.EXPRESS) {\n        throw new Error('State Machine must be of type \"EXPRESS\". Please use StateMachineType.EXPRESS as the stateMachineType');\n      }\n\n      //if not imported, extract the name from the CFN layer to reach the\n      //literal value if it is given (rather than a token)\n      stateMachineName = (this.stateMachine.node.defaultChild as sfn.CfnStateMachine).stateMachineName;\n    } else {\n      //imported state machine\n      stateMachineName = `StateMachine-${this.stateMachine.stack.node.addr}`;\n    }\n\n    let deploymentToken;\n\n    if (stateMachineName !== undefined && !Token.isUnresolved(stateMachineName)) {\n      deploymentToken = JSON.stringify({ stateMachineName });\n    }\n\n    for (const methodResponse of METHOD_RESPONSES) {\n      method.addMethodResponse(methodResponse);\n    }\n\n    return {\n      ...bindResult,\n      options: {\n        ...bindResult.options,\n        credentialsRole,\n      },\n      deploymentToken,\n    };\n  }\n}\n\n/**\n * Defines the integration response that passes the result on success,\n * or the error on failure, from the synchronous execution to the caller.\n *\n * @returns integrationResponse mapping\n */\nfunction integrationResponse() {\n  const errorResponse = [\n    {\n      /**\n       * Specifies the regular expression (regex) pattern used to choose\n       * an integration response based on the response from the back end.\n       * In this case it will match all '4XX' HTTP Errors\n       */\n      selectionPattern: '4\\\\d{2}',\n      statusCode: '400',\n      responseTemplates: {\n        'application/json': `{\n            \"error\": \"Bad request!\"\n          }`,\n      },\n    },\n    {\n      /**\n       * Match all '5XX' HTTP Errors\n       */\n      selectionPattern: '5\\\\d{2}',\n      statusCode: '500',\n      responseTemplates: {\n        'application/json': '\"error\": $input.path(\\'$.error\\')',\n      },\n    },\n  ];\n\n  const integResponse = [\n    {\n      statusCode: '200',\n      responseTemplates: {\n        /* eslint-disable */\n        'application/json': [\n          '#set($inputRoot = $input.path(\\'$\\'))',\n          '#if($input.path(\\'$.status\\').toString().equals(\"FAILED\"))',\n            '#set($context.responseOverride.status = 500)',\n            '{',\n              '\"error\": \"$input.path(\\'$.error\\')\",',\n              '\"cause\": \"$input.path(\\'$.cause\\')\"',\n            '}',\n          '#else',\n            '$input.path(\\'$.output\\')',\n          '#end',\n        /* eslint-enable */\n        ].join('\\n'),\n      },\n    },\n    ...errorResponse,\n  ];\n\n  return integResponse;\n}\n\n/**\n * Defines the request template that will be used for the integration\n * @param stateMachine\n * @param options\n * @returns requestTemplate\n */\nfunction requestTemplates(stateMachine: sfn.IStateMachine, options: StepFunctionsExecutionIntegrationOptions) {\n  const templateStr = templateString(stateMachine, options);\n\n  const requestTemplate: { [contentType:string] : string } =\n    {\n      'application/json': templateStr,\n    };\n\n  return requestTemplate;\n}\n\n/**\n * Reads the VTL template and returns the template string to be used\n * for the request template.\n *\n * @param stateMachine\n * @param includeRequestContext\n * @param options\n * @reutrns templateString\n */\nfunction templateString(\n  stateMachine: sfn.IStateMachine,\n  options: StepFunctionsExecutionIntegrationOptions): string {\n  let templateStr: string;\n\n  let requestContextStr = '';\n\n  const includeHeader = options.headers?? false;\n  const includeQueryString = options.querystring?? true;\n  const includePath = options.path?? true;\n  const includeAuthorizer = options.authorizer ?? false;\n\n  if (options.requestContext && Object.keys(options.requestContext).length > 0) {\n    requestContextStr = requestContext(options.requestContext);\n  }\n\n  templateStr = fs.readFileSync(path.join(__dirname, 'stepfunctions.vtl'), { encoding: 'utf-8' });\n  templateStr = templateStr.replace('%STATEMACHINE%', stateMachine.stateMachineArn);\n  templateStr = templateStr.replace('%INCLUDE_HEADERS%', String(includeHeader));\n  templateStr = templateStr.replace('%INCLUDE_QUERYSTRING%', String(includeQueryString));\n  templateStr = templateStr.replace('%INCLUDE_PATH%', String(includePath));\n  templateStr = templateStr.replace('%INCLUDE_AUTHORIZER%', String(includeAuthorizer));\n  templateStr = templateStr.replace('%REQUESTCONTEXT%', requestContextStr);\n\n  return templateStr;\n}\n\nfunction requestContext(requestContextObj: RequestContext | undefined): string {\n  const context = {\n    accountId: requestContextObj?.accountId? '$context.identity.accountId': undefined,\n    apiId: requestContextObj?.apiId? '$context.apiId': undefined,\n    apiKey: requestContextObj?.apiKey? '$context.identity.apiKey': undefined,\n    authorizerPrincipalId: requestContextObj?.authorizerPrincipalId? '$context.authorizer.principalId': undefined,\n    caller: requestContextObj?.caller? '$context.identity.caller': undefined,\n    cognitoAuthenticationProvider: requestContextObj?.cognitoAuthenticationProvider? '$context.identity.cognitoAuthenticationProvider': undefined,\n    cognitoAuthenticationType: requestContextObj?.cognitoAuthenticationType? '$context.identity.cognitoAuthenticationType': undefined,\n    cognitoIdentityId: requestContextObj?.cognitoIdentityId? '$context.identity.cognitoIdentityId': undefined,\n    cognitoIdentityPoolId: requestContextObj?.cognitoIdentityPoolId? '$context.identity.cognitoIdentityPoolId': undefined,\n    httpMethod: requestContextObj?.httpMethod? '$context.httpMethod': undefined,\n    stage: requestContextObj?.stage? '$context.stage': undefined,\n    sourceIp: requestContextObj?.sourceIp? '$context.identity.sourceIp': undefined,\n    user: requestContextObj?.user? '$context.identity.user': undefined,\n    userAgent: requestContextObj?.userAgent? '$context.identity.userAgent': undefined,\n    userArn: requestContextObj?.userArn? '$context.identity.userArn': undefined,\n    requestId: requestContextObj?.requestId? '$context.requestId': undefined,\n    resourceId: requestContextObj?.resourceId? '$context.resourceId': undefined,\n    resourcePath: requestContextObj?.resourcePath? '$context.resourcePath': undefined,\n  };\n\n  const contextAsString = JSON.stringify(context);\n\n  // The VTL Template conflicts with double-quotes (\") for strings.\n  // Before sending to the template, we replace double-quotes (\") with @@ and replace it back inside the .vtl file\n  const doublequotes = '\"';\n  const replaceWith = '@@';\n  return contextAsString.split(doublequotes).join(replaceWith);\n}\n\n/**\n * Method response model for each HTTP code response\n */\nconst METHOD_RESPONSES = [\n  {\n    statusCode: '200',\n    responseModels: {\n      'application/json': Model.EMPTY_MODEL,\n    },\n  },\n  {\n    statusCode: '400',\n    responseModels: {\n      'application/json': Model.ERROR_MODEL,\n    },\n  },\n  {\n    statusCode: '500',\n    responseModels: {\n      'application/json': Model.ERROR_MODEL,\n    },\n  },\n];\n"]} |
\ | No newline at end of file |