mirror of
				https://github.com/actions/node-versions.git
				synced 2025-10-26 15:26:41 +08:00 
			
		
		
		
	Compare commits
	
		
			304 Commits
		
	
	
		
			10.15.0-20
			...
			main
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | b9b8a06ea4 | ||
|  | 64801c3590 | ||
|  | a2eca09038 | ||
|  | fcfdeb317e | ||
|  | b32305dc94 | ||
|  | 018c97355d | ||
|  | 82b9d6eb7f | ||
|  | 6c1dcef7e3 | ||
|  | 414aa4b42d | ||
|  | 05932b3923 | ||
|  | 8bc7882285 | ||
|  | 447f233d8c | ||
|  | 48d06ac109 | ||
|  | 2f48f5f179 | ||
|  | 7bff9a2a32 | ||
|  | ef431e4d6b | ||
|  | 498365717c | ||
|  | 8c7740e607 | ||
|  | 6fda0935f2 | ||
|  | e25da9dc7c | ||
|  | 3de2515c74 | ||
|  | 89d81124fa | ||
|  | c5d5eb32a4 | ||
|  | 48b638b2a7 | ||
|  | 70176e8e7f | ||
|  | b8e2e0988f | ||
|  | 6f7e07ce98 | ||
|  | 759e7130a4 | ||
|  | 708c79f819 | ||
|  | ff0573dc52 | ||
|  | f3742acb8e | ||
|  | b5d073fd75 | ||
|  | 07972c08eb | ||
|  | 1089754c37 | ||
|  | d0b553a9eb | ||
|  | 09e0810728 | ||
|  | 83c0f2d854 | ||
|  | 1dda8b0c7d | ||
|  | a09329c1fa | ||
|  | 75581b6027 | ||
|  | bb9d406a6c | ||
|  | 4638058663 | ||
|  | c749aff12a | ||
|  | aacaf58011 | ||
|  | 5b159cbd4a | ||
|  | c8cf9cb4b5 | ||
|  | 3a7a3835dd | ||
|  | ec09b8ed02 | ||
|  | 22b86c61f6 | ||
|  | ee011d651d | ||
|  | f33462c8a2 | ||
|  | b8355a0fcc | ||
|  | d2561554de | ||
|  | ab329f0180 | ||
|  | cb84d2280a | ||
|  | a26a00efbb | ||
|  | 386b5d6f78 | ||
|  | fb3241c6f2 | ||
|  | c88b3f72e2 | ||
|  | 6c82900b55 | ||
|  | de463ffcdb | ||
|  | d51fc26ff7 | ||
|  | fee25d72a2 | ||
|  | bb598e8c31 | ||
|  | a798c8ed46 | ||
|  | 9d10c5b01a | ||
|  | a4a2f9a437 | ||
|  | 3d5333b8c1 | ||
|  | 897b0a01fc | ||
|  | 605dcd4078 | ||
|  | 705d5d4eb2 | ||
|  | aa0c3a667a | ||
|  | 989324aed0 | ||
|  | 59fe4e7639 | ||
|  | 5bebf2b61c | ||
|  | 015a819882 | ||
|  | bebc948c04 | ||
|  | 953fa628ab | ||
|  | ad868c9f05 | ||
|  | 5136b584d3 | ||
|  | 217a0f2329 | ||
|  | e1e54a94f4 | ||
|  | 802851f7e1 | ||
|  | 3985d2e8f1 | ||
|  | ed20ce4f29 | ||
|  | 86f7959999 | ||
|  | f9eb480338 | ||
|  | fc05df56de | ||
|  | 97eefec78a | ||
|  | a646a40627 | ||
|  | 742347d102 | ||
|  | c27b52d5a8 | ||
|  | 280ff21dab | ||
|  | d1c2cb96e1 | ||
|  | d4215ec43e | ||
|  | 9c020055eb | ||
|  | 80f2e07bc4 | ||
|  | 2a3603b0e8 | ||
|  | f3fac10826 | ||
|  | cfedc107f9 | ||
|  | a938f96cba | ||
|  | 3e6fdaa9dd | ||
|  | c8117e2a6f | ||
|  | a3474c6453 | ||
|  | 70b3a61620 | ||
|  | 5d308e3118 | ||
|  | 5ae9e2a880 | ||
|  | 85ccf2dd9b | ||
|  | 1563799473 | ||
|  | 32e71c3c3b | ||
|  | edcf67ef40 | ||
|  | 72d306dec2 | ||
|  | 7e8f83364e | ||
|  | d065c37c10 | ||
|  | 6f4ddf8c2c | ||
|  | deab417b89 | ||
|  | 46c45cb2bb | ||
|  | 80d54cdc81 | ||
|  | bcd6df9184 | ||
|  | 6751014c63 | ||
|  | fe2adddc4d | ||
|  | 4431f5baf6 | ||
|  | 4414c50507 | ||
|  | 5dd67ed7f9 | ||
|  | ec00a6e346 | ||
|  | 6c2489748b | ||
|  | 7ee585b554 | ||
|  | 86a16e3fbc | ||
|  | 618e16944c | ||
|  | cc0c4db785 | ||
|  | dd82515163 | ||
|  | 3f418bfd29 | ||
|  | fd15ad763a | ||
|  | 02a74240bb | ||
|  | 9b9e1bad68 | ||
|  | e379f2ee0d | ||
|  | e6e7799d67 | ||
|  | 11f6d8a2de | ||
|  | b03ec2b334 | ||
|  | 41fdef674f | ||
|  | bb5faa8461 | ||
|  | 914307fdad | ||
|  | 7a042473f1 | ||
|  | 46aed5d195 | ||
|  | 595b4b690a | ||
|  | 29a273c108 | ||
|  | bb9f13b224 | ||
|  | 453a014457 | ||
|  | cf85dc514f | ||
|  | 8acc56fc1b | ||
|  | a034d4c79d | ||
|  | 7ce0e62855 | ||
|  | ad4c15127c | ||
|  | e60aab1337 | ||
|  | fff8e8d2e7 | ||
|  | 9439938c24 | ||
|  | 7efd2b6d9c | ||
|  | 9f1448d546 | ||
|  | d6c0f3407a | ||
|  | 7ee0eb6d7f | ||
|  | 0e81926ec5 | ||
|  | 2b00037cf2 | ||
|  | 03a45f2d48 | ||
|  | 285c046a12 | ||
|  | d9cf0e3053 | ||
|  | d86ab03024 | ||
|  | ad9cb45c2a | ||
|  | aa33d265c4 | ||
|  | 58c3d40bb8 | ||
|  | 577e22bee8 | ||
|  | 2200e9ff30 | ||
|  | 09f8ea5c48 | ||
|  | d78f09bbe4 | ||
|  | acea7cad3a | ||
|  | 3257340283 | ||
|  | 833c6da64b | ||
|  | b2560ab6d9 | ||
|  | 84b6aaeb97 | ||
|  | b195064d9c | ||
|  | a72cfa7a7c | ||
|  | 63fc200ef6 | ||
|  | ad4714dee5 | ||
|  | 0287ac915c | ||
|  | 5c5f144d08 | ||
|  | 6e415f531b | ||
|  | 5a83490d4f | ||
|  | e5e3b83d13 | ||
|  | 9405eaa89a | ||
|  | 947a3202c2 | ||
|  | 5edc6dfe71 | ||
|  | c54f68be31 | ||
|  | dcd9e83d63 | ||
|  | e530a1172f | ||
|  | ee57453268 | ||
|  | c0fad79083 | ||
|  | 484fa88a69 | ||
|  | 8c866dde54 | ||
|  | 334a8d1394 | ||
|  | 72210eed7d | ||
|  | 2b595e0a51 | ||
|  | 7ac82af1b6 | ||
|  | 6d939f1dbe | ||
|  | 2b6ee4852f | ||
|  | b3a2ff96e2 | ||
|  | 0141b50275 | ||
|  | 7481b6b090 | ||
|  | 4fcc8f116d | ||
|  | e5f055b1c1 | ||
|  | 34f35b5cb6 | ||
|  | e21cafedd8 | ||
|  | b6e0bccc9f | ||
|  | 3e0c0eb0d6 | ||
|  | 62cc8fa6fb | ||
|  | 5fa4e704be | ||
|  | 60911f3d0a | ||
|  | 1aa2fc42b8 | ||
|  | 6a059f6c80 | ||
|  | 48cfd035f5 | ||
|  | 4066cc83f8 | ||
|  | 2277f8c8a4 | ||
|  | 28cdbd7c2e | ||
|  | a0e1e3c90e | ||
|  | 3716ed3c6e | ||
|  | 6fd0e71a0a | ||
|  | 5093d2b929 | ||
|  | 2fa5a9904c | ||
|  | ab76c432fe | ||
|  | ead568a146 | ||
|  | 6fdbc6deb3 | ||
|  | 69ae188e0e | ||
|  | 75b1a9f3f2 | ||
|  | ea5fefae18 | ||
|  | 8b33f33473 | ||
|  | d92adb7da1 | ||
|  | c2b002c470 | ||
|  | 649c1ec0c7 | ||
|  | fbeb7d747d | ||
|  | 445b5079f1 | ||
|  | 07c6e38e3e | ||
|  | 805c28590c | ||
|  | 72562f08b4 | ||
|  | adde90d079 | ||
|  | 6772ce992e | ||
|  | 196a7ffaab | ||
|  | 5803dd40ce | ||
|  | 15c3ac315a | ||
|  | 4790533e80 | ||
|  | c01749b1c4 | ||
|  | 01407a89d3 | ||
|  | ff6bcb4ec0 | ||
|  | a229d53240 | ||
|  | 593745d7c2 | ||
|  | e459abbdbe | ||
|  | f01c560f7f | ||
|  | 2b37bf1f1e | ||
|  | fe299f9388 | ||
|  | bbb4ceccb0 | ||
|  | d8a00be11c | ||
|  | cb6506c1cc | ||
|  | 69c87ed4ce | ||
|  | 36757fae02 | ||
|  | 05026253f5 | ||
|  | 2da30901b3 | ||
|  | d2b2d6cf72 | ||
|  | b6945a3f95 | ||
|  | 17f4719d01 | ||
|  | 19ff3f1812 | ||
|  | 6a38aba626 | ||
|  | 25173f9ae7 | ||
|  | c52043a439 | ||
|  | 70f1fd4673 | ||
|  | 02194c5c42 | ||
|  | 47471a527c | ||
|  | 978acceff5 | ||
|  | ed56a19091 | ||
|  | 1a9656d766 | ||
|  | 414a9c4d19 | ||
|  | dea9e4ccea | ||
|  | 070efd5a09 | ||
|  | 60042303e5 | ||
|  | 690c39139f | ||
|  | dbc462f6f0 | ||
|  | 2d85084834 | ||
|  | 71dc6cf704 | ||
|  | 2275f3fe19 | ||
|  | deeaedc1f4 | ||
|  | 82819ecf50 | ||
|  | 73c7229556 | ||
|  | a9761d076d | ||
|  | 63f56e381c | ||
|  | 9825d4703f | ||
|  | 376bb1b4eb | ||
|  | f675396c66 | ||
|  | 155e54b53a | ||
|  | 46502ad44b | ||
|  | ac6a66a1ac | ||
|  | ae3fa7b514 | ||
|  | 434ca92b37 | ||
|  | 7c335a2ccc | ||
|  | 891863d663 | ||
|  | a60b759ec9 | ||
|  | 66ca3f81b9 | ||
|  | ae23532a5f | ||
|  | 2ab95dc3b1 | 
							
								
								
									
										1
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | * @actions/setup-actions-team | ||||||
							
								
								
									
										31
									
								
								.github/workflows/build-node-packages.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								.github/workflows/build-node-packages.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,31 @@ | |||||||
