mirror of
				https://gitea.com/docker/build-push-action.git
				synced 2025-10-22 11:43:40 +08:00 
			
		
		
		
	Compare commits
	
		
			417 Commits
		
	
	
		
			v3.3.0
			...
			ci-oci-art
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 5ddb01429b | ||
|   | ccc2b40e9e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5dd7b9db36 | ||
|   | 55146d969b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 24e9b7e65f | ||
|   | 263435318d | ||
|   | c0432d2e01 | ||
|   | 0bb1f27d6b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5f9dbf956c | ||
|   | 0788c444d8 | ||
|   | aa179ca4f4 | ||
|   | 1dc7386353 | ||
|   | 9c9803f364 | ||
|   | db1f6c46e8 | ||
|   | 721e8c79de | ||
|   | 14487ce63c | ||
|   | 0ec91264d8 | ||
|   | b749522b90 | ||
|   | c566248492 | ||
|   | 13275dd76e | ||
|   | 67dc78bbaf | ||
|   | 0760504437 | ||
|   | 1c198f4467 | ||
|   | 288d9e2e4a | ||
|   | 88844b95d8 | ||
|   | 1be4244e8d | ||
|   | 094d2bc0cd | ||
|   | 548776e8d0 | ||
|   | 91838c2ba3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1332e65dc3 | ||
|   | 66147ca503 | ||
|   | 8ea72f78e8 | ||
|   | 6481840af9 | ||
|   | 84ad562665 | ||
|   | 9bea05fc44 | ||
|   | 471d1dc4e0 | ||
|   | b89ff0a6f2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1e3ae3a4d3 | ||
|   | b16f42f92a | ||
|   | dc0fea5e62 | ||
|   | 0adf995921 | ||
|   | d88cd289df | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3d09a6bd70 | ||
|   | ca877d9245 | ||
|   | d2fe919bb5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f0fc9ece82 | ||
|   | 67a2d409c0 | ||
|   | 0b1b1c9c43 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b6a7c2c4ee | ||
|   | 31ca4e5d51 | ||
|   | e613db9d5a | ||
|   | b32b51a8ed | ||
|   | 594bf46f0f | ||
|   | fd37bd55af | ||
|   | e6478a2405 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 78785bddff | ||
|   | 128779fed7 | ||
|   | 7e094594be | ||
|   | 32ee877a58 | ||
|   | d1a4129c41 | ||
|   | 49c623eaf8 | ||
|   | bcc9f4afe7 | ||
|   | da5b6c75b9 | ||
|   | 11be14d908 | ||
|   | f5a8591a7f | ||
|   | 8796455d32 | ||
|   | 750f367828 | ||
|   | 48aba3b46d | ||
|   | 678328cf8e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cdf0a37e6f | ||
|   | d719b79de1 | ||
|   | c333dfd43d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6b56a4c3f8 | ||
|   | 92fb0d73b6 | ||
|   | 40532c5d6f | ||
|   | 70dd953427 | ||
|   | 41b4e8020e | ||
|   | 22cd91d73b | ||
|   | 6bdcc4fb8c | ||
|   | b5e932e401 | ||
|   | 325a8bf0f1 | ||
|   | 0259cb088b | ||
|   | 1b8e4efb78 | ||
|   | 75ffdcce88 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 73b7dbf5d4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f79ca49284 | ||
|   | 5e99dacf67 | ||
|   | b249570060 | ||
|   | 4f58ea7922 | ||
|   | 49b5ea61c6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 13c9fddd72 | ||
|   | e44afff359 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 67ebad331f | ||
|   | 32945a3392 | ||
|   | e0fe9cf0f2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8f1ff6bf9a | ||
|   | 5cd11c3a4c | ||
|   | 0aba704831 | ||
|   | 23c657a01f | ||
|   | 16ebe778df | ||
|   | 646a62b4f2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d92ab1347f | ||
|   | 4f7cdeb0f0 | ||
|   | ad3cd774a4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3efbc13366 | ||
|   | 2dbe91db48 | ||
|   | 7de3854c4c | ||
|   | 175aa53a3f | ||
|   | 806a2a461f | ||
|   | a8d35412fb | ||
|   | 1672e74297 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a04f51d3b4 | ||
|   | 5176d81f87 | ||
|   | ec10ae8f96 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 597e8fc414 | ||
|   | e050dfa622 | ||
|   | d1fcdb6ee0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a6067b9a1a | ||
|   | 1ca370b3a9 | ||
|   | 2c95ebed5c | ||
|   | d189d0ef33 | ||
|   | a254f8ca60 | ||
|   | 94dae62c78 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 267a69d6cc | ||
|   | f23fb2a7cb | ||
|   | ef76d100ee | ||
|   | 522345f555 | ||
|   | 1a162644f9 | ||
|   | 9eea548195 | ||
|   | 11c2faaa9e | ||
|   | de2365af33 | ||
|   | bca5082da7 | ||
|   | e7aab408d9 | ||
|   | 63eb7590c6 | ||
|   | 53ec48606f | ||
|   | fe9d9f1d0c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ad37ba1ad0 | ||
|   | aabbd3f4d7 | ||
|   | 9c7c277413 | ||
|   | 987902144c | ||
|   | 709dde20a7 | ||
|   | ec2d640ea6 | ||
|   | 3dc0757c66 | ||
|   | dcb0699155 | ||
|   | a3c06e8698 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e75138d857 | ||
|   | 9552610e81 | ||
|   | 99c83871c1 | ||
|   | af64c4e18f | ||
|   | 6fdff4fb09 | ||
|   | 9e59c61762 | ||
|   | 49351df2b7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 723ff6ffad | ||
|   | 15560696de | ||
|   | 57e1d34ac3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 309982ebc9 | ||
|   | 9476c25b2a | ||
|   | 97be5a4928 | ||
|   | 9cac6c8ea0 | ||
|   | 31159d49c0 | ||
|   | 07e1c3e148 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f7febd621d | ||
|   | f6010ea701 | ||
|   | c0a6b9680f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0dfe9c3d41 | ||
|   | 94f8f8c2ee | ||
|   | 22f4433c58 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6721c56015 | ||
|   | 4367da978b | ||
|   | 0883ebe52d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 76e5c2d6ea | ||
|   | 29d67824d8 | ||
|   | c382f710d3 | ||
|   | 5a5b70d974 | ||
|   | dc24cf9e25 | ||
|   | 667cb22c52 | ||
|   | d880b1964b | ||
|   | e51051ad0b | ||
|   | 86c2bd0031 | ||
|   | 268d2b1611 | ||
|   | 2b8dc7f529 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 840c12be17 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 26368743c0 | ||
|   | cfed4e995e | ||
|   | dca03ca8fd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e030091ff4 | ||
|   | 84354d3b32 | ||
|   | de5408fe94 | ||
|   | cfc1555281 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f6a172d30e | ||
|   | ca052bb54a | ||
|   | 025c2051f3 | ||
|   | 12076d2fb1 | ||
|   | ef6cba3353 | ||
|   | 4c16cf906a | ||
|   | a3118a86c8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c86eb8b0f7 | ||
|   | 2a53c6ccda | ||
|   | ccef1f210d | ||
|   | 79117b6ea5 | ||
|   | df19a799eb | ||
|   | 0e2ab16cd2 | ||
|   | 54d0f58d64 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 563a2f55e4 | ||
|   | 6003d3266a | ||
|   | e1e22cdde8 | ||
|   | fc15b64049 | ||
|   | 6c58ea3670 | ||
|   | 729f7f4926 | ||
|   | 99d83235bc | ||
|   | 1d5307d7af | ||
|   | cf8d130912 | ||
|   | 8804d8e2ac | ||
|   | 1984549052 | ||
|   | 5bc9e2e9b9 | ||
|   | eb539f44b1 | ||
|   | b6ff9e5753 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 929fba6cce | ||
|   | 7f1f43ba33 | ||
|   | 40d6a900e0 | ||
|   | d56be63626 | ||
|   | eb3cfeaf00 | ||
|   | d0fc12d8a4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 68615d5b67 | ||
|   | c3b570184c | ||
|   | 7e6f77677b | ||
|   | 2ce6beaad4 | ||
|   | 4c8d1e6826 | ||
|   | b0312962ef | ||
|   | 96acf63e4c | ||
|   | f8bc7f4600 | ||
|   | c2064be02c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4f02f34098 | ||
|   | 090ca155fc | ||
|   | ec4854f780 | ||
|   | 2cdde995de | ||
|   | 008747aa03 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1580753126 | ||
|   | 2a7db1d68a | ||
|   | 35e7dd5921 | ||
|   | af5a7ed5ba | ||
|   | 2a85189a6c | ||
|   | 6c2079483e | ||
|   | afdf0c0a67 | ||
|   | 00ae31ab6e | ||
|   | 701942b6e5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 90e54d0b1d | ||
|   | 831ca179d3 | ||
|   | 6bd0e5492f | ||
|   | b3eddbb94c | ||
|   | ffd798c1f1 | ||
|   | 62d8db0960 | ||
|   | 8ab81cb898 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d47e7c357d | ||
|   | 4976231911 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d236adc992 | ||
|   | 94d76d3bc1 | ||
|   | 2b28f2a854 | ||
|   | 9f6f8c940b | ||
|   | 8411d080ee | ||
|   | 4a13e500e5 | ||
|   | 7416668686 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b4f76a5dc6 | ||
|   | b7feb766fa | ||
|   | fae8018297 | ||
|   | b625868b13 | ||
|   | 5193ef1da6 | ||
|   | d3afd779e4 | ||
|   | 7a786bb2b9 | ||
|   | c66ae3adcf | ||
|   | 248131c7bf | ||
|   | b425c4cd5a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9834ce5b4d | ||
|   | fdf7f43ecf | ||
|   | e3a4c332fb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c48d200483 | ||
|   | 8d2cf95286 | ||
|   | 3c7915695f | ||
|   | 0a283b683f | ||
|   | c544b50d70 | ||
|   | dd31262fa7 | ||
|   | 5f01267817 | ||
|   | 0f847266c3 | ||
|   | ea8499618b | ||
|   | 4c1b68d83a | ||
|   | 5909c5bffe | ||
|   | 285730d174 | ||
|   | 4bbe0177ef | ||
|   | cc4d1d4d5f | ||
|   | e7d3750abc | ||
|   | 4556201a14 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9fa62cfa91 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8026f009fc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6b35a7a7f1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c6e64b478a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e2505c6383 | ||
|   | 0565240e2d | ||
|   | 3ab07f8801 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b9e7e4daec | ||
|   | 04d1a3b049 | ||
|   | 1a4d1a13fb | ||
|   | 675965c0e1 | ||
|   | 58ee34cb6b | ||
|   | c97c4060bd | ||
|   | 47d5369e0b | ||
|   | 8895c7468f | ||
|   | 59ba712c53 | ||
|   | 0c20fff10d | ||
|   | 0a97817b6a | ||
|   | ec39ef320c | ||
|   | f46044b799 | ||
|   | 4e4ee680f6 | ||
|   | e86cf554b6 | ||
|   | daa0106f78 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ce51e905a6 | ||
|   | 1fde16337d | ||
|   | ae311c520f | ||
|   | 9311bf5263 | ||
|   | b1654941ef | ||
|   | 12a9f89349 | ||
|   | 2036a561be | ||
|   | b1d46f11a2 | ||
|   | e5b2fc7017 | ||
|   | 24216ba114 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | eb33afda71 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9407ba1305 | ||
|   | 429cdb70ad | ||
|   | 74a34eff3a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6787bde0a6 | ||
|   | 56932deb0a | ||
|   | 0681013357 | ||
|   | be4bf1099e | ||
|   | 9ec154c4b6 | ||
|   | 380260b6c7 | ||
|   | ac790be09a | ||
|   | dc0a85b056 | ||
|   | aca01f02d5 | ||
|   | 4b0752a2b1 | ||
|   | be06a9da57 | ||
|   | 19184b90ca | ||
|   | 57e90a56ab | ||
|   | 4fad532b9f | ||
|   | 413aee355f | ||
|   | f05b754b57 | ||
|   | 2f3765570b | ||
|   | 68d0dc20df | ||
|   | 1fd7f72e60 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ea907fb0a4 | ||
|   | 2eb1c1961a | ||
|   | 27376fe2fc | ||
|   | c9330004c2 | ||
|   | dac08d41ad | ||
|   | 44ea916f6c | ||
|   | 0167eef179 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 91bf8bfc4d | ||
|   | a799b4decf | ||
|   | 87480bdf69 | ||
|   | f9efed53cc | ||
|   | 3580b78e04 | ||
|   | 91df6b874e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ea92b18afb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6f91eb31f7 | ||
|   | eafaea8d0f | ||
|   | ddd2a92197 | ||
|   | a54198e85a | ||
|   | 7e3a79c50d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4a730ce64b | ||
|   | 817ed59f97 | ||
|   | a3646c08f8 | ||
|   | 5c3465b033 | ||
|   | e9c0697e5b | ||
|   | e090350180 | ||
|   | f2a1d5e99d | ||
|   | 81f5252b54 | ||
|   | b3435979d1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 80def7c74e | ||
|   | 35434f557a | ||
|   | d63c96254b | ||
|   | 21fe05ff59 | ||
|   | 097a3e23ac | ||
|   | d8823bfaed | ||
|   | 3a3620ed49 | ||
|   | 8082ebc6ea | ||
|   | a3ae207c14 | ||
|   | 4858a31f84 | ||
|   | d10433366f | ||
|   | ee5cabd9e3 | ||
|   | 7944f9a25b | ||
|   | c088e12d82 | ||
|   | e941dc0149 | ||
|   | 0d6f3e8936 | ||
|   | c779d8500d | ||
|   | b651cf69a6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0db6129a57 | ||
|   | 70cc701b9c | ||
|   | af7537dc3e | ||
|   | 2d8166c4b9 | ||
|   | 6a8fbf0dbc | ||
|   | f74fd039f3 | ||
|   | 695ee8547d | ||
|   | fd7264830a | ||
|   | 6842956e83 | ||
|   | 9d1b8fa4e2 | ||
|   | 17c742ea85 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 916f6873ae | ||
|   | a3f7a443f9 | ||
|   | 05e89e68aa | ||
|   | d16cc0b66f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 306a0a9f50 | ||
|   | 55a3485913 | ||
|   | 71e9aaaf29 | ||
|   | 3b5e8027fc | ||
|   | 02d3266a89 | ||
|   | f403dafe18 | ||
|   | 1104d47137 | ||
|   | 838bf90c88 | ||
|   | 337a09d182 | 
| @@ -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.* | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.eslintignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.eslintignore
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | /dist/** | ||||||
|  | /coverage/** | ||||||
|  | /node_modules/** | ||||||
| @@ -1,18 +1,19 @@ | |||||||
| { | { | ||||||
|   "env": { |   "env": { | ||||||
|     "node": true, |     "node": true, | ||||||
|     "es2021": true, |     "es6": true, | ||||||
|     "jest/globals": 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" | ||||||
|   ], |   ], | ||||||
|   "parser": "@typescript-eslint/parser", |   "parser": "@typescript-eslint/parser", | ||||||
|   "parserOptions": { |   "parserOptions": { | ||||||
|     "ecmaVersion": "latest", |     "ecmaVersion": 2023, | ||||||
|     "sourceType": "module" |     "sourceType": "module" | ||||||
|   }, |   }, | ||||||
|   "plugins": [ |   "plugins": [ | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							| @@ -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
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
								
							| @@ -1 +0,0 @@ | |||||||
| *	@crazy-max |  | ||||||
							
								
								
									
										3
									
								
								.github/CODE_OF_CONDUCT.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.github/CODE_OF_CONDUCT.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | # Code of conduct | ||||||
|  |  | ||||||
|  | - [Moby community guidelines](https://github.com/moby/moby/blob/master/CONTRIBUTING.md#moby-community-guidelines) | ||||||
							
								
								
									
										101
									
								
								.github/ISSUE_TEMPLATE/bug.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								.github/ISSUE_TEMPLATE/bug.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -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). | ||||||
|  |         Before submitting a bug report, check out the [Troubleshooting doc](https://github.com/docker/build-push-action/blob/master/TROUBLESHOOTING.md). | ||||||
|  |  | ||||||
|  |   - 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/build-push-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. | ||||||
|  |  | ||||||
|  |   - 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. | ||||||
							
								
								
									
										37
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										37
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,37 +0,0 @@ | |||||||
| --- |  | ||||||
| name: Bug report |  | ||||||
| about: Create a report to help us improve |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| ### Troubleshooting |  | ||||||
|  |  | ||||||
| Before submitting a bug report please read the [Troubleshooting doc](https://github.com/docker/build-push-action/blob/master/TROUBLESHOOTING.md). |  | ||||||
|  |  | ||||||
| ### 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. |  | ||||||
							
								
								
									
										9
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -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/build-push-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. | ||||||
							
								
								
									
										15
									
								
								.github/ISSUE_TEMPLATE/feature.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								.github/ISSUE_TEMPLATE/feature.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -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 | ||||||
							
								
								
									
										12
									
								
								.github/SECURITY.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.github/SECURITY.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -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. | ||||||
							
								
								
									
										31
									
								
								.github/SUPPORT.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										31
									
								
								.github/SUPPORT.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,31 +0,0 @@ | |||||||
| # Support [](https://isitmaintained.com/project/docker/build-push-action) |  | ||||||
|  |  | ||||||
| First, [be a good guy](https://github.com/kossnocorp/etiquette/blob/master/README.md). |  | ||||||
|  |  | ||||||
| ## Reporting an issue |  | ||||||
|  |  | ||||||
| Please do a search in [open issues](https://github.com/docker/build-push-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/build-push-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. |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								.github/build-push-summary.png
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								.github/build-push-summary.png
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 81 KiB | 
							
								
								
									
										1
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,6 +11,7 @@ updates: | |||||||
|     directory: "/" |     directory: "/" | ||||||
|     schedule: |     schedule: | ||||||
|       interval: "daily" |       interval: "daily" | ||||||
|  |     versioning-strategy: "increase" | ||||||
|     allow: |     allow: | ||||||
|       - dependency-type: "production" |       - dependency-type: "production" | ||||||
|     labels: |     labels: | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								.github/e2e/distribution/env
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.github/e2e/distribution/env
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | REGISTRY_FQDN=localhost:8080 | ||||||
|  | REGISTRY_SLUG=localhost:8080/test-docker-action | ||||||
|  |  | ||||||
|  | DISTRIBUTION_HOST=localhost | ||||||
|  | DISTRIBUTION_PORT=8080 | ||||||
							
								
								
									
										13
									
								
								.github/e2e/distribution/install.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										13
									
								
								.github/e2e/distribution/install.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  | set -eu | ||||||
|  |  | ||||||
|  | : "${DISTRIBUTION_VERSION:=2}" | ||||||
|  | : "${DISTRIBUTION_HOST:=localhost}" | ||||||
|  | : "${DISTRIBUTION_PORT:=8080}" | ||||||
|  |  | ||||||
|  | echo "::group::Starting registry:${DISTRIBUTION_VERSION}" | ||||||
|  | ( | ||||||
|  |   set -x | ||||||
|  |   docker run -d --name registry -p "${DISTRIBUTION_PORT}:5000" "registry:${DISTRIBUTION_VERSION}" | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
							
								
								
									
										8
									
								
								.github/e2e/harbor/env
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.github/e2e/harbor/env
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | REGISTRY_FQDN=localhost:8081 | ||||||
|  | REGISTRY_USER=admin | ||||||
|  | REGISTRY_PASSWORD=Harbor12345 | ||||||
|  | REGISTRY_SLUG=localhost:8081/test-docker-action/test-docker-action | ||||||
|  |  | ||||||
|  | HARBOR_HOST=localhost | ||||||
|  | HARBOR_PORT=8081 | ||||||
|  | HARBOR_PROJECT=test-docker-action | ||||||
							
								
								
									
										79
									
								
								.github/e2e/harbor/install.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										79
									
								
								.github/e2e/harbor/install.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,79 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  | set -eu | ||||||
|  |  | ||||||
|  | : "${HARBOR_VERSION:=v2.7.0}" | ||||||
|  | : "${HARBOR_HOST:=localhost}" | ||||||
|  | : "${HARBOR_PORT:=49154}" | ||||||
|  | : "${REGISTRY_USER:=admin}" | ||||||
|  | : "${REGISTRY_PASSWORD:=Harbor12345}" | ||||||
|  |  | ||||||
|  | : "${HARBOR_PROJECT:=test-docker-action}" | ||||||
|  |  | ||||||
|  | project_post_data() { | ||||||
|  |   cat <<EOF | ||||||
|  | { | ||||||
|  |   "project_name": "$HARBOR_PROJECT", | ||||||
|  |   "public": true | ||||||
|  | } | ||||||
|  | EOF | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export TERM=xterm | ||||||
|  |  | ||||||
|  | # download | ||||||
|  | echo "::group::Downloading Harbor $HARBOR_VERSION" | ||||||
|  | ( | ||||||
|  |   cd /tmp | ||||||
|  |   set -x | ||||||
|  |   wget -q "https://github.com/goharbor/harbor/releases/download/${HARBOR_VERSION}/harbor-offline-installer-${HARBOR_VERSION}.tgz" -O harbor-online-installer.tgz | ||||||
|  |   tar xvf harbor-online-installer.tgz | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | # config | ||||||
|  | echo "::group::Configuring Harbor" | ||||||
|  | ( | ||||||
|  |   cd /tmp/harbor | ||||||
|  |   set -x | ||||||
|  |   cp harbor.yml.tmpl harbor.yml | ||||||
|  |   harborConfig="$(harborHost="$HARBOR_HOST" harborPort="$HARBOR_PORT" harborPwd="$REGISTRY_PASSWORD" yq --no-colors '.hostname = env(harborHost) | .http.port = env(harborPort) | .harbor_admin_password = env(harborPwd) | del(.https)' harbor.yml)" | ||||||
|  |   tee harbor.yml <<<"$harborConfig" >/dev/null | ||||||
|  |   yq --no-colors harbor.yml | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | # install and start | ||||||
|  | echo "::group::Installing Harbor" | ||||||
|  | ( | ||||||
|  |   cd /tmp/harbor | ||||||
|  |   set -x | ||||||
|  |   ./install.sh | ||||||
|  |   sleep 10 | ||||||
|  |   netstat -aptn | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | # compose config | ||||||
|  | echo "::group::Compose config" | ||||||
|  | ( | ||||||
|  |   cd /tmp/harbor | ||||||
|  |   set -x | ||||||
|  |   docker compose config | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | # create project | ||||||
|  | echo "::group::Creating project" | ||||||
|  | ( | ||||||
|  |   set -x | ||||||
|  |   curl --fail -v -k --max-time 10 -u "$REGISTRY_USER:$REGISTRY_PASSWORD" -X POST -H "Content-Type: application/json" -d "$(project_post_data)" "http://$HARBOR_HOST:$HARBOR_PORT/api/v2.0/projects" | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | # list projects | ||||||
|  | echo "::group::List projects" | ||||||
|  | ( | ||||||
|  |   set -x | ||||||
|  |   curl --fail -s -k --max-time 10 -u "$REGISTRY_USER:$REGISTRY_PASSWORD" -H "Content-Type: application/json" "http://$HARBOR_HOST:$HARBOR_PORT/api/v2.0/projects" | jq | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
							
								
								
									
										8
									
								
								.github/e2e/nexus/docker-compose.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.github/e2e/nexus/docker-compose.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | services: | ||||||
|  |   nexus: | ||||||
|  |     image: sonatype/nexus3:${NEXUS_VERSION:-latest} | ||||||
|  |     volumes: | ||||||
|  |       - "./data:/nexus-data" | ||||||
|  |     ports: | ||||||
|  |       - "8081:8081" | ||||||
|  |       - "8082:8082" | ||||||
							
								
								
									
										9
									
								
								.github/e2e/nexus/env
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								.github/e2e/nexus/env
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | REGISTRY_FQDN=localhost:8082 | ||||||
|  | REGISTRY_USER=admin | ||||||
|  | REGISTRY_PASSWORD=Nexus12345 | ||||||
|  | REGISTRY_SLUG=localhost:8082/test-docker-action | ||||||
|  |  | ||||||
|  | NEXUS_HOST=localhost | ||||||
|  | NEXUS_PORT=8081 | ||||||
|  | NEXUS_REGISTRY_PORT=8082 | ||||||
|  | NEXUS_REPO=test-docker-action | ||||||
							
								
								
									
										93
									
								
								.github/e2e/nexus/install.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										93
									
								
								.github/e2e/nexus/install.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,93 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  | set -eu | ||||||
|  |  | ||||||
|  | SCRIPT_DIR=$(cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd) | ||||||
|  |  | ||||||
|  | : "${NEXUS_VERSION:=3.47.1}" | ||||||
|  | : "${NEXUS_HOST:=localhost}" | ||||||
|  | : "${NEXUS_PORT:=8081}" | ||||||
|  | : "${NEXUS_REGISTRY_PORT:=8082}" | ||||||
|  | : "${REGISTRY_USER:=admin}" | ||||||
|  | : "${REGISTRY_PASSWORD:=Nexus12345}" | ||||||
|  |  | ||||||
|  | : "${NEXUS_REPO:=test-docker-action}" | ||||||
|  |  | ||||||
|  | createrepo_post_data() { | ||||||
|  |   cat <<EOF | ||||||
|  | { | ||||||
|  |   "name": "${NEXUS_REPO}", | ||||||
|  |   "online": true, | ||||||
|  |   "storage": { | ||||||
|  |     "blobStoreName": "default", | ||||||
|  |     "strictContentTypeValidation": true, | ||||||
|  |     "writePolicy": "ALLOW" | ||||||
|  |   }, | ||||||
|  |   "docker": { | ||||||
|  |     "v1Enabled": false, | ||||||
|  |     "forceBasicAuth": true, | ||||||
|  |     "httpPort": ${NEXUS_REGISTRY_PORT}, | ||||||
|  |     "httpsPort": null, | ||||||
|  |     "subdomain": null | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | EOF | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export NEXUS_VERSION | ||||||
|  |  | ||||||
|  | mkdir -p /tmp/nexus/data | ||||||
|  | chown 200:200 /tmp/nexus/data | ||||||
|  | cp "${SCRIPT_DIR}/docker-compose.yml" /tmp/nexus/docker-compose.yml | ||||||
|  |  | ||||||
|  | echo "::group::Pulling Nexus $NEXUS_VERSION" | ||||||
|  | ( | ||||||
|  |   cd /tmp/nexus | ||||||
|  |   set -x | ||||||
|  |   docker compose pull | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | echo "::group::Compose config" | ||||||
|  | ( | ||||||
|  |   cd /tmp/nexus | ||||||
|  |   set -x | ||||||
|  |   docker compose config | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | echo "::group::Running Nexus" | ||||||
|  | ( | ||||||
|  |   cd /tmp/nexus | ||||||
|  |   set -x | ||||||
|  |   docker compose up -d | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | echo "::group::Running Nexus" | ||||||
|  | ( | ||||||
|  |   cd /tmp/nexus | ||||||
|  |   set -x | ||||||
|  |   docker compose up -d | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | echo "::group::Waiting for Nexus to be ready" | ||||||
|  | until $(curl --output /dev/null --silent --head --fail "http://$NEXUS_HOST:$NEXUS_PORT"); do | ||||||
|  |   printf '.' | ||||||
|  |   sleep 5 | ||||||
|  | done | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | echo "::group::Change user's password" | ||||||
|  | ( | ||||||
|  |   set -x | ||||||
|  |   curl --fail -v -k --max-time 10 -u "$REGISTRY_USER:$(cat /tmp/nexus/data/admin.password)" -X PUT -H 'Content-Type: text/plain' -d "$REGISTRY_PASSWORD" "http://$NEXUS_HOST:$NEXUS_PORT/service/rest/v1/security/users/$REGISTRY_USER/change-password" | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
|  |  | ||||||
|  | echo "::group::Create Docker repository" | ||||||
|  | ( | ||||||
|  |   set -x | ||||||
|  |   curl --fail -v -k --max-time 10 -u "$REGISTRY_USER:$REGISTRY_PASSWORD" -X POST -H 'Content-Type: application/json' -d "$(createrepo_post_data)" "http://$NEXUS_HOST:$NEXUS_PORT/service/rest/v1/repositories/docker/hosted" | ||||||
|  | ) | ||||||
|  | echo "::endgroup::" | ||||||
							
								
								
									
										132
									
								
								.github/workflows/.e2e-run.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								.github/workflows/.e2e-run.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,132 @@ | |||||||
|  | # reusable workflow | ||||||
|  | name: .e2e-run | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   workflow_call: | ||||||
|  |     inputs: | ||||||
|  |       id: | ||||||
|  |         required: false | ||||||
|  |         type: string | ||||||
|  |       type: | ||||||
|  |         required: true | ||||||
|  |         type: string | ||||||
|  |       name: | ||||||
|  |         required: true | ||||||
|  |         type: string | ||||||
|  |       registry: | ||||||
|  |         required: false | ||||||
|  |         type: string | ||||||
|  |       slug: | ||||||
|  |         required: false | ||||||
|  |         type: string | ||||||
|  |       username_secret: | ||||||
|  |         required: false | ||||||
|  |         type: string | ||||||
|  |       password_secret: | ||||||
|  |         required: false | ||||||
|  |         type: string | ||||||
|  |  | ||||||
|  | env: | ||||||
|  |   HARBOR_VERSION: v2.7.0 | ||||||
|  |   NEXUS_VERSION: 3.47.1 | ||||||
|  |   DISTRIBUTION_VERSION: 2.8.1 | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   run: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     strategy: | ||||||
|  |       fail-fast: false | ||||||
|  |       matrix: | ||||||
|  |         include: | ||||||
|  |           - | ||||||
|  |             buildx_version: edge | ||||||
|  |             buildkit_image: moby/buildkit:latest | ||||||
|  |           - | ||||||
|  |             buildx_version: latest | ||||||
|  |             buildkit_image: moby/buildkit:buildx-stable-1 | ||||||
|  |           - | ||||||
|  |             buildx_version: https://github.com/docker/buildx.git#master | ||||||
|  |             buildkit_image: moby/buildkit:master | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Checkout | ||||||
|  |         uses: actions/checkout@v5 | ||||||
|  |       - | ||||||
|  |         name: Set up env | ||||||
|  |         if: inputs.type == 'local' | ||||||
|  |         run: | | ||||||
|  |           cat ./.github/e2e/${{ inputs.id }}/env >> $GITHUB_ENV | ||||||
|  |       - | ||||||
|  |         name: Set up BuildKit config | ||||||
|  |         run: | | ||||||
|  |           touch /tmp/buildkitd.toml | ||||||
|  |           if [ "${{ inputs.type }}" = "local" ]; then | ||||||
|  |             echo -e "[registry.\"${{ env.REGISTRY_FQDN }}\"]\nhttp = true\ninsecure = true" > /tmp/buildkitd.toml | ||||||
|  |           fi | ||||||
|  |       - | ||||||
|  |         name: Set up Docker daemon | ||||||
|  |         if: inputs.type == 'local' | ||||||
|  |         run: | | ||||||
|  |           if [ ! -e /etc/docker/daemon.json ]; then | ||||||
|  |             echo '{}' | sudo tee /etc/docker/daemon.json >/dev/null | ||||||
|  |           fi | ||||||
|  |           DOCKERD_CONFIG=$(jq '.+{"insecure-registries":["http://${{ env.REGISTRY_FQDN }}"]}' /etc/docker/daemon.json) | ||||||
|  |           sudo tee /etc/docker/daemon.json <<<"$DOCKERD_CONFIG" >/dev/null | ||||||
|  |           cat /etc/docker/daemon.json | ||||||
|  |           sudo service docker restart | ||||||
|  |       - | ||||||
|  |         name: Install ${{ inputs.name }} | ||||||
|  |         if: inputs.type == 'local' | ||||||
|  |         run: | | ||||||
|  |           sudo -E bash ./.github/e2e/${{ inputs.id }}/install.sh | ||||||
|  |           sudo chown $(id -u):$(id -g) -R ~/.docker | ||||||
|  |       - | ||||||
|  |         name: Docker meta | ||||||
|  |         id: meta | ||||||
|  |         uses: docker/metadata-action@v5 | ||||||
|  |         with: | ||||||
|  |           images: ${{ env.REGISTRY_SLUG || inputs.slug }} | ||||||
|  |           tags: | | ||||||
|  |             type=raw,gh-runid-${{ github.run_id }} | ||||||
|  |       - | ||||||
|  |         name: Set up QEMU | ||||||
|  |         uses: docker/setup-qemu-action@v3 | ||||||
|  |       - | ||||||
|  |         name: Set up Docker Buildx | ||||||
|  |         uses: docker/setup-buildx-action@v3 | ||||||
|  |         with: | ||||||
|  |           version: ${{ matrix.buildx_version }} | ||||||
|  |           buildkitd-config: /tmp/buildkitd.toml | ||||||
|  |           buildkitd-flags: --debug --allow-insecure-entitlement security.insecure --allow-insecure-entitlement network.host | ||||||
|  |           driver-opts: | | ||||||
|  |             image=${{ matrix.buildkit_image }} | ||||||
|  |             network=host | ||||||
|  |       - | ||||||
|  |         name: Login to Registry | ||||||
|  |         if: github.event_name != 'pull_request' && (env.REGISTRY_USER || inputs.username_secret) != '' | ||||||
|  |         uses: docker/login-action@v3 | ||||||
|  |         with: | ||||||
|  |           registry: ${{ env.REGISTRY_FQDN || inputs.registry }} | ||||||
|  |           username: ${{ env.REGISTRY_USER || secrets[inputs.username_secret] }} | ||||||
|  |           password: ${{ env.REGISTRY_PASSWORD || secrets[inputs.password_secret] }} | ||||||
|  |       - | ||||||
|  |         name: Build and push | ||||||
|  |         uses: ./ | ||||||
|  |         with: | ||||||
|  |           context: ./test | ||||||
|  |           file: ./test/multi.Dockerfile | ||||||
|  |           platforms: linux/386,linux/amd64,linux/arm/v6,linux/arm/v7,linux/arm64,linux/ppc64le,linux/s390x | ||||||
|  |           outputs: type=image,oci-artifact=true,push=${{ github.event_name != 'pull_request' }} | ||||||
|  |           tags: ${{ steps.meta.outputs.tags }} | ||||||
|  |           labels: ${{ steps.meta.outputs.labels }} | ||||||
|  |           cache-from: type=registry,ref=${{ env.REGISTRY_SLUG || inputs.slug }}:master | ||||||
|  |           cache-to: type=inline | ||||||
|  |       - | ||||||
|  |         name: Inspect image | ||||||
|  |         run: | | ||||||
|  |           docker pull ${{ env.REGISTRY_SLUG || inputs.slug }}:${{ steps.meta.outputs.version }} | ||||||
|  |           docker image inspect ${{ env.REGISTRY_SLUG || inputs.slug }}:${{ steps.meta.outputs.version }} | ||||||
|  |       - | ||||||
|  |         name: Check manifest | ||||||
|  |         run: | | ||||||
|  |           docker buildx imagetools inspect ${{ env.REGISTRY_SLUG || inputs.slug }}:${{ steps.meta.outputs.version }} --format '{{json .}}' | ||||||
							
								
								
									
										939
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										939
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										120
									
								
								.github/workflows/e2e.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										120
									
								
								.github/workflows/e2e.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,123 +1,107 @@ | |||||||
| name: e2e | name: e2e | ||||||
|  |  | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.ref }} | ||||||
|  |   cancel-in-progress: true | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|     inputs: |  | ||||||
|       buildx-version: |  | ||||||
|         description: 'Buildx version or Git context' |  | ||||||
|         default: 'latest' |  | ||||||
|         required: false |  | ||||||
|       buildkit-image: |  | ||||||
|         description: 'BuildKit image' |  | ||||||
|         default: 'moby/buildkit:buildx-stable-1' |  | ||||||
|         required: false |  | ||||||
|   schedule: |   schedule: | ||||||
|     - cron: '0 10 * * *' |     - cron: '0 10 * * *' | ||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|       - 'master' |       - 'master' | ||||||
|     tags: |     tags: | ||||||
|       - v* |       - 'v*' | ||||||
|  |  | ||||||
| env: |  | ||||||
|   BUILDX_VERSION: latest |  | ||||||
|   BUILDKIT_IMAGE: moby/buildkit:buildx-stable-1 |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   docker: |   build: | ||||||
|     runs-on: ubuntu-latest |     uses: ./.github/workflows/.e2e-run.yml | ||||||
|     strategy: |     strategy: | ||||||
|       fail-fast: false |       fail-fast: false | ||||||
|       matrix: |       matrix: | ||||||
|         include: |         include: | ||||||
|           - |           - | ||||||
|  |             name: Distribution | ||||||
|  |             id: distribution | ||||||
|  |             type: local | ||||||
|  |           - | ||||||
|  |             name: Docker Hub | ||||||
|             registry: '' |             registry: '' | ||||||
|             slug: ghactionstest/ghactionstest |             slug: ghactionstest/ghactionstest | ||||||
|             username_secret: DOCKERHUB_USERNAME |             username_secret: DOCKERHUB_USERNAME | ||||||
|             password_secret: DOCKERHUB_TOKEN |             password_secret: DOCKERHUB_TOKEN | ||||||
|  |             type: remote | ||||||
|           - |           - | ||||||
|  |             name: GitHub | ||||||
|             registry: ghcr.io |             registry: ghcr.io | ||||||
|             slug: ghcr.io/docker-ghactiontest/test |             slug: ghcr.io/docker-ghactiontest/test | ||||||
|             username_secret: GHCR_USERNAME |             username_secret: GHCR_USERNAME | ||||||
|             password_secret: GHCR_PAT |             password_secret: GHCR_PAT | ||||||
|  |             type: remote | ||||||
|           - |           - | ||||||
|  |             name: GitLab | ||||||
|             registry: registry.gitlab.com |             registry: registry.gitlab.com | ||||||
|             slug: registry.gitlab.com/test1716/test |             slug: registry.gitlab.com/test1716/test | ||||||
|             username_secret: GITLAB_USERNAME |             username_secret: GITLAB_USERNAME | ||||||
|             password_secret: GITLAB_TOKEN |             password_secret: GITLAB_TOKEN | ||||||
|  |             type: remote | ||||||
|           - |           - | ||||||
|  |             name: AWS ECR | ||||||
|             registry: 175142243308.dkr.ecr.us-east-2.amazonaws.com |             registry: 175142243308.dkr.ecr.us-east-2.amazonaws.com | ||||||
|             slug: 175142243308.dkr.ecr.us-east-2.amazonaws.com/sandbox/test-docker-action |             slug: 175142243308.dkr.ecr.us-east-2.amazonaws.com/sandbox/test-docker-action | ||||||
|             username_secret: AWS_ACCESS_KEY_ID |             username_secret: AWS_ACCESS_KEY_ID | ||||||
|             password_secret: AWS_SECRET_ACCESS_KEY |             password_secret: AWS_SECRET_ACCESS_KEY | ||||||
|  |             type: remote | ||||||
|           - |           - | ||||||
|  |             name: AWS ECR Public | ||||||
|             registry: public.ecr.aws |             registry: public.ecr.aws | ||||||
|             slug: public.ecr.aws/q3b5f1u4/test-docker-action |             slug: public.ecr.aws/q3b5f1u4/test-docker-action | ||||||
|             username_secret: AWS_ACCESS_KEY_ID |             username_secret: AWS_ACCESS_KEY_ID | ||||||
|             password_secret: AWS_SECRET_ACCESS_KEY |             password_secret: AWS_SECRET_ACCESS_KEY | ||||||
|  |             type: remote | ||||||
|           - |           - | ||||||
|  |             name: Google Artifact Registry | ||||||
|             registry: us-east4-docker.pkg.dev |             registry: us-east4-docker.pkg.dev | ||||||
|             slug: us-east4-docker.pkg.dev/sandbox-298914/docker-official-github-actions/test-docker-action |             slug: us-east4-docker.pkg.dev/sandbox-298914/docker-official-github-actions/test-docker-action | ||||||
|             username_secret: GAR_USERNAME |             username_secret: GAR_USERNAME | ||||||
|             password_secret: GAR_JSON_KEY |             password_secret: GAR_JSON_KEY | ||||||
|  |             type: remote | ||||||
|           - |           - | ||||||
|             registry: gcr.io |             name: Azure Container Registry | ||||||
|             slug: gcr.io/sandbox-298914/test-docker-action |  | ||||||
|             username_secret: GCR_USERNAME |  | ||||||
|             password_secret: GCR_JSON_KEY |  | ||||||
|           - |  | ||||||
|             registry: officialgithubactions.azurecr.io |             registry: officialgithubactions.azurecr.io | ||||||
|             slug: officialgithubactions.azurecr.io/test-docker-action |             slug: officialgithubactions.azurecr.io/test-docker-action | ||||||
|             username_secret: AZURE_CLIENT_ID |             username_secret: AZURE_CLIENT_ID | ||||||
|             password_secret: AZURE_CLIENT_SECRET |             password_secret: AZURE_CLIENT_SECRET | ||||||
|     steps: |             type: remote | ||||||
|           - |           - | ||||||
|         name: Checkout |             name: Quay | ||||||
|         uses: actions/checkout@v3 |             registry: quay.io | ||||||
|  |             slug: quay.io/docker_build_team/ghactiontest | ||||||
|  |             username_secret: QUAY_USERNAME | ||||||
|  |             password_secret: QUAY_TOKEN | ||||||
|  |             type: remote | ||||||
|           - |           - | ||||||
|         name: Docker meta |             name: Artifactory | ||||||
|         id: meta |             registry: infradock.jfrog.io | ||||||
|         uses: docker/metadata-action@v4 |             slug: infradock.jfrog.io/test-ghaction/build-push-action | ||||||
|         with: |             username_secret: ARTIFACTORY_USERNAME | ||||||
|           images: ${{ matrix.slug }} |             password_secret: ARTIFACTORY_TOKEN | ||||||
|       - |             type: remote | ||||||
|         name: Set up QEMU |           - | ||||||
|         uses: docker/setup-qemu-action@v2 |             name: Harbor | ||||||
|       - |             id: harbor | ||||||
|         name: Set up Docker Buildx |             type: local | ||||||
|         uses: docker/setup-buildx-action@v2 |           - | ||||||
|         with: |             name: Nexus | ||||||
|           version: ${{ inputs.buildx-version || env.BUILDX_VERSION }} |             id: nexus | ||||||
|           driver-opts: | |             type: local | ||||||
|             image=${{ inputs.buildkit-image || env.BUILDKIT_IMAGE }} |  | ||||||
|       - |  | ||||||
|         name: Login to Registry |  | ||||||
|         if: github.event_name != 'pull_request' |  | ||||||
|         uses: docker/login-action@v2 |  | ||||||
|     with: |     with: | ||||||
|  |       id: ${{ matrix.id }} | ||||||
|  |       type: ${{ matrix.type }} | ||||||
|  |       name: ${{ matrix.name }} | ||||||
|       registry: ${{ matrix.registry }} |       registry: ${{ matrix.registry }} | ||||||
|           username: ${{ secrets[matrix.username_secret] }} |       slug: ${{ matrix.slug }} | ||||||
|           password: ${{ secrets[matrix.password_secret] }} |       username_secret: ${{ matrix.username_secret }} | ||||||
|       - |       password_secret: ${{ matrix.password_secret }} | ||||||
|         name: Build and push |     secrets: inherit | ||||||
|         uses: ./ |  | ||||||
|         with: |  | ||||||
|           context: ./test |  | ||||||
|           file: ./test/multi.Dockerfile |  | ||||||
|           platforms: linux/386,linux/amd64,linux/arm/v6,linux/arm/v7,linux/arm64,linux/ppc64le,linux/s390x |  | ||||||
|           push: ${{ github.event_name != 'pull_request' }} |  | ||||||
|           tags: ${{ steps.meta.outputs.tags }} |  | ||||||
|           labels: ${{ steps.meta.outputs.labels }} |  | ||||||
|           cache-from: type=registry,ref=${{ matrix.slug }}:master |  | ||||||
|           cache-to: type=inline |  | ||||||
|       - |  | ||||||
|         name: Inspect image |  | ||||||
|         if: github.event_name != 'pull_request' |  | ||||||
|         run: | |  | ||||||
|           docker pull ${{ matrix.slug }}:${{ steps.meta.outputs.version }} |  | ||||||
|           docker image inspect ${{ matrix.slug }}:${{ steps.meta.outputs.version }} |  | ||||||
|       - |  | ||||||
|         name: Check manifest |  | ||||||
|         if: github.event_name != 'pull_request' |  | ||||||
|         run: | |  | ||||||
|           docker buildx imagetools inspect ${{ matrix.slug }}:${{ steps.meta.outputs.version }} --format '{{json .}}' |  | ||||||
|   | |||||||
							
								
								
									
										74
									
								
								.github/workflows/example.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										74
									
								
								.github/workflows/example.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,74 +0,0 @@ | |||||||
| # This workflow is provided just as an example and not for repo testing/verification |  | ||||||
| name: example |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   schedule: |  | ||||||
|     - cron: '0 10 * * 0' |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - '**' |  | ||||||
|     tags: |  | ||||||
|       - 'v*.*.*' |  | ||||||
|   pull_request: |  | ||||||
|  |  | ||||||
| env: |  | ||||||
|   DOCKER_IMAGE: localhost:5000/name/app |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   docker: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     services: |  | ||||||
|       registry: |  | ||||||
|         image: registry:2 |  | ||||||
|         ports: |  | ||||||
|           - 5000:5000 |  | ||||||
|     steps: |  | ||||||
|       - |  | ||||||
|         name: Checkout |  | ||||||
|         uses: actions/checkout@v3 |  | ||||||
|       - |  | ||||||
|         name: Docker meta |  | ||||||
|         id: meta |  | ||||||
|         uses: docker/metadata-action@v4 |  | ||||||
|         with: |  | ||||||
|           images: ${{ env.DOCKER_IMAGE }} |  | ||||||
|           tags: | |  | ||||||
|             type=schedule |  | ||||||
|             type=ref,event=branch |  | ||||||
|             type=ref,event=pr |  | ||||||
|             type=semver,pattern={{version}} |  | ||||||
|             type=semver,pattern={{major}}.{{minor}} |  | ||||||
|             type=semver,pattern={{major}} |  | ||||||
|             type=sha |  | ||||||
|       - |  | ||||||
|         name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v2 |  | ||||||
|         with: |  | ||||||
|           driver-opts: network=host |  | ||||||
|       - |  | ||||||
|         name: Build and export to Docker client |  | ||||||
|         uses: ./ |  | ||||||
|         with: |  | ||||||
|           context: ./test |  | ||||||
|           file: ./test/Dockerfile |  | ||||||
|           load: true |  | ||||||
|           tags: ${{ steps.meta.outputs.tags }} |  | ||||||
|           labels: ${{ steps.meta.outputs.labels }} |  | ||||||
|       - |  | ||||||
|         name: Build and push to local registry |  | ||||||
|         uses: ./ |  | ||||||
|         with: |  | ||||||
|           context: ./test |  | ||||||
|           file: ./test/Dockerfile |  | ||||||
|           push: ${{ github.event_name != 'pull_request' }} |  | ||||||
|           tags: ${{ steps.meta.outputs.tags }} |  | ||||||
|           labels: ${{ steps.meta.outputs.labels }} |  | ||||||
|       - |  | ||||||
|         name: Inspect image |  | ||||||
|         run: | |  | ||||||
|           docker image inspect ${{ env.DOCKER_IMAGE }}:${{ steps.meta.outputs.version }} |  | ||||||
|       - |  | ||||||
|         name: Check manifest |  | ||||||
|         if: github.event_name != 'pull_request' |  | ||||||
|         run: | |  | ||||||
|           docker buildx imagetools inspect ${{ env.DOCKER_IMAGE }}:${{ steps.meta.outputs.version }} --format '{{json .}}' |  | ||||||
							
								
								
									
										17
									
								
								.github/workflows/pr-assign-author.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								.github/workflows/pr-assign-author.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -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 | ||||||
							
								
								
									
										21
									
								
								.github/workflows/publish.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								.github/workflows/publish.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -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 | ||||||
							
								
								
									
										22
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,12 +1,15 @@ | |||||||
| name: test | name: test | ||||||
|  |  | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.ref }} | ||||||
|  |   cancel-in-progress: true | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|       - 'master' |       - 'master' | ||||||
|  |       - 'releases/v*' | ||||||
|   pull_request: |   pull_request: | ||||||
|     branches: |  | ||||||
|       - 'master' |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   test: |   test: | ||||||
| @@ -14,19 +17,16 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v5 | ||||||
|       - |  | ||||||
|         name: Validate |  | ||||||
|         uses: docker/bake-action@v2 |  | ||||||
|         with: |  | ||||||
|           targets: validate |  | ||||||
|       - |       - | ||||||
|         name: Test |         name: Test | ||||||
|         uses: docker/bake-action@v2 |         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 }} | ||||||
|   | |||||||
							
								
								
									
										43
									
								
								.github/workflows/validate.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								.github/workflows/validate.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | |||||||
|  | name: validate | ||||||
|  |  | ||||||
|  | concurrency: | ||||||
|  |   group: ${{ github.workflow }}-${{ github.ref }} | ||||||
|  |   cancel-in-progress: true | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - 'master' | ||||||
|  |       - 'releases/v*' | ||||||
|  |   pull_request: | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   prepare: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     outputs: | ||||||
|  |       targets: ${{ steps.generate.outputs.targets }} | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Checkout | ||||||
|  |         uses: actions/checkout@v5 | ||||||
|  |       - | ||||||
|  |         name: List targets | ||||||
|  |         id: generate | ||||||
|  |         uses: docker/bake-action/subaction/list-targets@v6 | ||||||
|  |         with: | ||||||
|  |           target: validate | ||||||
|  |  | ||||||
|  |   validate: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: | ||||||
|  |       - prepare | ||||||
|  |     strategy: | ||||||
|  |       fail-fast: false | ||||||
|  |       matrix: | ||||||
|  |         target: ${{ fromJson(needs.prepare.outputs.targets) }} | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         name: Validate | ||||||
|  |         uses: docker/bake-action@v6 | ||||||
|  |         with: | ||||||
|  |           targets: ${{ matrix.target }} | ||||||
							
								
								
									
										72
									
								
								.github/workflows/virtual-env.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										72
									
								
								.github/workflows/virtual-env.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,72 +0,0 @@ | |||||||
| name: virtual-env |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   workflow_dispatch: |  | ||||||
|   schedule: |  | ||||||
|     - cron: '0 10 * * *' |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - 'master' |  | ||||||
|     paths: |  | ||||||
|       - '.github/workflows/virtual-env.yml' |  | ||||||
|   pull_request: |  | ||||||
|     branches: |  | ||||||
|       - 'master' |  | ||||||
|     paths: |  | ||||||
|       - '.github/workflows/virtual-env.yml' |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   os: |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     strategy: |  | ||||||
|       fail-fast: false |  | ||||||
|       matrix: |  | ||||||
|         os: |  | ||||||
|           - ubuntu-latest |  | ||||||
|           - ubuntu-22.04 |  | ||||||
|           - ubuntu-20.04 |  | ||||||
|           - ubuntu-18.04 |  | ||||||
|     steps: |  | ||||||
|       - |  | ||||||
|         name: File system |  | ||||||
|         run: df -ah |  | ||||||
|       - |  | ||||||
|         name: Mounts |  | ||||||
|         run: mount |  | ||||||
|       - |  | ||||||
|         name: Node info |  | ||||||
|         run: node -p process |  | ||||||
|       - |  | ||||||
|         name: NPM version |  | ||||||
|         run: npm version |  | ||||||
|       - |  | ||||||
|         name: List install packages |  | ||||||
|         run: apt list --installed |  | ||||||
|       - |  | ||||||
|         name: Docker daemon conf |  | ||||||
|         run: | |  | ||||||
|           cat /etc/docker/daemon.json |  | ||||||
|       - |  | ||||||
|         name: Docker info |  | ||||||
|         run: docker info |  | ||||||
|       - |  | ||||||
|         name: Docker version |  | ||||||
|         run: docker version |  | ||||||
|       - |  | ||||||
|         name: Cgroups |  | ||||||
|         run: | |  | ||||||
|           sudo apt-get install -y cgroup-tools |  | ||||||
|           lscgroup |  | ||||||
|       - |  | ||||||
|         name: buildx version |  | ||||||
|         run: docker buildx version |  | ||||||
|       - |  | ||||||
|         name: containerd version |  | ||||||
|         run: containerd --version |  | ||||||
|       - |  | ||||||
|         name: Docker images |  | ||||||
|         run: docker image ls |  | ||||||
|       - |  | ||||||
|         name: Dump context |  | ||||||
|         if: always() |  | ||||||
|         uses: crazy-max/ghaction-dump-context@v1 |  | ||||||
							
								
								
									
										70
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										70
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +1,5 @@ | |||||||
| node_modules | # https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore | ||||||
| lib |  | ||||||
|  |  | ||||||
| # Jetbrains |  | ||||||
| /.idea |  | ||||||
| /*.iml |  | ||||||
|  |  | ||||||
| # Rest of the file pulled from https://github.com/github/gitignore/blob/master/Node.gitignore |  | ||||||
| # Logs | # Logs | ||||||
| logs | logs | ||||||
| *.log | *.log | ||||||
| @@ -13,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 | ||||||
| @@ -23,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 | ||||||
|  |  | ||||||
| @@ -60,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/ |  | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.prettierignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.prettierignore
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | # Dependency directories | ||||||
|  | node_modules/ | ||||||
|  | jspm_packages/ | ||||||
|  |  | ||||||
|  | # yarn v2 | ||||||
|  | .yarn/ | ||||||
							
								
								
									
										541
									
								
								.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										541
									
								
								.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										13
									
								
								.yarnrc.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								.yarnrc.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | logFilters: | ||||||
|  |   - code: YN0013 | ||||||
|  |     level: discard | ||||||
|  |   - code: YN0019 | ||||||
|  |     level: discard | ||||||
|  |   - code: YN0076 | ||||||
|  |     level: discard | ||||||
|  |  | ||||||
|  | nodeLinker: node-modules | ||||||
|  |  | ||||||
|  | plugins: | ||||||
|  |   - path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs | ||||||
|  |     spec: "@yarnpkg/plugin-interactive-tools" | ||||||
							
								
								
									
										136
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										136
									
								
								README.md
									
									
									
									
									
								
							| @@ -19,9 +19,11 @@ ___ | |||||||
|   * [Git context](#git-context) |   * [Git context](#git-context) | ||||||
|   * [Path context](#path-context) |   * [Path context](#path-context) | ||||||
| * [Examples](#examples) | * [Examples](#examples) | ||||||
|  | * [Summaries](#summaries) | ||||||
| * [Customizing](#customizing) | * [Customizing](#customizing) | ||||||
|   * [inputs](#inputs) |   * [inputs](#inputs) | ||||||
|   * [outputs](#outputs) |   * [outputs](#outputs) | ||||||
|  |   * [environment variables](#environment-variables) | ||||||
| * [Troubleshooting](#troubleshooting) | * [Troubleshooting](#troubleshooting) | ||||||
| * [Contributing](#contributing) | * [Contributing](#contributing) | ||||||
|  |  | ||||||
| @@ -53,28 +55,26 @@ name: ci | |||||||
|  |  | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: |  | ||||||
|       - 'main' |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   docker: |   docker: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - |  | ||||||
|         name: Set up QEMU |  | ||||||
|         uses: docker/setup-qemu-action@v2 |  | ||||||
|       - |  | ||||||
|         name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v2 |  | ||||||
|       - |       - | ||||||
|         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 }} | ||||||
|  |       - | ||||||
|  |         name: Set up QEMU | ||||||
|  |         uses: docker/setup-qemu-action@v3 | ||||||
|  |       - | ||||||
|  |         name: Set up Docker Buildx | ||||||
|  |         uses: docker/setup-buildx-action@v3 | ||||||
|       - |       - | ||||||
|         name: Build and push |         name: Build and push | ||||||
|         uses: docker/build-push-action@v3 |         uses: docker/build-push-action@v6 | ||||||
|         with: |         with: | ||||||
|           push: true |           push: true | ||||||
|           tags: user/app:latest |           tags: user/app:latest | ||||||
| @@ -92,37 +92,24 @@ expression `{{defaultContext}}`. Here we can use it to provide a subdirectory | |||||||
| to the default Git context: | to the default Git context: | ||||||
|  |  | ||||||
| ```yaml | ```yaml | ||||||
|       - |  | ||||||
|         # Setting up Docker Buildx with docker-container driver is required |  | ||||||
|         # at the moment to be able to use a subdirectory with Git context |  | ||||||
|         name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v2 |  | ||||||
|       - |       - | ||||||
|         name: Build and push |         name: Build and push | ||||||
|         uses: docker/build-push-action@v3 |         uses: docker/build-push-action@v6 | ||||||
|         with: |         with: | ||||||
|           context: "{{defaultContext}}:mysubdir" |           context: "{{defaultContext}}:mysubdir" | ||||||
|           push: true |           push: true | ||||||
|           tags: user/app:latest |           tags: user/app:latest | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| > **Warning** |  | ||||||
| > |  | ||||||
| > Subdirectory for Git context is available from [BuildKit v0.9.0](https://github.com/moby/buildkit/releases/tag/v0.9.0). |  | ||||||
| > If you're using the `docker` builder (default if `setup-buildx-action` not used), |  | ||||||
| > then BuildKit in Docker Engine will be used. As Docker Engine < v22.x.x embeds |  | ||||||
| > Buildkit 0.8.2 at the moment, it does not support this feature. It's therefore |  | ||||||
| > required to use the `setup-buildx-action` at the moment. |  | ||||||
|  |  | ||||||
| Building from the current repository automatically uses the [GitHub Token](https://docs.github.com/en/actions/security-guides/automatic-token-authentication), | Building from the current repository automatically uses the [GitHub Token](https://docs.github.com/en/actions/security-guides/automatic-token-authentication), | ||||||
| so it does not need to be passed. If you want to authenticate against another | so it does not need to be passed. If you want to authenticate against another | ||||||
| private repository, you have to use a [secret](https://docs.docker.com/build/ci/github-actions/examples/#secrets) | private repository, you have to use a [secret](https://docs.docker.com/build/ci/github-actions/secrets) | ||||||
| named `GIT_AUTH_TOKEN` to be able to authenticate against it with Buildx: | named `GIT_AUTH_TOKEN` to be able to authenticate against it with Buildx: | ||||||
|  |  | ||||||
| ```yaml | ```yaml | ||||||
|       - |       - | ||||||
|         name: Build and push |         name: Build and push | ||||||
|         uses: docker/build-push-action@v3 |         uses: docker/build-push-action@v6 | ||||||
|         with: |         with: | ||||||
|           push: true |           push: true | ||||||
|           tags: user/app:latest |           tags: user/app:latest | ||||||
| @@ -137,8 +124,6 @@ name: ci | |||||||
|  |  | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: |  | ||||||
|       - 'main' |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   docker: |   docker: | ||||||
| @@ -146,22 +131,22 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v4 | ||||||
|       - |  | ||||||
|         name: Set up QEMU |  | ||||||
|         uses: docker/setup-qemu-action@v2 |  | ||||||
|       - |  | ||||||
|         name: Set up Docker Buildx |  | ||||||
|         uses: docker/setup-buildx-action@v2 |  | ||||||
|       - |       - | ||||||
|         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 }} | ||||||
|  |       - | ||||||
|  |         name: Set up QEMU | ||||||
|  |         uses: docker/setup-qemu-action@v3 | ||||||
|  |       - | ||||||
|  |         name: Set up Docker Buildx | ||||||
|  |         uses: docker/setup-buildx-action@v3 | ||||||
|       - |       - | ||||||
|         name: Build and push |         name: Build and push | ||||||
|         uses: docker/build-push-action@v3 |         uses: docker/build-push-action@v6 | ||||||
|         with: |         with: | ||||||
|           context: . |           context: . | ||||||
|           push: true |           push: true | ||||||
| @@ -170,13 +155,61 @@ jobs: | |||||||
|  |  | ||||||
| ## Examples | ## Examples | ||||||
|  |  | ||||||
| See https://docs.docker.com/build/ci/github-actions/examples/. | * [Multi-platform image](https://docs.docker.com/build/ci/github-actions/multi-platform/) | ||||||
|  | * [Secrets](https://docs.docker.com/build/ci/github-actions/secrets/) | ||||||
|  | * [Push to multi-registries](https://docs.docker.com/build/ci/github-actions/push-multi-registries/) | ||||||
|  | * [Manage tags and labels](https://docs.docker.com/build/ci/github-actions/manage-tags-labels/) | ||||||
|  | * [Cache management](https://docs.docker.com/build/ci/github-actions/cache/) | ||||||
|  | * [Export to Docker](https://docs.docker.com/build/ci/github-actions/export-docker/) | ||||||
|  | * [Test before push](https://docs.docker.com/build/ci/github-actions/test-before-push/) | ||||||
|  | * [Validating build configuration](https://docs.docker.com/build/ci/github-actions/checks/) | ||||||
|  | * [Local registry](https://docs.docker.com/build/ci/github-actions/local-registry/) | ||||||
|  | * [Share built image between jobs](https://docs.docker.com/build/ci/github-actions/share-image-jobs/) | ||||||
|  | * [Named contexts](https://docs.docker.com/build/ci/github-actions/named-contexts/) | ||||||
|  | * [Copy image between registries](https://docs.docker.com/build/ci/github-actions/copy-image-registries/) | ||||||
|  | * [Update Docker Hub repo description](https://docs.docker.com/build/ci/github-actions/update-dockerhub-desc/) | ||||||
|  | * [SBOM and provenance attestations](https://docs.docker.com/build/ci/github-actions/attestations/) | ||||||
|  | * [Annotations](https://docs.docker.com/build/ci/github-actions/annotations/) | ||||||
|  | * [Reproducible builds](https://docs.docker.com/build/ci/github-actions/reproducible-builds/) | ||||||
|  |  | ||||||
|  | ## Summaries | ||||||
|  |  | ||||||
|  | This action generates a [job summary](https://github.blog/2022-05-09-supercharging-github-actions-with-job-summaries/) | ||||||
|  | that provides a detailed overview of the build execution. The summary shows an | ||||||
|  | overview of all the steps executed during the build, including the build inputs | ||||||
|  | and eventual errors. | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | The summary also includes a link for downloading the build record with | ||||||
|  | additional details about the build, including build stats, logs, outputs, and | ||||||
|  | more. The build record can be imported to Docker Desktop for inspecting the | ||||||
|  | build in greater detail. | ||||||
|  |  | ||||||
|  | > [!WARNING] | ||||||
|  | > | ||||||
|  | > If you're using the [`actions/download-artifact`](https://github.com/actions/download-artifact) | ||||||
|  | > action in your workflow, you need to ignore the build record artifacts | ||||||
|  | > if `name` and `pattern` inputs are not specified ([defaults to download all artifacts](https://github.com/actions/download-artifact?tab=readme-ov-file#download-all-artifacts) of the workflow), | ||||||
|  | > otherwise the action will fail: | ||||||
|  | > ```yaml | ||||||
|  | > - uses: actions/download-artifact@v4 | ||||||
|  | >   with: | ||||||
|  | >     pattern: "!*.dockerbuild" | ||||||
|  | > ``` | ||||||
|  | > More info: https://github.com/actions/toolkit/pull/1874 | ||||||
|  |  | ||||||
|  | Summaries are enabled by default, but can be disabled with the | ||||||
|  | `DOCKER_BUILD_SUMMARY` [environment variable](#environment-variables). | ||||||
|  |  | ||||||
|  | For more information about summaries, refer to the | ||||||
|  | [documentation](https://docs.docker.com/go/build-summary/). | ||||||
|  |  | ||||||
| ## Customizing | ## Customizing | ||||||
|  |  | ||||||
| ### inputs | ### inputs | ||||||
|  |  | ||||||
| Following inputs can be used as `step.with` keys | The following inputs can be used as `step.with` keys: | ||||||
|  |  | ||||||
| > `List` type is a newline-delimited string | > `List` type is a newline-delimited string | ||||||
| > ```yaml | > ```yaml | ||||||
| @@ -194,12 +227,14 @@ Following inputs can be used as `step.with` keys | |||||||
| |--------------------|-------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | |--------------------|-------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | ||||||
| | `add-hosts`        | List/CSV    | List of [customs host-to-IP mapping](https://docs.docker.com/engine/reference/commandline/build/#add-entries-to-container-hosts-file---add-host) (e.g., `docker:10.180.0.1`)      | | | `add-hosts`        | List/CSV    | List of [customs host-to-IP mapping](https://docs.docker.com/engine/reference/commandline/build/#add-entries-to-container-hosts-file---add-host) (e.g., `docker:10.180.0.1`)      | | ||||||
| | `allow`            | List/CSV    | List of [extra privileged entitlement](https://docs.docker.com/engine/reference/commandline/buildx_build/#allow) (e.g., `network.host,security.insecure`)                         | | | `allow`            | List/CSV    | List of [extra privileged entitlement](https://docs.docker.com/engine/reference/commandline/buildx_build/#allow) (e.g., `network.host,security.insecure`)                         | | ||||||
|  | | `annotations`      | List        | List of annotation to set to the image                                                                                                                                            | | ||||||
| | `attests`          | List        | List of [attestation](https://docs.docker.com/build/attestations/) parameters (e.g., `type=sbom,generator=image`)                                                                 |  | | `attests`          | List        | List of [attestation](https://docs.docker.com/build/attestations/) parameters (e.g., `type=sbom,generator=image`)                                                                 |  | ||||||
| | `builder`          | String      | Builder instance (see [setup-buildx](https://github.com/docker/setup-buildx-action) action)                                                                                       | | | `builder`          | String      | Builder instance (see [setup-buildx](https://github.com/docker/setup-buildx-action) action)                                                                                       | | ||||||
| | `build-args`       | List        | List of [build-time variables](https://docs.docker.com/engine/reference/commandline/buildx_build/#build-arg)                                                                      | | | `build-args`       | List        | List of [build-time variables](https://docs.docker.com/engine/reference/commandline/buildx_build/#build-arg)                                                                      | | ||||||
| | `build-contexts`   | List        | List of additional [build contexts](https://docs.docker.com/engine/reference/commandline/buildx_build/#build-context) (e.g., `name=path`)                                         | | | `build-contexts`   | List        | List of additional [build contexts](https://docs.docker.com/engine/reference/commandline/buildx_build/#build-context) (e.g., `name=path`)                                         | | ||||||
| | `cache-from`       | List        | List of [external cache sources](https://docs.docker.com/engine/reference/commandline/buildx_build/#cache-from) (e.g., `type=local,src=path/to/dir`)                              | | | `cache-from`       | List        | List of [external cache sources](https://docs.docker.com/engine/reference/commandline/buildx_build/#cache-from) (e.g., `type=local,src=path/to/dir`)                              | | ||||||
| | `cache-to`         | List        | List of [cache export destinations](https://docs.docker.com/engine/reference/commandline/buildx_build/#cache-to) (e.g., `type=local,dest=path/to/dir`)                            | | | `cache-to`         | List        | List of [cache export destinations](https://docs.docker.com/engine/reference/commandline/buildx_build/#cache-to) (e.g., `type=local,dest=path/to/dir`)                            | | ||||||
|  | | `call`             | String      | Set [method for evaluating build](https://docs.docker.com/reference/cli/docker/buildx/build/#call) (e.g., `check`)                                                                | | ||||||
| | `cgroup-parent`    | String      | Optional [parent cgroup](https://docs.docker.com/engine/reference/commandline/build/#use-a-custom-parent-cgroup---cgroup-parent) for the container used in the build              | | | `cgroup-parent`    | String      | Optional [parent cgroup](https://docs.docker.com/engine/reference/commandline/build/#use-a-custom-parent-cgroup---cgroup-parent) for the container used in the build              | | ||||||
| | `context`          | String      | Build's context is the set of files located in the specified [`PATH` or `URL`](https://docs.docker.com/engine/reference/commandline/build/) (default [Git context](#git-context)) | | | `context`          | String      | Build's context is the set of files located in the specified [`PATH` or `URL`](https://docs.docker.com/engine/reference/commandline/build/) (default [Git context](#git-context)) | | ||||||
| | `file`             | String      | Path to the Dockerfile. (default `{context}/Dockerfile`)                                                                                                                          | | | `file`             | String      | Path to the Dockerfile. (default `{context}/Dockerfile`)                                                                                                                          | | ||||||
| @@ -208,13 +243,14 @@ Following inputs can be used as `step.with` keys | |||||||
| | `network`          | String      | Set the networking mode for the `RUN` instructions during build                                                                                                                   | | | `network`          | String      | Set the networking mode for the `RUN` instructions during build                                                                                                                   | | ||||||
| | `no-cache`         | Bool        | Do not use cache when building the image (default `false`)                                                                                                                        | | | `no-cache`         | Bool        | Do not use cache when building the image (default `false`)                                                                                                                        | | ||||||
| | `no-cache-filters` | List/CSV    | Do not cache specified stages                                                                                                                                                     | | | `no-cache-filters` | List/CSV    | Do not cache specified stages                                                                                                                                                     | | ||||||
| | `outputs`¹         | List        | List of [output destinations](https://docs.docker.com/engine/reference/commandline/buildx_build/#output) (format: `type=local,dest=path`)                                         | | | `outputs`          | List        | List of [output destinations](https://docs.docker.com/engine/reference/commandline/buildx_build/#output) (format: `type=local,dest=path`)                                         | | ||||||
| | `platforms`        | List/CSV    | List of [target platforms](https://docs.docker.com/engine/reference/commandline/buildx_build/#platform) for build                                                                 | | | `platforms`        | List/CSV    | List of [target platforms](https://docs.docker.com/engine/reference/commandline/buildx_build/#platform) for build                                                                 | | ||||||
| | `provenance`       | Bool/String | Generate [provenance](https://docs.docker.com/build/attestations/slsa-provenance/) attestation for the build (shorthand for `--attest=type=provenance`)                           | | | `provenance`       | Bool/String | Generate [provenance](https://docs.docker.com/build/attestations/slsa-provenance/) attestation for the build (shorthand for `--attest=type=provenance`)                           | | ||||||
| | `pull`             | Bool        | Always attempt to pull all referenced images (default `false`)                                                                                                                    | | | `pull`             | Bool        | Always attempt to pull all referenced images (default `false`)                                                                                                                    | | ||||||
| | `push`             | Bool        | [Push](https://docs.docker.com/engine/reference/commandline/buildx_build/#push) is a shorthand for `--output=type=registry` (default `false`)                                     | | | `push`             | Bool        | [Push](https://docs.docker.com/engine/reference/commandline/buildx_build/#push) is a shorthand for `--output=type=registry` (default `false`)                                     | | ||||||
| | `sbom`             | Bool/String | Generate [SBOM](https://docs.docker.com/build/attestations/sbom/) attestation for the build (shorthand for `--attest=type=sbom`)                                                  | | | `sbom`             | Bool/String | Generate [SBOM](https://docs.docker.com/build/attestations/sbom/) attestation for the build (shorthand for `--attest=type=sbom`)                                                  | | ||||||
| | `secrets`          | List        | List of [secrets](https://docs.docker.com/engine/reference/commandline/buildx_build/#secret) to expose to the build (e.g., `key=string`, `GIT_AUTH_TOKEN=mytoken`)                | | | `secrets`          | List        | List of [secrets](https://docs.docker.com/engine/reference/commandline/buildx_build/#secret) to expose to the build (e.g., `key=string`, `GIT_AUTH_TOKEN=mytoken`)                | | ||||||
|  | | `secret-envs`      | List/CSV    | List of [secret env vars](https://docs.docker.com/engine/reference/commandline/buildx_build/#secret) to expose to the build (e.g., `key=envname`, `MY_SECRET=MY_ENV_VAR`)         | | ||||||
| | `secret-files`     | List        | List of [secret files](https://docs.docker.com/engine/reference/commandline/buildx_build/#secret) to expose to the build (e.g., `key=filename`, `MY_SECRET=./secret.txt`)         | | | `secret-files`     | List        | List of [secret files](https://docs.docker.com/engine/reference/commandline/buildx_build/#secret) to expose to the build (e.g., `key=filename`, `MY_SECRET=./secret.txt`)         | | ||||||
| | `shm-size`         | String      | Size of [`/dev/shm`](https://docs.docker.com/engine/reference/commandline/buildx_build/#shm-size) (e.g., `2g`)                                                                    | | | `shm-size`         | String      | Size of [`/dev/shm`](https://docs.docker.com/engine/reference/commandline/buildx_build/#shm-size) (e.g., `2g`)                                                                    | | ||||||
| | `ssh`              | List        | List of [SSH agent socket or keys](https://docs.docker.com/engine/reference/commandline/buildx_build/#ssh) to expose to the build                                                 | | | `ssh`              | List        | List of [SSH agent socket or keys](https://docs.docker.com/engine/reference/commandline/buildx_build/#ssh) to expose to the build                                                 | | ||||||
| @@ -223,13 +259,9 @@ Following inputs can be used as `step.with` keys | |||||||
| | `ulimit`           | List        | [Ulimit](https://docs.docker.com/engine/reference/commandline/buildx_build/#ulimit) options (e.g., `nofile=1024:1024`)                                                            | | | `ulimit`           | List        | [Ulimit](https://docs.docker.com/engine/reference/commandline/buildx_build/#ulimit) options (e.g., `nofile=1024:1024`)                                                            | | ||||||
| | `github-token`     | String      | GitHub Token used to authenticate against a repository for [Git context](#git-context) (default `${{ github.token }}`)                                                            | | | `github-token`     | String      | GitHub Token used to authenticate against a repository for [Git context](#git-context) (default `${{ github.token }}`)                                                            | | ||||||
|  |  | ||||||
| > **Note** |  | ||||||
| > |  | ||||||
| > * ¹ multiple `outputs` are [not yet supported](https://github.com/moby/buildkit/issues/1555) |  | ||||||
|  |  | ||||||
| ### outputs | ### outputs | ||||||
|  |  | ||||||
| Following outputs are available | The following outputs are available: | ||||||
|  |  | ||||||
| | Name       | Type    | Description           | | | Name       | Type    | Description           | | ||||||
| |------------|---------|-----------------------| | |------------|---------|-----------------------| | ||||||
| @@ -237,6 +269,16 @@ Following outputs are available | |||||||
| | `digest`   | String  | Image digest          | | | `digest`   | String  | Image digest          | | ||||||
| | `metadata` | JSON    | Build result metadata | | | `metadata` | JSON    | Build result metadata | | ||||||
|  |  | ||||||
|  | ### environment variables | ||||||
|  |  | ||||||
|  | | Name                                 | Type   | Default | Description                                                                                                                                                                                                                                                        | | ||||||
|  | |--------------------------------------|--------|---------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | ||||||
|  | | `DOCKER_BUILD_CHECKS_ANNOTATIONS`    | Bool   | `true`  | If `false`, GitHub annotations are not generated for [build checks](https://docs.docker.com/build/checks/)                                                                                                                                                         | | ||||||
|  | | `DOCKER_BUILD_SUMMARY`               | Bool   | `true`  | If `false`, [build summary](https://docs.docker.com/build/ci/github-actions/build-summary/) generation is disabled                                                                                                                                                 | | ||||||
|  | | `DOCKER_BUILD_RECORD_UPLOAD`         | Bool   | `true`  | If `false`, build record upload as [GitHub artifact](https://docs.github.com/en/actions/using-workflows/storing-workflow-data-as-artifacts) is disabled                                                                                                            | | ||||||
|  | | `DOCKER_BUILD_RECORD_RETENTION_DAYS` | Number |         | Duration after which build record artifact will expire in days. Defaults to repository/org [retention settings](https://docs.github.com/en/actions/learn-github-actions/usage-limits-billing-and-administration#artifact-and-log-retention-policy) if unset or `0` | | ||||||
|  | | `DOCKER_BUILD_EXPORT_LEGACY`         | Bool   | `false` | If `true`, exports build using legacy export-build tool instead of [`buildx history export` command](https://docs.docker.com/reference/cli/docker/buildx/history/export/)                                                                                          | | ||||||
|  |  | ||||||
| ## Troubleshooting | ## Troubleshooting | ||||||
|  |  | ||||||
| See [TROUBLESHOOTING.md](TROUBLESHOOTING.md) | See [TROUBLESHOOTING.md](TROUBLESHOOTING.md) | ||||||
|   | |||||||
| @@ -44,13 +44,13 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - |       - | ||||||
|         name: Checkout |         name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v4 | ||||||
|       - |       - | ||||||
|         name: Set up QEMU |         name: Set up QEMU | ||||||
|         uses: docker/setup-qemu-action@v2 |         uses: docker/setup-qemu-action@v3 | ||||||
|       - |       - | ||||||
|         name: Set up Docker Buildx |         name: Set up Docker Buildx | ||||||
|         uses: docker/setup-buildx-action@v2 |         uses: docker/setup-buildx-action@v3 | ||||||
|         with: |         with: | ||||||
|           buildkitd-flags: --debug |           buildkitd-flags: --debug | ||||||
|       - |       - | ||||||
| @@ -58,7 +58,7 @@ jobs: | |||||||
|         uses: crazy-max/ghaction-setup-containerd@v2 |         uses: crazy-max/ghaction-setup-containerd@v2 | ||||||
|       - |       - | ||||||
|         name: Build Docker image |         name: Build Docker image | ||||||
|         uses: docker/build-push-action@v3 |         uses: docker/build-push-action@v6 | ||||||
|         with: |         with: | ||||||
|           context: . |           context: . | ||||||
|           platforms: linux/amd64,linux/arm64 |           platforms: linux/amd64,linux/arm64 | ||||||
| @@ -111,7 +111,7 @@ to generate sanitized tags: | |||||||
|     tags: latest |     tags: latest | ||||||
|  |  | ||||||
| - name: Build and push | - name: Build and push | ||||||
|   uses: docker/build-push-action@v3 |   uses: docker/build-push-action@v6 | ||||||
|   with: |   with: | ||||||
|     context: . |     context: . | ||||||
|     push: true |     push: true | ||||||
| @@ -129,7 +129,7 @@ Or a dedicated step to sanitize the slug: | |||||||
|     script: return 'ghcr.io/${{ github.repository }}'.toLowerCase() |     script: return 'ghcr.io/${{ github.repository }}'.toLowerCase() | ||||||
|  |  | ||||||
| - name: Build and push | - name: Build and push | ||||||
|   uses: docker/build-push-action@v3 |   uses: docker/build-push-action@v6 | ||||||
|   with: |   with: | ||||||
|     context: . |     context: . | ||||||
|     push: true |     push: true | ||||||
|   | |||||||
							
								
								
									
										207
									
								
								__mocks__/@actions/github.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										207
									
								
								__mocks__/@actions/github.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,207 @@ | |||||||
|  | import {jest} from '@jest/globals'; | ||||||
|  |  | ||||||
|  | export const context = { | ||||||
|  |   repo: { | ||||||
|  |     owner: 'docker', | ||||||
|  |     repo: 'build-push-action' | ||||||
|  |   }, | ||||||
|  |   ref: 'refs/heads/master', | ||||||
|  |   runId: 123456789, | ||||||
|  |   payload: { | ||||||
|  |     after: '860c1904a1ce19322e91ac35af1ab07466440c37', | ||||||
|  |     base_ref: null, | ||||||
|  |     before: '5f3331d7f7044c18ca9f12c77d961c4d7cf3276a', | ||||||
|  |     commits: [ | ||||||
|  |       { | ||||||
|  |         author: { | ||||||
|  |           email: 'crazy-max@users.noreply.github.com', | ||||||
|  |           name: 'CrazyMax', | ||||||
|  |           username: 'crazy-max' | ||||||
|  |         }, | ||||||
|  |         committer: { | ||||||
|  |           email: 'crazy-max@users.noreply.github.com', | ||||||
|  |           name: 'CrazyMax', | ||||||
|  |           username: 'crazy-max' | ||||||
|  |         }, | ||||||
|  |         distinct: true, | ||||||
|  |         id: '860c1904a1ce19322e91ac35af1ab07466440c37', | ||||||
|  |         message: 'hello dev', | ||||||
|  |         timestamp: '2022-04-19T11:27:24+02:00', | ||||||
|  |         tree_id: 'd2c60af597e863787d2d27f569e30495b0b92820', | ||||||
|  |         url: 'https://github.com/docker/test-docker-action/commit/860c1904a1ce19322e91ac35af1ab07466440c37' | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     compare: 'https://github.com/docker/test-docker-action/compare/5f3331d7f704...860c1904a1ce', | ||||||
|  |     created: false, | ||||||
|  |     deleted: false, | ||||||
|  |     forced: false, | ||||||
|  |     head_commit: { | ||||||
|  |       author: { | ||||||
|  |         email: 'crazy-max@users.noreply.github.com', | ||||||
|  |         name: 'CrazyMax', | ||||||
|  |         username: 'crazy-max' | ||||||
|  |       }, | ||||||
|  |       committer: { | ||||||
|  |         email: 'crazy-max@users.noreply.github.com', | ||||||
|  |         name: 'CrazyMax', | ||||||
|  |         username: 'crazy-max' | ||||||
|  |       }, | ||||||
|  |       distinct: true, | ||||||
|  |       id: '860c1904a1ce19322e91ac35af1ab07466440c37', | ||||||
|  |       message: 'hello dev', | ||||||
|  |       timestamp: '2022-04-19T11:27:24+02:00', | ||||||
|  |       tree_id: 'd2c60af597e863787d2d27f569e30495b0b92820', | ||||||
|  |       url: 'https://github.com/docker/test-docker-action/commit/860c1904a1ce19322e91ac35af1ab07466440c37' | ||||||
|  |     }, | ||||||
|  |     organization: { | ||||||
|  |       avatar_url: 'https://avatars.githubusercontent.com/u/5429470?v=4', | ||||||
|  |       description: 'Docker helps developers bring their ideas to life by conquering the complexity of app development.', | ||||||
|  |       events_url: 'https://api.github.com/orgs/docker/events', | ||||||
|  |       hooks_url: 'https://api.github.com/orgs/docker/hooks', | ||||||
|  |       id: 5429470, | ||||||
|  |       issues_url: 'https://api.github.com/orgs/docker/issues', | ||||||
|  |       login: 'docker', | ||||||
|  |       members_url: 'https://api.github.com/orgs/docker/members{/member}', | ||||||
|  |       node_id: 'MDEyOk9yZ2FuaXphdGlvbjU0Mjk0NzA=', | ||||||
|  |       public_members_url: 'https://api.github.com/orgs/docker/public_members{/member}', | ||||||
|  |       repos_url: 'https://api.github.com/orgs/docker/repos', | ||||||
|  |       url: 'https://api.github.com/orgs/docker' | ||||||
|  |     }, | ||||||
|  |     pusher: { | ||||||
|  |       email: 'github@crazymax.dev', | ||||||
|  |       name: 'crazy-max' | ||||||
|  |     }, | ||||||
|  |     ref: 'refs/heads/dev', | ||||||
|  |     repository: { | ||||||
|  |       allow_forking: true, | ||||||
|  |       archive_url: 'https://api.github.com/repos/docker/test-docker-action/{archive_format}{/ref}', | ||||||
|  |       archived: false, | ||||||
|  |       assignees_url: 'https://api.github.com/repos/docker/test-docker-action/assignees{/user}', | ||||||
|  |       blobs_url: 'https://api.github.com/repos/docker/test-docker-action/git/blobs{/sha}', | ||||||
|  |       branches_url: 'https://api.github.com/repos/docker/test-docker-action/branches{/branch}', | ||||||
|  |       clone_url: 'https://github.com/docker/test-docker-action.git', | ||||||
|  |       collaborators_url: 'https://api.github.com/repos/docker/test-docker-action/collaborators{/collaborator}', | ||||||
|  |       comments_url: 'https://api.github.com/repos/docker/test-docker-action/comments{/number}', | ||||||
|  |       commits_url: 'https://api.github.com/repos/docker/test-docker-action/commits{/sha}', | ||||||
|  |       compare_url: 'https://api.github.com/repos/docker/test-docker-action/compare/{base}...{head}', | ||||||
|  |       contents_url: 'https://api.github.com/repos/docker/test-docker-action/contents/{+path}', | ||||||
|  |       contributors_url: 'https://api.github.com/repos/docker/test-docker-action/contributors', | ||||||
|  |       created_at: 1596792180, | ||||||
|  |       default_branch: 'master', | ||||||
|  |       deployments_url: 'https://api.github.com/repos/docker/test-docker-action/deployments', | ||||||
|  |       description: 'Test "Docker" Actions', | ||||||
|  |       disabled: false, | ||||||
|  |       downloads_url: 'https://api.github.com/repos/docker/test-docker-action/downloads', | ||||||
|  |       events_url: 'https://api.github.com/repos/docker/test-docker-action/events', | ||||||
|  |       fork: false, | ||||||
|  |       forks: 1, | ||||||
|  |       forks_count: 1, | ||||||
|  |       forks_url: 'https://api.github.com/repos/docker/test-docker-action/forks', | ||||||
|  |       full_name: 'docker/test-docker-action', | ||||||
|  |       git_commits_url: 'https://api.github.com/repos/docker/test-docker-action/git/commits{/sha}', | ||||||
|  |       git_refs_url: 'https://api.github.com/repos/docker/test-docker-action/git/refs{/sha}', | ||||||
|  |       git_tags_url: 'https://api.github.com/repos/docker/test-docker-action/git/tags{/sha}', | ||||||
|  |       git_url: 'git://github.com/docker/test-docker-action.git', | ||||||
|  |       has_downloads: true, | ||||||
|  |       has_issues: true, | ||||||
|  |       has_pages: false, | ||||||
|  |       has_projects: true, | ||||||
|  |       has_wiki: true, | ||||||
|  |       homepage: '', | ||||||
|  |       hooks_url: 'https://api.github.com/repos/docker/test-docker-action/hooks', | ||||||
|  |       html_url: 'https://github.com/docker/test-docker-action', | ||||||
|  |       id: 285789493, | ||||||
|  |       is_template: false, | ||||||
|  |       issue_comment_url: 'https://api.github.com/repos/docker/test-docker-action/issues/comments{/number}', | ||||||
|  |       issue_events_url: 'https://api.github.com/repos/docker/test-docker-action/issues/events{/number}', | ||||||
|  |       issues_url: 'https://api.github.com/repos/docker/test-docker-action/issues{/number}', | ||||||
|  |       keys_url: 'https://api.github.com/repos/docker/test-docker-action/keys{/key_id}', | ||||||
|  |       labels_url: 'https://api.github.com/repos/docker/test-docker-action/labels{/name}', | ||||||
|  |       language: 'JavaScript', | ||||||
|  |       languages_url: 'https://api.github.com/repos/docker/test-docker-action/languages', | ||||||
|  |       license: { | ||||||
|  |         key: 'mit', | ||||||
|  |         name: 'MIT License', | ||||||
|  |         node_id: 'MDc6TGljZW5zZTEz', | ||||||
|  |         spdx_id: 'MIT', | ||||||
|  |         url: 'https://api.github.com/licenses/mit' | ||||||
|  |       }, | ||||||
|  |       master_branch: 'master', | ||||||
|  |       merges_url: 'https://api.github.com/repos/docker/test-docker-action/merges', | ||||||
|  |       milestones_url: 'https://api.github.com/repos/docker/test-docker-action/milestones{/number}', | ||||||
|  |       mirror_url: null, | ||||||
|  |       name: 'test-docker-action', | ||||||
|  |       node_id: 'MDEwOlJlcG9zaXRvcnkyODU3ODk0OTM=', | ||||||
|  |       notifications_url: 'https://api.github.com/repos/docker/test-docker-action/notifications{?since,all,participating}', | ||||||
|  |       open_issues: 6, | ||||||
|  |       open_issues_count: 6, | ||||||
|  |       organization: 'docker', | ||||||
|  |       owner: { | ||||||
|  |         avatar_url: 'https://avatars.githubusercontent.com/u/5429470?v=4', | ||||||
|  |         email: 'info@docker.com', | ||||||
|  |         events_url: 'https://api.github.com/users/docker/events{/privacy}', | ||||||
|  |         followers_url: 'https://api.github.com/users/docker/followers', | ||||||
|  |         following_url: 'https://api.github.com/users/docker/following{/other_user}', | ||||||
|  |         gists_url: 'https://api.github.com/users/docker/gists{/gist_id}', | ||||||
|  |         gravatar_id: '', | ||||||
|  |         html_url: 'https://github.com/docker', | ||||||
|  |         id: 5429470, | ||||||
|  |         login: 'docker', | ||||||
|  |         name: 'docker', | ||||||
|  |         node_id: 'MDEyOk9yZ2FuaXphdGlvbjU0Mjk0NzA=', | ||||||
|  |         organizations_url: 'https://api.github.com/users/docker/orgs', | ||||||
|  |         received_events_url: 'https://api.github.com/users/docker/received_events', | ||||||
|  |         repos_url: 'https://api.github.com/users/docker/repos', | ||||||
|  |         site_admin: false, | ||||||
|  |         starred_url: 'https://api.github.com/users/docker/starred{/owner}{/repo}', | ||||||
|  |         subscriptions_url: 'https://api.github.com/users/docker/subscriptions', | ||||||
|  |         type: 'Organization', | ||||||
|  |         url: 'https://api.github.com/users/docker' | ||||||
|  |       }, | ||||||
|  |       private: true, | ||||||
|  |       pulls_url: 'https://api.github.com/repos/docker/test-docker-action/pulls{/number}', | ||||||
|  |       pushed_at: 1650360446, | ||||||
|  |       releases_url: 'https://api.github.com/repos/docker/test-docker-action/releases{/id}', | ||||||
|  |       size: 796, | ||||||
|  |       ssh_url: 'git@github.com:docker/test-docker-action.git', | ||||||
|  |       stargazers: 0, | ||||||
|  |       stargazers_count: 0, | ||||||
|  |       stargazers_url: 'https://api.github.com/repos/docker/test-docker-action/stargazers', | ||||||
|  |       statuses_url: 'https://api.github.com/repos/docker/test-docker-action/statuses/{sha}', | ||||||
|  |       subscribers_url: 'https://api.github.com/repos/docker/test-docker-action/subscribers', | ||||||
|  |       subscription_url: 'https://api.github.com/repos/docker/test-docker-action/subscription', | ||||||
|  |       svn_url: 'https://github.com/docker/test-docker-action', | ||||||
|  |       tags_url: 'https://api.github.com/repos/docker/test-docker-action/tags', | ||||||
|  |       teams_url: 'https://api.github.com/repos/docker/test-docker-action/teams', | ||||||
|  |       topics: [], | ||||||
|  |       trees_url: 'https://api.github.com/repos/docker/test-docker-action/git/trees{/sha}', | ||||||
|  |       updated_at: '2022-04-19T09:05:09Z', | ||||||
|  |       url: 'https://github.com/docker/test-docker-action', | ||||||
|  |       visibility: 'private', | ||||||
|  |       watchers: 0, | ||||||
|  |       watchers_count: 0 | ||||||
|  |     }, | ||||||
|  |     sender: { | ||||||
|  |       avatar_url: 'https://avatars.githubusercontent.com/u/1951866?v=4', | ||||||
|  |       events_url: 'https://api.github.com/users/crazy-max/events{/privacy}', | ||||||
|  |       followers_url: 'https://api.github.com/users/crazy-max/followers', | ||||||
|  |       following_url: 'https://api.github.com/users/crazy-max/following{/other_user}', | ||||||
|  |       gists_url: 'https://api.github.com/users/crazy-max/gists{/gist_id}', | ||||||
|  |       gravatar_id: '', | ||||||
|  |       html_url: 'https://github.com/crazy-max', | ||||||
|  |       id: 1951866, | ||||||
|  |       login: 'crazy-max', | ||||||
|  |       node_id: 'MDQ6VXNlcjE5NTE4NjY=', | ||||||
|  |       organizations_url: 'https://api.github.com/users/crazy-max/orgs', | ||||||
|  |       received_events_url: 'https://api.github.com/users/crazy-max/received_events', | ||||||
|  |       repos_url: 'https://api.github.com/users/crazy-max/repos', | ||||||
|  |       site_admin: false, | ||||||
|  |       starred_url: 'https://api.github.com/users/crazy-max/starred{/owner}{/repo}', | ||||||
|  |       subscriptions_url: 'https://api.github.com/users/crazy-max/subscriptions', | ||||||
|  |       type: 'User', | ||||||
|  |       url: 'https://api.github.com/users/crazy-max' | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const getOctokit = jest.fn(); | ||||||
| @@ -1,146 +0,0 @@ | |||||||
| import {describe, expect, it, jest, test} from '@jest/globals'; |  | ||||||
| import * as fs from 'fs'; |  | ||||||
| import * as path from 'path'; |  | ||||||
| import * as semver from 'semver'; |  | ||||||
| import * as exec from '@actions/exec'; |  | ||||||
| import * as buildx from '../src/buildx'; |  | ||||||
| import * as context from '../src/context'; |  | ||||||
|  |  | ||||||
| const tmpNameSync = path.join('/tmp/.docker-build-push-jest', '.tmpname-jest').split(path.sep).join(path.posix.sep); |  | ||||||
| const imageID = 'sha256:bfb45ab72e46908183546477a08f8867fc40cebadd00af54b071b097aed127a9'; |  | ||||||
| const metadata = `{ |  | ||||||
|   "containerimage.config.digest": "sha256:059b68a595b22564a1cbc167af369349fdc2ecc1f7bc092c2235cbf601a795fd", |  | ||||||
|   "containerimage.digest": "sha256:b09b9482c72371486bb2c1d2c2a2633ed1d0b8389e12c8d52b9e052725c0c83c" |  | ||||||
| }`; |  | ||||||
|  |  | ||||||
| jest.spyOn(context, 'tmpDir').mockImplementation((): string => { |  | ||||||
|   const tmpDir = path.join('/tmp/.docker-build-push-jest').split(path.sep).join(path.posix.sep); |  | ||||||
|   if (!fs.existsSync(tmpDir)) { |  | ||||||
|     fs.mkdirSync(tmpDir, {recursive: true}); |  | ||||||
|   } |  | ||||||
|   return tmpDir; |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| jest.spyOn(context, 'tmpNameSync').mockImplementation((): string => { |  | ||||||
|   return tmpNameSync; |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('getImageID', () => { |  | ||||||
|   it('matches', async () => { |  | ||||||
|     const imageIDFile = await buildx.getImageIDFile(); |  | ||||||
|     await fs.writeFileSync(imageIDFile, imageID); |  | ||||||
|     const expected = await buildx.getImageID(); |  | ||||||
|     expect(expected).toEqual(imageID); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('getMetadata', () => { |  | ||||||
|   it('matches', async () => { |  | ||||||
|     const metadataFile = await buildx.getMetadataFile(); |  | ||||||
|     await fs.writeFileSync(metadataFile, metadata); |  | ||||||
|     const expected = await buildx.getMetadata(); |  | ||||||
|     expect(expected).toEqual(metadata); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('getDigest', () => { |  | ||||||
|   it('matches', async () => { |  | ||||||
|     const metadataFile = await buildx.getMetadataFile(); |  | ||||||
|     await fs.writeFileSync(metadataFile, metadata); |  | ||||||
|     const expected = await buildx.getDigest(metadata); |  | ||||||
|     expect(expected).toEqual('sha256:b09b9482c72371486bb2c1d2c2a2633ed1d0b8389e12c8d52b9e052725c0c83c'); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('isLocalOrTarExporter', () => { |  | ||||||
|   test.each([ |  | ||||||
|     [['type=registry,ref=user/app'], false], |  | ||||||
|     [['type=docker'], false], |  | ||||||
|     [['type=local,dest=./release-out'], true], |  | ||||||
|     [['type=tar,dest=/tmp/image.tar'], true], |  | ||||||
|     [['type=docker', 'type=tar,dest=/tmp/image.tar'], true], |  | ||||||
|     [['"type=tar","dest=/tmp/image.tar"'], true], |  | ||||||
|     [['" type= local" , dest=./release-out'], true], |  | ||||||
|     [['.'], true] |  | ||||||
|   ])('given %p returns %p', async (outputs: Array<string>, expected: boolean) => { |  | ||||||
|     expect(buildx.isLocalOrTarExporter(outputs)).toEqual(expected); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('isAvailable', () => { |  | ||||||
|   const execSpy = jest.spyOn(exec, 'getExecOutput'); |  | ||||||
|   buildx.isAvailable(); |  | ||||||
|  |  | ||||||
|   // eslint-disable-next-line jest/no-standalone-expect |  | ||||||
|   expect(execSpy).toHaveBeenCalledWith(`docker`, ['buildx'], { |  | ||||||
|     silent: true, |  | ||||||
|     ignoreReturnCode: true |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('isAvailable standalone', () => { |  | ||||||
|   const execSpy = jest.spyOn(exec, 'getExecOutput'); |  | ||||||
|   buildx.isAvailable(true); |  | ||||||
|  |  | ||||||
|   // eslint-disable-next-line jest/no-standalone-expect |  | ||||||
|   expect(execSpy).toHaveBeenCalledWith(`buildx`, [], { |  | ||||||
|     silent: true, |  | ||||||
|     ignoreReturnCode: true |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('getVersion', () => { |  | ||||||
|   it('valid', async () => { |  | ||||||
|     const version = await buildx.getVersion(); |  | ||||||
|     expect(semver.valid(version)).not.toBeNull(); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('parseVersion', () => { |  | ||||||
|   test.each([ |  | ||||||
|     ['github.com/docker/buildx 0.4.1+azure bda4882a65349ca359216b135896bddc1d92461c', '0.4.1'], |  | ||||||
|     ['github.com/docker/buildx v0.4.1 bda4882a65349ca359216b135896bddc1d92461c', '0.4.1'], |  | ||||||
|     ['github.com/docker/buildx v0.4.2 fb7b670b764764dc4716df3eba07ffdae4cc47b2', '0.4.2'], |  | ||||||
|     ['github.com/docker/buildx f117971 f11797113e5a9b86bd976329c5dbb8a8bfdfadfa', 'f117971'] |  | ||||||
|   ])('given %p', async (stdout, expected) => { |  | ||||||
|     expect(buildx.parseVersion(stdout)).toEqual(expected); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('satisfies', () => { |  | ||||||
|   test.each([ |  | ||||||
|     ['0.4.1', '>=0.3.2', true], |  | ||||||
|     ['bda4882a65349ca359216b135896bddc1d92461c', '>0.1.0', false], |  | ||||||
|     ['f117971', '>0.6.0', true] |  | ||||||
|   ])('given %p', async (version, range, expected) => { |  | ||||||
|     expect(buildx.satisfies(version, range)).toBe(expected); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('getSecret', () => { |  | ||||||
|   test.each([ |  | ||||||
|     ['A_SECRET=abcdef0123456789', false, 'A_SECRET', 'abcdef0123456789', false], |  | ||||||
|     ['GIT_AUTH_TOKEN=abcdefghijklmno=0123456789', false, 'GIT_AUTH_TOKEN', 'abcdefghijklmno=0123456789', false], |  | ||||||
|     ['MY_KEY=c3RyaW5nLXdpdGgtZXF1YWxzCg==', false, 'MY_KEY', 'c3RyaW5nLXdpdGgtZXF1YWxzCg==', false], |  | ||||||
|     ['aaaaaaaa', false, '', '', true], |  | ||||||
|     ['aaaaaaaa=', false, '', '', true], |  | ||||||
|     ['=bbbbbbb', false, '', '', true], |  | ||||||
|     [`foo=${path.join(__dirname, 'fixtures', 'secret.txt').split(path.sep).join(path.posix.sep)}`, true, 'foo', 'bar', false], |  | ||||||
|     [`notfound=secret`, true, '', '', true] |  | ||||||
|   ])('given %p key and %p secret', async (kvp, file, exKey, exValue, invalid) => { |  | ||||||
|     try { |  | ||||||
|       let secret: string; |  | ||||||
|       if (file) { |  | ||||||
|         secret = await buildx.getSecretFile(kvp); |  | ||||||
|       } else { |  | ||||||
|         secret = await buildx.getSecretString(kvp); |  | ||||||
|       } |  | ||||||
|       expect(true).toBe(!invalid); |  | ||||||
|       expect(secret).toEqual(`id=${exKey},src=${tmpNameSync}`); |  | ||||||
|       expect(fs.readFileSync(tmpNameSync, 'utf-8')).toEqual(exValue); |  | ||||||
|     } catch (err) { |  | ||||||
|       // eslint-disable-next-line jest/no-conditional-expect |  | ||||||
|       expect(true).toBe(invalid); |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,136 +1,74 @@ | |||||||
| import {beforeEach, describe, expect, it, jest, test} from '@jest/globals'; | import {afterEach, beforeEach, describe, expect, jest, test} from '@jest/globals'; | ||||||
| import * as fs from 'fs'; | import * as fs from 'fs'; | ||||||
| import * as path from 'path'; | import * as path from 'path'; | ||||||
|  |  | ||||||
| import * as buildx from '../src/buildx'; | import {Builder} from '@docker/actions-toolkit/lib/buildx/builder'; | ||||||
|  | import {Buildx} from '@docker/actions-toolkit/lib/buildx/buildx'; | ||||||
|  | import {Build} from '@docker/actions-toolkit/lib/buildx/build'; | ||||||
|  | import {Context} from '@docker/actions-toolkit/lib/context'; | ||||||
|  | import {Docker} from '@docker/actions-toolkit/lib/docker/docker'; | ||||||
|  | import {GitHub} from '@docker/actions-toolkit/lib/github'; | ||||||
|  | import {Toolkit} from '@docker/actions-toolkit/lib/toolkit'; | ||||||
|  |  | ||||||
|  | import {BuilderInfo} from '@docker/actions-toolkit/lib/types/buildx/builder'; | ||||||
|  | import {GitHubRepo} from '@docker/actions-toolkit/lib/types/github'; | ||||||
|  |  | ||||||
| import * as context from '../src/context'; | import * as context from '../src/context'; | ||||||
|  |  | ||||||
| const pgp = `-----BEGIN PGP PRIVATE KEY BLOCK----- | const tmpDir = path.join('/tmp', '.docker-build-push-jest'); | ||||||
|  | const tmpName = path.join(tmpDir, '.tmpname-jest'); | ||||||
|  |  | ||||||
| lQdGBF6tzaABEACjFbX7PFEG6vDPN2MPyxYW7/3o/sonORj4HXUFjFxxJxktJ3x3 | import repoFixture from './fixtures/github-repo.json'; | ||||||
| N1ayHPJ1lqIeoiY7jVbq0ZdEVGkd3YsKG9ZMdZkzGzY6PQPC/+M8OnzOiOPwUdWc | jest.spyOn(GitHub.prototype, 'repoData').mockImplementation((): Promise<GitHubRepo> => { | ||||||
| +Tdhh115LvVz0MMKYiab6Sn9cgxj9On3LCQKpjvMDpPo9Ttf6v2GQIw8h2ACvdzQ |   return <Promise<GitHubRepo>>(repoFixture as unknown); | ||||||
| 71LtIELS/I+dLbfZiwpUu2fhQT13EJkEnYMOYwM5jNUd66P9itUc7MrOWjkicrKP |  | ||||||
| oF1dQaCM+tuKuxvD8WLdiwU5x60NoGkJHHUehKQXl2dVzjpqEqHKEBJt9tfJ9lpE |  | ||||||
| YIisgwB8o3pes0fgCehjW2zI95/o9+ayJ6nl4g5+mSvWRXEu66h71nwM0Yuvquk8 |  | ||||||
| 3me7qhYfDrDdCwcxS5BS1hwakTgUQLD99FZjbx1j8sq96I65O0GRdyU2PR8KIjwu |  | ||||||
| JrkTH4ZlKxK3FQghUhFoA5GkiDb+eClmRMSni5qg+81T4XChmUkEprA3eWCHL+Ma |  | ||||||
| xRNNxLS+r6hH9HG5JBxpV3iaTI9HHpnQKhEeaLXqsUTDZliN9hP7Ywo8bpUB8j2d |  | ||||||
| oWYwDV4dPyMKr6Fb8RDCh2q5gJGbVp8w/NmmBTeL+IP2fFggJkRfyumv3Ul7x66L |  | ||||||
| tBFQ4rYo4JUUrGweSTneG6REIgxH66hIrNl6Vo/D1ZyknTe1dMOu/BTkkQARAQAB |  | ||||||
| /gcDAqra8KO+h3bfyu90vxTL1ro4x/x9il7VBcWlIR4cBP7Imgxv+T4hwPIu8P1x |  | ||||||
| lOlxLNWegFOV0idoTy1o3VLLBev/F+IlspX4A+2XEIddR6nZnKFi0Lv2L4TKgE9E |  | ||||||
| VJJTszmviDIRLMLN9dWzDfA8hj5tR5Inot92CHRF414AS22JHvlhbFSLQnjqsN+C |  | ||||||
| n1cQpNOJhkxsSfZsxjnFa/70y/u8v0o8mzyLZmk9HpzRHGzoz8IfpLp8OTqBR9u6 |  | ||||||
| zzoKLy16zZO55OKbj7h8uVZvDUq9l8iDICpqWMdZqBJIl56MBexYKgYxh3YO/8v2 |  | ||||||
| oXli+8Xuaq5QLiCN3yT7IbKoYzplnFfaJwFiMh7R1iPLXaYAZ0qdRijlbtseTK1m |  | ||||||
| oHNkwUbxVzjkh4LfE8UpmMwZn5ZjWni3230SoiXuKy0OHkGvwGvWWAL1mEuoYuUI |  | ||||||
| mFMcH5MnixP8oQYZKDj2IR/yEeOpdU6B/tr3Tk1NidLf7pUMqG7Ff1NU6dAUeBpa |  | ||||||
| 9xahITMjHvrhgMISY4IYZep5cEnVw8lQTpUJtW/ePMzrFhu3sA7oNdj9joW/VMfz |  | ||||||
| H7MHwwavtICsYqoqV3lnjX4EC9dW6o8PTUg2u956dmtK7KAyUK/+w2aLNGT28ChN |  | ||||||
| jhRYHvHzB9Kw5asqI/lTM49eqslBqYQMTTjdBphkYuSZQzNMf291j/ZmoLhD1A1a |  | ||||||
| S8tUnNygKV4D1cJYgSXfzhFoU8ib/0SPo+KqQ+CzGS+wxXg6WNBA6wepTjpnVVx3 |  | ||||||
| 4JADP8IJcDC3P0iwAreWjSy15F1cvemFFB0SLNUkyZGzsxtKzbM1+8khl68+eazC |  | ||||||
| LzRj0rxfIF5znWjX1QFhKxCk6eF0IWDY0+b3DBkmChME9YDXJ3TthcqA7JgcX4JI |  | ||||||
| M4/wdqhgerJYOmj+i2Q0M+Bu02icOJYMwTMMsDVl7XGHkaCuRgZ54eZAUH7JFwUm |  | ||||||
| 1Ct3tcaqiTMmz0ngHVqBTauzgqKDvzwdVqdfg05H364nJMay/3omR6GayIb5CwSo |  | ||||||
| xdNVwG3myPPradT9MP09mDr4ys2zcnQmCkvTVBF6cMZ1Eh6PQQ8CyQWv0zkaBnqj |  | ||||||
| JrM1hRpgW4ZlRosSIjCaaJjolN5QDcXBM9TbW9ww+ZYstazN2bV1ZQ7BEjlHQPa1 |  | ||||||
| BhzMsvqkbETHsIpDNF52gZKn3Q9eIX05BeadzpHUb5/XOheIHVIdhSaTlgl/qQW5 |  | ||||||
| hQgPGSzSV6KhXEY7aevTdvOgq++WiELkjfz2f2lQFesTjFoQWEvxVDUmLxHtEhaN |  | ||||||
| DOuh4H3mX5Opn3pLQmqWVhJTbFdx+g5qQd0NCW4mDaTFWTRLFLZQsSJxDSeg9xrY |  | ||||||
| gmaii8NhMZRwquADW+6iU6KfraBhngi7HRz4TfqPr9ma/KUY464cqim1fnwXejyx |  | ||||||
| jsb5YHR9R66i+F6P/ysF5w+QuVdDt1fnf9GLay0r6qxpA8ft2vGPcDs4806Huj+7 |  | ||||||
| Aq5VeJaNkCuh3GR3xVnCFAz/7AtkO6xKuZm8B3q904UuMdSmkhWbaobIuF/B2B6S |  | ||||||
| eawIXQHEOplK3ic26d8Ckf4gbjeORfELcMAEi5nGXpTThCdmxQApCLxAYYnTfQT1 |  | ||||||
| xhlDwT9xPEabo98mIwJJsAU5VsTDYW+qfo4qIx8gYoSKc9Xu3yVh3n+9k43Gcm5V |  | ||||||
| 9lvK1slijf+TzODZt/jsmkF8mPjXyP5KOI+xQp/m4PxW3pp57YrYj/Rnwga+8DKX |  | ||||||
| jMsW7mLAAZ/e+PY6z/s3x1Krfk+Bb5Ph4mI0zjw5weQdtyEToRgveda0GEpvZSBU |  | ||||||
| ZXN0ZXIgPGpvZUBmb28uYmFyPokCNgQQAQgAIAUCXq3NoAYLCQcIAwIEFQgKAgQW |  | ||||||
| AgEAAhkBAhsDAh4BAAoJEH2FHrctc72gxtQP/AulaClIcn/kDt43mhYnyLglPfbo |  | ||||||
| AqPlU26chXolBg0Wo0frFY3aIs5SrcWEf8aR4XLwCFGyi3vya0CUxjghN5tZBYqo |  | ||||||
| vswbT00zP3ohxxlJFCRRR9bc7OZXCgTddtfVf6EKrUAzIkbWyAhaJnwJy/1UGpSw |  | ||||||
| SEO/KpastrVKf3sv1wqOeFQ4DFyjaNda+xv3dVWS8db7KogqJiPFZXrQK3FKVIxS |  | ||||||
| fxRSmKaYN7//d+xwVAEY++RrnL/o8B2kV6N68cCpQWJELyYnJzis9LBcWd/3wiYh |  | ||||||
| efTyY+ePKUjcB+kEZnyJfLc7C2hll2e7UJ0fxv+k8vHReRhrNWmGRXsjNRxiw3U0 |  | ||||||
| hfvxD/C8nyqAbeTHp4XDX78Tc3XCysAqIYboIL+RyewDMjjLj5vzUYAdUdtyNaD7 |  | ||||||
| C6M2R6pN1GAt52CJmC/Z6F7W7GFGoYOdEkVdMQDsjCwScyEUNlGj9Zagw5M2EgSe |  | ||||||
| 6gaHgMgTzsMzCc4W6WV5RcS55cfDNOXtxPsMJTt4FmXrjl11prBzpMfpU5a9zxDZ |  | ||||||
| oi54ZZ8VPE6jsT4Lzw3sni3c83wm28ArM20AzZ1vh7fk3Sfd0u4Yaz7s9JlEm5+D |  | ||||||
| 34tEyli28+QjCQc18EfQUiJqiYEJRxJXJ3esvMHfYi45pV/Eh5DgRW1305fUJV/6 |  | ||||||
| +rGpg0NejsHoZdZPnQdGBF6tzaABEAC4mVXTkVk6Kdfa4r5zlzsoIrR27laUlMkb |  | ||||||
| OBMt+aokqS+BEbmTnMg6xIAmcUT5uvGAc8S/WhrPoYfc15fTUyHIz8ZbDoAg0LO6 |  | ||||||
| 0Io4VkAvNJNEnsSV9VdLBh/XYlc4K49JqKyWTL4/FJFAGbsmHY3b+QU90AS6FYRv |  | ||||||
| KeBAoiyebrjx0vmzb8E8h3xthVLN+AfMlR1ickY62zvnpkbncSMY/skur1D2KfbF |  | ||||||
| 3sFprty2pEtjFcyB5+18l2IyyHGOlEUw1PZdOAV4/Myh1EZRgYBPs80lYTJALCVF |  | ||||||
| IdOakH33WJCImtNZB0AbDTABG+JtMjQGscOa0qzf1Y/7tlhgCrynBBdaIJTx95TD |  | ||||||
| 21BUHcHOu5yTIS6Ulysxfkv611+BiOKHgdq7DVGP78VuzA7bCjlP1+vHqIt3cnIa |  | ||||||
| t2tEyuZ/XF4uc3/i4g0uP9r7AmtET7Z6SKECWjpVv+UEgLx5Cv+ql+LSKYQMvU9a |  | ||||||
| i3B1F9fatn3FSLVYrL4aRxu4TSw9POb0/lgDNmN3lGQOsjGCZPibkHjgPEVxKuiq |  | ||||||
| 9Oi38/VTQ0ZKAmHwBTq1WTZIrPrCW0/YMQ6yIJZulwQ9Yx1cgzYzEfg04fPXlXMi |  | ||||||
| vkvNpKbYIICzqj0/DVztz9wgpW6mnd0A2VX2dqbMM0fJUCHA6pj8AvXY4R+9Q4rj |  | ||||||
| eWRK9ycInQARAQAB/gcDApjt7biRO0PEyrrAiUwDMsJL4/CVMu11qUWEPjKe2Grh |  | ||||||
| ZTW3N+m3neKPRULu+LUtndUcEdVWUCoDzAJ7MwihZtV5vKST/5Scd2inonOaJqoA |  | ||||||
| nS3wnEMN/Sc93HAZiZnFx3NKjQVNCwbuEs45mXkkcjLm2iadrTL8fL4acsu5IsvD |  | ||||||
| LbDwVOPeNnHKl6Hr20e39fK0FuJEyH49JM6U3B1/8385sJB8+E24+hvSF81aMddh |  | ||||||
| Ne4Bc3ZYiYaKxe1quPNKC0CQhAZiT7LsMfkInXr0hY1I+kISNXEJ1dPYOEWiv0Ze |  | ||||||
| jD5Pupn34okKNEeBCx+dK8BmUCi6Jgs7McUA7hN0D/YUS++5fuR55UQq2j8Ui0tS |  | ||||||
| P8GDr86upH3PgEL0STh9fYfJ7TesxurwonWjlmmT62Myl4Pr+RmpS6PXOnhtcADm |  | ||||||
| eGLpzhTveFj4JBLMpyYHgBTqcs12zfprATOpsI/89kmQoGCZpG6+AbfSHqNNPdy2 |  | ||||||
| eqUCBhOZlIIda1z/cexmU3f/gBqyflFf8fkvmlO4AvI8aMH3OpgHdWnzh+AB51xj |  | ||||||
| kmdD/oWel9v7Dz4HoZUfwFaLZ0fE3P9voD8e+sCwqQwVqRY4L/BOYPD5noVOKgOj |  | ||||||
| ABNKu5uKrobj6rFUi6DTUCjFGcmoF1Sc06xFNaagUNggRbmlC/dz22RWdDUYv5ra |  | ||||||
| N6TxIDkGC0cK6ujyK0nes3DN0aHjgwWuMXDYkN3UckiebI4Cv/eF9jvUKOSiIcy1 |  | ||||||
| RtxdazZS4dYg2LBMeJKVkPi5elsNyw2812nEY3du/nEkQYXfYgWOF27OR+g4Y9Yw |  | ||||||
| 1BiqJ1TTjbQnd/khOCrrbzDH1mw00+1XVsT6wjObuYqqxPPS87UrqmMf6OdoYfPm |  | ||||||
| zEOnNLBnsJ5VQM3A3pcT40RfdBrZRO8LjGhzKTreyq3C+jz0RLa5HNE8GgOhGyck |  | ||||||
| ME4h+RhXlE8KGM+tTo6PA1NJSrEt+8kZzxjP4rIEn0aVthCkNXK12inuXtnHm0ao |  | ||||||
| iLUlQOsfPFEnzl0TUPd7+z7j/wB+XiKU/AyEUuB0mvdxdKtqXvajahOyhLjzHQhz |  | ||||||
| ZnNlgANGtiqcSoJmkJ8yAvhrtQX51fQLftxbArRW1RYk/5l+Gy3azR+gUC17M6JN |  | ||||||
| jrUYxn0zlAxDGFH7gACHUONwVekcuEffHzgu2lk7MyO1Y+lPnwabqjG0eWWHuU00 |  | ||||||
| hskJlXyhj7DeR12bwjYkyyjG62GvOH02g3OMvUgNGH+K321Dz539csCh/xwtg7Wt |  | ||||||
| U3YAphU7htQ1dPDfk1IRs7DQo2L+ZTE57vmL5m0l6fTataEWBPUXkygfQFUJOM6Q |  | ||||||
| yY76UEZww1OSDujNeY171NSTzXCVkUeAdAMXgjaHXWLK2QUQUoXbYX/Kr7Vvt9Fu |  | ||||||
| Jh6eGjjp7dSjQ9+DW8CAB8vxd93gsQQGWYjmGu8khkEmx6OdZhmSbDbe915LQTb9 |  | ||||||
| sPhk2s5/Szsvr5W2JJ2321JI6KXBJMZvPC5jEBWmRzOYkRd2vloft+CSMfXF+Zfd |  | ||||||
| nYtc6R3dvb9vcjo+a9wFtfcoDsO0MaPSM+9GB25MamdatmGX6iLOy9Re1UABwUi/ |  | ||||||
| VhTWNkP5uzqx0sDwHEIa2rYOwxpIZDwwjM3oOASCW1DDBQ0BI9KNjfIeL3ubx2mS |  | ||||||
| 2x8hFU9qSK4umoDNbzOqGPSlkdbiPcNjF2ZcSN1qQZiYdwLL5dw6APNyBVjxTN1J |  | ||||||
| gkCdJ/HwAY+r93Lbl5g8gz8d0vJEyfn//34sn9u+toSTw55GcG9Ks1kSKIeDNh0h |  | ||||||
| MiPm3HmJAh8EGAEIAAkFAl6tzaACGwwACgkQfYUety1zvaBV9hAAgliX36pXJ59g |  | ||||||
| 3I9/4R68e/fGg0FMM6D+01yCeiKApOYRrJ0cYKn7ITDYmHhlGGpBAie90UsqX12h |  | ||||||
| hdLP7LoQx7sjTyzQt6JmpA8krIwi2ON7FKBkdYb8IYx4mE/5vKnYT4/SFnwTmnZY |  | ||||||
| +m+NzK2U/qmhq8JyO8gozdAKJUcgz49IVv2Ij0tQ4qaPbyPwQxIDyKnT758nJhB1 |  | ||||||
| jTqo+oWtER8q3okzIlqcArqn5rDaNJx+DRYL4E/IddyHQAiUWUka8usIUqeW5reu |  | ||||||
| zoPUE2CCfOJSGArkqHQQqMx0WEzjQTwAPaHrQbera4SbiV/o4CLCV/u5p1Qnig+Q |  | ||||||
| iUsakmlD299t//125LIQEa5qzd9hRC7u1uJS7VdW8eGIEcZ0/XT/sr+z23z0kpZH |  | ||||||
| D3dXPX0BwM4IP9xu31CNg10x0rKwjbxy8VaskFEelpqpu+gpAnxqMd1evpeUHcOd |  | ||||||
| r5RgPgkNFfba9Nbxf7uEX+HOmsOM+kdtSmdGIvsBZjVnW31nnoDMp49jG4OynjrH |  | ||||||
| cRuoM9sxdr6UDqb22CZ3/e0YN4UaZM3YDWMVaP/QBVgvIFcdByqNWezpd9T4ZUII |  | ||||||
| MZlaV1uRnHg6B/zTzhIdMM80AXz6Uv6kw4S+Lt7HlbrnMT7uKLuvzH7cle0hcIUa |  | ||||||
| PejgXO0uIRolYQ3sz2tMGhx1MfBqH64= |  | ||||||
| =WbwB |  | ||||||
| -----END PGP PRIVATE KEY BLOCK-----`; |  | ||||||
|  |  | ||||||
| jest.spyOn(context, 'defaultContext').mockImplementation((): string => { |  | ||||||
|   return 'https://github.com/docker/build-push-action.git#refs/heads/test-jest'; |  | ||||||
| }); | }); | ||||||
|  |  | ||||||
| jest.spyOn(context, 'tmpDir').mockImplementation((): string => { | jest.spyOn(Context, 'tmpDir').mockImplementation((): string => { | ||||||
|   const tmpDir = path.join('/tmp/.docker-build-push-jest').split(path.sep).join(path.posix.sep); |  | ||||||
|   if (!fs.existsSync(tmpDir)) { |   if (!fs.existsSync(tmpDir)) { | ||||||
|     fs.mkdirSync(tmpDir, {recursive: true}); |     fs.mkdirSync(tmpDir, {recursive: true}); | ||||||
|   } |   } | ||||||
|   return tmpDir; |   return tmpDir; | ||||||
| }); | }); | ||||||
|  |  | ||||||
| jest.spyOn(context, 'tmpNameSync').mockImplementation((): string => { | jest.spyOn(Context, 'tmpName').mockImplementation((): string => { | ||||||
|   return path.join('/tmp/.docker-build-push-jest', '.tmpname-jest').split(path.sep).join(path.posix.sep); |   return tmpName; | ||||||
| }); | }); | ||||||
|  |  | ||||||
| jest.spyOn(buildx, 'satisfiesBuildKitVersion').mockResolvedValueOnce(true); | jest.spyOn(Docker, 'isAvailable').mockImplementation(async (): Promise<boolean> => { | ||||||
|  |   return true; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | const metadataJson = path.join(tmpDir, 'metadata.json'); | ||||||
|  | jest.spyOn(Build.prototype, 'getMetadataFilePath').mockImplementation((): string => { | ||||||
|  |   return metadataJson; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | const imageIDFilePath = path.join(tmpDir, 'iidfile.txt'); | ||||||
|  | jest.spyOn(Build.prototype, 'getImageIDFilePath').mockImplementation((): string => { | ||||||
|  |   return imageIDFilePath; | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | jest.spyOn(Builder.prototype, 'inspect').mockImplementation(async (): Promise<BuilderInfo> => { | ||||||
|  |   return { | ||||||
|  |     name: 'builder2', | ||||||
|  |     driver: 'docker-container', | ||||||
|  |     lastActivity: new Date('2023-01-16 09:45:23 +0000 UTC'), | ||||||
|  |     nodes: [ | ||||||
|  |       { | ||||||
|  |         buildkit: 'v0.11.0', | ||||||
|  |         'buildkitd-flags': '--debug --allow-insecure-entitlement security.insecure --allow-insecure-entitlement network.host', | ||||||
|  |         'driver-opts': ['BUILDKIT_STEP_LOG_MAX_SIZE=10485760', 'BUILDKIT_STEP_LOG_MAX_SPEED=10485760', 'JAEGER_TRACE=localhost:6831', 'image=moby/buildkit:latest', 'network=host'], | ||||||
|  |         endpoint: 'unix:///var/run/docker.sock', | ||||||
|  |         name: 'builder20', | ||||||
|  |         platforms: 'linux/amd64,linux/amd64/v2,linux/amd64/v3,linux/arm64,linux/riscv64,linux/ppc64le,linux/s390x,linux/386,linux/mips64le,linux/mips64,linux/arm/v7,linux/arm/v6', | ||||||
|  |         status: 'running' | ||||||
|  |       } | ||||||
|  |     ] | ||||||
|  |   }; | ||||||
|  | }); | ||||||
|  |  | ||||||
| describe('getArgs', () => { | describe('getArgs', () => { | ||||||
|  |   const originalEnv = process.env; | ||||||
|   beforeEach(() => { |   beforeEach(() => { | ||||||
|     process.env = Object.keys(process.env).reduce((object, key) => { |     process.env = Object.keys(process.env).reduce((object, key) => { | ||||||
|       if (!key.startsWith('INPUT_')) { |       if (!key.startsWith('INPUT_')) { | ||||||
| @@ -139,6 +77,9 @@ describe('getArgs', () => { | |||||||
|       return object; |       return object; | ||||||
|     }, {}); |     }, {}); | ||||||
|   }); |   }); | ||||||
|  |   afterEach(() => { | ||||||
|  |     process.env = originalEnv; | ||||||
|  |   }); | ||||||
|  |  | ||||||
|   // prettier-ignore |   // prettier-ignore | ||||||
|   test.each([ |   test.each([ | ||||||
| @@ -154,9 +95,10 @@ describe('getArgs', () => { | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       1, |       1, | ||||||
| @@ -177,9 +119,10 @@ ccc"`], | |||||||
|         '--build-arg', 'MY_ARG=val1,val2,val3', |         '--build-arg', 'MY_ARG=val1,val2,val3', | ||||||
|         '--build-arg', 'ARG=val', |         '--build-arg', 'ARG=val', | ||||||
|         '--build-arg', `MULTILINE=aaaa\nbbbb\nccc`, |         '--build-arg', `MULTILINE=aaaa\nbbbb\nccc`, | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         'https://github.com/docker/build-push-action.git#refs/heads/test-jest' |         'https://github.com/docker/build-push-action.git#refs/heads/master' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       2, |       2, | ||||||
| @@ -193,11 +136,12 @@ ccc"`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--tag', 'name/app:7.4', |         '--tag', 'name/app:7.4', | ||||||
|         '--tag', 'name/app:latest', |         '--tag', 'name/app:latest', | ||||||
|         'https://github.com/docker/build-push-action.git#refs/heads/test-jest' |         'https://github.com/docker/build-push-action.git#refs/heads/master' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       3, |       3, | ||||||
| @@ -217,7 +161,8 @@ ccc"`], | |||||||
|         '--label', 'org.opencontainers.image.description=concurrent, cache-efficient, and Dockerfile-agnostic builder toolkit', |         '--label', 'org.opencontainers.image.description=concurrent, cache-efficient, and Dockerfile-agnostic builder toolkit', | ||||||
|         '--output', 'type=local,dest=./release-out', |         '--output', 'type=local,dest=./release-out', | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       4, |       4, | ||||||
| @@ -234,7 +179,8 @@ ccc"`], | |||||||
|         'build', |         'build', | ||||||
|         '--platform', 'linux/amd64,linux/arm64', |         '--platform', 'linux/amd64,linux/arm64', | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       5, |       5, | ||||||
| @@ -248,9 +194,10 @@ ccc"`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       6, |       6, | ||||||
| @@ -265,10 +212,11 @@ ccc"`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--secret', 'id=GIT_AUTH_TOKEN,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`, | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       7, |       7, | ||||||
| @@ -284,9 +232,10 @@ ccc"`], | |||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--output', '.', |         '--output', '.', | ||||||
|         '--secret', 'id=GIT_AUTH_TOKEN,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`, | ||||||
|         'https://github.com/docker/build-push-action.git#refs/heads/test-jest' |         'https://github.com/docker/build-push-action.git#refs/heads/master' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       8, |       8, | ||||||
| @@ -306,13 +255,14 @@ ccc"`], | |||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--file', './test/Dockerfile', |         '--file', './test/Dockerfile', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--platform', 'linux/amd64,linux/arm64', |         '--platform', 'linux/amd64,linux/arm64', | ||||||
|         '--secret', 'id=GIT_AUTH_TOKEN,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`, | ||||||
|         '--builder', 'builder-git-context-2', |         '--builder', 'builder-git-context-2', | ||||||
|         '--push', |         '--push', | ||||||
|         'https://github.com/docker/build-push-action.git#refs/heads/master' |         'https://github.com/docker/build-push-action.git#refs/heads/master' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       9, |       9, | ||||||
| @@ -340,16 +290,17 @@ ccc"`], | |||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--file', './test/Dockerfile', |         '--file', './test/Dockerfile', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--platform', 'linux/amd64,linux/arm64', |         '--platform', 'linux/amd64,linux/arm64', | ||||||
|         '--secret', 'id=GIT_AUTH_TOKEN,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`, | ||||||
|         '--secret', 'id=MYSECRET,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=MYSECRET,src=${tmpName}`, | ||||||
|         '--secret', 'id=FOO,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=FOO,src=${tmpName}`, | ||||||
|         '--secret', 'id=EMPTYLINE,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=EMPTYLINE,src=${tmpName}`, | ||||||
|         '--builder', 'builder-git-context-2', |         '--builder', 'builder-git-context-2', | ||||||
|         '--push', |         '--push', | ||||||
|         'https://github.com/docker/build-push-action.git#refs/heads/master' |         'https://github.com/docker/build-push-action.git#refs/heads/master' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       10, |       10, | ||||||
| @@ -377,16 +328,17 @@ ccc`], | |||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--file', './test/Dockerfile', |         '--file', './test/Dockerfile', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--platform', 'linux/amd64,linux/arm64', |         '--platform', 'linux/amd64,linux/arm64', | ||||||
|         '--secret', 'id=GIT_AUTH_TOKEN,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`, | ||||||
|         '--secret', 'id=MYSECRET,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=MYSECRET,src=${tmpName}`, | ||||||
|         '--secret', 'id=FOO,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=FOO,src=${tmpName}`, | ||||||
|         '--secret', 'id=EMPTYLINE,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=EMPTYLINE,src=${tmpName}`, | ||||||
|         '--builder', 'builder-git-context-2', |         '--builder', 'builder-git-context-2', | ||||||
|         '--push', |         '--push', | ||||||
|         'https://github.com/docker/build-push-action.git#refs/heads/master' |         'https://github.com/docker/build-push-action.git#refs/heads/master' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       11, |       11, | ||||||
| @@ -394,7 +346,7 @@ ccc`], | |||||||
|       new Map<string, string>([ |       new Map<string, string>([ | ||||||
|         ['context', 'https://github.com/docker/build-push-action.git#refs/heads/master'], |         ['context', 'https://github.com/docker/build-push-action.git#refs/heads/master'], | ||||||
|         ['tag', 'localhost:5000/name/app:latest'], |         ['tag', 'localhost:5000/name/app:latest'], | ||||||
|         ['secret-files', `MY_SECRET=${path.join(__dirname, 'fixtures', 'secret.txt').split(path.sep).join(path.posix.sep)}`], |         ['secret-files', `MY_SECRET=${path.join(__dirname, 'fixtures', 'secret.txt')}`], | ||||||
|         ['file', './test/Dockerfile'], |         ['file', './test/Dockerfile'], | ||||||
|         ['builder', 'builder-git-context-2'], |         ['builder', 'builder-git-context-2'], | ||||||
|         ['network', 'host'], |         ['network', 'host'], | ||||||
| @@ -406,13 +358,14 @@ ccc`], | |||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--file', './test/Dockerfile', |         '--file', './test/Dockerfile', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--secret', 'id=MY_SECRET,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=MY_SECRET,src=${tmpName}`, | ||||||
|         '--builder', 'builder-git-context-2', |         '--builder', 'builder-git-context-2', | ||||||
|         '--network', 'host', |         '--network', 'host', | ||||||
|         '--push', |         '--push', | ||||||
|         'https://github.com/docker/build-push-action.git#refs/heads/master' |         'https://github.com/docker/build-push-action.git#refs/heads/master' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       12, |       12, | ||||||
| @@ -432,7 +385,8 @@ ccc`], | |||||||
|         '--label', 'org.opencontainers.image.description=Reference implementation of operation "filter results (top-n)"', |         '--label', 'org.opencontainers.image.description=Reference implementation of operation "filter results (top-n)"', | ||||||
|         '--output', 'type=local,dest=./release-out', |         '--output', 'type=local,dest=./release-out', | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       13, |       13, | ||||||
| @@ -453,12 +407,13 @@ ccc`], | |||||||
|         '--add-host', 'docker:10.180.0.1', |         '--add-host', 'docker:10.180.0.1', | ||||||
|         '--add-host', 'foo:10.0.0.1', |         '--add-host', 'foo:10.0.0.1', | ||||||
|         '--file', './test/Dockerfile', |         '--file', './test/Dockerfile', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--metadata-file', '/tmp/.docker-build-push-jest/metadata-file', |         '--metadata-file', metadataJson, | ||||||
|         '--network', 'host', |         '--network', 'host', | ||||||
|         '--push', |         '--push', | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       14, |       14, | ||||||
| @@ -482,13 +437,14 @@ nproc=3`], | |||||||
|         '--add-host', 'foo:10.0.0.1', |         '--add-host', 'foo:10.0.0.1', | ||||||
|         '--cgroup-parent', 'foo', |         '--cgroup-parent', 'foo', | ||||||
|         '--file', './test/Dockerfile', |         '--file', './test/Dockerfile', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--shm-size', '2g', |         '--shm-size', '2g', | ||||||
|         '--ulimit', 'nofile=1024:1024', |         '--ulimit', 'nofile=1024:1024', | ||||||
|         '--ulimit', 'nproc=3', |         '--ulimit', 'nproc=3', | ||||||
|         '--metadata-file', '/tmp/.docker-build-push-jest/metadata-file', |         '--metadata-file', metadataJson, | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       15, |       15, | ||||||
| @@ -502,10 +458,11 @@ nproc=3`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--metadata-file', '/tmp/.docker-build-push-jest/metadata-file', |         '--metadata-file', metadataJson, | ||||||
|         'https://github.com/docker/build-push-action.git#refs/heads/test-jest:docker' |         'https://github.com/docker/build-push-action.git#refs/heads/master:docker' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       16, |       16, | ||||||
| @@ -520,11 +477,12 @@ nproc=3`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--secret', 'id=GIT_AUTH_TOKEN,src=/tmp/.docker-build-push-jest/.tmpname-jest', |         '--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`, | ||||||
|         '--metadata-file', '/tmp/.docker-build-push-jest/metadata-file', |         '--metadata-file', metadataJson, | ||||||
|         'https://github.com/docker/build-push-action.git#refs/heads/test-jest:subdir' |         'https://github.com/docker/build-push-action.git#refs/heads/master:subdir' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       17, |       17, | ||||||
| @@ -539,10 +497,11 @@ nproc=3`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         '--metadata-file', '/tmp/.docker-build-push-jest/metadata-file', |         '--metadata-file', metadataJson, | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       18, |       18, | ||||||
| @@ -556,11 +515,12 @@ nproc=3`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         "--provenance", `mode=min,inline-only=true,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789`, |         '--attest', `type=provenance,mode=min,inline-only=true,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`, | ||||||
|         '--metadata-file', '/tmp/.docker-build-push-jest/metadata-file', |         '--metadata-file', metadataJson, | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       19, |       19, | ||||||
| @@ -575,11 +535,12 @@ nproc=3`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         "--provenance", `builder-id=https://github.com/docker/build-push-action/actions/runs/123456789`, |         '--attest', `type=provenance,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`, | ||||||
|         '--metadata-file', '/tmp/.docker-build-push-jest/metadata-file', |         '--metadata-file', metadataJson, | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       20, |       20, | ||||||
| @@ -594,11 +555,12 @@ nproc=3`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         "--provenance", `mode=max,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789`, |         '--attest', `type=provenance,mode=max,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`, | ||||||
|         '--metadata-file', '/tmp/.docker-build-push-jest/metadata-file', |         '--metadata-file', metadataJson, | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       21, |       21, | ||||||
| @@ -613,11 +575,12 @@ nproc=3`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         "--provenance", 'false', |         '--attest', 'type=provenance,disabled=true', | ||||||
|         '--metadata-file', '/tmp/.docker-build-push-jest/metadata-file', |         '--metadata-file', metadataJson, | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|     ], |     ], | ||||||
|     [ |     [ | ||||||
|       22, |       22, | ||||||
| @@ -632,183 +595,290 @@ nproc=3`], | |||||||
|       ]), |       ]), | ||||||
|       [ |       [ | ||||||
|         'build', |         'build', | ||||||
|         '--iidfile', '/tmp/.docker-build-push-jest/iidfile', |         '--iidfile', imageIDFilePath, | ||||||
|         "--provenance", 'builder-id=foo', |         '--attest', 'type=provenance,builder-id=foo', | ||||||
|         '--metadata-file', '/tmp/.docker-build-push-jest/metadata-file', |         '--metadata-file', metadataJson, | ||||||
|         '.' |         '.' | ||||||
|       ] |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       23, | ||||||
|  |       '0.10.0', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['load', 'false'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |         ['outputs', 'type=docker'], | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--iidfile', imageIDFilePath, | ||||||
|  |         "--output", 'type=docker', | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       24, | ||||||
|  |       '0.10.0', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['load', 'true'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--iidfile', imageIDFilePath, | ||||||
|  |         '--load', | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       25, | ||||||
|  |       '0.10.0', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['build-args', `FOO=bar#baz`], | ||||||
|  |         ['load', 'true'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--build-arg', 'FOO=bar#baz', | ||||||
|  |         '--iidfile', imageIDFilePath, | ||||||
|  |         '--load', | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       26, | ||||||
|  |       '0.10.0', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['load', 'true'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |         ['secret-envs', `MY_SECRET=MY_SECRET_ENV | ||||||
|  | ANOTHER_SECRET=ANOTHER_SECRET_ENV`] | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--secret', 'id=MY_SECRET,env=MY_SECRET_ENV', | ||||||
|  |         '--secret', 'id=ANOTHER_SECRET,env=ANOTHER_SECRET_ENV', | ||||||
|  |         '--iidfile', imageIDFilePath, | ||||||
|  |         '--load', | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       27, | ||||||
|  |       '0.10.0', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['load', 'true'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |         ['secret-envs', 'MY_SECRET=MY_SECRET_ENV,ANOTHER_SECRET=ANOTHER_SECRET_ENV'] | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--secret', 'id=MY_SECRET,env=MY_SECRET_ENV', | ||||||
|  |         '--secret', 'id=ANOTHER_SECRET,env=ANOTHER_SECRET_ENV', | ||||||
|  |         '--iidfile', imageIDFilePath, | ||||||
|  |         '--load', | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       28, | ||||||
|  |       '0.11.0', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['annotations', 'example1=www\nindex:example2=xxx\nmanifest:example3=yyy\nmanifest-descriptor[linux/amd64]:example4=zzz'], | ||||||
|  |         ['outputs', 'type=local,dest=./release-out'], | ||||||
|  |         ['load', 'false'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--output', 'type=local,dest=./release-out', | ||||||
|  |         '--attest', `type=provenance,mode=min,inline-only=true,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`, | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       29, | ||||||
|  |       '0.12.0', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['annotations', 'example1=www\nindex:example2=xxx\nmanifest:example3=yyy\nmanifest-descriptor[linux/amd64]:example4=zzz'], | ||||||
|  |         ['outputs', 'type=local,dest=./release-out'], | ||||||
|  |         ['load', 'false'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--annotation', 'example1=www', | ||||||
|  |         '--annotation', 'index:example2=xxx', | ||||||
|  |         '--annotation', 'manifest:example3=yyy', | ||||||
|  |         '--annotation', 'manifest-descriptor[linux/amd64]:example4=zzz', | ||||||
|  |         '--output', 'type=local,dest=./release-out', | ||||||
|  |         '--attest', `type=provenance,mode=min,inline-only=true,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`, | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       30, | ||||||
|  |       '0.12.0', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['outputs', `type=image,"name=localhost:5000/name/app:latest,localhost:5000/name/app:foo",push-by-digest=true,name-canonical=true,push=true`], | ||||||
|  |         ['load', 'false'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--iidfile', imageIDFilePath, | ||||||
|  |         "--output", `type=image,"name=localhost:5000/name/app:latest,localhost:5000/name/app:foo",push-by-digest=true,name-canonical=true,push=true`, | ||||||
|  |         '--attest', `type=provenance,mode=min,inline-only=true,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`, | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       31, | ||||||
|  |       '0.13.1', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['load', 'false'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |         ['provenance', 'mode=max'], | ||||||
|  |         ['sbom', 'true'], | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--iidfile', imageIDFilePath, | ||||||
|  |         '--attest', `type=provenance,mode=max,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`, | ||||||
|  |         '--attest', `type=sbom,disabled=false`, | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       32, | ||||||
|  |       '0.13.1', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['load', 'false'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |         ['attests', 'type=provenance,mode=min'], | ||||||
|  |         ['provenance', 'mode=max'], | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--iidfile', imageIDFilePath, | ||||||
|  |         '--attest', `type=provenance,mode=max,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`, | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       33, | ||||||
|  |       '0.13.1', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['load', 'false'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'], | ||||||
|  |         ['attests', 'type=provenance,mode=min'], | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--iidfile', imageIDFilePath, | ||||||
|  |         '--attest', `type=provenance,mode=min,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`, | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       undefined | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       34, | ||||||
|  |       '0.13.1', | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['context', '.'], | ||||||
|  |         ['load', 'false'], | ||||||
|  |         ['no-cache', 'false'], | ||||||
|  |         ['push', 'false'], | ||||||
|  |         ['pull', 'false'] | ||||||
|  |       ]), | ||||||
|  |       [ | ||||||
|  |         'build', | ||||||
|  |         '--iidfile', imageIDFilePath, | ||||||
|  |         '--metadata-file', metadataJson, | ||||||
|  |         '.' | ||||||
|  |       ], | ||||||
|  |       new Map<string, string>([ | ||||||
|  |         ['BUILDX_NO_DEFAULT_ATTESTATIONS', '1'] | ||||||
|  |       ]) | ||||||
|     ], |     ], | ||||||
|   ])( |   ])( | ||||||
|     '[%d] given %p with %p as inputs, returns %p', |     '[%d] given %p with %p as inputs, returns %p', | ||||||
|     async (num: number, buildxVersion: string, inputs: Map<string, string>, expected: Array<string>) => { |     async (num: number, buildxVersion: string, inputs: Map<string, string>, expected: Array<string>, envs: Map<string, string> | undefined) => { | ||||||
|  |       if (envs) { | ||||||
|  |         envs.forEach((value: string, name: string) => { | ||||||
|  |           process.env[name] = value; | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|       inputs.forEach((value: string, name: string) => { |       inputs.forEach((value: string, name: string) => { | ||||||
|         setInput(name, value); |         setInput(name, value); | ||||||
|       }); |       }); | ||||||
|       const defContext = context.defaultContext(); |       const toolkit = new Toolkit(); | ||||||
|       const inp = await context.getInputs(defContext); |       jest.spyOn(Buildx.prototype, 'version').mockImplementation(async (): Promise<string> => { | ||||||
|       const res = await context.getArgs(inp, defContext, buildxVersion); |         return buildxVersion; | ||||||
|  |       }); | ||||||
|  |       const inp = await context.getInputs(); | ||||||
|  |       const res = await context.getArgs(inp, toolkit); | ||||||
|       expect(res).toEqual(expected); |       expect(res).toEqual(expected); | ||||||
|     } |     } | ||||||
|   ); |   ); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| describe('getInputList', () => { |  | ||||||
|   it('single line correctly', async () => { |  | ||||||
|     await setInput('foo', 'bar'); |  | ||||||
|     const res = await context.getInputList('foo'); |  | ||||||
|     expect(res).toEqual(['bar']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('multiline correctly', async () => { |  | ||||||
|     setInput('foo', 'bar\nbaz'); |  | ||||||
|     const res = await context.getInputList('foo'); |  | ||||||
|     expect(res).toEqual(['bar', 'baz']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('empty lines correctly', async () => { |  | ||||||
|     setInput('foo', 'bar\n\nbaz'); |  | ||||||
|     const res = await context.getInputList('foo'); |  | ||||||
|     expect(res).toEqual(['bar', 'baz']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('comma correctly', async () => { |  | ||||||
|     setInput('foo', 'bar,baz'); |  | ||||||
|     const res = await context.getInputList('foo'); |  | ||||||
|     expect(res).toEqual(['bar', 'baz']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('empty result correctly', async () => { |  | ||||||
|     setInput('foo', 'bar,baz,'); |  | ||||||
|     const res = await context.getInputList('foo'); |  | ||||||
|     expect(res).toEqual(['bar', 'baz']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('different new lines correctly', async () => { |  | ||||||
|     setInput('foo', 'bar\r\nbaz'); |  | ||||||
|     const res = await context.getInputList('foo'); |  | ||||||
|     expect(res).toEqual(['bar', 'baz']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('different new lines and comma correctly', async () => { |  | ||||||
|     setInput('foo', 'bar\r\nbaz,bat'); |  | ||||||
|     const res = await context.getInputList('foo'); |  | ||||||
|     expect(res).toEqual(['bar', 'baz', 'bat']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('multiline and ignoring comma correctly', async () => { |  | ||||||
|     setInput('cache-from', 'user/app:cache\ntype=local,src=path/to/dir'); |  | ||||||
|     const res = await context.getInputList('cache-from', true); |  | ||||||
|     expect(res).toEqual(['user/app:cache', 'type=local,src=path/to/dir']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('different new lines and ignoring comma correctly', async () => { |  | ||||||
|     setInput('cache-from', 'user/app:cache\r\ntype=local,src=path/to/dir'); |  | ||||||
|     const res = await context.getInputList('cache-from', true); |  | ||||||
|     expect(res).toEqual(['user/app:cache', 'type=local,src=path/to/dir']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('multiline values', async () => { |  | ||||||
|     setInput( |  | ||||||
|       'secrets', |  | ||||||
|       `GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789 |  | ||||||
| "MYSECRET=aaaaaaaa |  | ||||||
| bbbbbbb |  | ||||||
| ccccccccc" |  | ||||||
| FOO=bar` |  | ||||||
|     ); |  | ||||||
|     const res = await context.getInputList('secrets', true); |  | ||||||
|     expect(res).toEqual([ |  | ||||||
|       'GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789', |  | ||||||
|       `MYSECRET=aaaaaaaa |  | ||||||
| bbbbbbb |  | ||||||
| ccccccccc`, |  | ||||||
|       'FOO=bar' |  | ||||||
|     ]); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('multiline values with empty lines', async () => { |  | ||||||
|     setInput( |  | ||||||
|       'secrets', |  | ||||||
|       `GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789 |  | ||||||
| "MYSECRET=aaaaaaaa |  | ||||||
| bbbbbbb |  | ||||||
| ccccccccc" |  | ||||||
| FOO=bar |  | ||||||
| "EMPTYLINE=aaaa |  | ||||||
|  |  | ||||||
| bbbb |  | ||||||
| ccc"` |  | ||||||
|     ); |  | ||||||
|     const res = await context.getInputList('secrets', true); |  | ||||||
|     expect(res).toEqual([ |  | ||||||
|       'GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789', |  | ||||||
|       `MYSECRET=aaaaaaaa |  | ||||||
| bbbbbbb |  | ||||||
| ccccccccc`, |  | ||||||
|       'FOO=bar', |  | ||||||
|       `EMPTYLINE=aaaa |  | ||||||
|  |  | ||||||
| bbbb |  | ||||||
| ccc` |  | ||||||
|     ]); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('multiline values without quotes', async () => { |  | ||||||
|     setInput( |  | ||||||
|       'secrets', |  | ||||||
|       `GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789 |  | ||||||
| MYSECRET=aaaaaaaa |  | ||||||
| bbbbbbb |  | ||||||
| ccccccccc |  | ||||||
| FOO=bar` |  | ||||||
|     ); |  | ||||||
|     const res = await context.getInputList('secrets', true); |  | ||||||
|     expect(res).toEqual(['GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789', 'MYSECRET=aaaaaaaa', 'bbbbbbb', 'ccccccccc', 'FOO=bar']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('large multiline values', async () => { |  | ||||||
|     setInput( |  | ||||||
|       'secrets', |  | ||||||
|       `"GPG_KEY=${pgp}" |  | ||||||
| FOO=bar` |  | ||||||
|     ); |  | ||||||
|     const res = await context.getInputList('secrets', true); |  | ||||||
|     expect(res).toEqual([`GPG_KEY=${pgp}`, 'FOO=bar']); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   it('multiline values escape quotes', async () => { |  | ||||||
|     setInput( |  | ||||||
|       'secrets', |  | ||||||
|       `GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789 |  | ||||||
| "MYSECRET=aaaaaaaa |  | ||||||
| bbbb""bbb |  | ||||||
| ccccccccc" |  | ||||||
| FOO=bar` |  | ||||||
|     ); |  | ||||||
|     const res = await context.getInputList('secrets', true); |  | ||||||
|     expect(res).toEqual([ |  | ||||||
|       'GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789', |  | ||||||
|       `MYSECRET=aaaaaaaa |  | ||||||
| bbbb"bbb |  | ||||||
| ccccccccc`, |  | ||||||
|       'FOO=bar' |  | ||||||
|     ]); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| describe('asyncForEach', () => { |  | ||||||
|   it('executes async tasks sequentially', async () => { |  | ||||||
|     const testValues = [1, 2, 3, 4, 5]; |  | ||||||
|     const results: number[] = []; |  | ||||||
|  |  | ||||||
|     await context.asyncForEach(testValues, async value => { |  | ||||||
|       results.push(value); |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     expect(results).toEqual(testValues); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| // See: https://github.com/actions/toolkit/blob/a1b068ec31a042ff1e10a522d8fdf0b8869d53ca/packages/core/src/core.ts#L89 | // See: https://github.com/actions/toolkit/blob/a1b068ec31a042ff1e10a522d8fdf0b8869d53ca/packages/core/src/core.ts#L89 | ||||||
| function getInputName(name: string): string { | function getInputName(name: string): string { | ||||||
|   return `INPUT_${name.replace(/ /g, '_').toUpperCase()}`; |   return `INPUT_${name.replace(/ /g, '_').toUpperCase()}`; | ||||||
|   | |||||||
| @@ -1,16 +0,0 @@ | |||||||
| import {describe, expect, it, jest} from '@jest/globals'; |  | ||||||
| import * as docker from '../src/docker'; |  | ||||||
| import * as exec from '@actions/exec'; |  | ||||||
|  |  | ||||||
| describe('isAvailable', () => { |  | ||||||
|   it('cli', () => { |  | ||||||
|     const execSpy = jest.spyOn(exec, 'getExecOutput'); |  | ||||||
|     docker.isAvailable(); |  | ||||||
|  |  | ||||||
|     // eslint-disable-next-line jest/no-standalone-expect |  | ||||||
|     expect(execSpy).toHaveBeenCalledWith(`docker`, undefined, { |  | ||||||
|       silent: true, |  | ||||||
|       ignoreReturnCode: true |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
							
								
								
									
										362
									
								
								__tests__/fixtures/github-repo.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										362
									
								
								__tests__/fixtures/github-repo.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,362 @@ | |||||||
|  | { | ||||||
|  |   "id": 1296269, | ||||||
|  |   "node_id": "MDEwOlJlcG9zaXRvcnkxMjk2MjY5", | ||||||
|  |   "name": "Hello-World", | ||||||
|  |   "full_name": "octocat/Hello-World", | ||||||
|  |   "owner": { | ||||||
|  |     "login": "octocat", | ||||||
|  |     "id": 1, | ||||||
|  |     "node_id": "MDQ6VXNlcjE=", | ||||||
|  |     "avatar_url": "https://github.com/images/error/octocat_happy.gif", | ||||||
|  |     "gravatar_id": "", | ||||||
|  |     "url": "https://api.github.com/users/octocat", | ||||||
|  |     "html_url": "https://github.com/octocat", | ||||||
|  |     "followers_url": "https://api.github.com/users/octocat/followers", | ||||||
|  |     "following_url": "https://api.github.com/users/octocat/following{/other_user}", | ||||||
|  |     "gists_url": "https://api.github.com/users/octocat/gists{/gist_id}", | ||||||
|  |     "starred_url": "https://api.github.com/users/octocat/starred{/owner}{/repo}", | ||||||
|  |     "subscriptions_url": "https://api.github.com/users/octocat/subscriptions", | ||||||
|  |     "organizations_url": "https://api.github.com/users/octocat/orgs", | ||||||
|  |     "repos_url": "https://api.github.com/users/octocat/repos", | ||||||
|  |     "events_url": "https://api.github.com/users/octocat/events{/privacy}", | ||||||
|  |     "received_events_url": "https://api.github.com/users/octocat/received_events", | ||||||
|  |     "type": "User", | ||||||
|  |     "site_admin": false | ||||||
|  |   }, | ||||||
|  |   "private": false, | ||||||
|  |   "html_url": "https://github.com/octocat/Hello-World", | ||||||
|  |   "description": "This your first repo!", | ||||||
|  |   "fork": false, | ||||||
|  |   "url": "https://api.github.com/repos/octocat/Hello-World", | ||||||
|  |   "archive_url": "http://api.github.com/repos/octocat/Hello-World/{archive_format}{/ref}", | ||||||
|  |   "assignees_url": "http://api.github.com/repos/octocat/Hello-World/assignees{/user}", | ||||||
|  |   "blobs_url": "http://api.github.com/repos/octocat/Hello-World/git/blobs{/sha}", | ||||||
|  |   "branches_url": "http://api.github.com/repos/octocat/Hello-World/branches{/branch}", | ||||||
|  |   "collaborators_url": "http://api.github.com/repos/octocat/Hello-World/collaborators{/collaborator}", | ||||||
|  |   "comments_url": "http://api.github.com/repos/octocat/Hello-World/comments{/number}", | ||||||
|  |   "commits_url": "http://api.github.com/repos/octocat/Hello-World/commits{/sha}", | ||||||
|  |   "compare_url": "http://api.github.com/repos/octocat/Hello-World/compare/{base}...{head}", | ||||||
|  |   "contents_url": "http://api.github.com/repos/octocat/Hello-World/contents/{+path}", | ||||||
|  |   "contributors_url": "http://api.github.com/repos/octocat/Hello-World/contributors", | ||||||
|  |   "deployments_url": "http://api.github.com/repos/octocat/Hello-World/deployments", | ||||||
|  |   "downloads_url": "http://api.github.com/repos/octocat/Hello-World/downloads", | ||||||
|  |   "events_url": "http://api.github.com/repos/octocat/Hello-World/events", | ||||||
|  |   "forks_url": "http://api.github.com/repos/octocat/Hello-World/forks", | ||||||
|  |   "git_commits_url": "http://api.github.com/repos/octocat/Hello-World/git/commits{/sha}", | ||||||
|  |   "git_refs_url": "http://api.github.com/repos/octocat/Hello-World/git/refs{/sha}", | ||||||
|  |   "git_tags_url": "http://api.github.com/repos/octocat/Hello-World/git/tags{/sha}", | ||||||
|  |   "git_url": "git:github.com/octocat/Hello-World.git", | ||||||
|  |   "issue_comment_url": "http://api.github.com/repos/octocat/Hello-World/issues/comments{/number}", | ||||||
|  |   "issue_events_url": "http://api.github.com/repos/octocat/Hello-World/issues/events{/number}", | ||||||
|  |   "issues_url": "http://api.github.com/repos/octocat/Hello-World/issues{/number}", | ||||||
|  |   "keys_url": "http://api.github.com/repos/octocat/Hello-World/keys{/key_id}", | ||||||
|  |   "labels_url": "http://api.github.com/repos/octocat/Hello-World/labels{/name}", | ||||||
|  |   "languages_url": "http://api.github.com/repos/octocat/Hello-World/languages", | ||||||
|  |   "merges_url": "http://api.github.com/repos/octocat/Hello-World/merges", | ||||||
|  |   "milestones_url": "http://api.github.com/repos/octocat/Hello-World/milestones{/number}", | ||||||
|  |   "notifications_url": "http://api.github.com/repos/octocat/Hello-World/notifications{?since,all,participating}", | ||||||
|  |   "pulls_url": "http://api.github.com/repos/octocat/Hello-World/pulls{/number}", | ||||||
|  |   "releases_url": "http://api.github.com/repos/octocat/Hello-World/releases{/id}", | ||||||
|  |   "ssh_url": "git@github.com:octocat/Hello-World.git", | ||||||
|  |   "stargazers_url": "http://api.github.com/repos/octocat/Hello-World/stargazers", | ||||||
|  |   "statuses_url": "http://api.github.com/repos/octocat/Hello-World/statuses/{sha}", | ||||||
|  |   "subscribers_url": "http://api.github.com/repos/octocat/Hello-World/subscribers", | ||||||
|  |   "subscription_url": "http://api.github.com/repos/octocat/Hello-World/subscription", | ||||||
|  |   "tags_url": "http://api.github.com/repos/octocat/Hello-World/tags", | ||||||
|  |   "teams_url": "http://api.github.com/repos/octocat/Hello-World/teams", | ||||||
|  |   "trees_url": "http://api.github.com/repos/octocat/Hello-World/git/trees{/sha}", | ||||||
|  |   "clone_url": "https://github.com/octocat/Hello-World.git", | ||||||
|  |   "mirror_url": "git:git.example.com/octocat/Hello-World", | ||||||
|  |   "hooks_url": "http://api.github.com/repos/octocat/Hello-World/hooks", | ||||||
|  |   "svn_url": "https://svn.github.com/octocat/Hello-World", | ||||||
|  |   "homepage": "https://github.com", | ||||||
|  |   "language": null, | ||||||
|  |   "forks_count": 9, | ||||||
|  |   "stargazers_count": 80, | ||||||
|  |   "watchers_count": 80, | ||||||
|  |   "size": 108, | ||||||
|  |   "default_branch": "master", | ||||||
|  |   "open_issues_count": 0, | ||||||
|  |   "is_template": true, | ||||||
|  |   "topics": [ | ||||||
|  |     "octocat", | ||||||
|  |     "atom", | ||||||
|  |     "electron", | ||||||
|  |     "api" | ||||||
|  |   ], | ||||||
|  |   "has_issues": true, | ||||||
|  |   "has_projects": true, | ||||||
|  |   "has_wiki": true, | ||||||
|  |   "has_pages": false, | ||||||
|  |   "has_downloads": true, | ||||||
|  |   "archived": false, | ||||||
|  |   "disabled": false, | ||||||
|  |   "visibility": "public", | ||||||
|  |   "pushed_at": "2011-01-26T19:06:43Z", | ||||||
|  |   "created_at": "2011-01-26T19:01:12Z", | ||||||
|  |   "updated_at": "2011-01-26T19:14:43Z", | ||||||
|  |   "permissions": { | ||||||
|  |     "pull": true, | ||||||
|  |     "triage": true, | ||||||
|  |     "push": false, | ||||||
|  |     "maintain": false, | ||||||
|  |     "admin": false | ||||||
|  |   }, | ||||||
|  |   "allow_rebase_merge": true, | ||||||
|  |   "template_repository": null, | ||||||
|  |   "temp_clone_token": "ABTLWHOULUVAXGTRYU7OC2876QJ2O", | ||||||
|  |   "allow_squash_merge": true, | ||||||
|  |   "delete_branch_on_merge": true, | ||||||
|  |   "allow_merge_commit": true, | ||||||
|  |   "subscribers_count": 42, | ||||||
|  |   "network_count": 0, | ||||||
|  |   "license": { | ||||||
|  |     "key": "mit", | ||||||
|  |     "name": "MIT License", | ||||||
|  |     "spdx_id": "MIT", | ||||||
|  |     "url": "https://api.github.com/licenses/mit", | ||||||
|  |     "node_id": "MDc6TGljZW5zZW1pdA==" | ||||||
|  |   }, | ||||||
|  |   "organization": { | ||||||
|  |     "login": "octocat", | ||||||
|  |     "id": 1, | ||||||
|  |     "node_id": "MDQ6VXNlcjE=", | ||||||
|  |     "avatar_url": "https://github.com/images/error/octocat_happy.gif", | ||||||
|  |     "gravatar_id": "", | ||||||
|  |     "url": "https://api.github.com/users/octocat", | ||||||
|  |     "html_url": "https://github.com/octocat", | ||||||
|  |     "followers_url": "https://api.github.com/users/octocat/followers", | ||||||
|  |     "following_url": "https://api.github.com/users/octocat/following{/other_user}", | ||||||
|  |     "gists_url": "https://api.github.com/users/octocat/gists{/gist_id}", | ||||||
|  |     "starred_url": "https://api.github.com/users/octocat/starred{/owner}{/repo}", | ||||||
|  |     "subscriptions_url": "https://api.github.com/users/octocat/subscriptions", | ||||||
|  |     "organizations_url": "https://api.github.com/users/octocat/orgs", | ||||||
|  |     "repos_url": "https://api.github.com/users/octocat/repos", | ||||||
|  |     "events_url": "https://api.github.com/users/octocat/events{/privacy}", | ||||||
|  |     "received_events_url": "https://api.github.com/users/octocat/received_events", | ||||||
|  |     "type": "Organization", | ||||||
|  |     "site_admin": false | ||||||
|  |   }, | ||||||
|  |   "parent": { | ||||||
|  |     "id": 1296269, | ||||||
|  |     "node_id": "MDEwOlJlcG9zaXRvcnkxMjk2MjY5", | ||||||
|  |     "name": "Hello-World", | ||||||
|  |     "full_name": "octocat/Hello-World", | ||||||
|  |     "owner": { | ||||||
|  |       "login": "octocat", | ||||||
|  |       "id": 1, | ||||||
|  |       "node_id": "MDQ6VXNlcjE=", | ||||||
|  |       "avatar_url": "https://github.com/images/error/octocat_happy.gif", | ||||||
|  |       "gravatar_id": "", | ||||||
|  |       "url": "https://api.github.com/users/octocat", | ||||||
|  |       "html_url": "https://github.com/octocat", | ||||||
|  |       "followers_url": "https://api.github.com/users/octocat/followers", | ||||||
|  |       "following_url": "https://api.github.com/users/octocat/following{/other_user}", | ||||||
|  |       "gists_url": "https://api.github.com/users/octocat/gists{/gist_id}", | ||||||
|  |       "starred_url": "https://api.github.com/users/octocat/starred{/owner}{/repo}", | ||||||
|  |       "subscriptions_url": "https://api.github.com/users/octocat/subscriptions", | ||||||
|  |       "organizations_url": "https://api.github.com/users/octocat/orgs", | ||||||
|  |       "repos_url": "https://api.github.com/users/octocat/repos", | ||||||
|  |       "events_url": "https://api.github.com/users/octocat/events{/privacy}", | ||||||
|  |       "received_events_url": "https://api.github.com/users/octocat/received_events", | ||||||
|  |       "type": "User", | ||||||
|  |       "site_admin": false | ||||||
|  |     }, | ||||||
|  |     "private": false, | ||||||
|  |     "html_url": "https://github.com/octocat/Hello-World", | ||||||
|  |     "description": "This your first repo!", | ||||||
|  |     "fork": false, | ||||||
|  |     "url": "https://api.github.com/repos/octocat/Hello-World", | ||||||
|  |     "archive_url": "http://api.github.com/repos/octocat/Hello-World/{archive_format}{/ref}", | ||||||
|  |     "assignees_url": "http://api.github.com/repos/octocat/Hello-World/assignees{/user}", | ||||||
|  |     "blobs_url": "http://api.github.com/repos/octocat/Hello-World/git/blobs{/sha}", | ||||||
|  |     "branches_url": "http://api.github.com/repos/octocat/Hello-World/branches{/branch}", | ||||||
|  |     "collaborators_url": "http://api.github.com/repos/octocat/Hello-World/collaborators{/collaborator}", | ||||||
|  |     "comments_url": "http://api.github.com/repos/octocat/Hello-World/comments{/number}", | ||||||
|  |     "commits_url": "http://api.github.com/repos/octocat/Hello-World/commits{/sha}", | ||||||
|  |     "compare_url": "http://api.github.com/repos/octocat/Hello-World/compare/{base}...{head}", | ||||||
|  |     "contents_url": "http://api.github.com/repos/octocat/Hello-World/contents/{+path}", | ||||||
|  |     "contributors_url": "http://api.github.com/repos/octocat/Hello-World/contributors", | ||||||
|  |     "deployments_url": "http://api.github.com/repos/octocat/Hello-World/deployments", | ||||||
|  |     "downloads_url": "http://api.github.com/repos/octocat/Hello-World/downloads", | ||||||
|  |     "events_url": "http://api.github.com/repos/octocat/Hello-World/events", | ||||||
|  |     "forks_url": "http://api.github.com/repos/octocat/Hello-World/forks", | ||||||
|  |     "git_commits_url": "http://api.github.com/repos/octocat/Hello-World/git/commits{/sha}", | ||||||
|  |     "git_refs_url": "http://api.github.com/repos/octocat/Hello-World/git/refs{/sha}", | ||||||
|  |     "git_tags_url": "http://api.github.com/repos/octocat/Hello-World/git/tags{/sha}", | ||||||
|  |     "git_url": "git:github.com/octocat/Hello-World.git", | ||||||
|  |     "issue_comment_url": "http://api.github.com/repos/octocat/Hello-World/issues/comments{/number}", | ||||||
|  |     "issue_events_url": "http://api.github.com/repos/octocat/Hello-World/issues/events{/number}", | ||||||
|  |     "issues_url": "http://api.github.com/repos/octocat/Hello-World/issues{/number}", | ||||||
|  |     "keys_url": "http://api.github.com/repos/octocat/Hello-World/keys{/key_id}", | ||||||
|  |     "labels_url": "http://api.github.com/repos/octocat/Hello-World/labels{/name}", | ||||||
|  |     "languages_url": "http://api.github.com/repos/octocat/Hello-World/languages", | ||||||
|  |     "merges_url": "http://api.github.com/repos/octocat/Hello-World/merges", | ||||||
|  |     "milestones_url": "http://api.github.com/repos/octocat/Hello-World/milestones{/number}", | ||||||
|  |     "notifications_url": "http://api.github.com/repos/octocat/Hello-World/notifications{?since,all,participating}", | ||||||
|  |     "pulls_url": "http://api.github.com/repos/octocat/Hello-World/pulls{/number}", | ||||||
|  |     "releases_url": "http://api.github.com/repos/octocat/Hello-World/releases{/id}", | ||||||
|  |     "ssh_url": "git@github.com:octocat/Hello-World.git", | ||||||
|  |     "stargazers_url": "http://api.github.com/repos/octocat/Hello-World/stargazers", | ||||||
|  |     "statuses_url": "http://api.github.com/repos/octocat/Hello-World/statuses/{sha}", | ||||||
|  |     "subscribers_url": "http://api.github.com/repos/octocat/Hello-World/subscribers", | ||||||
|  |     "subscription_url": "http://api.github.com/repos/octocat/Hello-World/subscription", | ||||||
|  |     "tags_url": "http://api.github.com/repos/octocat/Hello-World/tags", | ||||||
|  |     "teams_url": "http://api.github.com/repos/octocat/Hello-World/teams", | ||||||
|  |     "trees_url": "http://api.github.com/repos/octocat/Hello-World/git/trees{/sha}", | ||||||
|  |     "clone_url": "https://github.com/octocat/Hello-World.git", | ||||||
|  |     "mirror_url": "git:git.example.com/octocat/Hello-World", | ||||||
|  |     "hooks_url": "http://api.github.com/repos/octocat/Hello-World/hooks", | ||||||
|  |     "svn_url": "https://svn.github.com/octocat/Hello-World", | ||||||
|  |     "homepage": "https://github.com", | ||||||
|  |     "language": null, | ||||||
|  |     "forks_count": 9, | ||||||
|  |     "stargazers_count": 80, | ||||||
|  |     "watchers_count": 80, | ||||||
|  |     "size": 108, | ||||||
|  |     "default_branch": "master", | ||||||
|  |     "open_issues_count": 0, | ||||||
|  |     "is_template": true, | ||||||
|  |     "topics": [ | ||||||
|  |       "octocat", | ||||||
|  |       "atom", | ||||||
|  |       "electron", | ||||||
|  |       "api" | ||||||
|  |     ], | ||||||
|  |     "has_issues": true, | ||||||
|  |     "has_projects": true, | ||||||
|  |     "has_wiki": true, | ||||||
|  |     "has_pages": false, | ||||||
|  |     "has_downloads": true, | ||||||
|  |     "archived": false, | ||||||
|  |     "disabled": false, | ||||||
|  |     "visibility": "public", | ||||||
|  |     "pushed_at": "2011-01-26T19:06:43Z", | ||||||
|  |     "created_at": "2011-01-26T19:01:12Z", | ||||||
|  |     "updated_at": "2011-01-26T19:14:43Z", | ||||||
|  |     "permissions": { | ||||||
|  |       "admin": false, | ||||||
|  |       "push": false, | ||||||
|  |       "pull": true | ||||||
|  |     }, | ||||||
|  |     "allow_rebase_merge": true, | ||||||
|  |     "template_repository": null, | ||||||
|  |     "temp_clone_token": "ABTLWHOULUVAXGTRYU7OC2876QJ2O", | ||||||
|  |     "allow_squash_merge": true, | ||||||
|  |     "delete_branch_on_merge": true, | ||||||
|  |     "allow_merge_commit": true, | ||||||
|  |     "subscribers_count": 42, | ||||||
|  |     "network_count": 0 | ||||||
|  |   }, | ||||||
|  |   "source": { | ||||||
|  |     "id": 1296269, | ||||||
|  |     "node_id": "MDEwOlJlcG9zaXRvcnkxMjk2MjY5", | ||||||
|  |     "name": "Hello-World", | ||||||
|  |     "full_name": "octocat/Hello-World", | ||||||
|  |     "owner": { | ||||||
|  |       "login": "octocat", | ||||||
|  |       "id": 1, | ||||||
|  |       "node_id": "MDQ6VXNlcjE=", | ||||||
|  |       "avatar_url": "https://github.com/images/error/octocat_happy.gif", | ||||||
|  |       "gravatar_id": "", | ||||||
|  |       "url": "https://api.github.com/users/octocat", | ||||||
|  |       "html_url": "https://github.com/octocat", | ||||||
|  |       "followers_url": "https://api.github.com/users/octocat/followers", | ||||||
|  |       "following_url": "https://api.github.com/users/octocat/following{/other_user}", | ||||||
|  |       "gists_url": "https://api.github.com/users/octocat/gists{/gist_id}", | ||||||
|  |       "starred_url": "https://api.github.com/users/octocat/starred{/owner}{/repo}", | ||||||
|  |       "subscriptions_url": "https://api.github.com/users/octocat/subscriptions", | ||||||
|  |       "organizations_url": "https://api.github.com/users/octocat/orgs", | ||||||
|  |       "repos_url": "https://api.github.com/users/octocat/repos", | ||||||
|  |       "events_url": "https://api.github.com/users/octocat/events{/privacy}", | ||||||
|  |       "received_events_url": "https://api.github.com/users/octocat/received_events", | ||||||
|  |       "type": "User", | ||||||
|  |       "site_admin": false | ||||||
|  |     }, | ||||||
|  |     "private": false, | ||||||
|  |     "html_url": "https://github.com/octocat/Hello-World", | ||||||
|  |     "description": "This your first repo!", | ||||||
|  |     "fork": false, | ||||||
|  |     "url": "https://api.github.com/repos/octocat/Hello-World", | ||||||
|  |     "archive_url": "http://api.github.com/repos/octocat/Hello-World/{archive_format}{/ref}", | ||||||
|  |     "assignees_url": "http://api.github.com/repos/octocat/Hello-World/assignees{/user}", | ||||||
|  |     "blobs_url": "http://api.github.com/repos/octocat/Hello-World/git/blobs{/sha}", | ||||||
|  |     "branches_url": "http://api.github.com/repos/octocat/Hello-World/branches{/branch}", | ||||||
|  |     "collaborators_url": "http://api.github.com/repos/octocat/Hello-World/collaborators{/collaborator}", | ||||||
|  |     "comments_url": "http://api.github.com/repos/octocat/Hello-World/comments{/number}", | ||||||
|  |     "commits_url": "http://api.github.com/repos/octocat/Hello-World/commits{/sha}", | ||||||
|  |     "compare_url": "http://api.github.com/repos/octocat/Hello-World/compare/{base}...{head}", | ||||||
|  |     "contents_url": "http://api.github.com/repos/octocat/Hello-World/contents/{+path}", | ||||||
|  |     "contributors_url": "http://api.github.com/repos/octocat/Hello-World/contributors", | ||||||
|  |     "deployments_url": "http://api.github.com/repos/octocat/Hello-World/deployments", | ||||||
|  |     "downloads_url": "http://api.github.com/repos/octocat/Hello-World/downloads", | ||||||
|  |     "events_url": "http://api.github.com/repos/octocat/Hello-World/events", | ||||||
|  |     "forks_url": "http://api.github.com/repos/octocat/Hello-World/forks", | ||||||
|  |     "git_commits_url": "http://api.github.com/repos/octocat/Hello-World/git/commits{/sha}", | ||||||
|  |     "git_refs_url": "http://api.github.com/repos/octocat/Hello-World/git/refs{/sha}", | ||||||
|  |     "git_tags_url": "http://api.github.com/repos/octocat/Hello-World/git/tags{/sha}", | ||||||
|  |     "git_url": "git:github.com/octocat/Hello-World.git", | ||||||
|  |     "issue_comment_url": "http://api.github.com/repos/octocat/Hello-World/issues/comments{/number}", | ||||||
|  |     "issue_events_url": "http://api.github.com/repos/octocat/Hello-World/issues/events{/number}", | ||||||
|  |     "issues_url": "http://api.github.com/repos/octocat/Hello-World/issues{/number}", | ||||||
|  |     "keys_url": "http://api.github.com/repos/octocat/Hello-World/keys{/key_id}", | ||||||
|  |     "labels_url": "http://api.github.com/repos/octocat/Hello-World/labels{/name}", | ||||||
|  |     "languages_url": "http://api.github.com/repos/octocat/Hello-World/languages", | ||||||
|  |     "merges_url": "http://api.github.com/repos/octocat/Hello-World/merges", | ||||||
|  |     "milestones_url": "http://api.github.com/repos/octocat/Hello-World/milestones{/number}", | ||||||
|  |     "notifications_url": "http://api.github.com/repos/octocat/Hello-World/notifications{?since,all,participating}", | ||||||
|  |     "pulls_url": "http://api.github.com/repos/octocat/Hello-World/pulls{/number}", | ||||||
|  |     "releases_url": "http://api.github.com/repos/octocat/Hello-World/releases{/id}", | ||||||
|  |     "ssh_url": "git@github.com:octocat/Hello-World.git", | ||||||
|  |     "stargazers_url": "http://api.github.com/repos/octocat/Hello-World/stargazers", | ||||||
|  |     "statuses_url": "http://api.github.com/repos/octocat/Hello-World/statuses/{sha}", | ||||||
|  |     "subscribers_url": "http://api.github.com/repos/octocat/Hello-World/subscribers", | ||||||
|  |     "subscription_url": "http://api.github.com/repos/octocat/Hello-World/subscription", | ||||||
|  |     "tags_url": "http://api.github.com/repos/octocat/Hello-World/tags", | ||||||
|  |     "teams_url": "http://api.github.com/repos/octocat/Hello-World/teams", | ||||||
|  |     "trees_url": "http://api.github.com/repos/octocat/Hello-World/git/trees{/sha}", | ||||||
|  |     "clone_url": "https://github.com/octocat/Hello-World.git", | ||||||
|  |     "mirror_url": "git:git.example.com/octocat/Hello-World", | ||||||
|  |     "hooks_url": "http://api.github.com/repos/octocat/Hello-World/hooks", | ||||||
|  |     "svn_url": "https://svn.github.com/octocat/Hello-World", | ||||||
|  |     "homepage": "https://github.com", | ||||||
|  |     "language": null, | ||||||
|  |     "forks_count": 9, | ||||||
|  |     "stargazers_count": 80, | ||||||
|  |     "watchers_count": 80, | ||||||
|  |     "size": 108, | ||||||
|  |     "default_branch": "master", | ||||||
|  |     "open_issues_count": 0, | ||||||
|  |     "is_template": true, | ||||||
|  |     "topics": [ | ||||||
|  |       "octocat", | ||||||
|  |       "atom", | ||||||
|  |       "electron", | ||||||
|  |       "api" | ||||||
|  |     ], | ||||||
|  |     "has_issues": true, | ||||||
|  |     "has_projects": true, | ||||||
|  |     "has_wiki": true, | ||||||
|  |     "has_pages": false, | ||||||
|  |     "has_downloads": true, | ||||||
|  |     "archived": false, | ||||||
|  |     "disabled": false, | ||||||
|  |     "visibility": "public", | ||||||
|  |     "pushed_at": "2011-01-26T19:06:43Z", | ||||||
|  |     "created_at": "2011-01-26T19:01:12Z", | ||||||
|  |     "updated_at": "2011-01-26T19:14:43Z", | ||||||
|  |     "permissions": { | ||||||
|  |       "admin": false, | ||||||
|  |       "push": false, | ||||||
|  |       "pull": true | ||||||
|  |     }, | ||||||
|  |     "allow_rebase_merge": true, | ||||||
|  |     "template_repository": null, | ||||||
|  |     "temp_clone_token": "ABTLWHOULUVAXGTRYU7OC2876QJ2O", | ||||||
|  |     "allow_squash_merge": true, | ||||||
|  |     "delete_branch_on_merge": true, | ||||||
|  |     "allow_merge_commit": true, | ||||||
|  |     "subscribers_count": 42, | ||||||
|  |     "network_count": 0 | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								action.yml
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								action.yml
									
									
									
									
									
								
							| @@ -13,6 +13,9 @@ inputs: | |||||||
|   allow: |   allow: | ||||||
|     description: "List of extra privileged entitlement (e.g., network.host,security.insecure)" |     description: "List of extra privileged entitlement (e.g., network.host,security.insecure)" | ||||||
|     required: false |     required: false | ||||||
|  |   annotations: | ||||||
|  |     description: "List of annotation to set to the image" | ||||||
|  |     required: false | ||||||
|   attests: |   attests: | ||||||
|     description: "List of attestation parameters (e.g., type=sbom,generator=image)" |     description: "List of attestation parameters (e.g., type=sbom,generator=image)" | ||||||
|     required: false |     required: false | ||||||
| @@ -31,6 +34,9 @@ inputs: | |||||||
|   cache-to: |   cache-to: | ||||||
|     description: "List of cache export destinations for buildx (e.g., user/app:cache, type=local,dest=path/to/dir)" |     description: "List of cache export destinations for buildx (e.g., user/app:cache, type=local,dest=path/to/dir)" | ||||||
|     required: false |     required: false | ||||||
|  |   call: | ||||||
|  |     description: "Set method for evaluating build (e.g., check)" | ||||||
|  |     required: false | ||||||
|   cgroup-parent: |   cgroup-parent: | ||||||
|     description: "Optional parent cgroup for the container used in the build" |     description: "Optional parent cgroup for the container used in the build" | ||||||
|     required: false |     required: false | ||||||
| @@ -80,6 +86,9 @@ inputs: | |||||||
|   secrets: |   secrets: | ||||||
|     description: "List of secrets to expose to the build (e.g., key=string, GIT_AUTH_TOKEN=mytoken)" |     description: "List of secrets to expose to the build (e.g., key=string, GIT_AUTH_TOKEN=mytoken)" | ||||||
|     required: false |     required: false | ||||||
|  |   secret-envs: | ||||||
|  |     description: "List of secret env vars to expose to the build (e.g., key=envname, MY_SECRET=MY_ENV_VAR)" | ||||||
|  |     required: false | ||||||
|   secret-files: |   secret-files: | ||||||
|     description: "List of secret files to expose to the build (e.g., key=filename, MY_SECRET=./secret.txt)" |     description: "List of secret files to expose to the build (e.g., key=filename, MY_SECRET=./secret.txt)" | ||||||
|     required: false |     required: false | ||||||
| @@ -112,6 +121,6 @@ outputs: | |||||||
|     description: 'Build result metadata' |     description: 'Build result metadata' | ||||||
|  |  | ||||||
| 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.0 |  | ||||||
|  |  | ||||||
| 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-update"' | ||||||
|     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,39 +44,37 @@ 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 | ||||||
| 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 / | ||||||
|   | |||||||
							
								
								
									
										97
									
								
								dist/index.js
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										97
									
								
								dist/index.js
									
									
									
										generated
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										2
									
								
								dist/index.js.map
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								dist/index.js.map
									
									
									
										generated
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										3023
									
								
								dist/licenses.txt
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										3023
									
								
								dist/licenses.txt
									
									
									
										generated
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										2
									
								
								dist/sourcemap-register.js
									
									
									
										generated
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								dist/sourcemap-register.js
									
									
									
										generated
									
									
										vendored
									
									
								
							
										
											
												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"] | ||||||
|   | |||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Cache |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#cache) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Copy images between registries |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#copy-images-between-registries) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Update Docker Hub repo description |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#update-docker-hub-repository-description) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Export image to Docker |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#export-image-to-docker) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Isolated builders |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/configure-builder/#isolated-builders) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Local registry |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#local-registry) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Multi-platform image |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#multi-platform-images) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Named contexts |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#named-contexts) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Push to multi-registries |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#push-to-multi-registries) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Secrets |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#secrets) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Share built image between jobs |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#share-built-image-between-jobs) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Handle tags and labels |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#manage-tags-and-labels) |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Test your image before pushing it |  | ||||||
|  |  | ||||||
| This page has moved to [Docker Docs website](https://docs.docker.com/build/ci/github-actions/examples/#test-your-image-before-pushing-it) |  | ||||||
| @@ -1,8 +1,14 @@ | |||||||
|  | import fs from 'fs'; | ||||||
|  | import os from 'os'; | ||||||
|  | import path from 'path'; | ||||||
|  |  | ||||||
|  | const tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), 'docker-build-push-action-')); | ||||||
|  |  | ||||||
| process.env = Object.assign({}, process.env, { | process.env = Object.assign({}, process.env, { | ||||||
|   RUNNER_TEMP: '/tmp/github_runner', |   TEMP: tmpDir, | ||||||
|   RUNNER_TOOL_CACHE: '/tmp/github_tool_cache', |  | ||||||
|   GITHUB_REPOSITORY: 'docker/build-push-action', |   GITHUB_REPOSITORY: 'docker/build-push-action', | ||||||
|   GITHUB_RUN_ID: '123456789' |   RUNNER_TEMP: path.join(tmpDir, 'runner-temp'), | ||||||
|  |   RUNNER_TOOL_CACHE: path.join(tmpDir, 'runner-tool-cache') | ||||||
| }) as { | }) as { | ||||||
|   [key: string]: string; |   [key: string]: string; | ||||||
| }; | }; | ||||||
| @@ -11,7 +17,6 @@ module.exports = { | |||||||
|   clearMocks: false, |   clearMocks: false, | ||||||
|   testEnvironment: 'node', |   testEnvironment: 'node', | ||||||
|   moduleFileExtensions: ['js', 'ts'], |   moduleFileExtensions: ['js', 'ts'], | ||||||
|   setupFiles: ['dotenv/config'], |  | ||||||
|   testMatch: ['**/*.test.ts'], |   testMatch: ['**/*.test.ts'], | ||||||
|   transform: { |   transform: { | ||||||
|     '^.+\\.ts$': 'ts-jest' |     '^.+\\.ts$': 'ts-jest' | ||||||
| @@ -19,5 +24,7 @@ module.exports = { | |||||||
|   moduleNameMapper: { |   moduleNameMapper: { | ||||||
|     '^csv-parse/sync': '<rootDir>/node_modules/csv-parse/dist/cjs/sync.cjs' |     '^csv-parse/sync': '<rootDir>/node_modules/csv-parse/dist/cjs/sync.cjs' | ||||||
|   }, |   }, | ||||||
|  |   collectCoverageFrom: ['src/**/{!(main.ts),}.ts'], | ||||||
|  |   coveragePathIgnorePatterns: ['lib/', 'node_modules/', '__mocks__/', '__tests__/'], | ||||||
|   verbose: true |   verbose: true | ||||||
| }; | }; | ||||||
|   | |||||||
							
								
								
									
										65
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										65
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,13 +1,16 @@ | |||||||
| { | { | ||||||
|   "name": "docker-build-push", |   "name": "docker-build-push", | ||||||
|   "description": "Build and push Docker images", |   "description": "Build and push Docker images", | ||||||
|   "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", | ||||||
| @@ -19,41 +22,27 @@ | |||||||
|     "build", |     "build", | ||||||
|     "push" |     "push" | ||||||
|   ], |   ], | ||||||
|   "author": "Docker", |   "author": "Docker Inc.", | ||||||
|   "contributors": [ |  | ||||||
|     { |  | ||||||
|       "name": "CrazyMax", |  | ||||||
|       "url": "https://crazymax.dev" |  | ||||||
|     } |  | ||||||
|   ], |  | ||||||
|   "license": "Apache-2.0", |   "license": "Apache-2.0", | ||||||
|  |   "packageManager": "yarn@3.6.3", | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@actions/core": "^1.10.0", |     "@actions/core": "^1.11.1", | ||||||
|     "@actions/exec": "^1.1.1", |     "@docker/actions-toolkit": "0.62.1", | ||||||
|     "@actions/github": "^5.1.1", |     "handlebars": "^4.7.7" | ||||||
|     "csv-parse": "^5.3.3", |  | ||||||
|     "handlebars": "^4.7.7", |  | ||||||
|     "jwt-decode": "^3.1.2", |  | ||||||
|     "semver": "^7.3.7", |  | ||||||
|     "tmp": "^0.2.1" |  | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@types/csv-parse": "^1.2.2", |     "@types/node": "^20.12.12", | ||||||
|     "@types/node": "^16.11.26", |     "@typescript-eslint/eslint-plugin": "^7.9.0", | ||||||
|     "@types/semver": "^7.3.9", |     "@typescript-eslint/parser": "^7.9.0", | ||||||
|     "@types/tmp": "^0.2.3", |     "@vercel/ncc": "^0.38.1", | ||||||
|     "@typescript-eslint/eslint-plugin": "^5.14.0", |     "eslint": "^8.57.0", | ||||||
|     "@typescript-eslint/parser": "^5.14.0", |     "eslint-config-prettier": "^9.1.0", | ||||||
|     "@vercel/ncc": "^0.33.3", |     "eslint-plugin-jest": "^28.5.0", | ||||||
|     "dotenv": "^16.0.0", |     "eslint-plugin-prettier": "^5.1.3", | ||||||
|     "eslint": "^8.11.0", |     "jest": "^29.7.0", | ||||||
|     "eslint-config-prettier": "^8.5.0", |     "prettier": "^3.2.5", | ||||||
|     "eslint-plugin-jest": "^26.1.1", |     "ts-jest": "^29.1.2", | ||||||
|     "eslint-plugin-prettier": "^4.0.0", |     "ts-node": "^10.9.2", | ||||||
|     "jest": "^27.2.5", |     "typescript": "^5.4.5" | ||||||
|     "prettier": "^2.3.1", |  | ||||||
|     "ts-jest": "^27.1.2", |  | ||||||
|     "ts-node": "^10.7.0", |  | ||||||
|     "typescript": "^4.4.4" |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										282
									
								
								src/buildx.ts
									
									
									
									
									
								
							
							
						
						
									
										282
									
								
								src/buildx.ts
									
									
									
									
									
								
							| @@ -1,282 +0,0 @@ | |||||||
