@alliander-opensource/aws-jwt-sts 0.2.7 → 0.2.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,4 +1,5 @@
1
1
  # Use IAM roles to authenticate principals in workloads outside of AWS using JWT
2
+
2
3
  There is an inherent risk in maintaining and storing permanent credentials. In their lifetime, they are bound to be shared, compromised, and lost. When shared, it is often among a broader audience than initially intended. They can also be lost and found, sometimes by the wrong person. And when any of this occurs, it can put your systems, data or even organization at risk.
3
4
 
4
5
  Workloads running on AWS can communicate with each other or with AWS services without the need of storing permanent credentials by assuming roles or instance profiles. However, if one of the workloads lives outside of AWS, AWS principals can no longer be used for authentication.
@@ -6,6 +7,7 @@ Workloads running on AWS can communicate with each other or with AWS services wi
6
7
  An alternative to authenticating with external workloads is to use short-lived credentials issued by a trusted party, the issuer, that the target system can accept. JWTs (JSON Web Tokens), as used by the OIDC (OpenID Connect) standard, are an example of such credentials. JWTs are short-lived credentials that can be signed and verified using a public key in what is known as public-key cryptography.
7
8
 
8
9
  ## Secure Token Service (STS)
10
+
9
11
  Exchanging credentials from on form to the other is done with a Secure Token Service (STS) function. AWS also provides STS functions not the one we need. Only the other way around: to exchange a JWT to IAM Session which is called [AssumeRoleWithWebIdentity](https://docs.aws.amazon.com/STS/latest/APIReference/API_AssumeRoleWithWebIdentity.html).
10
12
  This repo contains a CDK Construct which will deploy a new function which adds the function to exchange an AWS IAM (Session) credential with a signed JWT.
11
13
 
@@ -21,13 +23,13 @@ On time based events, EventBridge will trigger a Step function rotation flow. Th
21
23
 
22
24
  ## Using the construct
23
25
 
24
- 1. Init a new typescript CDK project
26
+ 1. Init a new typescript CDK project
25
27
  `cdk init app --language typescript`
26
- 2. Config npm to retrieve packages from github package repository
28
+ 2. Config npm to retrieve packages from github package repository
27
29
  `echo @alliander-opensource:registry=https://npm.pkg.github.com > .npmrc`
28
- 3. Install the aws-jwt-sts construct
30
+ 3. Install the aws-jwt-sts construct
29
31
  `npm install @alliander-opensource/aws-jwt-sts`
30
- 4. Edit lib/my-sts-stack.ts to add the construct to the stack
32
+ 4. Edit lib/my-sts-stack.ts to add the construct to the stack
31
33
  See the comments in the code for possible options
32
34
 
33
35
  ```ts
@@ -102,6 +104,7 @@ export class MyStsStack extends cdk.Stack {
102
104
  The stack outputs the urls of the endpoints. So if no custom domain is provided observe the CDK Stack output.
103
105
 
104
106
  ## Using the STS function
107
+
105
108
  A token from the STS function can be obained by invoking the token endpoint.
106
109
  `GET https://$host/token`
107
110
  optionally an audience can be provided if this needs to be different than the installed default
@@ -110,6 +113,7 @@ optionally an audience can be provided if this needs to be different than the in
110
113
  > Note: The IAM Role / User invoking the endpoint must have *execute-api:Invoke* permissions
111
114
 
112
115
  In CDK these permission is added as followed:
116
+
113
117
  ```ts
114
118
  role.addToPolicy(new iam.PolicyStatement({
115
119
  actions: ['execute-api:Invoke'],
@@ -120,11 +124,13 @@ role.addToPolicy(new iam.PolicyStatement({
120
124
  > Note: keep in mind that *resource '\*'* should only be used if no other API GW's with IAM auth are used in that account.
121
125
 
122
126
  ### Test obtaining a JWT
123
- 1. Ensure the AWS IAM Role / User invoking the token endpoint has execute-api permissions. If you are using administrator access then that is more than sufficient.
124
- 2. Use a shell with AWS cli logged in, you can use your cli with which you deployed the stack or use cloudshell for this.
125
- 3. Install awscurl for authentication
127
+
128
+ 1. Ensure the AWS IAM Role / User invoking the token endpoint has execute-api permissions. If you are using administrator access then that is more than sufficient.
129
+ 2. Use a shell with AWS cli logged in, you can use your cli with which you deployed the stack or use cloudshell for this.
130
+ 3. Install awscurl for authentication
126
131
  `pip3 install awscurl`
127
- 4. Install jwt-cli for jwt formatting
132
+ 4. Install jwt-cli for jwt formatting
128
133
  `npm install -g jwt-cli`
129
- 5. Invoke the api: `awscurl {your token endpoint}/token --service execute-api --region {your_region} | jq -r .token | jwt decode – `
130
- 6. Observe the JWT
134
+ 5. Invoke the api
135
+ `awscurl {your_token_endpoint}/token --service execute-api --region {your_region} | jq -r .token | jwt decode –`
136
+ 6. Observe the JWT
package/dist/index.js CHANGED
@@ -37,6 +37,7 @@ var wafUsage;
37
37
  /* eslint-disable no-new */
38
38
  class AwsJwtSts extends constructs_1.Construct {
39
39
  constructor(app, id, props) {
40
+ var _a, _b, _c, _d;
40
41
  super(app, id);
41
42
  /** ---------------------- Custom domain thingies ----------------------- */
42
43
  let distributionDomainNames = [];
@@ -184,7 +185,7 @@ class AwsJwtSts extends constructs_1.Construct {
184
185
  snsFail.next(jobFailed);
185
186
  // Create state machine
186
187
  const rotateKeysMachine = new sfn.StateMachine(this, 'RotateKeys', {
187
- definition,
188
+ definitionBody: sfn.DefinitionBody.fromChainable(definition),
188
189
  timeout: cdk.Duration.minutes(5)
189
190
  });
190
191
  rotateKeys.grantInvoke(rotateKeysMachine.role);
@@ -220,7 +221,7 @@ class AwsJwtSts extends constructs_1.Construct {
220
221
  integrationPattern: sfn.IntegrationPattern.RUN_JOB
221
222
  });
222
223
  const populateKeys = new sfn.StateMachine(this, 'populateKeys', {
223
- definition: rotateOnce.next(rotateTwice),
224
+ definitionBody: sfn.DefinitionBody.fromChainable(rotateOnce.next(rotateTwice)),
224
225
  timeout: cdk.Duration.minutes(10)
225
226
  });
226
227
  const initialRunRule = new events.Rule(this, 'initialRunRule', {
@@ -292,20 +293,20 @@ class AwsJwtSts extends constructs_1.Construct {
292
293
  new cdk.CfnOutput(this, 'tokenEndpoint', {
293
294
  value: 'https://' + tokenDomainName + '/token',
294
295
  description: 'Url of the token endpoint',
295
- exportName: 'tokenEndpoint'
296
+ exportName: `${cdk.Stack.of(this)}-tokenEndpoint`
296
297
  });
297
298
  }
298
299
  else {
299
300
  new cdk.CfnOutput(this, 'tokenEndpoint', {
300
301
  value: api.url + 'token',
301
302
  description: 'Url of the token endpoint',
302
- exportName: 'tokenEndpoint'
303
+ exportName: `${cdk.Stack.of(this)}-tokenEndpoint`
303
304
  });
304
305
  }
305
306
  new cdk.CfnOutput(this, 'issuer', {
306
307
  value: issuer,
307
308
  description: 'Url of the issuer',
308
- exportName: 'issuer'
309
+ exportName: `${cdk.Stack.of(this)}-issuer`
309
310
  });
310
311
  /** ---------------------- WAF ----------------------- */
311
312
  if (props.apiGwWaf === wafUsage.ConstructProvided) {
@@ -404,7 +405,7 @@ class AwsJwtSts extends constructs_1.Construct {
404
405
  }
405
406
  /** ---------------------- Cloudwatch ----------------------- */
406
407
  new cloudwatch.Alarm(this, 'StepFunctionError', {
407
- alarmName: props.alarmNameKeyRotationStepFunctionFailed ?? 'sts-key_rotate_sfn-alarm',
408
+ alarmName: (_a = props.alarmNameKeyRotationStepFunctionFailed) !== null && _a !== void 0 ? _a : 'sts-key_rotate_sfn-alarm',
408
409
  comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,
409
410
  threshold: 1,
410
411
  evaluationPeriods: 1,
@@ -413,7 +414,7 @@ class AwsJwtSts extends constructs_1.Construct {
413
414
  treatMissingData: aws_cloudwatch_1.TreatMissingData.NOT_BREACHING
414
415
  });
415
416
  new cloudwatch.Alarm(this, 'ApiGateway5XXAlarm', {
416
- alarmName: props.alarmNameApiGateway5xx ?? 'sts-5xx_api_gw-alarm',
417
+ alarmName: (_b = props.alarmNameApiGateway5xx) !== null && _b !== void 0 ? _b : 'sts-5xx_api_gw-alarm',
417
418
  comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,
418
419
  threshold: 1,
419
420
  evaluationPeriods: 1,
@@ -428,7 +429,7 @@ class AwsJwtSts extends constructs_1.Construct {
428
429
  period: cdk.Duration.minutes(1)
429
430
  });
430
431
  new cloudwatch.Alarm(this, 'LambdaSignError', {
431
- alarmName: props.alarmNameSignLambdaFailed ?? 'sts-sign_errors_lambda-alarm',
432
+ alarmName: (_c = props.alarmNameSignLambdaFailed) !== null && _c !== void 0 ? _c : 'sts-sign_errors_lambda-alarm',
432
433
  comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,
433
434
  threshold: 1,
434
435
  evaluationPeriods: 1,
@@ -437,7 +438,7 @@ class AwsJwtSts extends constructs_1.Construct {
437
438
  treatMissingData: aws_cloudwatch_1.TreatMissingData.NOT_BREACHING
438
439
  });
439
440
  new cloudwatch.Alarm(this, 'LambdaRotateError', {
440
- alarmName: props.alarmNameKeyRotationLambdaFailed ?? 'sts-key_rotate_errors_lambda-alarm',
441
+ alarmName: (_d = props.alarmNameKeyRotationLambdaFailed) !== null && _d !== void 0 ? _d : 'sts-key_rotate_errors_lambda-alarm',
441
442
  comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,
442
443
  threshold: 1,
443
444
  evaluationPeriods: 1,
@@ -448,4 +449,4 @@ class AwsJwtSts extends constructs_1.Construct {
448
449
  }
449
450
  }
450
451
  exports.AwsJwtSts = AwsJwtSts;
451
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["// SPDX-FileCopyrightText: 2023 Alliander NV\n//\n// SPDX-License-Identifier: Apache-2.0\n\n/* eslint-disable no-unused-vars */\nimport * as cdk from 'aws-cdk-lib'\nimport * as lambda from 'aws-cdk-lib/aws-lambda'\nimport * as sfn from 'aws-cdk-lib/aws-stepfunctions'\nimport * as tasks from 'aws-cdk-lib/aws-stepfunctions-tasks'\nimport * as iam from 'aws-cdk-lib/aws-iam'\nimport { OrganizationPrincipal, PolicyDocument } from 'aws-cdk-lib/aws-iam'\nimport * as s3 from 'aws-cdk-lib/aws-s3'\nimport { BucketEncryption } from 'aws-cdk-lib/aws-s3'\nimport * as cloudfront from 'aws-cdk-lib/aws-cloudfront'\nimport * as cloudfrontOrigins from 'aws-cdk-lib/aws-cloudfront-origins'\nimport * as events from 'aws-cdk-lib/aws-events'\nimport * as targets from 'aws-cdk-lib/aws-events-targets'\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager'\nimport * as route53 from 'aws-cdk-lib/aws-route53'\nimport * as route53targets from 'aws-cdk-lib/aws-route53-targets'\nimport * as apigateway from 'aws-cdk-lib/aws-apigateway'\nimport { MethodLoggingLevel } from 'aws-cdk-lib/aws-apigateway'\nimport * as wafv2 from 'aws-cdk-lib/aws-wafv2'\nimport * as sns from 'aws-cdk-lib/aws-sns'\nimport * as logs from 'aws-cdk-lib/aws-logs'\nimport * as cloudwatch from 'aws-cdk-lib/aws-cloudwatch'\nimport { TreatMissingData } from 'aws-cdk-lib/aws-cloudwatch'\nimport * as lambdaNodejs from 'aws-cdk-lib/aws-lambda-nodejs'\nimport { Construct } from 'constructs'\nimport { ICertificate } from 'aws-cdk-lib/aws-certificatemanager'\nimport { IHostedZone } from 'aws-cdk-lib/aws-route53'\n\nexport enum wafUsage {\n  ConstructProvided,\n  ProvideWebAclArn\n}\n\nexport interface AwsJwtStsProps {\n  /**\n   * defaultAudience which is used in de JWT's\n   */\n  readonly defaultAudience: string;\n\n  /**\n   * HostedZoneId of the domain used for hosting the sts function\n   */\n  readonly hostedZoneId?: string;\n\n  /**\n   * Name of the hostedZone.\n   */\n  readonly hostedZoneName?: string;\n\n  /**\n  * Optional subdomain name of oidc discovery, default: oidc.\n  */\n  readonly oidcSubdomain?: string;\n\n  /**\n  * Optional subdomain name of the token api (on api gw), default: token.\n  */\n  readonly tokenSubdomain?: string;\n\n  /**\n   * If waf needs to be added to the API GW\n   *\n   * None: no waf is used\n   * ConstructProvided: the construct will deploy a wafAcl with opinionated rules\n   * ProvideWebAclArn: provide your own arn\n   */\n  readonly apiGwWaf?: wafUsage;\n\n  /**\n   * Arn of the waf webAcl rule to be associated with the API GW\n   *\n   */\n  readonly apiGwWafWebAclArn?: string;\n\n  /**\n   * The ID of the AWS Organization 0-xxxx\n   *\n   */\n  readonly orgId?: string;\n\n  /**\n   * CPU Architecture\n   */\n  readonly architecture?: lambda.Architecture\n\n  /**\n   * Optional boolean to specify if key rotation should be triggered on creation of the stack, default: false\n   */\n  readonly disableKeyRotateOnCreate?: boolean\n\n  /**\n   * Optional custom name for the CloudWatch Alarm monitoring Step Function failures, default: sts-key_rotate_sfn-alarm\n   */\n  readonly alarmNameKeyRotationStepFunctionFailed?: string\n\n  /**\n   * Optional custom name for the CloudWatch Alarm monitoring 5xx errors on the API Gateway, default: sts-5xx_api_gw-alarm\n   */\n  readonly alarmNameApiGateway5xx?: string\n\n  /**\n   * Optional custom name for the CloudWatch Alarm monitoring Sign Lambda failures, default: sts-sign_errors_lambda-alarm\n   */\n  readonly alarmNameSignLambdaFailed?: string\n\n  /**\n   * Optional custom name for the CloudWatch Alarm monitoring Key Rotation Lambda failures, default: sts-key_rotate_errors_lambda-alarm\n   */\n  readonly alarmNameKeyRotationLambdaFailed?: string\n}\n\n/* eslint-disable no-new */\nexport class AwsJwtSts extends Construct {\n  /**\n   * SNS topic used to publish errors from the Step Function rotation flow\n   */\n  public readonly failedRotationTopic: sns.Topic\n\n  constructor (app: Construct, id: string, props: AwsJwtStsProps) {\n    super(app, id)\n\n    /** ---------------------- Custom domain thingies ----------------------- */\n\n    let distributionDomainNames: string[] = []\n    let oidcCertificate: ICertificate | undefined\n    let tokenCertificate: ICertificate | undefined\n    let hostedZone: IHostedZone | undefined\n    const oidcSubdomain = props.oidcSubdomain ? props.oidcSubdomain : 'oidc'\n    const tokenSubdomain = props.tokenSubdomain ? props.tokenSubdomain : 'token'\n    const architecture = props.architecture ? props.architecture : lambda.Architecture.X86_64\n    let oidcDomainName = ''\n    let tokenDomainName = ''\n\n    const useCustomDomain = props.hostedZoneId && props.hostedZoneName\n\n    if (useCustomDomain) {\n      oidcDomainName = oidcSubdomain + '.' + props.hostedZoneName\n      tokenDomainName = tokenSubdomain + '.' + props.hostedZoneName\n\n      distributionDomainNames = [oidcDomainName]\n\n      hostedZone = route53.HostedZone.fromHostedZoneAttributes(\n        this,\n        'hostedZone',\n        {\n          zoneName: props.hostedZoneName!,\n          hostedZoneId: props.hostedZoneId!\n        }\n      )\n\n      oidcCertificate = new acm.DnsValidatedCertificate(this, 'CrossRegionCertificate', {\n        domainName: oidcDomainName,\n        hostedZone,\n        region: 'us-east-1'\n      })\n\n      tokenCertificate = new acm.Certificate(this, 'tokenCertificate', {\n        domainName: tokenDomainName,\n        validation: acm.CertificateValidation.fromDns(hostedZone)\n      })\n    }\n\n    /** ---------------------- S3 Definition ----------------------- */\n\n    // Create bucket where oidc information can be stored\n    const oidcbucket = new s3.Bucket(this, 'oidcbucket', {\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n      autoDeleteObjects: true,\n      encryption: BucketEncryption.S3_MANAGED,\n      versioned: true,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL\n    })\n\n    /** ------------------- Cloudfront Definition ------------------- */\n\n    const cloudfrontOAI = new cloudfront.OriginAccessIdentity(this, 'cloudfront-OAI', {\n      comment: 'OAI for oidc'\n    })\n\n    const distribution = new cloudfront.Distribution(this, 'oidcDistribution', {\n      domainNames: distributionDomainNames,\n      comment: 'Discovery endpoint for OIDC',\n      certificate: oidcCertificate,\n      defaultBehavior: {\n        origin: new cloudfrontOrigins.S3Origin(oidcbucket, { originAccessIdentity: cloudfrontOAI }),\n        compress: true,\n        allowedMethods: cloudfront.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n        viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS\n      }\n    })\n\n    /** ------------------ Lambda Handlers Definition ------------------ */\n\n    const issuer = useCustomDomain ? 'https://' + oidcDomainName : 'https://' + distribution.distributionDomainName\n\n    const rotateKeysRole = new iam.Role(this, 'rotateKeysRole', {\n      assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),\n      managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole')]\n    })\n    const rotateKeys = new lambdaNodejs.NodejsFunction(this, 'keyrotate', {\n      timeout: cdk.Duration.seconds(5),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      role: rotateKeysRole,\n      architecture,\n      environment: {\n        S3_BUCKET: oidcbucket.bucketName,\n        ISSUER: issuer\n      }\n    })\n\n    const signRole = new iam.Role(this, 'signRole', {\n      assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),\n      managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole')]\n    })\n    const sign = new lambdaNodejs.NodejsFunction(this, 'sign', {\n      timeout: cdk.Duration.seconds(5),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      role: signRole,\n      architecture,\n      environment: {\n        ISSUER: issuer,\n        DEFAULT_AUDIENCE: props.defaultAudience\n      }\n    })\n\n    /** ------------------------ SNS Topic ------------------------- */\n\n    this.failedRotationTopic = new sns.Topic(this, 'sts')\n    const snsFail = new tasks.SnsPublish(this, 'snsFailed', {\n      topic: this.failedRotationTopic,\n      subject: 'STS KeyRotate step function execution failed',\n      message: sfn.TaskInput.fromJsonPathAt('$')\n    })\n\n    /** ------------------ Step functions Definition ------------------ */\n\n    const deletePreviousStep = new tasks.LambdaInvoke(this, 'delete Previous', {\n      lambdaFunction: rotateKeys,\n      payload: sfn.TaskInput.fromObject({\n        step: 'deletePrevious'\n      }),\n      outputPath: '$.Payload'\n    })\n\n    const movePreviousStep = new tasks.LambdaInvoke(this, 'move Previous', {\n      lambdaFunction: rotateKeys,\n      payload: sfn.TaskInput.fromObject({\n        step: 'movePrevious'\n      }),\n      outputPath: '$.Payload'\n    })\n\n    const moveCurrentStep = new tasks.LambdaInvoke(this, 'move Current', {\n      lambdaFunction: rotateKeys,\n      payload: sfn.TaskInput.fromObject({\n        step: 'moveCurrent'\n      }),\n      outputPath: '$.Payload'\n    })\n\n    const createPendingStep = new tasks.LambdaInvoke(this, 'create Pending', {\n      lambdaFunction: rotateKeys,\n      payload: sfn.TaskInput.fromObject({\n        step: 'createPending'\n      }),\n      outputPath: '$.Payload'\n    })\n\n    const generateArtifactsStep = new tasks.LambdaInvoke(this, 'generate artifacts', {\n      lambdaFunction: rotateKeys,\n      payload: sfn.TaskInput.fromObject({\n        step: 'generateArtifacts'\n      }),\n      outputPath: '$.Payload'\n    })\n\n    const jobFailed = new sfn.Fail(this, 'Failed', {\n      cause: 'AWS Batch Job Failed',\n      error: 'DescribeJob returned FAILED'\n    })\n\n    const jobSuccess = new sfn.Succeed(this, 'Success!')\n\n    deletePreviousStep.addCatch(snsFail)\n    movePreviousStep.addCatch(snsFail)\n    moveCurrentStep.addCatch(snsFail)\n    createPendingStep.addCatch(snsFail)\n    generateArtifactsStep.addCatch(snsFail)\n\n    // Create chain\n    const definition = deletePreviousStep\n      .next(movePreviousStep)\n      .next(moveCurrentStep)\n      .next(createPendingStep)\n      .next(generateArtifactsStep)\n      .next(jobSuccess)\n\n    snsFail.next(jobFailed)\n\n    // Create state machine\n    const rotateKeysMachine = new sfn.StateMachine(this, 'RotateKeys', {\n      definition,\n      timeout: cdk.Duration.minutes(5)\n    })\n\n    rotateKeys.grantInvoke(rotateKeysMachine.role)\n    oidcbucket.grantReadWrite(rotateKeys)\n\n    const statementSign = new iam.PolicyStatement()\n    statementSign.addActions('kms:*')\n    statementSign.addResources('*')\n    const signPolicy = new iam.ManagedPolicy(this, 'SignPolicy', {\n      statements: [statementSign]\n    })\n    signRole.addManagedPolicy(signPolicy)\n\n    const statementRotateKeys = new iam.PolicyStatement()\n    statementRotateKeys.addActions('kms:*')\n    statementRotateKeys.addResources('*')\n    const rotateKeysPolicy = new iam.ManagedPolicy(this, 'RotateKeysPolicy', {\n      statements: [statementRotateKeys]\n    })\n    rotateKeysRole.addManagedPolicy(rotateKeysPolicy)\n\n    /** ------------------ Events Rule Definition ------------------ */\n\n    // Run every 3 months at 8 PM UTC\n    const scheduledRotateRule = new events.Rule(this, 'scheduledRotateRule', {\n      schedule: events.Schedule.expression('cron(0 20 1 */3 ? *)')\n    })\n    scheduledRotateRule.addTarget(new targets.SfnStateMachine(rotateKeysMachine))\n\n    // Create state machine and trigger to populate initial keys\n    if (!props.disableKeyRotateOnCreate) {\n      const rotateOnce = new tasks.StepFunctionsStartExecution(this, 'rotateOnce', {\n        stateMachine: rotateKeysMachine,\n        integrationPattern: sfn.IntegrationPattern.RUN_JOB\n      })\n\n      const rotateTwice = new tasks.StepFunctionsStartExecution(this, 'rotateTwice', {\n        stateMachine: rotateKeysMachine,\n        integrationPattern: sfn.IntegrationPattern.RUN_JOB\n      })\n\n      const populateKeys = new sfn.StateMachine(this, 'populateKeys', {\n        definition: rotateOnce.next(rotateTwice),\n        timeout: cdk.Duration.minutes(10)\n      })\n\n      const initialRunRule = new events.Rule(this, 'initialRunRule', {\n        eventPattern: {\n          source: ['aws.cloudformation'],\n          resources: [cdk.Stack.of(this).stackId],\n          detailType: ['CloudFormation Stack Status Change'],\n          detail: {\n            'status-details': {\n              status: ['CREATE_COMPLETE']\n            }\n          }\n        }\n      })\n\n      initialRunRule.addTarget(new targets.SfnStateMachine(populateKeys))\n    }\n\n    /** ---------------------- API Gateway ----------------------- */\n\n    // only set policy when orgId is set\n    let apiPolicy: PolicyDocument | undefined\n    if (props.orgId) {\n      apiPolicy = new iam.PolicyDocument({\n        statements: [\n          new iam.PolicyStatement({\n            actions: ['execute-api:Invoke'],\n            resources: ['*'],\n            principals: [\n              new OrganizationPrincipal(props.orgId)\n            ]\n          })\n        ]\n      })\n    }\n\n    const logGroup = new logs.LogGroup(this, 'APIGatewayAccessLogs', {\n      retention: 7\n    })\n\n    // Create API\n    const api = new apigateway.LambdaRestApi(this, 'jwk-sts-api', {\n      description: 'STS Token API Gateway',\n      handler: sign,\n      defaultMethodOptions: {\n        authorizationType: apigateway.AuthorizationType.IAM\n      },\n      endpointConfiguration: {\n        types: [apigateway.EndpointType.REGIONAL]\n      },\n      policy: apiPolicy,\n      deployOptions: {\n        loggingLevel: MethodLoggingLevel.INFO,\n        accessLogDestination: new apigateway.LogGroupLogDestination(logGroup)\n      }\n    })\n\n    /** ------------------- Route53 Definition for custom domain ------------------- */\n\n    if (useCustomDomain && hostedZone) {\n      api.addDomainName('apiCustomDomainName', {\n        domainName: tokenDomainName,\n        certificate: tokenCertificate!\n      })\n\n      // Add A record for cloudfront distribution\n\n      new route53.ARecord(this, 'oidcRecord', {\n        recordName: oidcDomainName,\n        zone: hostedZone,\n        target: route53.RecordTarget.fromAlias(new route53targets.CloudFrontTarget(distribution))\n      })\n\n      new route53.ARecord(this, 'tokenRecord', {\n        recordName: tokenDomainName,\n        zone: hostedZone,\n        target: route53.RecordTarget.fromAlias(new route53targets.ApiGateway(api))\n      })\n\n      new cdk.CfnOutput(this, 'tokenEndpoint', {\n        value: 'https://' + tokenDomainName + '/token',\n        description: 'Url of the token endpoint',\n        exportName: 'tokenEndpoint'\n      })\n    } else {\n      new cdk.CfnOutput(this, 'tokenEndpoint', {\n        value: api.url + 'token',\n        description: 'Url of the token endpoint',\n        exportName: 'tokenEndpoint'\n      })\n    }\n\n    new cdk.CfnOutput(this, 'issuer', {\n      value: issuer,\n      description: 'Url of the issuer',\n      exportName: 'issuer'\n    })\n\n    /** ---------------------- WAF ----------------------- */\n\n    if (props.apiGwWaf === wafUsage.ConstructProvided) {\n      // API gateway WAF ACL and rules\n      const APIGatewayWebACL = new wafv2.CfnWebACL(this, 'APIGatewayWebACL', {\n        description: 'This is WebACL for Auth APi Gateway',\n        scope: 'REGIONAL',\n        defaultAction: { allow: {} },\n        visibilityConfig: {\n          metricName: 'APIWebACL',\n          cloudWatchMetricsEnabled: true,\n          sampledRequestsEnabled: true\n        },\n        rules: [\n          {\n            name: 'AWS-AWSManagedRulesCommonRuleSet',\n            priority: 0,\n            statement: {\n              managedRuleGroupStatement: {\n                vendorName: 'AWS',\n                name: 'AWSManagedRulesCommonRuleSet'\n              }\n            },\n            overrideAction: {\n              none: {}\n            },\n            visibilityConfig: {\n              sampledRequestsEnabled: true,\n              cloudWatchMetricsEnabled: true,\n              metricName: 'AWS-AWSManagedRulesCommonRuleSet'\n            }\n          },\n          {\n            name: 'AWS-AWSManagedRulesAmazonIpReputationList',\n            priority: 1,\n            statement: {\n              managedRuleGroupStatement: {\n                vendorName: 'AWS',\n                name: 'AWSManagedRulesAmazonIpReputationList'\n              }\n            },\n            overrideAction: {\n              none: {}\n            },\n            visibilityConfig: {\n              sampledRequestsEnabled: true,\n              cloudWatchMetricsEnabled: true,\n              metricName: 'AWS-AWSManagedRulesAmazonIpReputationList'\n            }\n          },\n          {\n            name: 'api-gw-AuthAPIGeoLocation',\n            priority: 3,\n            action: { block: {} },\n            visibilityConfig: {\n              metricName: 'AuthAPIGeoLocation',\n              cloudWatchMetricsEnabled: true,\n              sampledRequestsEnabled: false\n            },\n            statement: {\n              geoMatchStatement: {\n                countryCodes: ['BY', 'CN', 'IR', 'RU', 'SY', 'KP']\n              }\n            }\n          },\n          {\n            name: 'api-gw-rateLimitRule',\n            priority: 4,\n            action: { block: {} },\n            visibilityConfig: {\n              metricName: 'rateLimitRule',\n              cloudWatchMetricsEnabled: true,\n              sampledRequestsEnabled: false\n            },\n            statement: {\n              rateBasedStatement: {\n                aggregateKeyType: 'IP',\n                limit: 100\n              }\n            }\n          }\n        ]\n      })\n\n      // Web ACL Association\n      new wafv2.CfnWebACLAssociation(this, 'APIGatewayWebACLAssociation', {\n        webAclArn: APIGatewayWebACL.attrArn,\n        resourceArn: api.deploymentStage.stageArn\n      })\n    } else if (props.apiGwWaf === wafUsage.ProvideWebAclArn && props.apiGwWafWebAclArn) {\n      // Web ACL Association\n      new wafv2.CfnWebACLAssociation(this, 'APIGatewayWebACLAssociation', {\n        webAclArn: props.apiGwWafWebAclArn,\n        resourceArn: api.deploymentStage.stageArn\n      })\n    }\n\n    /** ---------------------- Cloudwatch ----------------------- */\n\n    new cloudwatch.Alarm(this, 'StepFunctionError', {\n      alarmName: props.alarmNameKeyRotationStepFunctionFailed ?? 'sts-key_rotate_sfn-alarm',\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: 1,\n      evaluationPeriods: 1,\n      metric: rotateKeysMachine.metricFailed(),\n      alarmDescription: 'Key Rotation Failed',\n      treatMissingData: TreatMissingData.NOT_BREACHING\n    })\n\n    new cloudwatch.Alarm(this, 'ApiGateway5XXAlarm', {\n      alarmName: props.alarmNameApiGateway5xx ?? 'sts-5xx_api_gw-alarm',\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: 1,\n      evaluationPeriods: 1,\n      metric: api.metricServerError(),\n      alarmDescription: '5xx STS API gateway failures',\n      treatMissingData: TreatMissingData.NOT_BREACHING\n    })\n\n    const signErrors = sign.metricErrors({\n      period: cdk.Duration.minutes(1)\n    })\n\n    const rotateErrors = rotateKeys.metricErrors({\n      period: cdk.Duration.minutes(1)\n    })\n\n    new cloudwatch.Alarm(this, 'LambdaSignError', {\n      alarmName: props.alarmNameSignLambdaFailed ?? 'sts-sign_errors_lambda-alarm',\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: 1,\n      evaluationPeriods: 1,\n      metric: signErrors,\n      alarmDescription: 'Sign Lambda Failed',\n      treatMissingData: TreatMissingData.NOT_BREACHING\n    })\n\n    new cloudwatch.Alarm(this, 'LambdaRotateError', {\n      alarmName: props.alarmNameKeyRotationLambdaFailed ?? 'sts-key_rotate_errors_lambda-alarm',\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: 1,\n      evaluationPeriods: 1,\n      metric: rotateErrors,\n      alarmDescription: 'Key Rotation Lambda Failed',\n      treatMissingData: TreatMissingData.NOT_BREACHING\n    })\n  }\n}\n"]}
452
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["// SPDX-FileCopyrightText: 2023 Alliander NV\n//\n// SPDX-License-Identifier: Apache-2.0\n\n/* eslint-disable no-unused-vars */\nimport * as cdk from 'aws-cdk-lib'\nimport * as lambda from 'aws-cdk-lib/aws-lambda'\nimport * as sfn from 'aws-cdk-lib/aws-stepfunctions'\nimport * as tasks from 'aws-cdk-lib/aws-stepfunctions-tasks'\nimport * as iam from 'aws-cdk-lib/aws-iam'\nimport { OrganizationPrincipal, PolicyDocument } from 'aws-cdk-lib/aws-iam'\nimport * as s3 from 'aws-cdk-lib/aws-s3'\nimport { BucketEncryption } from 'aws-cdk-lib/aws-s3'\nimport * as cloudfront from 'aws-cdk-lib/aws-cloudfront'\nimport * as cloudfrontOrigins from 'aws-cdk-lib/aws-cloudfront-origins'\nimport * as events from 'aws-cdk-lib/aws-events'\nimport * as targets from 'aws-cdk-lib/aws-events-targets'\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager'\nimport * as route53 from 'aws-cdk-lib/aws-route53'\nimport * as route53targets from 'aws-cdk-lib/aws-route53-targets'\nimport * as apigateway from 'aws-cdk-lib/aws-apigateway'\nimport { MethodLoggingLevel } from 'aws-cdk-lib/aws-apigateway'\nimport * as wafv2 from 'aws-cdk-lib/aws-wafv2'\nimport * as sns from 'aws-cdk-lib/aws-sns'\nimport * as logs from 'aws-cdk-lib/aws-logs'\nimport * as cloudwatch from 'aws-cdk-lib/aws-cloudwatch'\nimport { TreatMissingData } from 'aws-cdk-lib/aws-cloudwatch'\nimport * as lambdaNodejs from 'aws-cdk-lib/aws-lambda-nodejs'\nimport { Construct } from 'constructs'\nimport { ICertificate } from 'aws-cdk-lib/aws-certificatemanager'\nimport { IHostedZone } from 'aws-cdk-lib/aws-route53'\n\nexport enum wafUsage {\n  ConstructProvided,\n  ProvideWebAclArn\n}\n\nexport interface AwsJwtStsProps {\n  /**\n   * defaultAudience which is used in de JWT's\n   */\n  readonly defaultAudience: string;\n\n  /**\n   * HostedZoneId of the domain used for hosting the sts function\n   */\n  readonly hostedZoneId?: string;\n\n  /**\n   * Name of the hostedZone.\n   */\n  readonly hostedZoneName?: string;\n\n  /**\n  * Optional subdomain name of oidc discovery, default: oidc.\n  */\n  readonly oidcSubdomain?: string;\n\n  /**\n  * Optional subdomain name of the token api (on api gw), default: token.\n  */\n  readonly tokenSubdomain?: string;\n\n  /**\n   * If waf needs to be added to the API GW\n   *\n   * None: no waf is used\n   * ConstructProvided: the construct will deploy a wafAcl with opinionated rules\n   * ProvideWebAclArn: provide your own arn\n   */\n  readonly apiGwWaf?: wafUsage;\n\n  /**\n   * Arn of the waf webAcl rule to be associated with the API GW\n   *\n   */\n  readonly apiGwWafWebAclArn?: string;\n\n  /**\n   * The ID of the AWS Organization 0-xxxx\n   *\n   */\n  readonly orgId?: string;\n\n  /**\n   * CPU Architecture\n   */\n  readonly architecture?: lambda.Architecture\n\n  /**\n   * Optional boolean to specify if key rotation should be triggered on creation of the stack, default: false\n   */\n  readonly disableKeyRotateOnCreate?: boolean\n\n  /**\n   * Optional custom name for the CloudWatch Alarm monitoring Step Function failures, default: sts-key_rotate_sfn-alarm\n   */\n  readonly alarmNameKeyRotationStepFunctionFailed?: string\n\n  /**\n   * Optional custom name for the CloudWatch Alarm monitoring 5xx errors on the API Gateway, default: sts-5xx_api_gw-alarm\n   */\n  readonly alarmNameApiGateway5xx?: string\n\n  /**\n   * Optional custom name for the CloudWatch Alarm monitoring Sign Lambda failures, default: sts-sign_errors_lambda-alarm\n   */\n  readonly alarmNameSignLambdaFailed?: string\n\n  /**\n   * Optional custom name for the CloudWatch Alarm monitoring Key Rotation Lambda failures, default: sts-key_rotate_errors_lambda-alarm\n   */\n  readonly alarmNameKeyRotationLambdaFailed?: string\n}\n\n/* eslint-disable no-new */\nexport class AwsJwtSts extends Construct {\n  /**\n   * SNS topic used to publish errors from the Step Function rotation flow\n   */\n  public readonly failedRotationTopic: sns.Topic\n\n  constructor (app: Construct, id: string, props: AwsJwtStsProps) {\n    super(app, id)\n\n    /** ---------------------- Custom domain thingies ----------------------- */\n\n    let distributionDomainNames: string[] = []\n    let oidcCertificate: ICertificate | undefined\n    let tokenCertificate: ICertificate | undefined\n    let hostedZone: IHostedZone | undefined\n    const oidcSubdomain = props.oidcSubdomain ? props.oidcSubdomain : 'oidc'\n    const tokenSubdomain = props.tokenSubdomain ? props.tokenSubdomain : 'token'\n    const architecture = props.architecture ? props.architecture : lambda.Architecture.X86_64\n    let oidcDomainName = ''\n    let tokenDomainName = ''\n\n    const useCustomDomain = props.hostedZoneId && props.hostedZoneName\n\n    if (useCustomDomain) {\n      oidcDomainName = oidcSubdomain + '.' + props.hostedZoneName\n      tokenDomainName = tokenSubdomain + '.' + props.hostedZoneName\n\n      distributionDomainNames = [oidcDomainName]\n\n      hostedZone = route53.HostedZone.fromHostedZoneAttributes(\n        this,\n        'hostedZone',\n        {\n          zoneName: props.hostedZoneName!,\n          hostedZoneId: props.hostedZoneId!\n        }\n      )\n\n      oidcCertificate = new acm.DnsValidatedCertificate(this, 'CrossRegionCertificate', {\n        domainName: oidcDomainName,\n        hostedZone,\n        region: 'us-east-1'\n      })\n\n      tokenCertificate = new acm.Certificate(this, 'tokenCertificate', {\n        domainName: tokenDomainName,\n        validation: acm.CertificateValidation.fromDns(hostedZone)\n      })\n    }\n\n    /** ---------------------- S3 Definition ----------------------- */\n\n    // Create bucket where oidc information can be stored\n    const oidcbucket = new s3.Bucket(this, 'oidcbucket', {\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n      autoDeleteObjects: true,\n      encryption: BucketEncryption.S3_MANAGED,\n      versioned: true,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL\n    })\n\n    /** ------------------- Cloudfront Definition ------------------- */\n\n    const cloudfrontOAI = new cloudfront.OriginAccessIdentity(this, 'cloudfront-OAI', {\n      comment: 'OAI for oidc'\n    })\n\n    const distribution = new cloudfront.Distribution(this, 'oidcDistribution', {\n      domainNames: distributionDomainNames,\n      comment: 'Discovery endpoint for OIDC',\n      certificate: oidcCertificate,\n      defaultBehavior: {\n        origin: new cloudfrontOrigins.S3Origin(oidcbucket, { originAccessIdentity: cloudfrontOAI }),\n        compress: true,\n        allowedMethods: cloudfront.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n        viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS\n      }\n    })\n\n    /** ------------------ Lambda Handlers Definition ------------------ */\n\n    const issuer = useCustomDomain ? 'https://' + oidcDomainName : 'https://' + distribution.distributionDomainName\n\n    const rotateKeysRole = new iam.Role(this, 'rotateKeysRole', {\n      assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),\n      managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole')]\n    })\n    const rotateKeys = new lambdaNodejs.NodejsFunction(this, 'keyrotate', {\n      timeout: cdk.Duration.seconds(5),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      role: rotateKeysRole,\n      architecture,\n      environment: {\n        S3_BUCKET: oidcbucket.bucketName,\n        ISSUER: issuer\n      }\n    })\n\n    const signRole = new iam.Role(this, 'signRole', {\n      assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),\n      managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole')]\n    })\n    const sign = new lambdaNodejs.NodejsFunction(this, 'sign', {\n      timeout: cdk.Duration.seconds(5),\n      runtime: lambda.Runtime.NODEJS_18_X,\n      role: signRole,\n      architecture,\n      environment: {\n        ISSUER: issuer,\n        DEFAULT_AUDIENCE: props.defaultAudience\n      }\n    })\n\n    /** ------------------------ SNS Topic ------------------------- */\n\n    this.failedRotationTopic = new sns.Topic(this, 'sts')\n    const snsFail = new tasks.SnsPublish(this, 'snsFailed', {\n      topic: this.failedRotationTopic,\n      subject: 'STS KeyRotate step function execution failed',\n      message: sfn.TaskInput.fromJsonPathAt('$')\n    })\n\n    /** ------------------ Step functions Definition ------------------ */\n\n    const deletePreviousStep = new tasks.LambdaInvoke(this, 'delete Previous', {\n      lambdaFunction: rotateKeys,\n      payload: sfn.TaskInput.fromObject({\n        step: 'deletePrevious'\n      }),\n      outputPath: '$.Payload'\n    })\n\n    const movePreviousStep = new tasks.LambdaInvoke(this, 'move Previous', {\n      lambdaFunction: rotateKeys,\n      payload: sfn.TaskInput.fromObject({\n        step: 'movePrevious'\n      }),\n      outputPath: '$.Payload'\n    })\n\n    const moveCurrentStep = new tasks.LambdaInvoke(this, 'move Current', {\n      lambdaFunction: rotateKeys,\n      payload: sfn.TaskInput.fromObject({\n        step: 'moveCurrent'\n      }),\n      outputPath: '$.Payload'\n    })\n\n    const createPendingStep = new tasks.LambdaInvoke(this, 'create Pending', {\n      lambdaFunction: rotateKeys,\n      payload: sfn.TaskInput.fromObject({\n        step: 'createPending'\n      }),\n      outputPath: '$.Payload'\n    })\n\n    const generateArtifactsStep = new tasks.LambdaInvoke(this, 'generate artifacts', {\n      lambdaFunction: rotateKeys,\n      payload: sfn.TaskInput.fromObject({\n        step: 'generateArtifacts'\n      }),\n      outputPath: '$.Payload'\n    })\n\n    const jobFailed = new sfn.Fail(this, 'Failed', {\n      cause: 'AWS Batch Job Failed',\n      error: 'DescribeJob returned FAILED'\n    })\n\n    const jobSuccess = new sfn.Succeed(this, 'Success!')\n\n    deletePreviousStep.addCatch(snsFail)\n    movePreviousStep.addCatch(snsFail)\n    moveCurrentStep.addCatch(snsFail)\n    createPendingStep.addCatch(snsFail)\n    generateArtifactsStep.addCatch(snsFail)\n\n    // Create chain\n    const definition = deletePreviousStep\n      .next(movePreviousStep)\n      .next(moveCurrentStep)\n      .next(createPendingStep)\n      .next(generateArtifactsStep)\n      .next(jobSuccess)\n\n    snsFail.next(jobFailed)\n\n    // Create state machine\n    const rotateKeysMachine = new sfn.StateMachine(this, 'RotateKeys', {\n      definitionBody: sfn.DefinitionBody.fromChainable(\n        definition\n      ),\n      timeout: cdk.Duration.minutes(5)\n    })\n\n    rotateKeys.grantInvoke(rotateKeysMachine.role)\n    oidcbucket.grantReadWrite(rotateKeys)\n\n    const statementSign = new iam.PolicyStatement()\n    statementSign.addActions('kms:*')\n    statementSign.addResources('*')\n    const signPolicy = new iam.ManagedPolicy(this, 'SignPolicy', {\n      statements: [statementSign]\n    })\n    signRole.addManagedPolicy(signPolicy)\n\n    const statementRotateKeys = new iam.PolicyStatement()\n    statementRotateKeys.addActions('kms:*')\n    statementRotateKeys.addResources('*')\n    const rotateKeysPolicy = new iam.ManagedPolicy(this, 'RotateKeysPolicy', {\n      statements: [statementRotateKeys]\n    })\n    rotateKeysRole.addManagedPolicy(rotateKeysPolicy)\n\n    /** ------------------ Events Rule Definition ------------------ */\n\n    // Run every 3 months at 8 PM UTC\n    const scheduledRotateRule = new events.Rule(this, 'scheduledRotateRule', {\n      schedule: events.Schedule.expression('cron(0 20 1 */3 ? *)')\n    })\n    scheduledRotateRule.addTarget(new targets.SfnStateMachine(rotateKeysMachine))\n\n    // Create state machine and trigger to populate initial keys\n    if (!props.disableKeyRotateOnCreate) {\n      const rotateOnce = new tasks.StepFunctionsStartExecution(this, 'rotateOnce', {\n        stateMachine: rotateKeysMachine,\n        integrationPattern: sfn.IntegrationPattern.RUN_JOB\n      })\n\n      const rotateTwice = new tasks.StepFunctionsStartExecution(this, 'rotateTwice', {\n        stateMachine: rotateKeysMachine,\n        integrationPattern: sfn.IntegrationPattern.RUN_JOB\n      })\n\n      const populateKeys = new sfn.StateMachine(this, 'populateKeys', {\n        definitionBody: sfn.DefinitionBody.fromChainable(\n          rotateOnce.next(rotateTwice)\n        ),\n        timeout: cdk.Duration.minutes(10)\n      })\n\n      const initialRunRule = new events.Rule(this, 'initialRunRule', {\n        eventPattern: {\n          source: ['aws.cloudformation'],\n          resources: [cdk.Stack.of(this).stackId],\n          detailType: ['CloudFormation Stack Status Change'],\n          detail: {\n            'status-details': {\n              status: ['CREATE_COMPLETE']\n            }\n          }\n        }\n      })\n\n      initialRunRule.addTarget(new targets.SfnStateMachine(populateKeys))\n    }\n\n    /** ---------------------- API Gateway ----------------------- */\n\n    // only set policy when orgId is set\n    let apiPolicy: PolicyDocument | undefined\n    if (props.orgId) {\n      apiPolicy = new iam.PolicyDocument({\n        statements: [\n          new iam.PolicyStatement({\n            actions: ['execute-api:Invoke'],\n            resources: ['*'],\n            principals: [\n              new OrganizationPrincipal(props.orgId)\n            ]\n          })\n        ]\n      })\n    }\n\n    const logGroup = new logs.LogGroup(this, 'APIGatewayAccessLogs', {\n      retention: 7\n    })\n\n    // Create API\n    const api = new apigateway.LambdaRestApi(this, 'jwk-sts-api', {\n      description: 'STS Token API Gateway',\n      handler: sign,\n      defaultMethodOptions: {\n        authorizationType: apigateway.AuthorizationType.IAM\n      },\n      endpointConfiguration: {\n        types: [apigateway.EndpointType.REGIONAL]\n      },\n      policy: apiPolicy,\n      deployOptions: {\n        loggingLevel: MethodLoggingLevel.INFO,\n        accessLogDestination: new apigateway.LogGroupLogDestination(logGroup)\n      }\n    })\n\n    /** ------------------- Route53 Definition for custom domain ------------------- */\n\n    if (useCustomDomain && hostedZone) {\n      api.addDomainName('apiCustomDomainName', {\n        domainName: tokenDomainName,\n        certificate: tokenCertificate!\n      })\n\n      // Add A record for cloudfront distribution\n\n      new route53.ARecord(this, 'oidcRecord', {\n        recordName: oidcDomainName,\n        zone: hostedZone,\n        target: route53.RecordTarget.fromAlias(new route53targets.CloudFrontTarget(distribution))\n      })\n\n      new route53.ARecord(this, 'tokenRecord', {\n        recordName: tokenDomainName,\n        zone: hostedZone,\n        target: route53.RecordTarget.fromAlias(new route53targets.ApiGateway(api))\n      })\n\n      new cdk.CfnOutput(this, 'tokenEndpoint', {\n        value: 'https://' + tokenDomainName + '/token',\n        description: 'Url of the token endpoint',\n        exportName: `${cdk.Stack.of(this)}-tokenEndpoint`\n      })\n    } else {\n      new cdk.CfnOutput(this, 'tokenEndpoint', {\n        value: api.url + 'token',\n        description: 'Url of the token endpoint',\n        exportName: `${cdk.Stack.of(this)}-tokenEndpoint`\n      })\n    }\n\n    new cdk.CfnOutput(this, 'issuer', {\n      value: issuer,\n      description: 'Url of the issuer',\n      exportName: `${cdk.Stack.of(this)}-issuer`\n    })\n\n    /** ---------------------- WAF ----------------------- */\n\n    if (props.apiGwWaf === wafUsage.ConstructProvided) {\n      // API gateway WAF ACL and rules\n      const APIGatewayWebACL = new wafv2.CfnWebACL(this, 'APIGatewayWebACL', {\n        description: 'This is WebACL for Auth APi Gateway',\n        scope: 'REGIONAL',\n        defaultAction: { allow: {} },\n        visibilityConfig: {\n          metricName: 'APIWebACL',\n          cloudWatchMetricsEnabled: true,\n          sampledRequestsEnabled: true\n        },\n        rules: [\n          {\n            name: 'AWS-AWSManagedRulesCommonRuleSet',\n            priority: 0,\n            statement: {\n              managedRuleGroupStatement: {\n                vendorName: 'AWS',\n                name: 'AWSManagedRulesCommonRuleSet'\n              }\n            },\n            overrideAction: {\n              none: {}\n            },\n            visibilityConfig: {\n              sampledRequestsEnabled: true,\n              cloudWatchMetricsEnabled: true,\n              metricName: 'AWS-AWSManagedRulesCommonRuleSet'\n            }\n          },\n          {\n            name: 'AWS-AWSManagedRulesAmazonIpReputationList',\n            priority: 1,\n            statement: {\n              managedRuleGroupStatement: {\n                vendorName: 'AWS',\n                name: 'AWSManagedRulesAmazonIpReputationList'\n              }\n            },\n            overrideAction: {\n              none: {}\n            },\n            visibilityConfig: {\n              sampledRequestsEnabled: true,\n              cloudWatchMetricsEnabled: true,\n              metricName: 'AWS-AWSManagedRulesAmazonIpReputationList'\n            }\n          },\n          {\n            name: 'api-gw-AuthAPIGeoLocation',\n            priority: 3,\n            action: { block: {} },\n            visibilityConfig: {\n              metricName: 'AuthAPIGeoLocation',\n              cloudWatchMetricsEnabled: true,\n              sampledRequestsEnabled: false\n            },\n            statement: {\n              geoMatchStatement: {\n                countryCodes: ['BY', 'CN', 'IR', 'RU', 'SY', 'KP']\n              }\n            }\n          },\n          {\n            name: 'api-gw-rateLimitRule',\n            priority: 4,\n            action: { block: {} },\n            visibilityConfig: {\n              metricName: 'rateLimitRule',\n              cloudWatchMetricsEnabled: true,\n              sampledRequestsEnabled: false\n            },\n            statement: {\n              rateBasedStatement: {\n                aggregateKeyType: 'IP',\n                limit: 100\n              }\n            }\n          }\n        ]\n      })\n\n      // Web ACL Association\n      new wafv2.CfnWebACLAssociation(this, 'APIGatewayWebACLAssociation', {\n        webAclArn: APIGatewayWebACL.attrArn,\n        resourceArn: api.deploymentStage.stageArn\n      })\n    } else if (props.apiGwWaf === wafUsage.ProvideWebAclArn && props.apiGwWafWebAclArn) {\n      // Web ACL Association\n      new wafv2.CfnWebACLAssociation(this, 'APIGatewayWebACLAssociation', {\n        webAclArn: props.apiGwWafWebAclArn,\n        resourceArn: api.deploymentStage.stageArn\n      })\n    }\n\n    /** ---------------------- Cloudwatch ----------------------- */\n\n    new cloudwatch.Alarm(this, 'StepFunctionError', {\n      alarmName: props.alarmNameKeyRotationStepFunctionFailed ?? 'sts-key_rotate_sfn-alarm',\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: 1,\n      evaluationPeriods: 1,\n      metric: rotateKeysMachine.metricFailed(),\n      alarmDescription: 'Key Rotation Failed',\n      treatMissingData: TreatMissingData.NOT_BREACHING\n    })\n\n    new cloudwatch.Alarm(this, 'ApiGateway5XXAlarm', {\n      alarmName: props.alarmNameApiGateway5xx ?? 'sts-5xx_api_gw-alarm',\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: 1,\n      evaluationPeriods: 1,\n      metric: api.metricServerError(),\n      alarmDescription: '5xx STS API gateway failures',\n      treatMissingData: TreatMissingData.NOT_BREACHING\n    })\n\n    const signErrors = sign.metricErrors({\n      period: cdk.Duration.minutes(1)\n    })\n\n    const rotateErrors = rotateKeys.metricErrors({\n      period: cdk.Duration.minutes(1)\n    })\n\n    new cloudwatch.Alarm(this, 'LambdaSignError', {\n      alarmName: props.alarmNameSignLambdaFailed ?? 'sts-sign_errors_lambda-alarm',\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: 1,\n      evaluationPeriods: 1,\n      metric: signErrors,\n      alarmDescription: 'Sign Lambda Failed',\n      treatMissingData: TreatMissingData.NOT_BREACHING\n    })\n\n    new cloudwatch.Alarm(this, 'LambdaRotateError', {\n      alarmName: props.alarmNameKeyRotationLambdaFailed ?? 'sts-key_rotate_errors_lambda-alarm',\n      comparisonOperator: cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD,\n      threshold: 1,\n      evaluationPeriods: 1,\n      metric: rotateErrors,\n      alarmDescription: 'Key Rotation Lambda Failed',\n      treatMissingData: TreatMissingData.NOT_BREACHING\n    })\n  }\n}\n"]}
@@ -107,10 +107,11 @@ async function updateOrCreateAlias(aliasName, keyId) {
107
107
  }
108
108
  }
109
109
  async function getKeyIdForAlias(keyId) {
110
+ var _a;
110
111
  try {
111
112
  const response = await client.send(new client_kms_1.DescribeKeyCommand({ KeyId: keyId }));
112
113
  console.log(response);
113
- return response.KeyMetadata?.KeyId;
114
+ return (_a = response.KeyMetadata) === null || _a === void 0 ? void 0 : _a.KeyId;
114
115
  }
115
116
  catch (err) {
116
117
  if (err instanceof client_kms_1.NotFoundException) {
@@ -184,10 +185,10 @@ async function uploadToS3(key, contents) {
184
185
  // Write jwk to s3 bucket
185
186
  await s3client.send(new client_s3_1.PutObjectCommand({
186
187
  Bucket: s3Bucket,
187
- Key: key,
188
+ Key: key, // File name you want to save as in S3
188
189
  Body: Buffer.from(JSON.stringify(contents)),
189
190
  ContentType: 'application/json',
190
191
  ContentEncoding: ''
191
192
  }));
192
193
  }
193
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.keyrotate.js","sourceRoot":"","sources":["../src/index.keyrotate.ts"],"names":[],"mappings":"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","sourcesContent":["// SPDX-FileCopyrightText: 2023 Alliander NV\n//\n// SPDX-License-Identifier: Apache-2.0\n\nimport {\n  KMSClient,\n  DescribeKeyCommand,\n  GetPublicKeyCommand,\n  CreateKeyCommand,\n  UpdateAliasCommand,\n  ScheduleKeyDeletionCommand,\n  TagResourceCommand,\n  Tag,\n  NotFoundException,\n  CreateAliasCommand\n} from '@aws-sdk/client-kms'\nimport { S3Client, PutObjectCommand } from '@aws-sdk/client-s3'\nimport { KEYUTIL, KJUR } from 'jsrsasign'\n\nconst client = new KMSClient({})\n\nconst ALIAS_PREVIOUS = 'alias/sts/PREVIOUS'\nconst ALIAS_CURRENT = 'alias/sts/CURRENT'\nconst ALIAS_PENDING = 'alias/sts/PENDING'\n\nconst ALIASES: string[] = [\n  ALIAS_PREVIOUS,\n  ALIAS_CURRENT,\n  ALIAS_PENDING\n]\n\nexport const handler = async (event: any): Promise<any> => {\n  // retrieve the step from the event\n  const step = (event.step)\n\n  // match the step with the corresponding function\n  switch (step) {\n    case 'deletePrevious':\n      await deletePrevious()\n      break\n    case 'movePrevious':\n      await movePrevious()\n      break\n    case 'moveCurrent':\n      await moveCurrent()\n      break\n    case 'createPending':\n      await createPending()\n      break\n    case 'generateArtifacts':\n      await generateJWKS()\n      await generateOpenIDConfiguration()\n      break\n\n    default:\n      console.log('invalid step')\n  }\n}\n\nasync function deletePrevious () {\n  console.log('Deleting PREVIOUS aliased key')\n\n  const prevKeyId = await getKeyIdForAlias(ALIAS_PREVIOUS)\n  if (prevKeyId) {\n    const ScheduleDeleteResponse = await client.send(\n      new ScheduleKeyDeletionCommand({ KeyId: prevKeyId })\n    )\n    console.log(ScheduleDeleteResponse)\n  } else {\n    console.log('No PREVIOUS key at the moment, skip deletion')\n  }\n}\n\nasync function movePrevious () {\n  console.log('moving PREVIOUS alias')\n  const currentKeyId = await getKeyIdForAlias(ALIAS_CURRENT)\n  if (currentKeyId) {\n    await updateOrCreateAlias(ALIAS_PREVIOUS, currentKeyId)\n  } else {\n    console.log('No CURRENT key at the moment, skip assigning the PREVIOUS alias to this key.')\n  }\n}\n\nasync function moveCurrent () {\n  console.log('Moving CURRENT alias')\n\n  const pendingKeyId = await getKeyIdForAlias(ALIAS_PENDING)\n  if (pendingKeyId) {\n    await updateOrCreateAlias(ALIAS_CURRENT, pendingKeyId)\n  } else {\n    console.log('No PENDING key at the moment, skip assigning the CURRENT alias to this key.')\n  }\n}\n\nasync function createPending () {\n  console.log('Creating new key for PENDING')\n\n  // Create new key\n  const createResponse = await client.send(new CreateKeyCommand({\n    KeySpec: 'RSA_2048',\n    KeyUsage: 'SIGN_VERIFY'\n  }))\n  console.log(createResponse)\n\n  // Update the new key with pending alias\n  await updateOrCreateAlias(ALIAS_PENDING, createResponse.KeyMetadata!.KeyId!)\n}\n\nasync function updateOrCreateAlias (aliasName: string, keyId: string) {\n  try {\n    const updateResponse = await client.send(new UpdateAliasCommand({\n      AliasName: aliasName,\n      TargetKeyId: keyId\n    }))\n    console.log(updateResponse)\n  } catch (err) {\n    if (err instanceof NotFoundException) {\n      console.log('ALIAS not found, creating it.')\n      const createResponse = await client.send(new CreateAliasCommand({\n        AliasName: aliasName,\n        TargetKeyId: keyId\n      }))\n      console.log(createResponse)\n    } else {\n      throw (err)\n    }\n  }\n}\n\nasync function getKeyIdForAlias (keyId: string) {\n  try {\n    const response = await client.send(new DescribeKeyCommand({ KeyId: keyId }))\n    console.log(response)\n    return response.KeyMetadata?.KeyId\n  } catch (err) {\n    if (err instanceof NotFoundException) {\n      return null\n    } else {\n      throw err\n    }\n  }\n}\n\nasync function generateJWKS () {\n  const allKeys: object[] = []\n\n  for (const keyAlias of ALIASES) {\n    const keyId = await getKeyIdForAlias(keyAlias)\n    if (keyId) {\n      const jwkContents = await generateJWK(keyAlias)\n      await setKMSKeyTags(keyId, [{ TagKey: 'jwk_kid', TagValue: jwkContents.kid }])\n      allKeys.push(jwkContents)\n    }\n  }\n\n  const result = { keys: allKeys }\n\n  await uploadToS3('discovery/keys', result)\n}\n\nasync function generateOpenIDConfiguration () {\n  const issuer = process.env.ISSUER\n\n  const openIdConfiguration = {\n    issuer,\n    jwks_uri: `${issuer}/discovery/keys`,\n    response_types_supported: [\n      'token'\n    ],\n    id_token_signing_alg_values_supported: [\n      'RS256'\n    ],\n    scopes_supported: [\n      'openid'\n    ],\n    token_endpoint_auth_methods_supported: [\n      'client_secret_basic'\n    ],\n    claims_supported: [\n      'aud',\n      'exp',\n      'iat',\n      'iss',\n      'sub'\n    ]\n  }\n\n  await uploadToS3('.well-known/openid-configuration', openIdConfiguration)\n}\n\nasync function generateJWK (keyAlias: string): Promise<any> {\n  // Get the public key from kms\n  const getPubKeyResponse = await client.send(new GetPublicKeyCommand({ KeyId: keyAlias }))\n\n  // generate HEX format from the response (DER)\n  const pubKeyHex = Buffer.from(getPubKeyResponse.PublicKey as Uint8Array).toString('hex')\n\n  // Get the pub key in pem format\n  const pubKeyPem = KJUR.asn1.ASN1Util.getPEMStringFromHex(pubKeyHex, 'PUBLIC KEY')\n\n  // return the JWK format for the key\n  const jwk: any = KEYUTIL.getJWK(pubKeyPem)\n\n  jwk.use = 'sig'\n  jwk.alg = 'RS256'\n\n  return jwk\n}\n\nasync function setKMSKeyTags (keyAlias: string, tags: Tag[]) {\n  return await client.send(new TagResourceCommand({ KeyId: keyAlias, Tags: tags }))\n}\n\nasync function uploadToS3 (key: string, contents: object) {\n  // get S3 bucket from environment variables\n  const s3Bucket = process.env.S3_BUCKET\n\n  const s3client = new S3Client({})\n\n  // Write jwk to s3 bucket\n  await s3client.send(new PutObjectCommand({\n    Bucket: s3Bucket,\n    Key: key, // File name you want to save as in S3\n    Body: Buffer.from(JSON.stringify(contents)),\n    ContentType: 'application/json',\n    ContentEncoding: ''\n  }))\n}\n"]}
194
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.keyrotate.js","sourceRoot":"","sources":["../src/index.keyrotate.ts"],"names":[],"mappings":"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","sourcesContent":["// SPDX-FileCopyrightText: 2023 Alliander NV\n//\n// SPDX-License-Identifier: Apache-2.0\n\nimport {\n  KMSClient,\n  DescribeKeyCommand,\n  GetPublicKeyCommand,\n  CreateKeyCommand,\n  UpdateAliasCommand,\n  ScheduleKeyDeletionCommand,\n  TagResourceCommand,\n  Tag,\n  NotFoundException,\n  CreateAliasCommand\n} from '@aws-sdk/client-kms'\nimport { S3Client, PutObjectCommand } from '@aws-sdk/client-s3'\nimport { KEYUTIL, KJUR } from 'jsrsasign'\n\nconst client = new KMSClient({})\n\nconst ALIAS_PREVIOUS = 'alias/sts/PREVIOUS'\nconst ALIAS_CURRENT = 'alias/sts/CURRENT'\nconst ALIAS_PENDING = 'alias/sts/PENDING'\n\nconst ALIASES: string[] = [\n  ALIAS_PREVIOUS,\n  ALIAS_CURRENT,\n  ALIAS_PENDING\n]\n\nexport const handler = async (event: any): Promise<any> => {\n  // retrieve the step from the event\n  const step = (event.step)\n\n  // match the step with the corresponding function\n  switch (step) {\n    case 'deletePrevious':\n      await deletePrevious()\n      break\n    case 'movePrevious':\n      await movePrevious()\n      break\n    case 'moveCurrent':\n      await moveCurrent()\n      break\n    case 'createPending':\n      await createPending()\n      break\n    case 'generateArtifacts':\n      await generateJWKS()\n      await generateOpenIDConfiguration()\n      break\n\n    default:\n      console.log('invalid step')\n  }\n}\n\nasync function deletePrevious () {\n  console.log('Deleting PREVIOUS aliased key')\n\n  const prevKeyId = await getKeyIdForAlias(ALIAS_PREVIOUS)\n  if (prevKeyId) {\n    const ScheduleDeleteResponse = await client.send(\n      new ScheduleKeyDeletionCommand({ KeyId: prevKeyId })\n    )\n    console.log(ScheduleDeleteResponse)\n  } else {\n    console.log('No PREVIOUS key at the moment, skip deletion')\n  }\n}\n\nasync function movePrevious () {\n  console.log('moving PREVIOUS alias')\n  const currentKeyId = await getKeyIdForAlias(ALIAS_CURRENT)\n  if (currentKeyId) {\n    await updateOrCreateAlias(ALIAS_PREVIOUS, currentKeyId)\n  } else {\n    console.log('No CURRENT key at the moment, skip assigning the PREVIOUS alias to this key.')\n  }\n}\n\nasync function moveCurrent () {\n  console.log('Moving CURRENT alias')\n\n  const pendingKeyId = await getKeyIdForAlias(ALIAS_PENDING)\n  if (pendingKeyId) {\n    await updateOrCreateAlias(ALIAS_CURRENT, pendingKeyId)\n  } else {\n    console.log('No PENDING key at the moment, skip assigning the CURRENT alias to this key.')\n  }\n}\n\nasync function createPending () {\n  console.log('Creating new key for PENDING')\n\n  // Create new key\n  const createResponse = await client.send(new CreateKeyCommand({\n    KeySpec: 'RSA_2048',\n    KeyUsage: 'SIGN_VERIFY'\n  }))\n  console.log(createResponse)\n\n  // Update the new key with pending alias\n  await updateOrCreateAlias(ALIAS_PENDING, createResponse.KeyMetadata!.KeyId!)\n}\n\nasync function updateOrCreateAlias (aliasName: string, keyId: string) {\n  try {\n    const updateResponse = await client.send(new UpdateAliasCommand({\n      AliasName: aliasName,\n      TargetKeyId: keyId\n    }))\n    console.log(updateResponse)\n  } catch (err) {\n    if (err instanceof NotFoundException) {\n      console.log('ALIAS not found, creating it.')\n      const createResponse = await client.send(new CreateAliasCommand({\n        AliasName: aliasName,\n        TargetKeyId: keyId\n      }))\n      console.log(createResponse)\n    } else {\n      throw (err)\n    }\n  }\n}\n\nasync function getKeyIdForAlias (keyId: string) {\n  try {\n    const response = await client.send(new DescribeKeyCommand({ KeyId: keyId }))\n    console.log(response)\n    return response.KeyMetadata?.KeyId\n  } catch (err) {\n    if (err instanceof NotFoundException) {\n      return null\n    } else {\n      throw err\n    }\n  }\n}\n\nasync function generateJWKS () {\n  const allKeys: object[] = []\n\n  for (const keyAlias of ALIASES) {\n    const keyId = await getKeyIdForAlias(keyAlias)\n    if (keyId) {\n      const jwkContents = await generateJWK(keyAlias)\n      await setKMSKeyTags(keyId, [{ TagKey: 'jwk_kid', TagValue: jwkContents.kid }])\n      allKeys.push(jwkContents)\n    }\n  }\n\n  const result = { keys: allKeys }\n\n  await uploadToS3('discovery/keys', result)\n}\n\nasync function generateOpenIDConfiguration () {\n  const issuer = process.env.ISSUER\n\n  const openIdConfiguration = {\n    issuer,\n    jwks_uri: `${issuer}/discovery/keys`,\n    response_types_supported: [\n      'token'\n    ],\n    id_token_signing_alg_values_supported: [\n      'RS256'\n    ],\n    scopes_supported: [\n      'openid'\n    ],\n    token_endpoint_auth_methods_supported: [\n      'client_secret_basic'\n    ],\n    claims_supported: [\n      'aud',\n      'exp',\n      'iat',\n      'iss',\n      'sub'\n    ]\n  }\n\n  await uploadToS3('.well-known/openid-configuration', openIdConfiguration)\n}\n\nasync function generateJWK (keyAlias: string): Promise<any> {\n  // Get the public key from kms\n  const getPubKeyResponse = await client.send(new GetPublicKeyCommand({ KeyId: keyAlias }))\n\n  // generate HEX format from the response (DER)\n  const pubKeyHex = Buffer.from(getPubKeyResponse.PublicKey as Uint8Array).toString('hex')\n\n  // Get the pub key in pem format\n  const pubKeyPem = KJUR.asn1.ASN1Util.getPEMStringFromHex(pubKeyHex, 'PUBLIC KEY')\n\n  // return the JWK format for the key\n  const jwk: any = KEYUTIL.getJWK(pubKeyPem)\n\n  jwk.use = 'sig'\n  jwk.alg = 'RS256'\n\n  return jwk\n}\n\nasync function setKMSKeyTags (keyAlias: string, tags: Tag[]) {\n  return await client.send(new TagResourceCommand({ KeyId: keyAlias, Tags: tags }))\n}\n\nasync function uploadToS3 (key: string, contents: object) {\n  // get S3 bucket from environment variables\n  const s3Bucket = process.env.S3_BUCKET\n\n  const s3client = new S3Client({})\n\n  // Write jwk to s3 bucket\n  await s3client.send(new PutObjectCommand({\n    Bucket: s3Bucket,\n    Key: key, // File name you want to save as in S3\n    Body: Buffer.from(JSON.stringify(contents)),\n    ContentType: 'application/json',\n    ContentEncoding: ''\n  }))\n}\n"]}
@@ -10,10 +10,11 @@ const logger_1 = require("@aws-lambda-powertools/logger");
10
10
  const KEY_ALIAS_CURRENT = 'alias/sts/CURRENT';
11
11
  const logger = new logger_1.Logger();
12
12
  const handler = async (apiEvent, context) => {
13
- const identityArn = getARNFromIdentity(apiEvent.requestContext.identity?.userArn);
13
+ var _a, _b, _c, _d;
14
+ const identityArn = getARNFromIdentity((_a = apiEvent.requestContext.identity) === null || _a === void 0 ? void 0 : _a.userArn);
14
15
  logger.debug(identityArn);
15
16
  if (identityArn === undefined || identityArn === null) {
16
- logger.info(`Unable to resolve identityArn for userArn: ${apiEvent.requestContext.identity?.userArn}`);
17
+ logger.info(`Unable to resolve identityArn for userArn: ${(_b = apiEvent.requestContext.identity) === null || _b === void 0 ? void 0 : _b.userArn}`);
17
18
  return respond('Unable to resolve identity', 400);
18
19
  }
19
20
  let aud = process.env.DEFAULT_AUDIENCE;
@@ -23,13 +24,13 @@ const handler = async (apiEvent, context) => {
23
24
  const kms = new client_kms_1.KMSClient({});
24
25
  // Get KeyID which will be sent as kid in JWT token
25
26
  const currentResponse = await kms.send(new client_kms_1.DescribeKeyCommand({ KeyId: `${KEY_ALIAS_CURRENT}` }));
26
- const currentKeyId = currentResponse.KeyMetadata?.KeyId;
27
+ const currentKeyId = (_c = currentResponse.KeyMetadata) === null || _c === void 0 ? void 0 : _c.KeyId;
27
28
  if (currentKeyId === undefined) {
28
29
  return respond('KMS key could not be retrieved', 500);
29
30
  }
30
31
  // Retrieve Tags for KMS Key - the key is tagged with the `kid` from the JWK which is used in the JWT headers
31
32
  const listResourceTagsResponse = await kms.send(new client_kms_1.ListResourceTagsCommand({ KeyId: currentKeyId }));
32
- const kid = getTagValueFromTags('jwk_kid', listResourceTagsResponse.Tags ?? []);
33
+ const kid = getTagValueFromTags('jwk_kid', (_d = listResourceTagsResponse.Tags) !== null && _d !== void 0 ? _d : []);
33
34
  if (kid == null) {
34
35
  return respond('KMS key is not correctly tagged', 500);
35
36
  }
@@ -48,7 +49,7 @@ const handler = async (apiEvent, context) => {
48
49
  notBeforeDate.setTime(notBeforeDate.getTime() - 5 * 60 * 1000); // 5m before issuedAtDate
49
50
  // JWT Token payload
50
51
  const payload = {
51
- sub: `${identityArn}`,
52
+ sub: `${identityArn}`, // Set role arn as message for payload
52
53
  aud,
53
54
  iss,
54
55
  iat: Math.floor(issuedAtDate.getTime() / 1000),
@@ -86,23 +87,24 @@ function respond(message, statusCode = 200) {
86
87
  };
87
88
  }
88
89
  function getARNFromIdentity(identityArn) {
90
+ var _a, _b;
89
91
  if (identityArn === undefined || identityArn === null) {
90
92
  return null;
91
93
  }
92
94
  // Regex for converting arn to base role
93
95
  const captGroups = [
94
96
  'arn:aws:sts:',
95
- '(?<regionName>[^:]*)',
97
+ '(?<regionName>[^:]*)', // group 1
96
98
  ':',
97
- '(?<accountId>\\d{12})',
99
+ '(?<accountId>\\d{12})', // group 2
98
100
  ':assumed-role\\/',
99
- '(?<roleName>[A-z0-9\\-]+?)',
101
+ '(?<roleName>[A-z0-9\\-]+?)', // group 3
100
102
  '\\/',
101
- '(?<user>[^:]*)',
103
+ '(?<user>[^:]*)', // group 4
102
104
  '$'
103
105
  ];
104
106
  const regex = new RegExp(captGroups.join(''));
105
- const { regionName, accountId, roleName } = regex.exec(identityArn)?.groups ?? {};
107
+ const { regionName, accountId, roleName } = (_b = (_a = regex.exec(identityArn)) === null || _a === void 0 ? void 0 : _a.groups) !== null && _b !== void 0 ? _b : {};
106
108
  if (regionName === undefined || accountId === undefined || roleName === undefined) {
107
109
  return null;
108
110
  }
@@ -117,4 +119,4 @@ function getTagValueFromTags(tagKey, tags) {
117
119
  }
118
120
  return null;
119
121
  }
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.sign.js","sourceRoot":"","sources":["../src/index.sign.ts"],"names":[],"mappings":"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","sourcesContent":["// SPDX-FileCopyrightText: 2023 Alliander NV\n//\n// SPDX-License-Identifier: Apache-2.0\n\nimport { Context, APIGatewayProxyResult, APIGatewayEvent } from 'aws-lambda'\nimport { KMSClient, SignCommand, DescribeKeyCommand, ListResourceTagsCommand, Tag } from '@aws-sdk/client-kms'\nimport base64url from 'base64url'\n\nimport { Logger } from '@aws-lambda-powertools/logger'\n\nconst KEY_ALIAS_CURRENT = 'alias/sts/CURRENT'\nconst logger = new Logger()\n\nexport const handler = async (apiEvent: APIGatewayEvent, context: Context): Promise<APIGatewayProxyResult> => {\n  const identityArn = getARNFromIdentity(apiEvent.requestContext.identity?.userArn)\n  logger.debug(identityArn!)\n\n  if (identityArn === undefined || identityArn === null) {\n    logger.info(`Unable to resolve identityArn for userArn: ${apiEvent.requestContext.identity?.userArn}`)\n    return respond('Unable to resolve identity', 400)\n  }\n\n  let aud = process.env.DEFAULT_AUDIENCE\n\n  if (apiEvent.queryStringParameters && apiEvent.queryStringParameters.aud) {\n    aud = apiEvent.queryStringParameters.aud\n  }\n\n  const kms = new KMSClient({})\n\n  // Get KeyID which will be sent as kid in JWT token\n  const currentResponse = await kms.send(new DescribeKeyCommand({ KeyId: `${KEY_ALIAS_CURRENT}` }))\n  const currentKeyId = currentResponse.KeyMetadata?.KeyId\n\n  if (currentKeyId === undefined) {\n    return respond('KMS key could not be retrieved', 500)\n  }\n\n  // Retrieve Tags for KMS Key - the key is tagged with the `kid` from the JWK which is used in the JWT headers\n  const listResourceTagsResponse = await kms.send(new ListResourceTagsCommand({ KeyId: currentKeyId }))\n  const kid = getTagValueFromTags('jwk_kid', listResourceTagsResponse.Tags ?? [])\n\n  if (kid == null) {\n    return respond('KMS key is not correctly tagged', 500)\n  }\n\n  const iss = process.env.ISSUER\n\n  // JWT Token headers\n  const headers: any = {\n    alg: 'RS256',\n    typ: 'JWT',\n    kid: `${kid}`\n  }\n\n  // prepare token lifetime property values\n  const issuedAtDate = new Date()\n  const expirationDate = new Date(issuedAtDate)\n  const notBeforeDate = new Date(issuedAtDate)\n  expirationDate.setTime(expirationDate.getTime() + 60 * 60 * 1000) // valid for one hour\n  notBeforeDate.setTime(notBeforeDate.getTime() - 5 * 60 * 1000) // 5m before issuedAtDate\n\n  // JWT Token payload\n  const payload: any = {\n    sub: `${identityArn}`, // Set role arn as message for payload\n    aud,\n    iss,\n    iat: Math.floor(issuedAtDate.getTime() / 1000),\n    exp: Math.floor(expirationDate.getTime() / 1000),\n    nbf: Math.floor(notBeforeDate.getTime() / 1000)\n  }\n\n  // Prepare message to be signed by KMS\n  const tokenHeaders = base64url(JSON.stringify(headers))\n  const tokenPayload = base64url(JSON.stringify(payload))\n\n  // Sign message with KMS\n  const signResponse = await kms.send(new SignCommand({\n    KeyId: currentKeyId,\n    Message: Buffer.from(`${tokenHeaders}.${tokenPayload}`),\n    SigningAlgorithm: 'RSASSA_PKCS1_V1_5_SHA_256',\n    MessageType: 'RAW'\n  }))\n  logger.debug(JSON.stringify(signResponse))\n\n  const signature = Buffer\n    .from(signResponse.Signature as Uint8Array)\n    .toString('base64')\n    .replace(/\\+/g, '-')\n    .replace(/\\//g, '_')\n    .replace(/=/g, '')\n\n  const token = `${tokenHeaders}.${tokenPayload}.${signature}`\n  logger.debug(token)\n\n  return respond(JSON.stringify({\n    token\n  }))\n}\n\nfunction respond (message: string, statusCode: number = 200) {\n  return {\n    statusCode,\n    body: message\n  }\n}\n\nfunction getARNFromIdentity (identityArn: string | null) {\n  if (identityArn === undefined || identityArn === null) {\n    return null\n  }\n\n  // Regex for converting arn to base role\n  const captGroups = [\n    'arn:aws:sts:',\n    '(?<regionName>[^:]*)', // group 1\n    ':',\n    '(?<accountId>\\\\d{12})', // group 2\n    ':assumed-role\\\\/',\n    '(?<roleName>[A-z0-9\\\\-]+?)', // group 3\n    '\\\\/',\n    '(?<user>[^:]*)', // group 4\n    '$'\n  ]\n\n  const regex = new RegExp(captGroups.join(''))\n  const { regionName, accountId, roleName } = regex.exec(identityArn)?.groups ?? {}\n\n  if (regionName === undefined || accountId === undefined || roleName === undefined) {\n    return null\n  }\n\n  // Build base role arn\n  return `arn:aws:iam:${regionName}:${accountId}:role/${roleName}`\n}\n\nfunction getTagValueFromTags (tagKey: string, tags: Tag[]) {\n  for (const tag of tags) {\n    if (tag.TagKey === tagKey) {\n      return tag.TagValue\n    }\n  }\n\n  return null\n}\n"]}
122
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.sign.js","sourceRoot":"","sources":["../src/index.sign.ts"],"names":[],"mappings":"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","sourcesContent":["// SPDX-FileCopyrightText: 2023 Alliander NV\n//\n// SPDX-License-Identifier: Apache-2.0\n\nimport { Context, APIGatewayProxyResult, APIGatewayEvent } from 'aws-lambda'\nimport { KMSClient, SignCommand, DescribeKeyCommand, ListResourceTagsCommand, Tag } from '@aws-sdk/client-kms'\nimport base64url from 'base64url'\n\nimport { Logger } from '@aws-lambda-powertools/logger'\n\nconst KEY_ALIAS_CURRENT = 'alias/sts/CURRENT'\nconst logger = new Logger()\n\nexport const handler = async (apiEvent: APIGatewayEvent, context: Context): Promise<APIGatewayProxyResult> => {\n  const identityArn = getARNFromIdentity(apiEvent.requestContext.identity?.userArn)\n  logger.debug(identityArn!)\n\n  if (identityArn === undefined || identityArn === null) {\n    logger.info(`Unable to resolve identityArn for userArn: ${apiEvent.requestContext.identity?.userArn}`)\n    return respond('Unable to resolve identity', 400)\n  }\n\n  let aud = process.env.DEFAULT_AUDIENCE\n\n  if (apiEvent.queryStringParameters && apiEvent.queryStringParameters.aud) {\n    aud = apiEvent.queryStringParameters.aud\n  }\n\n  const kms = new KMSClient({})\n\n  // Get KeyID which will be sent as kid in JWT token\n  const currentResponse = await kms.send(new DescribeKeyCommand({ KeyId: `${KEY_ALIAS_CURRENT}` }))\n  const currentKeyId = currentResponse.KeyMetadata?.KeyId\n\n  if (currentKeyId === undefined) {\n    return respond('KMS key could not be retrieved', 500)\n  }\n\n  // Retrieve Tags for KMS Key - the key is tagged with the `kid` from the JWK which is used in the JWT headers\n  const listResourceTagsResponse = await kms.send(new ListResourceTagsCommand({ KeyId: currentKeyId }))\n  const kid = getTagValueFromTags('jwk_kid', listResourceTagsResponse.Tags ?? [])\n\n  if (kid == null) {\n    return respond('KMS key is not correctly tagged', 500)\n  }\n\n  const iss = process.env.ISSUER\n\n  // JWT Token headers\n  const headers: any = {\n    alg: 'RS256',\n    typ: 'JWT',\n    kid: `${kid}`\n  }\n\n  // prepare token lifetime property values\n  const issuedAtDate = new Date()\n  const expirationDate = new Date(issuedAtDate)\n  const notBeforeDate = new Date(issuedAtDate)\n  expirationDate.setTime(expirationDate.getTime() + 60 * 60 * 1000) // valid for one hour\n  notBeforeDate.setTime(notBeforeDate.getTime() - 5 * 60 * 1000) // 5m before issuedAtDate\n\n  // JWT Token payload\n  const payload: any = {\n    sub: `${identityArn}`, // Set role arn as message for payload\n    aud,\n    iss,\n    iat: Math.floor(issuedAtDate.getTime() / 1000),\n    exp: Math.floor(expirationDate.getTime() / 1000),\n    nbf: Math.floor(notBeforeDate.getTime() / 1000)\n  }\n\n  // Prepare message to be signed by KMS\n  const tokenHeaders = base64url(JSON.stringify(headers))\n  const tokenPayload = base64url(JSON.stringify(payload))\n\n  // Sign message with KMS\n  const signResponse = await kms.send(new SignCommand({\n    KeyId: currentKeyId,\n    Message: Buffer.from(`${tokenHeaders}.${tokenPayload}`),\n    SigningAlgorithm: 'RSASSA_PKCS1_V1_5_SHA_256',\n    MessageType: 'RAW'\n  }))\n  logger.debug(JSON.stringify(signResponse))\n\n  const signature = Buffer\n    .from(signResponse.Signature as Uint8Array)\n    .toString('base64')\n    .replace(/\\+/g, '-')\n    .replace(/\\//g, '_')\n    .replace(/=/g, '')\n\n  const token = `${tokenHeaders}.${tokenPayload}.${signature}`\n  logger.debug(token)\n\n  return respond(JSON.stringify({\n    token\n  }))\n}\n\nfunction respond (message: string, statusCode: number = 200) {\n  return {\n    statusCode,\n    body: message\n  }\n}\n\nfunction getARNFromIdentity (identityArn: string | null) {\n  if (identityArn === undefined || identityArn === null) {\n    return null\n  }\n\n  // Regex for converting arn to base role\n  const captGroups = [\n    'arn:aws:sts:',\n    '(?<regionName>[^:]*)', // group 1\n    ':',\n    '(?<accountId>\\\\d{12})', // group 2\n    ':assumed-role\\\\/',\n    '(?<roleName>[A-z0-9\\\\-]+?)', // group 3\n    '\\\\/',\n    '(?<user>[^:]*)', // group 4\n    '$'\n  ]\n\n  const regex = new RegExp(captGroups.join(''))\n  const { regionName, accountId, roleName } = regex.exec(identityArn)?.groups ?? {}\n\n  if (regionName === undefined || accountId === undefined || roleName === undefined) {\n    return null\n  }\n\n  // Build base role arn\n  return `arn:aws:iam:${regionName}:${accountId}:role/${roleName}`\n}\n\nfunction getTagValueFromTags (tagKey: string, tags: Tag[]) {\n  for (const tag of tags) {\n    if (tag.TagKey === tagKey) {\n      return tag.TagValue\n    }\n  }\n\n  return null\n}\n"]}