|  | name: Generate Node.js packages | ||||||
|  | run-name: Generate Node.js ${{ inputs.VERSION || '18.12.0' }} | ||||||
|  | on: | ||||||
|  |   workflow_dispatch: | ||||||
|  |     inputs: | ||||||
|  |       VERSION: | ||||||
|  |         description: 'Node.js version to build and upload' | ||||||
|  |         required: true | ||||||
|  |         default: '18.12.0' | ||||||
|  |       PUBLISH_RELEASES: | ||||||
|  |         description: 'Whether to publish releases' | ||||||
|  |         required: true | ||||||
|  |         type: boolean | ||||||
|  |         default: false | ||||||
|  |   pull_request: | ||||||
|  |     paths-ignore: | ||||||
|  |     - 'versions-manifest.json' | ||||||
|  |     - 'LICENSE' | ||||||
|  |     - '**.md' | ||||||
|  |     branches: | ||||||
|  |     - 'main' | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   node: | ||||||
|  |     name: Node | ||||||
|  |     uses: actions/versions-package-tools/.github/workflows/build-tool-packages.yml@main | ||||||
|  |     with: | ||||||
|  |       tool-name: "node" | ||||||
|  |       tool-version: ${{ inputs.VERSION || '18.12.0' }} | ||||||
|  |       publish-release: ${{ inputs.PUBLISH_RELEASES || false }} | ||||||
|  |     secrets: inherit | ||||||
							
								
								
									
										14
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,14 @@ | |||||||
|  | name: CodeQL analysis | ||||||
|  | 
 | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: [ main ] | ||||||
|  |   pull_request: | ||||||
|  |     branches: [ main ] | ||||||
|  |   schedule: | ||||||
|  |     - cron: '0 3 * * 0' | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   call-codeQL-analysis: | ||||||
|  |     name: CodeQL analysis  | ||||||
|  |     uses: actions/reusable-workflows/.github/workflows/codeql-analysis.yml@main | ||||||
							
								
								
									
										10
									
								
								.github/workflows/create-pr.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/create-pr.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | name: Create Pull Request | ||||||
|  | on: | ||||||
|  |   workflow_dispatch: | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   create-pr: | ||||||
|  |     uses: actions/versions-package-tools/.github/workflows/create-pr-to-update-manifest.yml@main | ||||||
|  |     with: | ||||||
|  |       tool-name: "node" | ||||||
|  |     secrets: inherit | ||||||
							
								
								
									
										13
									
								
								.github/workflows/get-node-versions.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								.github/workflows/get-node-versions.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,13 @@ | |||||||
|  | name: Get Node versions | ||||||
|  | on: | ||||||
|  |   schedule: | ||||||
|  |     - cron: '0 0,12 * * *' | ||||||
|  |   workflow_dispatch: | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   get-new-node-versions: | ||||||
|  |     uses: actions/versions-package-tools/.github/workflows/get-new-tool-versions.yml@main | ||||||
|  |     with: | ||||||
|  |       tool-name: "Node" | ||||||
|  |       image-url: "https://nodejs.org/static/images/logo-hexagon-card.png" | ||||||
|  |     secrets: inherit | ||||||
							
								
								
									
										21
									
								
								.github/workflows/validate-manifest.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								.github/workflows/validate-manifest.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | name: Validate manifest | ||||||
|  | on: | ||||||
|  | # The GITHUB_TOKEN secret is used to create a PR | ||||||
|  | # The pull_request event will not be triggered by it | ||||||
|  | # That's one of the reasons we need the schedule to validate the versions-manifest.json file | ||||||
|  |   schedule: | ||||||
|  |     - cron: '0 8,20 * * *' | ||||||
|  |   workflow_dispatch: | ||||||
|  |   pull_request: | ||||||
|  |     branches: | ||||||
|  |       - main | ||||||
|  |     paths: | ||||||
|  |       - 'versions-manifest.json' | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   manifest: | ||||||
|  |     uses: actions/versions-package-tools/.github/workflows/validate-manifest.yml@main | ||||||
|  |     with: | ||||||
|  |       tool-name: "Node" | ||||||
|  |       image-url: "https://nodejs.org/static/images/logo-hexagon-card.png" | ||||||
|  |     secrets: inherit | ||||||
							
								
								
									
										4
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | [submodule "helpers"] | ||||||
|  | 	path = helpers | ||||||
|  | 	url = https://github.com/actions/versions-package-tools | ||||||
|  | 	branch = main | ||||||
| @ -29,22 +29,20 @@ Here are a few things you can do that will increase the likelihood of your pull | |||||||
| ### Directory structure | ### Directory structure | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ├── azure-pipelines/ | ├── .github/ | ||||||
| |   └──templates/ | |   └──workflows/ | ||||||
| ├── builders/ | ├── builders/ | ||||||
| ├── helpers/ | ├── helpers/ | ||||||
| ├── installers/ | ├── installers/ | ||||||
| └── tests/ | └── tests/ | ||||||
|     └──sources/ |     └──sources/ | ||||||
| ``` | ``` | ||||||
| - `azure-pipelines*` - contains global YAML definitions for build pipelines. Reusable templates for specific jobs are located in `templates` subfolder. | - `.github/workflows` - contains repository workflow files. | ||||||
| - `builders` - contains Node.js builder classes and functions. | - `builders` - contains Node.js builder classes and functions. | ||||||
| - `helpers` - contains global helper functions and functions. | - `helpers` - contains global helper classes and functions. | ||||||
| - `installers` - contains installation script templates. | - `installers` - contains installation script templates. | ||||||
| - `tests` - contains test scripts. Required tests sources are located in `sources` subfolder. | - `tests` - contains test scripts. Required tests sources are located in `sources` subfolder. | ||||||
| 
 | 
 | ||||||
| \* _We use Azure Pipelines because there are a few features that Actions is still missing, we'll move to Actions as soon as possible_. |  | ||||||
| 
 |  | ||||||
| ## Resources | ## Resources | ||||||
| 
 | 
 | ||||||
