construct-hub 0.2.3 → 0.2.7

Sign up to get free protection for your applications and to get access to all the features.
package/API.md CHANGED
@@ -323,6 +323,18 @@ The package sources to register with this ConstructHub instance.
323
323
 
324
324
  ---
325
325
 
326
+ ##### `packageTags`<sup>Optional</sup> <a name="construct-hub.ConstructHubProps.property.packageTags"></a>
327
+
328
+ ```typescript
329
+ public readonly packageTags: PackageTag[];
330
+ ```
331
+
332
+ - *Type:* [`construct-hub.PackageTag`](#construct-hub.PackageTag)[]
333
+
334
+ Configuration for custom package tags.
335
+
336
+ ---
337
+
326
338
  ### DenyListMap <a name="construct-hub.DenyListMap"></a>
327
339
 
328
340
  The contents of the deny list file in S3.
@@ -701,6 +713,155 @@ An optional list of linked resources to be displayed on the monitoring dashboard
701
713
 
702
714
  ---
703
715
 
716
+ ### PackageTag <a name="construct-hub.PackageTag"></a>
717
+
718
+ Configuration for applying custom tags to relevant packages.
719
+
720
+ Custom tags are
721
+ displayed on the package details page, and can be used for searching.
722
+
723
+ #### Initializer <a name="[object Object].Initializer"></a>
724
+
725
+ ```typescript
726
+ import { PackageTag } from 'construct-hub'
727
+
728
+ const packageTag: PackageTag = { ... }
729
+ ```
730
+
731
+ ##### `condition`<sup>Required</sup> <a name="construct-hub.PackageTag.property.condition"></a>
732
+
733
+ ```typescript
734
+ public readonly condition: TagCondition;
735
+ ```
736
+
737
+ - *Type:* [`construct-hub.TagCondition`](#construct-hub.TagCondition)
738
+
739
+ The description of the logic that dictates whether the package has the tag applied.
740
+
741
+ ---
742
+
743
+ ##### `label`<sup>Required</sup> <a name="construct-hub.PackageTag.property.label"></a>
744
+
745
+ ```typescript
746
+ public readonly label: string;
747
+ ```
748
+
749
+ - *Type:* `string`
750
+
751
+ The label for the tag being applied.
752
+
753
+ ---
754
+
755
+ ##### `color`<sup>Optional</sup> <a name="construct-hub.PackageTag.property.color"></a>
756
+
757
+ ```typescript
758
+ public readonly color: string;
759
+ ```
760
+
761
+ - *Type:* `string`
762
+
763
+ The hex value string for the color of the tag when displayed.
764
+
765
+ ---
766
+
767
+ ### PackageTagConfig <a name="construct-hub.PackageTagConfig"></a>
768
+
769
+ Serialized tag declaration to be passed to lambdas via environment variables.
770
+
771
+ #### Initializer <a name="[object Object].Initializer"></a>
772
+
773
+ ```typescript
774
+ import { PackageTagConfig } from 'construct-hub'
775
+
776
+ const packageTagConfig: PackageTagConfig = { ... }
777
+ ```
778
+
779
+ ##### `condition`<sup>Required</sup> <a name="construct-hub.PackageTagConfig.property.condition"></a>
780
+
781
+ ```typescript
782
+ public readonly condition: TagConditionConfig;
783
+ ```
784
+
785
+ - *Type:* [`construct-hub.TagConditionConfig`](#construct-hub.TagConditionConfig)
786
+
787
+ ---
788
+
789
+ ##### `label`<sup>Required</sup> <a name="construct-hub.PackageTagConfig.property.label"></a>
790
+
791
+ ```typescript
792
+ public readonly label: string;
793
+ ```
794
+
795
+ - *Type:* `string`
796
+
797
+ The label for the tag being applied.
798
+
799
+ ---
800
+
801
+ ##### `color`<sup>Optional</sup> <a name="construct-hub.PackageTagConfig.property.color"></a>
802
+
803
+ ```typescript
804
+ public readonly color: string;
805
+ ```
806
+
807
+ - *Type:* `string`
808
+
809
+ The hex value string for the color of the tag when displayed.
810
+
811
+ ---
812
+
813
+ ### TagConditionConfig <a name="construct-hub.TagConditionConfig"></a>
814
+
815
+ Serialized config for a tag condition.
816
+
817
+ #### Initializer <a name="[object Object].Initializer"></a>
818
+
819
+ ```typescript
820
+ import { TagConditionConfig } from 'construct-hub'
821
+
822
+ const tagConditionConfig: TagConditionConfig = { ... }
823
+ ```
824
+
825
+ ##### `type`<sup>Required</sup> <a name="construct-hub.TagConditionConfig.property.type"></a>
826
+
827
+ ```typescript
828
+ public readonly type: TagConditionLogicType;
829
+ ```
830
+
831
+ - *Type:* [`construct-hub.TagConditionLogicType`](#construct-hub.TagConditionLogicType)
832
+
833
+ ---
834
+
835
+ ##### `children`<sup>Optional</sup> <a name="construct-hub.TagConditionConfig.property.children"></a>
836
+
837
+ ```typescript
838
+ public readonly children: TagConditionConfig[];
839
+ ```
840
+
841
+ - *Type:* [`construct-hub.TagConditionConfig`](#construct-hub.TagConditionConfig)[]
842
+
843
+ ---
844
+
845
+ ##### `key`<sup>Optional</sup> <a name="construct-hub.TagConditionConfig.property.key"></a>
846
+
847
+ ```typescript
848
+ public readonly key: string[];
849
+ ```
850
+
851
+ - *Type:* `string`[]
852
+
853
+ ---
854
+
855
+ ##### `value`<sup>Optional</sup> <a name="construct-hub.TagConditionConfig.property.value"></a>
856
+
857
+ ```typescript
858
+ public readonly value: string;
859
+ ```
860
+
861
+ - *Type:* `string`
862
+
863
+ ---
864
+
704
865
  ## Classes <a name="Classes"></a>
705
866
 
706
867
  ### SpdxLicense <a name="construct-hub.SpdxLicense"></a>
@@ -5256,6 +5417,121 @@ Zope Public License 2.1.
5256
5417
 
5257
5418
  ---
5258
5419
 
5420
+ ### TagCondition <a name="construct-hub.TagCondition"></a>
5421
+
5422
+ Condition for applying a custom tag to a package.
5423
+
5424
+ #### Initializers <a name="construct-hub.TagCondition.Initializer"></a>
5425
+
5426
+ ```typescript
5427
+ import { TagCondition } from 'construct-hub'
5428
+
5429
+ new TagCondition()
5430
+ ```
5431
+
5432
+ #### Methods <a name="Methods"></a>
5433
+
5434
+ ##### `bind` <a name="construct-hub.TagCondition.bind"></a>
5435
+
5436
+ ```typescript
5437
+ public bind()
5438
+ ```
5439
+
5440
+ #### Static Functions <a name="Static Functions"></a>
5441
+
5442
+ ##### `and` <a name="construct-hub.TagCondition.and"></a>
5443
+
5444
+ ```typescript
5445
+ import { TagCondition } from 'construct-hub'
5446
+
5447
+ TagCondition.and(conds: TagCondition)
5448
+ ```
5449
+
5450
+ ###### `conds`<sup>Required</sup> <a name="construct-hub.TagCondition.parameter.conds"></a>
5451
+
5452
+ - *Type:* [`construct-hub.TagCondition`](#construct-hub.TagCondition)
5453
+
5454
+ ---
5455
+
5456
+ ##### `field` <a name="construct-hub.TagCondition.field"></a>
5457
+
5458
+ ```typescript
5459
+ import { TagCondition } from 'construct-hub'
5460
+
5461
+ TagCondition.field(keys: string)
5462
+ ```
5463
+
5464
+ ###### `keys`<sup>Required</sup> <a name="construct-hub.TagCondition.parameter.keys"></a>
5465
+
5466
+ - *Type:* `string`
5467
+
5468
+ ---
5469
+
5470
+ ##### `not` <a name="construct-hub.TagCondition.not"></a>
5471
+
5472
+ ```typescript
5473
+ import { TagCondition } from 'construct-hub'
5474
+
5475
+ TagCondition.not(conds: TagCondition)
5476
+ ```
5477
+
5478
+ ###### `conds`<sup>Required</sup> <a name="construct-hub.TagCondition.parameter.conds"></a>
5479
+
5480
+ - *Type:* [`construct-hub.TagCondition`](#construct-hub.TagCondition)
5481
+
5482
+ ---
5483
+
5484
+ ##### `or` <a name="construct-hub.TagCondition.or"></a>
5485
+
5486
+ ```typescript
5487
+ import { TagCondition } from 'construct-hub'
5488
+
5489
+ TagCondition.or(conds: TagCondition)
5490
+ ```
5491
+
5492
+ ###### `conds`<sup>Required</sup> <a name="construct-hub.TagCondition.parameter.conds"></a>
5493
+
5494
+ - *Type:* [`construct-hub.TagCondition`](#construct-hub.TagCondition)
5495
+
5496
+ ---
5497
+
5498
+
5499
+
5500
+ ### TagConditionField <a name="construct-hub.TagConditionField"></a>
5501
+
5502
+ Target a field to use in logic to dictate whether a tag is relevant.
5503
+
5504
+ #### Initializers <a name="construct-hub.TagConditionField.Initializer"></a>
5505
+
5506
+ ```typescript
5507
+ import { TagConditionField } from 'construct-hub'
5508
+
5509
+ new TagConditionField(field: string[])
5510
+ ```
5511
+
5512
+ ##### `field`<sup>Required</sup> <a name="construct-hub.TagConditionField.parameter.field"></a>
5513
+
5514
+ - *Type:* `string`[]
5515
+
5516
+ ---
5517
+
5518
+ #### Methods <a name="Methods"></a>
5519
+
5520
+ ##### `eq` <a name="construct-hub.TagConditionField.eq"></a>
5521
+
5522
+ ```typescript
5523
+ public eq(value: any)
5524
+ ```
5525
+
5526
+ ###### `value`<sup>Required</sup> <a name="construct-hub.TagConditionField.parameter.value"></a>
5527
+
5528
+ - *Type:* `any`
5529
+
5530
+ ---
5531
+
5532
+
5533
+
5534
+
5259
5535
  ## Protocols <a name="Protocols"></a>
5260
5536
 
5261
5537
  ### IDenyList <a name="construct-hub.IDenyList"></a>
@@ -5383,3 +5659,28 @@ the id of the external connection (i.e: `public:npmjs`).
5383
5659
  ---
5384
5660
 
5385
5661
 
5662
+ ## Enums <a name="Enums"></a>
5663
+
5664
+ ### TagConditionLogicType <a name="TagConditionLogicType"></a>
5665
+
5666
+ Logic operators for performing specific conditional logic.
5667
+
5668
+ #### `AND` <a name="construct-hub.TagConditionLogicType.AND"></a>
5669
+
5670
+ ---
5671
+
5672
+
5673
+ #### `OR` <a name="construct-hub.TagConditionLogicType.OR"></a>
5674
+
5675
+ ---
5676
+
5677
+
5678
+ #### `NOT` <a name="construct-hub.TagConditionLogicType.NOT"></a>
5679
+
5680
+ ---
5681
+
5682
+
5683
+ #### `EQUALS` <a name="construct-hub.TagConditionLogicType.EQUALS"></a>
5684
+
5685
+ ---
5686
+
package/README.md CHANGED
@@ -164,6 +164,57 @@ you may set the `isolateLambdas` setting to `false`.
164
164
  There are a number of customizations available in order to make your private
165
165
  construct hub better tailored to your organization.
166
166
 
167
+ #### Package Tags
168
+
169
+ Configuring package tags allows you to compute additional labels to be applied
170
+ to packages. These can be used to indicate to users which packages are owned by
171
+ trusted organizations, or any other arbitrary conditions, and can be referenced
172
+ while searching.
173
+
174
+ For example:
175
+ ```ts
176
+ new ConstructHub(this, "ConstructHub", {
177
+ ...myProps,
178
+ packageTags: [{
179
+ label: 'Official',
180
+ color: '#00FF00',
181
+ condition: TagCondition.field('name').eq('construct-hub'),
182
+ }]
183
+ });
184
+ ```
185
+
186
+ The above example will result in packages with the `name` of `construct-hub` to
187
+ receive the `Official` tag, which is colored green.
188
+
189
+ Combinations of conditions are also supported:
190
+ ```ts
191
+ new ConstructHub(this, "ConstructHub", {
192
+ ...myProps,
193
+ packageTags: [{
194
+ label: 'Official',
195
+ color: '#00FF00',
196
+ condition: TagCondition.or(
197
+ TagCondition.field('name').eq('construct-hub'),
198
+ TagCondition.field('name').eq('construct-hub-webapp'),
199
+ ),
200
+ }]
201
+ });
202
+
203
+ // or more succintly if you have a long list
204
+ condition: TagCondition.or(
205
+ ...['construct-hub', 'construct-hub-webapp', '...',]
206
+ .map(name => TagCondition.field('name').eq(name))
207
+ ),
208
+ ```
209
+
210
+ You can assert against any value within package json including nested ones.
211
+ ```ts
212
+ TagCondition.field('constructHub', 'nested', 'key').eq('value');
213
+
214
+ // checks
215
+ packageJson?.constructHub?.nested?.key === value;
216
+ ```
217
+
167
218
  #### Package Links
168
219
 
169
220
  Configuring package links allows you to replace the `Repository`, `License`,
@@ -6,6 +6,7 @@ import { IBucket } from '@aws-cdk/aws-s3';
6
6
  import { IQueue } from '@aws-cdk/aws-sqs';
7
7
  import { Construct, Duration } from '@aws-cdk/core';
8
8
  import { Monitoring } from '../../monitoring';
9
+ import { PackageTagConfig } from '../../package-tag';
9
10
  import type { PackageLinkConfig } from '../../webapp';
10
11
  import { Orchestration } from '../orchestration';
11
12
  export interface IngestionProps {
@@ -32,6 +33,10 @@ export interface IngestionProps {
32
33
  * Configuration for custom package page links.
33
34
  */
34
35
  readonly packageLinks?: PackageLinkConfig[];
36
+ /**
37
+ * Serialized configuration for custom package tags.
38
+ */
39
+ readonly packageTags?: PackageTagConfig[];
35
40
  }
36
41
  /**
37
42
  * The ingestion function receives messages from discovery integrations and
@@ -20,7 +20,7 @@ const ingestion_1 = require("./ingestion");
20
20
  */
21
21
  class Ingestion extends core_1.Construct {
22
22
  constructor(scope, id, props) {
23
- var _a, _b;
23
+ var _a, _b, _c;
24
24
  super(scope, id);
25
25
  this.queueRetentionPeriod = core_1.Duration.days(14);
26
26
  this.deadLetterQueue = new aws_sqs_1.Queue(this, 'DLQ', {
@@ -43,8 +43,9 @@ class Ingestion extends core_1.Construct {
43
43
  BUCKET_NAME: props.bucket.bucketName,
44
44
  STATE_MACHINE_ARN: props.orchestration.stateMachine.stateMachineArn,
45
45
  PACKAGE_LINKS: JSON.stringify((_a = props.packageLinks) !== null && _a !== void 0 ? _a : []),
46
+ PACKAGE_TAGS: JSON.stringify((_b = props.packageTags) !== null && _b !== void 0 ? _b : []),
46
47
  },
47
- logRetention: (_b = props.logRetention) !== null && _b !== void 0 ? _b : aws_logs_1.RetentionDays.TEN_YEARS,
48
+ logRetention: (_c = props.logRetention) !== null && _c !== void 0 ? _c : aws_logs_1.RetentionDays.TEN_YEARS,
48
49
  memorySize: 10240,
49
50
  timeout: core_1.Duration.minutes(15),
50
51
  tracing: aws_lambda_1.Tracing.ACTIVE,
@@ -146,4 +147,4 @@ class Ingestion extends core_1.Construct {
146
147
  }
147
148
  }
148
149
  exports.Ingestion = Ingestion;
149
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/backend/ingestion/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComparisonOperator, MathExpression, Metric, MetricOptions, Statistic, TreatMissingData } from '@aws-cdk/aws-cloudwatch';\nimport { IGrantable, IPrincipal } from '@aws-cdk/aws-iam';\nimport { IFunction, Tracing } from '@aws-cdk/aws-lambda';\nimport { SqsEventSource } from '@aws-cdk/aws-lambda-event-sources';\nimport { RetentionDays } from '@aws-cdk/aws-logs';\nimport { IBucket } from '@aws-cdk/aws-s3';\nimport { IQueue, Queue, QueueEncryption } from '@aws-cdk/aws-sqs';\nimport { Construct, Duration } from '@aws-cdk/core';\nimport { lambdaFunctionUrl, sqsQueueUrl } from '../../deep-link';\nimport { Monitoring } from '../../monitoring';\nimport { RUNBOOK_URL } from '../../runbook-url';\nimport type { PackageLinkConfig } from '../../webapp';\nimport { Orchestration } from '../orchestration';\nimport { MetricName, METRICS_NAMESPACE } from './constants';\nimport { Ingestion as Handler } from './ingestion';\n\nexport interface IngestionProps {\n  /**\n   * The bucket in which ingested objects are due to be inserted.\n   */\n  readonly bucket: IBucket;\n\n  /**\n   * The monitoring handler to register alarms with.\n   */\n  readonly monitoring: Monitoring;\n\n  /**\n   * The backend orchestration to invoke once the package metadata has been\n   * successfully registered.\n   */\n  readonly orchestration: Orchestration;\n\n  /**\n   * How long to retain the CloudWatch logs.\n   *\n   * @default RetentionDays.TEN_YEARS\n   */\n  readonly logRetention?: RetentionDays;\n\n  /**\n   * Configuration for custom package page links.\n   */\n  readonly packageLinks?: PackageLinkConfig[];\n}\n\n/**\n * The ingestion function receives messages from discovery integrations and\n * processes their payloads into the provided S3 Bucket.\n *\n * This function is also an `IGrantable`, so that it can be granted permissions\n * to read from the source S3 buckets.\n */\nexport class Ingestion extends Construct implements IGrantable {\n  public readonly grantPrincipal: IPrincipal;\n\n  /**\n   * The SQS queue that triggers the ingestion function.\n   */\n  public readonly queue: IQueue;\n\n  /**\n   * The ingestion dead letter queue, which will hold messages that failed\n   * ingestion one too many times, so that poison pills don't endlessly consume\n   * resources.\n   */\n  public readonly deadLetterQueue: IQueue;\n\n  public readonly queueRetentionPeriod = Duration.days(14);\n\n  public readonly function: IFunction;\n\n  public constructor(scope: Construct, id: string, props: IngestionProps) {\n    super(scope, id);\n\n    this.deadLetterQueue = new Queue(this, 'DLQ', {\n      encryption: QueueEncryption.KMS_MANAGED,\n      retentionPeriod: this.queueRetentionPeriod,\n      visibilityTimeout: Duration.minutes(15),\n    });\n\n    this.queue = new Queue(this, 'Queue', {\n      deadLetterQueue: {\n        maxReceiveCount: 5,\n        queue: this.deadLetterQueue,\n      },\n      encryption: QueueEncryption.KMS_MANAGED,\n      retentionPeriod: this.queueRetentionPeriod,\n      visibilityTimeout: Duration.minutes(15),\n    });\n\n    const handler = new Handler(this, 'Default', {\n      description: '[ConstructHub/Ingestion] Ingests new package versions into the Construct Hub',\n      environment: {\n        BUCKET_NAME: props.bucket.bucketName,\n        STATE_MACHINE_ARN: props.orchestration.stateMachine.stateMachineArn,\n        PACKAGE_LINKS: JSON.stringify(props.packageLinks ?? []),\n      },\n      logRetention: props.logRetention ?? RetentionDays.TEN_YEARS,\n      memorySize: 10_240, // Currently the maximum possible setting\n      timeout: Duration.minutes(15),\n      tracing: Tracing.ACTIVE,\n    });\n    this.function = handler;\n\n    props.bucket.grantWrite(this.function);\n    props.orchestration.stateMachine.grantStartExecution(this.function);\n\n    this.function.addEventSource(new SqsEventSource(this.queue, { batchSize: 1 }));\n    // This event source is disabled, and can be used to re-process dead-letter-queue messages\n    this.function.addEventSource(new SqsEventSource(this.deadLetterQueue, { batchSize: 1, enabled: false }));\n\n    this.grantPrincipal = this.function.grantPrincipal;\n\n    props.monitoring.addHighSeverityAlarm(\n      'Ingestion Dead-Letter Queue not empty',\n      new MathExpression({\n        expression: 'm1 + m2',\n        usingMetrics: {\n          m1: this.deadLetterQueue.metricApproximateNumberOfMessagesVisible({ period: Duration.minutes(1) }),\n          m2: this.deadLetterQueue.metricApproximateNumberOfMessagesNotVisible({ period: Duration.minutes(1) }),\n        },\n      }).createAlarm(this, 'DLQAlarm', {\n        alarmName: `${this.node.path}/DLQNotEmpty`,\n        alarmDescription: [\n          'The dead-letter queue for the Ingestion function is not empty!',\n          '',\n          `RunBook: ${RUNBOOK_URL}`,\n          '',\n          `Direct link to the queue: ${sqsQueueUrl(this.deadLetterQueue)}`,\n          `Direct link to the function: ${lambdaFunctionUrl(this.function)}`,\n        ].join('\\n'),\n        comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: 1,\n        threshold: 1,\n        // SQS does not emit metrics if the queue has been empty for a while, which is GOOD.\n        treatMissingData: TreatMissingData.NOT_BREACHING,\n      }),\n    );\n    props.monitoring.addHighSeverityAlarm(\n      'Ingestion failures',\n      this.function.metricErrors().createAlarm(this, 'FailureAlarm', {\n        alarmName: `${this.node.path}/Failure`,\n        alarmDescription: [\n          'The Ingestion function is failing!',\n          '',\n          `RunBook: ${RUNBOOK_URL}`,\n          '',\n          `Direct link to the function: ${lambdaFunctionUrl(this.function)}`,\n        ].join('\\n'),\n        comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: 2,\n        threshold: 1,\n        // Lambda only emits metrics when the function is invoked. No invokation => no errors.\n        treatMissingData: TreatMissingData.NOT_BREACHING,\n      }),\n    );\n  }\n\n  public metricFoundLicenseFile(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.FOUND_LICENSE_FILE,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricIneligibleLicense(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.INELIGIBLE_LICENSE,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricInvalidAssembly(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.INVALID_ASSEMBLY,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricInvalidTarball(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.INVALID_TARBALL,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * This metrics is the total count of packages that were rejected due to\n   * mismatched identity (name, version, license) between the `package.json`\n   * file and te `.jsii` attribute.\n   */\n  public metricMismatchedIdentityRejections(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.MISMATCHED_IDENTITY_REJECTIONS,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n}\n"]}
150
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/backend/ingestion/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComparisonOperator, MathExpression, Metric, MetricOptions, Statistic, TreatMissingData } from '@aws-cdk/aws-cloudwatch';\nimport { IGrantable, IPrincipal } from '@aws-cdk/aws-iam';\nimport { IFunction, Tracing } from '@aws-cdk/aws-lambda';\nimport { SqsEventSource } from '@aws-cdk/aws-lambda-event-sources';\nimport { RetentionDays } from '@aws-cdk/aws-logs';\nimport { IBucket } from '@aws-cdk/aws-s3';\nimport { IQueue, Queue, QueueEncryption } from '@aws-cdk/aws-sqs';\nimport { Construct, Duration } from '@aws-cdk/core';\nimport { lambdaFunctionUrl, sqsQueueUrl } from '../../deep-link';\nimport { Monitoring } from '../../monitoring';\nimport { PackageTagConfig } from '../../package-tag';\nimport { RUNBOOK_URL } from '../../runbook-url';\nimport type { PackageLinkConfig } from '../../webapp';\nimport { Orchestration } from '../orchestration';\nimport { MetricName, METRICS_NAMESPACE } from './constants';\nimport { Ingestion as Handler } from './ingestion';\n\nexport interface IngestionProps {\n  /**\n   * The bucket in which ingested objects are due to be inserted.\n   */\n  readonly bucket: IBucket;\n\n  /**\n   * The monitoring handler to register alarms with.\n   */\n  readonly monitoring: Monitoring;\n\n  /**\n   * The backend orchestration to invoke once the package metadata has been\n   * successfully registered.\n   */\n  readonly orchestration: Orchestration;\n\n  /**\n   * How long to retain the CloudWatch logs.\n   *\n   * @default RetentionDays.TEN_YEARS\n   */\n  readonly logRetention?: RetentionDays;\n\n  /**\n   * Configuration for custom package page links.\n   */\n  readonly packageLinks?: PackageLinkConfig[];\n\n  /**\n   * Serialized configuration for custom package tags.\n   */\n  readonly packageTags?: PackageTagConfig[];\n}\n\n/**\n * The ingestion function receives messages from discovery integrations and\n * processes their payloads into the provided S3 Bucket.\n *\n * This function is also an `IGrantable`, so that it can be granted permissions\n * to read from the source S3 buckets.\n */\nexport class Ingestion extends Construct implements IGrantable {\n  public readonly grantPrincipal: IPrincipal;\n\n  /**\n   * The SQS queue that triggers the ingestion function.\n   */\n  public readonly queue: IQueue;\n\n  /**\n   * The ingestion dead letter queue, which will hold messages that failed\n   * ingestion one too many times, so that poison pills don't endlessly consume\n   * resources.\n   */\n  public readonly deadLetterQueue: IQueue;\n\n  public readonly queueRetentionPeriod = Duration.days(14);\n\n  public readonly function: IFunction;\n\n  public constructor(scope: Construct, id: string, props: IngestionProps) {\n    super(scope, id);\n\n    this.deadLetterQueue = new Queue(this, 'DLQ', {\n      encryption: QueueEncryption.KMS_MANAGED,\n      retentionPeriod: this.queueRetentionPeriod,\n      visibilityTimeout: Duration.minutes(15),\n    });\n\n    this.queue = new Queue(this, 'Queue', {\n      deadLetterQueue: {\n        maxReceiveCount: 5,\n        queue: this.deadLetterQueue,\n      },\n      encryption: QueueEncryption.KMS_MANAGED,\n      retentionPeriod: this.queueRetentionPeriod,\n      visibilityTimeout: Duration.minutes(15),\n    });\n\n    const handler = new Handler(this, 'Default', {\n      description: '[ConstructHub/Ingestion] Ingests new package versions into the Construct Hub',\n      environment: {\n        BUCKET_NAME: props.bucket.bucketName,\n        STATE_MACHINE_ARN: props.orchestration.stateMachine.stateMachineArn,\n        PACKAGE_LINKS: JSON.stringify(props.packageLinks ?? []),\n        PACKAGE_TAGS: JSON.stringify(props.packageTags ?? []),\n      },\n      logRetention: props.logRetention ?? RetentionDays.TEN_YEARS,\n      memorySize: 10_240, // Currently the maximum possible setting\n      timeout: Duration.minutes(15),\n      tracing: Tracing.ACTIVE,\n    });\n    this.function = handler;\n\n    props.bucket.grantWrite(this.function);\n    props.orchestration.stateMachine.grantStartExecution(this.function);\n\n    this.function.addEventSource(new SqsEventSource(this.queue, { batchSize: 1 }));\n    // This event source is disabled, and can be used to re-process dead-letter-queue messages\n    this.function.addEventSource(new SqsEventSource(this.deadLetterQueue, { batchSize: 1, enabled: false }));\n\n    this.grantPrincipal = this.function.grantPrincipal;\n\n    props.monitoring.addHighSeverityAlarm(\n      'Ingestion Dead-Letter Queue not empty',\n      new MathExpression({\n        expression: 'm1 + m2',\n        usingMetrics: {\n          m1: this.deadLetterQueue.metricApproximateNumberOfMessagesVisible({ period: Duration.minutes(1) }),\n          m2: this.deadLetterQueue.metricApproximateNumberOfMessagesNotVisible({ period: Duration.minutes(1) }),\n        },\n      }).createAlarm(this, 'DLQAlarm', {\n        alarmName: `${this.node.path}/DLQNotEmpty`,\n        alarmDescription: [\n          'The dead-letter queue for the Ingestion function is not empty!',\n          '',\n          `RunBook: ${RUNBOOK_URL}`,\n          '',\n          `Direct link to the queue: ${sqsQueueUrl(this.deadLetterQueue)}`,\n          `Direct link to the function: ${lambdaFunctionUrl(this.function)}`,\n        ].join('\\n'),\n        comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: 1,\n        threshold: 1,\n        // SQS does not emit metrics if the queue has been empty for a while, which is GOOD.\n        treatMissingData: TreatMissingData.NOT_BREACHING,\n      }),\n    );\n    props.monitoring.addHighSeverityAlarm(\n      'Ingestion failures',\n      this.function.metricErrors().createAlarm(this, 'FailureAlarm', {\n        alarmName: `${this.node.path}/Failure`,\n        alarmDescription: [\n          'The Ingestion function is failing!',\n          '',\n          `RunBook: ${RUNBOOK_URL}`,\n          '',\n          `Direct link to the function: ${lambdaFunctionUrl(this.function)}`,\n        ].join('\\n'),\n        comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        evaluationPeriods: 2,\n        threshold: 1,\n        // Lambda only emits metrics when the function is invoked. No invokation => no errors.\n        treatMissingData: TreatMissingData.NOT_BREACHING,\n      }),\n    );\n  }\n\n  public metricFoundLicenseFile(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.FOUND_LICENSE_FILE,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricIneligibleLicense(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.INELIGIBLE_LICENSE,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricInvalidAssembly(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.INVALID_ASSEMBLY,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  public metricInvalidTarball(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.INVALID_TARBALL,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n\n  /**\n   * This metrics is the total count of packages that were rejected due to\n   * mismatched identity (name, version, license) between the `package.json`\n   * file and te `.jsii` attribute.\n   */\n  public metricMismatchedIdentityRejections(opts?: MetricOptions): Metric {\n    return new Metric({\n      period: Duration.minutes(5),\n      statistic: Statistic.SUM,\n      ...opts,\n      metricName: MetricName.MISMATCHED_IDENTITY_REJECTIONS,\n      namespace: METRICS_NAMESPACE,\n    });\n  }\n}\n"]}
@@ -25,6 +25,18 @@ var __markAsModule = (target) => __defProp(target, "__esModule", { value: true }
25
25
  var __require = typeof require !== "undefined" ? require : (x) => {
26
26
  throw new Error('Dynamic require of "' + x + '" is not supported');
27
27
  };
28
+ var __objRest = (source, exclude) => {
29
+ var target = {};
30
+ for (var prop in source)
31
+ if (__hasOwnProp.call(source, prop) && exclude.indexOf(prop) < 0)
32
+ target[prop] = source[prop];
33
+ if (source != null && __getOwnPropSymbols)
34
+ for (var prop of __getOwnPropSymbols(source)) {
35
+ if (exclude.indexOf(prop) < 0 && __propIsEnum.call(source, prop))
36
+ target[prop] = source[prop];
37
+ }
38
+ return target;
39
+ };
28
40
  var __commonJS = (cb, mod) => function __require2() {
29
41
  return mod || (0, cb[Object.keys(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports;
30
42
  };
@@ -16114,6 +16126,35 @@ function integrity(input, tarball, alg = ((_b) => (_b = ((_a) => (_a = input.int
16114
16126
  return `${alg}-${hash.digest("base64")}`;
16115
16127
  }
16116
16128
 
16129
+ // src/package-tag/index.ts
16130
+ var TagConditionLogicType;
16131
+ (function(TagConditionLogicType2) {
16132
+ TagConditionLogicType2["AND"] = "AND";
16133
+ TagConditionLogicType2["OR"] = "OR";
16134
+ TagConditionLogicType2["NOT"] = "NOT";
16135
+ TagConditionLogicType2["EQUALS"] = "EQUALS";
16136
+ })(TagConditionLogicType || (TagConditionLogicType = {}));
16137
+
16138
+ // src/backend/shared/tags.ts
16139
+ function isTagApplicable(config, pkg) {
16140
+ var _a, _b, _c, _d, _e, _f;
16141
+ if (config.type === TagConditionLogicType.AND) {
16142
+ return (_b = (_a = config.children) == null ? void 0 : _a.reduce((accum, cond) => accum && isTagApplicable(cond, pkg), true)) != null ? _b : true;
16143
+ } else if (config.type === TagConditionLogicType.OR) {
16144
+ return (_d = (_c = config.children) == null ? void 0 : _c.reduce((accum, cond) => accum || isTagApplicable(cond, pkg), false)) != null ? _d : true;
16145
+ } else if (config.type === TagConditionLogicType.NOT) {
16146
+ const cond = (_e = config.children) == null ? void 0 : _e[0];
16147
+ if (!cond) {
16148
+ throw new Error("NOT logical operator requires a single condition");
16149
+ }
16150
+ return !isTagApplicable(cond, pkg);
16151
+ } else if (config.type === TagConditionLogicType.EQUALS) {
16152
+ const val = (_f = config.key) == null ? void 0 : _f.reduce((accum, key) => accum ? accum[key] : void 0, pkg);
16153
+ return val === config.value;
16154
+ }
16155
+ return false;
16156
+ }
16157
+
16117
16158
  // src/backend/shared/tarball.lambda-shared.ts
16118
16159
  var import_zlib = __toModule(require("zlib"));
16119
16160
  var import_tar_stream = __toModule(require_tar_stream());
@@ -16189,6 +16230,7 @@ var handler = (0, import_aws_embedded_metrics.metricScope)((metrics) => async (e
16189
16230
  const BUCKET_NAME = requireEnv("BUCKET_NAME");
16190
16231
  const STATE_MACHINE_ARN = requireEnv("STATE_MACHINE_ARN");
16191
16232
  const PACKAGE_LINKS = requireEnv("PACKAGE_LINKS");
16233
+ const PACKAGE_TAGS = requireEnv("PACKAGE_TAGS");
16192
16234
  const result = new Array();
16193
16235
  for (const record of (_a = event.Records) != null ? _a : []) {
16194
16236
  const payload = JSON.parse(record.body);
@@ -16233,12 +16275,13 @@ var handler = (0, import_aws_embedded_metrics.metricScope)((metrics) => async (e
16233
16275
  metrics.putMetric(MetricName.INVALID_ASSEMBLY, 1, import_aws_embedded_metrics.Unit.Count);
16234
16276
  return;
16235
16277
  }
16278
+ const packageJsonObj = JSON.parse(packageJson.toString("utf-8"));
16236
16279
  const {
16237
16280
  name: packageJsonName,
16238
16281
  version: packageJsonVersion,
16239
16282
  license: packageJsonLicense,
16240
16283
  constructHub
16241
- } = JSON.parse(packageJson.toString("utf-8"));
16284
+ } = packageJsonObj;
16242
16285
  if (packageJsonName !== packageName || packageJsonVersion !== packageVersion || packageJsonLicense !== packageLicense) {
16243
16286
  console.log(`Ignoring package with mismatched name, version, and/or license (${packageJsonName}@${packageJsonVersion} is ${packageJsonLicense} !== ${packageName}@${packageVersion} is ${packageLicense})`);
16244
16287
  metrics.putMetric(MetricName.MISMATCHED_IDENTITY_REJECTIONS, 1, import_aws_embedded_metrics.Unit.Count);
@@ -16258,10 +16301,19 @@ var handler = (0, import_aws_embedded_metrics.metricScope)((metrics) => async (e
16258
16301
  }
16259
16302
  return __spreadProps(__spreadValues({}, accum), { [configKey]: pkgValue });
16260
16303
  }, {});
16304
+ const packageTagsConfig = JSON.parse(PACKAGE_TAGS);
16305
+ const packageTags = packageTagsConfig.reduce((accum, tagConfig) => {
16306
+ const _a2 = tagConfig, { condition } = _a2, tagData = __objRest(_a2, ["condition"]);
16307
+ if (isTagApplicable(condition, packageJsonObj)) {
16308
+ return [...accum, tagData];
16309
+ }
16310
+ return accum;
16311
+ }, []);
16261
16312
  const metadata = {
16262
16313
  date: payload.time,
16263
16314
  licenseText: licenseText == null ? void 0 : licenseText.toString("utf-8"),
16264
- packageLinks
16315
+ packageLinks,
16316
+ packageTags
16265
16317
  };
16266
16318
  const { assemblyKey, metadataKey, packageKey } = getObjectKeys(packageName, packageVersion);
16267
16319
  console.log(`Writing assembly at ${assemblyKey}`);