Compare commits
	
		
			215 Commits 
		
	
	
	| Author | SHA1 | Date | 
|---|---|---|
|  | 28fdb31ff3 | |
|  | 2064b45545 | |
|  | 5e57cd1181 | |
|  | 97e31439e8 | |
|  | 3a0796b57f | |
|  | 5b7b28b1cc | |
|  | abc9fb3154 | |
|  | d468688814 | |
|  | a99b2f88fc | |
|  | 0d7fae8057 | |
|  | 9832253cb7 | |
|  | 09e05bbdf6 | |
|  | 4384bb23be | |
|  | 722c6681a5 | |
|  | 0bd2ba43c1 | |
|  | 18836d8e24 | |
|  | 997e14a133 | |
|  | 67d87fe2c8 | |
|  | 1287044d73 | |
|  | 9fa22322ec | |
|  | 738e2ed2fd | |
|  | 5497b8b59c | |
|  | 2d98472de9 | |
|  | 3c28a369b4 | |
|  | 072e0d514c | |
|  | 65c0768122 | |
|  | 3d793fe111 | |
|  | 9fc23e88e3 | |
|  | bdf14dc173 | |
|  | 21ddfdf39a | |
|  | d5f9e03fca | |
|  | 9b3389107c | |
|  | 184bdaa072 | |
|  | 5c6bc94683 | |
|  | caf4058643 | |
|  | ef38ec311a | |
|  | d52e8ef81c | |
|  | 9644ab7025 | |
|  | 7abd1d5126 | |
|  | 1a81202c4f | |
|  | d1ab30dc54 | |
|  | f25ff28d1c | |
|  | e0c62a93a1 | |
|  | 34f6d346b5 | |
|  | a547b56e5b | |
|  | b7f56fce8a | |
|  | 3d100841f6 | |
|  | 5713f8474e | |
|  | 0a8c41d6e2 | |
|  | 6d4b68b490 | |
|  | b7f1bbcce7 | |
|  | abd3abc99c | |
|  | ac38c32963 | |
|  | 74a5d14239 | |
|  | 2f4f00e4c6 | |
|  | 67c184546c | |
|  | 3d4cc89e85 | |
|  | 6cc823a6c4 | |
|  | d94e792124 | |
|  | 033db0da30 | |
|  | 09c2ae9716 | |
|  | ba56f006fc | |
|  | 75bf9a79af | |
|  | ed2698b25f | |
|  | 52ad1d2e01 | |
|  | f18ea97ee4 | |
|  | 07d7ee0dc6 | |
|  | 327cd5a69d | |
|  | e217ef3a2d | |
|  | 407f438d11 | |
|  | 28e19ee314 | |
|  | 7ca345011a | |
|  | 02b671aa02 | |
|  | 06895751d1 | |
|  | 02c9ff3be2 | |
|  | 5d8785b43a | |
|  | 7a65d3de5d | |
|  | bd2f40996a | |
|  | 9fd0581bf0 | |
|  | eefb7e8744 | |
|  | cb13d66af0 | |
|  | 1f36f5b7a2 | |
|  | bcb47c2c49 | |
|  | 8165a5b270 | |
|  | 29df2a9f8c | |
|  | 2b0e0e02ba | |
|  | fe7fd06c5d | |
|  | 3b8fed7e4b | |
|  | 90de1764cc | |
|  | 9780b0c442 | |
|  | 2fa130caf4 | |
|  | 5e87b2aca7 | |
|  | e0394952ce | |
|  | 9af18aa7d8 | |
|  | 668190adc5 | |
|  | be5150d9fe | |
|  | e80ebcad71 | |
|  | 75ee3eaf53 | |
|  | 793c19c8fc | |
|  | 30f019fb76 | |
|  | 0d4c9c5ea7 | |
|  | b29e14f6a9 | |
|  | 218a70c516 | |
|  | b8200806cf | |
|  | 27530a9fbb | |
|  | d072a60421 | |
|  | 7c627b5124 | |
|  | 787cfc6623 | |
|  | 8e66e916f8 | |
|  | 5ba5e97350 | |
|  | db77945884 | |
|  | bfd28af29d | |
|  | 70fccc794a | |
|  | d5eb4a7b25 | |
|  | d9ec9539b3 | |
|  | 406e1d7119 | |
|  | 834ded21d8 | |
|  | c7cb6a492e | |
|  | 7840e6ddd4 | |
|  | d910b14072 | |
|  | 7c17210d47 | |
|  | 117949aa8e | |
|  | 225e47902b | |
|  | 5f4866a30a | |
|  | 28283a6966 | |
|  | e92390c5fb | |
|  | 1e752e2293 | |
|  | 51c60978b0 | |
|  | 8f079fbb6c | |
|  | 16fa7681c3 | |
|  | 46d1619226 | |
|  | 8c291c5677 | |
|  | ec726f409d | |
|  | 5139682d94 | |
|  | 6d4e2ba5df | |
|  | 583cbb6dfd | |
|  | 566711b9b9 | |
|  | f1ee1338e0 | |
|  | 83a00bc1ab | |
|  | f405860ca0 | |
|  | 3d58c274f1 | |
|  | a3484d8199 | |
|  | 1f401f745b | |
|  | adb236cb13 | |
|  | 1220aa36aa | |
|  | 21a24919f5 | |
|  | df0d2135ce | |
|  | 9ed9eb7c45 | |
|  | e4302a2b09 | |
|  | a174191dfd | |
|  | b5b3868f21 | |
|  | 7137ecbd58 | |
|  | 6452b9a4db | |
|  | 495b903b08 | |
|  | b96c2c0282 | |
|  | 3055c413af | |
|  | b4bedf8053 | |
|  | be2bd2e073 | |
|  | b1ee45f5c7 | |
|  | 601e14e667 | |
|  | 194e6ce729 | |
|  | ba38666a08 | |
|  | eb1ca82ebd | |
|  | 343f7c4344 | |
|  | aad0f974f2 | |
|  | 2e0cd39144 | |
|  | 203bc9c4ef | |
|  | 2199648fc8 | |
|  | b489376173 | |
|  | 7c309e74e6 | |
|  | 0ccf222961 | |
|  | 56d703e106 | |
|  | 24d3b3519e | |
|  | e7ebaec3b4 | |
|  | 357a886c44 | |
|  | d76f68242b | |
|  | 98faccfdb6 | |
|  | 5a4d18a7d0 | |
|  | 0e7803459f | |
|  | a64852414f | |
|  | 9b251271cc | |
|  | 94f6005b3f | |
|  | 3c2fe176ab | |
|  | a5609cb39f | |
|  | e67ee94f2c | |
|  | 98341b6c68 | |
|  | 69b57c9b4d | |
|  | cf8514a651 | |
|  | 721f0dbb33 | |
|  | 4690724d0f | |
|  | 6938c21f0b | |
|  | 3fcfd2761a | |
|  | 553b6f090f | |
|  | 7d6435bfdc | |
|  | 6d5b470689 | |
|  | ab756cbbaa | |
|  | 3e7d5660ba | |
|  | f244684852 | |
|  | 413775f4e3 | |
|  | 6f98ce7512 | |
|  | a979406458 | |
|  | 60b7f106cc | |
|  | ba3ca1d0ac | |
|  | 039d2a5a8d | |
|  | cccd53a808 | |
|  | 9338656050 | |
|  | cc693296f2 | |
|  | 939ed8cac4 | |
|  | 0b2ed364aa | |
|  | 315bd6c458 | |
|  | 243b4ca640 | |
|  | 1c5e4f407b | |
|  | 7124cda302 | |
|  | 0a5a6d545f | |
|  | e6f576c226 | 
|  | @ -1,2 +1,12 @@ | ||||||
| /coverage | /coverage | ||||||
| /node_modules | 
 | ||||||
|  | # Dependency directories | ||||||
|  | node_modules/ | ||||||
|  | jspm_packages/ | ||||||
|  | 
 | ||||||
