construct-hub 0.2.4 → 0.2.8
Sign up to get free protection for your applications and to get access to all the features.
- package/.jsii +519 -26
- package/API.md +301 -0
- package/README.md +51 -0
- package/lib/backend/ingestion/index.d.ts +5 -0
- package/lib/backend/ingestion/index.js +4 -3
- package/lib/backend/ingestion/ingestion.bundle/index.js +54 -2
- package/lib/backend/ingestion/ingestion.lambda.js +15 -2
- package/lib/backend/shared/tags.d.ts +7 -0
- package/lib/backend/shared/tags.js +32 -0
- package/lib/backend/transliterator/transliterator.bundle/Dockerfile +1 -1
- package/lib/construct-hub.d.ts +7 -0
- package/lib/construct-hub.js +14 -5
- package/lib/index.d.ts +1 -0
- package/lib/index.js +2 -1
- package/lib/package-sources/code-artifact.js +1 -1
- package/lib/package-sources/npmjs/npm-js-follower.bundle/index.js +1 -1
- package/lib/package-sources/npmjs/npm-js-follower.lambda.js +2 -2
- package/lib/package-sources/npmjs.js +1 -1
- package/lib/package-tag/index.d.ts +143 -0
- package/lib/package-tag/index.js +116 -0
- package/lib/spdx-license.js +1 -1
- package/lib/webapp/config.d.ts +7 -0
- package/lib/webapp/config.js +15 -3
- package/lib/webapp/index.d.ts +5 -0
- package/lib/webapp/index.js +2 -1
- package/package.json +1 -1
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: (
|
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
|
-
} =
|
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}`);
|