| import {parse} from 'csv-parse/sync'; |  | ||||||
| import fs from 'fs'; |  | ||||||
| import path from 'path'; |  | ||||||
| import * as semver from 'semver'; |  | ||||||
| import * as exec from '@actions/exec'; |  | ||||||
| import * as context from './context'; |  | ||||||
|  |  | ||||||
| export type Builder = { |  | ||||||
|   name?: string; |  | ||||||
|   driver?: string; |  | ||||||
|   nodes: Node[]; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export type Node = { |  | ||||||
|   name?: string; |  | ||||||
|   endpoint?: string; |  | ||||||
|   'driver-opts'?: Array<string>; |  | ||||||
|   status?: string; |  | ||||||
|   'buildkitd-flags'?: string; |  | ||||||
|   buildkit?: string; |  | ||||||
|   platforms?: string; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export async function getImageIDFile(): Promise<string> { |  | ||||||
|   return path.join(context.tmpDir(), 'iidfile').split(path.sep).join(path.posix.sep); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function getImageID(): Promise<string | undefined> { |  | ||||||
|   const iidFile = await getImageIDFile(); |  | ||||||
|   if (!fs.existsSync(iidFile)) { |  | ||||||
|     return undefined; |  | ||||||
|   } |  | ||||||
|   return fs.readFileSync(iidFile, {encoding: 'utf-8'}).trim(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function getMetadataFile(): Promise<string> { |  | ||||||
|   return path.join(context.tmpDir(), 'metadata-file').split(path.sep).join(path.posix.sep); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function getMetadata(): Promise<string | undefined> { |  | ||||||
|   const metadataFile = await getMetadataFile(); |  | ||||||
|   if (!fs.existsSync(metadataFile)) { |  | ||||||
|     return undefined; |  | ||||||
|   } |  | ||||||
|   const content = fs.readFileSync(metadataFile, {encoding: 'utf-8'}).trim(); |  | ||||||
|   if (content === 'null') { |  | ||||||
|     return undefined; |  | ||||||
|   } |  | ||||||
|   return content; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function getDigest(metadata: string | undefined): Promise<string | undefined> { |  | ||||||
|   if (metadata === undefined) { |  | ||||||
|     return undefined; |  | ||||||
|   } |  | ||||||
|   const metadataJSON = JSON.parse(metadata); |  | ||||||
|   if (metadataJSON['containerimage.digest']) { |  | ||||||
|     return metadataJSON['containerimage.digest']; |  | ||||||
|   } |  | ||||||
|   return undefined; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function getSecretString(kvp: string): Promise<string> { |  | ||||||
|   return getSecret(kvp, false); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function getSecretFile(kvp: string): Promise<string> { |  | ||||||
|   return getSecret(kvp, true); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function getSecret(kvp: string, file: boolean): Promise<string> { |  | ||||||
|   const delimiterIndex = kvp.indexOf('='); |  | ||||||
|   const key = kvp.substring(0, delimiterIndex); |  | ||||||
|   let value = kvp.substring(delimiterIndex + 1); |  | ||||||
|   if (key.length == 0 || value.length == 0) { |  | ||||||
|     throw new Error(`${kvp} is not a valid secret`); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (file) { |  | ||||||
|     if (!fs.existsSync(value)) { |  | ||||||
|       throw new Error(`secret file ${value} not found`); |  | ||||||
|     } |  | ||||||
|     value = fs.readFileSync(value, {encoding: 'utf-8'}); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const secretFile = context.tmpNameSync({ |  | ||||||
|     tmpdir: context.tmpDir() |  | ||||||
|   }); |  | ||||||
|   fs.writeFileSync(secretFile, value); |  | ||||||
|  |  | ||||||
|   return `id=${key},src=${secretFile}`; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isLocalOrTarExporter(outputs: string[]): boolean { |  | ||||||
|   const records = parse(outputs.join(`\n`), { |  | ||||||
|     delimiter: ',', |  | ||||||
|     trim: true, |  | ||||||
|     columns: false, |  | ||||||
|     relaxColumnCount: true |  | ||||||
|   }); |  | ||||||
|   for (const record of records) { |  | ||||||
|     // Local if no type is defined |  | ||||||
|     // https://github.com/docker/buildx/blob/d2bf42f8b4784d83fde17acb3ed84703ddc2156b/build/output.go#L29-L43 |  | ||||||
|     if (record.length == 1 && !record[0].startsWith('type=')) { |  | ||||||
|       return true; |  | ||||||
|     } |  | ||||||
|     for (const [key, value] of record.map(chunk => chunk.split('=').map(item => item.trim()))) { |  | ||||||
|       if (key == 'type' && (value == 'local' || value == 'tar')) { |  | ||||||
|         return true; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   return false; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function hasGitAuthToken(secrets: string[]): boolean { |  | ||||||
|   for (const secret of secrets) { |  | ||||||
|     if (secret.startsWith('GIT_AUTH_TOKEN=')) { |  | ||||||
|       return true; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   return false; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function isAvailable(standalone?: boolean): Promise<boolean> { |  | ||||||
|   const cmd = getCommand([], standalone); |  | ||||||
|   return await exec |  | ||||||
|     .getExecOutput(cmd.command, cmd.args, { |  | ||||||
|       ignoreReturnCode: true, |  | ||||||
|       silent: true |  | ||||||
|     }) |  | ||||||
|     .then(res => { |  | ||||||
|       if (res.stderr.length > 0 && res.exitCode != 0) { |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|       return res.exitCode == 0; |  | ||||||
|     }) |  | ||||||
|     // eslint-disable-next-line @typescript-eslint/no-unused-vars |  | ||||||
|     .catch(error => { |  | ||||||
|       return false; |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function satisfiesBuildKitVersion(builderName: string, range: string, standalone?: boolean): Promise<boolean> { |  | ||||||
|   const builderInspect = await inspect(builderName, standalone); |  | ||||||
|   for (const node of builderInspect.nodes) { |  | ||||||
|     if (!node.buildkit) { |  | ||||||
|       return false; |  | ||||||
|     } |  | ||||||
|     // BuildKit version reported by moby is in the format of `v0.11.0-moby` |  | ||||||
|     if (builderInspect.driver == 'docker' && !node.buildkit.endsWith('-moby')) { |  | ||||||
|       return false; |  | ||||||
|     } |  | ||||||
|     const version = node.buildkit.replace(/-moby$/, ''); |  | ||||||
|     if (!semver.satisfies(version, range)) { |  | ||||||
|       return false; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   return true; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function inspect(name: string, standalone?: boolean): Promise<Builder> { |  | ||||||
|   const cmd = getCommand(['inspect', name], standalone); |  | ||||||
|   return await exec |  | ||||||
|     .getExecOutput(cmd.command, cmd.args, { |  | ||||||
|       ignoreReturnCode: true, |  | ||||||
|       silent: true |  | ||||||
|     }) |  | ||||||
|     .then(res => { |  | ||||||
|       if (res.stderr.length > 0 && res.exitCode != 0) { |  | ||||||
|         throw new Error(res.stderr.trim()); |  | ||||||
|       } |  | ||||||
|       return parseInspect(res.stdout); |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function parseInspect(data: string): Promise<Builder> { |  | ||||||
|   const builder: Builder = { |  | ||||||
|     nodes: [] |  | ||||||
|   }; |  | ||||||
|   let node: Node = {}; |  | ||||||
|   for (const line of data.trim().split(`\n`)) { |  | ||||||
|     const [key, ...rest] = line.split(':'); |  | ||||||
|     const value = rest.map(v => v.trim()).join(':'); |  | ||||||
|     if (key.length == 0 || value.length == 0) { |  | ||||||
|       continue; |  | ||||||
|     } |  | ||||||
|     switch (key.toLowerCase()) { |  | ||||||
|       case 'name': { |  | ||||||
|         if (builder.name == undefined) { |  | ||||||
|           builder.name = value; |  | ||||||
|         } else { |  | ||||||
|           if (Object.keys(node).length > 0) { |  | ||||||
|             builder.nodes.push(node); |  | ||||||
|             node = {}; |  | ||||||
|           } |  | ||||||
|           node.name = value; |  | ||||||
|         } |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       case 'driver': { |  | ||||||
|         builder.driver = value; |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       case 'endpoint': { |  | ||||||
|         node.endpoint = value; |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       case 'driver options': { |  | ||||||
|         node['driver-opts'] = (value.match(/(\w+)="([^"]*)"/g) || []).map(v => v.replace(/^(.*)="(.*)"$/g, '$1=$2')); |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       case 'status': { |  | ||||||
|         node.status = value; |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       case 'flags': { |  | ||||||
|         node['buildkitd-flags'] = value; |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       case 'buildkit': { |  | ||||||
|         node.buildkit = value; |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|       case 'platforms': { |  | ||||||
|         let platforms: Array<string> = []; |  | ||||||
|         // if a preferred platform is being set then use only these |  | ||||||
|         // https://docs.docker.com/engine/reference/commandline/buildx_inspect/#get-information-about-a-builder-instance |  | ||||||
|         if (value.includes('*')) { |  | ||||||
|           for (const platform of value.split(', ')) { |  | ||||||
|             if (platform.includes('*')) { |  | ||||||
|               platforms.push(platform.replace('*', '')); |  | ||||||
|             } |  | ||||||
|           } |  | ||||||
|         } else { |  | ||||||
|           // otherwise set all platforms available |  | ||||||
|           platforms = value.split(', '); |  | ||||||
|         } |  | ||||||
|         node.platforms = platforms.join(','); |  | ||||||
|         break; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   if (Object.keys(node).length > 0) { |  | ||||||
|     builder.nodes.push(node); |  | ||||||
|   } |  | ||||||
|   return builder; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function getVersion(standalone?: boolean): Promise<string> { |  | ||||||
|   const cmd = getCommand(['version'], standalone); |  | ||||||
|   return await exec |  | ||||||
|     .getExecOutput(cmd.command, cmd.args, { |  | ||||||
|       ignoreReturnCode: true, |  | ||||||
|       silent: true |  | ||||||
|     }) |  | ||||||
|     .then(res => { |  | ||||||
|       if (res.stderr.length > 0 && res.exitCode != 0) { |  | ||||||
|         throw new Error(res.stderr.trim()); |  | ||||||
|       } |  | ||||||
|       return parseVersion(res.stdout.trim()); |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function parseVersion(stdout: string): string { |  | ||||||
|   const matches = /\sv?([0-9a-f]{7}|[0-9.]+)/.exec(stdout); |  | ||||||
|   if (!matches) { |  | ||||||
|     throw new Error(`Cannot parse buildx version`); |  | ||||||
|   } |  | ||||||
|   return matches[1]; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function satisfies(version: string, range: string): boolean { |  | ||||||
|   return semver.satisfies(version, range) || /^[0-9a-f]{7}$/.exec(version) !== null; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getCommand(args: Array<string>, standalone?: boolean) { |  | ||||||
|   return { |  | ||||||
|     command: standalone ? 'buildx' : 'docker', |  | ||||||
|     args: standalone ? args : ['buildx', ...args] |  | ||||||
|   }; |  | ||||||
| } |  | ||||||
							
								
								
									
										385
									
								
								src/context.ts
									
									
									
									
									
								
							
							
						
						
									
										385
									
								
								src/context.ts
									
									
									
									
									
								
							| @@ -1,32 +1,31 @@ | |||||||
| import * as fs from 'fs'; |  | ||||||
| import * as os from 'os'; |  | ||||||
| import * as path from 'path'; |  | ||||||
| import * as tmp from 'tmp'; |  | ||||||
| import * as buildx from './buildx'; |  | ||||||
| import * as core from '@actions/core'; | import * as core from '@actions/core'; | ||||||
| import * as github from '@actions/github'; |  | ||||||
| import {parse} from 'csv-parse/sync'; |  | ||||||
| import * as handlebars from 'handlebars'; | import * as handlebars from 'handlebars'; | ||||||
|  |  | ||||||
| let _defaultContext, _tmpDir: string; | import {Build} from '@docker/actions-toolkit/lib/buildx/build'; | ||||||
|  | import {Context} from '@docker/actions-toolkit/lib/context'; | ||||||
|  | import {GitHub} from '@docker/actions-toolkit/lib/github'; | ||||||
|  | import {Toolkit} from '@docker/actions-toolkit/lib/toolkit'; | ||||||
|  | import {Util} from '@docker/actions-toolkit/lib/util'; | ||||||
|  |  | ||||||
| export interface Inputs { | export interface Inputs { | ||||||
|   addHosts: string[]; |   'add-hosts': string[]; | ||||||
|   allow: string[]; |   allow: string[]; | ||||||
|  |   annotations: string[]; | ||||||
|   attests: string[]; |   attests: string[]; | ||||||
|   buildArgs: string[]; |   'build-args': string[]; | ||||||
|   buildContexts: string[]; |   'build-contexts': string[]; | ||||||
|   builder: string; |   builder: string; | ||||||
|   cacheFrom: string[]; |   'cache-from': string[]; | ||||||
|   cacheTo: string[]; |   'cache-to': string[]; | ||||||
|   cgroupParent: string; |   call: string; | ||||||
|  |   'cgroup-parent': string; | ||||||
|   context: string; |   context: string; | ||||||
|   file: string; |   file: string; | ||||||
|   labels: string[]; |   labels: string[]; | ||||||
|   load: boolean; |   load: boolean; | ||||||
|   network: string; |   network: string; | ||||||
|   noCache: boolean; |   'no-cache': boolean; | ||||||
|   noCacheFilters: string[]; |   'no-cache-filters': string[]; | ||||||
|   outputs: string[]; |   outputs: string[]; | ||||||
|   platforms: string[]; |   platforms: string[]; | ||||||
|   provenance: string; |   provenance: string; | ||||||
| @@ -34,193 +33,177 @@ export interface Inputs { | |||||||
|   push: boolean; |   push: boolean; | ||||||
|   sbom: string; |   sbom: string; | ||||||
|   secrets: string[]; |   secrets: string[]; | ||||||
|   secretFiles: string[]; |   'secret-envs': string[]; | ||||||
|   shmSize: string; |   'secret-files': string[]; | ||||||
|  |   'shm-size': string; | ||||||
|   ssh: string[]; |   ssh: string[]; | ||||||
|   tags: string[]; |   tags: string[]; | ||||||
|   target: string; |   target: string; | ||||||
|   ulimit: string[]; |   ulimit: string[]; | ||||||
|   githubToken: string; |   'github-token': string; | ||||||
| } | } | ||||||
|  |  | ||||||
| export function defaultContext(): string { | export async function getInputs(): Promise<Inputs> { | ||||||
|   if (!_defaultContext) { |  | ||||||
|     let ref = github.context.ref; |  | ||||||
|     if (github.context.sha && ref && !ref.startsWith('refs/')) { |  | ||||||
|       ref = `refs/heads/${github.context.ref}`; |  | ||||||
|     } |  | ||||||
|     if (github.context.sha && !ref.startsWith(`refs/pull/`)) { |  | ||||||
|       ref = github.context.sha; |  | ||||||
|     } |  | ||||||
|     _defaultContext = `${process.env.GITHUB_SERVER_URL || 'https://github.com'}/${github.context.repo.owner}/${github.context.repo.repo}.git#${ref}`; |  | ||||||
|   } |  | ||||||
|   return _defaultContext; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function tmpDir(): string { |  | ||||||
|   if (!_tmpDir) { |  | ||||||
|     _tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), 'docker-build-push-')).split(path.sep).join(path.posix.sep); |  | ||||||
|   } |  | ||||||
|   return _tmpDir; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function tmpNameSync(options?: tmp.TmpNameOptions): string { |  | ||||||
|   return tmp.tmpNameSync(options); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function provenanceBuilderID(): string { |  | ||||||
|   return `${process.env.GITHUB_SERVER_URL || 'https://github.com'}/${github.context.repo.owner}/${github.context.repo.repo}/actions/runs/${github.context.runId}`; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function getInputs(defaultContext: string): Promise<Inputs> { |  | ||||||
|   return { |   return { | ||||||
|     addHosts: await getInputList('add-hosts'), |     'add-hosts': Util.getInputList('add-hosts'), | ||||||
|     allow: await getInputList('allow'), |     allow: Util.getInputList('allow'), | ||||||
|     attests: await getInputList('attests', true), |     annotations: Util.getInputList('annotations', {ignoreComma: true}), | ||||||
|     buildArgs: await getInputList('build-args', true), |     attests: Util.getInputList('attests', {ignoreComma: true}), | ||||||
|     buildContexts: await getInputList('build-contexts', true), |     'build-args': Util.getInputList('build-args', {ignoreComma: true}), | ||||||
|  |     'build-contexts': Util.getInputList('build-contexts', {ignoreComma: true}), | ||||||
|     builder: core.getInput('builder'), |     builder: core.getInput('builder'), | ||||||
|     cacheFrom: await getInputList('cache-from', true), |     'cache-from': Util.getInputList('cache-from', {ignoreComma: true}), | ||||||
|     cacheTo: await getInputList('cache-to', true), |     'cache-to': Util.getInputList('cache-to', {ignoreComma: true}), | ||||||
|     cgroupParent: core.getInput('cgroup-parent'), |     call: core.getInput('call'), | ||||||
|     context: core.getInput('context') || defaultContext, |     'cgroup-parent': core.getInput('cgroup-parent'), | ||||||
|  |     context: core.getInput('context') || Context.gitContext(), | ||||||
|     file: core.getInput('file'), |     file: core.getInput('file'), | ||||||
|     labels: await getInputList('labels', true), |     labels: Util.getInputList('labels', {ignoreComma: true}), | ||||||
|     load: core.getBooleanInput('load'), |     load: core.getBooleanInput('load'), | ||||||
|     network: core.getInput('network'), |     network: core.getInput('network'), | ||||||
|     noCache: core.getBooleanInput('no-cache'), |     'no-cache': core.getBooleanInput('no-cache'), | ||||||
|     noCacheFilters: await getInputList('no-cache-filters'), |     'no-cache-filters': Util.getInputList('no-cache-filters'), | ||||||
|     outputs: await getInputList('outputs', true), |     outputs: Util.getInputList('outputs', {ignoreComma: true, quote: false}), | ||||||
|     platforms: await getInputList('platforms'), |     platforms: Util.getInputList('platforms'), | ||||||
|     provenance: getProvenanceInput('provenance'), |     provenance: Build.getProvenanceInput('provenance'), | ||||||
|     pull: core.getBooleanInput('pull'), |     pull: core.getBooleanInput('pull'), | ||||||
|     push: core.getBooleanInput('push'), |     push: core.getBooleanInput('push'), | ||||||
|     sbom: core.getInput('sbom'), |     sbom: core.getInput('sbom'), | ||||||
|     secrets: await getInputList('secrets', true), |     secrets: Util.getInputList('secrets', {ignoreComma: true}), | ||||||
|     secretFiles: await getInputList('secret-files', true), |     'secret-envs': Util.getInputList('secret-envs'), | ||||||
|     shmSize: core.getInput('shm-size'), |     'secret-files': Util.getInputList('secret-files', {ignoreComma: true}), | ||||||
|     ssh: await getInputList('ssh'), |     'shm-size': core.getInput('shm-size'), | ||||||
|     tags: await getInputList('tags'), |     ssh: Util.getInputList('ssh'), | ||||||
|  |     tags: Util.getInputList('tags'), | ||||||
|     target: core.getInput('target'), |     target: core.getInput('target'), | ||||||
|     ulimit: await getInputList('ulimit', true), |     ulimit: Util.getInputList('ulimit', {ignoreComma: true}), | ||||||
|     githubToken: core.getInput('github-token') |     'github-token': core.getInput('github-token') | ||||||
|   }; |   }; | ||||||
| } | } | ||||||
|  |  | ||||||
| export async function getArgs(inputs: Inputs, defaultContext: string, buildxVersion: string, standalone?: boolean): Promise<Array<string>> { | export async function getArgs(inputs: Inputs, toolkit: Toolkit): Promise<Array<string>> { | ||||||
|   const context = handlebars.compile(inputs.context)({defaultContext}); |   const context = handlebars.compile(inputs.context)({ | ||||||
|  |     defaultContext: Context.gitContext() | ||||||
|  |   }); | ||||||
|   // prettier-ignore |   // prettier-ignore | ||||||
|   return [ |   return [ | ||||||
|     ...await getBuildArgs(inputs, defaultContext, context, buildxVersion, standalone), |     ...await getBuildArgs(inputs, context, toolkit), | ||||||
|     ...await getCommonArgs(inputs, buildxVersion), |     ...await getCommonArgs(inputs, toolkit), | ||||||
|     context |     context | ||||||
|   ]; |   ]; | ||||||
| } | } | ||||||
|  |  | ||||||
| async function getBuildArgs(inputs: Inputs, defaultContext: string, context: string, buildxVersion: string, standalone?: boolean): Promise<Array<string>> { | async function getBuildArgs(inputs: Inputs, context: string, toolkit: Toolkit): Promise<Array<string>> { | ||||||
|   const args: Array<string> = ['build']; |   const args: Array<string> = ['build']; | ||||||
|   await asyncForEach(inputs.addHosts, async addHost => { |   await Util.asyncForEach(inputs['add-hosts'], async addHost => { | ||||||
|     args.push('--add-host', addHost); |     args.push('--add-host', addHost); | ||||||
|   }); |   }); | ||||||
|   if (inputs.allow.length > 0) { |   await Util.asyncForEach(inputs.allow, async allow => { | ||||||
|     args.push('--allow', inputs.allow.join(',')); |     args.push('--allow', allow); | ||||||
|   } |  | ||||||
|   if (buildx.satisfies(buildxVersion, '>=0.10.0')) { |  | ||||||
|     await asyncForEach(inputs.attests, async attest => { |  | ||||||
|       args.push('--attest', attest); |  | ||||||
|   }); |   }); | ||||||
|  |   if (await toolkit.buildx.versionSatisfies('>=0.12.0')) { | ||||||
|  |     await Util.asyncForEach(inputs.annotations, async annotation => { | ||||||
|  |       args.push('--annotation', annotation); | ||||||
|  |     }); | ||||||
|  |   } else if (inputs.annotations.length > 0) { | ||||||
|  |     core.warning("Annotations are only supported by buildx >= 0.12.0; the input 'annotations' is ignored."); | ||||||
|   } |   } | ||||||
|   await asyncForEach(inputs.buildArgs, async buildArg => { |   await Util.asyncForEach(inputs['build-args'], async buildArg => { | ||||||
|     args.push('--build-arg', buildArg); |     args.push('--build-arg', buildArg); | ||||||
|   }); |   }); | ||||||
|   if (buildx.satisfies(buildxVersion, '>=0.8.0')) { |   if (await toolkit.buildx.versionSatisfies('>=0.8.0')) { | ||||||
|     await asyncForEach(inputs.buildContexts, async buildContext => { |     await Util.asyncForEach(inputs['build-contexts'], async buildContext => { | ||||||
|       args.push('--build-context', buildContext); |       args.push( | ||||||
|  |         '--build-context', | ||||||
|  |         handlebars.compile(buildContext)({ | ||||||
|  |           defaultContext: Context.gitContext() | ||||||
|  |         }) | ||||||
|  |       ); | ||||||
|     }); |     }); | ||||||
|  |   } else if (inputs['build-contexts'].length > 0) { | ||||||
|  |     core.warning("Build contexts are only supported by buildx >= 0.8.0; the input 'build-contexts' is ignored."); | ||||||
|   } |   } | ||||||
|   await asyncForEach(inputs.cacheFrom, async cacheFrom => { |   await Util.asyncForEach(inputs['cache-from'], async cacheFrom => { | ||||||
|     args.push('--cache-from', cacheFrom); |     args.push('--cache-from', cacheFrom); | ||||||
|   }); |   }); | ||||||
|   await asyncForEach(inputs.cacheTo, async cacheTo => { |   await Util.asyncForEach(inputs['cache-to'], async cacheTo => { | ||||||
|     args.push('--cache-to', cacheTo); |     args.push('--cache-to', cacheTo); | ||||||
|   }); |   }); | ||||||
|   if (inputs.cgroupParent) { |   if (inputs.call) { | ||||||
|     args.push('--cgroup-parent', inputs.cgroupParent); |     if (!(await toolkit.buildx.versionSatisfies('>=0.15.0'))) { | ||||||
|  |       throw new Error(`Buildx >= 0.15.0 is required to use the call flag.`); | ||||||
|     } |     } | ||||||
|  |     args.push('--call', inputs.call); | ||||||
|  |   } | ||||||
|  |   if (inputs['cgroup-parent']) { | ||||||
|  |     args.push('--cgroup-parent', inputs['cgroup-parent']); | ||||||
|  |   } | ||||||
|  |   await Util.asyncForEach(inputs['secret-envs'], async secretEnv => { | ||||||
|  |     try { | ||||||
|  |       args.push('--secret', Build.resolveSecretEnv(secretEnv)); | ||||||
|  |     } catch (err) { | ||||||
|  |       core.warning(err.message); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|   if (inputs.file) { |   if (inputs.file) { | ||||||
|     args.push('--file', inputs.file); |     args.push('--file', inputs.file); | ||||||
|   } |   } | ||||||
|   if (!buildx.isLocalOrTarExporter(inputs.outputs) && (inputs.platforms.length == 0 || buildx.satisfies(buildxVersion, '>=0.4.2'))) { |   if (!Build.hasLocalExporter(inputs.outputs) && !Build.hasTarExporter(inputs.outputs) && (inputs.platforms.length == 0 || (await toolkit.buildx.versionSatisfies('>=0.4.2')))) { | ||||||
|     args.push('--iidfile', await buildx.getImageIDFile()); |     args.push('--iidfile', toolkit.buildxBuild.getImageIDFilePath()); | ||||||
|   } |   } | ||||||
|   await asyncForEach(inputs.labels, async label => { |   await Util.asyncForEach(inputs.labels, async label => { | ||||||
|     args.push('--label', label); |     args.push('--label', label); | ||||||
|   }); |   }); | ||||||
|   await asyncForEach(inputs.noCacheFilters, async noCacheFilter => { |   await Util.asyncForEach(inputs['no-cache-filters'], async noCacheFilter => { | ||||||
|     args.push('--no-cache-filter', noCacheFilter); |     args.push('--no-cache-filter', noCacheFilter); | ||||||
|   }); |   }); | ||||||
|   await asyncForEach(inputs.outputs, async output => { |   await Util.asyncForEach(inputs.outputs, async output => { | ||||||
|     args.push('--output', output); |     args.push('--output', output); | ||||||
|   }); |   }); | ||||||
|   if (inputs.platforms.length > 0) { |   if (inputs.platforms.length > 0) { | ||||||
|     args.push('--platform', inputs.platforms.join(',')); |     args.push('--platform', inputs.platforms.join(',')); | ||||||
|   } |   } | ||||||
|   if (buildx.satisfies(buildxVersion, '>=0.10.0')) { |   if (await toolkit.buildx.versionSatisfies('>=0.10.0')) { | ||||||
|     if (inputs.provenance) { |     args.push(...(await getAttestArgs(inputs, toolkit))); | ||||||
|       args.push('--provenance', inputs.provenance); |  | ||||||
|     } else if ((await buildx.satisfiesBuildKitVersion(inputs.builder, '>=0.11.0', standalone)) && !hasDockerExport(inputs)) { |  | ||||||
|       // if provenance not specified and BuildKit version compatible for |  | ||||||
|       // attestation, set default provenance. Also needs to make sure user |  | ||||||
|       // doesn't want to explicitly load the image to docker. |  | ||||||
|       if (fromPayload('repository.private') !== false) { |  | ||||||
|         // if this is a private repository, we set the default provenance |  | ||||||
|         // attributes being set in buildx: https://github.com/docker/buildx/blob/fb27e3f919dcbf614d7126b10c2bc2d0b1927eb6/build/build.go#L603 |  | ||||||
|         args.push('--provenance', getProvenanceAttrs(`mode=min,inline-only=true`)); |  | ||||||
|   } else { |   } else { | ||||||
|         // for a public repository, we set max provenance mode. |     core.warning("Attestations are only supported by buildx >= 0.10.0; the inputs 'attests', 'provenance' and 'sbom' are ignored."); | ||||||
|         args.push('--provenance', getProvenanceAttrs(`mode=max`)); |  | ||||||
|   } |   } | ||||||
|     } |   await Util.asyncForEach(inputs.secrets, async secret => { | ||||||
|     if (inputs.sbom) { |  | ||||||
|       args.push('--sbom', inputs.sbom); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   await asyncForEach(inputs.secrets, async secret => { |  | ||||||
|     try { |     try { | ||||||
|       args.push('--secret', await buildx.getSecretString(secret)); |       args.push('--secret', Build.resolveSecretString(secret)); | ||||||
|     } catch (err) { |     } catch (err) { | ||||||
|       core.warning(err.message); |       core.warning(err.message); | ||||||
|     } |     } | ||||||
|   }); |   }); | ||||||
|   await asyncForEach(inputs.secretFiles, async secretFile => { |   await Util.asyncForEach(inputs['secret-files'], async secretFile => { | ||||||
|     try { |     try { | ||||||
|       args.push('--secret', await buildx.getSecretFile(secretFile)); |       args.push('--secret', Build.resolveSecretFile(secretFile)); | ||||||
|     } catch (err) { |     } catch (err) { | ||||||
|       core.warning(err.message); |       core.warning(err.message); | ||||||
|     } |     } | ||||||
|   }); |   }); | ||||||
|   if (inputs.githubToken && !buildx.hasGitAuthToken(inputs.secrets) && context.startsWith(defaultContext)) { |   if (inputs['github-token'] && !Build.hasGitAuthTokenSecret(inputs.secrets) && context.startsWith(Context.gitContext())) { | ||||||
|     args.push('--secret', await buildx.getSecretString(`GIT_AUTH_TOKEN=${inputs.githubToken}`)); |     args.push('--secret', Build.resolveSecretString(`GIT_AUTH_TOKEN=${inputs['github-token']}`)); | ||||||
|   } |   } | ||||||
|   if (inputs.shmSize) { |   if (inputs['shm-size']) { | ||||||
|     args.push('--shm-size', inputs.shmSize); |     args.push('--shm-size', inputs['shm-size']); | ||||||
|   } |   } | ||||||
|   await asyncForEach(inputs.ssh, async ssh => { |   await Util.asyncForEach(inputs.ssh, async ssh => { | ||||||
|     args.push('--ssh', ssh); |     args.push('--ssh', ssh); | ||||||
|   }); |   }); | ||||||
|   await asyncForEach(inputs.tags, async tag => { |   await Util.asyncForEach(inputs.tags, async tag => { | ||||||
|     args.push('--tag', tag); |     args.push('--tag', tag); | ||||||
|   }); |   }); | ||||||
|   if (inputs.target) { |   if (inputs.target) { | ||||||
|     args.push('--target', inputs.target); |     args.push('--target', inputs.target); | ||||||
|   } |   } | ||||||
|   await asyncForEach(inputs.ulimit, async ulimit => { |   await Util.asyncForEach(inputs.ulimit, async ulimit => { | ||||||
|     args.push('--ulimit', ulimit); |     args.push('--ulimit', ulimit); | ||||||
|   }); |   }); | ||||||
|   return args; |   return args; | ||||||
| } | } | ||||||
|  |  | ||||||
| async function getCommonArgs(inputs: Inputs, buildxVersion: string): Promise<Array<string>> { | async function getCommonArgs(inputs: Inputs, toolkit: Toolkit): Promise<Array<string>> { | ||||||
|   const args: Array<string> = []; |   const args: Array<string> = []; | ||||||
|   if (inputs.builder) { |   if (inputs.builder) { | ||||||
|     args.push('--builder', inputs.builder); |     args.push('--builder', inputs.builder); | ||||||
| @@ -228,13 +211,13 @@ async function getCommonArgs(inputs: Inputs, buildxVersion: string): Promise<Arr | |||||||
|   if (inputs.load) { |   if (inputs.load) { | ||||||
|     args.push('--load'); |     args.push('--load'); | ||||||
|   } |   } | ||||||
|   if (buildx.satisfies(buildxVersion, '>=0.6.0')) { |   if (await toolkit.buildx.versionSatisfies('>=0.6.0')) { | ||||||
|     args.push('--metadata-file', await buildx.getMetadataFile()); |     args.push('--metadata-file', toolkit.buildxBuild.getMetadataFilePath()); | ||||||
|   } |   } | ||||||
|   if (inputs.network) { |   if (inputs.network) { | ||||||
|     args.push('--network', inputs.network); |     args.push('--network', inputs.network); | ||||||
|   } |   } | ||||||
|   if (inputs.noCache) { |   if (inputs['no-cache']) { | ||||||
|     args.push('--no-cache'); |     args.push('--no-cache'); | ||||||
|   } |   } | ||||||
|   if (inputs.pull) { |   if (inputs.pull) { | ||||||
| @@ -246,116 +229,58 @@ async function getCommonArgs(inputs: Inputs, buildxVersion: string): Promise<Arr | |||||||
|   return args; |   return args; | ||||||
| } | } | ||||||
|  |  | ||||||
| export async function getInputList(name: string, ignoreComma?: boolean): Promise<string[]> { | async function getAttestArgs(inputs: Inputs, toolkit: Toolkit): Promise<Array<string>> { | ||||||
|   const res: Array<string> = []; |   const args: Array<string> = []; | ||||||
|  |  | ||||||
|   const items = core.getInput(name); |   // check if provenance attestation is set in attests input | ||||||
|   if (items == '') { |   let hasAttestProvenance = false; | ||||||
|     return res; |   await Util.asyncForEach(inputs.attests, async (attest: string) => { | ||||||
|  |     if (Build.hasAttestationType('provenance', attest)) { | ||||||
|  |       hasAttestProvenance = true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   const records = await parse(items, { |  | ||||||
|     columns: false, |  | ||||||
|     relaxQuotes: true, |  | ||||||
|     relaxColumnCount: true, |  | ||||||
|     skipEmptyLines: true |  | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   for (const record of records as Array<string[]>) { |   let provenanceSet = false; | ||||||
|     if (record.length == 1) { |   let sbomSet = false; | ||||||
|       res.push(record[0]); |   if (inputs.provenance) { | ||||||
|       continue; |     args.push('--attest', Build.resolveAttestationAttrs(`type=provenance,${inputs.provenance}`)); | ||||||
|     } else if (!ignoreComma) { |     provenanceSet = true; | ||||||
|       res.push(...record); |   } else if (!hasAttestProvenance && !noDefaultAttestations() && (await toolkit.buildkit.versionSatisfies(inputs.builder, '>=0.11.0')) && !Build.hasDockerExporter(inputs.outputs, inputs.load)) { | ||||||
|       continue; |     // if provenance not specified in provenance or attests inputs and BuildKit | ||||||
|  |     // version compatible for attestation, set default provenance. Also needs | ||||||
|  |     // to make sure user doesn't want to explicitly load the image to docker. | ||||||
|  |     if (GitHub.context.payload.repository?.private ?? false) { | ||||||
|  |       // if this is a private repository, we set the default provenance | ||||||
|  |       // attributes being set in buildx: https://github.com/docker/buildx/blob/fb27e3f919dcbf614d7126b10c2bc2d0b1927eb6/build/build.go#L603 | ||||||
|  |       args.push('--attest', `type=provenance,${Build.resolveProvenanceAttrs(`mode=min,inline-only=true`)}`); | ||||||
|  |     } else { | ||||||
|  |       // for a public repository, we set max provenance mode. | ||||||
|  |       args.push('--attest', `type=provenance,${Build.resolveProvenanceAttrs(`mode=max`)}`); | ||||||
|     } |     } | ||||||
|     res.push(record.join(',')); |   } | ||||||
|  |   if (inputs.sbom) { | ||||||
|  |     args.push('--attest', Build.resolveAttestationAttrs(`type=sbom,${inputs.sbom}`)); | ||||||
|  |     sbomSet = true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return res.filter(item => item).map(pat => pat.trim()); |   // set attests but check if provenance or sbom types already set as | ||||||
|  |   // provenance and sbom inputs take precedence over attests input. | ||||||
|  |   await Util.asyncForEach(inputs.attests, async (attest: string) => { | ||||||
|  |     if (!Build.hasAttestationType('provenance', attest) && !Build.hasAttestationType('sbom', attest)) { | ||||||
|  |       args.push('--attest', Build.resolveAttestationAttrs(attest)); | ||||||
|  |     } else if (!provenanceSet && Build.hasAttestationType('provenance', attest)) { | ||||||
|  |       args.push('--attest', Build.resolveProvenanceAttrs(attest)); | ||||||
|  |     } else if (!sbomSet && Build.hasAttestationType('sbom', attest)) { | ||||||
|  |       args.push('--attest', attest); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   return args; | ||||||
| } | } | ||||||
|  |  | ||||||
| export const asyncForEach = async (array, callback) => { | function noDefaultAttestations(): boolean { | ||||||
|   for (let index = 0; index < array.length; index++) { |   if (process.env.BUILDX_NO_DEFAULT_ATTESTATIONS) { | ||||||
|     await callback(array[index], index, array); |     return Util.parseBool(process.env.BUILDX_NO_DEFAULT_ATTESTATIONS); | ||||||
|   } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| // eslint-disable-next-line @typescript-eslint/no-explicit-any |  | ||||||
| function fromPayload(path: string): any { |  | ||||||
|   return select(github.context.payload, path); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // eslint-disable-next-line @typescript-eslint/no-explicit-any |  | ||||||
| function select(obj: any, path: string): any { |  | ||||||
|   if (!obj) { |  | ||||||
|     return undefined; |  | ||||||
|   } |  | ||||||
|   const i = path.indexOf('.'); |  | ||||||
|   if (i < 0) { |  | ||||||
|     return obj[path]; |  | ||||||
|   } |  | ||||||
|   const key = path.slice(0, i); |  | ||||||
|   return select(obj[key], path.slice(i + 1)); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getProvenanceInput(name: string): string { |  | ||||||
|   const input = core.getInput(name); |  | ||||||
|   if (!input) { |  | ||||||
|     // if input is not set, default values will be set later. |  | ||||||
|     return input; |  | ||||||
|   } |  | ||||||
|   const builderID = provenanceBuilderID(); |  | ||||||
|   try { |  | ||||||
|     return core.getBooleanInput(name) ? `builder-id=${builderID}` : 'false'; |  | ||||||
|   } catch (err) { |  | ||||||
|     // not a valid boolean, so we assume it's a string |  | ||||||
|     return getProvenanceAttrs(input); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getProvenanceAttrs(input: string): string { |  | ||||||
|   const builderID = provenanceBuilderID(); |  | ||||||
|   // parse attributes from input |  | ||||||
|   const fields = parse(input, { |  | ||||||
|     relaxColumnCount: true, |  | ||||||
|     skipEmptyLines: true |  | ||||||
|   })[0]; |  | ||||||
|   // check if builder-id attribute exists in the input |  | ||||||
|   for (const field of fields) { |  | ||||||
|     const parts = field |  | ||||||
|       .toString() |  | ||||||
|       .split(/(?<=^[^=]+?)=/) |  | ||||||
|       .map(item => item.trim()); |  | ||||||
|     if (parts[0] == 'builder-id') { |  | ||||||
|       return input; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   // if not add builder-id attribute |  | ||||||
|   return `${input},builder-id=${builderID}`; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function hasDockerExport(inputs: Inputs): boolean { |  | ||||||
|   if (inputs.load) { |  | ||||||
|     return true; |  | ||||||
|   } |  | ||||||
|   for (const output of inputs.outputs) { |  | ||||||
|     const fields = parse(output, { |  | ||||||
|       relaxColumnCount: true, |  | ||||||
|       skipEmptyLines: true |  | ||||||
|     })[0]; |  | ||||||
|     for (const field of fields) { |  | ||||||
|       const parts = field |  | ||||||
|         .toString() |  | ||||||
|         .split(/(?<=^[^=]+?)=/) |  | ||||||
|         .map(item => item.trim()); |  | ||||||
|       if (parts.length != 2) { |  | ||||||
|         continue; |  | ||||||
|       } |  | ||||||
|       if (parts[0] == 'type' && parts[1] == 'docker') { |  | ||||||
|         return true; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|   return false; |   return false; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,19 +0,0 @@ | |||||||
| import * as exec from '@actions/exec'; |  | ||||||
|  |  | ||||||
| export async function isAvailable(): Promise<boolean> { |  | ||||||
|   return await exec |  | ||||||
|     .getExecOutput('docker', undefined, { |  | ||||||
|       ignoreReturnCode: true, |  | ||||||
|       silent: true |  | ||||||
|     }) |  | ||||||
|     .then(res => { |  | ||||||
|       if (res.stderr.length > 0 && res.exitCode != 0) { |  | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|       return res.exitCode == 0; |  | ||||||
|     }) |  | ||||||
|     // eslint-disable-next-line @typescript-eslint/no-unused-vars |  | ||||||
|     .catch(error => { |  | ||||||
|       return false; |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| import jwt_decode, {JwtPayload} from 'jwt-decode'; |  | ||||||
|  |  | ||||||
| interface Jwt extends JwtPayload { |  | ||||||
|   ac?: string; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export const parseRuntimeToken = (token: string): Jwt => { |  | ||||||
|   return jwt_decode<Jwt>(token); |  | ||||||
| }; |  | ||||||
							
								
								
									
										325
									
								
								src/main.ts
									
									
									
									
									
								
							
							
						
						
									
										325
									
								
								src/main.ts
									
									
									
									
									
								
							| @@ -1,72 +1,125 @@ | |||||||
| import * as fs from 'fs'; | import * as fs from 'fs'; | ||||||
| import * as buildx from './buildx'; | import * as path from 'path'; | ||||||
| import * as context from './context'; |  | ||||||
| import * as docker from './docker'; |  | ||||||
| import * as github from './github'; |  | ||||||
| import * as stateHelper from './state-helper'; | import * as stateHelper from './state-helper'; | ||||||
| import * as core from '@actions/core'; | import * as core from '@actions/core'; | ||||||
| import * as exec from '@actions/exec'; | import * as actionsToolkit from '@docker/actions-toolkit'; | ||||||
|  |  | ||||||
| async function run(): Promise<void> { | import {Buildx} from '@docker/actions-toolkit/lib/buildx/buildx'; | ||||||
|  | import {History as BuildxHistory} from '@docker/actions-toolkit/lib/buildx/history'; | ||||||
|  | import {Context} from '@docker/actions-toolkit/lib/context'; | ||||||
|  | import {Docker} from '@docker/actions-toolkit/lib/docker/docker'; | ||||||
|  | import {Exec} from '@docker/actions-toolkit/lib/exec'; | ||||||
|  | import {GitHub} from '@docker/actions-toolkit/lib/github'; | ||||||
|  | import {Toolkit} from '@docker/actions-toolkit/lib/toolkit'; | ||||||
|  | import {Util} from '@docker/actions-toolkit/lib/util'; | ||||||
|  |  | ||||||
|  | import {BuilderInfo} from '@docker/actions-toolkit/lib/types/buildx/builder'; | ||||||
|  | import {ConfigFile} from '@docker/actions-toolkit/lib/types/docker/docker'; | ||||||
|  | import {UploadArtifactResponse} from '@docker/actions-toolkit/lib/types/github'; | ||||||
|  |  | ||||||
|  | import * as context from './context'; | ||||||
|  |  | ||||||
|  | actionsToolkit.run( | ||||||
|  |   // main | ||||||
|  |   async () => { | ||||||
|  |     const startedTime = new Date(); | ||||||
|  |     const inputs: context.Inputs = await context.getInputs(); | ||||||
|  |     stateHelper.setSummaryInputs(inputs); | ||||||
|  |     core.debug(`inputs: ${JSON.stringify(inputs)}`); | ||||||
|  |  | ||||||
|  |     const toolkit = new Toolkit(); | ||||||
|  |  | ||||||
|  |     await core.group(`GitHub Actions runtime token ACs`, async () => { | ||||||
|       try { |       try { | ||||||
|     const defContext = context.defaultContext(); |         await GitHub.printActionsRuntimeTokenACs(); | ||||||
|     const inputs: context.Inputs = await context.getInputs(defContext); |       } catch (e) { | ||||||
|  |         core.warning(e.message); | ||||||
|     // standalone if docker cli not available |  | ||||||
|     const standalone = !(await docker.isAvailable()); |  | ||||||
|  |  | ||||||
|     await core.group(`GitHub Actions runtime token access controls`, async () => { |  | ||||||
|       const actionsRuntimeToken = process.env['ACTIONS_RUNTIME_TOKEN']; |  | ||||||
|       if (actionsRuntimeToken) { |  | ||||||
|         core.info(JSON.stringify(JSON.parse(github.parseRuntimeToken(actionsRuntimeToken).ac as string), undefined, 2)); |  | ||||||
|       } else { |  | ||||||
|         core.info(`ACTIONS_RUNTIME_TOKEN not set`); |  | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     core.startGroup(`Docker info`); |     await core.group(`Docker info`, async () => { | ||||||
|     if (standalone) { |       try { | ||||||
|       core.info(`Docker info skipped in standalone mode`); |         await Docker.printVersion(); | ||||||
|     } else { |         await Docker.printInfo(); | ||||||
|       await exec.exec('docker', ['version'], { |       } catch (e) { | ||||||
|         failOnStdErr: false |         core.info(e.message); | ||||||
|       }); |  | ||||||
|       await exec.exec('docker', ['info'], { |  | ||||||
|         failOnStdErr: false |  | ||||||
|       }); |  | ||||||
|       } |       } | ||||||
|     core.endGroup(); |     }); | ||||||
|  |  | ||||||
|     if (!(await buildx.isAvailable(standalone))) { |     await core.group(`Proxy configuration`, async () => { | ||||||
|  |       let dockerConfig: ConfigFile | undefined; | ||||||
|  |       let dockerConfigMalformed = false; | ||||||
|  |       try { | ||||||
|  |         dockerConfig = await Docker.configFile(); | ||||||
|  |       } catch (e) { | ||||||
|  |         dockerConfigMalformed = true; | ||||||
|  |         core.warning(`Unable to parse config file ${path.join(Docker.configDir, 'config.json')}: ${e}`); | ||||||
|  |       } | ||||||
|  |       if (dockerConfig && dockerConfig.proxies) { | ||||||
|  |         for (const host in dockerConfig.proxies) { | ||||||
|  |           let prefix = ''; | ||||||
|  |           if (Object.keys(dockerConfig.proxies).length > 1) { | ||||||
|  |             prefix = '  '; | ||||||
|  |             core.info(host); | ||||||
|  |           } | ||||||
|  |           for (const key in dockerConfig.proxies[host]) { | ||||||
|  |             core.info(`${prefix}${key}: ${dockerConfig.proxies[host][key]}`); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } else if (!dockerConfigMalformed) { | ||||||
|  |         core.info('No proxy configuration found'); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     if (!(await toolkit.buildx.isAvailable())) { | ||||||
|       core.setFailed(`Docker buildx is required. See https://github.com/docker/setup-buildx-action to set up buildx.`); |       core.setFailed(`Docker buildx is required. See https://github.com/docker/setup-buildx-action to set up buildx.`); | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|     stateHelper.setTmpDir(context.tmpDir()); |  | ||||||
|  |  | ||||||
|     const buildxVersion = await buildx.getVersion(standalone); |     stateHelper.setTmpDir(Context.tmpDir()); | ||||||
|  |  | ||||||
|     await core.group(`Buildx version`, async () => { |     await core.group(`Buildx version`, async () => { | ||||||
|       const versionCmd = buildx.getCommand(['version'], standalone); |       await toolkit.buildx.printVersion(); | ||||||
|       await exec.exec(versionCmd.command, versionCmd.args, { |  | ||||||
|         failOnStdErr: false |  | ||||||
|       }); |  | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     const args: string[] = await context.getArgs(inputs, defContext, buildxVersion, standalone); |     let builder: BuilderInfo; | ||||||
|     const buildCmd = buildx.getCommand(args, standalone); |     await core.group(`Builder info`, async () => { | ||||||
|     await exec |       builder = await toolkit.builder.inspect(inputs.builder); | ||||||
|       .getExecOutput(buildCmd.command, buildCmd.args, { |       stateHelper.setBuilderDriver(builder.driver ?? ''); | ||||||
|         ignoreReturnCode: true |       stateHelper.setBuilderEndpoint(builder.nodes?.[0]?.endpoint ?? ''); | ||||||
|       }) |       core.info(JSON.stringify(builder, null, 2)); | ||||||
|       .then(res => { |     }); | ||||||
|         if (res.stderr.length > 0 && res.exitCode != 0) { |  | ||||||
|           throw new Error(`buildx failed with: ${res.stderr.match(/(.*)\s*$/)?.[0]?.trim() ?? 'unknown error'}`); |     const args: string[] = await context.getArgs(inputs, toolkit); | ||||||
|  |     core.debug(`context.getArgs: ${JSON.stringify(args)}`); | ||||||
|  |  | ||||||
|  |     const buildCmd = await toolkit.buildx.getCommand(args); | ||||||
|  |     core.debug(`buildCmd.command: ${buildCmd.command}`); | ||||||
|  |     core.debug(`buildCmd.args: ${JSON.stringify(buildCmd.args)}`); | ||||||
|  |  | ||||||
|  |     let err: Error | undefined; | ||||||
|  |     await Exec.getExecOutput(buildCmd.command, buildCmd.args, { | ||||||
|  |       ignoreReturnCode: true, | ||||||
|  |       env: Object.assign({}, process.env, { | ||||||
|  |         BUILDX_METADATA_WARNINGS: 'true' | ||||||
|  |       }) as { | ||||||
|  |         [key: string]: string; | ||||||
|  |       } | ||||||
|  |     }).then(res => { | ||||||
|  |       if (res.exitCode != 0) { | ||||||
|  |         if (inputs.call && inputs.call === 'check' && res.stdout.length > 0) { | ||||||
|  |           // checks warnings are printed to stdout: https://github.com/docker/buildx/pull/2647 | ||||||
|  |           // take the first line with the message summaryzing the warnings | ||||||
|  |           err = new Error(res.stdout.split('\n')[0]?.trim()); | ||||||
|  |         } else if (res.stderr.length > 0) { | ||||||
|  |           err = new Error(`buildx failed with: ${res.stderr.match(/(.*)\s*$/)?.[0]?.trim() ?? 'unknown error'}`); | ||||||
|  |         } | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     const imageID = await buildx.getImageID(); |     const imageID = toolkit.buildxBuild.resolveImageID(); | ||||||
|     const metadata = await buildx.getMetadata(); |     const metadata = toolkit.buildxBuild.resolveMetadata(); | ||||||
|     const digest = await buildx.getDigest(metadata); |     const digest = toolkit.buildxBuild.resolveDigest(metadata); | ||||||
|  |  | ||||||
|     if (imageID) { |     if (imageID) { | ||||||
|       await core.group(`ImageID`, async () => { |       await core.group(`ImageID`, async () => { | ||||||
|         core.info(imageID); |         core.info(imageID); | ||||||
| @@ -81,25 +134,173 @@ async function run(): Promise<void> { | |||||||
|     } |     } | ||||||
|     if (metadata) { |     if (metadata) { | ||||||
|       await core.group(`Metadata`, async () => { |       await core.group(`Metadata`, async () => { | ||||||
|         core.info(metadata); |         const metadatadt = JSON.stringify(metadata, null, 2); | ||||||
|         core.setOutput('metadata', metadata); |         core.info(metadatadt); | ||||||
|  |         core.setOutput('metadata', metadatadt); | ||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|   } catch (error) { |  | ||||||
|     core.setFailed(error.message); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function cleanup(): Promise<void> { |     let ref: string | undefined; | ||||||
|  |     await core.group(`Reference`, async () => { | ||||||
|  |       ref = await buildRef(toolkit, startedTime, inputs.builder); | ||||||
|  |       if (ref) { | ||||||
|  |         core.info(ref); | ||||||
|  |         stateHelper.setBuildRef(ref); | ||||||
|  |       } else { | ||||||
|  |         core.info('No build reference found'); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     if (buildChecksAnnotationsEnabled()) { | ||||||
|  |       const warnings = toolkit.buildxBuild.resolveWarnings(metadata); | ||||||
|  |       if (ref && warnings && warnings.length > 0) { | ||||||
|  |         const annotations = await Buildx.convertWarningsToGitHubAnnotations(warnings, [ref]); | ||||||
|  |         core.debug(`annotations: ${JSON.stringify(annotations, null, 2)}`); | ||||||
|  |         if (annotations && annotations.length > 0) { | ||||||
|  |           await core.group(`Generating GitHub annotations (${annotations.length} build checks found)`, async () => { | ||||||
|  |             for (const annotation of annotations) { | ||||||
|  |               core.warning(annotation.message, annotation); | ||||||
|  |             } | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     await core.group(`Check build summary support`, async () => { | ||||||
|  |       if (!buildSummaryEnabled()) { | ||||||
|  |         core.info('Build summary disabled'); | ||||||
|  |       } else if (inputs.call && inputs.call !== 'build') { | ||||||
|  |         core.info(`Build summary skipped for ${inputs.call} subrequest`); | ||||||
|  |       } else if (GitHub.isGHES) { | ||||||
|  |         core.info('Build summary is not yet supported on GHES'); | ||||||
|  |       } else if (!(await toolkit.buildx.versionSatisfies('>=0.13.0'))) { | ||||||
|  |         core.info('Build summary requires Buildx >= 0.13.0'); | ||||||
|  |       } else if (!ref) { | ||||||
|  |         core.info('Build summary requires a build reference'); | ||||||
|  |       } else { | ||||||
|  |         core.info('Build summary supported!'); | ||||||
|  |         stateHelper.setSummarySupported(); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     if (err) { | ||||||
|  |       throw err; | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |   // post | ||||||
|  |   async () => { | ||||||
|  |     if (stateHelper.isSummarySupported) { | ||||||
|  |       await core.group(`Generating build summary`, async () => { | ||||||
|  |         try { | ||||||
|  |           const recordUploadEnabled = buildRecordUploadEnabled(); | ||||||
|  |           let recordRetentionDays: number | undefined; | ||||||
|  |           if (recordUploadEnabled) { | ||||||
|  |             recordRetentionDays = buildRecordRetentionDays(); | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |           const buildxHistory = new BuildxHistory(); | ||||||
|  |           const exportRes = await buildxHistory.export({ | ||||||
|  |             refs: stateHelper.buildRef ? [stateHelper.buildRef] : [], | ||||||
|  |             useContainer: buildExportLegacy() | ||||||
|  |           }); | ||||||
|  |           core.info(`Build record written to ${exportRes.dockerbuildFilename} (${Util.formatFileSize(exportRes.dockerbuildSize)})`); | ||||||
|  |  | ||||||
|  |           let uploadRes: UploadArtifactResponse | undefined; | ||||||
|  |           if (recordUploadEnabled) { | ||||||
|  |             uploadRes = await GitHub.uploadArtifact({ | ||||||
|  |               filename: exportRes.dockerbuildFilename, | ||||||
|  |               mimeType: 'application/gzip', | ||||||
|  |               retentionDays: recordRetentionDays | ||||||
|  |             }); | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |           await GitHub.writeBuildSummary({ | ||||||
|  |             exportRes: exportRes, | ||||||
|  |             uploadRes: uploadRes, | ||||||
|  |             inputs: stateHelper.summaryInputs, | ||||||
|  |             driver: stateHelper.builderDriver, | ||||||
|  |             endpoint: stateHelper.builderEndpoint | ||||||
|  |           }); | ||||||
|  |         } catch (e) { | ||||||
|  |           core.warning(e.message); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|     if (stateHelper.tmpDir.length > 0) { |     if (stateHelper.tmpDir.length > 0) { | ||||||
|     core.startGroup(`Removing temp folder ${stateHelper.tmpDir}`); |       await core.group(`Removing temp folder ${stateHelper.tmpDir}`, async () => { | ||||||
|  |         try { | ||||||
|           fs.rmSync(stateHelper.tmpDir, {recursive: true}); |           fs.rmSync(stateHelper.tmpDir, {recursive: true}); | ||||||
|     core.endGroup(); |         } catch (e) { | ||||||
|  |           core.warning(`Failed to remove temp folder ${stateHelper.tmpDir}`); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | async function buildRef(toolkit: Toolkit, since: Date, builder?: string): Promise<string> { | ||||||
|  |   // get ref from metadata file | ||||||
|  |   const ref = toolkit.buildxBuild.resolveRef(); | ||||||
|  |   if (ref) { | ||||||
|  |     return ref; | ||||||
|  |   } | ||||||
|  |   // otherwise, look for the very first build ref since the build has started | ||||||
|  |   if (!builder) { | ||||||
|  |     const currentBuilder = await toolkit.builder.inspect(); | ||||||
|  |     builder = currentBuilder.name; | ||||||
|  |   } | ||||||
|  |   const refs = Buildx.refs({ | ||||||
|  |     dir: Buildx.refsDir, | ||||||
|  |     builderName: builder, | ||||||
|  |     since: since | ||||||
|  |   }); | ||||||
|  |   return Object.keys(refs).length > 0 ? Object.keys(refs)[0] : ''; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function buildChecksAnnotationsEnabled(): boolean { | ||||||
|  |   if (process.env.DOCKER_BUILD_CHECKS_ANNOTATIONS) { | ||||||
|  |     return Util.parseBool(process.env.DOCKER_BUILD_CHECKS_ANNOTATIONS); | ||||||
|  |   } | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function buildSummaryEnabled(): boolean { | ||||||
|  |   if (process.env.DOCKER_BUILD_NO_SUMMARY) { | ||||||
|  |     core.warning('DOCKER_BUILD_NO_SUMMARY is deprecated. Set DOCKER_BUILD_SUMMARY to false instead.'); | ||||||
|  |     return !Util.parseBool(process.env.DOCKER_BUILD_NO_SUMMARY); | ||||||
|  |   } else if (process.env.DOCKER_BUILD_SUMMARY) { | ||||||
|  |     return Util.parseBool(process.env.DOCKER_BUILD_SUMMARY); | ||||||
|  |   } | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function buildRecordUploadEnabled(): boolean { | ||||||
|  |   if (process.env.DOCKER_BUILD_RECORD_UPLOAD) { | ||||||
|  |     return Util.parseBool(process.env.DOCKER_BUILD_RECORD_UPLOAD); | ||||||
|  |   } | ||||||
|  |   return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function buildRecordRetentionDays(): number | undefined { | ||||||
|  |   let val: string | undefined; | ||||||
|  |   if (process.env.DOCKER_BUILD_EXPORT_RETENTION_DAYS) { | ||||||
|  |     core.warning('DOCKER_BUILD_EXPORT_RETENTION_DAYS is deprecated. Use DOCKER_BUILD_RECORD_RETENTION_DAYS instead.'); | ||||||
|  |     val = process.env.DOCKER_BUILD_EXPORT_RETENTION_DAYS; | ||||||
|  |   } else if (process.env.DOCKER_BUILD_RECORD_RETENTION_DAYS) { | ||||||
|  |     val = process.env.DOCKER_BUILD_RECORD_RETENTION_DAYS; | ||||||
|  |   } | ||||||
|  |   if (val) { | ||||||
|  |     const res = parseInt(val); | ||||||
|  |     if (isNaN(res)) { | ||||||
|  |       throw new Error(`Invalid build record retention days: ${val}`); | ||||||
|  |     } | ||||||
|  |     return res; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| if (!stateHelper.IsPost) { | function buildExportLegacy(): boolean { | ||||||
|   run(); |   if (process.env.DOCKER_BUILD_EXPORT_LEGACY) { | ||||||
| } else { |     return Util.parseBool(process.env.DOCKER_BUILD_EXPORT_LEGACY); | ||||||
|   cleanup(); |   } | ||||||
|  |   return false; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,12 +1,70 @@ | |||||||
| import * as core from '@actions/core'; | import * as core from '@actions/core'; | ||||||
|  |  | ||||||
| export const IsPost = !!process.env['STATE_isPost']; | import {Build} from '@docker/actions-toolkit/lib/buildx/build'; | ||||||
|  |  | ||||||
|  | import {Inputs} from './context'; | ||||||
|  |  | ||||||
| export const tmpDir = process.env['STATE_tmpDir'] || ''; | export const tmpDir = process.env['STATE_tmpDir'] || ''; | ||||||
|  |  | ||||||
|  | export const builderDriver = process.env['STATE_builderDriver'] || ''; | ||||||
|  | export const builderEndpoint = process.env['STATE_builderEndpoint'] || ''; | ||||||
|  | export const summaryInputs = process.env['STATE_summaryInputs'] ? JSON.parse(process.env['STATE_summaryInputs']) : undefined; | ||||||
|  |  | ||||||
|  | export const buildRef = process.env['STATE_buildRef'] || ''; | ||||||
|  | export const isSummarySupported = !!process.env['STATE_isSummarySupported']; | ||||||
|  |  | ||||||
| export function setTmpDir(tmpDir: string) { | export function setTmpDir(tmpDir: string) { | ||||||
|   core.saveState('tmpDir', tmpDir); |   core.saveState('tmpDir', tmpDir); | ||||||
| } | } | ||||||
|  |  | ||||||
| if (!IsPost) { | export function setBuilderDriver(builderDriver: string) { | ||||||
|   core.saveState('isPost', 'true'); |   core.saveState('builderDriver', builderDriver); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export function setBuilderEndpoint(builderEndpoint: string) { | ||||||
|  |   core.saveState('builderEndpoint', builderEndpoint); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export function setBuildRef(buildRef: string) { | ||||||
|  |   core.saveState('buildRef', buildRef); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export function setSummarySupported() { | ||||||
|  |   core.saveState('isSummarySupported', 'true'); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export function setSummaryInputs(inputs: Inputs) { | ||||||
|  |   const res = {}; | ||||||
|  |   for (const key of Object.keys(inputs)) { | ||||||
|  |     if (key === 'github-token') { | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |     const value: string | string[] | boolean = inputs[key]; | ||||||
|  |     if (typeof value === 'boolean' && !value) { | ||||||
|  |       continue; | ||||||
|  |     } else if (Array.isArray(value)) { | ||||||
|  |       if (value.length === 0) { | ||||||
|  |         continue; | ||||||
|  |       } else if (key === 'secrets' && value.length > 0) { | ||||||
|  |         const secretKeys: string[] = []; | ||||||
|  |         for (const secret of value) { | ||||||
|  |           try { | ||||||
|  |             // eslint-disable-next-line @typescript-eslint/no-unused-vars | ||||||
|  |             const [skey, _] = Build.parseSecretKvp(secret, true); | ||||||
|  |             secretKeys.push(skey); | ||||||
|  |           } catch (err) { | ||||||
|  |             // ignore invalid secret | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |         if (secretKeys.length > 0) { | ||||||
|  |           res[key] = secretKeys; | ||||||
|  |         } | ||||||
|  |         continue; | ||||||
|  |       } | ||||||
|  |     } else if (!value) { | ||||||
|  |       continue; | ||||||
|  |     } | ||||||
|  |     res[key] = value; | ||||||
|  |   } | ||||||
|  |   core.saveState('summaryInputs', JSON.stringify(res)); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,16 +1,19 @@ | |||||||
| FROM golang:1.19-alpine AS base | # syntax=docker/dockerfile:1 | ||||||
|  |  | ||||||
|  | FROM golang:alpine AS base | ||||||
| ENV CGO_ENABLED=0 | ENV CGO_ENABLED=0 | ||||||
| RUN apk add --no-cache file git | RUN apk add --no-cache file git | ||||||
| WORKDIR /src | WORKDIR /src | ||||||
|  |  | ||||||
| FROM base as build | FROM base AS build | ||||||
| COPY go.mod go.sum ./ | RUN --mount=type=bind,target=/src \ | ||||||
| RUN go mod download -x |     --mount=type=cache,target=/root/.cache/go-build \ | ||||||
| COPY . . |     go build -ldflags "-s -w" -o /usr/bin/app . | ||||||
| RUN go build -ldflags "-s -w" -o /usr/bin/app . |  | ||||||
|  |  | ||||||
| FROM scratch AS binary | FROM scratch AS binary | ||||||
| COPY --from=build /usr/bin/app /bin/app | COPY --from=build /usr/bin/app /bin/app | ||||||
|  |  | ||||||
| FROM alpine:3.17 AS image | FROM alpine AS image | ||||||
| COPY --from=build /usr/bin/app /bin/app | COPY --from=build /usr/bin/app /bin/app | ||||||
|  | EXPOSE 8080 | ||||||
|  | ENTRYPOINT ["/bin/app"] | ||||||
|   | |||||||
| @@ -1,19 +1,3 @@ | |||||||
| module github.com/docker/build-push-action/test/go | module github.com/docker/build-push-action/test/go | ||||||
|  |  | ||||||
| go 1.18 | go 1.18 | ||||||
|  |  | ||||||
| require github.com/labstack/echo/v4 v4.9.1 |  | ||||||
|  |  | ||||||
| require ( |  | ||||||
| 	github.com/golang-jwt/jwt v3.2.2+incompatible // indirect |  | ||||||
| 	github.com/labstack/gommon v0.4.0 // indirect |  | ||||||
| 	github.com/mattn/go-colorable v0.1.11 // indirect |  | ||||||
| 	github.com/mattn/go-isatty v0.0.14 // indirect |  | ||||||
| 	github.com/valyala/bytebufferpool v1.0.0 // indirect |  | ||||||
| 	github.com/valyala/fasttemplate v1.2.1 // indirect |  | ||||||
| 	golang.org/x/crypto v0.0.0-20210817164053-32db794688a5 // indirect |  | ||||||
| 	golang.org/x/net v0.0.0-20211015210444-4f30a5c0130f // indirect |  | ||||||
| 	golang.org/x/sys v0.0.0-20211103235746-7861aae1554b // indirect |  | ||||||
| 	golang.org/x/text v0.3.7 // indirect |  | ||||||
| 	golang.org/x/time v0.0.0-20201208040808-7e3f01d25324 // indirect |  | ||||||
| ) |  | ||||||
|   | |||||||
| @@ -1,38 +0,0 @@ | |||||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= |  | ||||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= |  | ||||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= |  | ||||||
| github.com/golang-jwt/jwt v3.2.2+incompatible h1:IfV12K8xAKAnZqdXVzCZ+TOjboZ2keLg81eXfW3O+oY= |  | ||||||
| github.com/golang-jwt/jwt v3.2.2+incompatible/go.mod h1:8pz2t5EyA70fFQQSrl6XZXzqecmYZeUEB8OUGHkxJ+I= |  | ||||||
| github.com/labstack/echo/v4 v4.9.1 h1:GliPYSpzGKlyOhqIbG8nmHBo3i1saKWFOgh41AN3b+Y= |  | ||||||
| github.com/labstack/echo/v4 v4.9.1/go.mod h1:Pop5HLc+xoc4qhTZ1ip6C0RtP7Z+4VzRLWZZFKqbbjo= |  | ||||||
| github.com/labstack/gommon v0.4.0 h1:y7cvthEAEbU0yHOf4axH8ZG2NH8knB9iNSoTO8dyIk8= |  | ||||||
| github.com/labstack/gommon v0.4.0/go.mod h1:uW6kP17uPlLJsD3ijUYn3/M5bAxtlZhMI6m3MFxTMTM= |  | ||||||
| github.com/mattn/go-colorable v0.1.11 h1:nQ+aFkoE2TMGc0b68U2OKSexC+eq46+XwZzWXHRmPYs= |  | ||||||
| github.com/mattn/go-colorable v0.1.11/go.mod h1:u5H1YNBxpqRaxsYJYSkiCWKzEfiAb1Gb520KVy5xxl4= |  | ||||||
| github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y= |  | ||||||
| github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= |  | ||||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= |  | ||||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= |  | ||||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= |  | ||||||
| github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= |  | ||||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= |  | ||||||
| github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw= |  | ||||||
| github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= |  | ||||||
| github.com/valyala/fasttemplate v1.2.1 h1:TVEnxayobAdVkhQfrfes2IzOB6o+z4roRkPF52WA1u4= |  | ||||||
| github.com/valyala/fasttemplate v1.2.1/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= |  | ||||||
| golang.org/x/crypto v0.0.0-20210817164053-32db794688a5 h1:HWj/xjIHfjYU5nVXpTM0s39J9CbLn7Cc5a7IC5rwsMQ= |  | ||||||
| golang.org/x/crypto v0.0.0-20210817164053-32db794688a5/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= |  | ||||||
| golang.org/x/net v0.0.0-20211015210444-4f30a5c0130f h1:OfiFi4JbukWwe3lzw+xunroH1mnC1e2Gy5cxNJApiSY= |  | ||||||
| golang.org/x/net v0.0.0-20211015210444-4f30a5c0130f/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= |  | ||||||
| golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/sys v0.0.0-20211103235746-7861aae1554b h1:1VkfZQv42XQlA/jchYumAnv1UPo6RgF9rJFkTgZIxO4= |  | ||||||
| golang.org/x/sys v0.0.0-20211103235746-7861aae1554b/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= |  | ||||||
| golang.org/x/text v0.3.7 h1:olpwvP2KacW1ZWvsR7uQhoyTYvKAupfQrRGBFM352Gk= |  | ||||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= |  | ||||||
| golang.org/x/time v0.0.0-20201208040808-7e3f01d25324 h1:Hir2P/De0WpUhtrKGGjvSb2YxUgyZ7EFOSLIcSSpiwE= |  | ||||||
| golang.org/x/time v0.0.0-20201208040808-7e3f01d25324/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= |  | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= |  | ||||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= |  | ||||||
| gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b h1:h8qDotaEPuJATrMmW04NCwg7v22aHH28wwpauUhK9Oo= |  | ||||||
| gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= |  | ||||||
| @@ -1,31 +1,14 @@ | |||||||
| package main | package main | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"log" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"os" |  | ||||||
|  |  | ||||||
| 	"github.com/labstack/echo/v4" |  | ||||||
| 	"github.com/labstack/echo/v4/middleware" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func main() { | func main() { | ||||||
| 	e := echo.New() | 	http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) { | ||||||
|  | 		fmt.Fprintf(w, "Hello, Go!") | ||||||
| 	e.Use(middleware.Logger()) |  | ||||||
| 	e.Use(middleware.Recover()) |  | ||||||
|  |  | ||||||
| 	e.GET("/", func(c echo.Context) error { |  | ||||||
| 		return c.HTML(http.StatusOK, "Hello World") |  | ||||||
| 	}) | 	}) | ||||||
|  | 	log.Fatal(http.ListenAndServe(":8080", nil)) | ||||||
| 	e.GET("/ping", func(c echo.Context) error { |  | ||||||
| 		return c.JSON(http.StatusOK, struct{ Status string }{Status: "OK"}) |  | ||||||
| 	}) |  | ||||||
|  |  | ||||||
| 	httpPort := os.Getenv("HTTP_PORT") |  | ||||||
| 	if httpPort == "" { |  | ||||||
| 		httpPort = "8080" |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	e.Logger.Fatal(e.Start(":" + httpPort)) |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								test/lint.Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								test/lint.Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | frOM busybox as base | ||||||
|  | cOpy lint.Dockerfile . | ||||||
|  |  | ||||||
|  | from scratch | ||||||
|  | MAINTAINER moby@example.com | ||||||
|  | COPy --from=base \ | ||||||
|  |   /lint.Dockerfile \ | ||||||
|  |   / | ||||||
|  |  | ||||||
|  | CMD [ "echo", "Hello, Norway!" ] | ||||||
|  | CMD [ "echo", "Hello, Sweden!" ] | ||||||
|  | ENTRYPOINT my-program start | ||||||
| @@ -1,5 +1,6 @@ | |||||||
| # syntax=docker/dockerfile:1 | # syntax=docker/dockerfile:1 | ||||||
| FROM --platform=$BUILDPLATFORM golang:alpine AS build |  | ||||||
|  | FROM --platform=$BUILDPLATFORM alpine AS build | ||||||
| ARG TARGETPLATFORM | ARG TARGETPLATFORM | ||||||
| ARG BUILDPLATFORM | ARG BUILDPLATFORM | ||||||
| RUN echo "I am running on $BUILDPLATFORM, building for $TARGETPLATFORM" > /log | RUN echo "I am running on $BUILDPLATFORM, building for $TARGETPLATFORM" > /log | ||||||
| @@ -12,7 +13,7 @@ RUN apk --update --no-cache add \ | |||||||
|   && rm -rf /tmp/* /var/cache/apk/* |   && rm -rf /tmp/* /var/cache/apk/* | ||||||
|  |  | ||||||
| USER buildx | USER buildx | ||||||
| RUN sudo chown buildx. /log | RUN sudo chown buildx: /log | ||||||
| USER root | USER root | ||||||
|  |  | ||||||
| FROM alpine | FROM alpine | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								test/named-context-base.Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								test/named-context-base.Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | # syntax=docker/dockerfile:1 | ||||||
|  |  | ||||||
|  | FROM debian | ||||||
|  | RUN echo "Hello debian!" | ||||||
| @@ -1,3 +1,4 @@ | |||||||
| # syntax=docker/dockerfile:1 | # syntax=docker/dockerfile:1 | ||||||
|  | 
 | ||||||
| FROM alpine | FROM alpine | ||||||
| RUN cat /etc/*release | RUN cat /etc/*release | ||||||
							
								
								
									
										9
									
								
								test/proxy.Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								test/proxy.Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | # syntax=docker/dockerfile:1 | ||||||
|  | FROM alpine | ||||||
|  | RUN apk add --no-cache curl net-tools | ||||||
|  | ARG HTTP_PROXY | ||||||
|  | ARG HTTPS_PROXY | ||||||
|  | RUN printenv HTTP_PROXY | ||||||
|  | RUN printenv HTTPS_PROXY | ||||||
|  | RUN netstat -aptn | ||||||
|  | RUN curl --retry 5 --retry-all-errors --retry-delay 0 --connect-timeout 5 --proxy $HTTP_PROXY -v --insecure --head https://www.google.com | ||||||
| @@ -1,19 +1,22 @@ | |||||||
| { | { | ||||||
|   "compilerOptions": { |   "compilerOptions": { | ||||||
|  |     "esModuleInterop": true, | ||||||
|     "target": "es6", |     "target": "es6", | ||||||
|     "module": "commonjs", |     "module": "commonjs", | ||||||
|  |     "strict": true, | ||||||
|     "newLine": "lf", |     "newLine": "lf", | ||||||
|     "outDir": "./lib", |     "outDir": "./lib", | ||||||
|     "rootDir": "./src", |     "rootDir": "./src", | ||||||
|     "esModuleInterop": true, |  | ||||||
|     "forceConsistentCasingInFileNames": true, |     "forceConsistentCasingInFileNames": true, | ||||||
|     "strict": true, |  | ||||||
|     "noImplicitAny": false, |     "noImplicitAny": false, | ||||||
|  |     "resolveJsonModule": true, | ||||||
|     "useUnknownInCatchVariables": false, |     "useUnknownInCatchVariables": false, | ||||||
|   }, |   }, | ||||||
|   "exclude": [ |   "exclude": [ | ||||||
|  |     "./__mocks__/**/*", | ||||||
|  |     "./__tests__/**/*", | ||||||
|  |     "./lib/**/*", | ||||||
|     "node_modules", |     "node_modules", | ||||||
|     "**/*.test.ts", |  | ||||||
|     "jest.config.ts" |     "jest.config.ts" | ||||||
|   ] |   ] | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user