Initial commit
This commit is contained in:
commit
688a7bd017
11
.eslintrc.js
Normal file
11
.eslintrc.js
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
module.exports = {
|
||||||
|
root: true,
|
||||||
|
parser: '@typescript-eslint/parser',
|
||||||
|
plugins: ['@typescript-eslint'],
|
||||||
|
extends: [
|
||||||
|
'eslint:recommended',
|
||||||
|
'plugin:@typescript-eslint/eslint-recommended',
|
||||||
|
'plugin:@typescript-eslint/recommended',
|
||||||
|
'plugin:prettier/recommended',
|
||||||
|
],
|
||||||
|
};
|
45
.github/workflows/test.yaml
vendored
Normal file
45
.github/workflows/test.yaml
vendored
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
name: 'test'
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- 'main'
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- 'main'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
run:
|
||||||
|
name: 'test'
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
contents: read
|
||||||
|
runs-on: '${{ matrix.operating-system }}'
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
operating-system:
|
||||||
|
- 'ubuntu-latest'
|
||||||
|
- 'windows-latest'
|
||||||
|
- 'macos-latest'
|
||||||
|
steps:
|
||||||
|
- uses: 'actions/checkout@v2'
|
||||||
|
|
||||||
|
- uses: 'actions/setup-node@master'
|
||||||
|
with:
|
||||||
|
node-version: '12.x'
|
||||||
|
|
||||||
|
- id: 'integration'
|
||||||
|
name: 'integration'
|
||||||
|
uses: './'
|
||||||
|
with:
|
||||||
|
workload_identity_provider: 'projects/469401941463/locations/global/workloadIdentityPools/github-actions/providers/github-oidc-auth-google-cloud'
|
||||||
|
service_account: 'github-secret-accessor@actions-oidc-test.iam.gserviceaccount.com'
|
||||||
|
|
||||||
|
- name: 'npm install'
|
||||||
|
run: 'npm install'
|
||||||
|
|
||||||
|
- name: 'npm lint'
|
||||||
|
run: 'npm run lint'
|
||||||
|
|
||||||
|
- name: 'npm test'
|
||||||
|
run: 'npm run test'
|
45
.gitignore
vendored
Normal file
45
.gitignore
vendored
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
node_modules/
|
||||||
|
runner/
|
||||||
|
|
||||||
|
# Rest of the file pulled from https://github.com/github/gitignore/blob/master/Node.gitignore
|
||||||
|
# Logs
|
||||||
|
logs
|
||||||
|
*.log
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
lerna-debug.log*
|
||||||
|
|
||||||
|
# Diagnostic reports (https://nodejs.org/api/report.html)
|
||||||
|
report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json
|
||||||
|
|
||||||
|
# Runtime data
|
||||||
|
pids
|
||||||
|
*.pid
|
||||||
|
*.seed
|
||||||
|
*.pid.lock
|
||||||
|
|
||||||
|
# Directory for instrumented libs generated by jscoverage/JSCover
|
||||||
|
lib-cov
|
||||||
|
|
||||||
|
# Coverage directory used by tools like istanbul
|
||||||
|
coverage
|
||||||
|
*.lcov
|
||||||
|
|
||||||
|
# TypeScript v1 declaration files
|
||||||
|
typings/
|
||||||
|
|
||||||
|
# TypeScript cache
|
||||||
|
*.tsbuildinfo
|
||||||
|
|
||||||
|
# Optional npm cache directory
|
||||||
|
.npm
|
||||||
|
|
||||||
|
# Optional eslint cache
|
||||||
|
.eslintcache
|
||||||
|
|
||||||
|
# Optional REPL history
|
||||||
|
.node_repl_history
|
||||||
|
|
||||||
|
# Output of 'npm pack'
|
||||||
|
*.tgz
|
14
.prettierrc.js
Normal file
14
.prettierrc.js
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
module.exports = {
|
||||||
|
arrowParens: 'always',
|
||||||
|
bracketSpacing: true,
|
||||||
|
endOfLine: 'auto',
|
||||||
|
jsxBracketSameLine: true,
|
||||||
|
jsxSingleQuote: true,
|
||||||
|
printWidth: 100,
|
||||||
|
quoteProps: 'consistent',
|
||||||
|
semi: true,
|
||||||
|
singleQuote: true,
|
||||||
|
tabWidth: 2,
|
||||||
|
trailingComma: 'all',
|
||||||
|
useTabs: false,
|
||||||
|
};
|
201
LICENSE
Normal file
201
LICENSE
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction,
|
||||||
|
and distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by
|
||||||
|
the copyright owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
|
other entities that control, are controlled by, or are under common
|
||||||
|
control with that entity. For the purposes of this definition,
|
||||||
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
|
direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity
|
||||||
|
exercising permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications,
|
||||||
|
including but not limited to software source code, documentation
|
||||||
|
source, and configuration files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical
|
||||||
|
transformation or translation of a Source form, including but
|
||||||
|
not limited to compiled object code, generated documentation,
|
||||||
|
and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or
|
||||||
|
Object form, made available under the License, as indicated by a
|
||||||
|
copyright notice that is included in or attached to the work
|
||||||
|
(an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object
|
||||||
|
form, that is based on (or derived from) the Work and for which the
|
||||||
|
editorial revisions, annotations, elaborations, or other modifications
|
||||||
|
represent, as a whole, an original work of authorship. For the purposes
|
||||||
|
of this License, Derivative Works shall not include works that remain
|
||||||
|
separable from, or merely link (or bind by name) to the interfaces of,
|
||||||
|
the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including
|
||||||
|
the original version of the Work and any modifications or additions
|
||||||
|
to that Work or Derivative Works thereof, that is intentionally
|
||||||
|
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||||
|
or by an individual or Legal Entity authorized to submit on behalf of
|
||||||
|
the copyright owner. For the purposes of this definition, "submitted"
|
||||||
|
means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems,
|
||||||
|
and issue tracking systems that are managed by, or on behalf of, the
|
||||||
|
Licensor for the purpose of discussing and improving the Work, but
|
||||||
|
excluding communication that is conspicuously marked or otherwise
|
||||||
|
designated in writing by the copyright owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||||
|
on behalf of whom a Contribution has been received by Licensor and
|
||||||
|
subsequently incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the
|
||||||
|
Work and such Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
(except as stated in this section) patent license to make, have made,
|
||||||
|
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||||
|
where such license applies only to those patent claims licensable
|
||||||
|
by such Contributor that are necessarily infringed by their
|
||||||
|
Contribution(s) alone or by combination of their Contribution(s)
|
||||||
|
with the Work to which such Contribution(s) was submitted. If You
|
||||||
|
institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||||
|
or a Contribution incorporated within the Work constitutes direct
|
||||||
|
or contributory patent infringement, then any patent licenses
|
||||||
|
granted to You under this License for that Work shall terminate
|
||||||
|
as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution. You may reproduce and distribute copies of the
|
||||||
|
Work or Derivative Works thereof in any medium, with or without
|
||||||
|
modifications, and in Source or Object form, provided that You
|
||||||
|
meet the following conditions:
|
||||||
|
|
||||||
|
(a) You must give any other recipients of the Work or
|
||||||
|
Derivative Works a copy of this License; and
|
||||||
|
|
||||||
|
(b) You must cause any modified files to carry prominent notices
|
||||||
|
stating that You changed the files; and
|
||||||
|
|
||||||
|
(c) You must retain, in the Source form of any Derivative Works
|
||||||
|
that You distribute, all copyright, patent, trademark, and
|
||||||
|
attribution notices from the Source form of the Work,
|
||||||
|
excluding those notices that do not pertain to any part of
|
||||||
|
the Derivative Works; and
|
||||||
|
|
||||||
|
(d) If the Work includes a "NOTICE" text file as part of its
|
||||||
|
distribution, then any Derivative Works that You distribute must
|
||||||
|
include a readable copy of the attribution notices contained
|
||||||
|
within such NOTICE file, excluding those notices that do not
|
||||||
|
pertain to any part of the Derivative Works, in at least one
|
||||||
|
of the following places: within a NOTICE text file distributed
|
||||||
|
as part of the Derivative Works; within the Source form or
|
||||||
|
documentation, if provided along with the Derivative Works; or,
|
||||||
|
within a display generated by the Derivative Works, if and
|
||||||
|
wherever such third-party notices normally appear. The contents
|
||||||
|
of the NOTICE file are for informational purposes only and
|
||||||
|
do not modify the License. You may add Your own attribution
|
||||||
|
notices within Derivative Works that You distribute, alongside
|
||||||
|
or as an addendum to the NOTICE text from the Work, provided
|
||||||
|
that such additional attribution notices cannot be construed
|
||||||
|
as modifying the License.
|
||||||
|
|
||||||
|
You may add Your own copyright statement to Your modifications and
|
||||||
|
may provide additional or different license terms and conditions
|
||||||
|
for use, reproduction, or distribution of Your modifications, or
|
||||||
|
for any such Derivative Works as a whole, provided Your use,
|
||||||
|
reproduction, and distribution of the Work otherwise complies with
|
||||||
|
the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||||
|
any Contribution intentionally submitted for inclusion in the Work
|
||||||
|
by You to the Licensor shall be under the terms and conditions of
|
||||||
|
this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify
|
||||||
|
the terms of any separate license agreement you may have executed
|
||||||
|
with Licensor regarding such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks. This License does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or product names of the Licensor,
|
||||||
|
except as required for reasonable and customary use in describing the
|
||||||
|
origin of the Work and reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||||
|
agreed to in writing, Licensor provides the Work (and each
|
||||||
|
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
implied, including, without limitation, any warranties or conditions
|
||||||
|
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||||
|
appropriateness of using or redistributing the Work and assume any
|
||||||
|
risks associated with Your exercise of permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability. In no event and under no legal theory,
|
||||||
|
whether in tort (including negligence), contract, or otherwise,
|
||||||
|
unless required by applicable law (such as deliberate and grossly
|
||||||
|
negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special,
|
||||||
|
incidental, or consequential damages of any character arising as a
|
||||||
|
result of this License or out of the use or inability to use the
|
||||||
|
Work (including but not limited to damages for loss of goodwill,
|
||||||
|
work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses), even if such Contributor
|
||||||
|
has been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability. While redistributing
|
||||||
|
the Work or Derivative Works thereof, You may choose to offer,
|
||||||
|
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||||
|
or other liability obligations and/or rights consistent with this
|
||||||
|
License. However, in accepting such obligations, You may act only
|
||||||
|
on Your own behalf and on Your sole responsibility, not on behalf
|
||||||
|
of any other Contributor, and only if You agree to indemnify,
|
||||||
|
defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
APPENDIX: How to apply the Apache License to your work.
|
||||||
|
|
||||||
|
To apply the Apache License to your work, attach the following
|
||||||
|
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||||
|
replaced with your own identifying information. (Don't include
|
||||||
|
the brackets!) The text should be enclosed in the appropriate
|
||||||
|
comment syntax for the file format. We also recommend that a
|
||||||
|
file or class name and description of purpose be included on the
|
||||||
|
same "printed page" as the copyright notice for easier
|
||||||
|
identification within third-party archives.
|
||||||
|
|
||||||
|
Copyright [yyyy] [name of copyright owner]
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
201
README.md
Normal file
201
README.md
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
# oidc-auth-google-cloud
|
||||||
|
|
||||||
|
This GitHub Action exchanges a GitHub Actions OIDC token into a Google Cloud
|
||||||
|
access token using [Workload Identity Federation][wif]. This obviates the need
|
||||||
|
to export a long-lived Google Cloud service account key and establishes a trust
|
||||||
|
delegation relationship between a particular GitHub Actions workflow invocation
|
||||||
|
and permissions on Google Cloud.
|
||||||
|
|
||||||
|
#### Previously
|
||||||
|
|
||||||
|
1. Create a Google Cloud service account and grant IAM permissions
|
||||||
|
1. Export the long-lived JSON service account key
|
||||||
|
1. Upload the JSON service account key to a GitHub secret
|
||||||
|
|
||||||
|
#### With Workload Identity Federation
|
||||||
|
|
||||||
|
1. Create a Google Cloud service account and grant IAM permissions
|
||||||
|
1. Create and configure a Workload Identity Provider for GitHub
|
||||||
|
1. Exchange the GitHub Actions OIDC token for a short-lived Google Cloud access
|
||||||
|
token
|
||||||
|
|
||||||
|
## Prerequisites
|
||||||
|
|
||||||
|
- This action requires you to create and configure a Google Cloud Workload
|
||||||
|
Identity Provider. See [#setup] for instructions.
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
jobs:
|
||||||
|
run:
|
||||||
|
# ...
|
||||||
|
|
||||||
|
# Add "id-token" with the intended permissions.
|
||||||
|
permissions:
|
||||||
|
id-token: write
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- id: 'google-cloud-auth'
|
||||||
|
name: 'Authenticate to Google Cloud'
|
||||||
|
uses: 'github.com/sethvargo/oidc-auth-google-cloud'
|
||||||
|
with:
|
||||||
|
workload_identity_provider: 'projects/123456789/locations/global/workloadIdentityPools/my-pool/providers/my-provider'
|
||||||
|
service_account: 'my-service-account@my-project.iam.gserviceaccount.com'
|
||||||
|
|
||||||
|
# Example of using the output:
|
||||||
|
- id: 'access-secret'
|
||||||
|
run: |-
|
||||||
|
curl https://secretmanager.googleapis.com/v1/projects/my-project/secrets/my-secret/versions/1:access \
|
||||||
|
--header "Authorization: Bearer ${{ steps.integration.outputs.access_token }}"
|
||||||
|
```
|
||||||
|
|
||||||
|
## Inputs
|
||||||
|
|
||||||
|
- `workload_identity_provider`: (Required) The full identifier of the Workload
|
||||||
|
Identity Provider, including the project number, pool name, and provider
|
||||||
|
name. This must be the full identifier which includes all parts, for
|
||||||
|
example:
|
||||||
|
|
||||||
|
```text
|
||||||
|
projects/123456789/locations/global/workloadIdentityPools/my-pool/providers/my-provider
|
||||||
|
```
|
||||||
|
|
||||||
|
- `service_account`: (Required) Email address or unique identifier of the
|
||||||
|
Google Cloud service account for which to generate credentials. For example:
|
||||||
|
|
||||||
|
```text
|
||||||
|
my-service-account@my-project.iam.gserviceaccount.com
|
||||||
|
```
|
||||||
|
|
||||||
|
- `audience`: (Optional) The value for the audience (`aud`) parameter in the
|
||||||
|
generated GitHub Actions OIDC token. At present, the only valid value is
|
||||||
|
`"sigstore"`, but this variable exists in case custom values are permitted
|
||||||
|
in the future. The default value is `"sigstore"`.
|
||||||
|
|
||||||
|
- `delegates`: (Optional) List of additional service account emails or unique
|
||||||
|
identities to use for impersonation in the chain. By default there are no
|
||||||
|
delegates.
|
||||||
|
|
||||||
|
- `lifetime`: (Optional) Desired lifetime duration of the access token, in
|
||||||
|
seconds. This must be specified as the number of seconds with a trailing "s"
|
||||||
|
(e.g. 30s). The default value is 1 hour (3600s).
|
||||||
|
|
||||||
|
## Outputs
|
||||||
|
|
||||||
|
- `access_token`: The authenticated Google Cloud access token for calling
|
||||||
|
other Google Cloud APIs.
|
||||||
|
|
||||||
|
- `expiration`: The RFC3339 UTC "Zulu" format timestamp when the token
|
||||||
|
expires.
|
||||||
|
|
||||||
|
## Setup
|
||||||
|
|
||||||
|
To exchange a GitHub Actions OIDC token for a Google Cloud access token, you
|
||||||
|
must create and configure a Workload Identity Provider. These instructions use
|
||||||
|
the [gcloud][gcloud] command-line tool.
|
||||||
|
|
||||||
|
1. Create or use an existing Google Cloud project. You must have privileges to
|
||||||
|
create Workload Identity Pools, Workload Identity Providers, and to manage
|
||||||
|
Service Accounts and IAM permissions. Save your project ID as an environment
|
||||||
|
variable. The rest of these steps assume this environment variable is set:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
export PROJECT_ID="my-project" # update with your value
|
||||||
|
```
|
||||||
|
|
||||||
|
1. (Optional) Create a Google Cloud Service Account. If you already have a
|
||||||
|
Service Account, take note of the email address and skip this step.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
gcloud iam service-accounts create "my-service-account" \
|
||||||
|
--project "${PROJECT_ID}"
|
||||||
|
```
|
||||||
|
|
||||||
|
1. (Optional) Grant the Google Cloud Service Account permissions to access
|
||||||
|
Google Cloud resources. This step varies by use case. For demonstration
|
||||||
|
purposes, you could grant access to a Google Secret Manager secret or Google
|
||||||
|
Cloud Storage object.
|
||||||
|
|
||||||
|
1. Create a Workload Identity Pool:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
gcloud iam workload-identity-pools create "my-pool" \
|
||||||
|
--project="${PROJECT_ID}" \
|
||||||
|
--location="global" \
|
||||||
|
--display-name="Demo pool"
|
||||||
|
```
|
||||||
|
|
||||||
|
1. Create a Workload Identity Provider in that pool:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
gcloud iam workload-identity-pools providers create-oidc "my-provider" \
|
||||||
|
--project="${PROJECT_ID}" \
|
||||||
|
--location="global" \
|
||||||
|
--workload-identity-pool="my-pool" \
|
||||||
|
--display-name="Demo provider" \
|
||||||
|
--attribute-mapping="google.subject=assertion.sub,attribute.actor=assertion.actor,attribute.aud=assertion.aud" \
|
||||||
|
--issuer-uri="https://vstoken.actions.githubusercontent.com" \
|
||||||
|
--allowed-audiences="sigstore"
|
||||||
|
```
|
||||||
|
|
||||||
|
- The audience of "sigstore" is currently the only value GitHub allows.
|
||||||
|
- The attribute mappings map claims in the GitHub Actions JWT to
|
||||||
|
assertions you can make about the request (like the repository or GitHub
|
||||||
|
username of the principal invoking the GitHub Action). These can be used
|
||||||
|
to further restrict the authentication using `--attribute-condition`
|
||||||
|
flags.
|
||||||
|
|
||||||
|
1. Get the full ID for the Workload Identity Provider:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
gcloud iam workload-identity-pools providers describe "my-provider" \
|
||||||
|
--project="${PROJECT_ID}" \
|
||||||
|
--location="global" \
|
||||||
|
--workload-identity-pool="my-pool"
|
||||||
|
```
|
||||||
|
|
||||||
|
Take note of the `name` attribute. It will be of the format:
|
||||||
|
|
||||||
|
```text
|
||||||
|
projects/123456789/locations/global/workloadIdentityPools/my-pool/providers/my-provider
|
||||||
|
```
|
||||||
|
|
||||||
|
Save this value as an environment variable:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
export WORKLOAD_IDENTITY_PROVIDER_ID="..." # value from above
|
||||||
|
```
|
||||||
|
|
||||||
|
1. Allow authentications from the Workload Identity Provider to impersonate the
|
||||||
|
Service Account created above:
|
||||||
|
|
||||||
|
**Warning**: This grants access to any resource in the pool (all GitHub
|
||||||
|
repos). It's **strongly recommended** that you map to a specific attribute
|
||||||
|
such as the actor or repository name instead. See [mapping external
|
||||||
|
identities][map-external] for more information.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
gcloud iam service-accounts add-iam-policy-binding "my-service-account@${PROJECT_ID}.iam.gserviceaccount.com" \
|
||||||
|
--role="roles/iam.workloadIdentityUser" \
|
||||||
|
--member="principalSet://iam.googleapis.com/${WORKLOAD_IDENTITY_PROVIDER_ID}/*"
|
||||||
|
```
|
||||||
|
|
||||||
|
To map to a specific repository:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
gcloud iam service-accounts add-iam-policy-binding "my-service-account@${PROJECT_ID}.iam.gserviceaccount.com" \
|
||||||
|
--role="roles/iam.workloadIdentityUser" \
|
||||||
|
--member="principalSet://iam.googleapis.com/${WORKLOAD_IDENTITY_PROVIDER_ID}/attribute.repo/my-repo"
|
||||||
|
```
|
||||||
|
|
||||||
|
1. Use this GitHub Action with the Workload Identity Provider ID and Service
|
||||||
|
Account email. The GitHub Action will mint a GitHub OIDC token and exchange
|
||||||
|
the GitHub token for a Google Cloud access token (assuming the authorization
|
||||||
|
is correct). This all happens without exporting a Google Cloud service
|
||||||
|
account key JSON!
|
||||||
|
|
||||||
|
[wif]: https://cloud.google.com/iam/docs/workload-identity-federation
|
||||||
|
[gcloud]: https://cloud.google.com/sdk
|
||||||
|
[map-external]: https://cloud.google.com/iam/docs/access-resources-oidc#impersonate
|
68
action.yml
Normal file
68
action.yml
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
# Copyright 2021 Google LLC
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
name: 'OIDC Authenticate to Google Cloud'
|
||||||
|
author: 'sethvargo'
|
||||||
|
description: |-
|
||||||
|
Authenticate to Google Cloud from GitHub Actions using an OIDC token and
|
||||||
|
Workload Identity Federation.
|
||||||
|
|
||||||
|
inputs:
|
||||||
|
workload_identity_provider:
|
||||||
|
description: |-
|
||||||
|
The full identifier of the Workload Identity Provider, including the
|
||||||
|
project number, pool name, and provider name. This must be the full
|
||||||
|
identifier which includes all parts, for example:
|
||||||
|
"projects/123456789/locations/global/workloadIdentityPools/my-pool/providers/my-provider".
|
||||||
|
required: true
|
||||||
|
service_account:
|
||||||
|
description: |-
|
||||||
|
Email address or unique identifier of the Google Cloud service account for
|
||||||
|
which to generate credentials.
|
||||||
|
required: true
|
||||||
|
audience:
|
||||||
|
description: |-
|
||||||
|
The value for the audience (aud) parameter in GitHub's generated OIDC
|
||||||
|
token. At present, the only valid value is "sigstore", but this variable
|
||||||
|
exists in case custom values are permitted in the future.
|
||||||
|
default: 'sigstore'
|
||||||
|
required: false
|
||||||
|
delegates:
|
||||||
|
description: |-
|
||||||
|
List of additional service account emails or unique identities to use for
|
||||||
|
impersonation in the chain.
|
||||||
|
default: ''
|
||||||
|
required: false
|
||||||
|
lifetime:
|
||||||
|
description: |-
|
||||||
|
Desired lifetime duration of the access token, in seconds. This must be
|
||||||
|
specified as the number of seconds with a trailing "s" (e.g. 30s).
|
||||||
|
default: '3600s'
|
||||||
|
required: false
|
||||||
|
|
||||||
|
outputs:
|
||||||
|
access_token:
|
||||||
|
description: |-
|
||||||
|
The Google Cloud access token for calling other Google Cloud APIs.
|
||||||
|
expiration:
|
||||||
|
description: |-
|
||||||
|
The expiration timestamp for the access token.
|
||||||
|
|
||||||
|
branding:
|
||||||
|
icon: 'lock'
|
||||||
|
color: 'blue'
|
||||||
|
|
||||||
|
runs:
|
||||||
|
using: 'node12'
|
||||||
|
main: 'dist/index.js'
|
868
dist/index.js
vendored
Normal file
868
dist/index.js
vendored
Normal file
@ -0,0 +1,868 @@
|
|||||||
|
module.exports =
|
||||||
|
/******/ (function(modules, runtime) { // webpackBootstrap
|
||||||
|
/******/ "use strict";
|
||||||
|
/******/ // The module cache
|
||||||
|
/******/ var installedModules = {};
|
||||||
|
/******/
|
||||||
|
/******/ // The require function
|
||||||
|
/******/ function __webpack_require__(moduleId) {
|
||||||
|
/******/
|
||||||
|
/******/ // Check if module is in cache
|
||||||
|
/******/ if(installedModules[moduleId]) {
|
||||||
|
/******/ return installedModules[moduleId].exports;
|
||||||
|
/******/ }
|
||||||
|
/******/ // Create a new module (and put it into the cache)
|
||||||
|
/******/ var module = installedModules[moduleId] = {
|
||||||
|
/******/ i: moduleId,
|
||||||
|
/******/ l: false,
|
||||||
|
/******/ exports: {}
|
||||||
|
/******/ };
|
||||||
|
/******/
|
||||||
|
/******/ // Execute the module function
|
||||||
|
/******/ var threw = true;
|
||||||
|
/******/ try {
|
||||||
|
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
||||||
|
/******/ threw = false;
|
||||||
|
/******/ } finally {
|
||||||
|
/******/ if(threw) delete installedModules[moduleId];
|
||||||
|
/******/ }
|
||||||
|
/******/
|
||||||
|
/******/ // Flag the module as loaded
|
||||||
|
/******/ module.l = true;
|
||||||
|
/******/
|
||||||
|
/******/ // Return the exports of the module
|
||||||
|
/******/ return module.exports;
|
||||||
|
/******/ }
|
||||||
|
/******/
|
||||||
|
/******/
|
||||||
|
/******/ __webpack_require__.ab = __dirname + "/";
|
||||||
|
/******/
|
||||||
|
/******/ // the startup function
|
||||||
|
/******/ function startup() {
|
||||||
|
/******/ // Load entry module and return exports
|
||||||
|
/******/ return __webpack_require__(131);
|
||||||
|
/******/ };
|
||||||
|
/******/
|
||||||
|
/******/ // run startup
|
||||||
|
/******/ return startup();
|
||||||
|
/******/ })
|
||||||
|
/************************************************************************/
|
||||||
|
/******/ ({
|
||||||
|
|
||||||
|
/***/ 82:
|
||||||
|
/***/ (function(__unusedmodule, exports) {
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
// We use any as a valid input type
|
||||||
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
|
Object.defineProperty(exports, "__esModule", { value: true });
|
||||||
|
exports.toCommandProperties = exports.toCommandValue = void 0;
|
||||||
|
/**
|
||||||
|
* Sanitizes an input into a string so it can be passed into issueCommand safely
|
||||||
|
* @param input input to sanitize into a string
|
||||||
|
*/
|
||||||
|
function toCommandValue(input) {
|
||||||
|
if (input === null || input === undefined) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
else if (typeof input === 'string' || input instanceof String) {
|
||||||
|
return input;
|
||||||
|
}
|
||||||
|
return JSON.stringify(input);
|
||||||
|
}
|
||||||
|
exports.toCommandValue = toCommandValue;
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param annotationProperties
|
||||||
|
* @returns The command properties to send with the actual annotation command
|
||||||
|
* See IssueCommandProperties: https://github.com/actions/runner/blob/main/src/Runner.Worker/ActionCommandManager.cs#L646
|
||||||
|
*/
|
||||||
|
function toCommandProperties(annotationProperties) {
|
||||||
|
if (!Object.keys(annotationProperties).length) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
title: annotationProperties.title,
|
||||||
|
line: annotationProperties.startLine,
|
||||||
|
endLine: annotationProperties.endLine,
|
||||||
|
col: annotationProperties.startColumn,
|
||||||
|
endColumn: annotationProperties.endColumn
|
||||||
|
};
|
||||||
|
}
|
||||||
|
exports.toCommandProperties = toCommandProperties;
|
||||||
|
//# sourceMappingURL=utils.js.map
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ 87:
|
||||||
|
/***/ (function(module) {
|
||||||
|
|
||||||
|
module.exports = require("os");
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ 102:
|
||||||
|
/***/ (function(__unusedmodule, exports, __webpack_require__) {
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
// For internal use, subject to change.
|
||||||
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
||||||
|
if (k2 === undefined) k2 = k;
|
||||||
|
Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
|
||||||
|
}) : (function(o, m, k, k2) {
|
||||||
|
if (k2 === undefined) k2 = k;
|
||||||
|
o[k2] = m[k];
|
||||||
|
}));
|
||||||
|
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
||||||
|
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
||||||
|
}) : function(o, v) {
|
||||||
|
o["default"] = v;
|
||||||
|
});
|
||||||
|
var __importStar = (this && this.__importStar) || function (mod) {
|
||||||
|
if (mod && mod.__esModule) return mod;
|
||||||
|
var result = {};
|
||||||
|
if (mod != null) for (var k in mod) if (k !== "default" && Object.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
||||||
|
__setModuleDefault(result, mod);
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
Object.defineProperty(exports, "__esModule", { value: true });
|
||||||
|
exports.issueCommand = void 0;
|
||||||
|
// We use any as a valid input type
|
||||||
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
|
const fs = __importStar(__webpack_require__(747));
|
||||||
|
const os = __importStar(__webpack_require__(87));
|
||||||
|
const utils_1 = __webpack_require__(82);
|
||||||
|
function issueCommand(command, message) {
|
||||||
|
const filePath = process.env[`GITHUB_${command}`];
|
||||||
|
if (!filePath) {
|
||||||
|
throw new Error(`Unable to find environment variable for file command ${command}`);
|
||||||
|
}
|
||||||
|
if (!fs.existsSync(filePath)) {
|
||||||
|
throw new Error(`Missing file at path: ${filePath}`);
|
||||||
|
}
|
||||||
|
fs.appendFileSync(filePath, `${utils_1.toCommandValue(message)}${os.EOL}`, {
|
||||||
|
encoding: 'utf8'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
exports.issueCommand = issueCommand;
|
||||||
|
//# sourceMappingURL=file-command.js.map
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ 131:
|
||||||
|
/***/ (function(__unusedmodule, exports, __webpack_require__) {
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
||||||
|
if (k2 === undefined) k2 = k;
|
||||||
|
Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
|
||||||
|
}) : (function(o, m, k, k2) {
|
||||||
|
if (k2 === undefined) k2 = k;
|
||||||
|
o[k2] = m[k];
|
||||||
|
}));
|
||||||
|
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
||||||
|
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
||||||
|
}) : function(o, v) {
|
||||||
|
o["default"] = v;
|
||||||
|
});
|
||||||
|
var __importStar = (this && this.__importStar) || function (mod) {
|
||||||
|
if (mod && mod.__esModule) return mod;
|
||||||
|
var result = {};
|
||||||
|
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
||||||
|
__setModuleDefault(result, mod);
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
||||||
|
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
||||||
|
return new (P || (P = Promise))(function (resolve, reject) {
|
||||||
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
||||||
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
||||||
|
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
||||||
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
||||||
|
});
|
||||||
|
};
|
||||||
|
Object.defineProperty(exports, "__esModule", { value: true });
|
||||||
|
const core = __importStar(__webpack_require__(470));
|
||||||
|
const client_1 = __webpack_require__(976);
|
||||||
|
/**
|
||||||
|
* Converts a multi-line or comma-separated collection of strings into an array
|
||||||
|
* of trimmed strings.
|
||||||
|
*/
|
||||||
|
function explodeStrings(input) {
|
||||||
|
if (input == null || input.length === 0) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
const list = new Array();
|
||||||
|
for (const line of input.split(`\n`)) {
|
||||||
|
for (const piece of line.split(',')) {
|
||||||
|
const entry = piece.trim();
|
||||||
|
if (entry !== '') {
|
||||||
|
list.push(entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Executes the main action, documented inline.
|
||||||
|
*/
|
||||||
|
function run() {
|
||||||
|
return __awaiter(this, void 0, void 0, function* () {
|
||||||
|
try {
|
||||||
|
// Load configuration.
|
||||||
|
const workloadIdentityProvider = core.getInput('workload_identity_provider', {
|
||||||
|
required: true,
|
||||||
|
});
|
||||||
|
const serviceAccount = core.getInput('service_account', { required: true });
|
||||||
|
const audience = core.getInput('audience');
|
||||||
|
const delegates = explodeStrings(core.getInput('delegates'));
|
||||||
|
const lifetime = core.getInput('lifetime');
|
||||||
|
// Extract the GitHub Actions OIDC token.
|
||||||
|
const requestToken = process.env.ACTIONS_ID_TOKEN_REQUEST_TOKEN;
|
||||||
|
if (!requestToken) {
|
||||||
|
throw `missing ACTIONS_ID_TOKEN_REQUEST_TOKEN`;
|
||||||
|
}
|
||||||
|
const requestURL = process.env.ACTIONS_ID_TOKEN_REQUEST_URL;
|
||||||
|
if (!requestURL) {
|
||||||
|
throw `missing ACTIONS_ID_TOKEN_REQUEST_URL`;
|
||||||
|
}
|
||||||
|
const githubOIDCToken = yield client_1.Client.githubToken({
|
||||||
|
url: requestURL,
|
||||||
|
token: requestToken,
|
||||||
|
audience: audience,
|
||||||
|
});
|
||||||
|
core.setSecret(githubOIDCToken);
|
||||||
|
// Exchange the GitHub OIDC token for a Google Federated Token.
|
||||||
|
const googleFederatedToken = yield client_1.Client.googleFederatedToken({
|
||||||
|
providerID: workloadIdentityProvider,
|
||||||
|
token: githubOIDCToken,
|
||||||
|
});
|
||||||
|
core.setSecret(googleFederatedToken);
|
||||||
|
// Exchange the Google Federated Token for an access token.
|
||||||
|
const { accessToken, expiration } = yield client_1.Client.googleAccessToken({
|
||||||
|
token: googleFederatedToken,
|
||||||
|
serviceAccount: serviceAccount,
|
||||||
|
delegates: delegates,
|
||||||
|
lifetime: lifetime,
|
||||||
|
});
|
||||||
|
core.setSecret(accessToken);
|
||||||
|
core.setOutput('access_token', accessToken);
|
||||||
|
core.setOutput('expiration', expiration);
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
core.setFailed(`Action failed with error: ${err}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
run();
|
||||||
|
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ 211:
|
||||||
|
/***/ (function(module) {
|
||||||
|
|
||||||
|
module.exports = require("https");
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ 431:
|
||||||
|
/***/ (function(__unusedmodule, exports, __webpack_require__) {
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
||||||
|
if (k2 === undefined) k2 = k;
|
||||||
|
Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
|
||||||
|
}) : (function(o, m, k, k2) {
|
||||||
|
if (k2 === undefined) k2 = k;
|
||||||
|
o[k2] = m[k];
|
||||||
|
}));
|
||||||
|
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
||||||
|
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
||||||
|
}) : function(o, v) {
|
||||||
|
o["default"] = v;
|
||||||
|
});
|
||||||
|
var __importStar = (this && this.__importStar) || function (mod) {
|
||||||
|
if (mod && mod.__esModule) return mod;
|
||||||
|
var result = {};
|
||||||
|
if (mod != null) for (var k in mod) if (k !== "default" && Object.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
||||||
|
__setModuleDefault(result, mod);
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
Object.defineProperty(exports, "__esModule", { value: true });
|
||||||
|
exports.issue = exports.issueCommand = void 0;
|
||||||
|
const os = __importStar(__webpack_require__(87));
|
||||||
|
const utils_1 = __webpack_require__(82);
|
||||||
|
/**
|
||||||
|
* Commands
|
||||||
|
*
|
||||||
|
* Command Format:
|
||||||
|
* ::name key=value,key=value::message
|
||||||
|
*
|
||||||
|
* Examples:
|
||||||
|
* ::warning::This is the message
|
||||||
|
* ::set-env name=MY_VAR::some value
|
||||||
|
*/
|
||||||
|
function issueCommand(command, properties, message) {
|
||||||
|
const cmd = new Command(command, properties, message);
|
||||||
|
process.stdout.write(cmd.toString() + os.EOL);
|
||||||
|
}
|
||||||
|
exports.issueCommand = issueCommand;
|
||||||
|
function issue(name, message = '') {
|
||||||
|
issueCommand(name, {}, message);
|
||||||
|
}
|
||||||
|
exports.issue = issue;
|
||||||
|
const CMD_STRING = '::';
|
||||||
|
class Command {
|
||||||
|
constructor(command, properties, message) {
|
||||||
|
if (!command) {
|
||||||
|
command = 'missing.command';
|
||||||
|
}
|
||||||
|
this.command = command;
|
||||||
|
this.properties = properties;
|
||||||
|
this.message = message;
|
||||||
|
}
|
||||||
|
toString() {
|
||||||
|
let cmdStr = CMD_STRING + this.command;
|
||||||
|
if (this.properties && Object.keys(this.properties).length > 0) {
|
||||||
|
cmdStr += ' ';
|
||||||
|
let first = true;
|
||||||
|
for (const key in this.properties) {
|
||||||
|
if (this.properties.hasOwnProperty(key)) {
|
||||||
|
const val = this.properties[key];
|
||||||
|
if (val) {
|
||||||
|
if (first) {
|
||||||
|
first = false;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
cmdStr += ',';
|
||||||
|
}
|
||||||
|
cmdStr += `${key}=${escapeProperty(val)}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cmdStr += `${CMD_STRING}${escapeData(this.message)}`;
|
||||||
|
return cmdStr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function escapeData(s) {
|
||||||
|
return utils_1.toCommandValue(s)
|
||||||
|
.replace(/%/g, '%25')
|
||||||
|
.replace(/\r/g, '%0D')
|
||||||
|
.replace(/\n/g, '%0A');
|
||||||
|
}
|
||||||
|
function escapeProperty(s) {
|
||||||
|
return utils_1.toCommandValue(s)
|
||||||
|
.replace(/%/g, '%25')
|
||||||
|
.replace(/\r/g, '%0D')
|
||||||
|
.replace(/\n/g, '%0A')
|
||||||
|
.replace(/:/g, '%3A')
|
||||||
|
.replace(/,/g, '%2C');
|
||||||
|
}
|
||||||
|
//# sourceMappingURL=command.js.map
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ 470:
|
||||||
|
/***/ (function(__unusedmodule, exports, __webpack_require__) {
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
||||||
|
if (k2 === undefined) k2 = k;
|
||||||
|
Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
|
||||||
|
}) : (function(o, m, k, k2) {
|
||||||
|
if (k2 === undefined) k2 = k;
|
||||||
|
o[k2] = m[k];
|
||||||
|
}));
|
||||||
|
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
||||||
|
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
||||||
|
}) : function(o, v) {
|
||||||
|
o["default"] = v;
|
||||||
|
});
|
||||||
|
var __importStar = (this && this.__importStar) || function (mod) {
|
||||||
|
if (mod && mod.__esModule) return mod;
|
||||||
|
var result = {};
|
||||||
|
if (mod != null) for (var k in mod) if (k !== "default" && Object.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
||||||
|
__setModuleDefault(result, mod);
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
||||||
|
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
||||||
|
return new (P || (P = Promise))(function (resolve, reject) {
|
||||||
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
||||||
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
||||||
|
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
||||||
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
||||||
|
});
|
||||||
|
};
|
||||||
|
Object.defineProperty(exports, "__esModule", { value: true });
|
||||||
|
exports.getState = exports.saveState = exports.group = exports.endGroup = exports.startGroup = exports.info = exports.notice = exports.warning = exports.error = exports.debug = exports.isDebug = exports.setFailed = exports.setCommandEcho = exports.setOutput = exports.getBooleanInput = exports.getMultilineInput = exports.getInput = exports.addPath = exports.setSecret = exports.exportVariable = exports.ExitCode = void 0;
|
||||||
|
const command_1 = __webpack_require__(431);
|
||||||
|
const file_command_1 = __webpack_require__(102);
|
||||||
|
const utils_1 = __webpack_require__(82);
|
||||||
|
const os = __importStar(__webpack_require__(87));
|
||||||
|
const path = __importStar(__webpack_require__(622));
|
||||||
|
/**
|
||||||
|
* The code to exit an action
|
||||||
|
*/
|
||||||
|
var ExitCode;
|
||||||
|
(function (ExitCode) {
|
||||||
|
/**
|
||||||
|
* A code indicating that the action was successful
|
||||||
|
*/
|
||||||
|
ExitCode[ExitCode["Success"] = 0] = "Success";
|
||||||
|
/**
|
||||||
|
* A code indicating that the action was a failure
|
||||||
|
*/
|
||||||
|
ExitCode[ExitCode["Failure"] = 1] = "Failure";
|
||||||
|
})(ExitCode = exports.ExitCode || (exports.ExitCode = {}));
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
// Variables
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
/**
|
||||||
|
* Sets env variable for this action and future actions in the job
|
||||||
|
* @param name the name of the variable to set
|
||||||
|
* @param val the value of the variable. Non-string values will be converted to a string via JSON.stringify
|
||||||
|
*/
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
function exportVariable(name, val) {
|
||||||
|
const convertedVal = utils_1.toCommandValue(val);
|
||||||
|
process.env[name] = convertedVal;
|
||||||
|
const filePath = process.env['GITHUB_ENV'] || '';
|
||||||
|
if (filePath) {
|
||||||
|
const delimiter = '_GitHubActionsFileCommandDelimeter_';
|
||||||
|
const commandValue = `${name}<<${delimiter}${os.EOL}${convertedVal}${os.EOL}${delimiter}`;
|
||||||
|
file_command_1.issueCommand('ENV', commandValue);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
command_1.issueCommand('set-env', { name }, convertedVal);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
exports.exportVariable = exportVariable;
|
||||||
|
/**
|
||||||
|
* Registers a secret which will get masked from logs
|
||||||
|
* @param secret value of the secret
|
||||||
|
*/
|
||||||
|
function setSecret(secret) {
|
||||||
|
command_1.issueCommand('add-mask', {}, secret);
|
||||||
|
}
|
||||||
|
exports.setSecret = setSecret;
|
||||||
|
/**
|
||||||
|
* Prepends inputPath to the PATH (for this action and future actions)
|
||||||
|
* @param inputPath
|
||||||
|
*/
|
||||||
|
function addPath(inputPath) {
|
||||||
|
const filePath = process.env['GITHUB_PATH'] || '';
|
||||||
|
if (filePath) {
|
||||||
|
file_command_1.issueCommand('PATH', inputPath);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
command_1.issueCommand('add-path', {}, inputPath);
|
||||||
|
}
|
||||||
|
process.env['PATH'] = `${inputPath}${path.delimiter}${process.env['PATH']}`;
|
||||||
|
}
|
||||||
|
exports.addPath = addPath;
|
||||||
|
/**
|
||||||
|
* Gets the value of an input.
|
||||||
|
* Unless trimWhitespace is set to false in InputOptions, the value is also trimmed.
|
||||||
|
* Returns an empty string if the value is not defined.
|
||||||
|
*
|
||||||
|
* @param name name of the input to get
|
||||||
|
* @param options optional. See InputOptions.
|
||||||
|
* @returns string
|
||||||
|
*/
|
||||||
|
function getInput(name, options) {
|
||||||
|
const val = process.env[`INPUT_${name.replace(/ /g, '_').toUpperCase()}`] || '';
|
||||||
|
if (options && options.required && !val) {
|
||||||
|
throw new Error(`Input required and not supplied: ${name}`);
|
||||||
|
}
|
||||||
|
if (options && options.trimWhitespace === false) {
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
return val.trim();
|
||||||
|
}
|
||||||
|
exports.getInput = getInput;
|
||||||
|
/**
|
||||||
|
* Gets the values of an multiline input. Each value is also trimmed.
|
||||||
|
*
|
||||||
|
* @param name name of the input to get
|
||||||
|
* @param options optional. See InputOptions.
|
||||||
|
* @returns string[]
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
function getMultilineInput(name, options) {
|
||||||
|
const inputs = getInput(name, options)
|
||||||
|
.split('\n')
|
||||||
|
.filter(x => x !== '');
|
||||||
|
return inputs;
|
||||||
|
}
|
||||||
|
exports.getMultilineInput = getMultilineInput;
|
||||||
|
/**
|
||||||
|
* Gets the input value of the boolean type in the YAML 1.2 "core schema" specification.
|
||||||
|
* Support boolean input list: `true | True | TRUE | false | False | FALSE` .
|
||||||
|
* The return value is also in boolean type.
|
||||||
|
* ref: https://yaml.org/spec/1.2/spec.html#id2804923
|
||||||
|
*
|
||||||
|
* @param name name of the input to get
|
||||||
|
* @param options optional. See InputOptions.
|
||||||
|
* @returns boolean
|
||||||
|
*/
|
||||||
|
function getBooleanInput(name, options) {
|
||||||
|
const trueValue = ['true', 'True', 'TRUE'];
|
||||||
|
const falseValue = ['false', 'False', 'FALSE'];
|
||||||
|
const val = getInput(name, options);
|
||||||
|
if (trueValue.includes(val))
|
||||||
|
return true;
|
||||||
|
if (falseValue.includes(val))
|
||||||
|
return false;
|
||||||
|
throw new TypeError(`Input does not meet YAML 1.2 "Core Schema" specification: ${name}\n` +
|
||||||
|
`Support boolean input list: \`true | True | TRUE | false | False | FALSE\``);
|
||||||
|
}
|
||||||
|
exports.getBooleanInput = getBooleanInput;
|
||||||
|
/**
|
||||||
|
* Sets the value of an output.
|
||||||
|
*
|
||||||
|
* @param name name of the output to set
|
||||||
|
* @param value value to store. Non-string values will be converted to a string via JSON.stringify
|
||||||
|
*/
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
function setOutput(name, value) {
|
||||||
|
process.stdout.write(os.EOL);
|
||||||
|
command_1.issueCommand('set-output', { name }, value);
|
||||||
|
}
|
||||||
|
exports.setOutput = setOutput;
|
||||||
|
/**
|
||||||
|
* Enables or disables the echoing of commands into stdout for the rest of the step.
|
||||||
|
* Echoing is disabled by default if ACTIONS_STEP_DEBUG is not set.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
function setCommandEcho(enabled) {
|
||||||
|
command_1.issue('echo', enabled ? 'on' : 'off');
|
||||||
|
}
|
||||||
|
exports.setCommandEcho = setCommandEcho;
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
// Results
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
/**
|
||||||
|
* Sets the action status to failed.
|
||||||
|
* When the action exits it will be with an exit code of 1
|
||||||
|
* @param message add error issue message
|
||||||
|
*/
|
||||||
|
function setFailed(message) {
|
||||||
|
process.exitCode = ExitCode.Failure;
|
||||||
|
error(message);
|
||||||
|
}
|
||||||
|
exports.setFailed = setFailed;
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
// Logging Commands
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
/**
|
||||||
|
* Gets whether Actions Step Debug is on or not
|
||||||
|
*/
|
||||||
|
function isDebug() {
|
||||||
|
return process.env['RUNNER_DEBUG'] === '1';
|
||||||
|
}
|
||||||
|
exports.isDebug = isDebug;
|
||||||
|
/**
|
||||||
|
* Writes debug message to user log
|
||||||
|
* @param message debug message
|
||||||
|
*/
|
||||||
|
function debug(message) {
|
||||||
|
command_1.issueCommand('debug', {}, message);
|
||||||
|
}
|
||||||
|
exports.debug = debug;
|
||||||
|
/**
|
||||||
|
* Adds an error issue
|
||||||
|
* @param message error issue message. Errors will be converted to string via toString()
|
||||||
|
* @param properties optional properties to add to the annotation.
|
||||||
|
*/
|
||||||
|
function error(message, properties = {}) {
|
||||||
|
command_1.issueCommand('error', utils_1.toCommandProperties(properties), message instanceof Error ? message.toString() : message);
|
||||||
|
}
|
||||||
|
exports.error = error;
|
||||||
|
/**
|
||||||
|
* Adds a warning issue
|
||||||
|
* @param message warning issue message. Errors will be converted to string via toString()
|
||||||
|
* @param properties optional properties to add to the annotation.
|
||||||
|
*/
|
||||||
|
function warning(message, properties = {}) {
|
||||||
|
command_1.issueCommand('warning', utils_1.toCommandProperties(properties), message instanceof Error ? message.toString() : message);
|
||||||
|
}
|
||||||
|
exports.warning = warning;
|
||||||
|
/**
|
||||||
|
* Adds a notice issue
|
||||||
|
* @param message notice issue message. Errors will be converted to string via toString()
|
||||||
|
* @param properties optional properties to add to the annotation.
|
||||||
|
*/
|
||||||
|
function notice(message, properties = {}) {
|
||||||
|
command_1.issueCommand('notice', utils_1.toCommandProperties(properties), message instanceof Error ? message.toString() : message);
|
||||||
|
}
|
||||||
|
exports.notice = notice;
|
||||||
|
/**
|
||||||
|
* Writes info to log with console.log.
|
||||||
|
* @param message info message
|
||||||
|
*/
|
||||||
|
function info(message) {
|
||||||
|
process.stdout.write(message + os.EOL);
|
||||||
|
}
|
||||||
|
exports.info = info;
|
||||||
|
/**
|
||||||
|
* Begin an output group.
|
||||||
|
*
|
||||||
|
* Output until the next `groupEnd` will be foldable in this group
|
||||||
|
*
|
||||||
|
* @param name The name of the output group
|
||||||
|
*/
|
||||||
|
function startGroup(name) {
|
||||||
|
command_1.issue('group', name);
|
||||||
|
}
|
||||||
|
exports.startGroup = startGroup;
|
||||||
|
/**
|
||||||
|
* End an output group.
|
||||||
|
*/
|
||||||
|
function endGroup() {
|
||||||
|
command_1.issue('endgroup');
|
||||||
|
}
|
||||||
|
exports.endGroup = endGroup;
|
||||||
|
/**
|
||||||
|
* Wrap an asynchronous function call in a group.
|
||||||
|
*
|
||||||
|
* Returns the same type as the function itself.
|
||||||
|
*
|
||||||
|
* @param name The name of the group
|
||||||
|
* @param fn The function to wrap in the group
|
||||||
|
*/
|
||||||
|
function group(name, fn) {
|
||||||
|
return __awaiter(this, void 0, void 0, function* () {
|
||||||
|
startGroup(name);
|
||||||
|
let result;
|
||||||
|
try {
|
||||||
|
result = yield fn();
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
endGroup();
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
exports.group = group;
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
// Wrapper action state
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
/**
|
||||||
|
* Saves state for current action, the state can only be retrieved by this action's post job execution.
|
||||||
|
*
|
||||||
|
* @param name name of the state to store
|
||||||
|
* @param value value to store. Non-string values will be converted to a string via JSON.stringify
|
||||||
|
*/
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
function saveState(name, value) {
|
||||||
|
command_1.issueCommand('save-state', { name }, value);
|
||||||
|
}
|
||||||
|
exports.saveState = saveState;
|
||||||
|
/**
|
||||||
|
* Gets the value of an state set by this action's main execution.
|
||||||
|
*
|
||||||
|
* @param name name of the state to get
|
||||||
|
* @returns string
|
||||||
|
*/
|
||||||
|
function getState(name) {
|
||||||
|
return process.env[`STATE_${name}`] || '';
|
||||||
|
}
|
||||||
|
exports.getState = getState;
|
||||||
|
//# sourceMappingURL=core.js.map
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ 622:
|
||||||
|
/***/ (function(module) {
|
||||||
|
|
||||||
|
module.exports = require("path");
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ 747:
|
||||||
|
/***/ (function(module) {
|
||||||
|
|
||||||
|
module.exports = require("fs");
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ 835:
|
||||||
|
/***/ (function(module) {
|
||||||
|
|
||||||
|
module.exports = require("url");
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ 976:
|
||||||
|
/***/ (function(__unusedmodule, exports, __webpack_require__) {
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
||||||
|
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
||||||
|
return new (P || (P = Promise))(function (resolve, reject) {
|
||||||
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
||||||
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
||||||
|
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
||||||
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
||||||
|
});
|
||||||
|
};
|
||||||
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
||||||
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
||||||
|
};
|
||||||
|
Object.defineProperty(exports, "__esModule", { value: true });
|
||||||
|
exports.Client = void 0;
|
||||||
|
const https_1 = __importDefault(__webpack_require__(211));
|
||||||
|
const url_1 = __webpack_require__(835);
|
||||||
|
class Client {
|
||||||
|
/**
|
||||||
|
* request is a high-level helper that returns a promise from the executed
|
||||||
|
* request.
|
||||||
|
*/
|
||||||
|
static request(opts, data) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const req = https_1.default.request(opts, (res) => {
|
||||||
|
res.setEncoding('utf8');
|
||||||
|
let body = '';
|
||||||
|
res.on('data', (data) => {
|
||||||
|
body += data;
|
||||||
|
});
|
||||||
|
res.on('end', () => {
|
||||||
|
if (res.statusCode && res.statusCode >= 400) {
|
||||||
|
reject(body);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
resolve(body);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
req.on('error', (err) => {
|
||||||
|
reject(err);
|
||||||
|
});
|
||||||
|
if (data != null) {
|
||||||
|
req.write(data);
|
||||||
|
}
|
||||||
|
req.end();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* githubToken invokes the given URL, appending the audience parameter, using
|
||||||
|
* the provided token as authentication. This can only be run from inside a
|
||||||
|
* GitHub Action.
|
||||||
|
*/
|
||||||
|
static githubToken({ url, audience, token }) {
|
||||||
|
return __awaiter(this, void 0, void 0, function* () {
|
||||||
|
const requestURL = new url_1.URL(url);
|
||||||
|
// Append the audience value to the request.
|
||||||
|
const params = requestURL.searchParams;
|
||||||
|
params.set('audience', audience);
|
||||||
|
requestURL.search = params.toString();
|
||||||
|
// Make the request.
|
||||||
|
const opts = {
|
||||||
|
hostname: requestURL.hostname,
|
||||||
|
port: requestURL.port,
|
||||||
|
path: requestURL.pathname + requestURL.search,
|
||||||
|
method: 'GET',
|
||||||
|
headers: {
|
||||||
|
'Authorization': `Bearer ${token}`,
|
||||||
|
'Accept': 'application/json',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
try {
|
||||||
|
const resp = yield Client.request(opts);
|
||||||
|
const parsed = JSON.parse(resp);
|
||||||
|
return parsed['value'];
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
throw new Error(`failed to generate GitHub OIDC token via ${url} (aud: ${audience}): ${err}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* googleFederatedToken generates a Google Cloud federated token using the
|
||||||
|
* provided OIDC token and Workload Identity Provider.
|
||||||
|
*/
|
||||||
|
static googleFederatedToken({ providerID, token, }) {
|
||||||
|
return __awaiter(this, void 0, void 0, function* () {
|
||||||
|
const stsURL = new url_1.URL('https://sts.googleapis.com/v1/token');
|
||||||
|
const data = {
|
||||||
|
audience: '//iam.googleapis.com/' + providerID,
|
||||||
|
grantType: 'urn:ietf:params:oauth:grant-type:token-exchange',
|
||||||
|
requestedTokenType: 'urn:ietf:params:oauth:token-type:access_token',
|
||||||
|
scope: 'https://www.googleapis.com/auth/cloud-platform',
|
||||||
|
subjectTokenType: 'urn:ietf:params:oauth:token-type:jwt',
|
||||||
|
subjectToken: token,
|
||||||
|
};
|
||||||
|
const opts = {
|
||||||
|
hostname: stsURL.hostname,
|
||||||
|
port: stsURL.port,
|
||||||
|
path: stsURL.pathname + stsURL.search,
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Accept': 'application/json',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
try {
|
||||||
|
const resp = yield Client.request(opts, JSON.stringify(data));
|
||||||
|
const parsed = JSON.parse(resp);
|
||||||
|
return parsed['access_token'];
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
throw new Error(`failed to generate Google Cloud federated token for ${providerID}: ${err}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* googleAccessToken generates a Google Cloud access token for the provided
|
||||||
|
* service account email or unique id.
|
||||||
|
*/
|
||||||
|
static googleAccessToken({ token, serviceAccount, delegates, lifetime, }) {
|
||||||
|
return __awaiter(this, void 0, void 0, function* () {
|
||||||
|
const serviceAccountID = `projects/-/serviceAccounts/${serviceAccount}`;
|
||||||
|
const tokenURL = new url_1.URL(`https://iamcredentials.googleapis.com/v1/${serviceAccountID}:generateAccessToken`);
|
||||||
|
const data = {
|
||||||
|
delegates: delegates,
|
||||||
|
scope: 'https://www.googleapis.com/auth/cloud-platform',
|
||||||
|
lifetime: lifetime,
|
||||||
|
};
|
||||||
|
const opts = {
|
||||||
|
hostname: tokenURL.hostname,
|
||||||
|
port: tokenURL.port,
|
||||||
|
path: tokenURL.pathname + tokenURL.search,
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Authorization': `Bearer ${token}`,
|
||||||
|
'Accept': 'application/json',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
try {
|
||||||
|
const resp = yield Client.request(opts, JSON.stringify(data));
|
||||||
|
const parsed = JSON.parse(resp);
|
||||||
|
return {
|
||||||
|
accessToken: parsed['accessToken'],
|
||||||
|
expiration: parsed['expireTime'],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
throw new Error(`failed to generate Google Cloud access token for ${serviceAccount}: ${err}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
exports.Client = Client;
|
||||||
|
|
||||||
|
|
||||||
|
/***/ })
|
||||||
|
|
||||||
|
/******/ });
|
4561
package-lock.json
generated
Normal file
4561
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
45
package.json
Normal file
45
package.json
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
{
|
||||||
|
"name": "oidc-auth-gcp",
|
||||||
|
"version": "0.1.0",
|
||||||
|
"description": "Authenticate to Google Cloud using a GitHub Actions OIDC token.",
|
||||||
|
"main": "dist/index.js",
|
||||||
|
"scripts": {
|
||||||
|
"build": "ncc build src/main.ts",
|
||||||
|
"lint": "eslint . --ext .ts,.tsx",
|
||||||
|
"format": "prettier --write **/*.ts",
|
||||||
|
"test": "mocha -r ts-node/register -t 120s 'tests/*.test.ts'"
|
||||||
|
},
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/sethvargo/oidc-auth-gcp"
|
||||||
|
},
|
||||||
|
"keywords": [
|
||||||
|
"actions",
|
||||||
|
"google cloud",
|
||||||
|
"identity",
|
||||||
|
"auth",
|
||||||
|
"oidc"
|
||||||
|
],
|
||||||
|
"author": "GoogleCloudPlatform",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"dependencies": {
|
||||||
|
"@actions/core": "^1.5.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/chai": "^4.2.21",
|
||||||
|
"@types/mocha": "^9.0.0",
|
||||||
|
"@types/node": "^16.9.1",
|
||||||
|
"@typescript-eslint/eslint-plugin": "^4.31.0",
|
||||||
|
"@typescript-eslint/parser": "^4.31.0",
|
||||||
|
"@zeit/ncc": "^0.22.3",
|
||||||
|
"chai": "^4.3.4",
|
||||||
|
"eslint": "^7.32.0",
|
||||||
|
"eslint-config-prettier": "^8.3.0",
|
||||||
|
"eslint-plugin-prettier": "^4.0.0",
|
||||||
|
"husky": "^7.0.2",
|
||||||
|
"mocha": "^9.1.1",
|
||||||
|
"prettier": "^2.4.0",
|
||||||
|
"ts-node": "^10.2.1",
|
||||||
|
"typescript": "^4.3.5"
|
||||||
|
}
|
||||||
|
}
|
221
src/client.ts
Normal file
221
src/client.ts
Normal file
@ -0,0 +1,221 @@
|
|||||||
|
import https, { RequestOptions } from 'https';
|
||||||
|
import { URL } from 'url';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GitHubTokenParameters are the parameters to generate an OIDC token from
|
||||||
|
* within a GitHub Action.
|
||||||
|
*
|
||||||
|
* @param url URL endpoint from which to request the token.
|
||||||
|
* @param audience JWT aud value for the token.
|
||||||
|
* @param token Temporary token provided by the environment to request the real
|
||||||
|
* token.
|
||||||
|
*/
|
||||||
|
interface GitHubTokenParameters {
|
||||||
|
url: string;
|
||||||
|
audience: string;
|
||||||
|
token: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GoogleFederatedTokenParameters are the parameters to generate a Federated
|
||||||
|
* Identity Token as described in:
|
||||||
|
*
|
||||||
|
* https://cloud.google.com/iam/docs/access-resources-oidc#exchange-token
|
||||||
|
*
|
||||||
|
* @param providerID Full path (including project, location, etc) to the Google
|
||||||
|
* Cloud Workload Identity Provider.
|
||||||
|
* @param token OIDC token to exchange for a Google Cloud federated token.
|
||||||
|
*/
|
||||||
|
interface GoogleFederatedTokenParameters {
|
||||||
|
providerID: string;
|
||||||
|
token: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GoogleAccessTokenParameters are the parameters to generate a Google Cloud
|
||||||
|
* access token as described in:
|
||||||
|
*
|
||||||
|
* https://cloud.google.com/iam/docs/reference/credentials/rest/v1/projects.serviceAccounts/generateAccessToken
|
||||||
|
*
|
||||||
|
* @param token OAuth token or Federated access token with permissions to call
|
||||||
|
* the API.
|
||||||
|
* @param serviceAccount Email address or unique identifier of the service
|
||||||
|
* account.
|
||||||
|
* @param delegates Optional sequence of service accounts in the delegation
|
||||||
|
* chain.
|
||||||
|
* @param lifetime Optional validity period as a duration.
|
||||||
|
*/
|
||||||
|
interface GoogleAccessTokenParameters {
|
||||||
|
token: string;
|
||||||
|
serviceAccount: string;
|
||||||
|
delegates?: Array<string>;
|
||||||
|
lifetime?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GoogleAccessTokenResponse is the response from generating an access token.
|
||||||
|
*
|
||||||
|
* @param accessToken OAuth 2.0 access token.
|
||||||
|
* @param expiration A timestamp in RFC3339 UTC "Zulu" format when the token
|
||||||
|
* expires.
|
||||||
|
*/
|
||||||
|
interface GoogleAccessTokenResponse {
|
||||||
|
accessToken: string;
|
||||||
|
expiration: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class Client {
|
||||||
|
/**
|
||||||
|
* request is a high-level helper that returns a promise from the executed
|
||||||
|
* request.
|
||||||
|
*/
|
||||||
|
static request(opts: RequestOptions, data?: any): Promise<string> {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const req = https.request(opts, (res) => {
|
||||||
|
res.setEncoding('utf8');
|
||||||
|
|
||||||
|
let body = '';
|
||||||
|
res.on('data', (data) => {
|
||||||
|
body += data;
|
||||||
|
});
|
||||||
|
|
||||||
|
res.on('end', () => {
|
||||||
|
if (res.statusCode && res.statusCode >= 400) {
|
||||||
|
reject(body);
|
||||||
|
} else {
|
||||||
|
resolve(body);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
req.on('error', (err) => {
|
||||||
|
reject(err);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (data != null) {
|
||||||
|
req.write(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
req.end();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* githubToken invokes the given URL, appending the audience parameter, using
|
||||||
|
* the provided token as authentication. This can only be run from inside a
|
||||||
|
* GitHub Action.
|
||||||
|
*/
|
||||||
|
static async githubToken({ url, audience, token }: GitHubTokenParameters): Promise<string> {
|
||||||
|
const requestURL = new URL(url);
|
||||||
|
|
||||||
|
// Append the audience value to the request.
|
||||||
|
const params = requestURL.searchParams;
|
||||||
|
params.set('audience', audience);
|
||||||
|
requestURL.search = params.toString();
|
||||||
|
|
||||||
|
// Make the request.
|
||||||
|
const opts = {
|
||||||
|
hostname: requestURL.hostname,
|
||||||
|
port: requestURL.port,
|
||||||
|
path: requestURL.pathname + requestURL.search,
|
||||||
|
method: 'GET',
|
||||||
|
headers: {
|
||||||
|
'Authorization': `Bearer ${token}`,
|
||||||
|
'Accept': 'application/json',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const resp = await Client.request(opts);
|
||||||
|
const parsed = JSON.parse(resp);
|
||||||
|
return parsed['value'];
|
||||||
|
} catch (err) {
|
||||||
|
throw new Error(`failed to generate GitHub OIDC token via ${url} (aud: ${audience}): ${err}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* googleFederatedToken generates a Google Cloud federated token using the
|
||||||
|
* provided OIDC token and Workload Identity Provider.
|
||||||
|
*/
|
||||||
|
static async googleFederatedToken({
|
||||||
|
providerID,
|
||||||
|
token,
|
||||||
|
}: GoogleFederatedTokenParameters): Promise<string> {
|
||||||
|
const stsURL = new URL('https://sts.googleapis.com/v1/token');
|
||||||
|
|
||||||
|
const data = {
|
||||||
|
audience: '//iam.googleapis.com/' + providerID,
|
||||||
|
grantType: 'urn:ietf:params:oauth:grant-type:token-exchange',
|
||||||
|
requestedTokenType: 'urn:ietf:params:oauth:token-type:access_token',
|
||||||
|
scope: 'https://www.googleapis.com/auth/cloud-platform',
|
||||||
|
subjectTokenType: 'urn:ietf:params:oauth:token-type:jwt',
|
||||||
|
subjectToken: token,
|
||||||
|
};
|
||||||
|
|
||||||
|
const opts = {
|
||||||
|
hostname: stsURL.hostname,
|
||||||
|
port: stsURL.port,
|
||||||
|
path: stsURL.pathname + stsURL.search,
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Accept': 'application/json',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const resp = await Client.request(opts, JSON.stringify(data));
|
||||||
|
const parsed = JSON.parse(resp);
|
||||||
|
return parsed['access_token'];
|
||||||
|
} catch (err) {
|
||||||
|
throw new Error(`failed to generate Google Cloud federated token for ${providerID}: ${err}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* googleAccessToken generates a Google Cloud access token for the provided
|
||||||
|
* service account email or unique id.
|
||||||
|
*/
|
||||||
|
static async googleAccessToken({
|
||||||
|
token,
|
||||||
|
serviceAccount,
|
||||||
|
delegates,
|
||||||
|
lifetime,
|
||||||
|
}: GoogleAccessTokenParameters): Promise<GoogleAccessTokenResponse> {
|
||||||
|
const serviceAccountID = `projects/-/serviceAccounts/${serviceAccount}`;
|
||||||
|
const tokenURL = new URL(
|
||||||
|
`https://iamcredentials.googleapis.com/v1/${serviceAccountID}:generateAccessToken`,
|
||||||
|
);
|
||||||
|
|
||||||
|
const data = {
|
||||||
|
delegates: delegates,
|
||||||
|
scope: 'https://www.googleapis.com/auth/cloud-platform',
|
||||||
|
lifetime: lifetime,
|
||||||
|
};
|
||||||
|
|
||||||
|
const opts = {
|
||||||
|
hostname: tokenURL.hostname,
|
||||||
|
port: tokenURL.port,
|
||||||
|
path: tokenURL.pathname + tokenURL.search,
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Authorization': `Bearer ${token}`,
|
||||||
|
'Accept': 'application/json',
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const resp = await Client.request(opts, JSON.stringify(data));
|
||||||
|
const parsed = JSON.parse(resp);
|
||||||
|
return {
|
||||||
|
accessToken: parsed['accessToken'],
|
||||||
|
expiration: parsed['expireTime'],
|
||||||
|
};
|
||||||
|
} catch (err) {
|
||||||
|
throw new Error(`failed to generate Google Cloud access token for ${serviceAccount}: ${err}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
79
src/main.ts
Normal file
79
src/main.ts
Normal file
@ -0,0 +1,79 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
import * as core from '@actions/core';
|
||||||
|
import { Client } from './client';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts a multi-line or comma-separated collection of strings into an array
|
||||||
|
* of trimmed strings.
|
||||||
|
*/
|
||||||
|
function explodeStrings(input: string): Array<string> {
|
||||||
|
if (input == null || input.length === 0) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const list = new Array<string>();
|
||||||
|
for (const line of input.split(`\n`)) {
|
||||||
|
for (const piece of line.split(',')) {
|
||||||
|
const entry = piece.trim();
|
||||||
|
if (entry !== '') {
|
||||||
|
list.push(entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Executes the main action, documented inline.
|
||||||
|
*/
|
||||||
|
async function run(): Promise<void> {
|
||||||
|
try {
|
||||||
|
// Load configuration.
|
||||||
|
const workloadIdentityProvider = core.getInput('workload_identity_provider', {
|
||||||
|
required: true,
|
||||||
|
});
|
||||||
|
const serviceAccount = core.getInput('service_account', { required: true });
|
||||||
|
const audience = core.getInput('audience');
|
||||||
|
const delegates = explodeStrings(core.getInput('delegates'));
|
||||||
|
const lifetime = core.getInput('lifetime');
|
||||||
|
|
||||||
|
// Extract the GitHub Actions OIDC token.
|
||||||
|
const requestToken = process.env.ACTIONS_ID_TOKEN_REQUEST_TOKEN;
|
||||||
|
if (!requestToken) {
|
||||||
|
throw `missing ACTIONS_ID_TOKEN_REQUEST_TOKEN`;
|
||||||
|
}
|
||||||
|
const requestURL = process.env.ACTIONS_ID_TOKEN_REQUEST_URL;
|
||||||
|
if (!requestURL) {
|
||||||
|
throw `missing ACTIONS_ID_TOKEN_REQUEST_URL`;
|
||||||
|
}
|
||||||
|
const githubOIDCToken = await Client.githubToken({
|
||||||
|
url: requestURL,
|
||||||
|
token: requestToken,
|
||||||
|
audience: audience,
|
||||||
|
});
|
||||||
|
core.setSecret(githubOIDCToken);
|
||||||
|
|
||||||
|
// Exchange the GitHub OIDC token for a Google Federated Token.
|
||||||
|
const googleFederatedToken = await Client.googleFederatedToken({
|
||||||
|
providerID: workloadIdentityProvider,
|
||||||
|
token: githubOIDCToken,
|
||||||
|
});
|
||||||
|
core.setSecret(googleFederatedToken);
|
||||||
|
|
||||||
|
// Exchange the Google Federated Token for an access token.
|
||||||
|
const { accessToken, expiration } = await Client.googleAccessToken({
|
||||||
|
token: googleFederatedToken,
|
||||||
|
serviceAccount: serviceAccount,
|
||||||
|
delegates: delegates,
|
||||||
|
lifetime: lifetime,
|
||||||
|
});
|
||||||
|
core.setSecret(accessToken);
|
||||||
|
core.setOutput('access_token', accessToken);
|
||||||
|
core.setOutput('expiration', expiration);
|
||||||
|
} catch (err) {
|
||||||
|
core.setFailed(`Action failed with error: ${err}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
run();
|
6
tests/client.test.ts
Normal file
6
tests/client.test.ts
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
import { expect } from 'chai';
|
||||||
|
import 'mocha';
|
||||||
|
|
||||||
|
describe('Client', () => {
|
||||||
|
it('todo');
|
||||||
|
});
|
15
tsconfig.json
Normal file
15
tsconfig.json
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"target": "es6",
|
||||||
|
"module": "commonjs",
|
||||||
|
"lib": [
|
||||||
|
"es6"
|
||||||
|
],
|
||||||
|
"outDir": "./dist",
|
||||||
|
"rootDir": "./src",
|
||||||
|
"strict": true,
|
||||||
|
"noImplicitAny": true,
|
||||||
|
"esModuleInterop": true
|
||||||
|
},
|
||||||
|
"exclude": ["node_modules", "**/*.test.ts"]
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user