mirror of
https://github.com/actions/attest-build-provenance.git
synced 2025-12-14 11:22:22 +00:00
parent
e821771fc5
commit
9591ce3cf4
492
README.md
492
README.md
@ -1,231 +1,285 @@
|
|||||||
# Create a GitHub Action Using TypeScript
|
# attest-build-provenance
|
||||||
|
|
||||||
[](https://github.com/super-linter/super-linter)
|
GitHub Action to create, sign and upload a build provenance attestation for
|
||||||

|
artifacts built as part of a workflow.
|
||||||
[](https://github.com/actions/typescript-action/actions/workflows/check-dist.yml)
|
|
||||||
[](https://github.com/actions/typescript-action/actions/workflows/codeql-analysis.yml)
|
|
||||||
[](./badges/coverage.svg)
|
|
||||||
|
|
||||||
Use this template to bootstrap the creation of a TypeScript action. :rocket:
|
|
||||||
|
|
||||||
This template includes compilation support, tests, a validation workflow,
|
|
||||||
publishing, and versioning guidance.
|
|
||||||
|
|
||||||
If you are new, there's also a simpler introduction in the
|
|
||||||
[Hello world JavaScript action repository](https://github.com/actions/hello-world-javascript-action).
|
|
||||||
|
|
||||||
## Create Your Own Action
|
|
||||||
|
|
||||||
To create your own action, you can use this repository as a template! Just
|
|
||||||
follow the below instructions:
|
|
||||||
|
|
||||||
1. Click the **Use this template** button at the top of the repository
|
|
||||||
1. Select **Create a new repository**
|
|
||||||
1. Select an owner and name for your new repository
|
|
||||||
1. Click **Create repository**
|
|
||||||
1. Clone your new repository
|
|
||||||
|
|
||||||
> [!IMPORTANT]
|
|
||||||
>
|
|
||||||
> Make sure to remove or update the [`CODEOWNERS`](./CODEOWNERS) file! For
|
|
||||||
> details on how to use this file, see
|
|
||||||
> [About code owners](https://docs.github.com/en/repositories/managing-your-repositorys-settings-and-features/customizing-your-repository/about-code-owners).
|
|
||||||
|
|
||||||
## Initial Setup
|
|
||||||
|
|
||||||
After you've cloned the repository to your local machine or codespace, you'll
|
|
||||||
need to perform some initial setup steps before you can develop your action.
|
|
||||||
|
|
||||||
> [!NOTE]
|
|
||||||
>
|
|
||||||
> You'll need to have a reasonably modern version of
|
|
||||||
> [Node.js](https://nodejs.org) handy (20.x or later should work!). If you are
|
|
||||||
> using a version manager like [`nodenv`](https://github.com/nodenv/nodenv) or
|
|
||||||
> [`nvm`](https://github.com/nvm-sh/nvm), this template has a `.node-version`
|
|
||||||
> file at the root of the repository that will be used to automatically switch
|
|
||||||
> to the correct version when you `cd` into the repository. Additionally, this
|
|
||||||
> `.node-version` file is used by GitHub Actions in any `actions/setup-node`
|
|
||||||
> actions.
|
|
||||||
|
|
||||||
1. :hammer_and_wrench: Install the dependencies
|
|
||||||
|
|
||||||
```bash
|
|
||||||
npm install
|
|
||||||
```
|
|
||||||
|
|
||||||
1. :building_construction: Package the TypeScript for distribution
|
|
||||||
|
|
||||||
```bash
|
|
||||||
npm run bundle
|
|
||||||
```
|
|
||||||
|
|
||||||
1. :white_check_mark: Run the tests
|
|
||||||
|
|
||||||
```bash
|
|
||||||
$ npm test
|
|
||||||
|
|
||||||
PASS ./index.test.js
|
|
||||||
✓ throws invalid number (3ms)
|
|
||||||
✓ wait 500 ms (504ms)
|
|
||||||
✓ test runs (95ms)
|
|
||||||
|
|
||||||
...
|
|
||||||
```
|
|
||||||
|
|
||||||
## Update the Action Metadata
|
|
||||||
|
|
||||||
The [`action.yml`](action.yml) file defines metadata about your action, such as
|
|
||||||
input(s) and output(s). For details about this file, see
|
|
||||||
[Metadata syntax for GitHub Actions](https://docs.github.com/en/actions/creating-actions/metadata-syntax-for-github-actions).
|
|
||||||
|
|
||||||
When you copy this repository, update `action.yml` with the name, description,
|
|
||||||
inputs, and outputs for your action.
|
|
||||||
|
|
||||||
## Update the Action Code
|
|
||||||
|
|
||||||
The [`src/`](./src/) directory is the heart of your action! This contains the
|
|
||||||
source code that will be run when your action is invoked. You can replace the
|
|
||||||
contents of this directory with your own code.
|
|
||||||
|
|
||||||
There are a few things to keep in mind when writing your action code:
|
|
||||||
|
|
||||||
- Most GitHub Actions toolkit and CI/CD operations are processed asynchronously.
|
|
||||||
In `main.ts`, you will see that the action is run in an `async` function.
|
|
||||||
|
|
||||||
```javascript
|
|
||||||
import * as core from '@actions/core'
|
|
||||||
//...
|
|
||||||
|
|
||||||
async function run() {
|
|
||||||
try {
|
|
||||||
//...
|
|
||||||
} catch (error) {
|
|
||||||
core.setFailed(error.message)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
For more information about the GitHub Actions toolkit, see the
|
|
||||||
[documentation](https://github.com/actions/toolkit/blob/master/README.md).
|
|
||||||
|
|
||||||
So, what are you waiting for? Go ahead and start customizing your action!
|
|
||||||
|
|
||||||
1. Create a new branch
|
|
||||||
|
|
||||||
```bash
|
|
||||||
git checkout -b releases/v1
|
|
||||||
```
|
|
||||||
|
|
||||||
1. Replace the contents of `src/` with your action code
|
|
||||||
1. Add tests to `__tests__/` for your source code
|
|
||||||
1. Format, test, and build the action
|
|
||||||
|
|
||||||
```bash
|
|
||||||
npm run all
|
|
||||||
```
|
|
||||||
|
|
||||||
> [!WARNING]
|
|
||||||
>
|
|
||||||
> This step is important! It will run [`ncc`](https://github.com/vercel/ncc)
|
|
||||||
> to build the final JavaScript action code with all dependencies included.
|
|
||||||
> If you do not run this step, your action will not work correctly when it is
|
|
||||||
> used in a workflow. This step also includes the `--license` option for
|
|
||||||
> `ncc`, which will create a license file for all of the production node
|
|
||||||
> modules used in your project.
|
|
||||||
|
|
||||||
1. Commit your changes
|
|
||||||
|
|
||||||
```bash
|
|
||||||
git add .
|
|
||||||
git commit -m "My first action is ready!"
|
|
||||||
```
|
|
||||||
|
|
||||||
1. Push them to your repository
|
|
||||||
|
|
||||||
```bash
|
|
||||||
git push -u origin releases/v1
|
|
||||||
```
|
|
||||||
|
|
||||||
1. Create a pull request and get feedback on your action
|
|
||||||
1. Merge the pull request into the `main` branch
|
|
||||||
|
|
||||||
Your action is now published! :rocket:
|
|
||||||
|
|
||||||
For information about versioning your action, see
|
|
||||||
[Versioning](https://github.com/actions/toolkit/blob/master/docs/action-versioning.md)
|
|
||||||
in the GitHub Actions toolkit.
|
|
||||||
|
|
||||||
## Validate the Action
|
|
||||||
|
|
||||||
You can now validate the action by referencing it in a workflow file. For
|
|
||||||
example, [`ci.yml`](./.github/workflows/ci.yml) demonstrates how to reference an
|
|
||||||
action in the same repository.
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
id: checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Test Local Action
|
|
||||||
id: test-action
|
|
||||||
uses: ./
|
|
||||||
with:
|
|
||||||
milliseconds: 1000
|
|
||||||
|
|
||||||
- name: Print Output
|
|
||||||
id: output
|
|
||||||
run: echo "${{ steps.test-action.outputs.time }}"
|
|
||||||
```
|
|
||||||
|
|
||||||
For example workflow runs, check out the
|
|
||||||
[Actions tab](https://github.com/actions/typescript-action/actions)! :rocket:
|
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
After testing, you can create version tag(s) that developers can use to
|
Within the GitHub Actions workflow which builds some artifact you would like to
|
||||||
reference different stable versions of your action. For more information, see
|
attest,
|
||||||
[Versioning](https://github.com/actions/toolkit/blob/master/docs/action-versioning.md)
|
|
||||||
in the GitHub Actions toolkit.
|
|
||||||
|
|
||||||
To include the action in a workflow in another repository, you can use the
|
1. Ensure that the following permissions are set:
|
||||||
`uses` syntax with the `@` symbol to reference a specific branch, tag, or commit
|
|
||||||
hash.
|
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
steps:
|
permissions:
|
||||||
- name: Checkout
|
id-token: write
|
||||||
id: checkout
|
contents: write
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Test Local Action
|
|
||||||
id: test-action
|
|
||||||
uses: actions/typescript-action@v1 # Commit with the `v1` tag
|
|
||||||
with:
|
|
||||||
milliseconds: 1000
|
|
||||||
|
|
||||||
- name: Print Output
|
|
||||||
id: output
|
|
||||||
run: echo "${{ steps.test-action.outputs.time }}"
|
|
||||||
```
|
```
|
||||||
|
|
||||||
## Publishing a New Release
|
The `id-token` permission gives the action the ability to mint the OIDC token
|
||||||
|
necessary to request a Sigstore signing certificate. The `contents`
|
||||||
|
permission is necessary to persist the attestation.
|
||||||
|
|
||||||
This project includes a helper script, [`script/release`](./script/release)
|
> **NOTE**: The set of required permissions will be refined in a future
|
||||||
designed to streamline the process of tagging and pushing new releases for
|
> release.
|
||||||
GitHub Actions.
|
|
||||||
|
|
||||||
GitHub Actions allows users to select a specific version of the action to use,
|
1. After your artifact build step, add the following:
|
||||||
based on release tags. This script simplifies this process by performing the
|
|
||||||
following steps:
|
|
||||||
|
|
||||||
1. **Retrieving the latest release tag:** The script starts by fetching the most
|
```yaml
|
||||||
recent release tag by looking at the local data available in your repository.
|
- uses: actions/attest-build-provenance@main
|
||||||
1. **Prompting for a new release tag:** The user is then prompted to enter a new
|
with:
|
||||||
release tag. To assist with this, the script displays the latest release tag
|
subject-path: '${{ github.workspace }}/PATH_TO_FILE'
|
||||||
and provides a regular expression to validate the format of the new tag.
|
```
|
||||||
1. **Tagging the new release:** Once a valid new tag is entered, the script tags
|
|
||||||
the new release.
|
The `subject-path` parameter should identity the artifact for which you want
|
||||||
1. **Pushing the new tag to the remote:** Finally, the script pushes the new tag
|
to generate an attestation.
|
||||||
to the remote repository. From here, you will need to create a new release in
|
|
||||||
GitHub and users can easily reference the new tag in their workflows.
|
### What is being attested?
|
||||||
|
|
||||||
|
The generated attestation is a [SLSA provenance][2] document which captures
|
||||||
|
non-falsifiable information about the GitHub Actions run in which the subject
|
||||||
|
artifact was created:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"_type": "https://in-toto.io/Statement/v1",
|
||||||
|
"subject": [
|
||||||
|
{
|
||||||
|
"name": "some-app",
|
||||||
|
"digest": {
|
||||||
|
"sha256": "7d070f6b64d9bcc530fe99cc21eaaa4b3c364e0b2d367d7735671fa202a03b32"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"predicateType": "https://slsa.dev/provenance/v1",
|
||||||
|
"predicate": {
|
||||||
|
"buildDefinition": {
|
||||||
|
"buildType": "https://slsa-framework.github.io/github-actions-buildtypes/workflow/v1",
|
||||||
|
"externalParameters": {
|
||||||
|
"workflow": {
|
||||||
|
"ref": "refs/heads/main",
|
||||||
|
"repository": "https://github.com/user/app",
|
||||||
|
"path": ".github/workflows/build.yml"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"internalParameters": {
|
||||||
|
"github": {
|
||||||
|
"event_name": "push",
|
||||||
|
"repository_id": "706279790",
|
||||||
|
"repository_owner_id": "19792534"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"resolvedDependencies": [
|
||||||
|
{
|
||||||
|
"uri": "git+https://github.com/user/app@refs/heads/main",
|
||||||
|
"digest": {
|
||||||
|
"gitCommit": "c607ef44660b66df4d10b0dd6b01f56ec98f293f"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"runDetails": {
|
||||||
|
"builder": {
|
||||||
|
"id": "https://github.com/actions/runner/github-hosted"
|
||||||
|
},
|
||||||
|
"metadata": {
|
||||||
|
"invocationId": "https://github.com/user/app/actions/runs/1880241037/attempts/1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
The provenance statement is signed with a short-lived, [Sigstore][1]-issued
|
||||||
|
certificate.
|
||||||
|
|
||||||
|
If the repository initiating the GitHub Actions workflow is public, the public
|
||||||
|
instance of Sigstore will be used to generate the attestation signature. If the
|
||||||
|
repository is private, it will use the GitHub private Sigstore instance.
|
||||||
|
|
||||||
|
### Where does the attestation go?
|
||||||
|
|
||||||
|
On the actions summary page for a repository you'll see an "Attestations" link
|
||||||
|
which will take you to a list of all the attestations generated by workflows in
|
||||||
|
that repository.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
### How are attestations verified?
|
||||||
|
|
||||||
|
Attestations can be verified using the [gh-attestation][3] extension for the
|
||||||
|
[GitHub CLI][4].
|
||||||
|
|
||||||
|
## Customization
|
||||||
|
|
||||||
|
See [action.yml](action.yml)
|
||||||
|
|
||||||
|
### Inputs
|
||||||
|
|
||||||
|
- `subject-path` - Path to the artifact for which the provenance will be
|
||||||
|
generated.
|
||||||
|
|
||||||
|
Must specify exactly one of `subject-path` or `subject-digest`. Wildcards can
|
||||||
|
be used to identify more than one artifact.
|
||||||
|
|
||||||
|
- `subject-digest` - Digest of the subject for which the provenance will be
|
||||||
|
generated.
|
||||||
|
|
||||||
|
Only SHA-256 digests are accepted and the supplied value must be in the form
|
||||||
|
"sha256:\<hex-encoded-digest\>". Must specify exactly one of `subject-path` or
|
||||||
|
`subject-digest`.
|
||||||
|
|
||||||
|
- `subject-name` - Subject name as it should appear in the provenance statement.
|
||||||
|
|
||||||
|
Required when the subject is identified by the `subject-digest` parameter.
|
||||||
|
When attesting container images, the name should be the fully qualified image
|
||||||
|
name.
|
||||||
|
|
||||||
|
- `push-to-registry` - If true, the signed attestation is pushed to the
|
||||||
|
container registry identified by the `subject-name`. Default: `false`.
|
||||||
|
|
||||||
|
- `github-token` - Token used to make authenticated requests to the GitHub API.
|
||||||
|
Default: `${{ github.token }}`.
|
||||||
|
|
||||||
|
The supplied token must have the permissions necessary to write attestations
|
||||||
|
to the repository.
|
||||||
|
|
||||||
|
### Outputs
|
||||||
|
|
||||||
|
- `bundle-path` - The file path of JSON-serialized [Sigstore bundle][5] containing
|
||||||
|
the attestation and related verification material.
|
||||||
|
|
||||||
|
## Sample Workflows
|
||||||
|
|
||||||
|
### Identify Artifact by Path
|
||||||
|
|
||||||
|
For the basic use case, simply add the `attest-build-provenance` action to
|
||||||
|
your workflow and supply the path to the artifact for which you want to generate
|
||||||
|
build provenance.
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
name: build-with-provenance
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
- name: Build artifact
|
||||||
|
run: make some-app
|
||||||
|
- name: Attest artifact
|
||||||
|
uses: actions/attest-build-provenance@main
|
||||||
|
with:
|
||||||
|
subject-path: '${{ github.workspace }}/some-app'
|
||||||
|
```
|
||||||
|
|
||||||
|
### Identify Artifacts by Wildcard
|
||||||
|
|
||||||
|
If you are generating multiple artifacts, you can generate build provenance for
|
||||||
|
each artifact by using a wildcard in the `subject-path` input.
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
name: build-wildcard-with-provenance
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
- name: Set up Go
|
||||||
|
uses: actions/setup-go@v4
|
||||||
|
- name: Run GoReleaser
|
||||||
|
uses: goreleaser/goreleaser-action@v5
|
||||||
|
with:
|
||||||
|
distribution: goreleaser
|
||||||
|
version: latest
|
||||||
|
args: release --clean
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
- name: Attest artifact
|
||||||
|
uses: actions/attest-build-provenance@main
|
||||||
|
with:
|
||||||
|
subject-path: 'dist/**/my-bin-*'
|
||||||
|
```
|
||||||
|
|
||||||
|
For supported wildcards along with behavior and documentation, see
|
||||||
|
[@actions/glob][6] which is used internally to search for files.
|
||||||
|
|
||||||
|
### Container Image
|
||||||
|
|
||||||
|
When working with container images you may not have a `subject-path` value you
|
||||||
|
can supply. In this case you can invoke the action with the `subject-name` and
|
||||||
|
`subject-digest` inputs.
|
||||||
|
|
||||||
|
If you want to publish the attestation to the container registry with the
|
||||||
|
`push-to-registry` option, it is important that the `subject-name` specify the
|
||||||
|
fully-qualified image name (e.g. "ghcr.io/user/app" or
|
||||||
|
"acme.azurecr.io/user/app"). Do NOT include a tag as part of the image name --
|
||||||
|
the specific image being attested is identified by the supplied digest.
|
||||||
|
|
||||||
|
> **NOTE**: When pushing to Docker Hub, please use "index.docker.io" as the
|
||||||
|
> registry portion of the image name.
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
name: build-image-with-provenance
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [main]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
packages: write
|
||||||
|
contents: write
|
||||||
|
env:
|
||||||
|
REGISTRY: ghcr.io
|
||||||
|
IMAGE_NAME: ${{ github.repository }}
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
- name: Login to GitHub Container Registry
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
registry: ${{ env.REGISTRY }}
|
||||||
|
username: ${{ github.actor }}
|
||||||
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
- name: Build and push image
|
||||||
|
id: push
|
||||||
|
uses: docker/build-push-action@v5.0.0
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
push: true
|
||||||
|
tags: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:latest
|
||||||
|
- name: Attest image
|
||||||
|
uses: actions/attest-build-provenance@main
|
||||||
|
with:
|
||||||
|
subject-name: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
|
||||||
|
subject-digest: ${{ steps.push.outputs.digest }}
|
||||||
|
push-to-registry: true
|
||||||
|
```
|
||||||
|
|
||||||
|
[1]: https://www.sigstore.dev/
|
||||||
|
[2]: https://slsa.dev/spec/v1.0/provenance
|
||||||
|
[3]: https://github.com/github-early-access/gh-attestation
|
||||||
|
[4]: https://cli.github.com/
|
||||||
|
[5]:
|
||||||
|
https://github.com/sigstore/protobuf-specs/blob/main/protos/sigstore_bundle.proto
|
||||||
|
[6]: https://github.com/actions/toolkit/tree/main/packages/glob#patterns
|
||||||
|
|||||||
222
package-lock.json
generated
222
package-lock.json
generated
@ -24,6 +24,7 @@
|
|||||||
"eslint-plugin-jsonc": "^2.13.0",
|
"eslint-plugin-jsonc": "^2.13.0",
|
||||||
"eslint-plugin-prettier": "^5.1.3",
|
"eslint-plugin-prettier": "^5.1.3",
|
||||||
"jest": "^29.7.0",
|
"jest": "^29.7.0",
|
||||||
|
"markdownlint-cli": "^0.39.0",
|
||||||
"prettier": "^3.2.5",
|
"prettier": "^3.2.5",
|
||||||
"prettier-eslint": "^16.3.0",
|
"prettier-eslint": "^16.3.0",
|
||||||
"ts-jest": "^29.1.2",
|
"ts-jest": "^29.1.2",
|
||||||
@ -2761,6 +2762,15 @@
|
|||||||
"version": "1.1.4",
|
"version": "1.1.4",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/commander": {
|
||||||
|
"version": "11.1.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/commander/-/commander-11.1.0.tgz",
|
||||||
|
"integrity": "sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ==",
|
||||||
|
"dev": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=16"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/common-tags": {
|
"node_modules/common-tags": {
|
||||||
"version": "1.8.2",
|
"version": "1.8.2",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@ -2844,6 +2854,15 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/deep-extend": {
|
||||||
|
"version": "0.6.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
|
||||||
|
"integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
|
||||||
|
"dev": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=4.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/deep-is": {
|
"node_modules/deep-is": {
|
||||||
"version": "0.1.4",
|
"version": "0.1.4",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@ -2959,6 +2978,18 @@
|
|||||||
"iconv-lite": "^0.6.2"
|
"iconv-lite": "^0.6.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/entities": {
|
||||||
|
"version": "4.5.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz",
|
||||||
|
"integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==",
|
||||||
|
"dev": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.12"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/fb55/entities?sponsor=1"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/err-code": {
|
"node_modules/err-code": {
|
||||||
"version": "2.0.3",
|
"version": "2.0.3",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
@ -4065,6 +4096,18 @@
|
|||||||
"node": ">=8.0.0"
|
"node": ">=8.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/get-stdin": {
|
||||||
|
"version": "9.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-9.0.0.tgz",
|
||||||
|
"integrity": "sha512-dVKBjfWisLAicarI2Sf+JuBE/DghV4UzNAVe9yhEJuzeREd3JhOTE9cUaJTeSa77fsbQUK3pcOpJfM59+VKZaA==",
|
||||||
|
"dev": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=12"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/sindresorhus"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/get-stream": {
|
"node_modules/get-stream": {
|
||||||
"version": "6.0.1",
|
"version": "6.0.1",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@ -4333,9 +4376,10 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/ignore": {
|
"node_modules/ignore": {
|
||||||
"version": "5.2.4",
|
"version": "5.3.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.1.tgz",
|
||||||
|
"integrity": "sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">= 4"
|
"node": ">= 4"
|
||||||
}
|
}
|
||||||
@ -4401,6 +4445,15 @@
|
|||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "ISC"
|
"license": "ISC"
|
||||||
},
|
},
|
||||||
|
"node_modules/ini": {
|
||||||
|
"version": "4.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/ini/-/ini-4.1.1.tgz",
|
||||||
|
"integrity": "sha512-QQnnxNyfvmHFIsj7gkPcYymR8Jdw/o7mp5ZFihxn6h8Ci6fh3Dx4E1gPjpQEpIuPo9XVNY/ZUwh4BPMjGyL01g==",
|
||||||
|
"dev": true,
|
||||||
|
"engines": {
|
||||||
|
"node": "^14.17.0 || ^16.13.0 || >=18.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/internal-slot": {
|
"node_modules/internal-slot": {
|
||||||
"version": "1.0.5",
|
"version": "1.0.5",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@ -5370,6 +5423,12 @@
|
|||||||
"url": "https://github.com/sponsors/ota-meshi"
|
"url": "https://github.com/sponsors/ota-meshi"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/jsonc-parser": {
|
||||||
|
"version": "3.2.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/jsonc-parser/-/jsonc-parser-3.2.1.tgz",
|
||||||
|
"integrity": "sha512-AilxAyFOAcK5wA1+LeaySVBrHsGQvUFCDWXKpZjzaL0PqW+xfBOttn8GNtWKFWqneyMZj41MWF9Kl6iPWLwgOA==",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"node_modules/jsx-ast-utils": {
|
"node_modules/jsx-ast-utils": {
|
||||||
"version": "3.3.5",
|
"version": "3.3.5",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@ -5438,6 +5497,15 @@
|
|||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/linkify-it": {
|
||||||
|
"version": "5.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/linkify-it/-/linkify-it-5.0.0.tgz",
|
||||||
|
"integrity": "sha512-5aHCbzQRADcdP+ATqnDuhhJ/MRIqDkZX5pyjFHRRysS8vZ5AbqGEoFIb6pYHPZ+L/OC2Lc+xT8uHVVR5CAK/wQ==",
|
||||||
|
"dev": true,
|
||||||
|
"dependencies": {
|
||||||
|
"uc.micro": "^2.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/locate-path": {
|
"node_modules/locate-path": {
|
||||||
"version": "6.0.0",
|
"version": "6.0.0",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@ -5621,6 +5689,126 @@
|
|||||||
"tmpl": "1.0.5"
|
"tmpl": "1.0.5"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/markdown-it": {
|
||||||
|
"version": "14.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/markdown-it/-/markdown-it-14.0.0.tgz",
|
||||||
|
"integrity": "sha512-seFjF0FIcPt4P9U39Bq1JYblX0KZCjDLFFQPHpL5AzHpqPEKtosxmdq/LTVZnjfH7tjt9BxStm+wXcDBNuYmzw==",
|
||||||
|
"dev": true,
|
||||||
|
"dependencies": {
|
||||||
|
"argparse": "^2.0.1",
|
||||||
|
"entities": "^4.4.0",
|
||||||
|
"linkify-it": "^5.0.0",
|
||||||
|
"mdurl": "^2.0.0",
|
||||||
|
"punycode.js": "^2.3.1",
|
||||||
|
"uc.micro": "^2.0.0"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"markdown-it": "bin/markdown-it.mjs"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/markdownlint": {
|
||||||
|
"version": "0.33.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/markdownlint/-/markdownlint-0.33.0.tgz",
|
||||||
|
"integrity": "sha512-4lbtT14A3m0LPX1WS/3d1m7Blg+ZwiLq36WvjQqFGsX3Gik99NV+VXp/PW3n+Q62xyPdbvGOCfjPqjW+/SKMig==",
|
||||||
|
"dev": true,
|
||||||
|
"dependencies": {
|
||||||
|
"markdown-it": "14.0.0",
|
||||||
|
"markdownlint-micromark": "0.1.8"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/DavidAnson"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/markdownlint-cli": {
|
||||||
|
"version": "0.39.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/markdownlint-cli/-/markdownlint-cli-0.39.0.tgz",
|
||||||
|
"integrity": "sha512-ZuFN7Xpsbn1Nbp0YYkeLOfXOMOfLQBik2lKRy8pVI/llmKQ2uW7x+8k5OMgF6o7XCsTDSYC/OOmeJ+3qplvnJQ==",
|
||||||
|
"dev": true,
|
||||||
|
"dependencies": {
|
||||||
|
"commander": "~11.1.0",
|
||||||
|
"get-stdin": "~9.0.0",
|
||||||
|
"glob": "~10.3.10",
|
||||||
|
"ignore": "~5.3.0",
|
||||||
|
"js-yaml": "^4.1.0",
|
||||||
|
"jsonc-parser": "~3.2.1",
|
||||||
|
"markdownlint": "~0.33.0",
|
||||||
|
"minimatch": "~9.0.3",
|
||||||
|
"run-con": "~1.3.2"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"markdownlint": "markdownlint.js"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/markdownlint-cli/node_modules/brace-expansion": {
|
||||||
|
"version": "2.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz",
|
||||||
|
"integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==",
|
||||||
|
"dev": true,
|
||||||
|
"dependencies": {
|
||||||
|
"balanced-match": "^1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/markdownlint-cli/node_modules/glob": {
|
||||||
|
"version": "10.3.10",
|
||||||
|
"resolved": "https://registry.npmjs.org/glob/-/glob-10.3.10.tgz",
|
||||||
|
"integrity": "sha512-fa46+tv1Ak0UPK1TOy/pZrIybNNt4HCv7SDzwyfiOZkvZLEbjsZkJBPtDHVshZjbecAoAGSC20MjLDG/qr679g==",
|
||||||
|
"dev": true,
|
||||||
|
"dependencies": {
|
||||||
|
"foreground-child": "^3.1.0",
|
||||||
|
"jackspeak": "^2.3.5",
|
||||||
|
"minimatch": "^9.0.1",
|
||||||
|
"minipass": "^5.0.0 || ^6.0.2 || ^7.0.0",
|
||||||
|
"path-scurry": "^1.10.1"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"glob": "dist/esm/bin.mjs"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=16 || 14 >=14.17"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/isaacs"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/markdownlint-cli/node_modules/minimatch": {
|
||||||
|
"version": "9.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz",
|
||||||
|
"integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==",
|
||||||
|
"dev": true,
|
||||||
|
"dependencies": {
|
||||||
|
"brace-expansion": "^2.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=16 || 14 >=14.17"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/isaacs"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/markdownlint-micromark": {
|
||||||
|
"version": "0.1.8",
|
||||||
|
"resolved": "https://registry.npmjs.org/markdownlint-micromark/-/markdownlint-micromark-0.1.8.tgz",
|
||||||
|
"integrity": "sha512-1ouYkMRo9/6gou9gObuMDnvZM8jC/ly3QCFQyoSPCS2XV1ZClU0xpKbL1Ar3bWWRT1RnBZkWUEiNKrI2CwiBQA==",
|
||||||
|
"dev": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=16"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/DavidAnson"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/mdurl": {
|
||||||
|
"version": "2.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/mdurl/-/mdurl-2.0.0.tgz",
|
||||||
|
"integrity": "sha512-Lf+9+2r+Tdp5wXDXC4PcIBjTDtq4UKjCPMQhKIuzpJNW0b96kVqSwW0bT7FhRSfmAiFYgP+SCRvdrDozfh0U5w==",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"node_modules/merge-stream": {
|
"node_modules/merge-stream": {
|
||||||
"version": "2.0.0",
|
"version": "2.0.0",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@ -6322,6 +6510,15 @@
|
|||||||
"node": ">=6"
|
"node": ">=6"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/punycode.js": {
|
||||||
|
"version": "2.3.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/punycode.js/-/punycode.js-2.3.1.tgz",
|
||||||
|
"integrity": "sha512-uxFIHU0YlHYhDQtV4R9J6a52SLx28BCjT+4ieh7IGbgwVJWO+km431c4yRlREUAsAmt/uMjQUyQHNEPf0M39CA==",
|
||||||
|
"dev": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/pure-rand": {
|
"node_modules/pure-rand": {
|
||||||
"version": "6.0.3",
|
"version": "6.0.3",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@ -6476,6 +6673,21 @@
|
|||||||
"url": "https://github.com/sponsors/isaacs"
|
"url": "https://github.com/sponsors/isaacs"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/run-con": {
|
||||||
|
"version": "1.3.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/run-con/-/run-con-1.3.2.tgz",
|
||||||
|
"integrity": "sha512-CcfE+mYiTcKEzg0IqS08+efdnH0oJ3zV0wSUFBNrMHMuxCtXvBCLzCJHatwuXDcu/RlhjTziTo/a1ruQik6/Yg==",
|
||||||
|
"dev": true,
|
||||||
|
"dependencies": {
|
||||||
|
"deep-extend": "^0.6.0",
|
||||||
|
"ini": "~4.1.0",
|
||||||
|
"minimist": "^1.2.8",
|
||||||
|
"strip-json-comments": "~3.1.1"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"run-con": "cli.js"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/run-parallel": {
|
"node_modules/run-parallel": {
|
||||||
"version": "1.2.0",
|
"version": "1.2.0",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
@ -7177,6 +7389,12 @@
|
|||||||
"node": ">=14.17"
|
"node": ">=14.17"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/uc.micro": {
|
||||||
|
"version": "2.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/uc.micro/-/uc.micro-2.0.0.tgz",
|
||||||
|
"integrity": "sha512-DffL94LsNOccVn4hyfRe5rdKa273swqeA5DJpMOeFmEn1wCDc7nAbbB0gXlgBCL7TNzeTv6G7XVWzan7iJtfig==",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"node_modules/unbox-primitive": {
|
"node_modules/unbox-primitive": {
|
||||||
"version": "1.0.2",
|
"version": "1.0.2",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
|
|||||||
@ -28,7 +28,9 @@
|
|||||||
"ci-test": "jest",
|
"ci-test": "jest",
|
||||||
"format:write": "prettier --write **/*.ts",
|
"format:write": "prettier --write **/*.ts",
|
||||||
"format:check": "prettier --check **/*.ts",
|
"format:check": "prettier --check **/*.ts",
|
||||||
"lint": "npx eslint . -c ./.github/linters/.eslintrc.yml",
|
"lint:eslint": "npx eslint . -c ./.github/linters/.eslintrc.yml",
|
||||||
|
"lint:markdown": "npx markdownlint --config .github/linters/.markdown-lint.yml \"*.md\"",
|
||||||
|
"lint": "npm run lint:eslint && npm run lint:markdown",
|
||||||
"package": "ncc build src/index.ts --license licenses.txt",
|
"package": "ncc build src/index.ts --license licenses.txt",
|
||||||
"package:watch": "npm run package -- --watch",
|
"package:watch": "npm run package -- --watch",
|
||||||
"test": "jest",
|
"test": "jest",
|
||||||
@ -80,6 +82,7 @@
|
|||||||
"eslint-plugin-jsonc": "^2.13.0",
|
"eslint-plugin-jsonc": "^2.13.0",
|
||||||
"eslint-plugin-prettier": "^5.1.3",
|
"eslint-plugin-prettier": "^5.1.3",
|
||||||
"jest": "^29.7.0",
|
"jest": "^29.7.0",
|
||||||
|
"markdownlint-cli": "^0.39.0",
|
||||||
"prettier": "^3.2.5",
|
"prettier": "^3.2.5",
|
||||||
"prettier-eslint": "^16.3.0",
|
"prettier-eslint": "^16.3.0",
|
||||||
"ts-jest": "^29.1.2",
|
"ts-jest": "^29.1.2",
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user