|  | # yarn v2 | ||||||
|  | .yarn/cache | ||||||
|  | .yarn/unplugged | ||||||
|  | .yarn/build-state.yml | ||||||
|  | .yarn/install-state.gz | ||||||
|  | .pnp.* | ||||||
|  |  | ||||||
|  | @ -0,0 +1,3 @@ | ||||||
|  | /dist/** | ||||||
|  | /coverage/** | ||||||
|  | /node_modules/** | ||||||
|  | @ -1,11 +1,12 @@ | ||||||
| { | { | ||||||
|   "env": { |   "env": { | ||||||
|     "node": true, |     "node": true, | ||||||
|     "es2021": true, |     "es6": true, | ||||||
|     "jest": true |     "jest": true | ||||||
|   }, |   }, | ||||||
|   "extends": [ |   "extends": [ | ||||||
|     "eslint:recommended", |     "eslint:recommended", | ||||||
|  |     "plugin:@typescript-eslint/eslint-recommended", | ||||||
|     "plugin:@typescript-eslint/recommended", |     "plugin:@typescript-eslint/recommended", | ||||||
|     "plugin:jest/recommended", |     "plugin:jest/recommended", | ||||||
|     "plugin:prettier/recommended" |     "plugin:prettier/recommended" | ||||||
|  |  | ||||||
|  | @ -1,2 +1,4 @@ | ||||||
|  | /.yarn/releases/** binary | ||||||
|  | /.yarn/plugins/** binary | ||||||
| /dist/** linguist-generated=true | /dist/** linguist-generated=true | ||||||
| /lib/** linguist-generated=true | /lib/** linguist-generated=true | ||||||
|  |  | ||||||
|  | @ -1 +0,0 @@ | ||||||
| *	@crazy-max |  | ||||||
|  | @ -0,0 +1,3 @@ | ||||||
|  | # Code of conduct | ||||||
|  | 
 | ||||||
|  | - [Moby community guidelines](https://github.com/moby/moby/blob/master/CONTRIBUTING.md#moby-community-guidelines) | ||||||
|  | @ -0,0 +1,101 @@ | ||||||
|  | # https://docs.github.com/en/communities/using-templates-to-encourage-useful-issues-and-pull-requests/syntax-for-githubs-form-schema | ||||||
|  | name: Bug Report | ||||||
|  | description: Report a bug | ||||||
|  | labels: | ||||||
|  |   - status/triage | ||||||
|  | 
 | ||||||
|  | body: | ||||||
|  |   - type: markdown | ||||||
|  |     attributes: | ||||||
|  |       value: | | ||||||
|  |         Thank you for taking the time to report a bug! | ||||||
|  |         If this is a security issue please report it to the [Docker Security team](mailto:security@docker.com). | ||||||
|  | 
 | ||||||
|  |   - type: checkboxes | ||||||
|  |     attributes: | ||||||
|  |       label: Contributing guidelines | ||||||
|  |       description: > | ||||||
|  |         Make sure you've read the contributing guidelines before proceeding. | ||||||
|  |       options: | ||||||
|  |         - label: I've read the [contributing guidelines](https://github.com/docker/login-action/blob/master/.github/CONTRIBUTING.md) and wholeheartedly agree | ||||||
|  |           required: true | ||||||
|  | 
 | ||||||
|  |   - type: checkboxes | ||||||
|  |     attributes: | ||||||
|  |       label: "I've found a bug, and:" | ||||||
|  |       description: | | ||||||
|  |         Make sure that your request fulfills all of the following requirements. | ||||||
|  |         If one requirement cannot be satisfied, explain in detail why. | ||||||
|  |       options: | ||||||
|  |         - label: The documentation does not mention anything about my problem | ||||||
|  |         - label: There are no open or closed issues that are related to my problem | ||||||
|  | 
 | ||||||
|  |   - type: textarea | ||||||
|  |     attributes: | ||||||
|  |       label: Description | ||||||
|  |       description: > | ||||||
|  |         Provide a brief description of the bug in 1-2 sentences. | ||||||
|  |     validations: | ||||||
|  |       required: true | ||||||
|  | 
 | ||||||
|  |   - type: textarea | ||||||
|  |     attributes: | ||||||
|  |       label: Expected behaviour | ||||||
|  |       description: > | ||||||
|  |         Describe precisely what you'd expect to happen. | ||||||
|  |     validations: | ||||||
|  |       required: true | ||||||
|  | 
 | ||||||
|  |   - type: textarea | ||||||
|  |     attributes: | ||||||
|  |       label: Actual behaviour | ||||||
|  |       description: > | ||||||
|  |         Describe precisely what is actually happening. | ||||||
|  |     validations: | ||||||
|  |       required: true | ||||||
|  | 
 | ||||||
|  |   - type: input | ||||||
|  |     attributes: | ||||||
|  |       label: Repository URL | ||||||
|  |       description: > | ||||||
|  |         Enter the URL of the repository where you are experiencing the | ||||||
|  |         issue. If your repository is private, provide a link to a minimal | ||||||
|  |         repository that reproduces the issue. | ||||||
|  | 
 | ||||||
|  |   - type: input | ||||||
|  |     attributes: | ||||||
|  |       label: Workflow run URL | ||||||
|  |       description: > | ||||||
|  |         Enter the URL of the GitHub Action workflow run if public (e.g. | ||||||
|  |         `https://github.com/<user>/<repo>/actions/runs/<id>`) | ||||||
|  | 
 | ||||||
|  |   - type: textarea | ||||||
|  |     attributes: | ||||||
|  |       label: YAML workflow | ||||||
|  |       description: | | ||||||
|  |         Provide the YAML of the workflow that's causing the issue. | ||||||
|  |         Make sure to remove any sensitive information. | ||||||
|  |       render: yaml | ||||||
|  |     validations: | ||||||
|  |       required: true | ||||||
|  | 
 | ||||||
|  |   - type: textarea | ||||||
|  |     attributes: | ||||||
|  |       label: Workflow logs | ||||||
|  |       description: > | ||||||
|  |         [Attach](https://docs.github.com/en/get-started/writing-on-github/working-with-advanced-formatting/attaching-files) | ||||||
|  |         the [log file of your workflow run](https://docs.github.com/en/actions/managing-workflow-runs/using-workflow-run-logs#downloading-logs) | ||||||
|  |         and make sure to remove any sensitive information. | ||||||
|  | 
 | ||||||
|  |   - type: textarea | ||||||
|  |     attributes: | ||||||
|  |       label: BuildKit logs | ||||||
|  |       description: > | ||||||
|  |         If applicable, provide the [BuildKit container logs](https://docs.docker.com/build/ci/github-actions/configure-builder/#buildkit-container-logs) | ||||||
|  |       render: text | ||||||
|  | 
 | ||||||
|  |   - type: textarea | ||||||
|  |     attributes: | ||||||
|  |       label: Additional info | ||||||
|  |       description: | | ||||||
|  |         Provide any additional information that could be useful. | ||||||
|  | @ -1,34 +0,0 @@ | ||||||
| --- |  | ||||||
| name: Bug report |  | ||||||
| about: Create a report to help us improve |  | ||||||
| --- |  | ||||||
| 
 |  | ||||||
| ### Behaviour |  | ||||||
| 
 |  | ||||||
| #### Steps to reproduce this issue |  | ||||||
| 
 |  | ||||||
| 1. |  | ||||||
| 2. |  | ||||||
| 3. |  | ||||||
| 
 |  | ||||||
| #### Expected behaviour |  | ||||||
| 
 |  | ||||||
| > Tell us what should happen |  | ||||||
| 
 |  | ||||||
| #### Actual behaviour |  | ||||||
| 
 |  | ||||||
| > Tell us what happens instead |  | ||||||
| 
 |  | ||||||
| ### Configuration |  | ||||||
| 
 |  | ||||||
| * Repository URL (if public):  |  | ||||||
| * Build URL (if public):  |  | ||||||
| 
 |  | ||||||
| ```yml |  | ||||||
| # paste your YAML workflow file here and remove sensitive data |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ### Logs |  | ||||||
| 
 |  | ||||||
| > Download the [log file of your build](https://docs.github.com/en/actions/managing-workflow-runs/using-workflow-run-logs#downloading-logs) |  | ||||||
| > and [attach it](https://docs.github.com/en/github/managing-your-work-on-github/file-attachments-on-issues-and-pull-requests) to this issue. |  | ||||||
|  | @ -0,0 +1,9 @@ | ||||||
|  | # https://docs.github.com/en/communities/using-templates-to-encourage-useful-issues-and-pull-requests/configuring-issue-templates-for-your-repository#configuring-the-template-chooser | ||||||
|  | blank_issues_enabled: true | ||||||
|  | contact_links: | ||||||
|  |   - name: Questions and Discussions | ||||||
|  |     url: https://github.com/docker/login-action/discussions/new | ||||||
|  |     about: Use Github Discussions to ask questions and/or open discussion topics. | ||||||
|  |   - name: Documentation | ||||||
|  |     url: https://docs.docker.com/build/ci/github-actions/ | ||||||
|  |     about: Read the documentation. | ||||||
|  | @ -0,0 +1,15 @@ | ||||||
|  | # https://docs.github.com/en/communities/using-templates-to-encourage-useful-issues-and-pull-requests/syntax-for-githubs-form-schema | ||||||
|  | name: Feature request | ||||||
|  | description: Missing functionality? Come tell us about it! | ||||||
|  | labels: | ||||||
|  |   - kind/enhancement | ||||||
|  |   - status/triage | ||||||
|  | 
 | ||||||
|  | body: | ||||||
|  |   - type: textarea | ||||||
|  |     id: description | ||||||
|  |     attributes: | ||||||
|  |       label: Description | ||||||
|  |       description: What is the feature you want to see? | ||||||
|  |     validations: | ||||||
|  |       required: true | ||||||
|  | @ -0,0 +1,12 @@ | ||||||
|  | # Reporting security issues | ||||||
|  | 
 | ||||||
|  | The project maintainers take security seriously. If you discover a security | ||||||
|  | issue, please bring it to their attention right away! | ||||||
|  | 
 | ||||||
|  | **Please _DO NOT_ file a public issue**, instead send your report privately to | ||||||
|  | [security@docker.com](mailto:security@docker.com). | ||||||
|  | 
 | ||||||
|  | Security reports are greatly appreciated, and we will publicly thank you for it. | ||||||
|  | We also like to send gifts—if you'd like Docker swag, make sure to let | ||||||
|  | us know. We currently do not offer a paid security bounty program, but are not | ||||||
|  | ruling it out in the future. | ||||||
|  | @ -1,29 +0,0 @@ | ||||||
| # Support [](https://isitmaintained.com/project/docker/login-action) |  | ||||||
| 
 |  | ||||||
| ## Reporting an issue |  | ||||||
| 
 |  | ||||||
| Please do a search in [open issues](https://github.com/docker/login-action/issues?utf8=%E2%9C%93&q=) to see if the issue or feature request has already been filed. |  | ||||||
| 
 |  | ||||||
| If you find your issue already exists, make relevant comments and add your [reaction](https://github.com/blog/2119-add-reactions-to-pull-requests-issues-and-comments). Use a reaction in place of a "+1" comment. |  | ||||||
| 
 |  | ||||||
| :+1: - upvote |  | ||||||
| 
 |  | ||||||
| :-1: - downvote |  | ||||||
| 
 |  | ||||||
| If you cannot find an existing issue that describes your bug or feature, submit an issue using the guidelines below. |  | ||||||
| 
 |  | ||||||
| ## Writing good bug reports and feature requests |  | ||||||
| 
 |  | ||||||
| File a single issue per problem and feature request. |  | ||||||
| 
 |  | ||||||
| * Do not enumerate multiple bugs or feature requests in the same issue. |  | ||||||
| * Do not add your issue as a comment to an existing issue unless it's for the identical input. Many issues look similar, but have different causes. |  | ||||||
| 
 |  | ||||||
| The more information you can provide, the more likely someone will be successful reproducing the issue and finding a fix. |  | ||||||
| 
 |  | ||||||
| You are now ready to [create a new issue](https://github.com/docker/login-action/issues/new/choose)! |  | ||||||
| 
 |  | ||||||
| ## Closure policy |  | ||||||
| 
 |  | ||||||
| * Issues that don't have the information requested above (when applicable) will be closed immediately and the poster directed to the support guidelines. |  | ||||||
| * Issues that go a week without a response from original poster are subject to closure at our discretion. |  | ||||||
|  | @ -11,6 +11,14 @@ updates: | ||||||
|     directory: "/" |     directory: "/" | ||||||
|     schedule: |     schedule: | ||||||
|       interval: "daily" |       interval: "daily" | ||||||
|  |     versioning-strategy: "increase" | ||||||
|  |     groups: | ||||||
|  |       aws-sdk-dependencies: | ||||||
|  |         patterns: | ||||||
|  |           - "*aws-sdk*" | ||||||
|  |       proxy-agent-dependencies: | ||||||
|  |         patterns: | ||||||
|  |           - "*-proxy-agent" | ||||||
|     allow: |     allow: | ||||||
|       - dependency-type: "production" |       - dependency-type: "production" | ||||||
|     labels: |     labels: | ||||||
|  |  | ||||||
|  | @ -1,5 +1,9 @@ | ||||||
| name: ci | name: ci | ||||||
| 
 | 
 | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.ref }} | ||||||
|  |   cancel-in-progress: true | ||||||
|  | 
 | ||||||
| on: | on: | ||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|   schedule: |   schedule: | ||||||
|  | @ -15,7 +19,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Stop docker |         name: Stop docker | ||||||
|         run: | |         run: | | ||||||
|  | @ -39,7 +43,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Login to GitHub Container Registry |         name: Login to GitHub Container Registry | ||||||
|         uses: ./ |         uses: ./ | ||||||
|  | @ -56,7 +60,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Login to GitHub Container Registry |         name: Login to GitHub Container Registry | ||||||
|         uses: ./ |         uses: ./ | ||||||
|  | @ -81,7 +85,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Login to ACR |         name: Login to ACR | ||||||
|         uses: ./ |         uses: ./ | ||||||
|  | @ -101,7 +105,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Login to Docker Hub |         name: Login to Docker Hub | ||||||
|         uses: ./ |         uses: ./ | ||||||
|  | @ -120,7 +124,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Login to ECR |         name: Login to ECR | ||||||
|         uses: ./ |         uses: ./ | ||||||
|  | @ -140,10 +144,10 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Configure AWS Credentials |         name: Configure AWS Credentials | ||||||
|         uses: aws-actions/configure-aws-credentials@v2 |         uses: aws-actions/configure-aws-credentials@v5 | ||||||
|         with: |         with: | ||||||
|           aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }} |           aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||||
|           aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }} |           aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||||
|  | @ -165,9 +169,10 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Login to Public ECR |         name: Login to Public ECR | ||||||
|  |         continue-on-error: ${{ matrix.os == 'windows-latest' }} | ||||||
|         uses: ./ |         uses: ./ | ||||||
|         with: |         with: | ||||||
|           registry: public.ecr.aws |           registry: public.ecr.aws | ||||||
|  | @ -187,16 +192,17 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Configure AWS Credentials |         name: Configure AWS Credentials | ||||||
|         uses: aws-actions/configure-aws-credentials@v2 |         uses: aws-actions/configure-aws-credentials@v5 | ||||||
|         with: |         with: | ||||||
|           aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }} |           aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||||
|           aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }} |           aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||||
|           aws-region: us-east-1 |           aws-region: us-east-1 | ||||||
|       - |       - | ||||||
|         name: Login to ECR |         name: Login to Public ECR | ||||||
|  |         continue-on-error: ${{ matrix.os == 'windows-latest' }} | ||||||
|         uses: ./ |         uses: ./ | ||||||
|         with: |         with: | ||||||
|           registry: public.ecr.aws |           registry: public.ecr.aws | ||||||
|  | @ -212,7 +218,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Login to GitHub Container Registry |         name: Login to GitHub Container Registry | ||||||
|         uses: ./ |         uses: ./ | ||||||
|  | @ -232,7 +238,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Login to GitLab |         name: Login to GitLab | ||||||
|         uses: ./ |         uses: ./ | ||||||
|  | @ -252,7 +258,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Login to Google Artifact Registry |         name: Login to Google Artifact Registry | ||||||
|         uses: ./ |         uses: ./ | ||||||
|  | @ -272,7 +278,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Login to Google Container Registry |         name: Login to Google Container Registry | ||||||
|         uses: ./ |         uses: ./ | ||||||
|  | @ -280,3 +286,73 @@ jobs: | ||||||
|           registry: gcr.io |           registry: gcr.io | ||||||
|           username: _json_key |           username: _json_key | ||||||
|           password: ${{ secrets.GCR_JSON_KEY }} |           password: ${{ secrets.GCR_JSON_KEY }} | ||||||
|  | 
 | ||||||
|  |   registry-auth: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Checkout | ||||||
|  |         uses: actions/checkout@v5 | ||||||
|  |       - | ||||||
|  |         name: Login to registries | ||||||
|  |         uses: ./ | ||||||
|  |         with: | ||||||
|  |           registry-auth: | | ||||||
|  |             - username: ${{ secrets.DOCKERHUB_USERNAME }} | ||||||
|  |               password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||||
|  |             - registry: ghcr.io | ||||||
|  |               username: ${{ github.actor }} | ||||||
|  |               password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |             - registry: public.ecr.aws | ||||||
|  |               username: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||||
|  |               password: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||||
|  |             - registry: registry.gitlab.com | ||||||
|  |               username: ${{ secrets.GITLAB_USERNAME }} | ||||||
|  |               password: ${{ secrets.GITLAB_TOKEN }} | ||||||
|  | 
 | ||||||
|  |   registry-auth-dup: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Checkout | ||||||
|  |         uses: actions/checkout@v5 | ||||||
|  |       - | ||||||
|  |         name: Login to registries | ||||||
|  |         uses: ./ | ||||||
|  |         with: | ||||||
|  |           registry-auth: | | ||||||
|  |             - registry: ghcr.io | ||||||
|  |               username: ${{ github.actor }} | ||||||
|  |               password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |             - registry: public.ecr.aws | ||||||
|  |               username: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||||
|  |               password: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||||
|  |             - registry: ghcr.io | ||||||
|  |               username: ${{ github.actor }} | ||||||
|  |               password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  | 
 | ||||||
|  |   registry-auth-exclusive: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Checkout | ||||||
|  |         uses: actions/checkout@v5 | ||||||
|  |       - | ||||||
|  |         name: Login to registries | ||||||
|  |         id: login | ||||||
|  |         continue-on-error: true | ||||||
|  |         uses: ./ | ||||||
|  |         with: | ||||||
|  |           registry: ghcr.io | ||||||
|  |           username: ${{ github.actor }} | ||||||
|  |           password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |           registry-auth: | | ||||||
|  |             - username: ${{ secrets.DOCKERHUB_USERNAME }} | ||||||
|  |               password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||||
|  |       - | ||||||
|  |         name: Check | ||||||
|  |         run: | | ||||||
|  |           if [ "${{ steps.login.outcome }}" != "failure" ] || [ "${{ steps.login.conclusion }}" != "success" ]; then | ||||||
|  |             echo "::error::Should have failed" | ||||||
|  |             exit 1 | ||||||
|  |           fi | ||||||
|  |  | ||||||
|  | @ -0,0 +1,50 @@ | ||||||
|  | name: codeql | ||||||
|  | 
 | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - 'master' | ||||||
|  |       - 'releases/v*' | ||||||
|  |     paths: | ||||||
|  |       - '.github/workflows/codeql.yml' | ||||||
|  |       - 'dist/**' | ||||||
|  |       - 'src/**' | ||||||
|  |   pull_request: | ||||||
|  |     paths: | ||||||
|  |       - '.github/workflows/codeql.yml' | ||||||
|  |       - 'dist/**' | ||||||
|  |       - 'src/**' | ||||||
|  | 
 | ||||||
|  | permissions: | ||||||
|  |   actions: read | ||||||
|  |   contents: read | ||||||
|  |   security-events: write | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   analyze: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     strategy: | ||||||
|  |       fail-fast: false | ||||||
|  |       matrix: | ||||||
|  |         language: | ||||||
|  |           - javascript-typescript | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Checkout | ||||||
|  |         uses: actions/checkout@v5 | ||||||
|  |       - | ||||||
|  |         name: Initialize CodeQL | ||||||
|  |         uses: github/codeql-action/init@v4 | ||||||
|  |         with: | ||||||
|  |           languages: ${{ matrix.language }} | ||||||
|  |           config: | | ||||||
|  |             paths: | ||||||
|  |               - src | ||||||
|  |       - | ||||||
|  |         name: Autobuild | ||||||
|  |         uses: github/codeql-action/autobuild@v4 | ||||||
|  |       - | ||||||
|  |         name: Perform CodeQL Analysis | ||||||
|  |         uses: github/codeql-action/analyze@v4 | ||||||
|  |         with: | ||||||
|  |           category: "/language:${{matrix.language}}" | ||||||
|  | @ -0,0 +1,17 @@ | ||||||
|  | name: pr-assign-author | ||||||
|  | 
 | ||||||
|  | permissions: | ||||||
|  |   contents: read | ||||||
|  | 
 | ||||||
|  | on: | ||||||
|  |   pull_request_target: | ||||||
|  |     types: | ||||||
|  |       - opened | ||||||
|  |       - reopened | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   run: | ||||||
|  |     uses: crazy-max/.github/.github/workflows/pr-assign-author.yml@1b673f36fad86812f538c1df9794904038a23cbf | ||||||
|  |     permissions: | ||||||
|  |       contents: read | ||||||
|  |       pull-requests: write | ||||||
|  | @ -0,0 +1,21 @@ | ||||||
|  | name: publish | ||||||
|  | 
 | ||||||
|  | on: | ||||||
|  |   release: | ||||||
|  |     types: | ||||||
|  |       - published | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   publish: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     permissions: | ||||||
|  |       contents: read | ||||||
|  |       id-token: write | ||||||
|  |       packages: write | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Checkout | ||||||
|  |         uses: actions/checkout@v5 | ||||||
|  |       - | ||||||
|  |         name: Publish | ||||||
|  |         uses: actions/publish-immutable-action@v0.0.4 | ||||||
|  | @ -1,5 +1,9 @@ | ||||||
| name: test | name: test | ||||||
| 
 | 
 | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.ref }} | ||||||
|  |   cancel-in-progress: true | ||||||
|  | 
 | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|  | @ -13,14 +17,16 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Test |         name: Test | ||||||
|         uses: docker/bake-action@v3 |         uses: docker/bake-action@v6 | ||||||
|         with: |         with: | ||||||
|  |           source: . | ||||||
|           targets: test |           targets: test | ||||||
|       - |       - | ||||||
|         name: Upload coverage |         name: Upload coverage | ||||||
|         uses: codecov/codecov-action@v3 |         uses: codecov/codecov-action@v5 | ||||||
|         with: |         with: | ||||||
|           file: ./coverage/clover.xml |           files: ./coverage/clover.xml | ||||||
|  |           token: ${{ secrets.CODECOV_TOKEN }} | ||||||
|  |  | ||||||
|  | @ -1,5 +1,9 @@ | ||||||
| name: validate | name: validate | ||||||
| 
 | 
 | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.ref }} | ||||||
|  |   cancel-in-progress: true | ||||||
|  | 
 | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|  | @ -11,16 +15,17 @@ jobs: | ||||||
|   prepare: |   prepare: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     outputs: |     outputs: | ||||||
|       targets: ${{ steps.targets.outputs.matrix }} |       targets: ${{ steps.generate.outputs.targets }} | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |       - | ||||||
|         name: Targets matrix |         name: List targets | ||||||
|         id: targets |         id: generate | ||||||
|         run: | |         uses: docker/bake-action/subaction/list-targets@v6 | ||||||
|           echo "matrix=$(docker buildx bake validate --print | jq -cr '.group.validate.targets')" >> $GITHUB_OUTPUT |         with: | ||||||
|  |           target: validate | ||||||
| 
 | 
 | ||||||
|   validate: |   validate: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  | @ -31,11 +36,8 @@ jobs: | ||||||
|       matrix: |       matrix: | ||||||
|         target: ${{ fromJson(needs.prepare.outputs.targets) }} |         target: ${{ fromJson(needs.prepare.outputs.targets) }} | ||||||
|     steps: |     steps: | ||||||
|       - |  | ||||||
|         name: Checkout |  | ||||||
|         uses: actions/checkout@v3 |  | ||||||
|       - |       - | ||||||
|         name: Validate |         name: Validate | ||||||
|         uses: docker/bake-action@v3 |         uses: docker/bake-action@v6 | ||||||
|         with: |         with: | ||||||
|           targets: ${{ matrix.target }} |           targets: ${{ matrix.target }} | ||||||
|  |  | ||||||
|  | @ -1,7 +1,5 @@ | ||||||
| node_modules/ | # https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore | ||||||
| lib |  | ||||||
| 
 | 
 | ||||||
| # Rest of the file pulled from https://github.com/github/gitignore/blob/master/Node.gitignore |  | ||||||
| # Logs | # Logs | ||||||
| logs | logs | ||||||
| *.log | *.log | ||||||
|  | @ -9,6 +7,7 @@ npm-debug.log* | ||||||
| yarn-debug.log* | yarn-debug.log* | ||||||
| yarn-error.log* | yarn-error.log* | ||||||
| lerna-debug.log* | lerna-debug.log* | ||||||
|  | .pnpm-debug.log* | ||||||
| 
 | 
 | ||||||
| # Diagnostic reports (https://nodejs.org/api/report.html) | # Diagnostic reports (https://nodejs.org/api/report.html) | ||||||
| report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json | report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json | ||||||
|  | @ -19,34 +18,14 @@ pids | ||||||
| *.seed | *.seed | ||||||
| *.pid.lock | *.pid.lock | ||||||
| 
 | 
 | ||||||
| # Directory for instrumented libs generated by jscoverage/JSCover |  | ||||||
| lib-cov |  | ||||||
| 
 |  | ||||||
| # Coverage directory used by tools like istanbul | # Coverage directory used by tools like istanbul | ||||||
| coverage | coverage | ||||||
| *.lcov | *.lcov | ||||||
| 
 | 
 | ||||||
| # nyc test coverage |  | ||||||
| .nyc_output |  | ||||||
| 
 |  | ||||||
| # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) |  | ||||||
| .grunt |  | ||||||
| 
 |  | ||||||
| # Bower dependency directory (https://bower.io/) |  | ||||||
| bower_components |  | ||||||
| 
 |  | ||||||
| # node-waf configuration |  | ||||||
| .lock-wscript |  | ||||||
| 
 |  | ||||||
| # Compiled binary addons (https://nodejs.org/api/addons.html) |  | ||||||
| build/Release |  | ||||||
| 
 |  | ||||||
| # Dependency directories | # Dependency directories | ||||||
|  | node_modules/ | ||||||
| jspm_packages/ | jspm_packages/ | ||||||
| 
 | 
 | ||||||
| # TypeScript v1 declaration files |  | ||||||
| typings/ |  | ||||||
| 
 |  | ||||||
| # TypeScript cache | # TypeScript cache | ||||||
| *.tsbuildinfo | *.tsbuildinfo | ||||||
| 
 | 
 | ||||||
|  | @ -56,36 +35,19 @@ typings/ | ||||||
| # Optional eslint cache | # Optional eslint cache | ||||||
| .eslintcache | .eslintcache | ||||||
| 
 | 
 | ||||||
| # Optional REPL history |  | ||||||
| .node_repl_history |  | ||||||
| 
 |  | ||||||
| # Output of 'npm pack' |  | ||||||
| *.tgz |  | ||||||
| 
 |  | ||||||
| # Yarn Integrity file | # Yarn Integrity file | ||||||
| .yarn-integrity | .yarn-integrity | ||||||
| 
 | 
 | ||||||
| # dotenv environment variables file | # dotenv environment variable files | ||||||
| .env | .env | ||||||
| .env.test | .env.development.local | ||||||
|  | .env.test.local | ||||||
|  | .env.production.local | ||||||
|  | .env.local | ||||||
| 
 | 
 | ||||||
| # parcel-bundler cache (https://parceljs.org/) | # yarn v2 | ||||||
| .cache | .yarn/cache | ||||||
| 
 | .yarn/unplugged | ||||||
| # next.js build output | .yarn/build-state.yml | ||||||
| .next | .yarn/install-state.gz | ||||||
| 
 | .pnp.* | ||||||
| # nuxt.js build output |  | ||||||
| .nuxt |  | ||||||
| 
 |  | ||||||
| # vuepress build output |  | ||||||
| .vuepress/dist |  | ||||||
| 
 |  | ||||||
| # Serverless directories |  | ||||||
| .serverless/ |  | ||||||
| 
 |  | ||||||
| # FuseBox cache |  | ||||||
| .fusebox/ |  | ||||||
| 
 |  | ||||||
| # DynamoDB Local files |  | ||||||
| .dynamodb/ |  | ||||||
|  |  | ||||||
|  | @ -0,0 +1,6 @@ | ||||||
|  | # Dependency directories | ||||||
|  | node_modules/ | ||||||
|  | jspm_packages/ | ||||||
|  | 
 | ||||||
|  | # yarn v2 | ||||||
|  | .yarn/ | ||||||
|  | @ -0,0 +1,17 @@ | ||||||
|  | # https://yarnpkg.com/configuration/yarnrc | ||||||
|  | 
 | ||||||
|  | compressionLevel: mixed | ||||||
|  | enableGlobalCache: false | ||||||
|  | enableHardenedMode: true | ||||||
|  | 
 | ||||||
|  | logFilters: | ||||||
|  |   - code: YN0013 | ||||||
|  |     level: discard | ||||||
|  |   - code: YN0019 | ||||||
|  |     level: discard | ||||||
|  |   - code: YN0076 | ||||||
|  |     level: discard | ||||||
|  |   - code: YN0086 | ||||||
|  |     level: discard | ||||||
|  | 
 | ||||||
|  | nodeLinker: node-modules | ||||||
							
								
								
									
										303
									
								
								README.md
								
								
								
								
							
							
						
						
									
										303
									
								
								README.md
								
								
								
								
							|  | @ -23,16 +23,19 @@ ___ | ||||||
|   * [AWS Public Elastic Container Registry (ECR)](#aws-public-elastic-container-registry-ecr) |   * [AWS Public Elastic Container Registry (ECR)](#aws-public-elastic-container-registry-ecr) | ||||||
|   * [OCI Oracle Cloud Infrastructure Registry (OCIR)](#oci-oracle-cloud-infrastructure-registry-ocir) |   * [OCI Oracle Cloud Infrastructure Registry (OCIR)](#oci-oracle-cloud-infrastructure-registry-ocir) | ||||||
|   * [Quay.io](#quayio) |   * [Quay.io](#quayio) | ||||||
|  |   * [DigitalOcean](#digitalocean-container-registry) | ||||||
|  |   * [Authenticate to multiple registries](#authenticate-to-multiple-registries) | ||||||
| * [Customizing](#customizing) | * [Customizing](#customizing) | ||||||
|   * [inputs](#inputs) |   * [inputs](#inputs) | ||||||
| * [Keep up-to-date with GitHub Dependabot](#keep-up-to-date-with-github-dependabot) | * [Contributing](#contributing) | ||||||
| 
 | 
 | ||||||
| ## Usage | ## Usage | ||||||
| 
 | 
 | ||||||
| ### Docker Hub | ### Docker Hub | ||||||
| 
 | 
 | ||||||
| To authenticate against [Docker Hub](https://hub.docker.com) it's strongly recommended to create a | When authenticating to [Docker Hub](https://hub.docker.com) with GitHub Actions, | ||||||
| [personal access token](https://docs.docker.com/docker-hub/access-tokens/) as an alternative to your password. | use a [personal access token](https://docs.docker.com/docker-hub/access-tokens/). | ||||||
|  | Don't use your account password. | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
| name: ci | name: ci | ||||||
|  | @ -47,17 +50,17 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to Docker Hub |         name: Login to Docker Hub | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           username: ${{ secrets.DOCKERHUB_USERNAME }} |           username: ${{ vars.DOCKERHUB_USERNAME }} | ||||||
|           password: ${{ secrets.DOCKERHUB_TOKEN }} |           password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### GitHub Container Registry | ### GitHub Container Registry | ||||||
| 
 | 
 | ||||||
| To authenticate against the [GitHub Container Registry](https://docs.github.com/en/packages/working-with-a-github-packages-registry/working-with-the-container-registry), | To authenticate to the [GitHub Container Registry](https://docs.github.com/en/packages/working-with-a-github-packages-registry/working-with-the-container-registry), | ||||||
| use the [`GITHUB_TOKEN`](https://docs.github.com/en/actions/reference/authentication-in-a-workflow) for the best | use the [`GITHUB_TOKEN`](https://docs.github.com/en/actions/reference/authentication-in-a-workflow) | ||||||
| security and experience. | secret. | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
| name: ci | name: ci | ||||||
|  | @ -72,7 +75,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to GitHub Container Registry |         name: Login to GitHub Container Registry | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: ghcr.io |           registry: ghcr.io | ||||||
|           username: ${{ github.actor }} |           username: ${{ github.actor }} | ||||||
|  | @ -100,20 +103,23 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to GitLab |         name: Login to GitLab | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: registry.gitlab.com |           registry: registry.gitlab.com | ||||||
|           username: ${{ secrets.GITLAB_USERNAME }} |           username: ${{ vars.GITLAB_USERNAME }} | ||||||
|           password: ${{ secrets.GITLAB_PASSWORD }} |           password: ${{ secrets.GITLAB_PASSWORD }} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| If you have [Two-Factor Authentication](https://gitlab.com/help/user/profile/account/two_factor_authentication) enabled, use a [Personal Access Token](https://gitlab.com/help/user/profile/personal_access_tokens) instead of a password. | If you have [Two-Factor Authentication](https://gitlab.com/help/user/profile/account/two_factor_authentication) | ||||||
|  | enabled, use a [Personal Access Token](https://gitlab.com/help/user/profile/personal_access_tokens) | ||||||
|  | instead of a password. | ||||||
| 
 | 
 | ||||||
| ### Azure Container Registry (ACR) | ### Azure Container Registry (ACR) | ||||||
| 
 | 
 | ||||||
| [Create a service principal](https://docs.microsoft.com/en-us/azure/container-registry/container-registry-auth-service-principal#create-a-service-principal) | [Create a service principal](https://docs.microsoft.com/en-us/azure/container-registry/container-registry-auth-service-principal#create-a-service-principal) | ||||||
| with access to your container registry through the [Azure CLI](https://docs.microsoft.com/en-us/cli/azure/install-azure-cli) | with access to your container registry through the [Azure CLI](https://docs.microsoft.com/en-us/cli/azure/install-azure-cli) | ||||||
| and take note of the generated service principal's ID (also called _client ID_) and password (also called _client secret_). | and take note of the generated service principal's ID (also called _client ID_) | ||||||
|  | and password (also called _client secret_). | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
| name: ci | name: ci | ||||||
|  | @ -128,10 +134,10 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to ACR |         name: Login to ACR | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: <registry-name>.azurecr.io |           registry: <registry-name>.azurecr.io | ||||||
|           username: ${{ secrets.AZURE_CLIENT_ID }} |           username: ${{ vars.AZURE_CLIENT_ID }} | ||||||
|           password: ${{ secrets.AZURE_CLIENT_SECRET }} |           password: ${{ secrets.AZURE_CLIENT_SECRET }} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|  | @ -139,16 +145,21 @@ jobs: | ||||||
| 
 | 
 | ||||||
| ### Google Container Registry (GCR) | ### Google Container Registry (GCR) | ||||||
| 
 | 
 | ||||||
| > [Google Artifact Registry](#google-artifact-registry-gar) is the evolution of Google Container Registry. As a | > [Google Artifact Registry](#google-artifact-registry-gar) is the evolution of | ||||||
| > fully-managed service with support for both container images and non-container artifacts. If you currently use | > Google Container Registry. As a fully-managed service with support for both | ||||||
| > Google Container Registry, use the information [on this page](https://cloud.google.com/artifact-registry/docs/transition/transition-from-gcr) | > container images and non-container artifacts. If you currently use Google | ||||||
|  | > Container Registry, use the information [on this page](https://cloud.google.com/artifact-registry/docs/transition/transition-from-gcr) | ||||||
| > to learn about transitioning to Google Artifact Registry.  | > to learn about transitioning to Google Artifact Registry.  | ||||||
| 
 | 
 | ||||||
| You can use either workload identity federation based keyless authentication or service account based authentication. | You can authenticate with workload identity federation or a service account. | ||||||
| 
 | 
 | ||||||
| #### Workload identity federation based authentication | #### Workload identity federation | ||||||
| 
 | 
 | ||||||
| Configure the workload identity federation for github actions in gcloud (for steps, [refer here](https://github.com/google-github-actions/auth#setting-up-workload-identity-federation)). In the steps, your service account should the ability to push to GCR. Then use google-github-actions/auth action for authentication using workload identity like below: | Configure the workload identity federation for GitHub Actions in Google Cloud, | ||||||
|  | [see here](https://github.com/google-github-actions/auth#setting-up-workload-identity-federation). | ||||||
|  | Your service account must have permission to push to GCR. Use the | ||||||
|  | `google-github-actions/auth` action to authenticate using workload identity as | ||||||
|  | shown in the following example: | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
| name: ci | name: ci | ||||||
|  | @ -161,33 +172,35 @@ jobs: | ||||||
|   login: |   login: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - id: 'auth' |     - | ||||||
|       name: 'Authenticate to Google Cloud' |       name: Authenticate to Google Cloud | ||||||
|       uses: 'google-github-actions/auth@v0' |       id: auth | ||||||
|  |       uses: google-github-actions/auth@v1 | ||||||
|       with: |       with: | ||||||
|         token_format: 'access_token' |         token_format: access_token | ||||||
|         workload_identity_provider: '<workload_identity_provider>' |         workload_identity_provider: <workload_identity_provider> | ||||||
|         service_account: '<service_account>' |         service_account: <service_account> | ||||||
| 
 |     - | ||||||
|     - name: Login to GCR |       name: Login to GCR | ||||||
|       uses: docker/login-action@v2 |       uses: docker/login-action@v3 | ||||||
|       with: |       with: | ||||||
|         registry: gcr.io |         registry: gcr.io | ||||||
|         username: oauth2accesstoken |         username: oauth2accesstoken | ||||||
|         password: ${{ steps.auth.outputs.access_token }} |         password: ${{ steps.auth.outputs.access_token }} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| > Replace `<workload_identity_provider>` with configured workload identity provider. For steps to configure, [refer here](https://github.com/google-github-actions/auth#setting-up-workload-identity-federation). | > Replace `<workload_identity_provider>` with configured workload identity | ||||||
|  | > provider. For steps to configure, [see here](https://github.com/google-github-actions/auth#setting-up-workload-identity-federation). | ||||||
| 
 | 
 | ||||||
| > Replace `<service_account>` with configured service account in workload identity provider which has access to push to GCR | > Replace `<service_account>` with configured service account in workload | ||||||
|  | > identity provider which has access to push to GCR | ||||||
| 
 | 
 | ||||||
| #### Service account based authentication | #### Service account based authentication | ||||||
| 
 | 
 | ||||||
| Use a service account with the ability to push to GCR and [configure access control](https://cloud.google.com/container-registry/docs/access-control). | Use a service account with permission to push to GCR and [configure access control](https://cloud.google.com/container-registry/docs/access-control). | ||||||
| Then create and download the JSON key for this service account and save content of `.json` file | Download the key for the service account as a JSON file. Save the contents of | ||||||
| [as a secret](https://docs.github.com/en/actions/configuring-and-managing-workflows/creating-and-storing-encrypted-secrets#creating-encrypted-secrets-for-a-repository) | the file [as a secret](https://docs.github.com/en/actions/configuring-and-managing-workflows/creating-and-storing-encrypted-secrets#creating-encrypted-secrets-for-a-repository) | ||||||
| called `GCR_JSON_KEY` in your GitHub repo. Ensure you set the username to `_json_key`, | named `GCR_JSON_KEY` in your GitHub repository. Set the username to `_json_key`. | ||||||
| or `_json_key_base64` if you use a base64-encoded key. |  | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
| name: ci | name: ci | ||||||
|  | @ -202,7 +215,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to GCR |         name: Login to GCR | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: gcr.io |           registry: gcr.io | ||||||
|           username: _json_key |           username: _json_key | ||||||
|  | @ -211,11 +224,13 @@ jobs: | ||||||
| 
 | 
 | ||||||
| ### Google Artifact Registry (GAR) | ### Google Artifact Registry (GAR) | ||||||
| 
 | 
 | ||||||
| You can use either workload identity federation based keyless authentication or  service account based authentication. | You can authenticate with workload identity federation or a service account. | ||||||
| 
 | 
 | ||||||
| #### Workload identity federation based authentication | #### Workload identity federation | ||||||
| 
 | 
 | ||||||
| Configure the workload identity federation for github actions in gcloud (for steps, [refer here](https://github.com/google-github-actions/auth#setting-up-workload-identity-federation)). In the steps, your service account should the ability to push to GAR. Then use google-github-actions/auth action for authentication using workload identity like below: | Your service account must have permission to push to GAR. Use the | ||||||
|  | `google-github-actions/auth` action to authenticate using workload identity as | ||||||
|  | shown in the following example: | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
| name: ci | name: ci | ||||||
|  | @ -228,34 +243,38 @@ jobs: | ||||||
|   login: |   login: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - id: 'auth' |       - | ||||||
|         name: 'Authenticate to Google Cloud' |         name: Authenticate to Google Cloud | ||||||
|         uses: 'google-github-actions/auth@v0' |         id: auth | ||||||
|  |         uses: google-github-actions/auth@v1 | ||||||
|         with: |         with: | ||||||
|           token_format: 'access_token' |           token_format: access_token | ||||||
|           workload_identity_provider: '<workload_identity_provider>' |           workload_identity_provider: <workload_identity_provider> | ||||||
|           service_account: '<service_account>' |           service_account: <service_account> | ||||||
|        |       - | ||||||
|       - name: Login to GAR |         name: Login to GAR | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: <location>-docker.pkg.dev |           registry: <location>-docker.pkg.dev | ||||||
|           username: oauth2accesstoken |           username: oauth2accesstoken | ||||||
|           password: ${{ steps.auth.outputs.access_token }} |           password: ${{ steps.auth.outputs.access_token }} | ||||||
| ``` | ``` | ||||||
| > Replace `<workload_identity_provider>` with configured workload identity provider |  | ||||||
| 
 | 
 | ||||||
| > Replace `<service_account>` with configured service account in workload identity provider which has access to push to GCR | > Replace `<workload_identity_provider>` with configured workload identity | ||||||
|  | > provider | ||||||
|  | 
 | ||||||
|  | > Replace `<service_account>` with configured service account in workload | ||||||
|  | > identity provider which has access to push to GCR | ||||||
| 
 | 
 | ||||||
| > Replace `<location>` with the regional or multi-regional [location](https://cloud.google.com/artifact-registry/docs/repo-organize#locations) | > Replace `<location>` with the regional or multi-regional [location](https://cloud.google.com/artifact-registry/docs/repo-organize#locations) | ||||||
| > of the repository where the image is stored. | > of the repository where the image is stored. | ||||||
| 
 | 
 | ||||||
| #### Service account based authentication | #### Service account based authentication | ||||||
| 
 | 
 | ||||||
| Use a service account with the ability to push to GAR and [configure access control](https://cloud.google.com/artifact-registry/docs/access-control). | Use a service account with permission to push to GAR and [configure access control](https://cloud.google.com/artifact-registry/docs/access-control). | ||||||
| Then create and download the JSON key for this service account and save content of `.json` file | Download the key for the service account as a JSON file. Save the contents of | ||||||
| [as a secret](https://docs.github.com/en/actions/configuring-and-managing-workflows/creating-and-storing-encrypted-secrets#creating-encrypted-secrets-for-a-repository) | the file [as a secret](https://docs.github.com/en/actions/configuring-and-managing-workflows/creating-and-storing-encrypted-secrets#creating-encrypted-secrets-for-a-repository) | ||||||
| called `GAR_JSON_KEY` in your GitHub repo. Ensure you set the username to `_json_key`, | named `GAR_JSON_KEY` in your GitHub repository. Set the username to `_json_key`, | ||||||
| or `_json_key_base64` if you use a base64-encoded key. | or `_json_key_base64` if you use a base64-encoded key. | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
|  | @ -271,7 +290,7 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to GAR |         name: Login to GAR | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: <location>-docker.pkg.dev |           registry: <location>-docker.pkg.dev | ||||||
|           username: _json_key |           username: _json_key | ||||||
|  | @ -283,8 +302,8 @@ jobs: | ||||||
| 
 | 
 | ||||||
| ### AWS Elastic Container Registry (ECR) | ### AWS Elastic Container Registry (ECR) | ||||||
| 
 | 
 | ||||||
| Use an IAM user with the ability to [push to ECR with `AmazonEC2ContainerRegistryPowerUser` managed policy for example](https://docs.aws.amazon.com/AmazonECR/latest/userguide/ecr_managed_policies.html#AmazonEC2ContainerRegistryPowerUser). | Use an IAM user with the ability to [push to ECR with `AmazonEC2ContainerRegistryPowerUser` managed policy for example](https://docs.aws.amazon.com/AmazonECR/latest/userguide/security-iam-awsmanpol.html#security-iam-awsmanpol-AmazonEC2ContainerRegistryPowerUser). | ||||||
| Then create and download access keys and save `AWS_ACCESS_KEY_ID` and `AWS_SECRET_ACCESS_KEY` [as secrets](https://docs.github.com/en/actions/configuring-and-managing-workflows/creating-and-storing-encrypted-secrets#creating-encrypted-secrets-for-a-repository) | Download the access keys and save them as `AWS_ACCESS_KEY_ID` and `AWS_SECRET_ACCESS_KEY` [as secrets](https://docs.github.com/en/actions/configuring-and-managing-workflows/creating-and-storing-encrypted-secrets#creating-encrypted-secrets-for-a-repository) | ||||||
| in your GitHub repo. | in your GitHub repo. | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
|  | @ -300,15 +319,15 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to ECR |         name: Login to ECR | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: <aws-account-number>.dkr.ecr.<region>.amazonaws.com |           registry: <aws-account-number>.dkr.ecr.<region>.amazonaws.com | ||||||
|           username: ${{ secrets.AWS_ACCESS_KEY_ID }} |           username: ${{ vars.AWS_ACCESS_KEY_ID }} | ||||||
|           password: ${{ secrets.AWS_SECRET_ACCESS_KEY }} |           password: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| If you need to log in to Amazon ECR registries associated with other accounts, you can use the `AWS_ACCOUNT_IDS` | If you need to log in to Amazon ECR registries associated with other accounts, | ||||||
| environment variable: | you can use the `AWS_ACCOUNT_IDS` environment variable: | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
| name: ci | name: ci | ||||||
|  | @ -323,10 +342,10 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to ECR |         name: Login to ECR | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: <aws-account-number>.dkr.ecr.<region>.amazonaws.com |           registry: <aws-account-number>.dkr.ecr.<region>.amazonaws.com | ||||||
|           username: ${{ secrets.AWS_ACCESS_KEY_ID }} |           username: ${{ vars.AWS_ACCESS_KEY_ID }} | ||||||
|           password: ${{ secrets.AWS_SECRET_ACCESS_KEY }} |           password: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||||
|         env: |         env: | ||||||
|           AWS_ACCOUNT_IDS: 012345678910,023456789012 |           AWS_ACCOUNT_IDS: 012345678910,023456789012 | ||||||
|  | @ -334,8 +353,8 @@ jobs: | ||||||
| 
 | 
 | ||||||
| > Only available with [AWS CLI version 1](https://docs.aws.amazon.com/cli/latest/reference/ecr/get-login.html) | > Only available with [AWS CLI version 1](https://docs.aws.amazon.com/cli/latest/reference/ecr/get-login.html) | ||||||
| 
 | 
 | ||||||
| You can also use the [Configure AWS Credentials](https://github.com/aws-actions/configure-aws-credentials) action in | You can also use the [Configure AWS Credentials](https://github.com/aws-actions/configure-aws-credentials) | ||||||
| combination with this action: | action in combination with this action: | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
| name: ci | name: ci | ||||||
|  | @ -350,14 +369,14 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Configure AWS Credentials |         name: Configure AWS Credentials | ||||||
|         uses: aws-actions/configure-aws-credentials@v1 |         uses: aws-actions/configure-aws-credentials@v4 | ||||||
|         with: |         with: | ||||||
|           aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }} |           aws-access-key-id: ${{ vars.AWS_ACCESS_KEY_ID }} | ||||||
|           aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }} |           aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||||
|           aws-region: <region> |           aws-region: <region> | ||||||
|       - |       - | ||||||
|         name: Login to ECR |         name: Login to ECR | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: <aws-account-number>.dkr.ecr.<region>.amazonaws.com |           registry: <aws-account-number>.dkr.ecr.<region>.amazonaws.com | ||||||
| ``` | ``` | ||||||
|  | @ -366,9 +385,10 @@ jobs: | ||||||
| 
 | 
 | ||||||
| ### AWS Public Elastic Container Registry (ECR) | ### AWS Public Elastic Container Registry (ECR) | ||||||
| 
 | 
 | ||||||
| Use an IAM user with the ability to [push to ECR Public with `AmazonElasticContainerRegistryPublicPowerUser` managed policy for example](https://docs.aws.amazon.com/AmazonECR/latest/public/public-ecr-managed-policies.html#AmazonElasticContainerRegistryPublicPowerUser). | Use an IAM user with permission to push to ECR Public, for example using [managed policies](https://docs.aws.amazon.com/AmazonECR/latest/userguide/security-iam-awsmanpol.html#security-iam-awsmanpol-AmazonEC2ContainerRegistryPowerUser). | ||||||
| Then create and download access keys and save `AWS_ACCESS_KEY_ID` and `AWS_SECRET_ACCESS_KEY` [as secrets](https://docs.github.com/en/actions/configuring-and-managing-workflows/creating-and-storing-encrypted-secrets#creating-encrypted-secrets-for-a-repository) | Download the access keys and save them as `AWS_ACCESS_KEY_ID` and | ||||||
| in your GitHub repo. | `AWS_SECRET_ACCESS_KEY` [secrets](https://docs.github.com/en/actions/configuring-and-managing-workflows/creating-and-storing-encrypted-secrets#creating-encrypted-secrets-for-a-repository) | ||||||
|  | in your GitHub repository. | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
| name: ci | name: ci | ||||||
|  | @ -383,10 +403,10 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to Public ECR |         name: Login to Public ECR | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: public.ecr.aws |           registry: public.ecr.aws | ||||||
|           username: ${{ secrets.AWS_ACCESS_KEY_ID }} |           username: ${{ vars.AWS_ACCESS_KEY_ID }} | ||||||
|           password: ${{ secrets.AWS_SECRET_ACCESS_KEY }} |           password: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||||
|         env: |         env: | ||||||
|           AWS_REGION: <region> |           AWS_REGION: <region> | ||||||
|  | @ -417,10 +437,10 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to OCIR |         name: Login to OCIR | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: <region>.ocir.io |           registry: <region>.ocir.io | ||||||
|           username: ${{ secrets.OCI_USERNAME }} |           username: ${{ vars.OCI_USERNAME }} | ||||||
|           password: ${{ secrets.OCI_TOKEN }} |           password: ${{ secrets.OCI_TOKEN }} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|  | @ -428,7 +448,8 @@ jobs: | ||||||
| 
 | 
 | ||||||
| ### Quay.io | ### Quay.io | ||||||
| 
 | 
 | ||||||
| Use a [Robot account](https://docs.quay.io/glossary/robot-accounts.html) with the ability to push to a public/private Quay.io repository. | Use a [Robot account](https://docs.quay.io/glossary/robot-accounts.html) with | ||||||
|  | permission to push to a Quay.io repository. | ||||||
| 
 | 
 | ||||||
| ```yaml | ```yaml | ||||||
| name: ci | name: ci | ||||||
|  | @ -443,39 +464,119 @@ jobs: | ||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Login to Quay.io |         name: Login to Quay.io | ||||||
|         uses: docker/login-action@v2 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: quay.io |           registry: quay.io | ||||||
|           username: ${{ secrets.QUAY_USERNAME }} |           username: ${{ vars.QUAY_USERNAME }} | ||||||
|           password: ${{ secrets.QUAY_ROBOT_TOKEN }} |           password: ${{ secrets.QUAY_ROBOT_TOKEN }} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|  | ### DigitalOcean Container Registry | ||||||
|  | 
 | ||||||
|  | Use your DigitalOcean registered email address and an API access token to authenticate. | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | name: ci | ||||||
|  | 
 | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: main | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   login: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Login to DigitalOcean Container Registry | ||||||
|  |         uses: docker/login-action@v3 | ||||||
|  |         with: | ||||||
|  |           registry: registry.digitalocean.com | ||||||
|  |           username: ${{ vars.DIGITALOCEAN_USERNAME }} | ||||||
|  |           password: ${{ secrets.DIGITALOCEAN_ACCESS_TOKEN }} | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### Authenticate to multiple registries | ||||||
|  | 
 | ||||||
|  | To authenticate against multiple registries, you can specify the login-action | ||||||
|  | step multiple times in your workflow: | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | name: ci | ||||||
|  | 
 | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: main | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   login: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Login to Docker Hub | ||||||
|  |         uses: docker/login-action@v3 | ||||||
|  |         with: | ||||||
|  |           username: ${{ vars.DOCKERHUB_USERNAME }} | ||||||
|  |           password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||||
|  |       - | ||||||
|  |         name: Login to GitHub Container Registry | ||||||
|  |         uses: docker/login-action@v3 | ||||||
|  |         with: | ||||||
|  |           registry: ghcr.io | ||||||
|  |           username: ${{ github.actor }} | ||||||
|  |           password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | You can also use the `registry-auth` input for raw authentication to | ||||||
|  | registries, defined as YAML objects. Each object can contain `registry`, | ||||||
|  | `username`, `password` and `ecr` keys similar to current inputs: | ||||||
|  | 
 | ||||||
|  | > [!WARNING] | ||||||
|  | > We don't recommend using this method, it's better to use the action multiple | ||||||
|  | > times as shown above. | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | name: ci | ||||||
|  | 
 | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: main | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   login: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Login to registries | ||||||
|  |         uses: docker/login-action@v3 | ||||||
|  |         with: | ||||||
|  |           registry-auth: | | ||||||
|  |             - username: ${{ vars.DOCKERHUB_USERNAME }} | ||||||
|  |               password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||||
|  |             - registry: ghcr.io | ||||||
|  |               username: ${{ github.actor }} | ||||||
|  |               password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
| ## Customizing | ## Customizing | ||||||
| 
 | 
 | ||||||
| ### inputs | ### inputs | ||||||
| 
 | 
 | ||||||
| Following inputs can be used as `step.with` keys | The following inputs can be used as `step.with` keys: | ||||||
| 
 | 
 | ||||||
| | Name             | Type    | Default                     | Description                        | | | Name            | Type   | Default     | Description                                                                   | | ||||||
| |------------------|---------|-----------------------------|------------------------------------| | |-----------------|--------|-------------|-------------------------------------------------------------------------------| | ||||||
| | `registry`       | String  |                             | Server address of Docker registry. If not set then will default to Docker Hub | | | `registry`      | String | `docker.io` | Server address of Docker registry. If not set then will default to Docker Hub | | ||||||
| | `username`       | String  |                             | Username used to log against the Docker registry | | | `username`      | String |             | Username for authenticating to the Docker registry                            | | ||||||
| | `password`       | String  |                             | Password or personal access token used to log against the Docker registry | | | `password`      | String |             | Password or personal access token for authenticating the Docker registry      | | ||||||
| | `ecr`            | String  | `auto`                      | Specifies whether the given registry is ECR (`auto`, `true` or `false`) | | | `ecr`           | String | `auto`      | Specifies whether the given registry is ECR (`auto`, `true` or `false`)       | | ||||||
| | `logout`         | Bool    | `true`                      | Log out from the Docker registry at the end of a job | | | `logout`        | Bool   | `true`      | Log out from the Docker registry at the end of a job                          | | ||||||
|  | | `registry-auth` | YAML   |             | Raw authentication to registries, defined as YAML objects                     | | ||||||
| 
 | 
 | ||||||
| ## Keep up-to-date with GitHub Dependabot | > [!NOTE] | ||||||
|  | > The `registry-auth` input is mutually exclusive with `registry`, `username`, | ||||||
|  | > `password` and `ecr` inputs. | ||||||
| 
 | 
 | ||||||
| Since [Dependabot](https://docs.github.com/en/github/administering-a-repository/keeping-your-actions-up-to-date-with-github-dependabot) | ## Contributing | ||||||
| has [native GitHub Actions support](https://docs.github.com/en/github/administering-a-repository/configuration-options-for-dependency-updates#package-ecosystem), |  | ||||||
| to enable it on your GitHub repo all you need to do is add the `.github/dependabot.yml` file: |  | ||||||
| 
 | 
 | ||||||
| ```yaml | Want to contribute? Awesome! You can find information about contributing to | ||||||
| version: 2 | this project in the [CONTRIBUTING.md](/.github/CONTRIBUTING.md) | ||||||
| updates: |  | ||||||
|   # Maintain dependencies for GitHub Actions |  | ||||||
|   - package-ecosystem: "github-actions" |  | ||||||
|     directory: "/" |  | ||||||
|     schedule: |  | ||||||
|       interval: "daily" |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| import {beforeEach, describe, expect, jest, test} from '@jest/globals'; | import {beforeEach, describe, expect, jest, test} from '@jest/globals'; | ||||||
| import {AuthorizationData} from '@aws-sdk/client-ecr'; | import {AuthorizationData} from '@aws-sdk/client-ecr'; | ||||||
|  | 
 | ||||||
| import * as aws from '../src/aws'; | import * as aws from '../src/aws'; | ||||||
| 
 | 
 | ||||||
| describe('isECR', () => { | describe('isECR', () => { | ||||||
|  | @ -9,7 +10,9 @@ describe('isECR', () => { | ||||||
|     ['012345678901.dkr.ecr.eu-west-3.amazonaws.com', true], |     ['012345678901.dkr.ecr.eu-west-3.amazonaws.com', true], | ||||||
|     ['876820548815.dkr.ecr.cn-north-1.amazonaws.com.cn', true], |     ['876820548815.dkr.ecr.cn-north-1.amazonaws.com.cn', true], | ||||||
|     ['390948362332.dkr.ecr.cn-northwest-1.amazonaws.com.cn', true], |     ['390948362332.dkr.ecr.cn-northwest-1.amazonaws.com.cn', true], | ||||||
|     ['public.ecr.aws', true] |     ['012345678901.dkr-ecr.eu-north-1.on.aws', true], | ||||||
|  |     ['public.ecr.aws', true], | ||||||
|  |     ['ecr-public.aws.com', true] | ||||||
|   ])('given registry %p', async (registry, expected) => { |   ])('given registry %p', async (registry, expected) => { | ||||||
|     expect(aws.isECR(registry)).toEqual(expected); |     expect(aws.isECR(registry)).toEqual(expected); | ||||||
|   }); |   }); | ||||||
|  | @ -22,7 +25,9 @@ describe('isPubECR', () => { | ||||||
|     ['012345678901.dkr.ecr.eu-west-3.amazonaws.com', false], |     ['012345678901.dkr.ecr.eu-west-3.amazonaws.com', false], | ||||||
|     ['876820548815.dkr.ecr.cn-north-1.amazonaws.com.cn', false], |     ['876820548815.dkr.ecr.cn-north-1.amazonaws.com.cn', false], | ||||||
|     ['390948362332.dkr.ecr.cn-northwest-1.amazonaws.com.cn', false], |     ['390948362332.dkr.ecr.cn-northwest-1.amazonaws.com.cn', false], | ||||||
|     ['public.ecr.aws', true] |     ['012345678901.dkr-ecr.eu-north-1.on.aws', false], | ||||||
|  |     ['public.ecr.aws', true], | ||||||
|  |     ['ecr-public.aws.com', true] | ||||||
|   ])('given registry %p', async (registry, expected) => { |   ])('given registry %p', async (registry, expected) => { | ||||||
|     expect(aws.isPubECR(registry)).toEqual(expected); |     expect(aws.isPubECR(registry)).toEqual(expected); | ||||||
|   }); |   }); | ||||||
|  | @ -33,6 +38,7 @@ describe('getRegion', () => { | ||||||
|     ['012345678901.dkr.ecr.eu-west-3.amazonaws.com', 'eu-west-3'], |     ['012345678901.dkr.ecr.eu-west-3.amazonaws.com', 'eu-west-3'], | ||||||
|     ['876820548815.dkr.ecr.cn-north-1.amazonaws.com.cn', 'cn-north-1'], |     ['876820548815.dkr.ecr.cn-north-1.amazonaws.com.cn', 'cn-north-1'], | ||||||
|     ['390948362332.dkr.ecr.cn-northwest-1.amazonaws.com.cn', 'cn-northwest-1'], |     ['390948362332.dkr.ecr.cn-northwest-1.amazonaws.com.cn', 'cn-northwest-1'], | ||||||
|  |     ['012345678901.dkr-ecr.eu-north-1.on.aws', 'eu-north-1'], | ||||||
|     ['public.ecr.aws', 'us-east-1'] |     ['public.ecr.aws', 'us-east-1'] | ||||||
|   ])('given registry %p', async (registry, expected) => { |   ])('given registry %p', async (registry, expected) => { | ||||||
|     expect(aws.getRegion(registry)).toEqual(expected); |     expect(aws.getRegion(registry)).toEqual(expected); | ||||||
|  | @ -45,6 +51,7 @@ describe('getAccountIDs', () => { | ||||||
|     ['012345678901.dkr.ecr.eu-west-3.amazonaws.com', '012345678910,023456789012', ['012345678901', '012345678910', '023456789012']], |     ['012345678901.dkr.ecr.eu-west-3.amazonaws.com', '012345678910,023456789012', ['012345678901', '012345678910', '023456789012']], | ||||||
|     ['012345678901.dkr.ecr.eu-west-3.amazonaws.com', '012345678901,012345678910,023456789012', ['012345678901', '012345678910', '023456789012']], |     ['012345678901.dkr.ecr.eu-west-3.amazonaws.com', '012345678901,012345678910,023456789012', ['012345678901', '012345678910', '023456789012']], | ||||||
|     ['390948362332.dkr.ecr.cn-northwest-1.amazonaws.com.cn', '012345678910,023456789012', ['390948362332', '012345678910', '023456789012']], |     ['390948362332.dkr.ecr.cn-northwest-1.amazonaws.com.cn', '012345678910,023456789012', ['390948362332', '012345678910', '023456789012']], | ||||||
|  |     ['876820548815.dkr-ecr.eu-north-1.on.aws', '012345678910,023456789012', ['876820548815', '012345678910', '023456789012']], | ||||||
|     ['public.ecr.aws', undefined, []] |     ['public.ecr.aws', undefined, []] | ||||||
|   ])('given registry %p', async (registry, accountIDsEnv, expected) => { |   ])('given registry %p', async (registry, accountIDsEnv, expected) => { | ||||||
|     if (accountIDsEnv) { |     if (accountIDsEnv) { | ||||||
|  |  | ||||||
|  | @ -1,4 +1,5 @@ | ||||||
| import {expect, test} from '@jest/globals'; | import {expect, test} from '@jest/globals'; | ||||||
|  | 
 | ||||||
| import {getInputs} from '../src/context'; | import {getInputs} from '../src/context'; | ||||||
| 
 | 
 | ||||||
| test('with password and username getInputs does not throw error', async () => { | test('with password and username getInputs does not throw error', async () => { | ||||||
|  |  | ||||||
|  | @ -1,14 +1,16 @@ | ||||||
| import {expect, jest, test} from '@jest/globals'; | import {expect, jest, test} from '@jest/globals'; | ||||||
| import {loginStandard, logout} from '../src/docker'; |  | ||||||
| import * as path from 'path'; | import * as path from 'path'; | ||||||
| import * as exec from '@actions/exec'; | 
 | ||||||
|  | import {loginStandard, logout} from '../src/docker'; | ||||||
|  | 
 | ||||||
|  | import {Docker} from '@docker/actions-toolkit/lib/docker/docker'; | ||||||
| 
 | 
 | ||||||
| process.env['RUNNER_TEMP'] = path.join(__dirname, 'runner'); | process.env['RUNNER_TEMP'] = path.join(__dirname, 'runner'); | ||||||
| 
 | 
 | ||||||
| test('loginStandard calls exec', async () => { | test('loginStandard calls exec', async () => { | ||||||
|   // eslint-disable-next-line @typescript-eslint/ban-ts-comment
 |   // eslint-disable-next-line @typescript-eslint/ban-ts-comment
 | ||||||
|   // @ts-ignore
 |   // @ts-ignore
 | ||||||
|   const execSpy = jest.spyOn(exec, 'getExecOutput').mockImplementation(async () => { |   const execSpy = jest.spyOn(Docker, 'getExecOutput').mockImplementation(async () => { | ||||||
|     return { |     return { | ||||||
|       exitCode: expect.any(Number), |       exitCode: expect.any(Number), | ||||||
|       stdout: expect.any(Function), |       stdout: expect.any(Function), | ||||||
|  | @ -22,7 +24,13 @@ test('loginStandard calls exec', async () => { | ||||||
| 
 | 
 | ||||||
|   await loginStandard(registry, username, password); |   await loginStandard(registry, username, password); | ||||||
| 
 | 
 | ||||||
|   expect(execSpy).toHaveBeenCalledWith(`docker`, ['login', '--password-stdin', '--username', username, registry], { |   expect(execSpy).toHaveBeenCalledTimes(1); | ||||||
|  |   const callfunc = execSpy.mock.calls[0]; | ||||||
|  |   if (callfunc && callfunc[1]) { | ||||||
|  |     // we don't want to check env opt
 | ||||||
|  |     callfunc[1].env = undefined; | ||||||
|  |   } | ||||||
|  |   expect(execSpy).toHaveBeenCalledWith(['login', '--password-stdin', '--username', username, registry], { | ||||||
|     input: Buffer.from(password), |     input: Buffer.from(password), | ||||||
|     silent: true, |     silent: true, | ||||||
|     ignoreReturnCode: true |     ignoreReturnCode: true | ||||||
|  | @ -32,7 +40,7 @@ test('loginStandard calls exec', async () => { | ||||||
| test('logout calls exec', async () => { | test('logout calls exec', async () => { | ||||||
|   // eslint-disable-next-line @typescript-eslint/ban-ts-comment
 |   // eslint-disable-next-line @typescript-eslint/ban-ts-comment
 | ||||||
|   // @ts-ignore
 |   // @ts-ignore
 | ||||||
|   const execSpy = jest.spyOn(exec, 'getExecOutput').mockImplementation(async () => { |   const execSpy = jest.spyOn(Docker, 'getExecOutput').mockImplementation(async () => { | ||||||
|     return { |     return { | ||||||
|       exitCode: expect.any(Number), |       exitCode: expect.any(Number), | ||||||
|       stdout: expect.any(Function), |       stdout: expect.any(Function), | ||||||
|  | @ -44,7 +52,13 @@ test('logout calls exec', async () => { | ||||||
| 
 | 
 | ||||||
|   await logout(registry); |   await logout(registry); | ||||||
| 
 | 
 | ||||||
|   expect(execSpy).toHaveBeenCalledWith(`docker`, ['logout', registry], { |   expect(execSpy).toHaveBeenCalledTimes(1); | ||||||
|  |   const callfunc = execSpy.mock.calls[0]; | ||||||
|  |   if (callfunc && callfunc[1]) { | ||||||
|  |     // we don't want to check env opt
 | ||||||
|  |     callfunc[1].env = undefined; | ||||||
|  |   } | ||||||
|  |   expect(execSpy).toHaveBeenCalledWith(['logout', registry], { | ||||||
|     ignoreReturnCode: true |     ignoreReturnCode: true | ||||||
|   }); |   }); | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  | @ -1,65 +0,0 @@ | ||||||
| import {expect, jest, test} from '@jest/globals'; |  | ||||||
| import osm = require('os'); |  | ||||||
| 
 |  | ||||||
| import {main} from '../src/main'; |  | ||||||
| import * as docker from '../src/docker'; |  | ||||||
| import * as stateHelper from '../src/state-helper'; |  | ||||||
| 
 |  | ||||||
| test('errors without username and password', async () => { |  | ||||||
|   jest.spyOn(osm, 'platform').mockImplementation(() => 'linux'); |  | ||||||
|   process.env['INPUT_LOGOUT'] = 'true'; // default value
 |  | ||||||
|   await expect(main()).rejects.toThrow(new Error('Username and password required')); |  | ||||||
| }); |  | ||||||
| 
 |  | ||||||
| test('successful with username and password', async () => { |  | ||||||
|   jest.spyOn(osm, 'platform').mockImplementation(() => 'linux'); |  | ||||||
|   const setRegistrySpy = jest.spyOn(stateHelper, 'setRegistry'); |  | ||||||
|   const setLogoutSpy = jest.spyOn(stateHelper, 'setLogout'); |  | ||||||
|   const dockerSpy = jest.spyOn(docker, 'login').mockImplementation(() => Promise.resolve()); |  | ||||||
| 
 |  | ||||||
|   const username = 'dbowie'; |  | ||||||
|   process.env[`INPUT_USERNAME`] = username; |  | ||||||
| 
 |  | ||||||
|   const password = 'groundcontrol'; |  | ||||||
|   process.env[`INPUT_PASSWORD`] = password; |  | ||||||
| 
 |  | ||||||
|   const ecr = 'auto'; |  | ||||||
|   process.env['INPUT_ECR'] = ecr; |  | ||||||
| 
 |  | ||||||
|   const logout = false; |  | ||||||
|   process.env['INPUT_LOGOUT'] = String(logout); |  | ||||||
| 
 |  | ||||||
|   await main(); |  | ||||||
| 
 |  | ||||||
|   expect(setRegistrySpy).toHaveBeenCalledWith(''); |  | ||||||
|   expect(setLogoutSpy).toHaveBeenCalledWith(logout); |  | ||||||
|   expect(dockerSpy).toHaveBeenCalledWith('', username, password, ecr); |  | ||||||
| }); |  | ||||||
| 
 |  | ||||||
| test('calls docker login', async () => { |  | ||||||
|   jest.spyOn(osm, 'platform').mockImplementation(() => 'linux'); |  | ||||||
|   const setRegistrySpy = jest.spyOn(stateHelper, 'setRegistry'); |  | ||||||
|   const setLogoutSpy = jest.spyOn(stateHelper, 'setLogout'); |  | ||||||
|   const dockerSpy = jest.spyOn(docker, 'login').mockImplementation(() => Promise.resolve()); |  | ||||||
| 
 |  | ||||||
|   const username = 'dbowie'; |  | ||||||
|   process.env[`INPUT_USERNAME`] = username; |  | ||||||
| 
 |  | ||||||
|   const password = 'groundcontrol'; |  | ||||||
|   process.env[`INPUT_PASSWORD`] = password; |  | ||||||
| 
 |  | ||||||
|   const registry = 'ghcr.io'; |  | ||||||
|   process.env[`INPUT_REGISTRY`] = registry; |  | ||||||
| 
 |  | ||||||
|   const ecr = 'auto'; |  | ||||||
|   process.env['INPUT_ECR'] = ecr; |  | ||||||
| 
 |  | ||||||
|   const logout = true; |  | ||||||
|   process.env['INPUT_LOGOUT'] = String(logout); |  | ||||||
| 
 |  | ||||||
|   await main(); |  | ||||||
| 
 |  | ||||||
|   expect(setRegistrySpy).toHaveBeenCalledWith(registry); |  | ||||||
|   expect(setLogoutSpy).toHaveBeenCalledWith(logout); |  | ||||||
|   expect(dockerSpy).toHaveBeenCalledWith(registry, username, password, ecr); |  | ||||||
| }); |  | ||||||
|  | @ -18,14 +18,16 @@ inputs: | ||||||
|     required: false |     required: false | ||||||
|   ecr: |   ecr: | ||||||
|     description: 'Specifies whether the given registry is ECR (auto, true or false)' |     description: 'Specifies whether the given registry is ECR (auto, true or false)' | ||||||
|     default: 'auto' |  | ||||||
|     required: false |     required: false | ||||||
|   logout: |   logout: | ||||||
|     description: 'Log out from the Docker registry at the end of a job' |     description: 'Log out from the Docker registry at the end of a job' | ||||||
|     default: 'true' |     default: 'true' | ||||||
|     required: false |     required: false | ||||||
|  |   registry-auth: | ||||||
|  |     description: 'Raw authentication to registries, defined as YAML objects' | ||||||
|  |     required: false | ||||||
| 
 | 
 | ||||||
| runs: | runs: | ||||||
|   using: 'node16' |   using: 'node20' | ||||||
|   main: 'dist/index.js' |   main: 'dist/index.js' | ||||||
|   post: 'dist/index.js' |   post: 'dist/index.js' | ||||||
|  |  | ||||||
|  | @ -1,15 +1,20 @@ | ||||||
| # syntax=docker/dockerfile:1 | # syntax=docker/dockerfile:1 | ||||||
| 
 | 
 | ||||||
| ARG NODE_VERSION=16 | ARG NODE_VERSION=20 | ||||||
| ARG DOCKER_VERSION=20.10.13 |  | ||||||
| ARG BUILDX_VERSION=0.8.1 |  | ||||||
| 
 | 
 | ||||||
| FROM node:${NODE_VERSION}-alpine AS base | FROM node:${NODE_VERSION}-alpine AS base | ||||||
| RUN apk add --no-cache cpio findutils git | RUN apk add --no-cache cpio findutils git | ||||||
| WORKDIR /src | WORKDIR /src | ||||||
|  | RUN --mount=type=bind,target=.,rw \ | ||||||
|  |   --mount=type=cache,target=/src/.yarn/cache <<EOT | ||||||
|  |   corepack enable | ||||||
|  |   yarn --version | ||||||
|  |   yarn config set --home enableTelemetry 0 | ||||||
|  | EOT | ||||||
| 
 | 
 | ||||||
| FROM base AS deps | FROM base AS deps | ||||||
| RUN --mount=type=bind,target=.,rw \ | RUN --mount=type=bind,target=.,rw \ | ||||||
|  |   --mount=type=cache,target=/src/.yarn/cache \ | ||||||
|   --mount=type=cache,target=/src/node_modules \ |   --mount=type=cache,target=/src/node_modules \ | ||||||
|   yarn install && mkdir /vendor && cp yarn.lock /vendor |   yarn install && mkdir /vendor && cp yarn.lock /vendor | ||||||
| 
 | 
 | ||||||
|  | @ -18,18 +23,19 @@ COPY --from=deps /vendor / | ||||||
| 
 | 
 | ||||||
| FROM deps AS vendor-validate | FROM deps AS vendor-validate | ||||||
| RUN --mount=type=bind,target=.,rw <<EOT | RUN --mount=type=bind,target=.,rw <<EOT | ||||||
| set -e |   set -e | ||||||
| git add -A |   git add -A | ||||||
| cp -rf /vendor/* . |   cp -rf /vendor/* . | ||||||
| if [ -n "$(git status --porcelain -- yarn.lock)" ]; then |   if [ -n "$(git status --porcelain -- yarn.lock)" ]; then | ||||||
|   echo >&2 'ERROR: Vendor result differs. Please vendor your package with "docker buildx bake vendor-update"' |     echo >&2 'ERROR: Vendor result differs. Please vendor your package with "docker buildx bake vendor"' | ||||||
|   git status --porcelain -- yarn.lock |     git status --porcelain -- yarn.lock | ||||||
|   exit 1 |     exit 1 | ||||||
| fi |   fi | ||||||
| EOT | EOT | ||||||
| 
 | 
 | ||||||
| FROM deps AS build | FROM deps AS build | ||||||
| RUN --mount=type=bind,target=.,rw \ | RUN --mount=type=bind,target=.,rw \ | ||||||
|  |   --mount=type=cache,target=/src/.yarn/cache \ | ||||||
|   --mount=type=cache,target=/src/node_modules \ |   --mount=type=cache,target=/src/node_modules \ | ||||||
|   yarn run build && mkdir /out && cp -Rf dist /out/ |   yarn run build && mkdir /out && cp -Rf dist /out/ | ||||||
| 
 | 
 | ||||||
|  | @ -38,41 +44,39 @@ COPY --from=build /out / | ||||||
| 
 | 
 | ||||||
| FROM build AS build-validate | FROM build AS build-validate | ||||||
| RUN --mount=type=bind,target=.,rw <<EOT | RUN --mount=type=bind,target=.,rw <<EOT | ||||||
| set -e |   set -e | ||||||
| git add -A |   git add -A | ||||||
| cp -rf /out/* . |   cp -rf /out/* . | ||||||
| if [ -n "$(git status --porcelain -- dist)" ]; then |   if [ -n "$(git status --porcelain -- dist)" ]; then | ||||||
|   echo >&2 'ERROR: Build result differs. Please build first with "docker buildx bake build"' |     echo >&2 'ERROR: Build result differs. Please build first with "docker buildx bake build"' | ||||||
|   git status --porcelain -- dist |     git status --porcelain -- dist | ||||||
|   exit 1 |     exit 1 | ||||||
| fi |   fi | ||||||
| EOT | EOT | ||||||
| 
 | 
 | ||||||
| FROM deps AS format | FROM deps AS format | ||||||
| RUN --mount=type=bind,target=.,rw \ | RUN --mount=type=bind,target=.,rw \ | ||||||
|  |   --mount=type=cache,target=/src/.yarn/cache \ | ||||||
|   --mount=type=cache,target=/src/node_modules \ |   --mount=type=cache,target=/src/node_modules \ | ||||||
|   yarn run format \ |   yarn run format \ | ||||||
|   && mkdir /out && find . -name '*.ts' -not -path './node_modules/*' | cpio -pdm /out |   && mkdir /out && find . -name '*.ts' -not -path './node_modules/*' -not -path './.yarn/*' | cpio -pdm /out | ||||||
| 
 | 
 | ||||||
| FROM scratch AS format-update | FROM scratch AS format-update | ||||||
| COPY --from=format /out / | COPY --from=format /out / | ||||||
| 
 | 
 | ||||||
| FROM deps AS lint | FROM deps AS lint | ||||||
| RUN --mount=type=bind,target=.,rw \ | RUN --mount=type=bind,target=.,rw \ | ||||||
|  |   --mount=type=cache,target=/src/.yarn/cache \ | ||||||
|   --mount=type=cache,target=/src/node_modules \ |   --mount=type=cache,target=/src/node_modules \ | ||||||
|   yarn run lint |   yarn run lint | ||||||
| 
 | 
 | ||||||
| FROM docker:${DOCKER_VERSION} as docker |  | ||||||
| FROM docker/buildx-bin:${BUILDX_VERSION} as buildx |  | ||||||
| 
 |  | ||||||
| FROM deps AS test | FROM deps AS test | ||||||
| ENV RUNNER_TEMP=/tmp/github_runner | ENV RUNNER_TEMP=/tmp/github_runner | ||||||
| ENV RUNNER_TOOL_CACHE=/tmp/github_tool_cache | ENV RUNNER_TOOL_CACHE=/tmp/github_tool_cache | ||||||
| RUN --mount=type=bind,target=.,rw \ | RUN --mount=type=bind,target=.,rw \ | ||||||
|  |   --mount=type=cache,target=/src/.yarn/cache \ | ||||||
|   --mount=type=cache,target=/src/node_modules \ |   --mount=type=cache,target=/src/node_modules \ | ||||||
|   --mount=type=bind,from=docker,source=/usr/local/bin/docker,target=/usr/bin/docker \ |   yarn run test --coverage --coverageDirectory=/tmp/coverage | ||||||
|   --mount=type=bind,from=buildx,source=/buildx,target=/usr/libexec/docker/cli-plugins/docker-buildx \ |  | ||||||
|   yarn run test --coverageDirectory=/tmp/coverage |  | ||||||
| 
 | 
 | ||||||
| FROM scratch AS test-coverage | FROM scratch AS test-coverage | ||||||
| COPY --from=test /tmp/coverage / | COPY --from=test /tmp/coverage / | ||||||
|  |  | ||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							|  | @ -1,9 +1,15 @@ | ||||||
|  | target "_common" { | ||||||
|  |   args = { | ||||||
|  |     BUILDKIT_CONTEXT_KEEP_GIT_DIR = 1 | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| group "default" { | group "default" { | ||||||
|   targets = ["build"] |   targets = ["build"] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| group "pre-checkin" { | group "pre-checkin" { | ||||||
|   targets = ["vendor-update", "format", "build"] |   targets = ["vendor", "format", "build"] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| group "validate" { | group "validate" { | ||||||
|  | @ -11,42 +17,49 @@ group "validate" { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| target "build" { | target "build" { | ||||||
|  |   inherits = ["_common"] | ||||||
|   dockerfile = "dev.Dockerfile" |   dockerfile = "dev.Dockerfile" | ||||||
|   target = "build-update" |   target = "build-update" | ||||||
|   output = ["."] |   output = ["."] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| target "build-validate" { | target "build-validate" { | ||||||
|  |   inherits = ["_common"] | ||||||
|   dockerfile = "dev.Dockerfile" |   dockerfile = "dev.Dockerfile" | ||||||
|   target = "build-validate" |   target = "build-validate" | ||||||
|   output = ["type=cacheonly"] |   output = ["type=cacheonly"] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| target "format" { | target "format" { | ||||||
|  |   inherits = ["_common"] | ||||||
|   dockerfile = "dev.Dockerfile" |   dockerfile = "dev.Dockerfile" | ||||||
|   target = "format-update" |   target = "format-update" | ||||||
|   output = ["."] |   output = ["."] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| target "lint" { | target "lint" { | ||||||
|  |   inherits = ["_common"] | ||||||
|   dockerfile = "dev.Dockerfile" |   dockerfile = "dev.Dockerfile" | ||||||
|   target = "lint" |   target = "lint" | ||||||
|   output = ["type=cacheonly"] |   output = ["type=cacheonly"] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| target "vendor-update" { | target "vendor" { | ||||||
|  |   inherits = ["_common"] | ||||||
|   dockerfile = "dev.Dockerfile" |   dockerfile = "dev.Dockerfile" | ||||||
|   target = "vendor-update" |   target = "vendor-update" | ||||||
|   output = ["."] |   output = ["."] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| target "vendor-validate" { | target "vendor-validate" { | ||||||
|  |   inherits = ["_common"] | ||||||
|   dockerfile = "dev.Dockerfile" |   dockerfile = "dev.Dockerfile" | ||||||
|   target = "vendor-validate" |   target = "vendor-validate" | ||||||
|   output = ["type=cacheonly"] |   output = ["type=cacheonly"] | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| target "test" { | target "test" { | ||||||
|  |   inherits = ["_common"] | ||||||
|   dockerfile = "dev.Dockerfile" |   dockerfile = "dev.Dockerfile" | ||||||
|   target = "test-coverage" |   target = "test-coverage" | ||||||
|   output = ["./coverage"] |   output = ["./coverage"] | ||||||
|  |  | ||||||
|  | @ -15,6 +15,7 @@ process.env = Object.assign({}, process.env, { | ||||||
| 
 | 
 | ||||||
| module.exports = { | module.exports = { | ||||||
|   clearMocks: true, |   clearMocks: true, | ||||||
|  |   testEnvironment: 'node', | ||||||
|   moduleFileExtensions: ['js', 'ts'], |   moduleFileExtensions: ['js', 'ts'], | ||||||
|   testMatch: ['**/*.test.ts'], |   testMatch: ['**/*.test.ts'], | ||||||
|   transform: { |   transform: { | ||||||
|  |  | ||||||
							
								
								
									
										66
									
								
								package.json
								
								
								
								
							
							
						
						
									
										66
									
								
								package.json
								
								
								
								
							|  | @ -1,13 +1,16 @@ | ||||||
| { | { | ||||||
|   "name": "docker-login", |   "name": "docker-login", | ||||||
|   "description": "GitHub Action to login against a Docker registry", |   "description": "GitHub Action to login against a Docker registry", | ||||||
|   "main": "lib/main.js", |   "main": "src/main.ts", | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "build": "ncc build src/main.ts --source-map --minify --license licenses.txt", |     "build": "ncc build --source-map --minify --license licenses.txt", | ||||||
|     "lint": "eslint src/**/*.ts __tests__/**/*.ts", |     "lint": "yarn run prettier && yarn run eslint", | ||||||
|     "format": "eslint --fix src/**/*.ts __tests__/**/*.ts", |     "format": "yarn run prettier:fix && yarn run eslint:fix", | ||||||
|     "test": "jest --coverage", |     "eslint": "eslint --max-warnings=0 .", | ||||||
|     "all": "yarn run build && yarn run format && yarn test" |     "eslint:fix": "eslint --fix .", | ||||||
|  |     "prettier": "prettier --check \"./**/*.ts\"", | ||||||
|  |     "prettier:fix": "prettier --write \"./**/*.ts\"", | ||||||
|  |     "test": "jest" | ||||||
|   }, |   }, | ||||||
|   "repository": { |   "repository": { | ||||||
|     "type": "git", |     "type": "git", | ||||||
|  | @ -18,35 +21,32 @@ | ||||||
|     "docker", |     "docker", | ||||||
|     "login" |     "login" | ||||||
|   ], |   ], | ||||||
|   "author": "Docker", |   "author": "Docker Inc.", | ||||||
|   "contributors": [ |   "license": "Apache-2.0", | ||||||
|     { |   "packageManager": "yarn@4.9.2", | ||||||
|       "name": "CrazyMax", |  | ||||||
|       "url": "https://crazymax.dev" |  | ||||||
|     } |  | ||||||
|   ], |  | ||||||
|   "license": "MIT", |  | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@actions/core": "^1.10.0", |     "@actions/core": "^1.11.1", | ||||||
|     "@aws-sdk/client-ecr": "^3.347.1", |     "@aws-sdk/client-ecr": "^3.890.0", | ||||||
|     "@aws-sdk/client-ecr-public": "^3.347.1", |     "@aws-sdk/client-ecr-public": "^3.890.0", | ||||||
|     "@docker/actions-toolkit": "^0.2.0", |     "@docker/actions-toolkit": "^0.63.0", | ||||||
|     "http-proxy-agent": "^7.0.0", |     "http-proxy-agent": "^7.0.2", | ||||||
|     "https-proxy-agent": "^7.0.0" |     "https-proxy-agent": "^7.0.6", | ||||||
|  |     "js-yaml": "^4.1.0" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@types/node": "^16.18.21", |     "@types/js-yaml": "^4.0.9", | ||||||
|     "@typescript-eslint/eslint-plugin": "^5.56.0", |     "@types/node": "^20.19.9", | ||||||
|     "@typescript-eslint/parser": "^5.56.0", |     "@typescript-eslint/eslint-plugin": "^7.18.0", | ||||||
|     "@vercel/ncc": "^0.36.1", |     "@typescript-eslint/parser": "^7.18.0", | ||||||
|     "eslint": "^8.36.0", |     "@vercel/ncc": "^0.38.3", | ||||||
|     "eslint-config-prettier": "^8.8.0", |     "eslint": "^8.57.1", | ||||||
|     "eslint-plugin-jest": "^27.2.1", |     "eslint-config-prettier": "^9.1.2", | ||||||
|     "eslint-plugin-prettier": "^4.2.1", |     "eslint-plugin-jest": "^28.14.0", | ||||||
|     "jest": "^29.5.0", |     "eslint-plugin-prettier": "^5.5.4", | ||||||
|     "prettier": "^2.8.7", |     "jest": "^29.7.0", | ||||||
|     "ts-jest": "^29.0.5", |     "prettier": "^3.6.2", | ||||||
|     "ts-node": "^10.9.1", |     "ts-jest": "^29.4.1", | ||||||
|     "typescript": "^4.9.5" |     "ts-node": "^10.9.2", | ||||||
|  |     "typescript": "^5.9.2" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,18 +1,19 @@ | ||||||
| import * as core from '@actions/core'; | import * as core from '@actions/core'; | ||||||
| import {ECR} from '@aws-sdk/client-ecr'; | import {ECR} from '@aws-sdk/client-ecr'; | ||||||
| import {ECRPUBLIC} from '@aws-sdk/client-ecr-public'; | import {ECRPUBLIC} from '@aws-sdk/client-ecr-public'; | ||||||
| import {NodeHttpHandler} from '@aws-sdk/node-http-handler'; | import {NodeHttpHandler} from '@smithy/node-http-handler'; | ||||||
| import {HttpProxyAgent} from 'http-proxy-agent'; | import {HttpProxyAgent} from 'http-proxy-agent'; | ||||||
| import {HttpsProxyAgent} from 'https-proxy-agent'; | import {HttpsProxyAgent} from 'https-proxy-agent'; | ||||||
| 
 | 
 | ||||||