| - [How to Contribute to Open Source](https://opensource.guide/how-to-contribute/) | - [How to Contribute to Open Source](https://opensource.guide/how-to-contribute/) | ||||||
|  | |||||||
							
								
								
									
										11
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								README.md
									
									
									
									
									
								
							| @ -1,15 +1,18 @@ | |||||||
| # Node.js for Actions | # Node.js for Actions | ||||||
| This repository contains the code and scripts that we use to prepare Node.js packages used in [virtual-environments](https://github.com/actions/virtual-environments) and accessible through the [setup-node](https://github.com/actions/setup-node) Action.   | This repository contains the code and scripts that we use to prepare Node.js packages used in [runner-images](https://github.com/actions/runner-images) and accessible through the [setup-node](https://github.com/actions/setup-node) Action.   | ||||||
| The file [versions-manifest.json](./versions-manifest.json) contains the list of available and released versions.   | The file [versions-manifest.json](./versions-manifest.json) contains the list of available and released versions.   | ||||||
| 
 | 
 | ||||||
| > Caution: this is prepared for and only permitted for use by actions `virtual-environments` and `setup-node` action. | > Caution: this is prepared for and only permitted for use by actions `runner-images` and `setup-node` action. | ||||||
| 
 | 
 | ||||||
| **Status**: Currently under development and in use for beta and preview actions.  This repo is undergoing rapid changes. | **Status**: Currently under development and in use for beta and preview actions.  This repo is undergoing rapid changes. | ||||||
| 
 | 
 | ||||||
| Latest of LTS versions will be installed on the [virtual-environments](https://github.com/actions/virtual-environments) images.  Other versions will be pulled JIT using the [`setup-node`](https://github.com/actions/setup-node) action. | Latest of LTS versions will be installed on the [runner-images](https://github.com/actions/runner-images) images. Other versions will be pulled JIT using the [`setup-node`](https://github.com/actions/setup-node) action. | ||||||
| 
 | 
 | ||||||
| ## Adding new versions | ## Adding new versions | ||||||
| We are trying to prepare packages for new versions of Node.js as soon as they are released. Please open an issue if any versions are missing. | We are trying to prepare packages for new versions of Node.js as soon as they are released. Please open an issue in [actions/runner-images](https://github.com/actions/runner-images) if any versions are missing. | ||||||
|  | 
 | ||||||
|  | ## Support Notification Policy | ||||||
|  | Beginning **approximately six months prior** to the removal of a Node.js version from the [versions-manifest.json](https://github.com/actions/node-versions/blob/main/versions-manifest.json) file, a pinned issue will be created in the [setup-node](https://github.com/actions/setup-node) repository. This pinned issue will provide important details about the upcoming end of support, including the specific date, as well as any other notes, relevant updates or alternatives. We encourage users to regularly check pinned issues for updates on tool versions they are using for maximum transparency, security, performance and overall compatibility with their projects. | ||||||
| 
 | 
 | ||||||
| ## Contribution | ## Contribution | ||||||
| Contributions are welcome! See [Contributor's Guide](./CONTRIBUTING.md) for more details about contribution process and code structure | Contributions are welcome! See [Contributor's Guide](./CONTRIBUTING.md) for more details about contribution process and code structure | ||||||
|  | |||||||
| @ -1,65 +0,0 @@ | |||||||
| name: $(date:yyyyMMdd)$(rev:.r)-Node.js-$(VERSION) |  | ||||||
| trigger: none |  | ||||||
| pr: |  | ||||||
|   autoCancel: true |  | ||||||
|   branches: |  | ||||||
|     include: |  | ||||||
|     - master |  | ||||||
|   paths: |  | ||||||
|     exclude: |  | ||||||
|     - versions-manifest.json |  | ||||||
| 
 |  | ||||||
| stages: |  | ||||||
| - stage: Build_Node_Darwin |  | ||||||
|   dependsOn: [] |  | ||||||
|   variables: |  | ||||||
|     Platform: darwin |  | ||||||
|     Architecture: x64 |  | ||||||
|   jobs: |  | ||||||
|   - template: /azure-pipelines/templates/build-job.yml |  | ||||||
| 
 |  | ||||||
| - stage: Test_Node_Darwin |  | ||||||
|   condition: succeeded() |  | ||||||
|   dependsOn: Build_Node_Darwin |  | ||||||
|   variables: |  | ||||||
|     VmImage: macOS-latest |  | ||||||
|     Platform: darwin |  | ||||||
|     Architecture: x64 |  | ||||||
|   jobs: |  | ||||||
|   - template: /azure-pipelines/templates/test-job.yml |  | ||||||
| 
 |  | ||||||
| - stage: Build_Node_Linux |  | ||||||
|   dependsOn: [] |  | ||||||
|   variables: |  | ||||||
|     Platform: linux |  | ||||||
|     Architecture: x64 |  | ||||||
|   jobs: |  | ||||||
|   - template: /azure-pipelines/templates/build-job.yml |  | ||||||
| 
 |  | ||||||
| - stage: Test_Node_Linux |  | ||||||
|   condition: succeeded() |  | ||||||
|   dependsOn: Build_Node_Linux |  | ||||||
|   variables: |  | ||||||
|     VmImage: ubuntu-latest |  | ||||||
|     Platform: linux |  | ||||||
|     Architecture: x64 |  | ||||||
|   jobs: |  | ||||||
|   - template: /azure-pipelines/templates/test-job.yml |  | ||||||
| 
 |  | ||||||
| - stage: Build_Node_Windows |  | ||||||
|   dependsOn: [] |  | ||||||
|   variables: |  | ||||||
|     Platform: win32 |  | ||||||
|     Architecture: x64 |  | ||||||
|   jobs: |  | ||||||
|   - template: /azure-pipelines/templates/build-job.yml |  | ||||||
| 
 |  | ||||||
| - stage: Test_Node_Windows |  | ||||||
|   condition: succeeded() |  | ||||||
|   dependsOn: Build_Node_Windows |  | ||||||
|   variables: |  | ||||||
|     VmImage: windows-latest |  | ||||||
|     Platform: win32 |  | ||||||
|     Architecture: x64 |  | ||||||
|   jobs: |  | ||||||
|   - template: /azure-pipelines/templates/test-job.yml |  | ||||||
| @ -1,21 +0,0 @@ | |||||||
| jobs: |  | ||||||
| - job: Build_Node |  | ||||||
|   timeoutInMinutes: 90 |  | ||||||
|   pool: |  | ||||||
|     name: Azure Pipelines |  | ||||||
|     vmImage: ubuntu-latest |  | ||||||
|   steps: |  | ||||||
|   - checkout: self |  | ||||||
|      |  | ||||||
|   - task: PowerShell@2 |  | ||||||
|     displayName: 'Build Node $(Version)' |  | ||||||
|     inputs: |  | ||||||
|       targetType: filePath |  | ||||||
|       filePath: './builders/build-node.ps1' |  | ||||||
|       arguments: '-Version $(Version) -Platform $(Platform) -Architecture $(Architecture)' |  | ||||||
| 
 |  | ||||||
|   - task: PublishPipelineArtifact@1 |  | ||||||
|     displayName: 'Publish Artifact: Node.js $(Version)' |  | ||||||
|     inputs: |  | ||||||
|       targetPath: '$(Build.ArtifactStagingDirectory)'  |  | ||||||
|       artifactName: 'node-$(Version)-$(Platform)-$(Architecture)' |  | ||||||
| @ -1,78 +0,0 @@ | |||||||
| jobs: |  | ||||||
| - job: Test_Node |  | ||||||
|   pool:  |  | ||||||
|     name: Azure Pipelines |  | ||||||
|     vmImage: $(VmImage) |  | ||||||
|   steps: |  | ||||||
|   - checkout: self |  | ||||||
|     submodules: true |  | ||||||
| 
 |  | ||||||
|   - task: PowerShell@2 |  | ||||||
|     displayName: Fully cleanup the toolcache directory before testing |  | ||||||
|     inputs: |  | ||||||
|       TargetType: inline |  | ||||||
|       script: | |  | ||||||
|         $NodeToolcachePath = Join-Path -Path $env:AGENT_TOOLSDIRECTORY -ChildPath "node" |  | ||||||
|         if (Test-Path $NodeToolcachePath) { |  | ||||||
|           Remove-Item -Path $NodeToolcachePath -Recurse -Force |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|   - task: DownloadPipelineArtifact@2 |  | ||||||
|     inputs: |  | ||||||
|       source: 'current' |  | ||||||
|       artifact: 'node-$(Version)-$(Platform)-$(Architecture)' |  | ||||||
|       path: $(Build.ArtifactStagingDirectory) |  | ||||||
| 
 |  | ||||||
|   - task: ExtractFiles@1 |  | ||||||
|     inputs: |  | ||||||
|       archiveFilePatterns: '$(Build.ArtifactStagingDirectory)/node-$(Version)-$(Platform)-$(Architecture).*' |  | ||||||
|       destinationFolder: $(Build.BinariesDirectory) |  | ||||||
|       cleanDestinationFolder: false |  | ||||||
| 
 |  | ||||||
|   - task: PowerShell@2 |  | ||||||
|     displayName: 'Apply build artifact to the local machines' |  | ||||||
|     inputs: |  | ||||||
|       targetType: inline |  | ||||||
|       script: | |  | ||||||
|         if ("$(Platform)" -match 'win32') { powershell ./setup.ps1 } else { sh ./setup.sh } |  | ||||||
|       workingDirectory: '$(Build.BinariesDirectory)' |  | ||||||
| 
 |  | ||||||
|   - task: NodeTool@0 |  | ||||||
|     displayName: 'Use Node $(Version)' |  | ||||||
|     inputs: |  | ||||||
|       versionSpec: $(Version) |  | ||||||
| 
 |  | ||||||
|   - task: PowerShell@2 |  | ||||||
|     displayName: 'Wait for the logs' |  | ||||||
|     inputs: |  | ||||||
|       targetType: inline |  | ||||||
|       script: | |  | ||||||
|          Write-Host "Fake step that do nothing" |  | ||||||
|          Write-Host "We need it because log of previous step 'Use Node' is not available here yet." |  | ||||||
|          Write-Host "In testing step (Node.Tests.ps1) we analyze build log of 'Use Node' task" |  | ||||||
|          Write-Host "to determine if Node.js version was consumed from cache and was downloaded" |  | ||||||
|   - task: PowerShell@2 |  | ||||||
|     displayName: 'Run tests' |  | ||||||
|     inputs: |  | ||||||
|       TargetType: inline |  | ||||||
|       script: | |  | ||||||
|         Install-Module Pester -Force -Scope CurrentUser |  | ||||||
|         Import-Module Pester |  | ||||||
|         $pesterParams = @{ |  | ||||||
|           Path="./Node.Tests.ps1"; |  | ||||||
|           Parameters=@{ |  | ||||||
|             Version="$(Version)"; |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|         Invoke-Pester -Script $pesterParams -OutputFile "test_results.xml" -OutputFormat NUnitXml |  | ||||||
|       workingDirectory: '$(Build.SourcesDirectory)/tests' |  | ||||||
| 
 |  | ||||||
|   - task: PublishTestResults@2 |  | ||||||
|     displayName: 'Publish test results' |  | ||||||
|     inputs: |  | ||||||
|       testResultsFiles: '*.xml' |  | ||||||
|       testResultsFormat: NUnit |  | ||||||
|       searchFolder: 'tests' |  | ||||||
|       failTaskOnFailedTests: true |  | ||||||
|       testRunTitle: "Node.js $(Version)-$(Platform)" |  | ||||||
|     condition: always() |  | ||||||
| @ -1,5 +1,5 @@ | |||||||
| using module "./builders/win-node-builder.psm1" | using module "./win-node-builder.psm1" | ||||||
| using module "./builders/nix-node-builder.psm1" | using module "./nix-node-builder.psm1" | ||||||
| 
 | 
 | ||||||
| <# | <# | ||||||
| .SYNOPSIS | .SYNOPSIS | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| using module "./builders/node-builder.psm1" | using module "./node-builder.psm1" | ||||||
| 
 | 
 | ||||||
| class NixNodeBuilder : NodeBuilder { | class NixNodeBuilder : NodeBuilder { | ||||||
|     <# |     <# | ||||||
| @ -54,7 +54,7 @@ class NixNodeBuilder : NodeBuilder { | |||||||
|         $installationTemplateLocation = Join-Path -Path $this.InstallationTemplatesLocation -ChildPath $this.InstallationTemplateName |         $installationTemplateLocation = Join-Path -Path $this.InstallationTemplatesLocation -ChildPath $this.InstallationTemplateName | ||||||
| 
 | 
 | ||||||
|         $installationTemplateContent = Get-Content -Path $installationTemplateLocation -Raw |         $installationTemplateContent = Get-Content -Path $installationTemplateLocation -Raw | ||||||
|         $installationTemplateContent = $installationTemplateContent -f $this.Version.ToString(3) |         $installationTemplateContent = $installationTemplateContent -f $this.Version.ToString(3), $this.Architecture | ||||||
|         $installationTemplateContent | Out-File -FilePath $installationScriptLocation |         $installationTemplateContent | Out-File -FilePath $installationScriptLocation | ||||||
| 
 | 
 | ||||||
|         Write-Debug "Done; Installation script location: $installationScriptLocation)" |         Write-Debug "Done; Installation script location: $installationScriptLocation)" | ||||||
|  | |||||||
| @ -16,10 +16,13 @@ class NodeBuilder { | |||||||
|     The architecture with which Node.js should be built. |     The architecture with which Node.js should be built. | ||||||
| 
 | 
 | ||||||
|     .PARAMETER TempFolderLocation |     .PARAMETER TempFolderLocation | ||||||
|     The location of temporary files that will be used during Node.js package generation. Using system BUILD_STAGINGDIRECTORY variable value. |     The location of temporary files that will be used during Node.js package generation. | ||||||
| 
 | 
 | ||||||
|     .PARAMETER ArtifactLocation |     .PARAMETER WorkFolderLocation | ||||||
|     The location of generated Node.js artifact. Using system environment BUILD_BINARIESDIRECTORY variable value. |     The location of installation files. | ||||||
|  | 
 | ||||||
|  |     .PARAMETER ArtifactFolderLocation | ||||||
|  |     The location of generated Node.js artifact. | ||||||
| 
 | 
 | ||||||
|     .PARAMETER InstallationTemplatesLocation |     .PARAMETER InstallationTemplatesLocation | ||||||
|     The location of installation script template. Using "installers" folder from current repository. |     The location of installation script template. Using "installers" folder from current repository. | ||||||
| @ -40,9 +43,8 @@ class NodeBuilder { | |||||||
|         $this.Architecture = $architecture |         $this.Architecture = $architecture | ||||||
| 
 | 
 | ||||||
|         $this.TempFolderLocation = [IO.Path]::GetTempPath() |         $this.TempFolderLocation = [IO.Path]::GetTempPath() | ||||||
|         $this.WorkFolderLocation = $env:BUILD_BINARIESDIRECTORY |         $this.WorkFolderLocation = Join-Path $env:RUNNER_TEMP "binaries" | ||||||
|         $this.ArtifactFolderLocation = $env:BUILD_STAGINGDIRECTORY |         $this.ArtifactFolderLocation = Join-Path $env:RUNNER_TEMP "artifact" | ||||||
|          |  | ||||||
| 
 | 
 | ||||||
|         $this.InstallationTemplatesLocation = Join-Path -Path $PSScriptRoot -ChildPath "../installers" |         $this.InstallationTemplatesLocation = Join-Path -Path $PSScriptRoot -ChildPath "../installers" | ||||||
|     } |     } | ||||||
| @ -84,6 +86,10 @@ class NodeBuilder { | |||||||
|         Generates Node.js artifact from downloaded binaries. |         Generates Node.js artifact from downloaded binaries. | ||||||
|         #> |         #> | ||||||
| 
 | 
 | ||||||
|  |         Write-Host "Create WorkFolderLocation and ArtifactFolderLocation folders" | ||||||
|  |         New-Item -Path $this.WorkFolderLocation -ItemType "directory" | ||||||
|  |         New-Item -Path $this.ArtifactFolderLocation -ItemType "directory" | ||||||
|  | 
 | ||||||
|         Write-Host "Download Node.js $($this.Version) [$($this.Architecture)] executable..." |         Write-Host "Download Node.js $($this.Version) [$($this.Architecture)] executable..." | ||||||
|         $binariesArchivePath = $this.Download() |         $binariesArchivePath = $this.Download() | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| using module "./builders/node-builder.psm1" | using module "./node-builder.psm1" | ||||||
| 
 | 
 | ||||||
| class WinNodeBuilder : NodeBuilder { | class WinNodeBuilder : NodeBuilder { | ||||||
|     <# |     <# | ||||||
|  | |||||||
							
								
								
									
										8
									
								
								config/node-manifest-config.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								config/node-manifest-config.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | { | ||||||
|  |     "regex": "node-\\d+\\.\\d+\\.\\d+-(\\w+)-((x|arm)\\d+)", | ||||||
|  |     "groups": { | ||||||
|  |         "arch": 2, | ||||||
|  |         "platform": 1 | ||||||
|  |     }, | ||||||
|  |     "lts_rule_expression": "(Invoke-RestMethod 'https://raw.githubusercontent.com/nodejs/Release/main/schedule.json').PSObject.Properties | Where-Object { $_.Value.codename } | ForEach-Object { @{ Name = $_.Name.TrimStart('v'); Value = $_.Value.codename } }" | ||||||
|  | } | ||||||
							
								
								
									
										1
									
								
								helpers
									
									
									
									
									
										Submodule
									
								
							
							
								
								
								
								
								
								
									
									
								
							
						
						
									
										1
									
								
								helpers
									
									
									
									
									
										Submodule
									
								
							| @ -0,0 +1 @@ | |||||||
|  | Subproject commit 6fbb1f0f2098254142702dba05fe75cd8e77c4ae | ||||||
| @ -1,89 +0,0 @@ | |||||||
| class AzureDevOpsApi |  | ||||||
| { |  | ||||||
|     [string] $BaseUrl |  | ||||||
|     [string] $RepoOwner |  | ||||||
|     [object] $AuthHeader |  | ||||||
| 
 |  | ||||||
|     AzureDevOpsApi( |  | ||||||
|         [string] $TeamFoundationCollectionUri, |  | ||||||
|         [string] $ProjectName, |  | ||||||
|         [string] $AccessToken |  | ||||||
|     ) { |  | ||||||
|         $this.BaseUrl = $this.BuildBaseUrl($TeamFoundationCollectionUri, $ProjectName) |  | ||||||
|         $this.AuthHeader = $this.BuildAuth($AccessToken) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [object] hidden BuildAuth([string]$AccessToken) { |  | ||||||
|         if ([string]::IsNullOrEmpty($AccessToken)) { |  | ||||||
|             return $null |  | ||||||
|         } |  | ||||||
|         return @{ |  | ||||||
|             Authorization = "Bearer $AccessToken" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [string] hidden BuildBaseUrl([string]$TeamFoundationCollectionUri, [string]$ProjectName) { |  | ||||||
|         return "${TeamFoundationCollectionUri}/${ProjectName}/_apis" |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [object] QueueBuild([string]$ToolVersion, [string]$SourceBranch, [string]$SourceVersion, [UInt32]$DefinitionId){ |  | ||||||
|         $url = "build/builds" |  | ||||||
|          |  | ||||||
|         # The content of parameters field should be a json string |  | ||||||
|         $buildParameters = @{ VERSION = $ToolVersion } | ConvertTo-Json |  | ||||||
| 
 |  | ||||||
|         $body = @{ |  | ||||||
|             definition = @{ |  | ||||||
|                 id = $DefinitionId |  | ||||||
|             } |  | ||||||
|             sourceBranch = $SourceBranch |  | ||||||
|             sourceVersion = $SourceVersion |  | ||||||
|             parameters = $buildParameters |  | ||||||
|         } | ConvertTo-Json |  | ||||||
| 
 |  | ||||||
|         return $this.InvokeRestMethod($url, 'POST', $body) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [object] GetBuildInfo([UInt32]$BuildId){ |  | ||||||
|         $url = "build/builds/$BuildId" |  | ||||||
| 
 |  | ||||||
|         return $this.InvokeRestMethod($url, 'GET', $null) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [string] hidden BuildUrl([string]$Url) { |  | ||||||
|         return "$($this.BaseUrl)/${Url}/?api-version=5.1" |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [object] hidden InvokeRestMethod( |  | ||||||
|         [string] $Url, |  | ||||||
|         [string] $Method, |  | ||||||
|         [string] $Body |  | ||||||
|     ) { |  | ||||||
|         $requestUrl = $this.BuildUrl($Url) |  | ||||||
|         $params = @{ |  | ||||||
|             Method = $Method |  | ||||||
|             ContentType = "application/json" |  | ||||||
|             Uri = $requestUrl |  | ||||||
|             Headers = @{} |  | ||||||
|         } |  | ||||||
|         if ($this.AuthHeader) { |  | ||||||
|             $params.Headers += $this.AuthHeader |  | ||||||
|         } |  | ||||||
|         if (![string]::IsNullOrEmpty($body)) { |  | ||||||
|             $params.Body = $Body |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return Invoke-RestMethod @params |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Get-AzureDevOpsApi { |  | ||||||
|     param ( |  | ||||||
|         [string] $TeamFoundationCollectionUri, |  | ||||||
|         [string] $ProjectName, |  | ||||||
|         [string] $AccessToken |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     return [AzureDevOpsApi]::New($TeamFoundationCollectionUri, $ProjectName, $AccessToken) |  | ||||||
| } |  | ||||||
| @ -1,44 +0,0 @@ | |||||||
| Import-Module (Join-Path $PSScriptRoot "azure-devops-api.ps1") |  | ||||||
| 
 |  | ||||||
| class BuildInfo |  | ||||||
| { |  | ||||||
|     [AzureDevOpsApi] $AzureDevOpsApi |  | ||||||
|     [String] $Name |  | ||||||
|     [UInt32] $Id |  | ||||||
|     [String] $Status |  | ||||||
|     [String] $Result |  | ||||||
|     [String] $Link |  | ||||||
| 
 |  | ||||||
|     BuildInfo([AzureDevOpsApi] $AzureDevOpsApi, [object] $Build) |  | ||||||
|     { |  | ||||||
|         $this.AzureDevOpsApi = $AzureDevOpsApi |  | ||||||
|         $this.Id = $Build.id |  | ||||||
|         $this.Name = $Build.buildNumber |  | ||||||
|         $this.Link = $Build._links.web.href |  | ||||||
|         $this.Status = $Build.status |  | ||||||
|         $this.Result = $Build.result |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [boolean] IsFinished() { |  | ||||||
|         return ($this.Status -eq "completed") -or ($this.Status -eq "cancelling") |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [boolean] IsSuccess() { |  | ||||||
|         return $this.Result -eq "succeeded" |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [void] UpdateBuildInfo() { |  | ||||||
|         $buildInfo = $this.AzureDevOpsApi.GetBuildInfo($this.Id) |  | ||||||
|         $this.Status = $buildInfo.status |  | ||||||
|         $this.Result = $buildInfo.result |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Get-BuildInfo { |  | ||||||
|     param ( |  | ||||||
|         [AzureDevOpsApi] $AzureDevOpsApi, |  | ||||||
|         [object] $Build |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     return [BuildInfo]::New($AzureDevOpsApi, $Build) |  | ||||||
| } |  | ||||||
| @ -1,94 +0,0 @@ | |||||||
| param ( |  | ||||||
|     [Parameter(Mandatory)] [string] $TeamFoundationCollectionUri, |  | ||||||
|     [Parameter(Mandatory)] [string] $AzureDevOpsProjectName, |  | ||||||
|     [Parameter(Mandatory)] [string] $AzureDevOpsAccessToken, |  | ||||||
|     [Parameter(Mandatory)] [string] $SourceBranch, |  | ||||||
|     [Parameter(Mandatory)] [string] $ToolVersions, |  | ||||||
|     [Parameter(Mandatory)] [UInt32] $DefinitionId, |  | ||||||
|     [string] $SourceVersion |  | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| Import-Module (Join-Path $PSScriptRoot "azure-devops-api.ps1") |  | ||||||
| Import-Module (Join-Path $PSScriptRoot "build-info.ps1") |  | ||||||
| 
 |  | ||||||
| function Queue-Builds { |  | ||||||
|     param ( |  | ||||||
|         [Parameter(Mandatory)] [AzureDevOpsApi] $AzureDevOpsApi, |  | ||||||
|         [Parameter(Mandatory)] [string] $ToolVersions, |  | ||||||
|         [Parameter(Mandatory)] [string] $SourceBranch, |  | ||||||
|         [Parameter(Mandatory)] [string] $SourceVersion, |  | ||||||
|         [Parameter(Mandatory)] [string] $DefinitionId |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     [BuildInfo[]]$queuedBuilds = @() |  | ||||||
| 
 |  | ||||||
|     $ToolVersions.Split(',') | ForEach-Object {  |  | ||||||
|         $version = $_.Trim() |  | ||||||
|         Write-Host "Queue build for $version..." |  | ||||||
|         $queuedBuild = $AzureDevOpsApi.QueueBuild($version, $SourceBranch, $SourceVersion, $DefinitionId) |  | ||||||
|         $buildInfo = Get-BuildInfo -AzureDevOpsApi $AzureDevOpsApi -Build $queuedBuild |  | ||||||
|         Write-Host "Queued build: $($buildInfo.Link)" |  | ||||||
|         $queuedBuilds += $buildInfo |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return $queuedBuilds |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Wait-Builds { |  | ||||||
|     param ( |  | ||||||
|         [Parameter(Mandatory)] [BuildInfo[]] $Builds |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     $timeoutBetweenRefreshSec = 30 |  | ||||||
|      |  | ||||||
|     do { |  | ||||||
|         # If build is still running - refresh its status |  | ||||||
|         foreach($build in $builds) { |  | ||||||
|             if (!$build.IsFinished()) { |  | ||||||
|                 $build.UpdateBuildInfo() |  | ||||||
|                  |  | ||||||
|                 if ($build.IsFinished()) { |  | ||||||
|                    Write-Host "The $($build.Name) build was completed: $($build.Link)" |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|      |  | ||||||
|         $runningBuildsCount = ($builds | Where-Object { !$_.IsFinished() }).Length |  | ||||||
| 
 |  | ||||||
|         Start-Sleep -Seconds $timeoutBetweenRefreshSec |  | ||||||
|     } while($runningBuildsCount -gt 0) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Make-BuildsOutput { |  | ||||||
|     param ( |  | ||||||
|         [Parameter(Mandatory)] [BuildInfo[]] $Builds |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     Write-Host "Builds info:" |  | ||||||
|     $builds | Format-Table -AutoSize -Property Name,Id,Status,Result,Link | Out-String -Width 10000 |  | ||||||
| 
 |  | ||||||
|     # Return exit code based on status of builds |  | ||||||
|     $failedBuilds = ($builds | Where-Object { !$_.IsSuccess() }) |  | ||||||
|     if ($failedBuilds.Length -ne 0) { |  | ||||||
|         Write-Host "##vso[task.logissue type=error;]Builds failed" |  | ||||||
|         $failedBuilds | ForEach-Object -Process { Write-Host "##vso[task.logissue type=error;]Name: $($_.Name); Link: $($_.Link)" } |  | ||||||
|         Write-Host "##vso[task.complete result=Failed]" |  | ||||||
|     } else { |  | ||||||
|         Write-host "##[section] All builds have been passed successfully" |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| $azureDevOpsApi = Get-AzureDevOpsApi -TeamFoundationCollectionUri $TeamFoundationCollectionUri ` |  | ||||||
|                                      -ProjectName $AzureDevOpsProjectName ` |  | ||||||
|                                      -AccessToken $AzureDevOpsAccessToken |  | ||||||
| 
 |  | ||||||
| $queuedBuilds = Queue-Builds -AzureDevOpsApi $azureDevOpsApi ` |  | ||||||
|                              -ToolVersions $ToolVersions ` |  | ||||||
|                              -SourceBranch $SourceBranch ` |  | ||||||
|                              -SourceVersion $SourceVersion ` |  | ||||||
|                              -DefinitionId $DefinitionId |  | ||||||
| 
 |  | ||||||
| Write-Host "Waiting results of builds ..." |  | ||||||
| Wait-Builds -Builds $queuedBuilds |  | ||||||
| 
 |  | ||||||
| Make-BuildsOutput -Builds $queuedBuilds |  | ||||||
| @ -1,106 +0,0 @@ | |||||||
| <# |  | ||||||
| .SYNOPSIS |  | ||||||
| Create commit with all unstaged changes in repository and create pull-request |  | ||||||
| 
 |  | ||||||
| .PARAMETER RepositoryOwner |  | ||||||
| Required parameter. The organization which tool repository belongs |  | ||||||
| .PARAMETER RepositoryName |  | ||||||
| Optional parameter. The name of tool repository |  | ||||||
| .PARAMETER AccessToken |  | ||||||
| Required parameter. PAT Token to authorize |  | ||||||
| .PARAMETER BranchName |  | ||||||
| Required parameter. The name of branch where changes will be pushed |  | ||||||
| .PARAMETER CommitMessage |  | ||||||
| Required parameter. The commit message to push changes |  | ||||||
| .PARAMETER PullRequestTitle |  | ||||||
| Required parameter. The title of pull-request |  | ||||||
| .PARAMETER PullRequestBody |  | ||||||
| Required parameter. The description of pull-request |  | ||||||
| #> |  | ||||||
| param ( |  | ||||||
|     [Parameter(Mandatory)] [string] $RepositoryOwner, |  | ||||||
|     [Parameter(Mandatory)] [string] $RepositoryName, |  | ||||||
|     [Parameter(Mandatory)] [string] $AccessToken, |  | ||||||
|     [Parameter(Mandatory)] [string] $BranchName, |  | ||||||
|     [Parameter(Mandatory)] [string] $CommitMessage, |  | ||||||
|     [Parameter(Mandatory)] [string] $PullRequestTitle, |  | ||||||
|     [Parameter(Mandatory)] [string] $PullRequestBody |  | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| Import-Module (Join-Path $PSScriptRoot "github-api.psm1") |  | ||||||
| Import-Module (Join-Path $PSScriptRoot "git.psm1") |  | ||||||
| 
 |  | ||||||
| function Update-PullRequest { |  | ||||||
|     Param ( |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [object] $GitHubApi, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $Title, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $Body, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $BranchName, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [object] $PullRequest |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     $updatedPullRequest = $GitHubApi.UpdatePullRequest($Title, $Body, $BranchName, $PullRequest.number) |  | ||||||
| 
 |  | ||||||
|     if (($updatedPullRequest -eq $null) -or ($updatedPullRequest.html_url -eq $null)) { |  | ||||||
|         Write-Host "##vso[task.logissue type=error;] Unexpected error occurs while updating pull request." |  | ||||||
|         exit 1 |  | ||||||
|     } |  | ||||||
|     Write-host "##[section] Pull request updated: $($updatedPullRequest.html_url)" |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Create-PullRequest { |  | ||||||
|     Param ( |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [object] $GitHubApi, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $Title, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $Body, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $BranchName |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     $createdPullRequest = $GitHubApi.CreateNewPullRequest($Title, $Body, $BranchName) |  | ||||||
| 
 |  | ||||||
|     if (($createdPullRequest -eq $null) -or ($createdPullRequest.html_url -eq $null)) { |  | ||||||
|         Write-Host "##vso[task.logissue type=error;] Unexpected error occurs while creating pull request." |  | ||||||
|         exit 1 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     Write-host "##[section] Pull request created: $($createdPullRequest.html_url)" |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| Write-Host "Configure local git preferences" |  | ||||||
| Git-ConfigureUser -Name "Service account" -Email "no-reply@microsoft.com" |  | ||||||
| 
 |  | ||||||
| Write-Host "Create branch: $BranchName" |  | ||||||
| Git-CreateBranch -Name $BranchName |  | ||||||
|      |  | ||||||
| Write-Host "Create commit" |  | ||||||
| Git-CommitAllChanges -Message $CommitMessage |  | ||||||
| 
 |  | ||||||
| Write-Host "Push branch: $BranchName" |  | ||||||
| Git-PushBranch -Name $BranchName -Force $true |  | ||||||
| 
 |  | ||||||
| $gitHubApi = Get-GitHubApi -AccountName $RepositoryOwner -ProjectName $RepositoryName -AccessToken $AccessToken |  | ||||||
| $pullRequest = $gitHubApi.GetPullRequest($BranchName, $RepositoryOwner) |  | ||||||
| 
 |  | ||||||
| if ($pullRequest.Count -gt 0) { |  | ||||||
|     Write-Host "Update pull request" |  | ||||||
|     Update-PullRequest -GitHubApi $gitHubApi ` |  | ||||||
|                        -Title $PullRequestTitle ` |  | ||||||
|                        -Body $PullRequestBody ` |  | ||||||
|                        -BranchName $BranchName ` |  | ||||||
|                        -PullRequest $pullRequest[0] |  | ||||||
| } else { |  | ||||||
|     Write-Host "Create pull request" |  | ||||||
|     Create-PullRequest -GitHubApi $gitHubApi ` |  | ||||||
|                        -Title $PullRequestTitle ` |  | ||||||
|                        -Body $PullRequestBody ` |  | ||||||
|                        -BranchName $BranchName |  | ||||||
| } |  | ||||||
| @ -1,81 +0,0 @@ | |||||||
| <# |  | ||||||
| .SYNOPSIS |  | ||||||
| Configure git credentials to use with commits |  | ||||||
| #> |  | ||||||
| function Git-ConfigureUser { |  | ||||||
|     Param ( |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $Name, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $Email |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     git config --global user.name $Name | Out-Host |  | ||||||
|     git config --global user.email $Email | Out-Host |  | ||||||
| 
 |  | ||||||
|     if ($LASTEXITCODE -ne 0) { |  | ||||||
|         Write-Host "##vso[task.logissue type=error;] Unexpected failure occurs while configuring git preferences." |  | ||||||
|         exit 1 |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| <# |  | ||||||
| .SYNOPSIS |  | ||||||
| Create new branch |  | ||||||
| #> |  | ||||||
| function Git-CreateBranch { |  | ||||||
|     Param ( |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $Name |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     git checkout -b $Name | Out-Host |  | ||||||
| 
 |  | ||||||
|     if ($LASTEXITCODE -ne 0) { |  | ||||||
|         Write-Host "##vso[task.logissue type=error;] Unexpected failure occurs while creating new branch: $Name." |  | ||||||
|         exit 1 |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| <# |  | ||||||
| .SYNOPSIS |  | ||||||
| Commit all staged and unstaged changes |  | ||||||
| #> |  | ||||||
| function Git-CommitAllChanges { |  | ||||||
|     Param ( |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $Message |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     git add -A | Out-Host |  | ||||||
|     git commit -m "$Message" | Out-Host |  | ||||||
| 
 |  | ||||||
|     if ($LASTEXITCODE -ne 0) { |  | ||||||
|         Write-Host "##vso[task.logissue type=error;] Unexpected failure occurs while commiting changes." |  | ||||||
|         exit 1 |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| <# |  | ||||||
| .SYNOPSIS |  | ||||||
| Push branch to remote repository |  | ||||||
| #> |  | ||||||
| function Git-PushBranch { |  | ||||||
|     Param ( |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [string] $Name, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [boolean] $Force |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     if ($Force) { |  | ||||||
|         git push --set-upstream origin $Name --force | Out-Host |  | ||||||
|     } else { |  | ||||||
|         git push --set-upstream origin $Name | Out-Host |  | ||||||
|     } |  | ||||||
|      |  | ||||||
|     if ($LASTEXITCODE -ne 0) { |  | ||||||
|         Write-Host "##vso[task.logissue type=error;] Unexpected failure occurs while pushing changes." |  | ||||||
|         exit 1 |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -1,126 +0,0 @@ | |||||||
| <# |  | ||||||
| .SYNOPSIS |  | ||||||
| The module that contains a bunch of methods to interact with GitHub API V3 |  | ||||||
| #> |  | ||||||
| class GitHubApi |  | ||||||
| { |  | ||||||
|     [string] $BaseUrl |  | ||||||
|     [string] $RepoOwner |  | ||||||
|     [object] $AuthHeader |  | ||||||
| 
 |  | ||||||
|     GitHubApi( |  | ||||||
|         [string] $AccountName, |  | ||||||
|         [string] $ProjectName, |  | ||||||
|         [string] $AccessToken |  | ||||||
|     ) { |  | ||||||
|         $this.BaseUrl = $this.BuildBaseUrl($AccountName, $ProjectName) |  | ||||||
|         $this.AuthHeader = $this.BuildAuth($AccessToken) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [object] hidden BuildAuth([string]$AccessToken) { |  | ||||||
|         if ([string]::IsNullOrEmpty($AccessToken)) { |  | ||||||
|             return $null |  | ||||||
|         } |  | ||||||
|         $base64AuthInfo = [Convert]::ToBase64String([Text.Encoding]::ASCII.GetBytes("'':${AccessToken}")) |  | ||||||
|         return @{ |  | ||||||
|             Authorization = "Basic ${base64AuthInfo}" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [string] hidden BuildBaseUrl([string]$RepositoryOwner, [string]$RepositoryName) { |  | ||||||
|         return "https://api.github.com/repos/$RepositoryOwner/$RepositoryName" |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [object] CreateNewPullRequest([string]$Title, [string]$Body, [string]$BranchName){ |  | ||||||
|         $requestBody = @{ |  | ||||||
|             title = $Title |  | ||||||
|             body = $Body |  | ||||||
|             head = $BranchName |  | ||||||
|             base = "master" |  | ||||||
|         } | ConvertTo-Json |  | ||||||
| 
 |  | ||||||
|         $url = "pulls" |  | ||||||
|         return $this.InvokeRestMethod($url, 'Post', $null, $requestBody) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [object] GetPullRequest([string]$BranchName, [string]$RepositoryOwner){ |  | ||||||
|         $url = "pulls" |  | ||||||
|         return $this.InvokeRestMethod($url, 'GET', "head=${RepositoryOwner}:$BranchName&base=master", $null) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [object] UpdatePullRequest([string]$Title, [string]$Body, [string]$BranchName, [string]$PullRequestNumber){ |  | ||||||
|         $requestBody = @{ |  | ||||||
|             title = $Title |  | ||||||
|             body = $Body |  | ||||||
|             head = $BranchName |  | ||||||
|             base = "master" |  | ||||||
|         } | ConvertTo-Json |  | ||||||
| 
 |  | ||||||
|         $url = "pulls/$PullRequestNumber" |  | ||||||
|         return $this.InvokeRestMethod($url, 'Post', $null, $requestBody) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [array] GetReleases(){ |  | ||||||
|         $url = "releases" |  | ||||||
|         $releases = @() |  | ||||||
|         $pageNumber = 1 |  | ||||||
|         $releaseNumberLimit = 10000 |  | ||||||
| 
 |  | ||||||
|         while ($releases.Count -le $releaseNumberLimit) |  | ||||||
|         { |  | ||||||
|             $requestParams = "page=${pageNumber}&per_page=100" |  | ||||||
|             [array] $response = $this.InvokeRestMethod($url, 'GET', $requestParams, $null) |  | ||||||
|              |  | ||||||
|             if ($response.Count -eq 0) { |  | ||||||
|                 break |  | ||||||
|             } else { |  | ||||||
|                 $releases += $response |  | ||||||
|                 $pageNumber++ |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return $releases |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [string] hidden BuildUrl([string]$Url, [string]$RequestParams) { |  | ||||||
|         if ([string]::IsNullOrEmpty($RequestParams)) { |  | ||||||
|             return "$($this.BaseUrl)/$($Url)" |  | ||||||
|         } else { |  | ||||||
|             return "$($this.BaseUrl)/$($Url)?$($RequestParams)" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [object] hidden InvokeRestMethod( |  | ||||||
|         [string] $Url, |  | ||||||
|         [string] $Method, |  | ||||||
|         [string] $RequestParams, |  | ||||||
|         [string] $Body |  | ||||||
|     ) { |  | ||||||
|         $requestUrl = $this.BuildUrl($Url, $RequestParams) |  | ||||||
|         $params = @{ |  | ||||||
|             Method = $Method |  | ||||||
|             ContentType = "application/json" |  | ||||||
|             Uri = $requestUrl |  | ||||||
|             Headers = @{} |  | ||||||
|         } |  | ||||||
|         if ($this.AuthHeader) { |  | ||||||
|             $params.Headers += $this.AuthHeader |  | ||||||
|         } |  | ||||||
|         if (![string]::IsNullOrEmpty($Body)) { |  | ||||||
|             $params.Body = $Body |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return Invoke-RestMethod @params |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Get-GitHubApi { |  | ||||||
|     param ( |  | ||||||
|         [string] $AccountName, |  | ||||||
|         [string] $ProjectName, |  | ||||||
|         [string] $AccessToken |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     return [GitHubApi]::New($AccountName, $ProjectName, $AccessToken) |  | ||||||
| } |  | ||||||
| @ -1,32 +0,0 @@ | |||||||
| <# |  | ||||||
| .SYNOPSIS |  | ||||||
| Unpack *.tar file |  | ||||||
| #> |  | ||||||
| function Extract-TarArchive { |  | ||||||
|     param( |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [String]$ArchivePath, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [String]$OutputDirectory |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     Write-Debug "Extract $ArchivePath to $OutputDirectory" |  | ||||||
|     tar -C $OutputDirectory -xzf $ArchivePath --strip 1 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Create-TarArchive { |  | ||||||
|     param( |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [String]$SourceFolder, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [String]$ArchivePath, |  | ||||||
|         [string]$CompressionType = "gz" |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     $CompressionTypeArgument = If ([string]::IsNullOrWhiteSpace($CompressionType)) { "" } else { "--${CompressionType}" } |  | ||||||
| 
 |  | ||||||
|     Push-Location $SourceFolder |  | ||||||
|     Write-Debug "tar -c $CompressionTypeArgument -f $ArchivePath ." |  | ||||||
|     tar -c $CompressionTypeArgument -f $ArchivePath . |  | ||||||
|     Pop-Location |  | ||||||
| } |  | ||||||
| @ -1,158 +0,0 @@ | |||||||
| <# |  | ||||||
| .SYNOPSIS |  | ||||||
| Generate versions manifest based on repository releases |  | ||||||
| 
 |  | ||||||
| .DESCRIPTION |  | ||||||
| Versions manifest is needed to find the latest assets for particular version of tool |  | ||||||
| .PARAMETER GitHubRepositoryOwner |  | ||||||
| Required parameter. The organization which tool repository belongs |  | ||||||
| .PARAMETER GitHubRepositoryName |  | ||||||
| Optional parameter. The name of tool repository |  | ||||||
| .PARAMETER GitHubAccessToken |  | ||||||
| Required parameter. PAT Token to overcome GitHub API Rate limit |  | ||||||
| .PARAMETER OutputFile |  | ||||||
| Required parameter. File "*.json" where generated results will be saved |  | ||||||
| .PARAMETER PlatformMapFile |  | ||||||
| Optional parameter. Path to the json file with platform map |  | ||||||
| Structure example: |  | ||||||
| { |  | ||||||
|     "macos-1014": [ |  | ||||||
|         { |  | ||||||
|             "platform": "darwin", |  | ||||||
|             "platform_version": "10.14" |  | ||||||
|         }, ... |  | ||||||
|     ], ... |  | ||||||
| } |  | ||||||
| #> |  | ||||||
| 
 |  | ||||||
| param ( |  | ||||||
|     [Parameter(Mandatory)] [string] $GitHubRepositoryOwner, |  | ||||||
|     [Parameter(Mandatory)] [string] $GitHubRepositoryName, |  | ||||||
|     [Parameter(Mandatory)] [string] $GitHubAccessToken, |  | ||||||
|     [Parameter(Mandatory)] [string] $OutputFile, |  | ||||||
|     [string] $PlatformMapFile |  | ||||||
| ) |  | ||||||
| 
 |  | ||||||
| Import-Module (Join-Path $PSScriptRoot "../github/github-api.psm1") |  | ||||||
| 
 |  | ||||||
| if ($PlatformMapFile -and (Test-Path $PlatformMapFile)) { |  | ||||||
|     $PlatformMap = Get-Content $PlatformMapFile -Raw | ConvertFrom-Json -AsHashtable |  | ||||||
| } else { |  | ||||||
|     $PlatformMap = @{} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Get-FileNameWithoutExtension { |  | ||||||
|     param ( |  | ||||||
|         [Parameter(Mandatory)][string]$Filename |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     if ($Filename.EndsWith(".tar.gz")) { |  | ||||||
|         $Filename = [IO.path]::GetFileNameWithoutExtension($Filename) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return [IO.path]::GetFileNameWithoutExtension($Filename) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function New-AssetItem { |  | ||||||
|     param ( |  | ||||||
|         [Parameter(Mandatory)][string]$Filename, |  | ||||||
|         [Parameter(Mandatory)][string]$DownloadUrl, |  | ||||||
|         [Parameter(Mandatory)][string]$Arch, |  | ||||||
|         [Parameter(Mandatory)][string]$Platform, |  | ||||||
|         [string]$PlatformVersion |  | ||||||
|     ) |  | ||||||
|     $asset = New-Object PSObject |  | ||||||
| 
 |  | ||||||
|     $asset | Add-Member -Name "filename" -Value $Filename -MemberType NoteProperty |  | ||||||
|     $asset | Add-Member -Name "arch" -Value $Arch -MemberType NoteProperty |  | ||||||
|     $asset | Add-Member -Name "platform" -Value $Platform -MemberType NoteProperty |  | ||||||
|     if ($PlatformVersion) { $asset | Add-Member -Name "platform_version" -Value $PlatformVersion -MemberType NoteProperty } |  | ||||||
|     $asset | Add-Member -Name "download_url" -Value $DownloadUrl -MemberType NoteProperty |  | ||||||
| 
 |  | ||||||
|     return $asset |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Build-AssetsList { |  | ||||||
|     param ( |  | ||||||
|         [AllowEmptyCollection()] |  | ||||||
|         [Parameter(Mandatory)][array]$ReleaseAssets |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|      |  | ||||||
|     $assets = @() |  | ||||||
|     foreach($releaseAsset in $ReleaseAssets) { |  | ||||||
|         $filename = Get-FileNameWithoutExtension -Filename $releaseAsset.name |  | ||||||
|         $parts = $filename.Split("-") |  | ||||||
|         $arch = $parts[-1] |  | ||||||
|         $buildPlatform = [string]::Join("-", $parts[2..($parts.Length-2)]) |  | ||||||
| 
 |  | ||||||
|         if ($PlatformMap[$buildPlatform]) { |  | ||||||
|             $PlatformMap[$buildPlatform] | ForEach-Object { |  | ||||||
|                 $assets += New-AssetItem -Filename $releaseAsset.name ` |  | ||||||
|                                          -DownloadUrl $releaseAsset.browser_download_url ` |  | ||||||
|                                          -Arch $arch ` |  | ||||||
|                                          -Platform $_.platform ` |  | ||||||
|                                          -PlatformVersion $_.platform_version |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|         } else { |  | ||||||
|             $assets += New-AssetItem -Filename $releaseAsset.name ` |  | ||||||
|                                      -DownloadUrl $releaseAsset.browser_download_url ` |  | ||||||
|                                      -Arch $arch ` |  | ||||||
|                                      -Platform $buildPlatform |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return $assets |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Get-VersionFromRelease { |  | ||||||
|     param ( |  | ||||||
|         [Parameter(Mandatory)][object]$Release |  | ||||||
|     ) |  | ||||||
|     # Release name can contain additional information after ':' so filter it |  | ||||||
|     [string]$releaseName = $Release.name.Split(':')[0] |  | ||||||
|     [Version]$version = $null |  | ||||||
|     if (![Version]::TryParse($releaseName, [ref]$version)) { |  | ||||||
|         throw "Release '$($Release.id)' has invalid title '$($Release.name)'. It can't be parsed as version. ( $($Release.html_url) )" |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return $version |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Build-VersionsManifest { |  | ||||||
|     param ( |  | ||||||
|         [Parameter(Mandatory)][array]$Releases |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     $Releases = $Releases | Sort-Object -Property "published_at" -Descending |  | ||||||
| 
 |  | ||||||
|     $versionsHash = @{} |  | ||||||
|     foreach ($release in $Releases) { |  | ||||||
|         if (($release.draft -eq $true) -or ($release.prerelease -eq $true)) { |  | ||||||
|             continue |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         [Version]$version = Get-VersionFromRelease $release |  | ||||||
|         $versionKey = $version.ToString() |  | ||||||
| 
 |  | ||||||
|         if ($versionsHash.ContainsKey($versionKey)) { |  | ||||||
|             continue |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         $versionsHash.Add($versionKey, [PSCustomObject]@{ |  | ||||||
|             version = $versionKey |  | ||||||
|             stable = $true |  | ||||||
|             release_url = $release.html_url |  | ||||||
|             files = Build-AssetsList $release.assets |  | ||||||
|         }) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     # Sort versions by descending |  | ||||||
|     return $versionsHash.Values | Sort-Object -Property @{ Expression = { [Version]$_.version }; Descending = $true } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| $gitHubApi = Get-GitHubApi -AccountName $GitHubRepositoryOwner -ProjectName $GitHubRepositoryName -AccessToken $GitHubAccessToken |  | ||||||
| $releases = $gitHubApi.GetReleases() |  | ||||||
| $versionIndex = Build-VersionsManifest $releases |  | ||||||
| $versionIndex | ConvertTo-Json -Depth 5 | Out-File $OutputFile -Encoding UTF8NoBOM -Force |  | ||||||
| @ -1,33 +0,0 @@ | |||||||
| <# |  | ||||||
| .SYNOPSIS |  | ||||||
| Pester extension that allows to run command and validate exit code |  | ||||||
| .EXAMPLE |  | ||||||
| "python file.py" | Should -ReturnZeroExitCode |  | ||||||
| #> |  | ||||||
| function ShouldReturnZeroExitCode { |  | ||||||
|     Param( |  | ||||||
|         [Parameter (Mandatory = $true)] [ValidateNotNullOrEmpty()] |  | ||||||
|         [String]$ActualValue, |  | ||||||
|         [switch]$Negate |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     Write-Host "Run command '${ActualValue}'" |  | ||||||
|     Invoke-Expression -Command $ActualValue | ForEach-Object { Write-Host $_ } |  | ||||||
|     $actualExitCode = $LASTEXITCODE |  | ||||||
| 
 |  | ||||||
|     [bool]$succeeded = $actualExitCode -eq 0 |  | ||||||
|     if ($Negate) { $succeeded = -not $succeeded } |  | ||||||
| 
 |  | ||||||
|     if (-not $succeeded) |  | ||||||
|     { |  | ||||||
|         $failureMessage = "Command '${ActualValue}' has finished with exit code ${actualExitCode}" |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return New-Object PSObject -Property @{ |  | ||||||
|         Succeeded      = $succeeded |  | ||||||
|         FailureMessage = $failureMessage |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| Add-AssertionOperator -Name ReturnZeroExitCode ` |  | ||||||
|                     -Test  $function:ShouldReturnZeroExitCode |  | ||||||
| @ -1,34 +0,0 @@ | |||||||
| <# |  | ||||||
| .SYNOPSIS |  | ||||||
| Unpack *.7z file |  | ||||||
| #> |  | ||||||
| function Extract-SevenZipArchive { |  | ||||||
|     param( |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [String]$ArchivePath, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [String]$OutputDirectory |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     Write-Debug "Extract $ArchivePath to $OutputDirectory" |  | ||||||
|     7z x $ArchivePath -o"$OutputDirectory" -y | Out-Null |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function Create-SevenZipArchive { |  | ||||||
|     param( |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [String]$SourceFolder, |  | ||||||
|         [Parameter(Mandatory=$true)] |  | ||||||
|         [String]$ArchivePath, |  | ||||||
|         [String]$ArchiveType = "zip", |  | ||||||
|         [String]$CompressionLevel = 5 |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     $ArchiveTypeArgument = "-t${ArchiveType}" |  | ||||||
|     $CompressionLevelArgument = "-mx=${CompressionLevel}" |  | ||||||
|      |  | ||||||
|     Push-Location $SourceFolder |  | ||||||
|     Write-Debug "7z a $ArchiveTypeArgument $CompressionLevelArgument $ArchivePath @$SourceFolder" |  | ||||||
|     7z a $ArchiveTypeArgument $CompressionLevelArgument $ArchivePath $SourceFolder\* |  | ||||||
|     Pop-Location |  | ||||||
| } |  | ||||||
| @ -1,10 +1,11 @@ | |||||||
| set -e | set -e | ||||||
| 
 | 
 | ||||||
| NODE_VERSION={0} | NODE_VERSION={0} | ||||||
|  | ARCH={1} | ||||||
| 
 | 
 | ||||||
| NODE_TOOLCACHE_PATH=$AGENT_TOOLSDIRECTORY/node | NODE_TOOLCACHE_PATH=$AGENT_TOOLSDIRECTORY/node | ||||||
| NODE_TOOLCACHE_VERSION_PATH=$NODE_TOOLCACHE_PATH/$NODE_VERSION | NODE_TOOLCACHE_VERSION_PATH=$NODE_TOOLCACHE_PATH/$NODE_VERSION | ||||||
| NODE_TOOLCACHE_VERSION_ARCH_PATH=$NODE_TOOLCACHE_VERSION_PATH/x64 | NODE_TOOLCACHE_VERSION_ARCH_PATH=$NODE_TOOLCACHE_VERSION_PATH/$ARCH | ||||||
| 
 | 
 | ||||||
| echo "Check if Node.js hostedtoolcache folder exist..." | echo "Check if Node.js hostedtoolcache folder exist..." | ||||||
| if [ ! -d $NODE_TOOLCACHE_PATH ]; then | if [ ! -d $NODE_TOOLCACHE_PATH ]; then | ||||||
| @ -22,4 +23,4 @@ cp -R ./* $NODE_TOOLCACHE_VERSION_ARCH_PATH | |||||||
| rm $NODE_TOOLCACHE_VERSION_ARCH_PATH/setup.sh | rm $NODE_TOOLCACHE_VERSION_ARCH_PATH/setup.sh | ||||||
| 
 | 
 | ||||||
| echo "Create complete file" | echo "Create complete file" | ||||||
| touch $NODE_TOOLCACHE_VERSION_PATH/x64.complete | touch $NODE_TOOLCACHE_VERSION_PATH/$ARCH.complete | ||||||
|  | |||||||
| @ -1,43 +1,75 @@ | |||||||
| param ( | Import-Module (Join-Path $PSScriptRoot "../helpers/pester-extensions.psm1") | ||||||
|     [Version] [Parameter (Mandatory = $true)] [ValidateNotNullOrEmpty()] |  | ||||||
|     $Version |  | ||||||
| ) |  | ||||||
| 
 | 
 | ||||||
| Import-Module (Join-Path $PSScriptRoot "../helpers/packages-generation/pester-extensions.psm1") |  | ||||||
| 
 | 
 | ||||||
| function Get-UseNodeLogs { |  | ||||||
|     $logsFolderPath = Join-Path -Path $env:AGENT_HOMEDIRECTORY -ChildPath "_diag" | Join-Path -ChildPath "pages" |  | ||||||
| 
 |  | ||||||
|     $useNodeLogFile = Get-ChildItem -Path $logsFolderPath | Where-Object { |  | ||||||
|         $logContent = Get-Content $_.Fullname -Raw |  | ||||||
|         return $logContent -match "Use Node" |  | ||||||
|     } | Select-Object -First 1 |  | ||||||
|     return $useNodeLogFile.Fullname |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| Describe "Node.js" { | Describe "Node.js" { | ||||||
|  | 
 | ||||||
|  |     BeforeAll { | ||||||
|  |         function Get-UseNodeLogs { | ||||||
|  |             # GitHub Windows images don't have `HOME` variable | ||||||
|  |             $homeDir = $env:HOME ?? $env:HOMEDRIVE | ||||||
|  |              | ||||||
|  |             $possiblePaths = @( | ||||||
|  |                 Join-Path -Path $homeDir -ChildPath "actions-runner/cached/_diag/pages" | ||||||
|  |                 Join-Path -Path $homeDir -ChildPath "runners/*/_diag/pages" | ||||||
|  |             ) | ||||||
|  |              | ||||||
|  |             $logsFolderPath = $possiblePaths | Where-Object { Test-Path $_ } | Select-Object -First 1 | ||||||
|  |             $resolvedPath = Resolve-Path -Path $logsFolderPath -ErrorAction SilentlyContinue | ||||||
|  | 
 | ||||||
|  |             if ($resolvedPath -and -not [string]::IsNullOrEmpty($resolvedPath.Path) -and (Test-Path $resolvedPath.Path)) {                 | ||||||
|  |                 $useNodeLogFile = Get-ChildItem -Path $resolvedPath | Where-Object { | ||||||
|  |                             $logContent = Get-Content $_.Fullname -Raw | ||||||
|  |                             return $logContent -match "setup-node@v"                      | ||||||
|  |                     } | Select-Object -First 1                 | ||||||
|  |                  | ||||||
|  |               # Return the file name if a match is found | ||||||
|  |                 if ($useNodeLogFile) { | ||||||
|  |                     return $useNodeLogFile.FullName | ||||||
|  |                 } else { | ||||||
|  |                     Write-Error "No matching log file found in the specified path: $($resolvedPath.Path)" | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 Write-Error "The provided logs folder path is null, empty, or does not exist: $logsFolderPath" | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |      | ||||||
|     It "is available" { |     It "is available" { | ||||||
|         "node --version" | Should -ReturnZeroExitCode |         "node --version" | Should -ReturnZeroExitCode | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     It "version is correct" { |     It "version is correct" { | ||||||
|         $versionOutput = Invoke-Expression "node --version" |         $versionOutput = Invoke-Expression "node --version" | ||||||
|         $versionOutput | Should -Match $Version |         $versionOutput | Should -Match $env:VERSION | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     It "is used from tool-cache" { |     It "is used from tool-cache" { | ||||||
|         $nodePath = (Get-Command "node").Path |         $nodePath = (Get-Command "node").Path | ||||||
|         $nodePath | Should -Not -BeNullOrEmpty |         $nodePath | Should -Not -BeNullOrEmpty | ||||||
|         $expectedPath = Join-Path -Path $env:AGENT_TOOLSDIRECTORY -ChildPath "node" |          | ||||||
|  |         # GitHub Windows images don't have `AGENT_TOOLSDIRECTORY` variable | ||||||
|  |         $toolcacheDir = $env:AGENT_TOOLSDIRECTORY ?? $env:RUNNER_TOOL_CACHE | ||||||
|  |         $expectedPath = Join-Path -Path $toolcacheDir -ChildPath "node" | ||||||
|         $nodePath.startsWith($expectedPath) | Should -BeTrue -Because "'$nodePath' is not started with '$expectedPath'" |         $nodePath.startsWith($expectedPath) | Should -BeTrue -Because "'$nodePath' is not started with '$expectedPath'" | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     It "cached version is used without downloading" { |      It "cached version is used without downloading" { | ||||||
|         # Analyze output of previous steps to check if Node.js was consumed from cache or downloaded | 
 | ||||||
|         $useNodeLogFile = Get-UseNodeLogs |        if ($env:RUNNER_TYPE -eq "self-hosted") { | ||||||
|         $useNodeLogFile | Should -Exist |             # Get the installed version of Node.js | ||||||
|         $useNodeLogContent = Get-Content $useNodeLogFile -Raw |             $nodeVersion = Invoke-Expression "node --version" | ||||||
|         $useNodeLogContent | Should -Match "Found tool in cache" |             # Check if Node.js is installed | ||||||
|  |             $nodeVersion | Should -Not -BeNullOrEmpty | ||||||
|  |             # Check if the installed version of Node.js is the expected version | ||||||
|  |             $nodeVersion | Should -Match $env:VERSION | ||||||
|  |         }else { | ||||||
|  |             # Analyze output of previous steps to check if Node.js was consumed from cache or downloaded | ||||||
|  |             $useNodeLogFile = Get-UseNodeLogs | ||||||
|  |             $useNodeLogFile | Should -Exist | ||||||
|  |             $useNodeLogContent = Get-Content $useNodeLogFile -Raw | ||||||
|  |             $useNodeLogContent | Should -Match "Found in cache" | ||||||
|  |         }  | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     It "Run simple code" { |     It "Run simple code" { | ||||||
|  | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user