| const ecrRegistryRegex = /^(([0-9]{12})\.dkr\.ecr\.(.+)\.amazonaws\.com(.cn)?)(\/([^:]+)(:.+)?)?$/; | const ecrRegistryRegex = /^(([0-9]{12})\.(dkr\.ecr|dkr-ecr)\.(.+)\.(on\.aws|amazonaws\.com(.cn)?))(\/([^:]+)(:.+)?)?$/; | ||||||
|  | const ecrPublicRegistryRegex = /public\.ecr\.aws|ecr-public\.aws\.com/; | ||||||
| 
 | 
 | ||||||
| export const isECR = (registry: string): boolean => { | export const isECR = (registry: string): boolean => { | ||||||
|   return ecrRegistryRegex.test(registry) || isPubECR(registry); |   return ecrRegistryRegex.test(registry) || isPubECR(registry); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export const isPubECR = (registry: string): boolean => { | export const isPubECR = (registry: string): boolean => { | ||||||
|   return registry === 'public.ecr.aws'; |   return ecrPublicRegistryRegex.test(registry); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export const getRegion = (registry: string): string => { | export const getRegion = (registry: string): string => { | ||||||
|  | @ -23,7 +24,7 @@ export const getRegion = (registry: string): string => { | ||||||
|   if (!matches) { |   if (!matches) { | ||||||
|     return ''; |     return ''; | ||||||
|   } |   } | ||||||
|   return matches[3]; |   return matches[4]; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export const getAccountIDs = (registry: string): string[] => { | export const getAccountIDs = (registry: string): string[] => { | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ export interface Inputs { | ||||||
|   password: string; |   password: string; | ||||||
|   ecr: string; |   ecr: string; | ||||||
|   logout: boolean; |   logout: boolean; | ||||||
|  |   registryAuth: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export function getInputs(): Inputs { | export function getInputs(): Inputs { | ||||||
|  | @ -14,6 +15,7 @@ export function getInputs(): Inputs { | ||||||
|     username: core.getInput('username'), |     username: core.getInput('username'), | ||||||
|     password: core.getInput('password'), |     password: core.getInput('password'), | ||||||
|     ecr: core.getInput('ecr'), |     ecr: core.getInput('ecr'), | ||||||
|     logout: core.getBooleanInput('logout') |     logout: core.getBooleanInput('logout'), | ||||||
|  |     registryAuth: core.getInput('registry-auth') | ||||||
|   }; |   }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,6 +1,7 @@ | ||||||
| import * as aws from './aws'; | import * as aws from './aws'; | ||||||
| import * as core from '@actions/core'; | import * as core from '@actions/core'; | ||||||
| import {Exec} from '@docker/actions-toolkit/lib/exec'; | 
 | ||||||
|  | import {Docker} from '@docker/actions-toolkit/lib/docker/docker'; | ||||||
| 
 | 
 | ||||||
| export async function login(registry: string, username: string, password: string, ecr: string): Promise<void> { | export async function login(registry: string, username: string, password: string, ecr: string): Promise<void> { | ||||||
|   if (/true/i.test(ecr) || (ecr == 'auto' && aws.isECR(registry))) { |   if (/true/i.test(ecr) || (ecr == 'auto' && aws.isECR(registry))) { | ||||||
|  | @ -11,7 +12,7 @@ export async function login(registry: string, username: string, password: string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export async function logout(registry: string): Promise<void> { | export async function logout(registry: string): Promise<void> { | ||||||
|   await Exec.getExecOutput('docker', ['logout', registry], { |   await Docker.getExecOutput(['logout', registry], { | ||||||
|     ignoreReturnCode: true |     ignoreReturnCode: true | ||||||
|   }).then(res => { |   }).then(res => { | ||||||
|     if (res.stderr.length > 0 && res.exitCode != 0) { |     if (res.stderr.length > 0 && res.exitCode != 0) { | ||||||
|  | @ -21,20 +22,22 @@ export async function logout(registry: string): Promise<void> { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export async function loginStandard(registry: string, username: string, password: string): Promise<void> { | export async function loginStandard(registry: string, username: string, password: string): Promise<void> { | ||||||
|   if (!username || !password) { |   if (!username && !password) { | ||||||
|     throw new Error('Username and password required'); |     throw new Error('Username and password required'); | ||||||
|   } |   } | ||||||
|  |   if (!username) { | ||||||
|  |     throw new Error('Username required'); | ||||||
|  |   } | ||||||
|  |   if (!password) { | ||||||
|  |     throw new Error('Password required'); | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|   const loginArgs: Array<string> = ['login', '--password-stdin']; |   const loginArgs: Array<string> = ['login', '--password-stdin']; | ||||||
|   loginArgs.push('--username', username); |   loginArgs.push('--username', username); | ||||||
|   loginArgs.push(registry); |   loginArgs.push(registry); | ||||||
| 
 | 
 | ||||||
|   if (registry) { |   core.info(`Logging into ${registry}...`); | ||||||
|     core.info(`Logging into ${registry}...`); |   await Docker.getExecOutput(loginArgs, { | ||||||
|   } else { |  | ||||||
|     core.info(`Logging into Docker Hub...`); |  | ||||||
|   } |  | ||||||
|   await Exec.getExecOutput('docker', loginArgs, { |  | ||||||
|     ignoreReturnCode: true, |     ignoreReturnCode: true, | ||||||
|     silent: true, |     silent: true, | ||||||
|     input: Buffer.from(password) |     input: Buffer.from(password) | ||||||
|  | @ -51,7 +54,7 @@ export async function loginECR(registry: string, username: string, password: str | ||||||
|   const regDatas = await aws.getRegistriesData(registry, username, password); |   const regDatas = await aws.getRegistriesData(registry, username, password); | ||||||
|   for (const regData of regDatas) { |   for (const regData of regDatas) { | ||||||
|     core.info(`Logging into ${regData.registry}...`); |     core.info(`Logging into ${regData.registry}...`); | ||||||
|     await Exec.getExecOutput('docker', ['login', '--password-stdin', '--username', regData.username, regData.registry], { |     await Docker.getExecOutput(['login', '--password-stdin', '--username', regData.username, regData.registry], { | ||||||
|       ignoreReturnCode: true, |       ignoreReturnCode: true, | ||||||
|       silent: true, |       silent: true, | ||||||
|       input: Buffer.from(regData.password) |       input: Buffer.from(regData.password) | ||||||
|  |  | ||||||
							
								
								
									
										50
									
								
								src/main.ts
								
								
								
								
							
							
						
						
									
										50
									
								
								src/main.ts
								
								
								
								
							|  | @ -1,21 +1,61 @@ | ||||||
|  | import * as yaml from 'js-yaml'; | ||||||
|  | import * as core from '@actions/core'; | ||||||
| import * as actionsToolkit from '@docker/actions-toolkit'; | import * as actionsToolkit from '@docker/actions-toolkit'; | ||||||
| 
 | 
 | ||||||
| import * as context from './context'; | import * as context from './context'; | ||||||
| import * as docker from './docker'; | import * as docker from './docker'; | ||||||
| import * as stateHelper from './state-helper'; | import * as stateHelper from './state-helper'; | ||||||
| 
 | 
 | ||||||
|  | interface Auth { | ||||||
|  |   registry: string; | ||||||
|  |   username: string; | ||||||
|  |   password: string; | ||||||
|  |   ecr: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export async function main(): Promise<void> { | export async function main(): Promise<void> { | ||||||
|   const input: context.Inputs = context.getInputs(); |   const inputs: context.Inputs = context.getInputs(); | ||||||
|   stateHelper.setRegistry(input.registry); |   stateHelper.setLogout(inputs.logout); | ||||||
|   stateHelper.setLogout(input.logout); | 
 | ||||||
|   await docker.login(input.registry, input.username, input.password, input.ecr); |   if (inputs.registryAuth && (inputs.registry || inputs.username || inputs.password || inputs.ecr)) { | ||||||
|  |     throw new Error('Cannot use registry-auth with other inputs'); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   if (!inputs.registryAuth) { | ||||||
|  |     stateHelper.setRegistries([inputs.registry || 'docker.io']); | ||||||
|  |     await docker.login(inputs.registry || 'docker.io', inputs.username, inputs.password, inputs.ecr || 'auto'); | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const auths = yaml.load(inputs.registryAuth) as Auth[]; | ||||||
|  |   if (auths.length == 0) { | ||||||
|  |     throw new Error('No registry to login'); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const registries: string[] = []; | ||||||
|  |   for (const auth of auths) { | ||||||
|  |     if (!auth.registry) { | ||||||
|  |       registries.push('docker.io'); | ||||||
|  |     } else { | ||||||
|  |       registries.push(auth.registry); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   stateHelper.setRegistries(registries.filter((value, index, self) => self.indexOf(value) === index)); | ||||||
|  | 
 | ||||||
|  |   for (const auth of auths) { | ||||||
|  |     await core.group(`Login to ${auth.registry || 'docker.io'}`, async () => { | ||||||
|  |       await docker.login(auth.registry || 'docker.io', auth.username, auth.password, auth.ecr || 'auto'); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| async function post(): Promise<void> { | async function post(): Promise<void> { | ||||||
|   if (!stateHelper.logout) { |   if (!stateHelper.logout) { | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|   await docker.logout(stateHelper.registry); |   for (const registry of stateHelper.registries.split(',')) { | ||||||
|  |     await docker.logout(registry); | ||||||
|  |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| actionsToolkit.run(main, post); | actionsToolkit.run(main, post); | ||||||
|  |  | ||||||
|  | @ -1,10 +1,10 @@ | ||||||
| import * as core from '@actions/core'; | import * as core from '@actions/core'; | ||||||
| 
 | 
 | ||||||
| export const registry = process.env['STATE_registry'] || ''; | export const registries = process.env['STATE_registries'] || ''; | ||||||
| export const logout = /true/i.test(process.env['STATE_logout'] || ''); | export const logout = /true/i.test(process.env['STATE_logout'] || ''); | ||||||
| 
 | 
 | ||||||
| export function setRegistry(registry: string) { | export function setRegistries(registries: string[]) { | ||||||
|   core.saveState('registry', registry); |   core.saveState('registries', registries.join(',')); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export function setLogout(logout: boolean) { | export function setLogout(logout: boolean) { | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue