Compare commits
	
		
			105 Commits 
		
	
	
	| Author | SHA1 | Date | 
|---|---|---|
|  | cfdebb67c1 | |
|  | ab64768ceb | |
|  | 00fa0d3772 | |
|  | d587557287 | |
|  | 6e0bf17345 | |
|  | 2822af505e | |
|  | 8ec161b981 | |
|  | 88a89c94a0 | |
|  | f7f3f70c57 | |
|  | c6c8236152 | |
|  | 9e91ca9749 | |
|  | 9cd9f7e0e7 | |
|  | 0b822c28fb | |
|  | 65607f9ebb | |
|  | a1ebdb7abd | |
|  | 3b26780ddf | |
|  | 819c2511e0 | |
|  | 81ed299e10 | |
|  | 9d6bdcbdc5 | |
|  | 3ae9ec8536 | |
|  | 83c85328dd | |
|  | b11b6a6f2c | |
|  | 461ecf7cea | |
|  | f2250e958e | |
|  | dd427466ce | |
|  | 0c16aab353 | |
|  | fc0a52b805 | |
|  | e820c9ce7b | |
|  | f4d2cceeb5 | |
|  | 4ae184ca89 | |
|  | 082ea39498 | |
|  | e73b48fb38 | |
|  | 2800d14403 | |
|  | 5ba81971e2 | |
|  | ff23166e30 | |
|  | 9406bce875 | |
|  | bbd713b05a | |
|  | 96cfb845ae | |
|  | 8ca1282c9e | |
|  | 8da77ace98 | |
|  | 2afd9cd86f | |
|  | caa0a81b47 | |
|  | 7afabe74da | |
|  | 4c4611c021 | |
|  | 6419c8742b | |
|  | a13443a746 | |
|  | 2190fd5667 | |
|  | a073719c29 | |
|  | ff551fe06d | |
|  | 43e128087c | |
|  | 5f7828371a | |
|  | b567eb9057 | |
|  | f43335663c | |
|  | 21da302529 | |
|  | 857a41e877 | |
|  | 3032a4ab97 | |
|  | 7abb3a409d | |
|  | 309d668d63 | |
|  | ef38f5a88a | |
|  | 3de97ed24a | |
|  | c7a5c0640a | |
|  | 7f2782d3ed | |
|  | f58ac29d5b | |
|  | 0efa1855a3 | |
|  | 8c9dcf076d | |
|  | f35829a9d4 | |
|  | fb5b03c3b8 | |
|  | 584c0366c6 | |
|  | 28147e5e1b | |
|  | 00c5685d03 | |
|  | e334dc785a | |
|  | a45155c578 | |
|  | f68f14ff9f | |
|  | 9329b7369c | |
|  | 5bd589e19f | |
|  | f03bee03f6 | |
|  | 2a32a9d870 | |
|  | d6ac850da1 | |
|  | f300dd27bb | |
|  | 4cca069ebb | |
|  | 96555a0945 | |
|  | 4cb3e593f5 | |
|  | 9f79830454 | |
|  | f054aaa35d | |
|  | 0756841374 | |
|  | 579daa93a6 | |
|  | 1adfabf512 | |
|  | cdee7d1d9a | |
|  | fb633660a4 | |
|  | c15f955669 | |
|  | 2684be2dee | |
|  | db2d8b6dbd | |
|  | c068855899 | |
|  | 5b8bf1c8b7 | |
|  | 47b25cf3b1 | |
|  | 36891ec921 | |
|  | 2652cb78a7 | |
|  | f77696efae | |
|  | 4556fc4ff1 | |
|  | d066039c26 | |
|  | 8abce48a48 | |
|  | 5ae03dfef6 | |
|  | 4be5d2ddf4 | |
|  | a61c02481f | |
|  | 2b399b2641 | 
|  | @ -1,20 +1,82 @@ | |||
| { | ||||
|   "plugins": ["jest", "@typescript-eslint", "prettier", "unicorn"], | ||||
|   "extends": ["plugin:unicorn/recommended", "plugin:github/recommended", "prettier"], | ||||
|   "extends": ["plugin:unicorn/recommended", "plugin:github/recommended", "plugin:prettier/recommended"], | ||||
|   "parser": "@typescript-eslint/parser", | ||||
|   "parserOptions": { | ||||
|     "ecmaVersion": 9, | ||||
|     "sourceType": "module" | ||||
|     "ecmaVersion": 2020, | ||||
|     "sourceType": "module", | ||||
|     "extraFileExtensions": [".mjs"], | ||||
|     "ecmaFeatures": { | ||||
|       "impliedStrict": true | ||||
|     }, | ||||
|     "project": "./tsconfig.json" | ||||
|   }, | ||||
|   "env": { | ||||
|     "node": true, | ||||
|     "es6": true, | ||||
|     "jest/globals": true | ||||
|     "jest/globals": true, | ||||
|     "es2020": true | ||||
|   }, | ||||
|   "rules": { | ||||
|     // Error out for code formatting errors | ||||
|     "prettier/prettier": "error", | ||||
|     "import/no-extraneous-dependencies": 0, | ||||
|     // Namespaces or sometimes needed | ||||
|     "import/no-namespace": "off", | ||||
|     "no-undef": "off" // TODO: REMOVE THIS LINE WHEN UPDATING ESLINT RULES | ||||
|     // Properly format comments | ||||
|     "spaced-comment": ["error", "always"], | ||||
|     "lines-around-comment": [ | ||||
|       "error", | ||||
|       { | ||||
|         "beforeBlockComment": true, | ||||
|         "beforeLineComment": true, | ||||
|         "allowBlockStart": true, | ||||
|         "allowObjectStart": true, | ||||
|         "allowArrayStart": true, | ||||
|         "allowClassStart": true, | ||||
|         "ignorePattern": "pragma|ts-ignore" | ||||
|       } | ||||
|     ], | ||||
|     // Mandatory spacing | ||||
|     "padding-line-between-statements": [ | ||||
|       "error", | ||||
|       { | ||||
|         "blankLine": "always", | ||||
|         "prev": "*", | ||||
|         "next": "return" | ||||
|       }, | ||||
|       { | ||||
|         "blankLine": "always", | ||||
|         "prev": "directive", | ||||
|         "next": "*" | ||||
|       }, | ||||
|       { | ||||
|         "blankLine": "any", | ||||
|         "prev": "directive", | ||||
|         "next": "directive" | ||||
|       } | ||||
|     ], | ||||
|     // Enforce camelCase | ||||
|     "camelcase": "error", | ||||
|     // Allow forOfStatements | ||||
|     "no-restricted-syntax": ["error", "ForInStatement", "LabeledStatement", "WithStatement"], | ||||
|     // Continue is viable in forOf loops in generators | ||||
|     "no-continue": "off", | ||||
|     // From experience, named exports are almost always desired. I got tired of this rule | ||||
|     "import/prefer-default-export": "off", | ||||
|     // Unused vars are useful to keep method signatures consistent and documented | ||||
|     "@typescript-eslint/no-unused-vars": "off", | ||||
|     // For this project only use kebab-case | ||||
|     "unicorn/filename-case": [ | ||||
|       "error", | ||||
|       { | ||||
|         "cases": { | ||||
|           "kebabCase": true | ||||
|         } | ||||
|       } | ||||
|     ], | ||||
|     // Allow Array.from(set) mitigate TS2569 which would require '--downlevelIteration' | ||||
|     "unicorn/prefer-spread": "off", | ||||
|     // Temp disable to prevent mixing changes with other PRs | ||||
|     "i18n-text/no-en": "off" | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -1,8 +1,8 @@ | |||
| # These are supported funding model platforms | ||||
| 
 | ||||
| github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2] | ||||
| github: game-ci | ||||
| patreon: # Replace with a single Patreon username | ||||
| open_collective: game-ci | ||||
| open_collective: # replace with a single OpenCollective username | ||||
| ko_fi: # Replace with a single Ko-fi username | ||||
| tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel | ||||
| community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry | ||||
|  |  | |||
|  | @ -2,10 +2,28 @@ | |||
| 
 | ||||
| - ... | ||||
| 
 | ||||
| #### Related Issues | ||||
| 
 | ||||
| - ... | ||||
| 
 | ||||
| #### Related PRs | ||||
| 
 | ||||
| - ... | ||||
| 
 | ||||
| #### Successful Workflow Run Link | ||||
| 
 | ||||
| PRs don't have access to secrets so you will need to provide a link to a successful run of the workflows from your own | ||||
| repo. | ||||
| 
 | ||||
| - ... | ||||
| 
 | ||||
| #### Checklist | ||||
| 
 | ||||
| <!-- please check all items and add your own --> | ||||
| 
 | ||||
| - [x] Read the contribution [guide](../CONTRIBUTING.md) and accept the [code](../CODE_OF_CONDUCT.md) of conduct | ||||
| - [x] Read the contribution [guide](https://github.com/game-ci/unity-builder/blob/main/CONTRIBUTING.md) and accept the | ||||
|       [code](https://github.com/game-ci/unity-builder/blob/main/CODE_OF_CONDUCT.md) of conduct | ||||
| - [ ] Docs (If new inputs or outputs have been added or changes to behavior that should be documented. Please make a PR | ||||
|       in the [documentation repo](https://github.com/game-ci/documentation)) | ||||
| - [ ] Readme (updated or not needed) | ||||
| - [ ] Tests (added, updated or not needed) | ||||
|  |  | |||
|  | @ -13,7 +13,7 @@ jobs: | |||
|         id: requestActivationFile | ||||
|         uses: game-ci/unity-request-activation-file@v2.0-alpha-1 | ||||
|       - name: Upload activation file | ||||
|         uses: actions/upload-artifact@v2 | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: ${{ steps.requestActivationFile.outputs.filePath }} | ||||
|           path: ${{ steps.requestActivationFile.outputs.filePath }} | ||||
|  |  | |||
|  | @ -0,0 +1,87 @@ | |||
| name: Builds - MacOS | ||||
| 
 | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   push: | ||||
| 
 | ||||
| concurrency: | ||||
|   group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} | ||||
|   cancel-in-progress: true | ||||
| 
 | ||||
| jobs: | ||||
|   buildForAllPlatformsMacOS: | ||||
|     name: ${{ matrix.targetPlatform }} on ${{ matrix.unityVersion }} | ||||
|     runs-on: macos-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         projectPath: | ||||
|           - test-project | ||||
|         unityVersion: | ||||
|           - 2021.3.45f1 | ||||
|           - 2022.3.13f1 | ||||
|           - 2023.2.2f1 | ||||
|         targetPlatform: | ||||
|           - StandaloneOSX # Build a MacOS executable | ||||
|           - iOS # Build an iOS executable | ||||
|         include: | ||||
|           # Additionally test enableGpu build for a standalone windows target | ||||
|           - unityVersion: 6000.0.36f1 | ||||
|             targetPlatform: StandaloneOSX | ||||
|           - unityVersion: 6000.0.36f1 | ||||
|             targetPlatform: StandaloneOSX | ||||
|             buildProfile: 'Assets/Settings/Build Profiles/Sample macOS Build Profile.asset' | ||||
| 
 | ||||
|     steps: | ||||
|       ########################### | ||||
|       #         Checkout        # | ||||
|       ########################### | ||||
|       - uses: actions/checkout@v4 | ||||
|         with: | ||||
|           lfs: true | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Cache          # | ||||
|       ########################### | ||||
|       - uses: actions/cache@v4 | ||||
|         with: | ||||
|           path: ${{ matrix.projectPath }}/Library | ||||
|           key: Library-${{ matrix.projectPath }}-macos-${{ matrix.targetPlatform }} | ||||
|           restore-keys: | | ||||
|             Library-${{ matrix.projectPath }}-macos- | ||||
|             Library- | ||||
| 
 | ||||
|       ########################### | ||||
|       #  Set Scripting Backend  # | ||||
|       ########################### | ||||
|       - name: Set Scripting Backend To il2cpp | ||||
|         run: | | ||||
|           mv -f "./test-project/ProjectSettings/ProjectSettingsIl2cpp.asset" "./test-project/ProjectSettings/ProjectSettings.asset" | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Build          # | ||||
|       ########################### | ||||
|       - uses: ./ | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|           UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
|           UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
|         with: | ||||
|           buildName: 'GameCI Test Build' | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           buildProfile: ${{ matrix.buildProfile }} | ||||
|           customParameters: -profile SomeProfile -someBoolean -someValue exampleValue | ||||
|           # We use dirty build because we are replacing the default project settings file above | ||||
|           allowDirtyBuild: true | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Upload         # | ||||
|       ########################### | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: Build ${{ matrix.targetPlatform }} on MacOS (${{ matrix.unityVersion }})${{ matrix.buildProfile && '  With Build Profile' || '' }} | ||||
|           path: build | ||||
|           retention-days: 14 | ||||
|  | @ -0,0 +1,206 @@ | |||
| name: Builds - Ubuntu | ||||
| 
 | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   push: | ||||
| 
 | ||||
| concurrency: | ||||
|   group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} | ||||
|   cancel-in-progress: true | ||||
| 
 | ||||
| env: | ||||
|   UNITY_LICENSE: | ||||
|     "<?xml version=\"1.0\" encoding=\"UTF-8\"?><root>\n    <License | ||||
|     id=\"Terms\">\n        <MachineBindings>\n            <Binding Key=\"1\" | ||||
|     Value=\"576562626572264761624c65526f7578\"/>\n            <Binding Key=\"2\" | ||||
|     Value=\"576562626572264761624c65526f7578\"/>\n        </MachineBindings>\n        <MachineID | ||||
|     Value=\"D7nTUnjNAmtsUMcnoyrqkgIbYdM=\"/>\n        <SerialHash | ||||
|     Value=\"2033b8ac3e6faa3742ca9f0bfae44d18f2a96b80\"/>\n        <Features>\n            <Feature | ||||
|     Value=\"33\"/>\n            <Feature Value=\"1\"/>\n            <Feature Value=\"12\"/>\n            <Feature | ||||
|     Value=\"2\"/>\n            <Feature Value=\"24\"/>\n            <Feature Value=\"3\"/>\n            <Feature | ||||
|     Value=\"36\"/>\n            <Feature Value=\"17\"/>\n            <Feature Value=\"19\"/>\n            <Feature | ||||
|     Value=\"62\"/>\n        </Features>\n        <DeveloperData | ||||
|     Value=\"AQAAAEY0LUJHUlgtWEQ0RS1aQ1dWLUM1SlctR0RIQg==\"/>\n        <SerialMasked | ||||
|     Value=\"F4-BGRX-XD4E-ZCWV-C5JW-XXXX\"/>\n        <StartDate Value=\"2021-02-08T00:00:00\"/>\n        <UpdateDate | ||||
|     Value=\"2021-02-09T00:34:57\"/>\n        <InitialActivationDate | ||||
|     Value=\"2021-02-08T00:34:56\"/>\n        <LicenseVersion Value=\"6.x\"/>\n        <ClientProvidedVersion | ||||
|     Value=\"2018.4.30f1\"/>\n        <AlwaysOnline Value=\"false\"/>\n        <Entitlements>\n            <Entitlement | ||||
|     Ns=\"unity_editor\" Tag=\"UnityPersonal\" Type=\"EDITOR\" | ||||
|     ValidTo=\"9999-12-31T00:00:00\"/>\n            <Entitlement Ns=\"unity_editor\" Tag=\"DarkSkin\" | ||||
|     Type=\"EDITOR_FEATURE\" ValidTo=\"9999-12-31T00:00:00\"/>\n        </Entitlements>\n    </License>\n<Signature | ||||
|     xmlns=\"http://www.w3.org/2000/09/xmldsig#\"><SignedInfo><CanonicalizationMethod | ||||
|     Algorithm=\"http://www.w3.org/TR/2001/REC-xml-c14n-20010315#WithComments\"/><SignatureMethod | ||||
|     Algorithm=\"http://www.w3.org/2000/09/xmldsig#rsa-sha1\"/><Reference URI=\"#Terms\"><Transforms><Transform | ||||
|     Algorithm=\"http://www.w3.org/2000/09/xmldsig#enveloped-signature\"/></Transforms><DigestMethod | ||||
|     Algorithm=\"http://www.w3.org/2000/09/xmldsig#sha1\"/><DigestValue>m0Db8UK+ktnOLJBtHybkfetpcKo=</DigestValue></Reference></SignedInfo><SignatureValue>o/pUbSQAukz7+ZYAWhnA0AJbIlyyCPL7bKVEM2lVqbrXt7cyey+umkCXamuOgsWPVUKBMkXtMH8L\n5etLmD0getWIhTGhzOnDCk+gtIPfL4jMo9tkEuOCROQAXCci23VFscKcrkB+3X6h4wEOtA2APhOY\nB+wvC794o8/82ffjP79aVAi57rp3Wmzx+9pe9yMwoJuljAy2sc2tIMgdQGWVmOGBpQm3JqsidyzI\nJWG2kjnc7pDXK9pwYzXoKiqUqqrut90d+kQqRyv7MSZXR50HFqD/LI69h68b7P8Bjo3bPXOhNXGR\n9YCoemH6EkfCJxp2gIjzjWW+l2Hj2EsFQi8YXw==</SignatureValue></Signature></root>" | ||||
| 
 | ||||
| jobs: | ||||
|   buildForAllPlatformsUbuntu: | ||||
|     name: | ||||
|       "${{ matrix.targetPlatform }} on ${{ matrix.unityVersion}}${{startsWith(matrix.buildProfile, 'Assets') && ' (via Build Profile)' || '' }}" | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         providerStrategy: | ||||
|           # - local-docker | ||||
|           - local | ||||
|         projectPath: | ||||
|           - test-project | ||||
|         unityVersion: | ||||
|           - 2021.3.32f1 | ||||
|           - 2022.3.13f1 | ||||
|           - 2023.2.2f1 | ||||
|         targetPlatform: | ||||
|           - StandaloneOSX # Build a macOS standalone (Intel 64-bit) with mono backend. | ||||
|           - StandaloneWindows64 # Build a Windows 64-bit standalone with mono backend. | ||||
|           - StandaloneLinux64 # Build a Linux 64-bit standalone with mono/il2cpp backend. | ||||
|           - iOS # Build an iOS project. | ||||
|           - Android # Build an Android .apk. | ||||
|           - WebGL # WebGL. | ||||
|         buildWithIl2cpp: | ||||
|           - false | ||||
|           - true | ||||
|         additionalParameters: | ||||
|           - -param value | ||||
|           - -standaloneBuildSubtarget Server | ||||
|         # Skipping configurations that are not supported | ||||
|         exclude: | ||||
|           # No il2cpp support on Linux Host | ||||
|           - targetPlatform: StandaloneOSX | ||||
|             buildWithIl2cpp: true | ||||
|           - targetPlatform: StandaloneWindows64 | ||||
|             buildWithIl2cpp: true | ||||
|           # Only builds with Il2cpp | ||||
|           - targetPlatform: iOS | ||||
|             buildWithIl2cpp: false | ||||
|           - targetPlatform: Android | ||||
|             buildWithIl2cpp: false | ||||
|           - targetPlatform: WebGL | ||||
|             buildWithIl2cpp: false | ||||
|           # No dedicated server support | ||||
|           - targetPlatform: WebGL | ||||
|             additionalParameters: -standaloneBuildSubtarget Server | ||||
|           - targetPlatform: Android | ||||
|             additionalParameters: -standaloneBuildSubtarget Server | ||||
|           - targetPlatform: iOS | ||||
|             additionalParameters: -standaloneBuildSubtarget Server | ||||
|           # No dedicated server support on Linux Host | ||||
|           - targetPlatform: StandaloneOSX | ||||
|             additionalParameters: -standaloneBuildSubtarget Server | ||||
|           # No il2cpp dedicated server support on Linux Host | ||||
|           - targetPlatform: StandaloneWindows64 | ||||
|             additionalParameters: -standaloneBuildSubtarget Server | ||||
|             buildWithIl2cpp: true | ||||
|         include: | ||||
|           - unityVersion: 6000.0.36f1 | ||||
|             targetPlatform: WebGL | ||||
|           - unityVersion: 6000.0.36f1 | ||||
|             targetPlatform: WebGL | ||||
|             buildProfile: 'Assets/Settings/Build Profiles/Sample WebGL Build Profile.asset' | ||||
| 
 | ||||
|     steps: | ||||
|       - name: Clear Space for Android Build | ||||
|         if: matrix.targetPlatform == 'Android' | ||||
|         uses: jlumbroso/free-disk-space@v1.3.1 | ||||
| 
 | ||||
|       ########################### | ||||
|       #         Checkout        # | ||||
|       ########################### | ||||
|       - uses: actions/checkout@v4 | ||||
|         with: | ||||
|           lfs: true | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Cache          # | ||||
|       ########################### | ||||
|       - uses: actions/cache@v4 | ||||
|         with: | ||||
|           path: ${{ matrix.projectPath }}/Library | ||||
|           key: Library-${{ matrix.projectPath }}-ubuntu-${{ matrix.targetPlatform }} | ||||
|           restore-keys: | | ||||
|             Library-${{ matrix.projectPath }}-ubuntu- | ||||
|             Library- | ||||
| 
 | ||||
|       ########################### | ||||
|       #  Set Scripting Backend  # | ||||
|       ########################### | ||||
|       - name: Set Scripting Backend To il2cpp | ||||
|         if: matrix.buildWithIl2cpp == true | ||||
|         run: | | ||||
|           mv -f "./test-project/ProjectSettings/ProjectSettingsIl2cpp.asset" "./test-project/ProjectSettings/ProjectSettings.asset" | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Build          # | ||||
|       ########################### | ||||
|       - name: Build | ||||
|         uses: ./ | ||||
|         id: build-1 | ||||
|         continue-on-error: true | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|         with: | ||||
|           buildName: 'GameCI Test Build' | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           buildProfile: ${{ matrix.buildProfile }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           customParameters: -profile SomeProfile -someBoolean -someValue exampleValue ${{ matrix.additionalParameters }} | ||||
|           providerStrategy: ${{ matrix.providerStrategy }} | ||||
|           allowDirtyBuild: true | ||||
| 
 | ||||
|       - name: Sleep for Retry | ||||
|         if: ${{ steps.build-1.outcome == 'failure' }} | ||||
|         run: | | ||||
|           sleep 60 | ||||
| 
 | ||||
|       - name: Build (Retry 1) | ||||
|         uses: ./ | ||||
|         id: build-2 | ||||
|         if: ${{ steps.build-1.outcome == 'failure' }} | ||||
|         continue-on-error: true | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|         with: | ||||
|           buildName: 'GameCI Test Build' | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           buildProfile: ${{ matrix.buildProfile }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           customParameters: -profile SomeProfile -someBoolean -someValue exampleValue ${{ matrix.additionalParameters }} | ||||
|           providerStrategy: ${{ matrix.providerStrategy }} | ||||
|           allowDirtyBuild: true | ||||
| 
 | ||||
|       - name: Sleep for Retry | ||||
|         if: ${{ steps.build-2.outcome == 'failure' }} | ||||
|         run: | | ||||
|           sleep 240 | ||||
| 
 | ||||
|       - name: Build (Retry 2) | ||||
|         uses: ./ | ||||
|         id: build-3 | ||||
|         if: ${{ steps.build-2.outcome == 'failure' }} | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|         with: | ||||
|           buildName: 'GameCI Test Build' | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           buildProfile: ${{ matrix.buildProfile }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           customParameters: -profile SomeProfile -someBoolean -someValue exampleValue ${{ matrix.additionalParameters }} | ||||
|           providerStrategy: ${{ matrix.providerStrategy }} | ||||
|           allowDirtyBuild: true | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Upload         # | ||||
|       ########################### | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: | ||||
|             "Build ${{ matrix.targetPlatform }}${{ startsWith(matrix.buildProfile, 'Assets') && ' (via Build Profile)' || '' }} on Ubuntu (${{ matrix.unityVersion }}_il2cpp_${{ matrix.buildWithIl2cpp }}_params_${{ matrix.additionalParameters }})" | ||||
|           path: build | ||||
|           retention-days: 14 | ||||
|  | @ -0,0 +1,151 @@ | |||
| name: Builds - Windows | ||||
| 
 | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   push: | ||||
| 
 | ||||
| concurrency: | ||||
|   group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} | ||||
|   cancel-in-progress: true | ||||
| 
 | ||||
| jobs: | ||||
|   buildForAllPlatformsWindows: | ||||
|     name: ${{ matrix.targetPlatform }} on ${{ matrix.unityVersion }} | ||||
|     runs-on: windows-2022 | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         projectPath: | ||||
|           - test-project | ||||
|         unityVersion: | ||||
|           - 2021.3.32f1 | ||||
|           - 2022.3.13f1 | ||||
|           - 2023.2.2f1 | ||||
|         targetPlatform: | ||||
|           - Android # Build an Android apk. | ||||
|           - StandaloneWindows64 # Build a Windows 64-bit standalone. | ||||
|           - WSAPlayer # Build a UWP App | ||||
|           - tvOS # Build an Apple TV XCode project | ||||
|         enableGpu: | ||||
|           - false | ||||
|         include: | ||||
|           # Additionally test enableGpu build for a standalone windows target | ||||
|           - projectPath: test-project | ||||
|             unityVersion: 2023.2.2f1 | ||||
|             targetPlatform: StandaloneWindows64 | ||||
|             enableGpu: true | ||||
|           - unityVersion: 6000.0.36f1 | ||||
|             targetPlatform: StandaloneWindows64 | ||||
|           - unityVersion: 6000.0.36f1 | ||||
|             targetPlatform: StandaloneWindows64 | ||||
|             buildProfile: 'Assets/Settings/Build Profiles/Sample Windows Build Profile.asset' | ||||
|    | ||||
|     steps: | ||||
|       ########################### | ||||
|       #         Checkout        # | ||||
|       ########################### | ||||
|       - uses: actions/checkout@v4 | ||||
|         with: | ||||
|           lfs: true | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Cache          # | ||||
|       ########################### | ||||
|       - uses: actions/cache@v4 | ||||
|         with: | ||||
|           path: ${{ matrix.projectPath }}/Library | ||||
|           key: Library-${{ matrix.projectPath }}-windows-${{ matrix.targetPlatform }} | ||||
|           restore-keys: | | ||||
|             Library-${{ matrix.projectPath }}-windows- | ||||
|             Library- | ||||
| 
 | ||||
|       ########################### | ||||
|       #  Set Scripting Backend  # | ||||
|       ########################### | ||||
|       - name: Set Scripting Backend To il2cpp | ||||
|         run: | | ||||
|           Move-Item -Path "./test-project/ProjectSettings/ProjectSettingsIl2cpp.asset" -Destination "./test-project/ProjectSettings/ProjectSettings.asset" -Force | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Build          # | ||||
|       ########################### | ||||
|       - name: Build | ||||
|         uses: ./ | ||||
|         id: build-1 | ||||
|         continue-on-error: true | ||||
|         timeout-minutes: 30 | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|           UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
|           UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
|         with: | ||||
|           buildName: 'GameCI Test Build' | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           buildProfile: ${{ matrix.buildProfile }} | ||||
|           enableGpu: ${{ matrix.enableGpu }} | ||||
|           customParameters: -profile SomeProfile -someBoolean -someValue exampleValue | ||||
|           allowDirtyBuild: true | ||||
|       # We use dirty build because we are replacing the default project settings file above | ||||
| 
 | ||||
|       - name: Sleep for Retry | ||||
|         if: ${{ steps.build-1.outcome == 'failure' }} | ||||
|         run: | | ||||
|           Start-Sleep -s 120 | ||||
| 
 | ||||
|       - name: Build Retry 1 | ||||
|         uses: ./ | ||||
|         id: build-2 | ||||
|         continue-on-error: true | ||||
|         timeout-minutes: 30 | ||||
|         if: steps.build-1.outcome == 'failure' | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|           UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
|           UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
|         with: | ||||
|           buildName: 'GameCI Test Build' | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           enableGpu: ${{ matrix.enableGpu }} | ||||
|           customParameters: -profile SomeProfile -someBoolean -someValue exampleValue | ||||
|           allowDirtyBuild: true | ||||
|       # We use dirty build because we are replacing the default project settings file above | ||||
| 
 | ||||
|       - name: Sleep for Retry | ||||
|         if: ${{ steps.build-1.outcome == 'failure' && steps.build-2.outcome == 'failure' }} | ||||
|         run: | | ||||
|           Start-Sleep -s 240 | ||||
| 
 | ||||
|       - name: Build Retry 2 | ||||
|         uses: ./ | ||||
|         id: build-3 | ||||
|         timeout-minutes: 30 | ||||
|         if: ${{ steps.build-1.outcome == 'failure' && steps.build-2.outcome == 'failure' }} | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|           UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
|           UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
|         with: | ||||
|           buildName: 'GameCI Test Build' | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           enableGpu: ${{ matrix.enableGpu }} | ||||
|           customParameters: -profile SomeProfile -someBoolean -someValue exampleValue | ||||
|           allowDirtyBuild: true | ||||
|       # We use dirty build because we are replacing the default project settings file above | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Upload         # | ||||
|       ########################### | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: Build ${{ matrix.targetPlatform }} on Windows (${{ matrix.unityVersion }})${{ matrix.enableGpu && ' With GPU' || '' }}${{ matrix.buildProfile && '  With Build Profile' || '' }} | ||||
|           path: build | ||||
|           retention-days: 14 | ||||
|  | @ -1,73 +0,0 @@ | |||
| name: Builds | ||||
| 
 | ||||
| on: | ||||
|   push: { branches: [main] } | ||||
|   pull_request: | ||||
|     paths-ignore: | ||||
|       - '.github/**' | ||||
| 
 | ||||
| env: | ||||
|   UNITY_LICENSE: "<?xml version=\"1.0\" encoding=\"UTF-8\"?><root>\n    <License id=\"Terms\">\n        <MachineBindings>\n            <Binding Key=\"1\" Value=\"576562626572264761624c65526f7578\"/>\n            <Binding Key=\"2\" Value=\"576562626572264761624c65526f7578\"/>\n        </MachineBindings>\n        <MachineID Value=\"D7nTUnjNAmtsUMcnoyrqkgIbYdM=\"/>\n        <SerialHash Value=\"2033b8ac3e6faa3742ca9f0bfae44d18f2a96b80\"/>\n        <Features>\n            <Feature Value=\"33\"/>\n            <Feature Value=\"1\"/>\n            <Feature Value=\"12\"/>\n            <Feature Value=\"2\"/>\n            <Feature Value=\"24\"/>\n            <Feature Value=\"3\"/>\n            <Feature Value=\"36\"/>\n            <Feature Value=\"17\"/>\n            <Feature Value=\"19\"/>\n            <Feature Value=\"62\"/>\n        </Features>\n        <DeveloperData Value=\"AQAAAEY0LUJHUlgtWEQ0RS1aQ1dWLUM1SlctR0RIQg==\"/>\n        <SerialMasked Value=\"F4-BGRX-XD4E-ZCWV-C5JW-XXXX\"/>\n        <StartDate Value=\"2021-02-08T00:00:00\"/>\n        <UpdateDate Value=\"2021-02-09T00:34:57\"/>\n        <InitialActivationDate Value=\"2021-02-08T00:34:56\"/>\n        <LicenseVersion Value=\"6.x\"/>\n        <ClientProvidedVersion Value=\"2018.4.30f1\"/>\n        <AlwaysOnline Value=\"false\"/>\n        <Entitlements>\n            <Entitlement Ns=\"unity_editor\" Tag=\"UnityPersonal\" Type=\"EDITOR\" ValidTo=\"9999-12-31T00:00:00\"/>\n            <Entitlement Ns=\"unity_editor\" Tag=\"DarkSkin\" Type=\"EDITOR_FEATURE\" ValidTo=\"9999-12-31T00:00:00\"/>\n        </Entitlements>\n    </License>\n<Signature xmlns=\"http://www.w3.org/2000/09/xmldsig#\"><SignedInfo><CanonicalizationMethod Algorithm=\"http://www.w3.org/TR/2001/REC-xml-c14n-20010315#WithComments\"/><SignatureMethod Algorithm=\"http://www.w3.org/2000/09/xmldsig#rsa-sha1\"/><Reference URI=\"#Terms\"><Transforms><Transform Algorithm=\"http://www.w3.org/2000/09/xmldsig#enveloped-signature\"/></Transforms><DigestMethod Algorithm=\"http://www.w3.org/2000/09/xmldsig#sha1\"/><DigestValue>m0Db8UK+ktnOLJBtHybkfetpcKo=</DigestValue></Reference></SignedInfo><SignatureValue>o/pUbSQAukz7+ZYAWhnA0AJbIlyyCPL7bKVEM2lVqbrXt7cyey+umkCXamuOgsWPVUKBMkXtMH8L\n5etLmD0getWIhTGhzOnDCk+gtIPfL4jMo9tkEuOCROQAXCci23VFscKcrkB+3X6h4wEOtA2APhOY\nB+wvC794o8/82ffjP79aVAi57rp3Wmzx+9pe9yMwoJuljAy2sc2tIMgdQGWVmOGBpQm3JqsidyzI\nJWG2kjnc7pDXK9pwYzXoKiqUqqrut90d+kQqRyv7MSZXR50HFqD/LI69h68b7P8Bjo3bPXOhNXGR\n9YCoemH6EkfCJxp2gIjzjWW+l2Hj2EsFQi8YXw==</SignatureValue></Signature></root>" | ||||
| 
 | ||||
| jobs: | ||||
|   buildForAllPlatformsUbuntu: | ||||
|     name: Build for ${{ matrix.targetPlatform }} on version ${{ matrix.unityVersion }} | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         projectPath: | ||||
|           - test-project | ||||
|         unityVersion: | ||||
|           - 2019.2.11f1 | ||||
|           - 2019.3.15f1 | ||||
|         targetPlatform: | ||||
|           - StandaloneOSX # Build a macOS standalone (Intel 64-bit). | ||||
|           - StandaloneWindows64 # Build a Windows 64-bit standalone. | ||||
|           - StandaloneLinux64 # Build a Linux 64-bit standalone. | ||||
|           - iOS # Build an iOS player. | ||||
|           - Android # Build an Android .apk. | ||||
|           - WebGL # WebGL. | ||||
|           #          - StandaloneWindows # Build a Windows standalone. | ||||
|           #          - WSAPlayer # Build an Windows Store Apps player. | ||||
|           #          - PS4 # Build a PS4 Standalone. | ||||
|           #          - XboxOne # Build a Xbox One Standalone. | ||||
|           #          - tvOS # Build to Apple's tvOS platform. | ||||
|           #          - Switch # Build a Nintendo Switch player | ||||
|     steps: | ||||
|       ########################### | ||||
|       #         Checkout        # | ||||
|       ########################### | ||||
|       - uses: actions/checkout@v2 | ||||
|         with: | ||||
|           lfs: true | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Cache          # | ||||
|       ########################### | ||||
|       - uses: actions/cache@v2 | ||||
|         with: | ||||
|           path: ${{ matrix.projectPath }}/Library | ||||
|           key: Library-${{ matrix.projectPath }}-ubuntu-${{ matrix.targetPlatform }} | ||||
|           restore-keys: | | ||||
|             Library-${{ matrix.projectPath }}-ubuntu- | ||||
|             Library- | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Build          # | ||||
|       ########################### | ||||
|       - uses: ./ | ||||
|         with: | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           customParameters: -profile SomeProfile -someBoolean -someValue exampleValue | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Upload         # | ||||
|       ########################### | ||||
|       - uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: Build Ubuntu (${{ matrix.unityVersion }}) | ||||
|           path: build | ||||
|           retention-days: 14 | ||||
|  | @ -11,6 +11,6 @@ jobs: | |||
|     name: A cat for your effort! | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: ruairidhwm/action-cats@1.0.1 | ||||
|       - uses: ruairidhwm/action-cats@1.0.2 | ||||
|         with: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  |  | |||
|  | @ -12,3 +12,26 @@ jobs: | |||
|         with: | ||||
|           token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           expire-in: 21 days | ||||
|   cleanupCloudRunner: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|         if: github.event.event_type != 'pull_request_target' | ||||
|         with: | ||||
|           lfs: true | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: '18' | ||||
|       - run: yarn | ||||
|       - run: yarn run cli --help | ||||
|         env: | ||||
|           AWS_REGION: eu-west-2 | ||||
|           AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|           AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|           AWS_DEFAULT_REGION: eu-west-2 | ||||
|       - run: yarn run cli -m list-resources | ||||
|         env: | ||||
|           AWS_REGION: eu-west-2 | ||||
|           AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|           AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|           AWS_DEFAULT_REGION: eu-west-2 | ||||
|  |  | |||
|  | @ -0,0 +1,59 @@ | |||
| name: Async Checks API | ||||
| 
 | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|     inputs: | ||||
|       checksObject: | ||||
|         description: '' | ||||
|         required: false | ||||
|         default: '' | ||||
| 
 | ||||
| permissions: | ||||
|   checks: write | ||||
| 
 | ||||
| env: | ||||
|   GKE_ZONE: 'us-central1' | ||||
|   GKE_REGION: 'us-central1' | ||||
|   GKE_PROJECT: 'unitykubernetesbuilder' | ||||
|   GKE_CLUSTER: 'game-ci-github-pipelines' | ||||
|   GCP_LOGGING: true | ||||
|   GCP_PROJECT: unitykubernetesbuilder | ||||
|   GCP_LOG_FILE: ${{ github.workspace }}/cloud-runner-logs.txt | ||||
|   AWS_REGION: eu-west-2 | ||||
|   AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|   AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|   AWS_DEFAULT_REGION: eu-west-2 | ||||
|   AWS_STACK_NAME: game-ci-github-pipelines | ||||
|   CLOUD_RUNNER_BRANCH: ${{ github.ref }} | ||||
|   CLOUD_RUNNER_DEBUG: true | ||||
|   CLOUD_RUNNER_DEBUG_TREE: true | ||||
|   DEBUG: true | ||||
|   UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
|   PROJECT_PATH: test-project | ||||
|   UNITY_VERSION: 2019.3.15f1 | ||||
|   USE_IL2CPP: false | ||||
| 
 | ||||
| jobs: | ||||
|   asyncChecks: | ||||
|     name: Async Checks | ||||
|     if: github.event.event_type != 'pull_request_target' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - timeout-minutes: 180 | ||||
|         env: | ||||
|           UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
|           PROJECT_PATH: test-project | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|           GIT_PRIVATE_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|           TARGET_PLATFORM: StandaloneWindows64 | ||||
|           cloudRunnerTests: true | ||||
|           versioning: None | ||||
|           CLOUD_RUNNER_CLUSTER: local-docker | ||||
|           AWS_STACK_NAME: game-ci-github-pipelines | ||||
|           CHECKS_UPDATE: ${{ github.event.inputs.checksObject }} | ||||
|         run: | | ||||
|           git clone -b cloud-runner-develop https://github.com/game-ci/unity-builder | ||||
|           cd unity-builder | ||||
|           yarn | ||||
|           ls | ||||
|           yarn run cli -m checks-update | ||||
|  | @ -1,111 +0,0 @@ | |||
| name: Cloud Runner - AWS Tests | ||||
| 
 | ||||
| on: | ||||
|   push: { branches: [main, cloud-runner-develop] } | ||||
| 
 | ||||
| env: | ||||
|   GKE_ZONE: 'us-central1' | ||||
|   GKE_REGION: 'us-central1' | ||||
|   GKE_PROJECT: 'unitykubernetesbuilder' | ||||
|   GKE_CLUSTER: 'unity-builder-cluster' | ||||
|   GCP_LOGGING: true | ||||
|   GCP_PROJECT: unitykubernetesbuilder | ||||
|   AWS_REGION: eu-west-2 | ||||
|   AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|   AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|   AWS_DEFAULT_REGION: eu-west-2 | ||||
|   AWS_BASE_STACK_NAME: game-ci-github-pipelines | ||||
|   CLOUD_RUNNER_BRANCH: ${{ github.ref }} | ||||
|   CLOUD_RUNNER_TESTS: true | ||||
|   DEBUG: true | ||||
|   UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
| 
 | ||||
| jobs: | ||||
|   buildForAllPlatforms: | ||||
|     name: AWS Fargate Build | ||||
|     if: github.event.pull_request.draft == false | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         projectPath: | ||||
|           - test-project | ||||
|         unityVersion: | ||||
|           #  - 2019.2.11f1 | ||||
|           - 2019.3.15f1 | ||||
|         targetPlatform: | ||||
|           #- StandaloneOSX # Build a macOS standalone (Intel 64-bit). | ||||
|           - StandaloneWindows64 # Build a Windows 64-bit standalone. | ||||
|           - StandaloneLinux64 # Build a Linux 64-bit standalone. | ||||
|           #- iOS # Build an iOS player. | ||||
|           #- Android # Build an Android .apk. | ||||
|           #- WebGL # WebGL. | ||||
|           #          - StandaloneWindows # Build a Windows standalone. | ||||
|           #          - WSAPlayer # Build an Windows Store Apps player. | ||||
|           #          - PS4 # Build a PS4 Standalone. | ||||
|           #          - XboxOne # Build a Xbox One Standalone. | ||||
|           #          - tvOS # Build to Apple's tvOS platform. | ||||
|           #          - Switch # Build a Nintendo Switch player | ||||
|     # steps | ||||
|     steps: | ||||
|       - name: Checkout (default) | ||||
|         uses: actions/checkout@v2 | ||||
|         if: github.event.event_type != 'pull_request_target' | ||||
|         with: | ||||
|           lfs: true | ||||
|       - name: Configure AWS Credentials | ||||
|         uses: aws-actions/configure-aws-credentials@v1 | ||||
|         with: | ||||
|           aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|           aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|           aws-region: eu-west-2 | ||||
|       - run: yarn | ||||
|       - run: yarn run cli --help | ||||
|       - run: yarn run test-i-aws | ||||
|         env: | ||||
|           UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
|           PROJECT_PATH: ${{ matrix.projectPath }} | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|           TARGET_PLATFORM: ${{ matrix.targetPlatform }} | ||||
|       - uses: ./ | ||||
|         id: aws-fargate-unity-build | ||||
|         timeout-minutes: 25 | ||||
|         with: | ||||
|           cloudRunnerCluster: aws | ||||
|           versioning: None | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           githubToken: ${{ secrets.GITHUB_TOKEN }} | ||||
|           postBuildSteps: | | ||||
|             - name: upload | ||||
|               image: amazon/aws-cli | ||||
|               commands: | | ||||
|                 aws configure set aws_access_key_id $AWS_ACCESS_KEY_ID --profile default | ||||
|                 aws configure set aws_secret_access_key $AWS_SECRET_ACCESS_KEY --profile default | ||||
|                 aws configure set region $AWS_DEFAULT_REGION --profile default | ||||
|                 aws s3 ls | ||||
|                 aws s3 ls game-ci-test-storage | ||||
|                 ls /data/cache/$CACHE_KEY | ||||
|                 echo "/data/cache/$CACHE_KEY/build-$BUILD_GUID.zip s3://game-ci-test-storage/$CACHE_KEY/$BUILD_FILE" | ||||
|                 aws s3 cp /data/cache/$CACHE_KEY/build-$BUILD_GUID.zip s3://game-ci-test-storage/$CACHE_KEY/build-$BUILD_GUID.zip | ||||
|                 aws s3 cp /data/cache/$CACHE_KEY s3://game-ci-test-storage/$CACHE_KEY/$BUILD_GUID | ||||
|               secrets: | ||||
|               - name: awsAccessKeyId | ||||
|                 value: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|               - name: awsSecretAccessKey | ||||
|                 value: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|               - name: awsDefaultRegion | ||||
|                 value: eu-west-2 | ||||
|       - run: | | ||||
|           aws s3 cp s3://game-ci-test-storage/${{ steps.aws-fargate-unity-build.outputs.BRANCH }}/build-${{ steps.aws-fargate-unity-build.outputs.BUILD_GUID }}.zip build-${{ steps.aws-fargate-unity-build.outputs.BUILD_GUID }}.zip | ||||
|           ls | ||||
|       ########################### | ||||
|       #          Upload         # | ||||
|       ########################### | ||||
|       # download from cloud storage | ||||
|       - uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: AWS Build (${{ matrix.targetPlatform }}) | ||||
|           path: build-${{ steps.aws-fargate-unity-build.outputs.BUILD_GUID }}.zip | ||||
|           retention-days: 14 | ||||
|  | @ -0,0 +1,231 @@ | |||
| name: Cloud Runner CI Pipeline | ||||
| 
 | ||||
| on: | ||||
|   push: { branches: [cloud-runner-develop, cloud-runner-preview, main] } | ||||
|   workflow_dispatch: | ||||
|     inputs: | ||||
|       runGithubIntegrationTests: | ||||
|         description: 'Run GitHub Checks integration tests' | ||||
|         required: false | ||||
|         default: 'false' | ||||
| 
 | ||||
| permissions: | ||||
|   checks: write | ||||
|   contents: read | ||||
|   actions: write | ||||
| 
 | ||||
| env: | ||||
|   GKE_ZONE: 'us-central1' | ||||
|   GKE_REGION: 'us-central1' | ||||
|   GKE_PROJECT: 'unitykubernetesbuilder' | ||||
|   GKE_CLUSTER: 'game-ci-github-pipelines' | ||||
|   GCP_LOGGING: true | ||||
|   GCP_PROJECT: unitykubernetesbuilder | ||||
|   GCP_LOG_FILE: ${{ github.workspace }}/cloud-runner-logs.txt | ||||
|   AWS_REGION: eu-west-2 | ||||
|   AWS_DEFAULT_REGION: eu-west-2 | ||||
|   AWS_STACK_NAME: game-ci-team-pipelines | ||||
|   CLOUD_RUNNER_BRANCH: ${{ github.ref }} | ||||
|   DEBUG: true | ||||
|   UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|   UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|   UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
|   PROJECT_PATH: test-project | ||||
|   UNITY_VERSION: 2019.3.15f1 | ||||
|   USE_IL2CPP: false | ||||
|   USE_GKE_GCLOUD_AUTH_PLUGIN: true | ||||
| 
 | ||||
| jobs: | ||||
|   tests: | ||||
|     name: Tests | ||||
|     if: github.event.event_type != 'pull_request_target' | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         test: | ||||
|           - 'cloud-runner-end2end-locking' | ||||
|           - 'cloud-runner-end2end-caching' | ||||
|           - 'cloud-runner-end2end-retaining' | ||||
|           - 'cloud-runner-caching' | ||||
|           - 'cloud-runner-environment' | ||||
|           - 'cloud-runner-image' | ||||
|           - 'cloud-runner-hooks' | ||||
|           - 'cloud-runner-local-persistence' | ||||
|           - 'cloud-runner-locking-core' | ||||
|           - 'cloud-runner-locking-get-locked' | ||||
|     steps: | ||||
|       - name: Checkout (default) | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           lfs: false | ||||
|       - name: Configure AWS Credentials | ||||
|         uses: aws-actions/configure-aws-credentials@v1 | ||||
|         with: | ||||
|           aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|           aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|           aws-region: eu-west-2 | ||||
|       - run: yarn | ||||
|       - run: yarn run test "${{ matrix.test }}" --detectOpenHandles --forceExit --runInBand | ||||
|         timeout-minutes: 60 | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|           UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
|           PROJECT_PATH: test-project | ||||
|           TARGET_PLATFORM: StandaloneWindows64 | ||||
|           cloudRunnerTests: true | ||||
|           versioning: None | ||||
|           KUBE_STORAGE_CLASS: local-path | ||||
|           PROVIDER_STRATEGY: local-docker | ||||
|           AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|           AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|           GIT_PRIVATE_TOKEN: ${{ secrets.GIT_PRIVATE_TOKEN }} | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|   k8sTests: | ||||
|     name: K8s Tests | ||||
|     if: github.event.event_type != 'pull_request_target' | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         test: | ||||
|           # - 'cloud-runner-async-workflow' | ||||
|           - 'cloud-runner-end2end-locking' | ||||
|           - 'cloud-runner-end2end-caching' | ||||
|           - 'cloud-runner-end2end-retaining' | ||||
|           - 'cloud-runner-kubernetes' | ||||
|           - 'cloud-runner-environment' | ||||
|           - 'cloud-runner-github-checks' | ||||
|     steps: | ||||
|       - name: Checkout (default) | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           lfs: false | ||||
|       - run: yarn | ||||
|       - name: actions-k3s | ||||
|         uses: debianmaster/actions-k3s@v1.0.5 | ||||
|         with: | ||||
|           version: 'latest' | ||||
|       - run: yarn run test "${{ matrix.test }}" --detectOpenHandles --forceExit --runInBand | ||||
|         timeout-minutes: 60 | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|           UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
|           PROJECT_PATH: test-project | ||||
|           TARGET_PLATFORM: StandaloneWindows64 | ||||
|           cloudRunnerTests: true | ||||
|           versioning: None | ||||
|           KUBE_STORAGE_CLASS: local-path | ||||
|           PROVIDER_STRATEGY: k8s | ||||
|           AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|           AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|           GIT_PRIVATE_TOKEN: ${{ secrets.GIT_PRIVATE_TOKEN }} | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|   awsTests: | ||||
|     name: AWS Tests | ||||
|     if: github.event.event_type != 'pull_request_target' | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         test: | ||||
|           - 'cloud-runner-end2end-locking' | ||||
|           - 'cloud-runner-end2end-caching' | ||||
|           - 'cloud-runner-end2end-retaining' | ||||
|           - 'cloud-runner-environment' | ||||
|           - 'cloud-runner-s3-steps' | ||||
|     steps: | ||||
|       - name: Checkout (default) | ||||
|         uses: actions/checkout@v2 | ||||
|         with: | ||||
|           lfs: false | ||||
|       - name: Configure AWS Credentials | ||||
|         uses: aws-actions/configure-aws-credentials@v1 | ||||
|         with: | ||||
|           aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|           aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|           aws-region: eu-west-2 | ||||
|       - run: yarn | ||||
|       - run: yarn run test "${{ matrix.test }}" --detectOpenHandles --forceExit --runInBand | ||||
|         timeout-minutes: 60 | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|           UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
|           PROJECT_PATH: test-project | ||||
|           TARGET_PLATFORM: StandaloneWindows64 | ||||
|           cloudRunnerTests: true | ||||
|           versioning: None | ||||
|           KUBE_STORAGE_CLASS: local-path | ||||
|           PROVIDER_STRATEGY: aws | ||||
|           AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|           AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|           GIT_PRIVATE_TOKEN: ${{ secrets.GIT_PRIVATE_TOKEN }} | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
| 
 | ||||
|   buildTargetTests: | ||||
|     name: Local Build Target Tests | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         providerStrategy: | ||||
|           #- aws | ||||
|           - local-docker | ||||
|           #- k8s | ||||
|         targetPlatform: | ||||
|           - StandaloneOSX # Build a macOS standalone (Intel 64-bit). | ||||
|           - StandaloneWindows64 # Build a Windows 64-bit standalone. | ||||
|           - StandaloneLinux64 # Build a Linux 64-bit standalone. | ||||
|           - WebGL # WebGL. | ||||
|           - iOS # Build an iOS player. | ||||
|           # - Android # Build an Android .apk. | ||||
|     steps: | ||||
|       - name: Checkout (default) | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           lfs: false | ||||
|       - run: yarn | ||||
|       - uses: ./ | ||||
|         id: unity-build | ||||
|         timeout-minutes: 30 | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|           UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
| 
 | ||||
|           AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|           AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|           GIT_PRIVATE_TOKEN: ${{ secrets.GIT_PRIVATE_TOKEN }} | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|         with: | ||||
|           cloudRunnerTests: true | ||||
|           versioning: None | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           providerStrategy: ${{ matrix.providerStrategy }} | ||||
|       - run: | | ||||
|           cp ./cloud-runner-cache/cache/${{ steps.unity-build.outputs.CACHE_KEY }}/build/${{ steps.unity-build.outputs.BUILD_ARTIFACT }} ${{ steps.unity-build.outputs.BUILD_ARTIFACT }} | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: ${{ matrix.providerStrategy }} Build (${{ matrix.targetPlatform }}) | ||||
|           path: ${{ steps.unity-build.outputs.BUILD_ARTIFACT }} | ||||
|           retention-days: 14 | ||||
| 
 | ||||
|   githubChecksIntegration: | ||||
|     name: GitHub Checks Integration | ||||
|     runs-on: ubuntu-latest | ||||
|     if: github.event_name == 'workflow_dispatch' && github.event.inputs.runGithubIntegrationTests == 'true' | ||||
|     env: | ||||
|       RUN_GITHUB_INTEGRATION_TESTS: true | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: 20 | ||||
|           cache: 'yarn' | ||||
|       - run: yarn install --frozen-lockfile | ||||
|       - run: yarn test cloud-runner-github-checks-integration-test --detectOpenHandles --forceExit --runInBand | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | @ -1,125 +0,0 @@ | |||
| name: Cloud Runner - K8s Tests | ||||
| 
 | ||||
| on: | ||||
|   push: { branches: [cloud-runner-develop] } | ||||
| #  push: { branches: [main] } | ||||
| #  pull_request: | ||||
| #    paths-ignore: | ||||
| #      - '.github/**' | ||||
| 
 | ||||
| env: | ||||
|   GKE_ZONE: 'us-central1' | ||||
|   GKE_REGION: 'us-central1' | ||||
|   GKE_PROJECT: 'unitykubernetesbuilder' | ||||
|   GKE_CLUSTER: 'game-ci-github-pipelines' | ||||
|   GCP_LOGGING: true | ||||
|   GCP_PROJECT: unitykubernetesbuilder | ||||
|   GCP_LOG_FILE: ${{ github.workspace }}/cloud-runner-logs.txt | ||||
|   AWS_REGION: eu-west-2 | ||||
|   AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|   AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|   AWS_DEFAULT_REGION: eu-west-2 | ||||
|   AWS_BASE_STACK_NAME: game-ci-github-pipelines | ||||
|   CLOUD_RUNNER_BRANCH: ${{ github.ref }} | ||||
|   CLOUD_RUNNER_TESTS: true | ||||
|   DEBUG: true | ||||
|   UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
| 
 | ||||
| jobs: | ||||
|   k8sBuilds: | ||||
|     name: K8s (GKE Autopilot) build for ${{ matrix.targetPlatform }} on version ${{ matrix.unityVersion }} | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         unityVersion: | ||||
|           #  - 2019.2.11f1 | ||||
|           - 2019.3.15f1 | ||||
|         targetPlatform: | ||||
|           #  - StandaloneWindows64 | ||||
|           - StandaloneLinux64 | ||||
|     steps: | ||||
|       ########################### | ||||
|       #         Checkout        # | ||||
|       ########################### | ||||
|       - uses: actions/checkout@v2 | ||||
|         if: github.event.event_type != 'pull_request_target' | ||||
|         with: | ||||
|           lfs: true | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Setup          # | ||||
|       ########################### | ||||
|       - uses: google-github-actions/setup-gcloud@master | ||||
|         with: | ||||
|           version: '288.0.0' | ||||
|           service_account_email: ${{ secrets.GOOGLE_SERVICE_ACCOUNT_EMAIL }} | ||||
|           service_account_key: ${{ secrets.GOOGLE_SERVICE_ACCOUNT_KEY }} | ||||
|       - name: Get GKE cluster credentials | ||||
|         run: gcloud container clusters get-credentials $GKE_CLUSTER --zone $GKE_ZONE --project $GKE_PROJECT | ||||
| 
 | ||||
|       ########################### | ||||
|       # Cloud Runner Test Suite # | ||||
|       ########################### | ||||
|       - uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 12.x | ||||
|       - run: yarn | ||||
|       - run: yarn run cli --help | ||||
|       - name: Cloud Runner Test Suite | ||||
|         run: yarn run test-i-k8s --detectOpenHandles --forceExit | ||||
|         env: | ||||
|           UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
|           PROJECT_PATH: ${{ matrix.projectPath }} | ||||
|           TARGET_PLATFORM: ${{ matrix.targetPlatform }} | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|           KUBE_CONFIG: ${{ steps.read-base64.outputs.base64 }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
| 
 | ||||
|       ########################### | ||||
|       # Cloud Runner Build Test # | ||||
|       ########################### | ||||
|       - name: Cloud Runner Build Test | ||||
|         uses: ./ | ||||
|         id: k8s-unity-build | ||||
|         timeout-minutes: 30 | ||||
|         with: | ||||
|           cloudRunnerCluster: k8s | ||||
|           UNITY_LICENSE: ${{ secrets.UNITY_LICENSE }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           kubeConfig: ${{ steps.read-base64.outputs.base64 }} | ||||
|           githubToken: ${{ secrets.GITHUB_TOKEN }} | ||||
|           projectPath: test-project | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           versioning: None | ||||
|           postBuildSteps: | | ||||
|             - name: upload | ||||
|               image: amazon/aws-cli | ||||
|               commands: | | ||||
|                 aws configure set aws_access_key_id $AWS_ACCESS_KEY_ID --profile default | ||||
|                 aws configure set aws_secret_access_key $AWS_SECRET_ACCESS_KEY --profile default | ||||
|                 aws configure set region $AWS_DEFAULT_REGION --profile default | ||||
|                 aws s3 ls | ||||
|                 aws s3 ls game-ci-test-storage | ||||
|                 ls /data/cache/$BRANCH | ||||
|                 echo "/data/cache/$BRANCH/build-$BUILD_GUID.zip s3://game-ci-test-storage/$BRANCH/$BUILD_FILE" | ||||
|                 aws s3 cp /data/cache/$BRANCH/build-$BUILD_GUID.zip s3://game-ci-test-storage/$BRANCH/build-$BUILD_GUID.zip | ||||
|               secrets: | ||||
|               - name: awsAccessKeyId | ||||
|                 value: ${{ secrets.AWS_ACCESS_KEY_ID }} | ||||
|               - name: awsSecretAccessKey | ||||
|                 value: ${{ secrets.AWS_SECRET_ACCESS_KEY }} | ||||
|               - name: awsDefaultRegion | ||||
|                 value: eu-west-2 | ||||
|       - run: | | ||||
|           aws s3 cp s3://game-ci-test-storage/${{ steps.k8s-unity-build.outputs.BRANCH }}/build-${{ steps.k8s-unity-build.outputs.BUILD_GUID }}.zip build-${{ steps.k8s-unity-build.outputs.BUILD_GUID }}.zip | ||||
|           ls | ||||
|       ########################### | ||||
|       #          Upload         # | ||||
|       ########################### | ||||
|       # download from cloud storage | ||||
|       - uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: K8s Build (${{ matrix.targetPlatform }}) | ||||
|           path: build-${{ steps.k8s-unity-build.outputs.BUILD_GUID }}.zip | ||||
|           retention-days: 14 | ||||
|  | @ -7,15 +7,19 @@ on: | |||
| env: | ||||
|   CODECOV_TOKEN: '2f2eb890-30e2-4724-83eb-7633832cf0de' | ||||
| 
 | ||||
| concurrency: | ||||
|   group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} | ||||
|   cancel-in-progress: true | ||||
| 
 | ||||
| jobs: | ||||
|   tests: | ||||
|     name: Tests | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v2 | ||||
|       - uses: actions/setup-node@v2 | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: 12.x | ||||
|           node-version: '18' | ||||
|       - run: yarn | ||||
|       - run: yarn lint | ||||
|       - run: yarn test --coverage | ||||
|  |  | |||
|  | @ -1,74 +0,0 @@ | |||
| name: Mac Builds | ||||
| 
 | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - main | ||||
| 
 | ||||
| env: | ||||
|   UNITY_LICENSE: "<?xml version=\"1.0\" encoding=\"UTF-8\"?><root>\n    <License id=\"Terms\">\n        <MachineBindings>\n            <Binding Key=\"1\" Value=\"576562626572264761624c65526f7578\"/>\n            <Binding Key=\"2\" Value=\"576562626572264761624c65526f7578\"/>\n        </MachineBindings>\n        <MachineID Value=\"D7nTUnjNAmtsUMcnoyrqkgIbYdM=\"/>\n        <SerialHash Value=\"2033b8ac3e6faa3742ca9f0bfae44d18f2a96b80\"/>\n        <Features>\n            <Feature Value=\"33\"/>\n            <Feature Value=\"1\"/>\n            <Feature Value=\"12\"/>\n            <Feature Value=\"2\"/>\n            <Feature Value=\"24\"/>\n            <Feature Value=\"3\"/>\n            <Feature Value=\"36\"/>\n            <Feature Value=\"17\"/>\n            <Feature Value=\"19\"/>\n            <Feature Value=\"62\"/>\n        </Features>\n        <DeveloperData Value=\"AQAAAEY0LUJHUlgtWEQ0RS1aQ1dWLUM1SlctR0RIQg==\"/>\n        <SerialMasked Value=\"F4-BGRX-XD4E-ZCWV-C5JW-XXXX\"/>\n        <StartDate Value=\"2021-02-08T00:00:00\"/>\n        <UpdateDate Value=\"2021-02-09T00:34:57\"/>\n        <InitialActivationDate Value=\"2021-02-08T00:34:56\"/>\n        <LicenseVersion Value=\"6.x\"/>\n        <ClientProvidedVersion Value=\"2018.4.30f1\"/>\n        <AlwaysOnline Value=\"false\"/>\n        <Entitlements>\n            <Entitlement Ns=\"unity_editor\" Tag=\"UnityPersonal\" Type=\"EDITOR\" ValidTo=\"9999-12-31T00:00:00\"/>\n            <Entitlement Ns=\"unity_editor\" Tag=\"DarkSkin\" Type=\"EDITOR_FEATURE\" ValidTo=\"9999-12-31T00:00:00\"/>\n        </Entitlements>\n    </License>\n<Signature xmlns=\"http://www.w3.org/2000/09/xmldsig#\"><SignedInfo><CanonicalizationMethod Algorithm=\"http://www.w3.org/TR/2001/REC-xml-c14n-20010315#WithComments\"/><SignatureMethod Algorithm=\"http://www.w3.org/2000/09/xmldsig#rsa-sha1\"/><Reference URI=\"#Terms\"><Transforms><Transform Algorithm=\"http://www.w3.org/2000/09/xmldsig#enveloped-signature\"/></Transforms><DigestMethod Algorithm=\"http://www.w3.org/2000/09/xmldsig#sha1\"/><DigestValue>m0Db8UK+ktnOLJBtHybkfetpcKo=</DigestValue></Reference></SignedInfo><SignatureValue>o/pUbSQAukz7+ZYAWhnA0AJbIlyyCPL7bKVEM2lVqbrXt7cyey+umkCXamuOgsWPVUKBMkXtMH8L\n5etLmD0getWIhTGhzOnDCk+gtIPfL4jMo9tkEuOCROQAXCci23VFscKcrkB+3X6h4wEOtA2APhOY\nB+wvC794o8/82ffjP79aVAi57rp3Wmzx+9pe9yMwoJuljAy2sc2tIMgdQGWVmOGBpQm3JqsidyzI\nJWG2kjnc7pDXK9pwYzXoKiqUqqrut90d+kQqRyv7MSZXR50HFqD/LI69h68b7P8Bjo3bPXOhNXGR\n9YCoemH6EkfCJxp2gIjzjWW+l2Hj2EsFQi8YXw==</SignatureValue></Signature></root>" | ||||
| 
 | ||||
| jobs: | ||||
|   buildForAllPlatformsWindows: | ||||
|     name: Build for ${{ matrix.targetPlatform }} on version ${{ matrix.unityVersion }} | ||||
|     runs-on: macos-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         projectPath: | ||||
|           - test-project | ||||
|         unityVersion: | ||||
|           - 2020.3.24f1 | ||||
|         targetPlatform: | ||||
|           - StandaloneOSX # Build a MacOS executable | ||||
| 
 | ||||
|     steps: | ||||
|       ########################### | ||||
|       #         Checkout        # | ||||
|       ########################### | ||||
|       - uses: actions/checkout@v2 | ||||
|         with: | ||||
|           lfs: true | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Cache          # | ||||
|       ########################### | ||||
|       - uses: actions/cache@v2 | ||||
|         with: | ||||
|           path: ${{ matrix.projectPath }}/Library | ||||
|           key: Library-${{ matrix.projectPath }}-macos-${{ matrix.targetPlatform }} | ||||
|           restore-keys: | | ||||
|             Library-${{ matrix.projectPath }}-macos- | ||||
|             Library- | ||||
| 
 | ||||
|       ########################### | ||||
|       #  Set Scripting Backend  # | ||||
|       ########################### | ||||
|       - name: Set Scripting Backend To il2cpp | ||||
|         run: | | ||||
|           mv -f "./test-project/ProjectSettings/ProjectSettingsIl2cpp.asset" "./test-project/ProjectSettings/ProjectSettings.asset" | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Build          # | ||||
|       ########################### | ||||
|       - uses: ./ | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|           UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
|         with: | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           customParameters: -profile SomeProfile -someBoolean -someValue exampleValue | ||||
|           # We use dirty build because we are replacing the default project settings file above | ||||
|           allowDirtyBuild: true | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Upload         # | ||||
|       ########################### | ||||
|       - uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: Build MacOS (${{ matrix.unityVersion }}) | ||||
|           path: build | ||||
|           retention-days: 14 | ||||
|  | @ -1,77 +0,0 @@ | |||
| name: Windows Builds | ||||
| 
 | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - main | ||||
| 
 | ||||
| env: | ||||
|   UNITY_LICENSE: "<?xml version=\"1.0\" encoding=\"UTF-8\"?><root>\n    <License id=\"Terms\">\n        <MachineBindings>\n            <Binding Key=\"1\" Value=\"576562626572264761624c65526f7578\"/>\n            <Binding Key=\"2\" Value=\"576562626572264761624c65526f7578\"/>\n        </MachineBindings>\n        <MachineID Value=\"D7nTUnjNAmtsUMcnoyrqkgIbYdM=\"/>\n        <SerialHash Value=\"2033b8ac3e6faa3742ca9f0bfae44d18f2a96b80\"/>\n        <Features>\n            <Feature Value=\"33\"/>\n            <Feature Value=\"1\"/>\n            <Feature Value=\"12\"/>\n            <Feature Value=\"2\"/>\n            <Feature Value=\"24\"/>\n            <Feature Value=\"3\"/>\n            <Feature Value=\"36\"/>\n            <Feature Value=\"17\"/>\n            <Feature Value=\"19\"/>\n            <Feature Value=\"62\"/>\n        </Features>\n        <DeveloperData Value=\"AQAAAEY0LUJHUlgtWEQ0RS1aQ1dWLUM1SlctR0RIQg==\"/>\n        <SerialMasked Value=\"F4-BGRX-XD4E-ZCWV-C5JW-XXXX\"/>\n        <StartDate Value=\"2021-02-08T00:00:00\"/>\n        <UpdateDate Value=\"2021-02-09T00:34:57\"/>\n        <InitialActivationDate Value=\"2021-02-08T00:34:56\"/>\n        <LicenseVersion Value=\"6.x\"/>\n        <ClientProvidedVersion Value=\"2018.4.30f1\"/>\n        <AlwaysOnline Value=\"false\"/>\n        <Entitlements>\n            <Entitlement Ns=\"unity_editor\" Tag=\"UnityPersonal\" Type=\"EDITOR\" ValidTo=\"9999-12-31T00:00:00\"/>\n            <Entitlement Ns=\"unity_editor\" Tag=\"DarkSkin\" Type=\"EDITOR_FEATURE\" ValidTo=\"9999-12-31T00:00:00\"/>\n        </Entitlements>\n    </License>\n<Signature xmlns=\"http://www.w3.org/2000/09/xmldsig#\"><SignedInfo><CanonicalizationMethod Algorithm=\"http://www.w3.org/TR/2001/REC-xml-c14n-20010315#WithComments\"/><SignatureMethod Algorithm=\"http://www.w3.org/2000/09/xmldsig#rsa-sha1\"/><Reference URI=\"#Terms\"><Transforms><Transform Algorithm=\"http://www.w3.org/2000/09/xmldsig#enveloped-signature\"/></Transforms><DigestMethod Algorithm=\"http://www.w3.org/2000/09/xmldsig#sha1\"/><DigestValue>m0Db8UK+ktnOLJBtHybkfetpcKo=</DigestValue></Reference></SignedInfo><SignatureValue>o/pUbSQAukz7+ZYAWhnA0AJbIlyyCPL7bKVEM2lVqbrXt7cyey+umkCXamuOgsWPVUKBMkXtMH8L\n5etLmD0getWIhTGhzOnDCk+gtIPfL4jMo9tkEuOCROQAXCci23VFscKcrkB+3X6h4wEOtA2APhOY\nB+wvC794o8/82ffjP79aVAi57rp3Wmzx+9pe9yMwoJuljAy2sc2tIMgdQGWVmOGBpQm3JqsidyzI\nJWG2kjnc7pDXK9pwYzXoKiqUqqrut90d+kQqRyv7MSZXR50HFqD/LI69h68b7P8Bjo3bPXOhNXGR\n9YCoemH6EkfCJxp2gIjzjWW+l2Hj2EsFQi8YXw==</SignatureValue></Signature></root>" | ||||
| 
 | ||||
| jobs: | ||||
|   buildForAllPlatformsWindows: | ||||
|     name: Build for ${{ matrix.targetPlatform }} on version ${{ matrix.unityVersion }} | ||||
|     runs-on: windows-2019 | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         projectPath: | ||||
|           - test-project | ||||
|         unityVersion: | ||||
|           - 2020.3.24f1 | ||||
|         targetPlatform: | ||||
|           - StandaloneWindows64 # Build a Windows 64-bit standalone. | ||||
|           - StandaloneWindows # Build a Windows 32-bit standalone. | ||||
|           - WSAPlayer # Build a UWP App | ||||
|           - tvOS # Build an Apple TV XCode project | ||||
| 
 | ||||
|     steps: | ||||
|       ########################### | ||||
|       #         Checkout        # | ||||
|       ########################### | ||||
|       - uses: actions/checkout@v2 | ||||
|         with: | ||||
|           lfs: true | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Cache          # | ||||
|       ########################### | ||||
|       - uses: actions/cache@v2 | ||||
|         with: | ||||
|           path: ${{ matrix.projectPath }}/Library | ||||
|           key: Library-${{ matrix.projectPath }}-windows-${{ matrix.targetPlatform }} | ||||
|           restore-keys: | | ||||
|             Library-${{ matrix.projectPath }}-windows- | ||||
|             Library- | ||||
| 
 | ||||
|       ########################### | ||||
|       #  Set Scripting Backend  # | ||||
|       ########################### | ||||
|       - name: Set Scripting Backend To il2cpp | ||||
|         run: | | ||||
|           Move-Item -Path "./test-project/ProjectSettings/ProjectSettingsIl2cpp.asset" -Destination "./test-project/ProjectSettings/ProjectSettings.asset" -Force | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Build          # | ||||
|       ########################### | ||||
|       - uses: ./ | ||||
|         env: | ||||
|           UNITY_EMAIL: ${{ secrets.UNITY_EMAIL }} | ||||
|           UNITY_PASSWORD: ${{ secrets.UNITY_PASSWORD }} | ||||
|           UNITY_SERIAL: ${{ secrets.UNITY_SERIAL }} | ||||
|         with: | ||||
|           projectPath: ${{ matrix.projectPath }} | ||||
|           unityVersion: ${{ matrix.unityVersion }} | ||||
|           targetPlatform: ${{ matrix.targetPlatform }} | ||||
|           customParameters: -profile SomeProfile -someBoolean -someValue exampleValue | ||||
|           allowDirtyBuild: true | ||||
|       # We use dirty build because we are replacing the default project settings file above | ||||
| 
 | ||||
|       ########################### | ||||
|       #          Upload         # | ||||
|       ########################### | ||||
|       - uses: actions/upload-artifact@v2 | ||||
|         with: | ||||
|           name: Build Windows (${{ matrix.unityVersion }}) | ||||
|           path: build | ||||
|           retention-days: 14 | ||||
|  | @ -2,5 +2,6 @@ | |||
|   "semi": true, | ||||
|   "singleQuote": true, | ||||
|   "trailingComma": "all", | ||||
|   "printWidth": 120 | ||||
|   "printWidth": 120, | ||||
|   "proseWrap": "always" | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,32 @@ | |||
| { | ||||
|   "configurations": [ | ||||
|     { | ||||
|       "name": "PowerShell Launch Current File", | ||||
|       "type": "PowerShell", | ||||
|       "request": "launch", | ||||
|       "script": "${file}", | ||||
|       "cwd": "${cwd}" | ||||
|     }, | ||||
|     { | ||||
|       "type": "node", | ||||
|       "request": "launch", | ||||
|       "name": "Debug Jest Test", | ||||
|       "program": "${workspaceRoot}/node_modules/jest/bin/jest.js", | ||||
|       "args": [ | ||||
|         "--collectCoverage=false", | ||||
|         "--colors", | ||||
|         "--config", | ||||
|         "${workspaceRoot}/jest.config.js", | ||||
|         "--runInBand", | ||||
|         "--runTestsByPath", | ||||
|         "${relativeFile}", | ||||
|         "--testPathPattern=${fileDirname}", | ||||
|         "--testTimeout=10000000" | ||||
|       ], | ||||
|       "outputCapture": "std", | ||||
|       "internalConsoleOptions": "openOnSessionStart", | ||||
|       "envFile": "${workspaceRoot}/.env", | ||||
|       "skipFiles": ["${workspaceRoot}/../../node_modules/**/*", "<node_internals>/**/*"] | ||||
|     } | ||||
|   ] | ||||
| } | ||||
|  | @ -1,4 +1,5 @@ | |||
| { | ||||
|   "files.eol": "\n", | ||||
|   "god.tsconfig": "./tsconfig.json", | ||||
|   "yaml.customTags": [ | ||||
|     "!And", | ||||
|  |  | |||
|  | @ -4,13 +4,12 @@ | |||
| 
 | ||||
| #### Code of Conduct | ||||
| 
 | ||||
| This repository has adopted the Contributor Covenant as it's | ||||
| Code of Conduct. It is expected that participants adhere to it. | ||||
| This repository has adopted the Contributor Covenant as it's Code of Conduct. It is expected that participants adhere to | ||||
| it. | ||||
| 
 | ||||
| #### Proposing a Change | ||||
| 
 | ||||
| If you are unsure about whether or not a change is desired, | ||||
| you can create an issue. This is useful because it creates | ||||
| If you are unsure about whether or not a change is desired, you can create an issue. This is useful because it creates | ||||
| the possibility for a discussion that's visible to everyone. | ||||
| 
 | ||||
| When fixing a bug it is fine to submit a pull request right away. | ||||
|  | @ -26,7 +25,7 @@ Steps to be performed to submit a pull request: | |||
| 
 | ||||
| #### Pull Request Prerequisites | ||||
| 
 | ||||
| You have [Node](https://nodejs.org/) installed at v12.2.0+ and [Yarn](https://yarnpkg.com/) at v1.18.0+. | ||||
| You have [Node](https://nodejs.org/) installed at v18+ and [Yarn](https://yarnpkg.com/) at v1.22.0+. | ||||
| 
 | ||||
| Please note that commit hooks will run automatically to perform some tasks; | ||||
| 
 | ||||
|  | @ -34,6 +33,12 @@ Please note that commit hooks will run automatically to perform some tasks; | |||
| - run tests | ||||
| - build distributable files | ||||
| 
 | ||||
| #### Windows users | ||||
| 
 | ||||
| Make sure your editor and terminal that run the tests are set to `Powershell 7` or above with | ||||
| `Git's Unix tools for Windows` installed. This is because some tests require you to be able to run `sh` and other | ||||
| unix commands. | ||||
| 
 | ||||
| #### License | ||||
| 
 | ||||
| By contributing to this repository, you agree that your contributions will be licensed under its MIT license. | ||||
|  |  | |||
|  | @ -10,8 +10,9 @@ Part of the <a href="https://game.ci">GameCI</a> open source project. | |||
| <br /> | ||||
| <br /> | ||||
| 
 | ||||
| [](https://github.com/game-ci/unity-builder/actions?query=branch%3Amain+event%3Apush+workflow%3A%22Builds) | ||||
| [](https://lgtm.com/projects/g/webbertakken/unity-builder/context:javascript) | ||||
| [](https://github.com/game-ci/unity-builder/actions/workflows/build-tests-ubuntu.yml) | ||||
| [](https://github.com/game-ci/unity-builder/actions/workflows/build-tests-windows.yml) | ||||
| [](https://github.com/game-ci/unity-builder/actions/workflows/build-tests-mac.yml) | ||||
| [](https://codecov.io/gh/game-ci/unity-builder) | ||||
| <br /> | ||||
| <br /> | ||||
|  |  | |||
							
								
								
									
										199
									
								
								action.yml
								
								
								
								
							
							
						
						
									
										199
									
								
								action.yml
								
								
								
								
							|  | @ -9,7 +9,8 @@ inputs: | |||
|   unityVersion: | ||||
|     required: false | ||||
|     default: 'auto' | ||||
|     description: 'Version of unity to use for building the project. Use "auto" to get from your ProjectSettings/ProjectVersion.txt' | ||||
|     description: | ||||
|       'Version of unity to use for building the project. Use "auto" to get from your ProjectSettings/ProjectVersion.txt' | ||||
|   customImage: | ||||
|     required: false | ||||
|     default: '' | ||||
|  | @ -17,11 +18,15 @@ inputs: | |||
|   projectPath: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Relative path to the project to be built.' | ||||
|     description: 'Path to the project to be built, relative to the repository root.' | ||||
|   buildProfile: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Path to the build profile to activate, relative to the project root.' | ||||
|   buildName: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Name of the build.' | ||||
|     description: 'Name of the build. Should not include a file extension.' | ||||
|   buildsPath: | ||||
|     required: false | ||||
|     default: '' | ||||
|  | @ -30,6 +35,14 @@ inputs: | |||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Path to a Namespace.Class.StaticMethod to run to perform the build.' | ||||
|   manualExit: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Suppresses `-quit`. Exit your build method using `EditorApplication.Exit(0)` instead.' | ||||
|   enableGpu: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Launches unity without specifying `-nographics`.' | ||||
|   customParameters: | ||||
|     required: false | ||||
|     default: '' | ||||
|  | @ -46,10 +59,12 @@ inputs: | |||
|     required: false | ||||
|     default: '' | ||||
|     description: 'The android versionCode' | ||||
|   androidAppBundle: | ||||
|   androidExportType: | ||||
|     required: false | ||||
|     default: 'false' | ||||
|     description: 'Whether to build .aab instead of .apk' | ||||
|     default: 'androidPackage' | ||||
|     description: | ||||
|       'The android export type. Should be androidPackage for apk, androidAppBundle for aab, or androidStudioProject for | ||||
|       an android studio project.' | ||||
|   androidKeystoreName: | ||||
|     required: false | ||||
|     default: '' | ||||
|  | @ -74,74 +89,198 @@ inputs: | |||
|     required: false | ||||
|     default: '' | ||||
|     description: 'The android target API level.' | ||||
|   androidSymbolType: | ||||
|     required: false | ||||
|     default: 'none' | ||||
|     description: 'The android symbol type to export. Should be "none", "public" or "debugging".' | ||||
|   sshAgent: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'SSH Agent path to forward to the container' | ||||
|   sshPublicKeysDirectoryPath: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Path to a directory containing SSH public keys to forward to the container.' | ||||
|   gitPrivateToken: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Github private token to pull from github' | ||||
|     description: '[CloudRunner] Github private token to pull from github' | ||||
|   githubOwner: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: '[CloudRunner] GitHub owner name or organization/team name' | ||||
|   runAsHostUser: | ||||
|     required: false | ||||
|     default: 'false' | ||||
|     description: | ||||
|       'Whether to run as a user that matches the host system or the default root container user. Only applicable to | ||||
|       Linux hosts and containers. This is useful for fixing permission errors on Self-Hosted runners.' | ||||
|   chownFilesTo: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'User and optionally group (user or user:group or uid:gid) to give ownership of the resulting build artifacts' | ||||
|     description: | ||||
|       'User and optionally group (user or user:group or uid:gid) to give ownership of the resulting build artifacts' | ||||
|   dockerCpuLimit: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Number of CPU cores to assign the docker container. Defaults to all available cores on all platforms.' | ||||
|   dockerMemoryLimit: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: | ||||
|       'Amount of memory to assign the docker container. Defaults to 95% of total system memory rounded down to the | ||||
|       nearest megabyte on Linux and 80% on Windows. On unrecognized platforms, defaults to 75% of total system memory. | ||||
|       To manually specify a value, use the format <number><unit>, where unit is either m or g. ie: 512m = 512 megabytes' | ||||
|   dockerIsolationMode: | ||||
|     required: false | ||||
|     default: 'default' | ||||
|     description: | ||||
|       'Isolation mode to use for the docker container. Can be one of process, hyperv, or default. Default will pick the | ||||
|       default mode as described by Microsoft where server versions use process and desktop versions use hyperv. Only | ||||
|       applicable on Windows' | ||||
|   containerRegistryRepository: | ||||
|     required: false | ||||
|     default: 'unityci/editor' | ||||
|     description: 'Container registry and repository to pull image from. Only applicable if customImage is not set.' | ||||
|   containerRegistryImageVersion: | ||||
|     required: false | ||||
|     default: '3' | ||||
|     description: 'Container registry image version. Only applicable if customImage is not set.' | ||||
|   allowDirtyBuild: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Allows the branch of the build to be dirty, and still generate the build.' | ||||
|     description: '[CloudRunner] Allows the branch of the build to be dirty, and still generate the build.' | ||||
|   postBuildSteps: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'run a post build job in yaml format with the keys image, secrets (name, value object array), command string' | ||||
|     description: | ||||
|       '[CloudRunner] run a post build job in yaml format with the keys image, secrets (name, value object array), | ||||
|       command string' | ||||
|   preBuildSteps: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Run a pre build job after the repository setup but before the build job (in yaml format with the keys image, secrets (name, value object array), command line string)' | ||||
|     description: | ||||
|       '[CloudRunner] Run a pre build job after the repository setup but before the build job (in yaml format with the | ||||
|       keys image, secrets (name, value object array), command line string)' | ||||
|   containerHookFiles: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: | ||||
|       '[CloudRunner] Specify the names (by file name) of custom steps to run before or after cloud runner jobs, must | ||||
|       match a yaml step file inside your repo in the folder .game-ci/steps/' | ||||
|   customHookFiles: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: | ||||
|       '[CloudRunner] Specify the names (by file name) of custom hooks to run before or after cloud runner jobs, must | ||||
|       match a yaml step file inside your repo in the folder .game-ci/hooks/' | ||||
|   customCommandHooks: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: '[CloudRunner] Specify custom commands and trigger hooks (injects commands into jobs)' | ||||
|   customJob: | ||||
|     required: false | ||||
|     default: '' | ||||
|     description: 'Run a custom job instead of the standard build automation for cloud runner (in yaml format with the keys image, secrets (name, value object array), command line string)' | ||||
|   awsBaseStackName: | ||||
|     description: | ||||
|       '[CloudRunner] Run a custom job instead of the standard build automation for cloud runner (in yaml format with the | ||||
|       keys image, secrets (name, value object array), command line string)' | ||||
|   awsStackName: | ||||
|     default: 'game-ci' | ||||
|     required: false | ||||
|     description: 'The Cloud Formation stack name that must be setup before using this option.' | ||||
|   cloudRunnerCluster: | ||||
|     description: '[CloudRunner] The Cloud Formation stack name that must be setup before using this option.' | ||||
|   providerStrategy: | ||||
|     default: 'local' | ||||
|     required: false | ||||
|     description: 'Either local, k8s or aws can be used to run builds on a remote cluster. Additional parameters must be configured.' | ||||
|   cloudRunnerCpu: | ||||
|     default: '1.0' | ||||
|     required: false | ||||
|     description: 'Amount of CPU time to assign the remote build container' | ||||
|   cloudRunnerMemory: | ||||
|     default: '750M' | ||||
|     required: false | ||||
|     description: 'Amount of memory to assign the remote build container' | ||||
|   githubToken: | ||||
|     description: | ||||
|       '[CloudRunner] Either local, k8s or aws can be used to run builds on a remote cluster. Additional parameters must | ||||
|       be configured.' | ||||
|   containerCpu: | ||||
|     default: '' | ||||
|     required: false | ||||
|     description: 'GitHub token for cloning, only needed when kubeconfig is used.' | ||||
|     description: '[CloudRunner] Amount of CPU time to assign the remote build container' | ||||
|   containerMemory: | ||||
|     default: '' | ||||
|     required: false | ||||
|     description: '[CloudRunner] Amount of memory to assign the remote build container' | ||||
|   readInputFromOverrideList: | ||||
|     default: '' | ||||
|     required: false | ||||
|     description: '[CloudRunner] Comma separated list of input value names to read from "input override command"' | ||||
|   readInputOverrideCommand: | ||||
|     default: '' | ||||
|     required: false | ||||
|     description: | ||||
|       '[CloudRunner] Extend game ci by specifying a command to execute to pull input from external source e.g cloud | ||||
|       provider secret managers' | ||||
|   kubeConfig: | ||||
|     default: '' | ||||
|     required: false | ||||
|     description: 'Supply a base64 encoded kubernetes config to run builds on kubernetes and stream logs until completion.' | ||||
|     description: | ||||
|       '[CloudRunner] Supply a base64 encoded kubernetes config to run builds on kubernetes and stream logs until | ||||
|       completion.' | ||||
|   kubeVolume: | ||||
|     default: '' | ||||
|     required: false | ||||
|     description: 'Supply a Persistent Volume Claim name to use for the Unity build.' | ||||
|     description: '[CloudRunner] Supply a Persistent Volume Claim name to use for the Unity build.' | ||||
|   kubeStorageClass: | ||||
|     default: '' | ||||
|     required: false | ||||
|     description: | ||||
|       '[CloudRunner] Kubernetes storage class to use for cloud runner jobs, leave empty to install rook cluster.' | ||||
|   kubeVolumeSize: | ||||
|     default: '5Gi' | ||||
|     required: false | ||||
|     description: 'Amount of disc space to assign the Kubernetes Persistent Volume' | ||||
|     description: '[CloudRunner] Amount of disc space to assign the Kubernetes Persistent Volume' | ||||
|   cacheKey: | ||||
|     default: '' | ||||
|     required: false | ||||
|     description: '[CloudRunner] Cache key to indicate bucket for cache' | ||||
|   watchToEnd: | ||||
|     default: 'true' | ||||
|     required: false | ||||
|     description: | ||||
|       '[CloudRunner] Whether or not to watch the build to the end. Can be used for especially long running jobs e.g | ||||
|       imports or self-hosted ephemeral runners.' | ||||
|   cacheUnityInstallationOnMac: | ||||
|     default: 'false' | ||||
|     required: false | ||||
|     description: 'Whether to cache the Unity hub and editor installation on MacOS' | ||||
|   unityHubVersionOnMac: | ||||
|     default: '' | ||||
|     required: false | ||||
|     description: | ||||
|       'The version of Unity Hub to install on MacOS (e.g. 3.4.0). Defaults to latest available on brew if empty string | ||||
|       or nothing is specified.' | ||||
|   unityLicensingServer: | ||||
|     default: '' | ||||
|     required: false | ||||
|     description: 'The Unity licensing server address to use for activating Unity.' | ||||
|   dockerWorkspacePath: | ||||
|     default: '/github/workspace' | ||||
|     required: false | ||||
|     description: | ||||
|       'The path to mount the workspace inside the docker container. For windows, leave out the drive letter. For example | ||||
|       c:/github/workspace should be defined as /github/workspace' | ||||
|   skipActivation: | ||||
|     default: 'false' | ||||
|     required: false | ||||
|     description: 'Skip the activation/deactivation of Unity. This assumes Unity is already activated.' | ||||
| 
 | ||||
| outputs: | ||||
|   volume: | ||||
|     description: 'The Persistent Volume (PV) where the build artifacts have been stored by Kubernetes' | ||||
|   buildVersion: | ||||
|     description: 'The generated version used for the Unity build' | ||||
|   androidVersionCode: | ||||
|     description: 'The generated versionCode used for the Android Unity build' | ||||
|   engineExitCode: | ||||
|     description: | ||||
|       'Returns the exit code from the build scripts. This code is 0 if the build was successful. If there was an error | ||||
|       during activation, the code is from the activation step. If activation is successful, the code is from the project | ||||
|       build step.' | ||||
| branding: | ||||
|   icon: 'box' | ||||
|   color: 'gray-dark' | ||||
| runs: | ||||
|   using: 'node12' | ||||
|   using: 'node20' | ||||
|   main: 'dist/index.js' | ||||
|  |  | |||
|  | @ -1,13 +0,0 @@ | |||
| Cloud Runner platform selected AWS | ||||
| Cloud Runner is running in custom job mode | ||||
| AWS Region: eu-west-2 | ||||
| Parsing build steps:  | ||||
|     - name: 'step 1' | ||||
|       image: 'alpine' | ||||
|       commands: 'printenv' | ||||
|       secrets: | ||||
|         - name: 'testSecretName' | ||||
|           value: 'testSecretValue' | ||||
|      | ||||
| game-ci stack does not exist (["game-ci-github-automation-424-linux64-a9hz-cleanup","game-ci-github-automation-423-linux64-v34g-cleanup","game-ci-github-automation-423-linux64-v34g","game-ci-github-automation-422-linux64-7x6i-cleanup","game-ci-github-automation-422-linux64-7x6i","game-ci-github-automation-414-linux64-j21p-cleanup","game-ci-github-automation-414-linux64-j21p","game-ci-github-automation-413-linux64-tcih-cleanup","game-ci-github-automation-413-linux64-tcih","game-ci-github-automation-411-linux64-0s69-cleanup","game-ci-github-automation-411-linux64-0s69","game-ci-github-automation-410-linux64-1tli-cleanup","game-ci-github-automation-410-linux64-1tli","game-ci-github-automation-408-linux64-8pbw-cleanup","game-ci-github-automation-408-linux64-8pbw","game-ci-github-automation-407-linux64-21un-cleanup","game-ci-github-automation-407-linux64-21un","game-ci-github-automation-406-linux64-dizb-cleanup","game-ci-github-automation-406-linux64-dizb","game-ci-github-automation-405-linux64-9xj5-cleanup","game-ci-github-automation-405-linux64-9xj5","game-ci-github-automation-402-linux64-0bym-cleanup","game-ci-github-automation-402-linux64-0bym","game-ci-github-automation-400-linux64-arqv-cleanup","game-ci-github-automation-400-linux64-arqv","game-ci-github-automation-399-linux64-utkt-cleanup","game-ci-github-automation-399-linux64-utkt","game-ci-github-automation-397-linux64-xwfu-cleanup","game-ci-github-automation-397-linux64-xwfu","game-ci-github-automation-396-linux64-2g3q-cleanup","game-ci-github-automation-396-linux64-2g3q","game-ci-github-automation","game-ci-stack-integration-tests-390-linux64-mcdw-cleanup","game-ci-stack-integration-tests-390-linux64-mcdw","game-ci-stack-integration-tests-391-linux64-2arq-cleanup","game-ci-stack-integration-tests-391-linux64-2arq","game-ci-stack-integration-tests-390-linux64-awd0-cleanup","game-ci-stack-integration-tests-390-linux64-awd0","game-ci-stack-integration-tests"]) | ||||
| created stack (version: eedce7440581ab2e8a80cee59e34ed64) | ||||
										
											Binary file not shown.
										
									
								
							|  | @ -1,10 +0,0 @@ | |||
| { | ||||
|     "m_SettingKeys": [ | ||||
|         "VR Device Disabled", | ||||
|         "VR Device User Alert" | ||||
|     ], | ||||
|     "m_SettingValues": [ | ||||
|         "False", | ||||
|         "False" | ||||
|     ] | ||||
| } | ||||
|  | @ -1,416 +0,0 @@ | |||
| AWSTemplateFormatVersion: '2010-09-09' | ||||
| Description: AWS Fargate cluster that can span public and private subnets. Supports | ||||
|              public facing load balancers, private internal load balancers, and | ||||
|              both internal and external service discovery namespaces. | ||||
| Parameters: | ||||
|   EnvironmentName: | ||||
|     Type: String | ||||
|     Default: development | ||||
|     Description: "Your deployment environment: DEV, QA , PROD" | ||||
|   Version: | ||||
|     Type: String | ||||
|     Description: "hash of template" | ||||
| 
 | ||||
|   # ContainerPort: | ||||
|   #   Type: Number | ||||
|   #   Default: 80 | ||||
|   #   Description: What port number the application inside the docker container is binding to | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| Mappings: | ||||
|   # Hard values for the subnet masks. These masks define | ||||
|   # the range of internal IP addresses that can be assigned. | ||||
|   # The VPC can have all IP's from 10.0.0.0 to 10.0.255.255 | ||||
|   # There are four subnets which cover the ranges: | ||||
|   # | ||||
|   # 10.0.0.0 - 10.0.0.255 | ||||
|   # 10.0.1.0 - 10.0.1.255 | ||||
|   # 10.0.2.0 - 10.0.2.255 | ||||
|   # 10.0.3.0 - 10.0.3.255 | ||||
| 
 | ||||
|   SubnetConfig: | ||||
|     VPC: | ||||
|       CIDR: '10.0.0.0/16' | ||||
|     PublicOne: | ||||
|       CIDR: '10.0.0.0/24' | ||||
|     PublicTwo: | ||||
|       CIDR: '10.0.1.0/24' | ||||
| 
 | ||||
| Resources: | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|   # VPC in which containers will be networked. | ||||
|   # It has two public subnets, and two private subnets. | ||||
|   # We distribute the subnets across the first two available subnets | ||||
|   # for the region, for high availability. | ||||
|   VPC: | ||||
|     Type: AWS::EC2::VPC | ||||
|     Properties: | ||||
|       EnableDnsSupport: true | ||||
|       EnableDnsHostnames: true | ||||
|       CidrBlock: !FindInMap ['SubnetConfig', 'VPC', 'CIDR'] | ||||
| 
 | ||||
|   EFSServerSecurityGroup: | ||||
|     Type: AWS::EC2::SecurityGroup | ||||
|     Properties: | ||||
|       GroupName: "efs-server-endpoints" | ||||
|       GroupDescription: Which client ip addrs are allowed to access EFS server | ||||
|       VpcId: !Ref 'VPC' | ||||
|       SecurityGroupIngress: | ||||
|         - IpProtocol: tcp | ||||
|           FromPort: 2049 | ||||
|           ToPort: 2049 | ||||
|           SourceSecurityGroupId: !Ref ContainerSecurityGroup | ||||
|           #CidrIp: !FindInMap ['SubnetConfig', 'VPC', 'CIDR'] | ||||
|     # A security group for the containers we will run in Fargate. | ||||
|   # Rules are added to this security group based on what ingress you | ||||
|   # add for the cluster. | ||||
|   ContainerSecurityGroup: | ||||
|     Type: AWS::EC2::SecurityGroup | ||||
|     Properties: | ||||
|       GroupName: "task security group" | ||||
|       GroupDescription: Access to the Fargate containers | ||||
|       VpcId: !Ref 'VPC' | ||||
|       # SecurityGroupIngress: | ||||
|       #   - IpProtocol: tcp | ||||
|       #     FromPort: !Ref ContainerPort | ||||
|       #     ToPort: !Ref ContainerPort | ||||
|       #     CidrIp: 0.0.0.0/0 | ||||
|       SecurityGroupEgress: | ||||
|         - IpProtocol: -1 | ||||
|           FromPort: 2049 | ||||
|           ToPort: 2049 | ||||
|           CidrIp: "0.0.0.0/0" | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|   # Two public subnets, where containers can have public IP addresses | ||||
|   PublicSubnetOne: | ||||
|     Type: AWS::EC2::Subnet | ||||
|     Properties: | ||||
|       AvailabilityZone: !Select | ||||
|         - 0 | ||||
|         - Fn::GetAZs: !Ref 'AWS::Region' | ||||
|       VpcId: !Ref 'VPC' | ||||
|       CidrBlock: !FindInMap ['SubnetConfig', 'PublicOne', 'CIDR'] | ||||
|     #  MapPublicIpOnLaunch: true | ||||
| 
 | ||||
|   PublicSubnetTwo: | ||||
|     Type: AWS::EC2::Subnet | ||||
|     Properties: | ||||
|       AvailabilityZone: !Select | ||||
|         - 1 | ||||
|         - Fn::GetAZs: !Ref 'AWS::Region' | ||||
|       VpcId: !Ref 'VPC' | ||||
|       CidrBlock: !FindInMap ['SubnetConfig', 'PublicTwo', 'CIDR'] | ||||
|     #  MapPublicIpOnLaunch: true | ||||
| 
 | ||||
| 
 | ||||
|   # Setup networking resources for the public subnets. Containers | ||||
|   # in the public subnets have public IP addresses and the routing table | ||||
|   # sends network traffic via the internet gateway. | ||||
|   InternetGateway: | ||||
|     Type: AWS::EC2::InternetGateway | ||||
|   GatewayAttachement: | ||||
|     Type: AWS::EC2::VPCGatewayAttachment | ||||
|     Properties: | ||||
|       VpcId: !Ref 'VPC' | ||||
|       InternetGatewayId: !Ref 'InternetGateway' | ||||
| 
 | ||||
|   # Attaching a Internet Gateway to route table makes it public. | ||||
|   PublicRouteTable: | ||||
|     Type: AWS::EC2::RouteTable | ||||
|     Properties: | ||||
|       VpcId: !Ref 'VPC' | ||||
|   PublicRoute: | ||||
|     Type: AWS::EC2::Route | ||||
|     DependsOn: GatewayAttachement | ||||
|     Properties: | ||||
|       RouteTableId: !Ref 'PublicRouteTable' | ||||
|       DestinationCidrBlock: '0.0.0.0/0' | ||||
|       GatewayId: !Ref 'InternetGateway' | ||||
| 
 | ||||
|   # Attaching a public route table makes a subnet public. | ||||
|   PublicSubnetOneRouteTableAssociation: | ||||
|     Type: AWS::EC2::SubnetRouteTableAssociation | ||||
|     Properties: | ||||
|       SubnetId: !Ref PublicSubnetOne | ||||
|       RouteTableId: !Ref PublicRouteTable | ||||
|   PublicSubnetTwoRouteTableAssociation: | ||||
|     Type: AWS::EC2::SubnetRouteTableAssociation | ||||
|     Properties: | ||||
|       SubnetId: !Ref PublicSubnetTwo | ||||
|       RouteTableId: !Ref PublicRouteTable | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|   # ECS Resources | ||||
|   ECSCluster: | ||||
|     Type: AWS::ECS::Cluster | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|   # A role used to allow AWS Autoscaling to inspect stats and adjust scaleable targets | ||||
|   # on your AWS account | ||||
|   AutoscalingRole: | ||||
|     Type: AWS::IAM::Role | ||||
|     Properties: | ||||
|       AssumeRolePolicyDocument: | ||||
|         Statement: | ||||
|         - Effect: Allow | ||||
|           Principal: | ||||
|             Service: [application-autoscaling.amazonaws.com] | ||||
|           Action: ['sts:AssumeRole'] | ||||
|       Path: / | ||||
|       Policies: | ||||
|       - PolicyName: service-autoscaling | ||||
|         PolicyDocument: | ||||
|           Statement: | ||||
|           - Effect: Allow | ||||
|             Action: | ||||
|               - 'application-autoscaling:*' | ||||
|               - 'cloudwatch:DescribeAlarms' | ||||
|               - 'cloudwatch:PutMetricAlarm' | ||||
|               - 'ecs:DescribeServices' | ||||
|               - 'ecs:UpdateService' | ||||
|             Resource: '*' | ||||
| 
 | ||||
|   # This is an IAM role which authorizes ECS to manage resources on your | ||||
|   # account on your behalf, such as updating your load balancer with the | ||||
|   # details of where your containers are, so that traffic can reach your | ||||
|   # containers. | ||||
|   ECSRole: | ||||
|     Type: AWS::IAM::Role | ||||
|     Properties: | ||||
|       AssumeRolePolicyDocument: | ||||
|         Statement: | ||||
|         - Effect: Allow | ||||
|           Principal: | ||||
|             Service: [ecs.amazonaws.com] | ||||
|           Action: ['sts:AssumeRole'] | ||||
|       Path: / | ||||
|       Policies: | ||||
|       - PolicyName: ecs-service | ||||
|         PolicyDocument: | ||||
|           Statement: | ||||
|           - Effect: Allow | ||||
|             Action: | ||||
|               # Rules which allow ECS to attach network interfaces to instances | ||||
|               # on your behalf in order for awsvpc networking mode to work right | ||||
|               - 'ec2:AttachNetworkInterface' | ||||
|               - 'ec2:CreateNetworkInterface' | ||||
|               - 'ec2:CreateNetworkInterfacePermission' | ||||
|               - 'ec2:DeleteNetworkInterface' | ||||
|               - 'ec2:DeleteNetworkInterfacePermission' | ||||
|               - 'ec2:Describe*' | ||||
|               - 'ec2:DetachNetworkInterface' | ||||
| 
 | ||||
|               # Rules which allow ECS to update load balancers on your behalf | ||||
|               # with the information sabout how to send traffic to your containers | ||||
|               - 'elasticloadbalancing:DeregisterInstancesFromLoadBalancer' | ||||
|               - 'elasticloadbalancing:DeregisterTargets' | ||||
|               - 'elasticloadbalancing:Describe*' | ||||
|               - 'elasticloadbalancing:RegisterInstancesWithLoadBalancer' | ||||
|               - 'elasticloadbalancing:RegisterTargets' | ||||
|             Resource: '*' | ||||
| 
 | ||||
|   # This is a role which is used by the ECS tasks themselves. | ||||
|   ECSTaskExecutionRole: | ||||
|     Type: AWS::IAM::Role | ||||
|     Properties: | ||||
|       AssumeRolePolicyDocument: | ||||
|         Statement: | ||||
|         - Effect: Allow | ||||
|           Principal: | ||||
|             Service: [ecs-tasks.amazonaws.com] | ||||
|           Action: ['sts:AssumeRole'] | ||||
|       Path: / | ||||
|       Policies: | ||||
|         - PolicyName: AmazonECSTaskExecutionRolePolicy | ||||
|           PolicyDocument: | ||||
|             Statement: | ||||
|             - Effect: Allow | ||||
|               Action: | ||||
| 
 | ||||
|                 # Allow the use of secret manager | ||||
|                 - 'secretsmanager:GetSecretValue' | ||||
|                 - 'kms:Decrypt' | ||||
| 
 | ||||
|                 # Allow the ECS Tasks to download images from ECR | ||||
|                 - 'ecr:GetAuthorizationToken' | ||||
|                 - 'ecr:BatchCheckLayerAvailability' | ||||
|                 - 'ecr:GetDownloadUrlForLayer' | ||||
|                 - 'ecr:BatchGetImage' | ||||
| 
 | ||||
|                 # Allow the ECS tasks to upload logs to CloudWatch | ||||
|                 - 'logs:CreateLogStream' | ||||
|                 - 'logs:PutLogEvents' | ||||
|               Resource: '*' | ||||
| 
 | ||||
|   DeleteCFNLambdaExecutionRole: | ||||
|     Type: "AWS::IAM::Role" | ||||
|     Properties: | ||||
|       AssumeRolePolicyDocument: | ||||
|         Version: "2012-10-17" | ||||
|         Statement: | ||||
|         - Effect: "Allow" | ||||
|           Principal: | ||||
|             Service: ["lambda.amazonaws.com"] | ||||
|           Action: "sts:AssumeRole" | ||||
|       Path: "/" | ||||
|       Policies: | ||||
|       - PolicyName: DeleteCFNLambdaExecutionRole | ||||
|         PolicyDocument: | ||||
|           Version: "2012-10-17" | ||||
|           Statement: | ||||
|           - Effect: "Allow" | ||||
|             Action: | ||||
|             - "logs:CreateLogGroup" | ||||
|             - "logs:CreateLogStream" | ||||
|             - "logs:PutLogEvents" | ||||
|             Resource: "arn:aws:logs:*:*:*" | ||||
|           - Effect: "Allow" | ||||
|             Action: | ||||
|             - "cloudformation:DeleteStack" | ||||
|             - "kinesis:DeleteStream" | ||||
|             - "secretsmanager:DeleteSecret" | ||||
|             - "kinesis:DescribeStreamSummary" | ||||
|             - "logs:DeleteLogGroup" | ||||
|             - "logs:DeleteSubscriptionFilter" | ||||
|             - "ecs:DeregisterTaskDefinition" | ||||
|             - "lambda:DeleteFunction" | ||||
|             - "lambda:InvokeFunction" | ||||
|             - "events:RemoveTargets" | ||||
|             - "events:DeleteRule" | ||||
|             - "lambda:RemovePermission" | ||||
|             Resource: "*" | ||||
| 
 | ||||
| ### cloud watch to kinesis role | ||||
| 
 | ||||
|   CloudWatchIAMRole: | ||||
|     Type: AWS::IAM::Role | ||||
|     Properties: | ||||
|       AssumeRolePolicyDocument: | ||||
|         Statement: | ||||
|         - Effect: Allow | ||||
|           Principal: | ||||
|             Service: [logs.amazonaws.com] | ||||
|           Action: ['sts:AssumeRole'] | ||||
|       Path: / | ||||
|       Policies: | ||||
|       - PolicyName: service-autoscaling | ||||
|         PolicyDocument: | ||||
|           Statement: | ||||
|           - Effect: Allow | ||||
|             Action: | ||||
|               - 'kinesis:PutRecord' | ||||
|             Resource: '*' | ||||
| #####################EFS##################### | ||||
| 
 | ||||
|   EfsFileStorage: | ||||
|     Type: 'AWS::EFS::FileSystem' | ||||
|     Properties: | ||||
|       BackupPolicy: | ||||
|         Status: ENABLED | ||||
|       PerformanceMode: maxIO | ||||
|       Encrypted: false | ||||
| 
 | ||||
| 
 | ||||
|       FileSystemPolicy: | ||||
|         Version: "2012-10-17" | ||||
|         Statement: | ||||
|           - Effect: "Allow" | ||||
|             Action: | ||||
|               - "elasticfilesystem:ClientMount" | ||||
|               - "elasticfilesystem:ClientWrite" | ||||
|               - "elasticfilesystem:ClientRootAccess" | ||||
|             Principal: | ||||
|               AWS: "*" | ||||
| 
 | ||||
| 
 | ||||
|   MountTargetResource1: | ||||
|     Type: AWS::EFS::MountTarget | ||||
|     Properties: | ||||
|       FileSystemId: !Ref EfsFileStorage | ||||
|       SubnetId: !Ref PublicSubnetOne | ||||
|       SecurityGroups: | ||||
|       - !Ref EFSServerSecurityGroup | ||||
| 
 | ||||
|   MountTargetResource2: | ||||
|     Type: AWS::EFS::MountTarget | ||||
|     Properties: | ||||
|       FileSystemId: !Ref EfsFileStorage | ||||
|       SubnetId: !Ref PublicSubnetTwo | ||||
|       SecurityGroups: | ||||
|       - !Ref EFSServerSecurityGroup | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| Outputs: | ||||
| 
 | ||||
|   EfsFileStorageId: | ||||
|     Description: 'The connection endpoint for the database.' | ||||
|     Value: !Ref EfsFileStorage | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:EfsFileStorageId | ||||
|   ClusterName: | ||||
|     Description: The name of the ECS cluster | ||||
|     Value: !Ref 'ECSCluster' | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:ClusterName | ||||
|   AutoscalingRole: | ||||
|     Description: The ARN of the role used for autoscaling | ||||
|     Value: !GetAtt 'AutoscalingRole.Arn' | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:AutoscalingRole | ||||
|   ECSRole: | ||||
|     Description: The ARN of the ECS role | ||||
|     Value: !GetAtt 'ECSRole.Arn' | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:ECSRole | ||||
|   ECSTaskExecutionRole: | ||||
|     Description: The ARN of the ECS role tsk execution role | ||||
|     Value: !GetAtt 'ECSTaskExecutionRole.Arn' | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:ECSTaskExecutionRole | ||||
| 
 | ||||
|   DeleteCFNLambdaExecutionRole: | ||||
|     Description: Lambda execution role for cleaning up cloud formations | ||||
|     Value: !GetAtt 'DeleteCFNLambdaExecutionRole.Arn' | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:DeleteCFNLambdaExecutionRole | ||||
| 
 | ||||
|   CloudWatchIAMRole: | ||||
|     Description: The ARN of the CloudWatch role for subscription filter | ||||
|     Value: !GetAtt 'CloudWatchIAMRole.Arn' | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:CloudWatchIAMRole | ||||
|   VpcId: | ||||
|     Description: The ID of the VPC that this stack is deployed in | ||||
|     Value: !Ref 'VPC' | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:VpcId | ||||
|   PublicSubnetOne: | ||||
|     Description: Public subnet one | ||||
|     Value: !Ref 'PublicSubnetOne' | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:PublicSubnetOne | ||||
|   PublicSubnetTwo: | ||||
|     Description: Public subnet two | ||||
|     Value: !Ref 'PublicSubnetTwo' | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:PublicSubnetTwo | ||||
| 
 | ||||
|   ContainerSecurityGroup: | ||||
|     Description: A security group used to allow Fargate containers to receive traffic | ||||
|     Value: !Ref 'ContainerSecurityGroup' | ||||
|     Export: | ||||
|       Name: !Sub ${EnvironmentName}:ContainerSecurityGroup | ||||
|  | @ -1,221 +0,0 @@ | |||
| AWSTemplateFormatVersion: 2010-09-09 | ||||
| Description: >- | ||||
|   AWS Fargate cluster that can span public and private subnets. Supports public | ||||
|   facing load balancers, private internal load balancers, and both internal and | ||||
|   external service discovery namespaces. | ||||
| Parameters: | ||||
|   EnvironmentName: | ||||
|     Type: String | ||||
|     Default: development | ||||
|     Description: 'Your deployment environment: DEV, QA , PROD' | ||||
|   ServiceName: | ||||
|     Type: String | ||||
|     Default: example | ||||
|     Description: A name for the service | ||||
|   ImageUrl: | ||||
|     Type: String | ||||
|     Default: nginx | ||||
|     Description: >- | ||||
|       The url of a docker image that contains the application process that will | ||||
|       handle the traffic for this service | ||||
|   ContainerPort: | ||||
|     Type: Number | ||||
|     Default: 80 | ||||
|     Description: What port number the application inside the docker container is binding to | ||||
|   ContainerCpu: | ||||
|     Type: Number | ||||
|     Default: 1024 | ||||
|     Description: How much CPU to give the container. 1024 is 1 CPU | ||||
|   ContainerMemory: | ||||
|     Type: Number | ||||
|     Default: 2048 | ||||
|     Description: How much memory in megabytes to give the container | ||||
|   BUILDGUID: | ||||
|     Type: String | ||||
|     Default: '' | ||||
|   Command: | ||||
|     Type: String | ||||
|     Default: 'ls' | ||||
|   EntryPoint: | ||||
|     Type: String | ||||
|     Default: '/bin/sh' | ||||
|   WorkingDirectory: | ||||
|     Type: String | ||||
|     Default: '/efsdata/' | ||||
|   Role: | ||||
|     Type: String | ||||
|     Default: '' | ||||
|     Description: >- | ||||
|       (Optional) An IAM role to give the service's containers if the code within | ||||
|       needs to access other AWS resources | ||||
|   EFSMountDirectory: | ||||
|     Type: String | ||||
|     Default: '/efsdata' | ||||
|   # template secrets p1 - input | ||||
| Mappings: | ||||
|   SubnetConfig: | ||||
|     VPC: | ||||
|       CIDR: 10.0.0.0/16 | ||||
|     PublicOne: | ||||
|       CIDR: 10.0.0.0/24 | ||||
|     PublicTwo: | ||||
|       CIDR: 10.0.1.0/24 | ||||
| Conditions: | ||||
|   HasCustomRole: !Not | ||||
|     - !Equals | ||||
|       - Ref: Role | ||||
|       - '' | ||||
| Resources: | ||||
|   LogGroup: | ||||
|     Type: 'AWS::Logs::LogGroup' | ||||
|     Properties: | ||||
|       LogGroupName: !Ref ServiceName | ||||
|     Metadata: | ||||
|       'AWS::CloudFormation::Designer': | ||||
|         id: aece53ae-b82d-4267-bc16-ed964b05db27 | ||||
|   SubscriptionFilter: | ||||
|     Type: 'AWS::Logs::SubscriptionFilter' | ||||
|     Properties: | ||||
|       FilterPattern: '' | ||||
|       RoleArn: | ||||
|         'Fn::ImportValue': !Sub '${EnvironmentName}:CloudWatchIAMRole' | ||||
|       LogGroupName: !Ref ServiceName | ||||
|       DestinationArn: | ||||
|         'Fn::GetAtt': | ||||
|           - KinesisStream | ||||
|           - Arn | ||||
|     Metadata: | ||||
|       'AWS::CloudFormation::Designer': | ||||
|         id: 7f809e91-9e5d-4678-98c1-c5085956c480 | ||||
|     DependsOn: | ||||
|       - LogGroup | ||||
|       - KinesisStream | ||||
|   KinesisStream: | ||||
|     Type: 'AWS::Kinesis::Stream' | ||||
|     Properties: | ||||
|       Name: !Ref ServiceName | ||||
|       ShardCount: 1 | ||||
|     Metadata: | ||||
|       'AWS::CloudFormation::Designer': | ||||
|         id: c6f18447-b879-4696-8873-f981b2cedd2b | ||||
| 
 | ||||
|   # template secrets p2 - secret | ||||
| 
 | ||||
|   TaskDefinition: | ||||
|     Type: 'AWS::ECS::TaskDefinition' | ||||
|     Properties: | ||||
|       Family: !Ref ServiceName | ||||
|       Cpu: !Ref ContainerCpu | ||||
|       Memory: !Ref ContainerMemory | ||||
|       NetworkMode: awsvpc | ||||
|       Volumes: | ||||
|         - Name: efs-data | ||||
|           EFSVolumeConfiguration: | ||||
|             FilesystemId: | ||||
|               'Fn::ImportValue': !Sub '${EnvironmentName}:EfsFileStorageId' | ||||
|             TransitEncryption: ENABLED | ||||
|       RequiresCompatibilities: | ||||
|         - FARGATE | ||||
|       ExecutionRoleArn: | ||||
|         'Fn::ImportValue': !Sub '${EnvironmentName}:ECSTaskExecutionRole' | ||||
|       TaskRoleArn: | ||||
|         'Fn::If': | ||||
|           - HasCustomRole | ||||
|           - !Ref Role | ||||
|           - !Ref 'AWS::NoValue' | ||||
|       ContainerDefinitions: | ||||
|         - Name: !Ref ServiceName | ||||
|           Cpu: !Ref ContainerCpu | ||||
|           Memory: !Ref ContainerMemory | ||||
|           Image: !Ref ImageUrl | ||||
|           EntryPoint: | ||||
|             Fn::Split: | ||||
|                 - "," | ||||
|                 - !Ref EntryPoint | ||||
|           Command: | ||||
|             Fn::Split: | ||||
|                 - "," | ||||
|                 - !Ref Command | ||||
|           WorkingDirectory: !Ref WorkingDirectory | ||||
|           Environment: | ||||
|             - Name: ALLOW_EMPTY_PASSWORD | ||||
|               Value: 'yes' | ||||
|             # template - env vars | ||||
|           MountPoints: | ||||
|             - SourceVolume: efs-data | ||||
|               ContainerPath: !Ref EFSMountDirectory | ||||
|               ReadOnly: false | ||||
|           Secrets: | ||||
|             # template secrets p3 - container def | ||||
|           LogConfiguration: | ||||
|             LogDriver: awslogs | ||||
|             Options: | ||||
|               awslogs-group: !Ref ServiceName | ||||
|               awslogs-region: !Ref 'AWS::Region' | ||||
|               awslogs-stream-prefix: !Ref ServiceName | ||||
|     Metadata: | ||||
|       'AWS::CloudFormation::Designer': | ||||
|         id: dabb0116-abe0-48a6-a8af-cf9111c879a5 | ||||
|     DependsOn: | ||||
|       - LogGroup | ||||
| Metadata: | ||||
|   'AWS::CloudFormation::Designer': | ||||
|     dabb0116-abe0-48a6-a8af-cf9111c879a5: | ||||
|       size: | ||||
|         width: 60 | ||||
|         height: 60 | ||||
|       position: | ||||
|         x: 270 | ||||
|         'y': 90 | ||||
|       z: 1 | ||||
|       embeds: [] | ||||
|       dependson: | ||||
|         - aece53ae-b82d-4267-bc16-ed964b05db27 | ||||
|     c6f18447-b879-4696-8873-f981b2cedd2b: | ||||
|       size: | ||||
|         width: 60 | ||||
|         height: 60 | ||||
|       position: | ||||
|         x: 270 | ||||
|         'y': 210 | ||||
|       z: 1 | ||||
|       embeds: [] | ||||
|     7f809e91-9e5d-4678-98c1-c5085956c480: | ||||
|       size: | ||||
|         width: 60 | ||||
|         height: 60 | ||||
|       position: | ||||
|         x: 60 | ||||
|         'y': 300 | ||||
|       z: 1 | ||||
|       embeds: [] | ||||
|       dependson: | ||||
|         - aece53ae-b82d-4267-bc16-ed964b05db27 | ||||
|         - c6f18447-b879-4696-8873-f981b2cedd2b | ||||
|     aece53ae-b82d-4267-bc16-ed964b05db27: | ||||
|       size: | ||||
|         width: 150 | ||||
|         height: 150 | ||||
|       position: | ||||
|         x: 60 | ||||
|         'y': 90 | ||||
|       z: 1 | ||||
|       embeds: [] | ||||
|     4d2da56c-3643-46b8-aaee-e46e19f95fcc: | ||||
|       source: | ||||
|         id: 7f809e91-9e5d-4678-98c1-c5085956c480 | ||||
|       target: | ||||
|         id: aece53ae-b82d-4267-bc16-ed964b05db27 | ||||
|       z: 11 | ||||
|     14eb957b-f094-4653-93c4-77b2f851953c: | ||||
|       source: | ||||
|         id: 7f809e91-9e5d-4678-98c1-c5085956c480 | ||||
|       target: | ||||
|         id: c6f18447-b879-4696-8873-f981b2cedd2b | ||||
|       z: 12 | ||||
|     85c57444-e5bb-4230-bc85-e545cd4558f6: | ||||
|       source: | ||||
|         id: dabb0116-abe0-48a6-a8af-cf9111c879a5 | ||||
|       target: | ||||
|         id: aece53ae-b82d-4267-bc16-ed964b05db27 | ||||
|       z: 13 | ||||
|  | @ -1,709 +0,0 @@ | |||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <Project ToolsVersion="4.0" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003"> | ||||
|   <PropertyGroup> | ||||
|     <LangVersion>latest</LangVersion> | ||||
|     <_TargetFrameworkDirectories>non_empty_path_generated_by_unity.rider.package</_TargetFrameworkDirectories> | ||||
|     <_FullFrameworkReferenceAssemblyPaths>non_empty_path_generated_by_unity.rider.package</_FullFrameworkReferenceAssemblyPaths> | ||||
|     <DisableHandlePackageFileConflicts>true</DisableHandlePackageFileConflicts> | ||||
|   </PropertyGroup> | ||||
|   <PropertyGroup> | ||||
|     <Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration> | ||||
|     <Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform> | ||||
|     <ProductVersion>10.0.20506</ProductVersion> | ||||
|     <SchemaVersion>2.0</SchemaVersion> | ||||
|     <RootNamespace></RootNamespace> | ||||
|     <ProjectGuid>{B7F8614B-1EC2-9D3A-DA1C-4D279A867D74}</ProjectGuid> | ||||
|     <OutputType>Library</OutputType> | ||||
|     <AppDesignerFolder>Properties</AppDesignerFolder> | ||||
|     <AssemblyName>Assembly-CSharp-Editor</AssemblyName> | ||||
|     <TargetFrameworkVersion>v4.7.1</TargetFrameworkVersion> | ||||
|     <FileAlignment>512</FileAlignment> | ||||
|     <BaseDirectory>.</BaseDirectory> | ||||
|   </PropertyGroup> | ||||
|   <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' "> | ||||
|     <DebugSymbols>true</DebugSymbols> | ||||
|     <DebugType>full</DebugType> | ||||
|     <Optimize>false</Optimize> | ||||
|     <OutputPath>Temp\bin\Debug\</OutputPath> | ||||
|     <DefineConstants>DEBUG;TRACE;UNITY_2019_2_11;UNITY_2019_2;UNITY_2019;UNITY_5_3_OR_NEWER;UNITY_5_4_OR_NEWER;UNITY_5_5_OR_NEWER;UNITY_5_6_OR_NEWER;UNITY_2017_1_OR_NEWER;UNITY_2017_2_OR_NEWER;UNITY_2017_3_OR_NEWER;UNITY_2017_4_OR_NEWER;UNITY_2018_1_OR_NEWER;UNITY_2018_2_OR_NEWER;UNITY_2018_3_OR_NEWER;UNITY_2018_4_OR_NEWER;UNITY_2019_1_OR_NEWER;UNITY_2019_2_OR_NEWER;PLATFORM_ARCH_64;UNITY_64;UNITY_INCLUDE_TESTS;ENABLE_AUDIO;ENABLE_CACHING;ENABLE_CLOTH;ENABLE_MICROPHONE;ENABLE_MULTIPLE_DISPLAYS;ENABLE_PHYSICS;ENABLE_TEXTURE_STREAMING;ENABLE_UNET;ENABLE_LZMA;ENABLE_UNITYEVENTS;ENABLE_WEBCAM;ENABLE_WWW;ENABLE_CLOUD_SERVICES_COLLAB;ENABLE_CLOUD_SERVICES_COLLAB_SOFTLOCKS;ENABLE_CLOUD_SERVICES_ADS;ENABLE_CLOUD_SERVICES_USE_WEBREQUEST;ENABLE_CLOUD_SERVICES_UNET;ENABLE_CLOUD_SERVICES_BUILD;ENABLE_CLOUD_LICENSE;ENABLE_EDITOR_HUB_LICENSE;ENABLE_WEBSOCKET_CLIENT;ENABLE_DIRECTOR_AUDIO;ENABLE_DIRECTOR_TEXTURE;ENABLE_MANAGED_JOBS;ENABLE_MANAGED_TRANSFORM_JOBS;ENABLE_MANAGED_ANIMATION_JOBS;ENABLE_MANAGED_AUDIO_JOBS;INCLUDE_DYNAMIC_GI;ENABLE_MONO_BDWGC;ENABLE_SCRIPTING_GC_WBARRIERS;PLATFORM_SUPPORTS_MONO;RENDER_SOFTWARE_CURSOR;ENABLE_VIDEO;PLATFORM_STANDALONE_WIN;PLATFORM_STANDALONE;UNITY_STANDALONE_WIN;UNITY_STANDALONE;ENABLE_RUNTIME_GI;ENABLE_MOVIES;ENABLE_NETWORK;ENABLE_CRUNCH_TEXTURE_COMPRESSION;ENABLE_UNITYWEBREQUEST;ENABLE_CLOUD_SERVICES;ENABLE_CLOUD_SERVICES_ANALYTICS;ENABLE_CLOUD_SERVICES_PURCHASING;ENABLE_CLOUD_SERVICES_CRASH_REPORTING;ENABLE_OUT_OF_PROCESS_CRASH_HANDLER;ENABLE_EVENT_QUEUE;ENABLE_CLUSTER_SYNC;ENABLE_CLUSTERINPUT;ENABLE_VR;ENABLE_AR;ENABLE_WEBSOCKET_HOST;ENABLE_MONO;NET_STANDARD_2_0;ENABLE_PROFILER;UNITY_ASSERTIONS;UNITY_EDITOR;UNITY_EDITOR_64;UNITY_EDITOR_WIN;ENABLE_UNITY_COLLECTIONS_CHECKS;ENABLE_BURST_AOT;UNITY_TEAM_LICENSE;ENABLE_CUSTOM_RENDER_TEXTURE;ENABLE_DIRECTOR;ENABLE_LOCALIZATION;ENABLE_SPRITES;ENABLE_TERRAIN;ENABLE_TILEMAP;ENABLE_TIMELINE;ENABLE_LEGACY_INPUT_MANAGER;NET_4_6;CSHARP_7_OR_LATER;CSHARP_7_3_OR_NEWER</DefineConstants> | ||||
|     <ErrorReport>prompt</ErrorReport> | ||||
|     <WarningLevel>4</WarningLevel> | ||||
|     <NoWarn>0169</NoWarn> | ||||
|     <AllowUnsafeBlocks>False</AllowUnsafeBlocks> | ||||
|   </PropertyGroup> | ||||
|   <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' "> | ||||
|     <DebugType>pdbonly</DebugType> | ||||
|     <Optimize>true</Optimize> | ||||
|     <OutputPath>Temp\bin\Release\</OutputPath> | ||||
|     <ErrorReport>prompt</ErrorReport> | ||||
|     <WarningLevel>4</WarningLevel> | ||||
|     <NoWarn>0169</NoWarn> | ||||
|     <AllowUnsafeBlocks>False</AllowUnsafeBlocks> | ||||
|   </PropertyGroup> | ||||
|   <PropertyGroup> | ||||
|     <NoConfig>true</NoConfig> | ||||
|     <NoStdLib>true</NoStdLib> | ||||
|     <AddAdditionalExplicitAssemblyReferences>false</AddAdditionalExplicitAssemblyReferences> | ||||
|     <ImplicitlyExpandNETStandardFacades>false</ImplicitlyExpandNETStandardFacades> | ||||
|     <ImplicitlyExpandDesignTimeFacades>false</ImplicitlyExpandDesignTimeFacades> | ||||
|   </PropertyGroup> | ||||
|   <ItemGroup> | ||||
|     <Reference Include="UnityEngine"> | ||||
|       <HintPath>C:\Program Files\Unity\2019.2.11f1\Editor\Data\Managed/UnityEngine/UnityEngine.dll</HintPath> | ||||
|     </Reference> | ||||
|     <Reference Include="UnityEditor"> | ||||
|       <HintPath>C:\Program Files\Unity\2019.2.11f1\Editor\Data\Managed/UnityEditor.dll</HintPath> | ||||
|     </Reference> | ||||
|   </ItemGroup> | ||||
|   <ItemGroup> | ||||
|      <Compile Include="Assets\Editor\Builder.cs" /> | ||||
|      <Compile Include="Assets\Editor\Input\ArgumentsParser.cs" /> | ||||
|      <Compile Include="Assets\Editor\Reporting\StdOutReporter.cs" /> | ||||
|      <Compile Include="Assets\Editor\System\ProcessExtensions.cs" /> | ||||
|      <Compile Include="Assets\Editor\Versioning\VersionApplicator.cs" /> | ||||
|      <Compile Include="Assets\Editor\Versioning\Git.cs" /> | ||||
|      <Compile Include="Assets\Editor\Versioning\VersionGenerator.cs" /> | ||||
|      <Compile Include="Assets\Editor\Versioning\GitException.cs" /> | ||||
|  <Reference Include="UnityEditor.TestRunner"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/UnityEditor.TestRunner.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.TestRunner"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/UnityEngine.TestRunner.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.Timeline.Editor"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/Unity.Timeline.Editor.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="com.unity.multiplayer-hlapi.Editor"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/com.unity.multiplayer-hlapi.Editor.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.VSCode.Editor"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/Unity.VSCode.Editor.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.TextMeshPro.Editor"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/Unity.TextMeshPro.Editor.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UI"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/UnityEngine.UI.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.Timeline"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/Unity.Timeline.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.CollabProxy.Editor"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/Unity.CollabProxy.Editor.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="com.unity.multiplayer-weaver.Editor"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/com.unity.multiplayer-weaver.Editor.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.XR.LegacyInputHelpers"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/UnityEngine.XR.LegacyInputHelpers.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.Rider.Editor"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/Unity.Rider.Editor.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.2D.Sprite.Editor"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/Unity.2D.Sprite.Editor.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.2D.Tilemap.Editor"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/Unity.2D.Tilemap.Editor.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.SpatialTracking"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/UnityEditor.SpatialTracking.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.SpatialTracking"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/UnityEngine.SpatialTracking.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.TextMeshPro"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/Unity.TextMeshPro.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.Analytics.DataPrivacy"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/Unity.Analytics.DataPrivacy.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.XR.LegacyInputHelpers"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/UnityEditor.XR.LegacyInputHelpers.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.UI"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/UnityEditor.UI.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="com.unity.multiplayer-hlapi.Runtime"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/ScriptAssemblies/com.unity.multiplayer-hlapi.Runtime.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.AIModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.AIModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.ARModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.ARModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.AccessibilityModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.AccessibilityModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.AndroidJNIModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.AndroidJNIModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.AnimationModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.AnimationModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.AssetBundleModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.AssetBundleModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.AudioModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.AudioModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.ClothModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.ClothModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.ClusterInputModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.ClusterInputModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.ClusterRendererModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.ClusterRendererModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.CoreModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.CoreModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.CrashReportingModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.CrashReportingModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.DSPGraphModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.DSPGraphModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.DirectorModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.DirectorModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.FileSystemHttpModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.FileSystemHttpModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.GameCenterModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.GameCenterModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.GridModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.GridModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.HotReloadModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.HotReloadModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.IMGUIModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.IMGUIModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.ImageConversionModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.ImageConversionModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.InputModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.InputModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.InputLegacyModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.InputLegacyModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.JSONSerializeModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.JSONSerializeModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.LocalizationModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.LocalizationModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.ParticleSystemModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.ParticleSystemModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.PerformanceReportingModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.PerformanceReportingModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.PhysicsModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.PhysicsModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.Physics2DModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.Physics2DModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.ProfilerModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.ProfilerModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.ScreenCaptureModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.ScreenCaptureModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.SharedInternalsModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.SharedInternalsModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.SpriteMaskModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.SpriteMaskModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.SpriteShapeModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.SpriteShapeModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.StreamingModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.StreamingModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.SubstanceModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.SubstanceModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.TLSModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.TLSModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.TerrainModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.TerrainModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.TerrainPhysicsModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.TerrainPhysicsModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.TextCoreModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.TextCoreModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.TextRenderingModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.TextRenderingModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.TilemapModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.TilemapModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UIModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UIModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UIElementsModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UIElementsModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UNETModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UNETModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UmbraModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UmbraModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UnityAnalyticsModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UnityAnalyticsModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UnityConnectModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UnityConnectModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UnityTestProtocolModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UnityTestProtocolModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UnityWebRequestModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UnityWebRequestModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UnityWebRequestAssetBundleModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UnityWebRequestAssetBundleModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UnityWebRequestAudioModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UnityWebRequestAudioModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UnityWebRequestTextureModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UnityWebRequestTextureModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.UnityWebRequestWWWModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.UnityWebRequestWWWModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.VFXModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.VFXModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.VRModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.VRModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.VehiclesModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.VehiclesModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.VideoModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.VideoModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.WindModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.WindModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEngine.XRModule"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEngine/UnityEngine.XRModule.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.VR"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/UnityExtensions/Unity/UnityVR/Editor/UnityEditor.VR.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.Graphs"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/Managed/UnityEditor.Graphs.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.WindowsStandalone.Extensions"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/PlaybackEngines/WindowsStandaloneSupport/UnityEditor.WindowsStandalone.Extensions.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.WebGL.Extensions"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/PlaybackEngines/WebGLSupport/UnityEditor.WebGL.Extensions.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.Android.Extensions"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/PlaybackEngines/AndroidPlayer/UnityEditor.Android.Extensions.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.UWP.Extensions"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/PlaybackEngines/MetroSupport/UnityEditor.UWP.Extensions.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.Advertisements"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/PackageCache/com.unity.ads@2.0.8/Editor/UnityEditor.Advertisements.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.Analytics.Editor"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/PackageCache/com.unity.analytics@3.3.2/Unity.Analytics.Editor.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.Analytics.StandardEvents"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/PackageCache/com.unity.analytics@3.3.2/Unity.Analytics.StandardEvents.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Unity.Analytics.Tracker"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/PackageCache/com.unity.analytics@3.3.2/Unity.Analytics.Tracker.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityEditor.Purchasing"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/PackageCache/com.unity.purchasing@2.0.6/Editor/UnityEditor.Purchasing.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="nunit.framework"> | ||||
|  <HintPath>C:/Repositories/unity-builder/builder/default-build-script/Library/PackageCache/com.unity.ext.nunit@1.0.0/net35/unity-custom/nunit.framework.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="mscorlib"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/mscorlib.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/System.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Core"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/System.Core.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.Serialization"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/System.Runtime.Serialization.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Xml"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/System.Xml.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Xml.Linq"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/System.Xml.Linq.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Numerics"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/System.Numerics.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Numerics.Vectors"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/System.Numerics.Vectors.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.Http"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/System.Net.Http.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Microsoft.CSharp"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Microsoft.CSharp.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Data"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/System.Data.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Microsoft.Win32.Primitives"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/Microsoft.Win32.Primitives.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="netstandard"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/netstandard.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.AppContext"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.AppContext.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Collections.Concurrent"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Collections.Concurrent.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Collections"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Collections.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Collections.NonGeneric"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Collections.NonGeneric.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Collections.Specialized"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Collections.Specialized.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ComponentModel.Annotations"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ComponentModel.Annotations.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ComponentModel"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ComponentModel.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ComponentModel.EventBasedAsync"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ComponentModel.EventBasedAsync.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ComponentModel.Primitives"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ComponentModel.Primitives.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ComponentModel.TypeConverter"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ComponentModel.TypeConverter.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Console"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Console.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Data.Common"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Data.Common.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Diagnostics.Contracts"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Diagnostics.Contracts.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Diagnostics.Debug"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Diagnostics.Debug.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Diagnostics.FileVersionInfo"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Diagnostics.FileVersionInfo.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Diagnostics.Process"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Diagnostics.Process.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Diagnostics.StackTrace"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Diagnostics.StackTrace.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Diagnostics.TextWriterTraceListener"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Diagnostics.TextWriterTraceListener.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Diagnostics.Tools"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Diagnostics.Tools.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Diagnostics.TraceSource"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Diagnostics.TraceSource.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Drawing.Primitives"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Drawing.Primitives.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Dynamic.Runtime"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Dynamic.Runtime.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Globalization.Calendars"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Globalization.Calendars.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Globalization"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Globalization.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Globalization.Extensions"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Globalization.Extensions.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.IO.Compression.ZipFile"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.IO.Compression.ZipFile.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.IO"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.IO.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.IO.FileSystem"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.IO.FileSystem.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.IO.FileSystem.DriveInfo"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.IO.FileSystem.DriveInfo.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.IO.FileSystem.Primitives"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.IO.FileSystem.Primitives.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.IO.FileSystem.Watcher"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.IO.FileSystem.Watcher.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.IO.IsolatedStorage"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.IO.IsolatedStorage.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.IO.MemoryMappedFiles"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.IO.MemoryMappedFiles.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.IO.Pipes"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.IO.Pipes.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.IO.UnmanagedMemoryStream"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.IO.UnmanagedMemoryStream.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Linq"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Linq.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Linq.Expressions"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Linq.Expressions.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Linq.Parallel"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Linq.Parallel.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Linq.Queryable"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Linq.Queryable.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.Http.Rtc"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.Http.Rtc.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.NameResolution"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.NameResolution.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.NetworkInformation"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.NetworkInformation.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.Ping"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.Ping.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.Primitives"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.Primitives.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.Requests"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.Requests.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.Security"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.Security.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.Sockets"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.Sockets.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.WebHeaderCollection"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.WebHeaderCollection.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.WebSockets.Client"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.WebSockets.Client.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Net.WebSockets"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Net.WebSockets.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ObjectModel"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ObjectModel.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Reflection"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Reflection.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Reflection.Emit"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Reflection.Emit.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Reflection.Emit.ILGeneration"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Reflection.Emit.ILGeneration.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Reflection.Emit.Lightweight"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Reflection.Emit.Lightweight.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Reflection.Extensions"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Reflection.Extensions.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Reflection.Primitives"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Reflection.Primitives.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Resources.Reader"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Resources.Reader.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Resources.ResourceManager"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Resources.ResourceManager.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Resources.Writer"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Resources.Writer.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.CompilerServices.VisualC"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.CompilerServices.VisualC.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.Extensions"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.Extensions.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.Handles"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.Handles.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.InteropServices"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.InteropServices.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.InteropServices.RuntimeInformation"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.InteropServices.RuntimeInformation.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.InteropServices.WindowsRuntime"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.InteropServices.WindowsRuntime.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.Numerics"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.Numerics.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.Serialization.Formatters"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.Serialization.Formatters.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.Serialization.Json"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.Serialization.Json.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.Serialization.Primitives"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.Serialization.Primitives.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Runtime.Serialization.Xml"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Runtime.Serialization.Xml.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Security.Claims"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Security.Claims.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Security.Cryptography.Algorithms"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Security.Cryptography.Algorithms.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Security.Cryptography.Csp"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Security.Cryptography.Csp.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Security.Cryptography.Encoding"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Security.Cryptography.Encoding.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Security.Cryptography.Primitives"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Security.Cryptography.Primitives.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Security.Cryptography.X509Certificates"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Security.Cryptography.X509Certificates.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Security.Principal"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Security.Principal.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Security.SecureString"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Security.SecureString.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ServiceModel.Duplex"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ServiceModel.Duplex.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ServiceModel.Http"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ServiceModel.Http.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ServiceModel.NetTcp"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ServiceModel.NetTcp.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ServiceModel.Primitives"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ServiceModel.Primitives.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ServiceModel.Security"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ServiceModel.Security.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Text.Encoding"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Text.Encoding.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Text.Encoding.Extensions"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Text.Encoding.Extensions.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Text.RegularExpressions"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Text.RegularExpressions.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Threading"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Threading.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Threading.Overlapped"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Threading.Overlapped.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Threading.Tasks"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Threading.Tasks.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Threading.Tasks.Parallel"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Threading.Tasks.Parallel.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Threading.Thread"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Threading.Thread.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Threading.ThreadPool"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Threading.ThreadPool.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Threading.Timer"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Threading.Timer.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.ValueTuple"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.ValueTuple.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Xml.ReaderWriter"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Xml.ReaderWriter.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Xml.XDocument"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Xml.XDocument.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Xml.XmlDocument"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Xml.XmlDocument.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Xml.XmlSerializer"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Xml.XmlSerializer.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Xml.XPath"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Xml.XPath.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="System.Xml.XPath.XDocument"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/4.7.1-api/Facades/System.Xml.XPath.XDocument.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityScript"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/unityscript/UnityScript.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="UnityScript.Lang"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/unityscript/UnityScript.Lang.dll</HintPath> | ||||
|  </Reference> | ||||
|  <Reference Include="Boo.Lang"> | ||||
|  <HintPath>C:/Program Files/Unity/2019.2.11f1/Editor/Data/MonoBleedingEdge/lib/mono/unityscript/Boo.Lang.dll</HintPath> | ||||
|  </Reference> | ||||
|   </ItemGroup> | ||||
|   <ItemGroup> | ||||
|     <Content Include=".editorconfig" /> | ||||
|   </ItemGroup> | ||||
|   <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" /> | ||||
|   <!-- To modify your build process, add your task inside one of the targets below and uncomment it.  | ||||
|        Other similar extension points exist, see Microsoft.Common.targets. | ||||
|   <Target Name="BeforeBuild"> | ||||
|   </Target> | ||||
|   <Target Name="AfterBuild"> | ||||
|   </Target> | ||||
|   --> | ||||
| </Project> | ||||
|  | @ -6,6 +6,9 @@ using UnityBuilderAction.Reporting; | |||
| using UnityBuilderAction.Versioning; | ||||
| using UnityEditor; | ||||
| using UnityEditor.Build.Reporting; | ||||
| #if UNITY_6000_0_OR_NEWER | ||||
| using UnityEditor.Build.Profile; | ||||
| #endif | ||||
| using UnityEngine; | ||||
| 
 | ||||
| namespace UnityBuilderAction | ||||
|  | @ -17,36 +20,9 @@ namespace UnityBuilderAction | |||
|       // Gather values from args | ||||
|       var options = ArgumentsParser.GetValidatedOptions(); | ||||
| 
 | ||||
|       // Gather values from project | ||||
|       var scenes = EditorBuildSettings.scenes.Where(scene => scene.enabled).Select(s => s.path).ToArray(); | ||||
|        | ||||
|       // Get all buildOptions from options | ||||
|       BuildOptions buildOptions = BuildOptions.None; | ||||
|       foreach (string buildOptionString in Enum.GetNames(typeof(BuildOptions))) { | ||||
|         if (options.ContainsKey(buildOptionString)) { | ||||
|           BuildOptions buildOptionEnum = (BuildOptions) Enum.Parse(typeof(BuildOptions), buildOptionString); | ||||
|           buildOptions |= buildOptionEnum; | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       // Define BuildPlayer Options | ||||
|       var buildPlayerOptions = new BuildPlayerOptions { | ||||
|         scenes = scenes, | ||||
|         locationPathName = options["customBuildPath"], | ||||
|         target = (BuildTarget) Enum.Parse(typeof(BuildTarget), options["buildTarget"]), | ||||
|         options = buildOptions | ||||
|       }; | ||||
| 
 | ||||
|       // Set version for this build | ||||
|       VersionApplicator.SetVersion(options["buildVersion"]); | ||||
|        | ||||
|       // Apply Android settings | ||||
|       if (buildPlayerOptions.target == BuildTarget.Android) | ||||
|       { | ||||
|         VersionApplicator.SetAndroidVersionCode(options["androidVersionCode"]); | ||||
|         AndroidSettings.Apply(options); | ||||
|       } | ||||
|        | ||||
| 
 | ||||
|       // Execute default AddressableAsset content build, if the package is installed. | ||||
|       // Version defines would be the best solution here, but Unity 2018 doesn't support that, | ||||
|       // so we fall back to using reflection instead. | ||||
|  | @ -63,10 +39,89 @@ namespace UnityBuilderAction | |||
|         } | ||||
|         catch (Exception e) | ||||
|         { | ||||
|           Debug.LogError($"Failed to run default addressables build:\n{e}"); | ||||
|           Debug.LogError("Failed to run default addressables build:\n" + e); | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       // Get all buildOptions from options | ||||
|       BuildOptions buildOptions = BuildOptions.None; | ||||
|       foreach (string buildOptionString in Enum.GetNames(typeof(BuildOptions))) { | ||||
|         if (options.ContainsKey(buildOptionString)) { | ||||
|           BuildOptions buildOptionEnum = (BuildOptions) Enum.Parse(typeof(BuildOptions), buildOptionString); | ||||
|           buildOptions |= buildOptionEnum; | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       // Depending on whether the build is using a build profile, `buildPlayerOptions` will an instance | ||||
|       // of either `UnityEditor.BuildPlayerOptions` or `UnityEditor.BuildPlayerWithProfileOptions` | ||||
|       dynamic buildPlayerOptions; | ||||
| 
 | ||||
|       if (options.TryGetValue("activeBuildProfile", out var buildProfilePath)) { | ||||
|         if (string.IsNullOrEmpty(buildProfilePath)) { | ||||
|           throw new Exception("`-activeBuildProfile` is set but with an empty value; this shouldn't happen"); | ||||
|         } | ||||
| 
 | ||||
| #if UNITY_6000_0_OR_NEWER | ||||
|         // Load build profile from Assets folder | ||||
|         var buildProfile = AssetDatabase.LoadAssetAtPath<BuildProfile>(buildProfilePath) | ||||
|                            ?? throw new Exception("Build profile file not found at path: " + buildProfilePath); | ||||
| 
 | ||||
| #if !BUILD_PROFILE_LOADED | ||||
|         throw new Exception("Build profile's define symbol not present before script execution; shouldn't happen"); | ||||
| #endif // BUILD_PROFILE_LOADED | ||||
| 
 | ||||
|         // no need to set active profile, as already set by `-activeBuildProfile` CLI argument | ||||
|         // BuildProfile.SetActiveBuildProfile(buildProfile); | ||||
|         Debug.Log($"build profile: {buildProfile.name}"); | ||||
| 
 | ||||
|         // Define BuildPlayerWithProfileOptions | ||||
|         buildPlayerOptions = new BuildPlayerWithProfileOptions { | ||||
|             buildProfile = buildProfile, | ||||
|             locationPathName = options["customBuildPath"], | ||||
|             options = buildOptions, | ||||
|         }; | ||||
| #else // UNITY_6000_0_OR_NEWER | ||||
|         throw new Exception("Build profiles are not supported by this version of Unity (" + Application.unityVersion +")"); | ||||
| #endif // UNITY_6000_0_OR_NEWER | ||||
| 
 | ||||
|       } else { | ||||
| 
 | ||||
| #if BUILD_PROFILE_LOADED | ||||
|         throw new Exception("Build profile's define symbol present; shouldn't happen"); | ||||
| #endif // BUILD_PROFILE_LOADED | ||||
| 
 | ||||
|         // Gather values from project | ||||
|         var scenes = EditorBuildSettings.scenes.Where(scene => scene.enabled).Select(s => s.path).ToArray(); | ||||
| 
 | ||||
| #if UNITY_2021_2_OR_NEWER | ||||
|         // Determine subtarget | ||||
|         StandaloneBuildSubtarget buildSubtarget; | ||||
|         if (!options.TryGetValue("standaloneBuildSubtarget", out var subtargetValue) || !Enum.TryParse(subtargetValue, out buildSubtarget)) { | ||||
|           buildSubtarget = default; | ||||
|         } | ||||
| #endif | ||||
| 
 | ||||
|         BuildTarget buildTarget = (BuildTarget) Enum.Parse(typeof(BuildTarget), options["buildTarget"]); | ||||
| 
 | ||||
|         // Define BuildPlayerOptions | ||||
|         buildPlayerOptions = new BuildPlayerOptions { | ||||
|           scenes = scenes, | ||||
|           locationPathName = options["customBuildPath"], | ||||
|           target = buildTarget, | ||||
|           options = buildOptions, | ||||
| #if UNITY_2021_2_OR_NEWER | ||||
|           subtarget = (int) buildSubtarget | ||||
| #endif | ||||
|         }; | ||||
| 
 | ||||
|         // Apply Android settings | ||||
|         if (buildTarget == BuildTarget.Android) { | ||||
|           VersionApplicator.SetAndroidVersionCode(options["androidVersionCode"]); | ||||
|           AndroidSettings.Apply(options); | ||||
|         } | ||||
| 
 | ||||
|       } | ||||
| 
 | ||||
|       // Perform build | ||||
|       BuildReport buildReport = BuildPipeline.BuildPlayer(buildPlayerOptions); | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,24 +1,150 @@ | |||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using UnityEditor; | ||||
| using System.Reflection; | ||||
| 
 | ||||
| namespace UnityBuilderAction.Input | ||||
| { | ||||
|   public class AndroidSettings | ||||
|   public static class AndroidSettings | ||||
|   { | ||||
|     public static void Apply(Dictionary<string, string> options) | ||||
|     { | ||||
|       EditorUserBuildSettings.buildAppBundle = options["customBuildPath"].EndsWith(".aab"); | ||||
| #if UNITY_2019_1_OR_NEWER | ||||
|       if (options.TryGetValue("androidKeystoreName", out string keystoreName) && !string.IsNullOrEmpty(keystoreName)) | ||||
|       { | ||||
|         PlayerSettings.Android.useCustomKeystore = true; | ||||
|         PlayerSettings.Android.keystoreName = keystoreName; | ||||
|       if (options.TryGetValue("androidKeystorePass", out string keystorePass) && !string.IsNullOrEmpty(keystorePass)) | ||||
|       } | ||||
| #endif | ||||
|       // Can't use out variable declaration as Unity 2018 doesn't support it | ||||
|       string keystorePass; | ||||
|       if (options.TryGetValue("androidKeystorePass", out keystorePass) && !string.IsNullOrEmpty(keystorePass)) | ||||
|         PlayerSettings.Android.keystorePass = keystorePass; | ||||
|       if (options.TryGetValue("androidKeyaliasName", out string keyaliasName) && !string.IsNullOrEmpty(keyaliasName)) | ||||
|        | ||||
|       string keyaliasName; | ||||
|       if (options.TryGetValue("androidKeyaliasName", out keyaliasName) && !string.IsNullOrEmpty(keyaliasName)) | ||||
|         PlayerSettings.Android.keyaliasName = keyaliasName; | ||||
|       if (options.TryGetValue("androidKeyaliasPass", out string keyaliasPass) && !string.IsNullOrEmpty(keyaliasPass)) | ||||
| 
 | ||||
|       string keyaliasPass; | ||||
|       if (options.TryGetValue("androidKeyaliasPass", out keyaliasPass) && !string.IsNullOrEmpty(keyaliasPass)) | ||||
|         PlayerSettings.Android.keyaliasPass = keyaliasPass; | ||||
|       if (options.TryGetValue("androidTargetSdkVersion", out string androidTargetSdkVersion) && !string.IsNullOrEmpty(androidTargetSdkVersion)) | ||||
|         PlayerSettings.Android.targetSdkVersion = (AndroidSdkVersions) Enum.Parse(typeof(AndroidSdkVersions), androidTargetSdkVersion); | ||||
|        | ||||
|       string androidTargetSdkVersion; | ||||
|       if (options.TryGetValue("androidTargetSdkVersion", out androidTargetSdkVersion) && !string.IsNullOrEmpty(androidTargetSdkVersion)) | ||||
|       { | ||||
|           var targetSdkVersion = AndroidSdkVersions.AndroidApiLevelAuto; | ||||
|           try | ||||
|           { | ||||
|               targetSdkVersion = | ||||
|                   (AndroidSdkVersions) Enum.Parse(typeof(AndroidSdkVersions), androidTargetSdkVersion); | ||||
|           } | ||||
|           catch | ||||
|           { | ||||
|               UnityEngine.Debug.Log("Failed to parse androidTargetSdkVersion! Fallback to AndroidApiLevelAuto"); | ||||
|           } | ||||
|           PlayerSettings.Android.targetSdkVersion = targetSdkVersion; | ||||
|       } | ||||
| 
 | ||||
|       string androidExportType; | ||||
|       if (options.TryGetValue("androidExportType", out androidExportType) && !string.IsNullOrEmpty(androidExportType)) | ||||
|       { | ||||
|         // Only exists in 2018.3 and above | ||||
|         PropertyInfo buildAppBundle = typeof(EditorUserBuildSettings) | ||||
|               .GetProperty("buildAppBundle", BindingFlags.Public | BindingFlags.Static); | ||||
|         switch (androidExportType) | ||||
|         { | ||||
|           case "androidStudioProject": | ||||
|             EditorUserBuildSettings.exportAsGoogleAndroidProject = true; | ||||
|             if (buildAppBundle != null) | ||||
|               buildAppBundle.SetValue(null, false, null); | ||||
|             break; | ||||
|           case "androidAppBundle": | ||||
|             EditorUserBuildSettings.exportAsGoogleAndroidProject = false; | ||||
|             if (buildAppBundle != null) | ||||
|               buildAppBundle.SetValue(null, true, null); | ||||
|             break; | ||||
|           case "androidPackage": | ||||
|             EditorUserBuildSettings.exportAsGoogleAndroidProject = false; | ||||
|             if (buildAppBundle != null) | ||||
|               buildAppBundle.SetValue(null, false, null); | ||||
|             break; | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       string symbolType; | ||||
|       if (options.TryGetValue("androidSymbolType", out symbolType) && !string.IsNullOrEmpty(symbolType)) | ||||
|       { | ||||
| #if UNITY_6000_0_OR_NEWER | ||||
|         switch (symbolType) | ||||
|         { | ||||
|           case "public": | ||||
|             SetDebugSymbols("SymbolTable"); | ||||
|             break; | ||||
|           case "debugging": | ||||
|             SetDebugSymbols("Full"); | ||||
|             break; | ||||
|           case "none": | ||||
|             SetDebugSymbols("None"); | ||||
|             break; | ||||
|         } | ||||
| #elif UNITY_2021_1_OR_NEWER | ||||
|         switch (symbolType) | ||||
|         { | ||||
|           case "public": | ||||
|             EditorUserBuildSettings.androidCreateSymbols = AndroidCreateSymbols.Public; | ||||
|             break; | ||||
|           case "debugging": | ||||
|             EditorUserBuildSettings.androidCreateSymbols = AndroidCreateSymbols.Debugging; | ||||
|             break; | ||||
|           case "none": | ||||
|             EditorUserBuildSettings.androidCreateSymbols = AndroidCreateSymbols.Disabled; | ||||
|             break; | ||||
|         } | ||||
| #elif UNITY_2019_2_OR_NEWER | ||||
|         switch (symbolType) | ||||
|         { | ||||
|           case "public": | ||||
|           case "debugging": | ||||
|             EditorUserBuildSettings.androidCreateSymbolsZip = true; | ||||
|             break; | ||||
|           case "none": | ||||
|             EditorUserBuildSettings.androidCreateSymbolsZip = false; | ||||
|             break; | ||||
|         } | ||||
| #endif | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
| #if UNITY_6000_0_OR_NEWER | ||||
|     private static void SetDebugSymbols(string enumValueName) | ||||
|     { | ||||
|       // UnityEditor.Android.UserBuildSettings and Unity.Android.Types.DebugSymbolLevel are part of the Unity Android module. | ||||
|       // Reflection is used here to ensure the code works even if the module is not installed. | ||||
| 
 | ||||
|       var debugSymbolsType = Type.GetType("UnityEditor.Android.UserBuildSettings+DebugSymbols, UnityEditor.Android.Extensions"); | ||||
|       if (debugSymbolsType == null) | ||||
|       { | ||||
|         return; | ||||
|       } | ||||
| 
 | ||||
|       var levelProp = debugSymbolsType.GetProperty("level", BindingFlags.Static | BindingFlags.Public); | ||||
|       if (levelProp == null) | ||||
|       { | ||||
|         return; | ||||
|       } | ||||
| 
 | ||||
|       var enumType = Type.GetType("Unity.Android.Types.DebugSymbolLevel, Unity.Android.Types"); | ||||
|       if (enumType == null) | ||||
|       { | ||||
|         return; | ||||
|       } | ||||
| 
 | ||||
|       if (!Enum.TryParse(enumType, enumValueName, false , out var enumValue)) | ||||
|       { | ||||
|         return; | ||||
|       } | ||||
|       levelProp.SetValue(null, enumValue); | ||||
|     } | ||||
| #endif | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -12,33 +12,53 @@ namespace UnityBuilderAction.Input | |||
| 
 | ||||
|     public static Dictionary<string, string> GetValidatedOptions() | ||||
|     { | ||||
|       ParseCommandLineArguments(out var validatedOptions); | ||||
|       Dictionary<string, string> validatedOptions; | ||||
|       ParseCommandLineArguments(out validatedOptions); | ||||
| 
 | ||||
|       if (!validatedOptions.TryGetValue("projectPath", out var projectPath)) { | ||||
|       string projectPath; | ||||
|       if (!validatedOptions.TryGetValue("projectPath", out projectPath)) { | ||||
|         Console.WriteLine("Missing argument -projectPath"); | ||||
|         EditorApplication.Exit(110); | ||||
|       } | ||||
| 
 | ||||
|       if (!validatedOptions.TryGetValue("buildTarget", out var buildTarget)) { | ||||
| #if UNITY_6000_0_OR_NEWER | ||||
|       var buildProfileSupport = true; | ||||
| #else | ||||
|       var buildProfileSupport = false; | ||||
| #endif // UNITY_6000_0_OR_NEWER | ||||
| 
 | ||||
|       string buildProfile; | ||||
|       if (buildProfileSupport && validatedOptions.TryGetValue("activeBuildProfile", out buildProfile)) { | ||||
|         if (validatedOptions.ContainsKey("buildTarget")) { | ||||
|           Console.WriteLine("Extra argument -buildTarget"); | ||||
|           EditorApplication.Exit(122); | ||||
|         } | ||||
|       } else { | ||||
|       string buildTarget; | ||||
|       if (!validatedOptions.TryGetValue("buildTarget", out buildTarget)) { | ||||
|         Console.WriteLine("Missing argument -buildTarget"); | ||||
|         EditorApplication.Exit(120); | ||||
|       } | ||||
| 
 | ||||
|       if (!Enum.IsDefined(typeof(BuildTarget), buildTarget)) { | ||||
|         Console.WriteLine(buildTarget + " is not a defined " + typeof(BuildTarget).Name); | ||||
|         EditorApplication.Exit(121); | ||||
|       } | ||||
|       } | ||||
| 
 | ||||
|       if (!validatedOptions.TryGetValue("customBuildPath", out var customBuildPath)) { | ||||
|       string customBuildPath; | ||||
|       if (!validatedOptions.TryGetValue("customBuildPath", out customBuildPath)) { | ||||
|         Console.WriteLine("Missing argument -customBuildPath"); | ||||
|         EditorApplication.Exit(130); | ||||
|       } | ||||
| 
 | ||||
|       const string defaultCustomBuildName = "TestBuild"; | ||||
|       if (!validatedOptions.TryGetValue("customBuildName", out var customBuildName)) { | ||||
|         Console.WriteLine($"Missing argument -customBuildName, defaulting to {defaultCustomBuildName}."); | ||||
|       string customBuildName; | ||||
|       if (!validatedOptions.TryGetValue("customBuildName", out customBuildName)) { | ||||
|         Console.WriteLine("Missing argument -customBuildName, defaulting to" + defaultCustomBuildName); | ||||
|         validatedOptions.Add("customBuildName", defaultCustomBuildName); | ||||
|       } else if (customBuildName == "") { | ||||
|         Console.WriteLine($"Invalid argument -customBuildName, defaulting to {defaultCustomBuildName}."); | ||||
|         Console.WriteLine("Invalid argument -customBuildName, defaulting to" + defaultCustomBuildName); | ||||
|         validatedOptions.Add("customBuildName", defaultCustomBuildName); | ||||
|       } | ||||
| 
 | ||||
|  | @ -51,11 +71,11 @@ namespace UnityBuilderAction.Input | |||
|       string[] args = Environment.GetCommandLineArgs(); | ||||
| 
 | ||||
|       Console.WriteLine( | ||||
|         $"{EOL}" + | ||||
|         $"###########################{EOL}" + | ||||
|         $"#    Parsing settings     #{EOL}" + | ||||
|         $"###########################{EOL}" + | ||||
|         $"{EOL}" | ||||
|         EOL + | ||||
|         "###########################" + EOL + | ||||
|         "#    Parsing settings     #" + EOL + | ||||
|         "###########################" + EOL + | ||||
|         EOL | ||||
|       ); | ||||
| 
 | ||||
|       // Extract flags with optional values | ||||
|  | @ -72,7 +92,7 @@ namespace UnityBuilderAction.Input | |||
|         string displayValue = secret ? "*HIDDEN*" : "\"" + value + "\""; | ||||
| 
 | ||||
|         // Assign | ||||
|         Console.WriteLine($"Found flag \"{flag}\" with value {displayValue}."); | ||||
|         Console.WriteLine("Found flag \"" + flag + "\" with value " + displayValue); | ||||
|         providedArguments.Add(flag, value); | ||||
|       } | ||||
|     } | ||||
|  |  | |||
|  | @ -0,0 +1,36 @@ | |||
| using System; | ||||
| using UnityEngine; | ||||
| using UnityEditor; | ||||
| 
 | ||||
| namespace UnityBuilderAction.Reporting | ||||
| { | ||||
|     [InitializeOnLoad] | ||||
|     static class CompileListener | ||||
|     { | ||||
|         static CompileListener() | ||||
|         { | ||||
|             if (Application.isBatchMode) | ||||
|             { | ||||
|                 Application.logMessageReceived += Application_logMessageReceived; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         private static void Application_logMessageReceived(string condition, string stackTrace, LogType type) | ||||
|         { | ||||
|             string prefix = ""; | ||||
|             switch (type) | ||||
|             { | ||||
|                 case LogType.Error: | ||||
|                     prefix = "error"; | ||||
|                     break; | ||||
|                 case LogType.Warning: | ||||
|                     prefix = "warning"; | ||||
|                     break; | ||||
|                 case LogType.Exception: | ||||
|                     prefix = "error"; | ||||
|                     break; | ||||
|             } | ||||
|             Console.WriteLine(Environment.NewLine + "::" + prefix + "::" + condition + Environment.NewLine + stackTrace); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,11 @@ | |||
| fileFormatVersion: 2 | ||||
| guid: fad44373fb7b61a4bb584e2675795aca | ||||
| MonoImporter: | ||||
|   externalObjects: {} | ||||
|   serializedVersion: 2 | ||||
|   defaultReferences: [] | ||||
|   executionOrder: 0 | ||||
|   icon: {instanceID: 0} | ||||
|   userData:  | ||||
|   assetBundleName:  | ||||
|   assetBundleVariant:  | ||||
|  | @ -11,16 +11,16 @@ namespace UnityBuilderAction.Reporting | |||
|     public static void ReportSummary(BuildSummary summary) | ||||
|     { | ||||
|       Console.WriteLine( | ||||
|         $"{EOL}" + | ||||
|         $"###########################{EOL}" + | ||||
|         $"#      Build results      #{EOL}" + | ||||
|         $"###########################{EOL}" + | ||||
|         $"{EOL}" + | ||||
|         $"Duration: {summary.totalTime.ToString()}{EOL}" + | ||||
|         $"Warnings: {summary.totalWarnings.ToString()}{EOL}" + | ||||
|         $"Errors: {summary.totalErrors.ToString()}{EOL}" + | ||||
|         $"Size: {summary.totalSize.ToString()} bytes{EOL}" + | ||||
|         $"{EOL}" | ||||
|         EOL + | ||||
|         "###########################" + EOL + | ||||
|         "#      Build results      #" + EOL + | ||||
|         "###########################" + EOL + | ||||
|         EOL + | ||||
|         "Duration: " + summary.totalTime.ToString() + EOL + | ||||
|         "Warnings: " + summary.totalWarnings.ToString() + EOL + | ||||
|         "Errors: " + summary.totalErrors.ToString() + EOL + | ||||
|         "Size: " + summary.totalSize.ToString() + " bytes" + EOL + | ||||
|         EOL | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|  |  | |||
|  | @ -21,11 +21,11 @@ namespace UnityBuilderAction.Versioning | |||
|         version = GetSemanticCommitVersion(); | ||||
|         Console.WriteLine("Repository has a valid version tag."); | ||||
|       } else { | ||||
|         version = $"0.0.{GetTotalNumberOfCommits()}"; | ||||
|         version = "0.0." + GetTotalNumberOfCommits(); | ||||
|         Console.WriteLine("Repository does not have tags to base the version on."); | ||||
|       } | ||||
| 
 | ||||
|       Console.WriteLine($"Version is {version}"); | ||||
|       Console.WriteLine("Version is " + version); | ||||
| 
 | ||||
|       return version; | ||||
|     } | ||||
|  | @ -106,7 +106,8 @@ namespace UnityBuilderAction.Versioning | |||
|       using (var process = new System.Diagnostics.Process()) { | ||||
|         string workingDirectory = UnityEngine.Application.dataPath; | ||||
| 
 | ||||
|         int exitCode = process.Run(application, arguments, workingDirectory, out string output, out string errors); | ||||
|         string output, errors; | ||||
|         int exitCode = process.Run(application, arguments, workingDirectory, out output, out errors); | ||||
|         if (exitCode != 0) { throw new GitException(exitCode, errors); } | ||||
| 
 | ||||
|         return output; | ||||
|  |  | |||
|  | @ -1,10 +0,0 @@ | |||
| { | ||||
|     "m_SettingKeys": [ | ||||
|         "VR Device Disabled", | ||||
|         "VR Device User Alert" | ||||
|     ], | ||||
|     "m_SettingValues": [ | ||||
|         "False", | ||||
|         "False" | ||||
|     ] | ||||
| } | ||||
|  | @ -1,20 +0,0 @@ | |||
|  | ||||
| Microsoft Visual Studio Solution File, Format Version 11.00 | ||||
| # Visual Studio 2010 | ||||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Assembly-CSharp-Editor", "Assembly-CSharp-Editor.csproj", "{B7F8614B-1EC2-9D3A-DA1C-4D279A867D74}" | ||||
| EndProject | ||||
| Global | ||||
| 	GlobalSection(SolutionConfigurationPlatforms) = preSolution | ||||
| 		Debug|Any CPU = Debug|Any CPU | ||||
| 		Release|Any CPU = Release|Any CPU | ||||
| 	EndGlobalSection | ||||
| 	GlobalSection(ProjectConfigurationPlatforms) = postSolution | ||||
| 		{B7F8614B-1EC2-9D3A-DA1C-4D279A867D74}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | ||||
| 		{B7F8614B-1EC2-9D3A-DA1C-4D279A867D74}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||
| 		{B7F8614B-1EC2-9D3A-DA1C-4D279A867D74}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||
| 		{B7F8614B-1EC2-9D3A-DA1C-4D279A867D74}.Release|Any CPU.Build.0 = Release|Any CPU | ||||
| 	EndGlobalSection | ||||
| 	GlobalSection(SolutionProperties) = preSolution | ||||
| 		HideSolutionNode = FALSE | ||||
| 	EndGlobalSection | ||||
| EndGlobal | ||||
|  | @ -1,3 +0,0 @@ | |||
| <wpf:ResourceDictionary xml:space="preserve" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:s="clr-namespace:System;assembly=mscorlib" xmlns:ss="urn:shemas-jetbrains-com:settings-storage-xaml" xmlns:wpf="http://schemas.microsoft.com/winfx/2006/xaml/presentation"> | ||||
| 	<s:Boolean x:Key="/Default/UserDictionary/Words/=Untracked/@EntryIndexedValue">True</s:Boolean> | ||||
| 	<s:Boolean x:Key="/Default/UserDictionary/Words/=Versioning/@EntryIndexedValue">True</s:Boolean></wpf:ResourceDictionary> | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -1,28 +1,40 @@ | |||
| #!/usr/bin/env bash | ||||
| 
 | ||||
| # | ||||
| # Create directories for license activation | ||||
| # Perform Activation | ||||
| # | ||||
| 
 | ||||
| sudo mkdir /Library/Application\ Support/Unity | ||||
| sudo chmod -R 777 /Library/Application\ Support/Unity | ||||
| if [ "$SKIP_ACTIVATION" != "true" ]; then | ||||
|   UNITY_LICENSE_PATH="/Library/Application Support/Unity" | ||||
| 
 | ||||
| ACTIVATE_LICENSE_PATH="$ACTION_FOLDER/BlankProject" | ||||
| mkdir -p "$ACTIVATE_LICENSE_PATH" | ||||
|   if [ ! -d "$UNITY_LICENSE_PATH" ]; then | ||||
|     echo "Creating Unity License Directory" | ||||
|     sudo mkdir -p "$UNITY_LICENSE_PATH" | ||||
|     sudo chmod -R 777 "$UNITY_LICENSE_PATH" | ||||
|   fi; | ||||
| 
 | ||||
|   ACTIVATE_LICENSE_PATH="$ACTION_FOLDER/BlankProject" | ||||
|   mkdir -p "$ACTIVATE_LICENSE_PATH" | ||||
| 
 | ||||
|   source $ACTION_FOLDER/platforms/mac/steps/activate.sh | ||||
| else | ||||
|   echo "Skipping activation" | ||||
| fi | ||||
| 
 | ||||
| # | ||||
| # Run steps | ||||
| # Run Build | ||||
| # | ||||
| source $ACTION_FOLDER/platforms/mac/steps/activate.sh | ||||
| 
 | ||||
| source $ACTION_FOLDER/platforms/mac/steps/build.sh | ||||
| source $ACTION_FOLDER/platforms/mac/steps/return_license.sh | ||||
| 
 | ||||
| # | ||||
| # Remove license activation directory | ||||
| # License Cleanup | ||||
| # | ||||
| 
 | ||||
| sudo rm -r /Library/Application\ Support/Unity | ||||
| rm -r "$ACTIVATE_LICENSE_PATH" | ||||
| if [ "$SKIP_ACTIVATION" != "true" ]; then | ||||
|   source $ACTION_FOLDER/platforms/mac/steps/return_license.sh | ||||
|   rm -r "$ACTIVATE_LICENSE_PATH" | ||||
| fi | ||||
| 
 | ||||
| # | ||||
| # Instructions for debugging | ||||
|  | @ -37,7 +49,7 @@ echo "" | |||
| echo "Please note that the exit code is not very descriptive." | ||||
| echo "Most likely it will not help you solve the issue." | ||||
| echo "" | ||||
| echo "To find the reason for failure: please search for errors in the log above." | ||||
| echo "To find the reason for failure: please search for errors in the log above and check for annotations in the summary view." | ||||
| echo "" | ||||
| fi; | ||||
| 
 | ||||
|  |  | |||
|  | @ -4,21 +4,69 @@ | |||
| echo "Changing to \"$ACTIVATE_LICENSE_PATH\" directory." | ||||
| pushd "$ACTIVATE_LICENSE_PATH" | ||||
| 
 | ||||
| echo "Requesting activation" | ||||
| if [[ -n "$UNITY_SERIAL" && -n "$UNITY_EMAIL" && -n "$UNITY_PASSWORD" ]]; then | ||||
|   # | ||||
|   # SERIAL LICENSE MODE | ||||
|   # | ||||
|   # This will activate unity, using the serial activation process. | ||||
|   # | ||||
| 
 | ||||
| # Activate license | ||||
| /Applications/Unity/Hub/Editor/$UNITY_VERSION/Unity.app/Contents/MacOS/Unity \ | ||||
|   -logFile /dev/stdout \ | ||||
|   -batchmode \ | ||||
|   -nographics \ | ||||
|   -quit \ | ||||
|   -serial "$UNITY_SERIAL" \ | ||||
|   -username "$UNITY_EMAIL" \ | ||||
|   -password "$UNITY_PASSWORD" \ | ||||
|   -projectPath "$ACTIVATE_LICENSE_PATH" | ||||
|   echo "Requesting activation" | ||||
| 
 | ||||
| # Store the exit code from the verify command | ||||
| UNITY_EXIT_CODE=$? | ||||
|   # Activate license | ||||
|   /Applications/Unity/Hub/Editor/$UNITY_VERSION/Unity.app/Contents/MacOS/Unity \ | ||||
|     -logFile - \ | ||||
|     -batchmode \ | ||||
|     -nographics \ | ||||
|     -quit \ | ||||
|     -serial "$UNITY_SERIAL" \ | ||||
|     -username "$UNITY_EMAIL" \ | ||||
|     -password "$UNITY_PASSWORD" \ | ||||
|     -projectPath "$ACTIVATE_LICENSE_PATH" | ||||
| 
 | ||||
|   # Store the exit code from the verify command | ||||
|   UNITY_EXIT_CODE=$? | ||||
| 
 | ||||
| elif [[ -n "$UNITY_LICENSING_SERVER" ]]; then | ||||
|   # | ||||
|   # Custom Unity License Server | ||||
|   # | ||||
|   echo "Adding licensing server config" | ||||
|   mkdir -p "$UNITY_LICENSE_PATH/config/" | ||||
|   cp "$ACTION_FOLDER/unity-config/services-config.json" "$UNITY_LICENSE_PATH/config/services-config.json" | ||||
| 
 | ||||
|   /Applications/Unity/Hub/Editor/$UNITY_VERSION/Unity.app/Contents/Frameworks/UnityLicensingClient.app/Contents/MacOS/Unity.Licensing.Client \ | ||||
|     --acquire-floating > license.txt | ||||
| 
 | ||||
|   # Store the exit code from the verify command | ||||
|   UNITY_EXIT_CODE=$? | ||||
| 
 | ||||
|   if [ $UNITY_EXIT_CODE -eq 0 ]; then | ||||
|     PARSEDFILE=$(grep -oE '\"[^"]*\"' < license.txt | tr -d '"') | ||||
|     export FLOATING_LICENSE | ||||
|     FLOATING_LICENSE=$(sed -n 2p <<< "$PARSEDFILE") | ||||
|     FLOATING_LICENSE_TIMEOUT=$(sed -n 4p <<< "$PARSEDFILE") | ||||
| 
 | ||||
|     echo "Acquired floating license: \"$FLOATING_LICENSE\" with timeout $FLOATING_LICENSE_TIMEOUT" | ||||
|   fi | ||||
| else | ||||
|   # | ||||
|   # NO LICENSE ACTIVATION STRATEGY MATCHED | ||||
|   # | ||||
|   # This will exit since no activation strategies could be matched. | ||||
|   # | ||||
|   echo "License activation strategy could not be determined." | ||||
|   echo "" | ||||
|   echo "Visit https://game.ci/docs/github/activation for more" | ||||
|   echo "details on how to set up one of the possible activation strategies." | ||||
| 
 | ||||
|   echo "::error ::No valid license activation strategy could be determined. Make sure to provide UNITY_EMAIL, UNITY_PASSWORD, and either a UNITY_SERIAL \ | ||||
| or UNITY_LICENSE. Otherwise please use UNITY_LICENSING_SERVER. See more info at https://game.ci/docs/github/activation" | ||||
| 
 | ||||
|   # Immediately exit as no UNITY_EXIT_CODE can be derived. | ||||
|   exit 1; | ||||
| 
 | ||||
| fi | ||||
| 
 | ||||
| # | ||||
| # Display information about the result | ||||
|  | @ -30,6 +78,7 @@ else | |||
|   # Activation failed so exit with the code from the license verification step | ||||
|   echo "Unclassified error occured while trying to activate license." | ||||
|   echo "Exit code was: $UNITY_EXIT_CODE" | ||||
|   echo "::error ::There was an error while trying to activate the Unity license." | ||||
|   exit $UNITY_EXIT_CODE | ||||
| fi | ||||
| 
 | ||||
|  |  | |||
|  | @ -19,6 +19,23 @@ echo "Using build name \"$BUILD_NAME\"." | |||
| 
 | ||||
| echo "Using build target \"$BUILD_TARGET\"." | ||||
| 
 | ||||
| # | ||||
| # Display the build profile | ||||
| # | ||||
| 
 | ||||
| if [ -z "$BUILD_PROFILE" ]; then | ||||
|   # User has not provided a build profile | ||||
|   # | ||||
|   echo "Doing a default \"$BUILD_TARGET\" platform build." | ||||
|   # | ||||
| else | ||||
|   # User has provided a path to a build profile `.asset` file | ||||
|   # | ||||
|   echo "Using build profile \"$BUILD_PROFILE\" relative to \"$UNITY_PROJECT_PATH\"." | ||||
|   # | ||||
| fi | ||||
| 
 | ||||
| 
 | ||||
| # | ||||
| # Display build path and file | ||||
| # | ||||
|  | @ -76,8 +93,10 @@ fi | |||
| 
 | ||||
| if [[ "$BUILD_TARGET" == "Android" && -n "$ANDROID_SDK_MANAGER_PARAMETERS" ]]; then | ||||
|   echo "Updating Android SDK with parameters: $ANDROID_SDK_MANAGER_PARAMETERS" | ||||
|   export JAVA_HOME="$(awk -F'=' '/JAVA_HOME=/{print $2}' /usr/bin/unity-editor.d/*)" | ||||
|   "$(awk -F'=' '/ANDROID_HOME=/{print $2}' /usr/bin/unity-editor.d/*)/tools/bin/sdkmanager" "$ANDROID_SDK_MANAGER_PARAMETERS" | ||||
|   ANDROID_INSTALL_LOCATION="/Applications/Unity/Hub/Editor/$UNITY_VERSION/PlaybackEngines/AndroidPlayer" | ||||
|   export JAVA_HOME="$ANDROID_INSTALL_LOCATION/OpenJDK" | ||||
|   export ANDROID_HOME="$ANDROID_INSTALL_LOCATION/SDK" | ||||
|   yes | "$ANDROID_HOME/tools/bin/sdkmanager" "$ANDROID_SDK_MANAGER_PARAMETERS" | ||||
|   echo "Updated Android SDK." | ||||
| else | ||||
|   echo "Not updating Android SDK." | ||||
|  | @ -126,15 +145,17 @@ echo "" | |||
| # Reference: https://docs.unity3d.com/2019.3/Documentation/Manual/CommandLineArguments.html | ||||
| 
 | ||||
| /Applications/Unity/Hub/Editor/$UNITY_VERSION/Unity.app/Contents/MacOS/Unity \ | ||||
|   -logfile /dev/stdout \ | ||||
|   -quit \ | ||||
|   -logFile - \ | ||||
|   $( [ "${MANUAL_EXIT}" == "true" ] || echo "-quit" ) \ | ||||
|   -batchmode \ | ||||
|   -nographics \ | ||||
|   $( [ "${ENABLE_GPU}" == "true" ] || echo "-nographics" ) \ | ||||
|   -customBuildName "$BUILD_NAME" \ | ||||
|   -projectPath "$UNITY_PROJECT_PATH" \ | ||||
|   -buildTarget "$BUILD_TARGET" \ | ||||
|   $( [ -z "$BUILD_PROFILE" ] && echo "-buildTarget $BUILD_TARGET") \ | ||||
|   -customBuildTarget "$BUILD_TARGET" \ | ||||
|   -customBuildPath "$CUSTOM_BUILD_PATH" \ | ||||
|   -customBuildProfile "$BUILD_PROFILE" \ | ||||
|   ${BUILD_PROFILE:+-activeBuildProfile} ${BUILD_PROFILE:+"$BUILD_PROFILE"} \ | ||||
|   -executeMethod "$BUILD_METHOD" \ | ||||
|   -buildVersion "$VERSION" \ | ||||
|   -androidVersionCode "$ANDROID_VERSION_CODE" \ | ||||
|  | @ -143,11 +164,16 @@ echo "" | |||
|   -androidKeyaliasName "$ANDROID_KEYALIAS_NAME" \ | ||||
|   -androidKeyaliasPass "$ANDROID_KEYALIAS_PASS" \ | ||||
|   -androidTargetSdkVersion "$ANDROID_TARGET_SDK_VERSION" \ | ||||
|   -androidExportType "$ANDROID_EXPORT_TYPE" \ | ||||
|   -androidSymbolType "$ANDROID_SYMBOL_TYPE" \ | ||||
|   $CUSTOM_PARAMETERS | ||||
| 
 | ||||
| # Catch exit code | ||||
| BUILD_EXIT_CODE=$? | ||||
| 
 | ||||
| # Display logs | ||||
| cat "$UNITY_PROJECT_PATH/out.log" | ||||
| 
 | ||||
| # Display results | ||||
| if [ $BUILD_EXIT_CODE -eq 0 ]; then | ||||
|   echo "Build succeeded"; | ||||
|  |  | |||
|  | @ -4,13 +4,29 @@ | |||
| echo "Changing to \"$ACTIVATE_LICENSE_PATH\" directory." | ||||
| pushd "$ACTIVATE_LICENSE_PATH" | ||||
| 
 | ||||
| /Applications/Unity/Hub/Editor/$UNITY_VERSION/Unity.app/Contents/MacOS/Unity \ | ||||
|   -logFile /dev/stdout \ | ||||
|   -batchmode \ | ||||
|   -nographics \ | ||||
|   -quit \ | ||||
|   -returnlicense \ | ||||
|   -projectPath "$ACTIVATE_LICENSE_PATH" | ||||
| if [[ -n "$UNITY_LICENSING_SERVER" ]]; then | ||||
|   # | ||||
|   # Return any floating license used. | ||||
|   # | ||||
|   echo "Returning floating license: \"$FLOATING_LICENSE\"" | ||||
|   /Applications/Unity/Hub/Editor/$UNITY_VERSION/Unity.app/Contents/Frameworks/UnityLicensingClient.app/Contents/MacOS/Unity.Licensing.Client \ | ||||
|     --return-floating "$FLOATING_LICENSE" | ||||
| elif [[ -n "$UNITY_SERIAL" ]]; then | ||||
|   # | ||||
|   # SERIAL LICENSE MODE | ||||
|   # | ||||
|   # This will return the license that is currently in use. | ||||
|   # | ||||
|   /Applications/Unity/Hub/Editor/$UNITY_VERSION/Unity.app/Contents/MacOS/Unity \ | ||||
|     -logFile - \ | ||||
|     -batchmode \ | ||||
|     -nographics \ | ||||
|     -quit \ | ||||
|     -username "$UNITY_EMAIL" \ | ||||
|     -password "$UNITY_PASSWORD" \ | ||||
|     -returnlicense \ | ||||
|     -projectPath "$ACTIVATE_LICENSE_PATH" | ||||
| fi | ||||
| 
 | ||||
| # Return to previous working directory | ||||
| popd | ||||
|  |  | |||
|  | @ -1,45 +1,83 @@ | |||
| #!/usr/bin/env bash | ||||
| 
 | ||||
| # | ||||
| # Create directory for license activation | ||||
| # | ||||
| 
 | ||||
| ACTIVATE_LICENSE_PATH="$GITHUB_WORKSPACE/_activate-license~" | ||||
| mkdir -p "$ACTIVATE_LICENSE_PATH" | ||||
| # Ensure machine ID is randomized for personal license activation | ||||
| if [[ "$UNITY_SERIAL" = F* ]]; then | ||||
|   echo "Randomizing machine ID for personal license activation" | ||||
|   dbus-uuidgen > /etc/machine-id && mkdir -p /var/lib/dbus/ && ln -sf /etc/machine-id /var/lib/dbus/machine-id | ||||
| fi | ||||
| 
 | ||||
| # | ||||
| # Run steps | ||||
| # | ||||
| source /steps/set_gitcredential.sh | ||||
| source /steps/activate.sh | ||||
| source /steps/build.sh | ||||
| source /steps/return_license.sh | ||||
| 
 | ||||
| # | ||||
| # Remove license activation directory | ||||
| # Prepare Android SDK, if needed | ||||
| # We do this here to ensure it has root permissions | ||||
| # | ||||
| 
 | ||||
| rm -r "$ACTIVATE_LICENSE_PATH" | ||||
| fullProjectPath="$GITHUB_WORKSPACE/$PROJECT_PATH" | ||||
| 
 | ||||
| # | ||||
| # Instructions for debugging | ||||
| # | ||||
| if [[ "$BUILD_TARGET" == "Android" ]]; then | ||||
|   export JAVA_HOME="$(awk -F'=' '/JAVA_HOME=/{print $2}' /usr/bin/unity-editor.d/*)" | ||||
|   ANDROID_HOME_DIRECTORY="$(awk -F'=' '/ANDROID_HOME=/{print $2}' /usr/bin/unity-editor.d/*)" | ||||
|   SDKMANAGER=$(find $ANDROID_HOME_DIRECTORY/cmdline-tools -name sdkmanager) | ||||
|   if [ -z "${SDKMANAGER}" ] | ||||
|   then | ||||
|     SDKMANAGER=$(find $ANDROID_HOME_DIRECTORY/tools/bin -name sdkmanager) | ||||
|     if [ -z "${SDKMANAGER}" ] | ||||
|     then | ||||
|       echo "No sdkmanager found" | ||||
|       exit 1 | ||||
|     fi | ||||
|   fi | ||||
| 
 | ||||
| if [[ $BUILD_EXIT_CODE -gt 0 ]]; then | ||||
| echo "" | ||||
| echo "###########################" | ||||
| echo "#         Failure         #" | ||||
| echo "###########################" | ||||
| echo "" | ||||
| echo "Please note that the exit code is not very descriptive." | ||||
| echo "Most likely it will not help you solve the issue." | ||||
| echo "" | ||||
| echo "To find the reason for failure: please search for errors in the log above." | ||||
| echo "" | ||||
| fi; | ||||
|   if [[ -n "$ANDROID_SDK_MANAGER_PARAMETERS" ]]; then | ||||
|     echo "Updating Android SDK with parameters: $ANDROID_SDK_MANAGER_PARAMETERS" | ||||
|     $SDKMANAGER "$ANDROID_SDK_MANAGER_PARAMETERS" | ||||
|   else | ||||
|     echo "Updating Android SDK with auto detected target API version" | ||||
|     # Read the line containing AndroidTargetSdkVersion from the file | ||||
|     targetAPILine=$(grep 'AndroidTargetSdkVersion' "$fullProjectPath/ProjectSettings/ProjectSettings.asset") | ||||
| 
 | ||||
| # | ||||
| # Exit with code from the build step. | ||||
| # | ||||
|     # Extract the number after the semicolon | ||||
|     targetAPI=$(echo "$targetAPILine" | cut -d':' -f2 | tr -d '[:space:]') | ||||
| 
 | ||||
| exit $BUILD_EXIT_CODE | ||||
|     $SDKMANAGER "platforms;android-$targetAPI" | ||||
|   fi | ||||
| 
 | ||||
|   echo "Updated Android SDK." | ||||
| else | ||||
|   echo "Not updating Android SDK." | ||||
| fi | ||||
| 
 | ||||
| if [[ "$RUN_AS_HOST_USER" == "true" ]]; then | ||||
|   echo "Running as host user" | ||||
| 
 | ||||
|   # Stop on error if we can't set up the user | ||||
|   set -e | ||||
| 
 | ||||
|   # Get host user/group info so we create files with the correct ownership | ||||
|   USERNAME=$(stat -c '%U' "$fullProjectPath") | ||||
|   USERID=$(stat -c '%u' "$fullProjectPath") | ||||
|   GROUPNAME=$(stat -c '%G' "$fullProjectPath") | ||||
|   GROUPID=$(stat -c '%g' "$fullProjectPath") | ||||
| 
 | ||||
|   groupadd -g $GROUPID $GROUPNAME | ||||
|   useradd -u $USERID -g $GROUPID $USERNAME | ||||
|   usermod -aG $GROUPNAME $USERNAME | ||||
|   mkdir -p "/home/$USERNAME" | ||||
|   chown $USERNAME:$GROUPNAME "/home/$USERNAME" | ||||
| 
 | ||||
|   # Normally need root permissions to access when using su | ||||
|   chmod 777 /dev/stdout | ||||
|   chmod 777 /dev/stderr | ||||
| 
 | ||||
|   # Don't stop on error when running our scripts as error handling is baked in | ||||
|   set +e | ||||
| 
 | ||||
|   # Switch to the host user so we can create files with the correct ownership | ||||
|   su $USERNAME -c "$SHELL -c 'source /steps/runsteps.sh'" | ||||
| else | ||||
|   echo "Running as root" | ||||
| 
 | ||||
|   # Run as root | ||||
|   source /steps/runsteps.sh | ||||
| fi | ||||
| 
 | ||||
| exit $? | ||||
|  |  | |||
|  | @ -1,79 +1,85 @@ | |||
| #!/usr/bin/env bash | ||||
| 
 | ||||
| # Run in ACTIVATE_LICENSE_PATH directory | ||||
| echo "Changing to \"$ACTIVATE_LICENSE_PATH\" directory." | ||||
| pushd "$ACTIVATE_LICENSE_PATH" | ||||
| # if blankproject folder doesn't exist create it | ||||
| if [ ! -d "/BlankProject" ]; then | ||||
|   mkdir /BlankProject | ||||
| fi | ||||
| # if blankproject folder doesn't exist create it | ||||
| if [ ! -d "/BlankProject/Assets" ]; then | ||||
|   mkdir /BlankProject/Assets | ||||
| fi | ||||
| 
 | ||||
| if [[ -n "$UNITY_LICENSE" ]] || [[ -n "$UNITY_LICENSE_FILE" ]]; then | ||||
| if [[ -n "$UNITY_SERIAL" && -n "$UNITY_EMAIL" && -n "$UNITY_PASSWORD" ]]; then | ||||
|   # | ||||
|   # PERSONAL LICENSE MODE | ||||
|   # SERIAL LICENSE MODE | ||||
|   # | ||||
|   # This will activate Unity, using a license file | ||||
|   # This will activate unity, using the serial activation process. | ||||
|   # | ||||
|   # Note that this is the ONLY WAY for PERSONAL LICENSES in 2020. | ||||
|   #   * See for more details: https://gitlab.com/gableroux/unity3d-gitlab-ci-example/issues/5#note_72815478 | ||||
|   # | ||||
|   # The license file can be acquired using `webbertakken/request-manual-activation-file` action. | ||||
|   echo "Requesting activation (personal license)" | ||||
|   echo "Requesting activation" | ||||
| 
 | ||||
|   # Set the license file path | ||||
|   FILE_PATH=UnityLicenseFile.ulf | ||||
|   # Loop the unity-editor call until the license is activated with exponential backoff and a maximum of 5 retries | ||||
|   retry_count=0 | ||||
| 
 | ||||
|   if [[ -n "$UNITY_LICENSE" ]]; then | ||||
|     # Copy license file from Github variables | ||||
|     echo "$UNITY_LICENSE" | tr -d '\r' > $FILE_PATH | ||||
|   elif [[ -n "$UNITY_LICENSE_FILE" ]]; then | ||||
|     # Copy license file from file system | ||||
|     cat "$UNITY_LICENSE_FILE" | tr -d '\r' > $FILE_PATH | ||||
|   fi | ||||
|   # Initialize delay to 15 seconds | ||||
|   delay=15 | ||||
| 
 | ||||
|   # Activate license | ||||
|   ACTIVATION_OUTPUT=$(unity-editor \ | ||||
|   # Loop until UNITY_EXIT_CODE is 0 or retry count reaches 5 | ||||
|   while [[ $retry_count -lt 5 ]] | ||||
|   do | ||||
|     # Activate license | ||||
|     unity-editor \ | ||||
|       -logFile /dev/stdout \ | ||||
|       -quit \ | ||||
|       -manualLicenseFile $FILE_PATH) | ||||
|       -serial "$UNITY_SERIAL" \ | ||||
|       -username "$UNITY_EMAIL" \ | ||||
|       -password "$UNITY_PASSWORD" \ | ||||
|       -projectPath "/BlankProject" | ||||
| 
 | ||||
|     # Store the exit code from the verify command | ||||
|     UNITY_EXIT_CODE=$? | ||||
| 
 | ||||
|     # Check if UNITY_EXIT_CODE is 0 | ||||
|     if [[ $UNITY_EXIT_CODE -eq 0 ]] | ||||
|     then | ||||
|       echo "Activation successful" | ||||
|       break | ||||
|     else | ||||
|       # Increment retry count | ||||
|       ((retry_count++)) | ||||
| 
 | ||||
|       echo "::warning ::Activation failed, attempting retry #$retry_count" | ||||
|       echo "Activation failed, retrying in $delay seconds..." | ||||
|       sleep $delay | ||||
| 
 | ||||
|       # Double the delay for the next iteration | ||||
|       delay=$((delay * 2)) | ||||
|     fi | ||||
|   done | ||||
| 
 | ||||
|   if [[ $retry_count -eq 5 ]] | ||||
|   then | ||||
|     echo "Activation failed after 5 retries" | ||||
|   fi | ||||
| 
 | ||||
| elif [[ -n "$UNITY_LICENSING_SERVER" ]]; then | ||||
|   # | ||||
|   # Custom Unity License Server | ||||
|   # | ||||
|   echo "Adding licensing server config" | ||||
| 
 | ||||
|   /opt/unity/Editor/Data/Resources/Licensing/Client/Unity.Licensing.Client --acquire-floating > license.txt #is this accessible in a env variable? | ||||
| 
 | ||||
|   # Store the exit code from the verify command | ||||
|   UNITY_EXIT_CODE=$? | ||||
| 
 | ||||
|   # The exit code for personal activation is always 1; | ||||
|   # Determine whether activation was successful. | ||||
|   # | ||||
|   # Successful output should include the following: | ||||
|   # | ||||
|   #   "LICENSE SYSTEM [2020120 18:51:20] Next license update check is after 2019-11-25T18:23:38" | ||||
|   # | ||||
|   ACTIVATION_SUCCESSFUL=$(echo $ACTIVATION_OUTPUT | grep 'Next license update check is after' | wc -l) | ||||
| 
 | ||||
|   # Set exit code to 0 if activation was successful | ||||
|   if [[ $ACTIVATION_SUCCESSFUL -eq 1 ]]; then | ||||
|     UNITY_EXIT_CODE=0 | ||||
|   fi; | ||||
| 
 | ||||
|   # Remove license file | ||||
|   rm -f $FILE_PATH | ||||
| 
 | ||||
| elif [[ -n "$UNITY_SERIAL" && -n "$UNITY_EMAIL" && -n "$UNITY_PASSWORD" ]]; then | ||||
|   # | ||||
|   # PROFESSIONAL (SERIAL) LICENSE MODE | ||||
|   # | ||||
|   # This will activate unity, using the activating process. | ||||
|   # | ||||
|   # Note: This is the preferred way for PROFESSIONAL LICENSES. | ||||
|   # | ||||
|   echo "Requesting activation (professional license)" | ||||
| 
 | ||||
|   # Activate license | ||||
|   unity-editor \ | ||||
|     -logFile /dev/stdout \ | ||||
|     -quit \ | ||||
|     -serial "$UNITY_SERIAL" \ | ||||
|     -username "$UNITY_EMAIL" \ | ||||
|     -password "$UNITY_PASSWORD" | ||||
| 
 | ||||
|   # Store the exit code from the verify command | ||||
|   UNITY_EXIT_CODE=$? | ||||
|   if [ $UNITY_EXIT_CODE -eq 0 ]; then | ||||
|     PARSEDFILE=$(grep -oP '\".*?\"' < license.txt | tr -d '"') | ||||
|     export FLOATING_LICENSE | ||||
|     FLOATING_LICENSE=$(sed -n 2p <<< "$PARSEDFILE") | ||||
|     FLOATING_LICENSE_TIMEOUT=$(sed -n 4p <<< "$PARSEDFILE") | ||||
| 
 | ||||
|     echo "Acquired floating license: \"$FLOATING_LICENSE\" with timeout $FLOATING_LICENSE_TIMEOUT" | ||||
|   fi | ||||
| else | ||||
|   # | ||||
|   # NO LICENSE ACTIVATION STRATEGY MATCHED | ||||
|  | @ -82,10 +88,13 @@ else | |||
|   # | ||||
|   echo "License activation strategy could not be determined." | ||||
|   echo "" | ||||
|   echo "Visit https://game.ci/docs/github/getting-started for more" | ||||
|   echo "Visit https://game.ci/docs/github/activation for more" | ||||
|   echo "details on how to set up one of the possible activation strategies." | ||||
| 
 | ||||
|   # Immediately exit as no UNITY_EXIT_CODE can be derrived. | ||||
|   echo "::error ::No valid license activation strategy could be determined. Make sure to provide UNITY_EMAIL, UNITY_PASSWORD, and either a UNITY_SERIAL \ | ||||
| or UNITY_LICENSE. Otherwise please use UNITY_LICENSING_SERVER. See more info at https://game.ci/docs/github/activation" | ||||
| 
 | ||||
|   # Immediately exit as no UNITY_EXIT_CODE can be derived. | ||||
|   exit 1; | ||||
| 
 | ||||
| fi | ||||
|  | @ -100,8 +109,6 @@ else | |||
|   # Activation failed so exit with the code from the license verification step | ||||
|   echo "Unclassified error occured while trying to activate license." | ||||
|   echo "Exit code was: $UNITY_EXIT_CODE" | ||||
|   echo "::error ::There was an error while trying to activate the Unity license." | ||||
|   exit $UNITY_EXIT_CODE | ||||
| fi | ||||
| 
 | ||||
| # Return to previous working directory | ||||
| popd | ||||
|  |  | |||
|  | @ -19,6 +19,22 @@ echo "Using build name \"$BUILD_NAME\"." | |||
| 
 | ||||
| echo "Using build target \"$BUILD_TARGET\"." | ||||
| 
 | ||||
| # | ||||
| # Display the build profile | ||||
| # | ||||
| 
 | ||||
| if [ -z "$BUILD_PROFILE" ]; then | ||||
|   # User has not provided a build profile | ||||
|   # | ||||
|   echo "Doing a default \"$BUILD_TARGET\" platform build." | ||||
|   # | ||||
| else | ||||
|   # User has provided a path to a build profile `.asset` file | ||||
|   # | ||||
|   echo "Using build profile \"$BUILD_PROFILE\" relative to \"$UNITY_PROJECT_PATH\"." | ||||
|   # | ||||
| fi | ||||
| 
 | ||||
| # | ||||
| # Display build path and file | ||||
| # | ||||
|  | @ -62,27 +78,6 @@ else | |||
|   # | ||||
| fi | ||||
| 
 | ||||
| # | ||||
| # Prepare Android keystore and SDK, if needed | ||||
| # | ||||
| 
 | ||||
| if [[ "$BUILD_TARGET" == "Android" && -n "$ANDROID_KEYSTORE_NAME" && -n "$ANDROID_KEYSTORE_BASE64" ]]; then | ||||
|   echo "Creating Android keystore." | ||||
|   echo "$ANDROID_KEYSTORE_BASE64" | base64 --decode > "$UNITY_PROJECT_PATH/$ANDROID_KEYSTORE_NAME" | ||||
|   echo "Created Android keystore." | ||||
| else | ||||
|   echo "Not creating Android keystore." | ||||
| fi | ||||
| 
 | ||||
| if [[ "$BUILD_TARGET" == "Android" && -n "$ANDROID_SDK_MANAGER_PARAMETERS" ]]; then | ||||
|   echo "Updating Android SDK with parameters: $ANDROID_SDK_MANAGER_PARAMETERS" | ||||
|   export JAVA_HOME="$(awk -F'=' '/JAVA_HOME=/{print $2}' /usr/bin/unity-editor.d/*)" | ||||
|   "$(awk -F'=' '/ANDROID_HOME=/{print $2}' /usr/bin/unity-editor.d/*)/tools/bin/sdkmanager" "$ANDROID_SDK_MANAGER_PARAMETERS" | ||||
|   echo "Updated Android SDK." | ||||
| else | ||||
|   echo "Not updating Android SDK." | ||||
| fi | ||||
| 
 | ||||
| # | ||||
| # Pre-build debug information | ||||
| # | ||||
|  | @ -127,12 +122,14 @@ echo "" | |||
| 
 | ||||
| unity-editor \ | ||||
|   -logfile /dev/stdout \ | ||||
|   -quit \ | ||||
|   $( [ "${MANUAL_EXIT}" == "true" ] || echo "-quit" ) \ | ||||
|   -customBuildName "$BUILD_NAME" \ | ||||
|   -projectPath "$UNITY_PROJECT_PATH" \ | ||||
|   -buildTarget "$BUILD_TARGET" \ | ||||
|   $( [ -z "$BUILD_PROFILE" ] && echo "-buildTarget $BUILD_TARGET" ) \ | ||||
|   -customBuildTarget "$BUILD_TARGET" \ | ||||
|   -customBuildPath "$CUSTOM_BUILD_PATH" \ | ||||
|   -customBuildProfile "$BUILD_PROFILE" \ | ||||
|   ${BUILD_PROFILE:+-activeBuildProfile} ${BUILD_PROFILE:+"$BUILD_PROFILE"} \ | ||||
|   -executeMethod "$BUILD_METHOD" \ | ||||
|   -buildVersion "$VERSION" \ | ||||
|   -androidVersionCode "$ANDROID_VERSION_CODE" \ | ||||
|  | @ -141,6 +138,8 @@ unity-editor \ | |||
|   -androidKeyaliasName "$ANDROID_KEYALIAS_NAME" \ | ||||
|   -androidKeyaliasPass "$ANDROID_KEYALIAS_PASS" \ | ||||
|   -androidTargetSdkVersion "$ANDROID_TARGET_SDK_VERSION" \ | ||||
|   -androidExportType "$ANDROID_EXPORT_TYPE" \ | ||||
|   -androidSymbolType "$ANDROID_SYMBOL_TYPE" \ | ||||
|   $CUSTOM_PARAMETERS | ||||
| 
 | ||||
| # Catch exit code | ||||
|  | @ -159,6 +158,7 @@ fi | |||
| 
 | ||||
| # Make a given user owner of all artifacts | ||||
| if [[ -n "$CHOWN_FILES_TO" ]]; then | ||||
|   echo "Changing ownership of files to $CHOWN_FILES_TO for $BUILD_PATH_FULL and $UNITY_PROJECT_PATH" | ||||
|   chown -R "$CHOWN_FILES_TO" "$BUILD_PATH_FULL" | ||||
|   chown -R "$CHOWN_FILES_TO" "$UNITY_PROJECT_PATH" | ||||
| fi | ||||
|  |  | |||
|  | @ -1,20 +1,22 @@ | |||
| #!/usr/bin/env bash | ||||
| 
 | ||||
| # Run in ACTIVATE_LICENSE_PATH directory | ||||
| echo "Changing to \"$ACTIVATE_LICENSE_PATH\" directory." | ||||
| pushd "$ACTIVATE_LICENSE_PATH" | ||||
| 
 | ||||
| if [[ -n "$UNITY_SERIAL" ]]; then | ||||
| if [[ -n "$UNITY_LICENSING_SERVER" ]]; then | ||||
|   # | ||||
|   # PROFESSIONAL (SERIAL) LICENSE MODE | ||||
|   # Return any floating license used. | ||||
|   # | ||||
|   echo "Returning floating license: \"$FLOATING_LICENSE\"" | ||||
|   /opt/unity/Editor/Data/Resources/Licensing/Client/Unity.Licensing.Client --return-floating "$FLOATING_LICENSE" | ||||
| elif [[ -n "$UNITY_SERIAL" ]]; then | ||||
|   # | ||||
|   # SERIAL LICENSE MODE | ||||
|   # | ||||
|   # This will return the license that is currently in use. | ||||
|   # | ||||
|   unity-editor \ | ||||
|     -logFile /dev/stdout \ | ||||
|     -quit \ | ||||
|     -returnlicense | ||||
|     -returnlicense \ | ||||
|     -username "$UNITY_EMAIL" \ | ||||
|     -password "$UNITY_PASSWORD" \ | ||||
|     -projectPath "/BlankProject" | ||||
| fi | ||||
| 
 | ||||
| # Return to previous working directory | ||||
| popd | ||||
|  |  | |||
|  | @ -0,0 +1,48 @@ | |||
| #!/usr/bin/env bash | ||||
| 
 | ||||
| # | ||||
| # Run steps | ||||
| # | ||||
| source /steps/set_extra_git_configs.sh | ||||
| source /steps/set_gitcredential.sh | ||||
| 
 | ||||
| if [ "$SKIP_ACTIVATION" != "true" ]; then | ||||
|   source /steps/activate.sh | ||||
| 
 | ||||
|   # If we didn't activate successfully, exit with the exit code from the activation step. | ||||
|   if [[ $UNITY_EXIT_CODE -ne 0 ]]; then | ||||
|     exit $UNITY_EXIT_CODE | ||||
|   fi | ||||
| else | ||||
|   echo "Skipping activation" | ||||
| fi | ||||
| 
 | ||||
| source /steps/build.sh | ||||
| 
 | ||||
| if [ "$SKIP_ACTIVATION" != "true" ]; then | ||||
|   source /steps/return_license.sh | ||||
| fi | ||||
| 
 | ||||
| # | ||||
| # Instructions for debugging | ||||
| # | ||||
| 
 | ||||
| if [[ $BUILD_EXIT_CODE -gt 0 ]]; then | ||||
| echo "" | ||||
| echo "###########################" | ||||
| echo "#         Failure         #" | ||||
| echo "###########################" | ||||
| echo "" | ||||
| echo "Please note that the exit code is not very descriptive." | ||||
| echo "Most likely it will not help you solve the issue." | ||||
| echo "" | ||||
| echo "To find the reason for failure: please search for errors in the log above and check for annotations in the summary view." | ||||
| echo "" | ||||
| fi; | ||||
| 
 | ||||
| # | ||||
| # Exit with code from the build step. | ||||
| # | ||||
| 
 | ||||
| # Exiting su | ||||
| exit $BUILD_EXIT_CODE | ||||
|  | @ -0,0 +1,29 @@ | |||
| #!/usr/bin/env bash | ||||
| 
 | ||||
| if [ -z "${GIT_CONFIG_EXTENSIONS}" ] | ||||
| then | ||||
|   echo "GIT_CONFIG_EXTENSIONS unset skipping" | ||||
| else | ||||
|   echo "GIT_CONFIG_EXTENSIONS is set configuring extra git configs" | ||||
| 
 | ||||
|   IFS=$'\n' | ||||
|   for config in $(echo "${GIT_CONFIG_EXTENSIONS}" | sed 's/\(.*\)=\(.*\)/"\1" "\2"/g'); do | ||||
|     if [[ $config =~ \"([^\"]+)\"\ \"([^\"]+)\" ]]; then | ||||
|       key="${BASH_REMATCH[1]}" | ||||
|       value="${BASH_REMATCH[2]}" | ||||
|     else | ||||
|       echo "Error parsing config: $config" | ||||
|       exit 1 | ||||
|     fi | ||||
|     echo "Adding extra git config: \"$key\" = \"$value\"" | ||||
|     git config --global --add "$key" "$value" | ||||
|   done | ||||
|   unset IFS | ||||
| 
 | ||||
| fi | ||||
| 
 | ||||
| echo "---------- git config --list -------------" | ||||
| git config --list | ||||
| 
 | ||||
| echo "---------- git config --list --show-origin -------------" | ||||
| git config --list --show-origin | ||||
|  | @ -7,10 +7,10 @@ else | |||
|   echo "GIT_PRIVATE_TOKEN is set configuring git credentials" | ||||
| 
 | ||||
| 	git config --global credential.helper store | ||||
| 	git config --global --replace-all url.https://github.com/.insteadOf ssh://git@github.com/ | ||||
| 	git config --global --add url.https://github.com/.insteadOf git@github.com | ||||
| 
 | ||||
|   git config --global url."https://token:$GIT_PRIVATE_TOKEN@github.com/".insteadOf "https://github.com/" | ||||
| 	git config --global --replace-all url."https://token:$GIT_PRIVATE_TOKEN@github.com/".insteadOf ssh://git@github.com/ | ||||
| 	git config --global --add url."https://token:$GIT_PRIVATE_TOKEN@github.com/".insteadOf git@github.com | ||||
|   git config --global --add url."https://token:$GIT_PRIVATE_TOKEN@github.com/".insteadOf "https://github.com/" | ||||
|    | ||||
|   git config --global url."https://ssh:$GIT_PRIVATE_TOKEN@github.com/".insteadOf "ssh://git@github.com/" | ||||
|   git config --global url."https://git:$GIT_PRIVATE_TOKEN@github.com/".insteadOf "git@github.com:" | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,7 +1,93 @@ | |||
| # Activates Unity | ||||
| & "C:\Program Files\Unity\Hub\Editor\$Env:UNITY_VERSION\Editor\Unity.exe" -batchmode -quit -nographics ` | ||||
|                                                                           -username $Env:UNITY_EMAIL ` | ||||
|                                                                           -password $Env:UNITY_PASSWORD ` | ||||
|                                                                           -serial $Env:UNITY_SERIAL ` | ||||
|                                                                           -projectPath "c:/BlankProject" ` | ||||
|                                                                           -logfile | Out-Host | ||||
| 
 | ||||
| Write-Output "" | ||||
| Write-Output "###########################" | ||||
| Write-Output "#        Activating       #" | ||||
| Write-Output "###########################" | ||||
| Write-Output "" | ||||
| 
 | ||||
| if ( ($null -ne ${env:UNITY_SERIAL}) -and ($null -ne ${env:UNITY_EMAIL}) -and ($null -ne ${env:UNITY_PASSWORD}) ) | ||||
| { | ||||
|   # | ||||
|   # SERIAL LICENSE MODE | ||||
|   # | ||||
|   # This will activate unity, using the serial activation process. | ||||
|   # | ||||
|   Write-Output "Requesting activation" | ||||
| 
 | ||||
|   $ACTIVATION_OUTPUT = Start-Process -FilePath "$Env:UNITY_PATH/Editor/Unity.exe" ` | ||||
|                                      -NoNewWindow ` | ||||
|                                      -PassThru ` | ||||
|                                      -ArgumentList  "-batchmode ` | ||||
|                                                      -quit ` | ||||
|                                                      -nographics ` | ||||
|                                                      -username $Env:UNITY_EMAIL ` | ||||
|                                                      -password $Env:UNITY_PASSWORD ` | ||||
|                                                      -serial $Env:UNITY_SERIAL ` | ||||
|                                                      -projectPath c:/BlankProject ` | ||||
|                                                      -logfile -" | ||||
| 
 | ||||
|   # Cache the handle so exit code works properly | ||||
|   # https://stackoverflow.com/questions/10262231/obtaining-exitcode-using-start-process-and-waitforexit-instead-of-wait | ||||
|   $unityHandle = $ACTIVATION_OUTPUT.Handle | ||||
| 
 | ||||
|   while ($true) { | ||||
|       if ($ACTIVATION_OUTPUT.HasExited) { | ||||
|         $ACTIVATION_EXIT_CODE = $ACTIVATION_OUTPUT.ExitCode | ||||
| 
 | ||||
|         # Display results | ||||
|         if ($ACTIVATION_EXIT_CODE -eq 0) | ||||
|         { | ||||
|             Write-Output "Activation Succeeded" | ||||
|         } else | ||||
|         { | ||||
|             Write-Output "Activation failed, with exit code $ACTIVATION_EXIT_CODE" | ||||
|         } | ||||
| 
 | ||||
|         break | ||||
|       } | ||||
| 
 | ||||
|       Start-Sleep -Seconds 3 | ||||
|   } | ||||
| } | ||||
| elseif( ($null -ne ${env:UNITY_LICENSING_SERVER})) | ||||
| { | ||||
|     # | ||||
|     # Custom Unity License Server | ||||
|     # | ||||
| 
 | ||||
|     Write-Output "Adding licensing server config" | ||||
| 
 | ||||
|     $ACTIVATION_OUTPUT = Start-Process -FilePath "$Env:UNITY_PATH\Editor\Data\Resources\Licensing\Client\Unity.Licensing.Client.exe" ` | ||||
|                                        -ArgumentList "--acquire-floating" ` | ||||
|                                        -NoNewWindow ` | ||||
|                                        -PassThru ` | ||||
|                                        -Wait ` | ||||
|                                        -RedirectStandardOutput "license.txt" | ||||
| 
 | ||||
|     $PARSEDFILE = (Get-Content "license.txt" | Select-String -AllMatches -Pattern '\".*?\"' | ForEach-Object { $_.Matches.Value }) -replace '"' | ||||
| 
 | ||||
|     $env:FLOATING_LICENSE = $PARSEDFILE[1] | ||||
|     $FLOATING_LICENSE_TIMEOUT = $PARSEDFILE[3] | ||||
| 
 | ||||
|     Write-Output "Acquired floating license: ""$env:FLOATING_LICENSE"" with timeout $FLOATING_LICENSE_TIMEOUT" | ||||
|     # Store the exit code from the verify command | ||||
|     $ACTIVATION_EXIT_CODE = $ACTIVATION_OUTPUT.ExitCode | ||||
| } | ||||
| else | ||||
| { | ||||
|     # | ||||
|     # NO LICENSE ACTIVATION STRATEGY MATCHED | ||||
|     # | ||||
|     # This will exit since no activation strategies could be matched. | ||||
|     # | ||||
|     Write-Output "License activation strategy could not be determined." | ||||
|     Write-Output "" | ||||
|     Write-Output "Visit https://game.ci/docs/github/activation for more" | ||||
|     Write-Output "details on how to set up one of the possible activation strategies." | ||||
| 
 | ||||
|     Write-Output "::error ::No valid license activation strategy could be determined. Make sure to provide UNITY_EMAIL, UNITY_PASSWORD, and either a UNITY_SERIAL \ | ||||
| or UNITY_LICENSE. See more info at https://game.ci/docs/github/activation" | ||||
| 
 | ||||
|     $ACTIVATION_EXIT_CODE = 1; | ||||
| } | ||||
|  |  | |||
|  | @ -16,6 +16,25 @@ Write-Output "$('Using build name "')$($Env:BUILD_NAME)$('".')" | |||
| 
 | ||||
| Write-Output "$('Using build target "')$($Env:BUILD_TARGET)$('".')" | ||||
| 
 | ||||
| # | ||||
| # Display the build profile | ||||
| # | ||||
| 
 | ||||
| if ($Env:BUILD_PROFILE) | ||||
| { | ||||
|   # User has provided a path to a build profile `.asset` file | ||||
|   # | ||||
|   Write-Output "$('Using build profile "')$($Env:BUILD_PROFILE)$('" relative to "')$($Env:UNITY_PROJECT_PATH)$('".')" | ||||
|   # | ||||
| } | ||||
| else | ||||
| { | ||||
|   # User has not provided a build profile | ||||
|   # | ||||
|   Write-Output "$('Doing a default "')$($Env:BUILD_TARGET)$('" platform build.')" | ||||
|   # | ||||
| } | ||||
| 
 | ||||
| # | ||||
| # Display build path and file | ||||
| # | ||||
|  | @ -66,6 +85,26 @@ else | |||
|     Get-ChildItem -Path $Env:UNITY_PROJECT_PATH\Assets\Editor -Recurse | ||||
| } | ||||
| 
 | ||||
| if ( "$Env:BUILD_TARGET" -eq "Android" -and -not ([string]::IsNullOrEmpty("$Env:ANDROID_KEYSTORE_BASE64")) ) | ||||
| { | ||||
|     Write-Output "Creating Android keystore." | ||||
| 
 | ||||
|     # Write to consistent location as Windows Unity seems to have issues with pwd and can't find the keystore | ||||
|     $keystorePath = "C:/android.keystore" | ||||
|     [System.IO.File]::WriteAllBytes($keystorePath, [System.Convert]::FromBase64String($Env:ANDROID_KEYSTORE_BASE64)) | ||||
| 
 | ||||
|     # Ensure the project settings are pointed at the correct path | ||||
|     $unitySettingsPath = "$Env:UNITY_PROJECT_PATH\ProjectSettings\ProjectSettings.asset" | ||||
|     $fileContent = Get-Content -Path "$unitySettingsPath" | ||||
|     $fileContent = $fileContent -replace "AndroidKeystoreName:\s+.*", "AndroidKeystoreName: $keystorePath" | ||||
|     $fileContent | Set-Content -Path "$unitySettingsPath" | ||||
| 
 | ||||
|     Write-Output "Created Android keystore." | ||||
| } | ||||
| else { | ||||
|     Write-Output "Not creating Android keystore." | ||||
| } | ||||
| 
 | ||||
| # | ||||
| # Pre-build debug information | ||||
| # | ||||
|  | @ -109,39 +148,84 @@ Write-Output "#    Building project     #" | |||
| Write-Output "###########################" | ||||
| Write-Output "" | ||||
| 
 | ||||
| & "C:\Program Files\Unity\Hub\Editor\$Env:UNITY_VERSION\Editor\Unity.exe" -quit -batchmode -nographics ` | ||||
|                                                                           -projectPath $Env:UNITY_PROJECT_PATH ` | ||||
|                                                                           -executeMethod $Env:BUILD_METHOD ` | ||||
|                                                                           -buildTarget $Env:BUILD_TARGET ` | ||||
|                                                                           -customBuildTarget $Env:BUILD_TARGET ` | ||||
|                                                                           -customBuildPath $Env:CUSTOM_BUILD_PATH ` | ||||
|                                                                           -buildVersion $Env:VERSION ` | ||||
|                                                                           $Env:CUSTOM_PARAMETERS ` | ||||
|                                                                           -logfile | Out-Host | ||||
| $unityGraphics = "-nographics" | ||||
| 
 | ||||
| # Catch exit code | ||||
| $Env:BUILD_EXIT_CODE=$? | ||||
| 
 | ||||
| # Display results | ||||
| if ($Env:BUILD_EXIT_CODE -eq 0) | ||||
| if ($LLVMPIPE_INSTALLED -eq "true") | ||||
| { | ||||
|     Write-Output "Build Succeeded!" | ||||
| } else | ||||
| { | ||||
|     Write-Output "$('Build failed, with exit code ')$($Env:BUILD_EXIT_CODE)$('"')" | ||||
|   $unityGraphics = "-force-opengl" | ||||
| } | ||||
| 
 | ||||
| # TODO: Determine if we need to set permissions on any files | ||||
| # If $Env:CUSTOM_PARAMETERS contains spaces and is passed directly on the command line to Unity, powershell will wrap it | ||||
| # in double quotes.  To avoid this, parse $Env:CUSTOM_PARAMETERS into an array, while respecting any quotations within the string. | ||||
| $_, $customParametersArray = Invoke-Expression('Write-Output -- "" ' + $Env:CUSTOM_PARAMETERS) | ||||
| $unityArgs = @( | ||||
|     "-quit", | ||||
|     "-batchmode", | ||||
|     $unityGraphics, | ||||
|     "-silent-crashes", | ||||
|     "-customBuildName", "`"$Env:BUILD_NAME`"", | ||||
|     "-projectPath", "`"$Env:UNITY_PROJECT_PATH`"", | ||||
|     "-executeMethod", "`"$Env:BUILD_METHOD`"", | ||||
|     "-customBuildTarget", "`"$Env:BUILD_TARGET`"", | ||||
|     "-customBuildPath", "`"$Env:CUSTOM_BUILD_PATH`"", | ||||
|     "-customBuildProfile", "`"$Env:BUILD_PROFILE`"", | ||||
|     "-buildVersion", "`"$Env:VERSION`"", | ||||
|     "-androidVersionCode", "`"$Env:ANDROID_VERSION_CODE`"", | ||||
|     "-androidKeystorePass", "`"$Env:ANDROID_KEYSTORE_PASS`"", | ||||
|     "-androidKeyaliasName", "`"$Env:ANDROID_KEYALIAS_NAME`"", | ||||
|     "-androidKeyaliasPass", "`"$Env:ANDROID_KEYALIAS_PASS`"", | ||||
|     "-androidTargetSdkVersion", "`"$Env:ANDROID_TARGET_SDK_VERSION`"", | ||||
|     "-androidExportType", "`"$Env:ANDROID_EXPORT_TYPE`"", | ||||
|     "-androidSymbolType", "`"$Env:ANDROID_SYMBOL_TYPE`"", | ||||
|     "-logfile", "-" | ||||
| ) + $customParametersArray | ||||
| 
 | ||||
| # | ||||
| # Results | ||||
| # | ||||
| if (-not $Env:BUILD_PROFILE) { | ||||
|     $unityArgs += @("-buildTarget", "`"$Env:BUILD_TARGET`"") | ||||
| } | ||||
| if ($Env:BUILD_PROFILE) { | ||||
|     $unityArgs += @("-activeBuildProfile", "`"$Env:BUILD_PROFILE`"") | ||||
| } | ||||
| 
 | ||||
| Write-Output "" | ||||
| Write-Output "###########################" | ||||
| Write-Output "#       Build output      #" | ||||
| Write-Output "###########################" | ||||
| Write-Output "" | ||||
| # Remove null items as that will fail the Start-Process call | ||||
| $unityArgs = $unityArgs | Where-Object { $_ -ne $null } | ||||
| 
 | ||||
| Get-ChildItem $Env:BUILD_PATH_FULL | ||||
| Write-Output "" | ||||
| $unityProcess = Start-Process -FilePath "$Env:UNITY_PATH/Editor/Unity.exe" ` | ||||
|                               -ArgumentList $unityArgs ` | ||||
|                               -PassThru ` | ||||
|                               -NoNewWindow | ||||
| 
 | ||||
| # Cache the handle so exit code works properly | ||||
| # https://stackoverflow.com/questions/10262231/obtaining-exitcode-using-start-process-and-waitforexit-instead-of-wait | ||||
| $unityHandle = $unityProcess.Handle | ||||
| 
 | ||||
| while ($true) { | ||||
|     if ($unityProcess.HasExited) { | ||||
|       Start-Sleep -Seconds 3 | ||||
|       Get-Process | ||||
| 
 | ||||
|       $BUILD_EXIT_CODE = $unityProcess.ExitCode | ||||
| 
 | ||||
|       # Display results | ||||
|       if ($BUILD_EXIT_CODE -eq 0) | ||||
|       { | ||||
|           Write-Output "Build Succeeded!!" | ||||
|       } else | ||||
|       { | ||||
|           Write-Output "Build failed, with exit code $BUILD_EXIT_CODE" | ||||
|       } | ||||
| 
 | ||||
|       Write-Output "" | ||||
|       Write-Output "###########################" | ||||
|       Write-Output "#       Build output      #" | ||||
|       Write-Output "###########################" | ||||
|       Write-Output "" | ||||
| 
 | ||||
|       Get-ChildItem $Env:BUILD_PATH_FULL | ||||
|       Write-Output "" | ||||
| 
 | ||||
|       break | ||||
|     } | ||||
| 
 | ||||
|     Start-Sleep -Seconds 3 | ||||
| } | ||||
|  |  | |||
|  | @ -1,15 +1,52 @@ | |||
| # Activate Unity | ||||
| & "c:\steps\activate.ps1" | ||||
| Get-Process | ||||
| 
 | ||||
| # Copy .upmconfig.toml if it exists | ||||
| if (Test-Path "C:\githubhome\.upmconfig.toml") { | ||||
|   Write-Host "Copying .upmconfig.toml to $Env:USERPROFILE\.upmconfig.toml" | ||||
|   Copy-Item -Path "C:\githubhome\.upmconfig.toml" -Destination "$Env:USERPROFILE\.upmconfig.toml" -Force | ||||
| } else { | ||||
|   Write-Host "No .upmconfig.toml found at C:\githubhome" | ||||
| } | ||||
| 
 | ||||
| # Import any necessary registry keys, ie: location of windows 10 sdk | ||||
| # No guarantee that there will be any necessary registry keys, ie: tvOS | ||||
| Get-ChildItem -Path c:\regkeys -File | Foreach {reg import $_.fullname} | ||||
| Get-ChildItem -Path c:\regkeys -File | ForEach-Object { reg import $_.fullname } | ||||
| 
 | ||||
| # Register the Visual Studio installation so Unity can find it | ||||
| regsvr32 C:\ProgramData\Microsoft\VisualStudio\Setup\x64\Microsoft.VisualStudio.Setup.Configuration.Native.dll | ||||
| 
 | ||||
| # Kill the regsvr process | ||||
| Get-Process -Name regsvr32 | ForEach-Object { Stop-Process -Id $_.Id -Force } | ||||
| 
 | ||||
| # Setup Git Credentials | ||||
| . "c:\steps\set_gitcredential.ps1" | ||||
| 
 | ||||
| if ($env:ENABLE_GPU -eq "true") { | ||||
|   # Install LLVMpipe software graphics driver | ||||
|   . "c:\steps\install_llvmpipe.ps1" | ||||
| } | ||||
| 
 | ||||
| # Activate Unity | ||||
| if ($env:SKIP_ACTIVATION -ne "true") { | ||||
|   . "c:\steps\activate.ps1" | ||||
| 
 | ||||
|   # If we didn't activate successfully, exit with the exit code from the activation step. | ||||
|   if ($ACTIVATION_EXIT_CODE -ne 0) { | ||||
|     exit $ACTIVATION_EXIT_CODE | ||||
|   } | ||||
| } | ||||
| else { | ||||
|   Write-Host "Skipping activation" | ||||
| } | ||||
| 
 | ||||
| # Build the project | ||||
| & "c:\steps\build.ps1" | ||||
| . "c:\steps\build.ps1" | ||||
| 
 | ||||
| # Free the seat for the activated license | ||||
| & "c:\steps\return_license.ps1" | ||||
| if ($env:SKIP_ACTIVATION -ne "true") { | ||||
|   . "c:\steps\return_license.ps1" | ||||
| } | ||||
| 
 | ||||
| Get-Process | ||||
| 
 | ||||
| exit $BUILD_EXIT_CODE | ||||
|  |  | |||
|  | @ -0,0 +1,56 @@ | |||
| $Private:repo = "mmozeiko/build-mesa" | ||||
| $Private:downloadPath = "$Env:TEMP\mesa.zip" | ||||
| $Private:extractPath = "$Env:TEMP\mesa" | ||||
| $Private:destinationPath = "$Env:UNITY_PATH\Editor\" | ||||
| $Private:version = "25.1.0" | ||||
| 
 | ||||
| $LLVMPIPE_INSTALLED = "false" | ||||
| 
 | ||||
| try { | ||||
|     # Get the release info from GitHub API (version fixed to decrease probability of breakage) | ||||
|     $releaseUrl = "https://api.github.com/repos/$repo/releases/tags/$version" | ||||
|     $release = Invoke-RestMethod -Uri $releaseUrl -Headers @{ "User-Agent" = "PowerShell" } | ||||
| 
 | ||||
|     # Get the download URL for the zip asset | ||||
|     $zipUrl = $release.assets | Where-Object { $_.name -like "mesa-llvmpipe-x64*.zip" } | Select-Object -First 1 -ExpandProperty browser_download_url | ||||
| 
 | ||||
|     if (-not $zipUrl) { | ||||
|         throw "No zip file found in the latest release." | ||||
|     } | ||||
| 
 | ||||
|     # Download the zip file | ||||
|     Write-Host "Downloading $zipUrl..." | ||||
|     Invoke-WebRequest -Uri $zipUrl -OutFile $downloadPath | ||||
| 
 | ||||
|     # Create extraction directory if it doesn't exist | ||||
|     if (-not (Test-Path $extractPath)) { | ||||
|         New-Item -ItemType Directory -Path $extractPath | Out-Null | ||||
|     } | ||||
| 
 | ||||
|     # Extract the zip file | ||||
|     Write-Host "Extracting $downloadPath to $extractPath..." | ||||
|     Expand-Archive -Path $downloadPath -DestinationPath $extractPath -Force | ||||
| 
 | ||||
|     # Create destination directory if it doesn't exist | ||||
|     if (-not (Test-Path $destinationPath)) { | ||||
|         New-Item -ItemType Directory -Path $destinationPath | Out-Null | ||||
|     } | ||||
| 
 | ||||
|     # Copy extracted files to destination | ||||
|     Write-Host "Copying files to $destinationPath..." | ||||
|     Copy-Item -Path "$extractPath\*" -Destination $destinationPath -Recurse -Force | ||||
| 
 | ||||
|     Write-Host "Successfully downloaded, extracted, and copied Mesa files to $destinationPath" | ||||
| 
 | ||||
|     $LLVMPIPE_INSTALLED = "true" | ||||
| } catch { | ||||
|     Write-Error "An error occurred: $_" | ||||
| } finally { | ||||
|     # Clean up temporary files | ||||
|     if (Test-Path $downloadPath) { | ||||
|         Remove-Item $downloadPath -Force | ||||
|     } | ||||
|     if (Test-Path $extractPath) { | ||||
|         Remove-Item $extractPath -Recurse -Force | ||||
|     } | ||||
| } | ||||
|  | @ -1,7 +1,61 @@ | |||
| # Return the active Unity license | ||||
| & "C:\Program Files\Unity\Hub\Editor\$Env:UNITY_VERSION\Editor\Unity.exe" -batchmode -quit -nographics ` | ||||
|                                                                           -username $Env:UNITY_EMAIL ` | ||||
|                                                                           -password $Env:UNITY_PASSWORD ` | ||||
|                                                                           -returnlicense ` | ||||
|                                                                           -projectPath "c:/BlankProject" ` | ||||
|                                                                           -logfile | Out-Host | ||||
| 
 | ||||
| Write-Output "" | ||||
| Write-Output "###########################" | ||||
| Write-Output "#      Return License     #" | ||||
| Write-Output "###########################" | ||||
| Write-Output "" | ||||
| 
 | ||||
| if (($null -ne ${env:UNITY_LICENSING_SERVER})) | ||||
| { | ||||
|   Write-Output "Returning floating license: ""$env:FLOATING_LICENSE""" | ||||
|   Start-Process -FilePath "$Env:UNITY_PATH\Editor\Data\Resources\Licensing\Client\Unity.Licensing.Client.exe" ` | ||||
|                 -ArgumentList "--return-floating ""$env:FLOATING_LICENSE"" " ` | ||||
|                 -NoNewWindow ` | ||||
|                 -Wait | ||||
| } | ||||
| 
 | ||||
| elseif (($null -ne ${env:UNITY_SERIAL}) -and ($null -ne ${env:UNITY_EMAIL}) -and ($null -ne ${env:UNITY_PASSWORD})) | ||||
| { | ||||
|   # | ||||
|   # SERIAL LICENSE MODE | ||||
|   # | ||||
|   # This will return the license that is currently in use. | ||||
|   # | ||||
|   $RETURN_LICENSE_OUTPUT = Start-Process -FilePath "$Env:UNITY_PATH/Editor/Unity.exe" ` | ||||
|                                          -NoNewWindow ` | ||||
|                                          -PassThru ` | ||||
|                                          -ArgumentList "-batchmode ` | ||||
|                                                          -quit ` | ||||
|                                                          -nographics ` | ||||
|                                                          -username $Env:UNITY_EMAIL ` | ||||
|                                                          -password $Env:UNITY_PASSWORD ` | ||||
|                                                          -returnlicense ` | ||||
|                                                          -projectPath c:/BlankProject ` | ||||
|                                                          -logfile -" | ||||
| 
 | ||||
|   # Cache the handle so exit code works properly | ||||
|   # https://stackoverflow.com/questions/10262231/obtaining-exitcode-using-start-process-and-waitforexit-instead-of-wait | ||||
|   $unityHandle = $RETURN_LICENSE_OUTPUT.Handle | ||||
| 
 | ||||
|   while ($true) { | ||||
|       if ($RETURN_LICENSE_OUTPUT.HasExited) { | ||||
|         $RETURN_LICENSE_EXIT_CODE = $RETURN_LICENSE_OUTPUT.ExitCode | ||||
| 
 | ||||
|         # Display results | ||||
|         if ($RETURN_LICENSE_EXIT_CODE -eq 0) | ||||
|         { | ||||
|             Write-Output "License Return Succeeded" | ||||
|         } else | ||||
|         { | ||||
|             Write-Output "License Return failed, with exit code $RETURN_LICENSE_EXIT_CODE" | ||||
|             Write-Output "::warning ::License Return failed! If this is a Pro License you might need to manually ` | ||||
| free the seat in your Unity admin panel or you might run out of seats to activate with." | ||||
|         } | ||||
| 
 | ||||
|         break | ||||
|       } | ||||
| 
 | ||||
|       Start-Sleep -Seconds 3 | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,20 @@ | |||
| if ($null -eq ${env:GIT_PRIVATE_TOKEN}) { | ||||
|     Write-Host "GIT_PRIVATE_TOKEN unset skipping" | ||||
| } | ||||
| else { | ||||
|     Write-Host "GIT_PRIVATE_TOKEN is set configuring git credentials" | ||||
| 
 | ||||
|     git config --global credential.helper store | ||||
|     git config --global --replace-all url."https://token:$env:GIT_PRIVATE_TOKEN@github.com/".insteadOf "ssh://git@github.com/" | ||||
|     git config --global --add url."https://token:$env:GIT_PRIVATE_TOKEN@github.com/".insteadOf "git@github.com" | ||||
|     git config --global --add url."https://token:$env:GIT_PRIVATE_TOKEN@github.com/".insteadOf "https://github.com/" | ||||
| 
 | ||||
|     git config --global url."https://ssh:$env:GIT_PRIVATE_TOKEN@github.com/".insteadOf "ssh://git@github.com/" | ||||
|     git config --global url."https://git:$env:GIT_PRIVATE_TOKEN@github.com/".insteadOf "git@github.com:" | ||||
| } | ||||
| 
 | ||||
| Write-Host "---------- git config --list -------------" | ||||
| git config --list | ||||
| 
 | ||||
| Write-Host "---------- git config --list --show-origin -------------" | ||||
| git config --list --show-origin | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							|  | @ -0,0 +1,7 @@ | |||
| { | ||||
|   "licensingServiceBaseUrl": "%URL%", | ||||
|   "enableEntitlementLicensing": true, | ||||
|   "enableFloatingApi": true, | ||||
|   "clientConnectTimeoutSec": 5, | ||||
|   "clientHandshakeTimeoutSec": 10 | ||||
| } | ||||
|  | @ -1,60 +0,0 @@ | |||
| "use strict"; | ||||
| /* eslint-disable no-process-exit */ | ||||
| const util = require("util"); | ||||
| const { JSDOM } = require("../../../.."); | ||||
| const { READY_STATES } = require("./xhr-utils"); | ||||
| const idlUtils = require("../generated/utils"); | ||||
| const tough = require("tough-cookie"); | ||||
| 
 | ||||
| const dom = new JSDOM(); | ||||
| const xhr = new dom.window.XMLHttpRequest(); | ||||
| const xhrImpl = idlUtils.implForWrapper(xhr); | ||||
| 
 | ||||
| const chunks = []; | ||||
| 
 | ||||
| process.stdin.on("data", chunk => { | ||||
|   chunks.push(chunk); | ||||
| }); | ||||
| 
 | ||||
| process.stdin.on("end", () => { | ||||
|   const buffer = Buffer.concat(chunks); | ||||
| 
 | ||||
|   const flag = JSON.parse(buffer.toString()); | ||||
|   if (flag.body && flag.body.type === "Buffer" && flag.body.data) { | ||||
|     flag.body = Buffer.from(flag.body.data); | ||||
|   } | ||||
|   if (flag.cookieJar) { | ||||
|     flag.cookieJar = tough.CookieJar.fromJSON(flag.cookieJar); | ||||
|   } | ||||
| 
 | ||||
|   flag.synchronous = false; | ||||
|   Object.assign(xhrImpl.flag, flag); | ||||
|   const { properties } = xhrImpl; | ||||
|   xhrImpl.readyState = READY_STATES.OPENED; | ||||
|   try { | ||||
|     xhr.addEventListener("loadend", () => { | ||||
|       if (properties.error) { | ||||
|         properties.error = properties.error.stack || util.inspect(properties.error); | ||||
|       } | ||||
|       process.stdout.write(JSON.stringify({ | ||||
|         responseURL: xhrImpl.responseURL, | ||||
|         status: xhrImpl.status, | ||||
|         statusText: xhrImpl.statusText, | ||||
|         properties | ||||
|       }), () => { | ||||
|         process.exit(0); | ||||
|       }); | ||||
|     }, false); | ||||
|     xhr.send(flag.body); | ||||
|   } catch (error) { | ||||
|     properties.error += error.stack || util.inspect(error); | ||||
|     process.stdout.write(JSON.stringify({ | ||||
|       responseURL: xhrImpl.responseURL, | ||||
|       status: xhrImpl.status, | ||||
|       statusText: xhrImpl.statusText, | ||||
|       properties | ||||
|     }), () => { | ||||
|       process.exit(0); | ||||
|     }); | ||||
|   } | ||||
| }); | ||||
|  | @ -0,0 +1,3 @@ | |||
| hook: after | ||||
| commands: | | ||||
|   echo "after-build hook test!" | ||||
|  | @ -0,0 +1,3 @@ | |||
| hook: before | ||||
| commands: | | ||||
|   echo "before-build hook test!!" | ||||
|  | @ -0,0 +1,3 @@ | |||
| hook: after | ||||
| commands: | | ||||
|   echo "after-build step test!" | ||||
|  | @ -0,0 +1,3 @@ | |||
| hook: before | ||||
| commands: | | ||||
|   echo "before-build step test!" | ||||
|  | @ -25,6 +25,8 @@ module.exports = { | |||
|   // An array of regexp pattern strings, matched against all module paths before considered 'visible' to the module loader
 | ||||
|   modulePathIgnorePatterns: ['<rootDir>/lib/', '<rootDir>/dist/'], | ||||
| 
 | ||||
|   // A list of paths to modules that run some code to configure or set up the testing framework before each test
 | ||||
|   // Files that will be run before Jest is loaded to set globals like fetch
 | ||||
|   setupFiles: ['<rootDir>/src/jest.globals.ts'], | ||||
|   // A list of paths to modules that run some code to configure or set up the testing framework after the environment is ready
 | ||||
|   setupFilesAfterEnv: ['<rootDir>/src/jest.setup.ts'], | ||||
| }; | ||||
|  |  | |||
|  | @ -6,13 +6,6 @@ | |||
| color: true | ||||
| extends: {} | ||||
| 
 | ||||
| pre-push: | ||||
|   parallel: true | ||||
|   commands: | ||||
|     packages-audit: | ||||
|       tags: security | ||||
|       run: yarn audit | ||||
| 
 | ||||
| pre-commit: | ||||
|   parallel: true | ||||
|   commands: | ||||
|  |  | |||
							
								
								
									
										64
									
								
								package.json
								
								
								
								
							
							
						
						
									
										64
									
								
								package.json
								
								
								
								
							|  | @ -1,6 +1,6 @@ | |||
| { | ||||
|   "name": "unity-builder", | ||||
|   "version": "2.0.0", | ||||
|   "version": "3.0.0", | ||||
|   "description": "Build Unity projects for different platforms.", | ||||
|   "main": "dist/index.js", | ||||
|   "repository": "git@github.com:game-ci/unity-builder.git", | ||||
|  | @ -12,42 +12,58 @@ | |||
|     "lint": "prettier --check \"src/**/*.{js,ts}\" && eslint src/**/*.ts", | ||||
|     "format": "prettier --write \"src/**/*.{js,ts}\"", | ||||
|     "cli": "yarn ts-node src/index.ts -m cli", | ||||
|     "cli-aws": "cross-env cloudRunnerCluster=aws yarn run test-cli", | ||||
|     "cli-k8s": "cross-env cloudRunnerCluster=k8s yarn run test-cli", | ||||
|     "gcp-secrets-tests": "cross-env providerStrategy=aws cloudRunnerTests=true inputPullCommand=\"gcp-secret-manager\" populateOverride=true pullInputList=UNITY_EMAIL,UNITY_SERIAL,UNITY_PASSWORD yarn test -i -t \"cloud runner\"", | ||||
|     "gcp-secrets-cli": "cross-env cloudRunnerTests=true USE_IL2CPP=false inputPullCommand=\"gcp-secret-manager\" yarn ts-node src/index.ts -m cli --populateOverride true --pullInputList UNITY_EMAIL,UNITY_SERIAL,UNITY_PASSWORD", | ||||
|     "aws-secrets-cli": "cross-env cloudRunnerTests=true inputPullCommand=\"aws-secret-manager\" yarn ts-node src/index.ts -m cli --populateOverride true --pullInputList UNITY_EMAIL,UNITY_SERIAL,UNITY_PASSWORD", | ||||
|     "cli-aws": "cross-env providerStrategy=aws yarn run test-cli", | ||||
|     "cli-k8s": "cross-env providerStrategy=k8s yarn run test-cli", | ||||
|     "test-cli": "cross-env cloudRunnerTests=true yarn ts-node src/index.ts -m cli --projectPath test-project", | ||||
|     "test": "jest", | ||||
|     "test-i": "yarn run test-i-aws && yarn run test-i-k8s", | ||||
|     "test-i-f": "yarn run test-i-aws && yarn run test-i-k8s && yarn run cli-k8s && yarn run cli-aws", | ||||
|     "test-i-aws": "cross-env cloudRunnerTests=true cloudRunnerCluster=aws yarn test -i -t \"cloud runner\"", | ||||
|     "test-i-k8s": "cross-env cloudRunnerTests=true cloudRunnerCluster=k8s yarn test -i -t \"cloud runner\"" | ||||
|     "test-i": "cross-env cloudRunnerTests=true yarn test -i -t \"cloud runner\"", | ||||
|     "test-i-*": "yarn run test-i-aws && yarn run test-i-k8s", | ||||
|     "test-i-aws": "cross-env cloudRunnerTests=true providerStrategy=aws yarn test -i -t \"cloud runner\"", | ||||
|     "test-i-k8s": "cross-env cloudRunnerTests=true providerStrategy=k8s yarn test -i -t \"cloud runner\"" | ||||
|   }, | ||||
|   "engines": { | ||||
|     "node": ">=18.x" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@actions/core": "^1.6.0", | ||||
|     "@actions/exec": "^1.1.0", | ||||
|     "@actions/github": "^5.0.0", | ||||
|     "@actions/cache": "^4.0.0", | ||||
|     "@actions/core": "^1.11.1", | ||||
|     "@actions/exec": "^1.1.1", | ||||
|     "@actions/github": "^6.0.0", | ||||
|     "@aws-sdk/client-cloudformation": "^3.777.0", | ||||
|     "@aws-sdk/client-cloudwatch-logs": "^3.777.0", | ||||
|     "@aws-sdk/client-ecs": "^3.778.0", | ||||
|     "@aws-sdk/client-kinesis": "^3.777.0", | ||||
|     "@aws-sdk/client-s3": "^3.779.0", | ||||
|     "@kubernetes/client-node": "^0.16.3", | ||||
|     "@octokit/core": "^3.5.1", | ||||
|     "@octokit/core": "^5.1.0", | ||||
|     "async-wait-until": "^2.0.12", | ||||
|     "aws-sdk": "^2.1081.0", | ||||
|     "base-64": "^1.0.0", | ||||
|     "commander": "^9.0.0", | ||||
|     "commander-ts": "^0.2.0", | ||||
|     "kubernetes-client": "^9.0.0", | ||||
|     "md5": "^2.3.0", | ||||
|     "nanoid": "^3.3.1", | ||||
|     "reflect-metadata": "^0.1.13", | ||||
|     "semver": "^7.3.5", | ||||
|     "unity-changeset": "^1.6.0", | ||||
|     "yaml": "^1.10.2" | ||||
|     "semver": "^7.5.2", | ||||
|     "ts-md5": "^1.3.1", | ||||
|     "unity-changeset": "^3.1.0", | ||||
|     "uuid": "^9.0.0", | ||||
|     "yaml": "^2.2.2" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@arkweid/lefthook": "^0.7.7", | ||||
|     "@types/base-64": "^1.0.0", | ||||
|     "@types/jest": "^27.4.1", | ||||
|     "@types/node": "^17.0.21", | ||||
|     "@types/node": "^17.0.23", | ||||
|     "@types/semver": "^7.3.9", | ||||
|     "@types/uuid": "^9.0.0", | ||||
|     "@typescript-eslint/parser": "4.8.1", | ||||
|     "@vercel/ncc": "^0.33.3", | ||||
|     "@vercel/ncc": "^0.36.1", | ||||
|     "cross-env": "^7.0.3", | ||||
|     "eslint": "7.17.0", | ||||
|     "eslint": "^7.23.0", | ||||
|     "eslint-config-prettier": "8.1.0", | ||||
|     "eslint-plugin-github": "^4.1.1", | ||||
|     "eslint-plugin-jest": "24.1.3", | ||||
|  | @ -55,11 +71,17 @@ | |||
|     "eslint-plugin-unicorn": "28.0.2", | ||||
|     "jest": "^27.5.1", | ||||
|     "jest-circus": "^27.5.1", | ||||
|     "jest-fail-on-console": "^2.3.0", | ||||
|     "jest-fail-on-console": "^3.0.2", | ||||
|     "js-yaml": "^4.1.0", | ||||
|     "lefthook": "^1.6.1", | ||||
|     "prettier": "^2.5.1", | ||||
|     "ts-jest": "^27.1.3", | ||||
|     "ts-node": "10.4.0", | ||||
|     "typescript": "4.1.3" | ||||
|     "ts-node": "10.8.1", | ||||
|     "typescript": "4.7.4", | ||||
|     "yarn-audit-fix": "^9.3.8" | ||||
|   }, | ||||
|   "volta": { | ||||
|     "node": "20.5.1", | ||||
|     "yarn": "1.22.19" | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,15 @@ | |||
| echo "installing game-ci cli" | ||||
| if exist %UserProfile%\AppData\LocalLow\game-ci\ ( | ||||
|   echo Installed Updating | ||||
|   git -C %UserProfile%\AppData\LocalLow\game-ci\ fetch | ||||
|   git -C %UserProfile%\AppData\LocalLow\game-ci\ reset --hard | ||||
|   git -C %UserProfile%\AppData\LocalLow\game-ci\ pull | ||||
|   git -C %UserProfile%\AppData\LocalLow\game-ci\ branch | ||||
| ) else ( | ||||
|   echo Not Installed Downloading... | ||||
|   mkdir %UserProfile%\AppData\LocalLow\game-ci\ | ||||
|   git clone https://github.com/game-ci/unity-builder %UserProfile%\AppData\LocalLow\game-ci\ | ||||
| ) | ||||
| 
 | ||||
| call yarn --cwd %UserProfile%\AppData\LocalLow\game-ci\ install | ||||
| call yarn --cwd %UserProfile%\AppData\LocalLow\game-ci\ run gcp-secrets-cli %* --projectPath %cd% --awsStackName game-ci-cli | ||||
							
								
								
									
										50
									
								
								src/index.ts
								
								
								
								
							
							
						
						
									
										50
									
								
								src/index.ts
								
								
								
								
							|  | @ -1,10 +1,16 @@ | |||
| import * as core from '@actions/core'; | ||||
| import { Action, BuildParameters, Cache, Docker, ImageTag, Output, CloudRunner } from './model'; | ||||
| import { CLI } from './model/cli/cli'; | ||||
| import { Action, BuildParameters, Cache, CloudRunner, Docker, ImageTag, Output } from './model'; | ||||
| import { Cli } from './model/cli/cli'; | ||||
| import MacBuilder from './model/mac-builder'; | ||||
| import PlatformSetup from './model/platform-setup'; | ||||
| 
 | ||||
| async function runMain() { | ||||
|   try { | ||||
|     if (Cli.InitCliMode()) { | ||||
|       await Cli.RunCli(); | ||||
| 
 | ||||
|       return; | ||||
|     } | ||||
|     Action.checkCompatibility(); | ||||
|     Cache.verify(); | ||||
| 
 | ||||
|  | @ -13,31 +19,35 @@ async function runMain() { | |||
|     const buildParameters = await BuildParameters.create(); | ||||
|     const baseImage = new ImageTag(buildParameters); | ||||
| 
 | ||||
|     if ( | ||||
|       buildParameters.cloudRunnerCluster && | ||||
|       buildParameters.cloudRunnerCluster !== '' && | ||||
|       buildParameters.cloudRunnerCluster !== 'local' | ||||
|     ) { | ||||
|       await CloudRunner.run(buildParameters, baseImage.toString()); | ||||
|     } else { | ||||
|     let exitCode = -1; | ||||
| 
 | ||||
|     if (buildParameters.providerStrategy === 'local') { | ||||
|       core.info('Building locally'); | ||||
|       await PlatformSetup.setup(buildParameters, actionFolder); | ||||
|       if (process.platform === 'darwin') { | ||||
|         MacBuilder.run(actionFolder, workspace, buildParameters); | ||||
|       } else { | ||||
|         await Docker.run(baseImage, { workspace, actionFolder, ...buildParameters }); | ||||
|       } | ||||
|       exitCode = | ||||
|         process.platform === 'darwin' | ||||
|           ? await MacBuilder.run(actionFolder) | ||||
|           : await Docker.run(baseImage.toString(), { | ||||
|               workspace, | ||||
|               actionFolder, | ||||
|               ...buildParameters, | ||||
|             }); | ||||
|     } else { | ||||
|       await CloudRunner.run(buildParameters, baseImage.toString()); | ||||
|       exitCode = 0; | ||||
|     } | ||||
| 
 | ||||
|     // Set output
 | ||||
|     await Output.setBuildVersion(buildParameters.buildVersion); | ||||
|     await Output.setAndroidVersionCode(buildParameters.androidVersionCode); | ||||
|     await Output.setEngineExitCode(exitCode); | ||||
| 
 | ||||
|     if (exitCode !== 0) { | ||||
|       core.setFailed(`Build failed with exit code ${exitCode}`); | ||||
|     } | ||||
|   } catch (error) { | ||||
|     core.setFailed((error as Error).message); | ||||
|   } | ||||
| } | ||||
| const options = CLI.SetupCli(); | ||||
| if (CLI.isCliMode(options)) { | ||||
|   CLI.RunCli(options); | ||||
| } else { | ||||
|   runMain(); | ||||
| } | ||||
| 
 | ||||
| runMain(); | ||||
|  |  | |||
|  | @ -0,0 +1,29 @@ | |||
| // Integration test for exercising real GitHub check creation and updates.
 | ||||
| import CloudRunner from '../model/cloud-runner/cloud-runner'; | ||||
| import UnityVersioning from '../model/unity-versioning'; | ||||
| import GitHub from '../model/github'; | ||||
| import { TIMEOUT_INFINITE, createParameters } from '../test-utils/cloud-runner-test-helpers'; | ||||
| 
 | ||||
| const runIntegration = process.env.RUN_GITHUB_INTEGRATION_TESTS === 'true'; | ||||
| const describeOrSkip = runIntegration ? describe : describe.skip; | ||||
| 
 | ||||
| describeOrSkip('Cloud Runner Github Checks Integration', () => { | ||||
|   it( | ||||
|     'creates and updates a real GitHub check', | ||||
|     async () => { | ||||
|       const buildParameter = await createParameters({ | ||||
|         versioning: 'None', | ||||
|         projectPath: 'test-project', | ||||
|         unityVersion: UnityVersioning.read('test-project'), | ||||
|         asyncCloudRunner: `true`, | ||||
|         githubChecks: `true`, | ||||
|       }); | ||||
|       await CloudRunner.setup(buildParameter); | ||||
|       const checkId = await GitHub.createGitHubCheck(`integration create`); | ||||
|       expect(checkId).not.toEqual(''); | ||||
|       await GitHub.updateGitHubCheck(`1 ${new Date().toISOString()}`, `integration`); | ||||
|       await GitHub.updateGitHubCheck(`2 ${new Date().toISOString()}`, `integration`, `success`, `completed`); | ||||
|     }, | ||||
|     TIMEOUT_INFINITE, | ||||
|   ); | ||||
| }); | ||||
|  | @ -0,0 +1,9 @@ | |||
| import { stat } from 'node:fs/promises'; | ||||
| 
 | ||||
| describe('Integrity tests', () => { | ||||
|   describe('package-lock.json', () => { | ||||
|     it('does not exist', async () => { | ||||
|       await expect(stat(`${process.cwd()}/package-lock.json`)).rejects.toThrowError(); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|  | @ -0,0 +1,3 @@ | |||
| import { fetch as undiciFetch, Headers, Request, Response } from 'undici'; | ||||
| 
 | ||||
| Object.assign(globalThis, { fetch: undiciFetch, Headers, Request, Response }); | ||||
|  | @ -0,0 +1,112 @@ | |||
| export const completelyValidSemanticVersions = [ | ||||
|   '0.0.4', | ||||
|   '1.2.3', | ||||
|   '10.20.30', | ||||
|   '1.1.2-prerelease+meta', | ||||
|   '1.1.2+meta', | ||||
|   '1.1.2+meta-valid', | ||||
|   '1.0.0-alpha', | ||||
|   '1.0.0-beta', | ||||
|   '1.0.0-alpha.beta', | ||||
|   '1.0.0-alpha.beta.1', | ||||
|   '1.0.0-alpha.1', | ||||
|   '1.0.0-alpha0.valid', | ||||
|   '1.0.0-alpha.0valid', | ||||
|   '1.0.0-alpha-a.b-c-somethinglong+build.1-aef.1-its-okay', | ||||
|   '1.0.0-rc.1+build.1', | ||||
|   '2.0.0-rc.1+build.123', | ||||
|   '1.2.3-beta', | ||||
|   '10.2.3-DEV-SNAPSHOT', | ||||
|   '1.2.3-SNAPSHOT-123', | ||||
|   '1.0.0', | ||||
|   '2.0.0', | ||||
|   '1.1.7', | ||||
|   '2.0.0+build.1848', | ||||
|   '2.0.1-alpha.1227', | ||||
|   '1.0.0-alpha+beta', | ||||
|   '1.2.3----RC-SNAPSHOT.12.9.1--.12+788', | ||||
|   '1.2.3----R-S.12.9.1--.12+meta', | ||||
|   '1.2.3----RC-SNAPSHOT.12.9.1--.12', | ||||
|   '1.0.0+0.build.1-rc.10000aaa-kk-0.1', | ||||
|   '99999999999999999999999.999999999999999999.99999999999999999', | ||||
|   '1.0.0-0A.is.legal', | ||||
| ]; | ||||
| 
 | ||||
| export const notCompletelyValidSemanticVersions = [ | ||||
|   '1', | ||||
|   '1.2', | ||||
|   '1.2.3-0123', | ||||
|   '1.2.3-0123.0123', | ||||
|   '1.1.2+.123', | ||||
|   '+invalid', | ||||
|   '-invalid', | ||||
|   '-invalid+invalid', | ||||
|   '-invalid.01', | ||||
|   'alpha', | ||||
|   'alpha.beta', | ||||
|   'alpha.beta.1', | ||||
|   'alpha.1', | ||||
|   'alpha+beta', | ||||
|   'alpha_beta', | ||||
|   'alpha.', | ||||
|   'alpha..', | ||||
|   'beta', | ||||
|   '1.0.0-alpha_beta', | ||||
|   '-alpha.', | ||||
|   '1.0.0-alpha..', | ||||
|   '1.0.0-alpha..1', | ||||
|   '1.0.0-alpha...1', | ||||
|   '1.0.0-alpha....1', | ||||
|   '1.0.0-alpha.....1', | ||||
|   '1.0.0-alpha......1', | ||||
|   '1.0.0-alpha.......1', | ||||
|   '01.1.1', | ||||
|   '1.01.1', | ||||
|   '1.1.01', | ||||
|   '1.2', | ||||
|   '1.2.3.DEV', | ||||
|   '1.2-SNAPSHOT', | ||||
|   '1.2.31.2.3----RC-SNAPSHOT.12.09.1--..12+788', | ||||
|   '1.2-RC-SNAPSHOT', | ||||
|   '-1.0.3-gamma+b7718', | ||||
|   '+justmeta', | ||||
|   '9.8.7+meta+meta', | ||||
|   '9.8.7-whatever+meta+meta', | ||||
|   '99999999999999999999999.999999999999999999.99999999999999999----RC-SNAPSHOT.12.09.1--------------------------------..12', | ||||
| ]; | ||||
| 
 | ||||
| const addVariantsPrependingV = (array: string[]) => array.map((tag) => [tag, `v${tag}`]).flat(); | ||||
| 
 | ||||
| /** | ||||
|  * Array of versions that will be detected as version tags. Not all of these are | ||||
|  * "semantic versions", but can be used to generate one. Especially using the | ||||
|  * `versioning: Semantic` option. | ||||
|  */ | ||||
| export const validVersionTagInputs = addVariantsPrependingV([ | ||||
|   '0', | ||||
|   '1', | ||||
|   '0.1', | ||||
|   '1.0', | ||||
|   '1.1.0', | ||||
|   '1.2.3', | ||||
|   ...completelyValidSemanticVersions, | ||||
| ]); | ||||
| 
 | ||||
| export const invalidVersionTagInputs = addVariantsPrependingV([ | ||||
|   '+invalid', | ||||
|   '-invalid', | ||||
|   '-invalid+invalid', | ||||
|   '-invalid.01', | ||||
|   'alpha', | ||||
|   'alpha.beta', | ||||
|   'alpha.beta.1', | ||||
|   'alpha.1', | ||||
|   'alpha+beta', | ||||
|   'alpha_beta', | ||||
|   'alpha.', | ||||
|   'alpha..', | ||||
|   'beta', | ||||
|   '-alpha.', | ||||
|   '-1.0.3-gamma+b7718', | ||||
|   '+justmeta', | ||||
| ]); | ||||
|  | @ -1,5 +1,5 @@ | |||
| import path from 'path'; | ||||
| import fs from 'fs'; | ||||
| import path from 'node:path'; | ||||
| import fs from 'node:fs'; | ||||
| import Action from './action'; | ||||
| 
 | ||||
| describe('Action', () => { | ||||
|  |  | |||
|  | @ -1,23 +1,27 @@ | |||
| import path from 'path'; | ||||
| import path from 'node:path'; | ||||
| 
 | ||||
| class Action { | ||||
|   static get supportedPlatforms() { | ||||
|   static get supportedPlatforms(): string[] { | ||||
|     return ['linux', 'win32', 'darwin']; | ||||
|   } | ||||
| 
 | ||||
|   static get isRunningLocally() { | ||||
|   static get isRunningLocally(): boolean { | ||||
|     return process.env.RUNNER_WORKSPACE === undefined; | ||||
|   } | ||||
| 
 | ||||
|   static get isRunningFromSource() { | ||||
|   static get isRunningFromSource(): boolean { | ||||
|     return path.basename(__dirname) === 'model'; | ||||
|   } | ||||
| 
 | ||||
|   static get canonicalName() { | ||||
|   static get canonicalName(): string { | ||||
|     if (Action.isRunningFromSource) { | ||||
|       return path.basename(path.dirname(path.join(path.dirname(__filename), '/..'))); | ||||
|     } | ||||
| 
 | ||||
|     return 'unity-builder'; | ||||
|   } | ||||
| 
 | ||||
|   static get rootFolder() { | ||||
|   static get rootFolder(): string { | ||||
|     if (Action.isRunningFromSource) { | ||||
|       return path.dirname(path.dirname(path.dirname(__filename))); | ||||
|     } | ||||
|  | @ -25,12 +29,12 @@ class Action { | |||
|     return path.dirname(path.dirname(__filename)); | ||||
|   } | ||||
| 
 | ||||
|   static get actionFolder() { | ||||
|   static get actionFolder(): string { | ||||
|     return `${Action.rootFolder}/dist`; | ||||
|   } | ||||
| 
 | ||||
|   static get workspace() { | ||||
|     return process.env.GITHUB_WORKSPACE; | ||||
|   static get workspace(): string { | ||||
|     return process.env.GITHUB_WORKSPACE!; | ||||
|   } | ||||
| 
 | ||||
|   static checkCompatibility() { | ||||
|  |  | |||
|  | @ -3,15 +3,15 @@ import AndroidVersioning from './android-versioning'; | |||
| describe('Android Versioning', () => { | ||||
|   describe('versionToVersionCode', () => { | ||||
|     it('defaults to 0 when versioning strategy is none', () => { | ||||
|       expect(AndroidVersioning.versionToVersionCode('none')).toBe(0); | ||||
|       expect(AndroidVersioning.versionToVersionCode('none')).toBe('0'); | ||||
|     }); | ||||
| 
 | ||||
|     it('defaults to 1 when version is not a valid semver', () => { | ||||
|       expect(AndroidVersioning.versionToVersionCode('abcd')).toBe(1); | ||||
|       expect(AndroidVersioning.versionToVersionCode('abcd')).toBe('1'); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns a number', () => { | ||||
|       expect(AndroidVersioning.versionToVersionCode('123.456.789')).toBe(123456789); | ||||
|       expect(AndroidVersioning.versionToVersionCode('123.456.789')).toBe('123456789'); | ||||
|     }); | ||||
| 
 | ||||
|     it('throw when generated version code is too large', () => { | ||||
|  | @ -21,11 +21,11 @@ describe('Android Versioning', () => { | |||
| 
 | ||||
|   describe('determineVersionCode', () => { | ||||
|     it('defaults to parsed version', () => { | ||||
|       expect(AndroidVersioning.determineVersionCode('1.2.3', '')).toBe(1002003); | ||||
|       expect(AndroidVersioning.determineVersionCode('1.2.3', '')).toBe('1002003'); | ||||
|     }); | ||||
| 
 | ||||
|     it('use specified code', () => { | ||||
|       expect(AndroidVersioning.determineVersionCode('1.2.3', 2)).toBe(2); | ||||
|       expect(AndroidVersioning.determineVersionCode('1.2.3', '2')).toBe('2'); | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|  |  | |||
|  | @ -2,24 +2,27 @@ import * as core from '@actions/core'; | |||
| import * as semver from 'semver'; | ||||
| 
 | ||||
| export default class AndroidVersioning { | ||||
|   static determineVersionCode(version, inputVersionCode) { | ||||
|     if (!inputVersionCode) { | ||||
|   static determineVersionCode(version: string, inputVersionCode: string): string { | ||||
|     if (inputVersionCode === '') { | ||||
|       return AndroidVersioning.versionToVersionCode(version); | ||||
|     } | ||||
| 
 | ||||
|     return inputVersionCode; | ||||
|   } | ||||
| 
 | ||||
|   static versionToVersionCode(version) { | ||||
|   static versionToVersionCode(version: string): string { | ||||
|     if (version === 'none') { | ||||
|       core.info(`Versioning strategy is set to ${version}, so android version code should not be applied.`); | ||||
|       return 0; | ||||
| 
 | ||||
|       return '0'; | ||||
|     } | ||||
| 
 | ||||
|     const parsedVersion = semver.parse(version); | ||||
| 
 | ||||
|     if (!parsedVersion) { | ||||
|       core.warning(`Could not parse "${version}" to semver, defaulting android version code to 1`); | ||||
|       return 1; | ||||
| 
 | ||||
|       return '1'; | ||||
|     } | ||||
| 
 | ||||
|     // The greatest value Google Plays allows is 2100000000.
 | ||||
|  | @ -32,11 +35,13 @@ export default class AndroidVersioning { | |||
|       ); | ||||
|     } | ||||
|     core.info(`Using android versionCode ${versionCode}`); | ||||
|     return versionCode; | ||||
| 
 | ||||
|     return versionCode.toString(); | ||||
|   } | ||||
| 
 | ||||
|   static determineSdkManagerParameters(targetSdkVersion) { | ||||
|   static determineSdkManagerParameters(targetSdkVersion: string) { | ||||
|     const parsedVersion = Number.parseInt(targetSdkVersion.slice(-2), 10); | ||||
| 
 | ||||
|     return Number.isNaN(parsedVersion) ? '' : `platforms;android-${parsedVersion}`; | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -5,21 +5,17 @@ import BuildParameters from './build-parameters'; | |||
| import Input from './input'; | ||||
| import Platform from './platform'; | ||||
| 
 | ||||
| // Todo - Don't use process.env directly, that's what the input model class is for.
 | ||||
| const testLicense = | ||||
|   '<?xml version="1.0" encoding="UTF-8"?><root>\n    <License id="Terms">\n        <MachineBindings>\n            <Binding Key="1" Value="576562626572264761624c65526f7578"/>\n            <Binding Key="2" Value="576562626572264761624c65526f7578"/>\n        </MachineBindings>\n        <MachineID Value="D7nTUnjNAmtsUMcnoyrqkgIbYdM="/>\n        <SerialHash Value="2033b8ac3e6faa3742ca9f0bfae44d18f2a96b80"/>\n        <Features>\n            <Feature Value="33"/>\n            <Feature Value="1"/>\n            <Feature Value="12"/>\n            <Feature Value="2"/>\n            <Feature Value="24"/>\n            <Feature Value="3"/>\n            <Feature Value="36"/>\n            <Feature Value="17"/>\n            <Feature Value="19"/>\n            <Feature Value="62"/>\n        </Features>\n        <DeveloperData Value="AQAAAEY0LUJHUlgtWEQ0RS1aQ1dWLUM1SlctR0RIQg=="/>\n        <SerialMasked Value="F4-BGRX-XD4E-ZCWV-C5JW-XXXX"/>\n        <StartDate Value="2021-02-08T00:00:00"/>\n        <UpdateDate Value="2021-02-09T00:34:57"/>\n        <InitialActivationDate Value="2021-02-08T00:34:56"/>\n        <LicenseVersion Value="6.x"/>\n        <ClientProvidedVersion Value="2018.4.30f1"/>\n        <AlwaysOnline Value="false"/>\n        <Entitlements>\n            <Entitlement Ns="unity_editor" Tag="UnityPersonal" Type="EDITOR" ValidTo="9999-12-31T00:00:00"/>\n            <Entitlement Ns="unity_editor" Tag="DarkSkin" Type="EDITOR_FEATURE" ValidTo="9999-12-31T00:00:00"/>\n        </Entitlements>\n    </License>\n<Signature xmlns="http://www.w3.org/2000/09/xmldsig#"><SignedInfo><CanonicalizationMethod Algorithm="http://www.w3.org/TR/2001/REC-xml-c14n-20010315#WithComments"/><SignatureMethod Algorithm="http://www.w3.org/2000/09/xmldsig#rsa-sha1"/><Reference URI="#Terms"><Transforms><Transform Algorithm="http://www.w3.org/2000/09/xmldsig#enveloped-signature"/></Transforms><DigestMethod Algorithm="http://www.w3.org/2000/09/xmldsig#sha1"/><DigestValue>m0Db8UK+ktnOLJBtHybkfetpcKo=</DigestValue></Reference></SignedInfo><SignatureValue>o/pUbSQAukz7+ZYAWhnA0AJbIlyyCPL7bKVEM2lVqbrXt7cyey+umkCXamuOgsWPVUKBMkXtMH8L\n5etLmD0getWIhTGhzOnDCk+gtIPfL4jMo9tkEuOCROQAXCci23VFscKcrkB+3X6h4wEOtA2APhOY\nB+wvC794o8/82ffjP79aVAi57rp3Wmzx+9pe9yMwoJuljAy2sc2tIMgdQGWVmOGBpQm3JqsidyzI\nJWG2kjnc7pDXK9pwYzXoKiqUqqrut90d+kQqRyv7MSZXR50HFqD/LI69h68b7P8Bjo3bPXOhNXGR\n9YCoemH6EkfCJxp2gIjzjWW+l2Hj2EsFQi8YXw==</SignatureValue></Signature></root>'; | ||||
| process.env.UNITY_LICENSE = testLicense; | ||||
| 
 | ||||
| const determineVersion = jest.spyOn(Versioning, 'determineBuildVersion').mockImplementation(async () => '1.3.37'); | ||||
| const determineUnityVersion = jest | ||||
|   .spyOn(UnityVersioning, 'determineUnityVersion') | ||||
|   .mockImplementation(() => '2019.2.11f1'); | ||||
| const determineSdkManagerParameters = jest | ||||
|   .spyOn(AndroidVersioning, 'determineSdkManagerParameters') | ||||
|   .mockImplementation(() => 'platforms;android-30'); | ||||
| 
 | ||||
| afterEach(() => { | ||||
|   jest.clearAllMocks(); | ||||
|   jest.restoreAllMocks(); | ||||
| }); | ||||
| 
 | ||||
| beforeEach(() => { | ||||
|   jest.spyOn(Versioning, 'determineBuildVersion').mockImplementation(async () => '1.3.37'); | ||||
|   process.env.UNITY_LICENSE = testLicense; // Todo - Don't use process.env directly, that's what the input model class is for.
 | ||||
| }); | ||||
| 
 | ||||
| describe('BuildParameters', () => { | ||||
|  | @ -29,48 +25,62 @@ describe('BuildParameters', () => { | |||
|     }); | ||||
| 
 | ||||
|     it('determines the version only once', async () => { | ||||
|       jest.spyOn(Versioning, 'determineBuildVersion').mockImplementation(async () => '1.3.37'); | ||||
|       await BuildParameters.create(); | ||||
|       expect(determineVersion).toHaveBeenCalledTimes(1); | ||||
|       await expect(Versioning.determineBuildVersion).toHaveBeenCalledTimes(1); | ||||
|     }); | ||||
| 
 | ||||
|     it('determines the unity version only once', async () => { | ||||
|       jest.spyOn(UnityVersioning, 'determineUnityVersion').mockImplementation(() => '2019.2.11f1'); | ||||
|       await BuildParameters.create(); | ||||
|       expect(determineUnityVersion).toHaveBeenCalledTimes(1); | ||||
|       expect(UnityVersioning.determineUnityVersion).toHaveBeenCalledTimes(1); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the android version code with provided input', async () => { | ||||
|       const mockValue = '42'; | ||||
|       jest.spyOn(Input, 'androidVersionCode', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidVersionCode: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ androidVersionCode: mockValue }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the android version code from version by default', async () => { | ||||
|       const mockValue = ''; | ||||
|       jest.spyOn(Input, 'androidVersionCode', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidVersionCode: 1003037 })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ androidVersionCode: '1003037' }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     it('determines the android sdk manager parameters only once', async () => { | ||||
|       jest.spyOn(AndroidVersioning, 'determineSdkManagerParameters').mockImplementation(() => 'platforms;android-30'); | ||||
|       await BuildParameters.create(); | ||||
|       expect(determineSdkManagerParameters).toHaveBeenCalledTimes(1); | ||||
|       expect(AndroidVersioning.determineSdkManagerParameters).toHaveBeenCalledTimes(1); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the targetPlatform', async () => { | ||||
|       const mockValue = 'somePlatform'; | ||||
|       jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ targetPlatform: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ targetPlatform: mockValue })); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the project path', async () => { | ||||
|       const mockValue = 'path/to/project'; | ||||
|       jest.spyOn(UnityVersioning, 'determineUnityVersion').mockImplementation(() => '2019.2.11f1'); | ||||
|       jest.spyOn(Input, 'projectPath', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ projectPath: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ projectPath: mockValue })); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the build profile', async () => { | ||||
|       const mockValue = 'path/to/build_profile.asset'; | ||||
|       jest.spyOn(Input, 'buildProfile', 'get').mockReturnValue(mockValue); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildProfile: mockValue })); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the build name', async () => { | ||||
|       const mockValue = 'someBuildName'; | ||||
|       jest.spyOn(Input, 'buildName', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildName: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildName: mockValue })); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the build path', async () => { | ||||
|  | @ -79,92 +89,135 @@ describe('BuildParameters', () => { | |||
|       const expectedBuildPath = `${mockPath}/${mockPlatform}`; | ||||
|       jest.spyOn(Input, 'buildsPath', 'get').mockReturnValue(mockPath); | ||||
|       jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(mockPlatform); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildPath: expectedBuildPath })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ buildPath: expectedBuildPath }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the build file', async () => { | ||||
|       const mockValue = 'someBuildName'; | ||||
|       const mockPlatform = 'somePlatform'; | ||||
| 
 | ||||
|       jest.spyOn(Input, 'buildName', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildFile: mockValue })); | ||||
|       jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(mockPlatform); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildFile: mockValue })); | ||||
|     }); | ||||
| 
 | ||||
|     test.each([Platform.types.StandaloneWindows, Platform.types.StandaloneWindows64])( | ||||
|       'appends exe for %s', | ||||
|       async (targetPlatform) => { | ||||
|     test.each` | ||||
|       targetPlatform                        | expectedExtension | androidExportType | ||||
|       ${Platform.types.Android}             | ${'.apk'}         | ${'androidPackage'} | ||||
|       ${Platform.types.Android}             | ${'.aab'}         | ${'androidAppBundle'} | ||||
|       ${Platform.types.Android}             | ${''}             | ${'androidStudioProject'} | ||||
|       ${Platform.types.StandaloneWindows}   | ${'.exe'}         | ${'n/a'} | ||||
|       ${Platform.types.StandaloneWindows64} | ${'.exe'}         | ${'n/a'} | ||||
|     `(
 | ||||
|       'appends $expectedExtension for $targetPlatform with androidExportType $androidExportType', | ||||
|       async ({ targetPlatform, expectedExtension, androidExportType }) => { | ||||
|         jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(targetPlatform); | ||||
|         jest.spyOn(Input, 'buildName', 'get').mockReturnValue(targetPlatform); | ||||
|         expect(BuildParameters.create()).resolves.toEqual( | ||||
|           expect.objectContaining({ buildFile: `${targetPlatform}.exe` }), | ||||
|         jest.spyOn(Input, 'androidExportType', 'get').mockReturnValue(androidExportType); | ||||
|         await expect(BuildParameters.create()).resolves.toEqual( | ||||
|           expect.objectContaining({ buildFile: `${targetPlatform}${expectedExtension}` }), | ||||
|         ); | ||||
|       }, | ||||
|     ); | ||||
| 
 | ||||
|     test.each([Platform.types.Android])('appends apk for %s', async (targetPlatform) => { | ||||
|       jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(targetPlatform); | ||||
|       jest.spyOn(Input, 'buildName', 'get').mockReturnValue(targetPlatform); | ||||
|       jest.spyOn(Input, 'androidAppBundle', 'get').mockReturnValue(false); | ||||
|       expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ buildFile: `${targetPlatform}.apk` }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     test.each([Platform.types.Android])('appends aab for %s', async (targetPlatform) => { | ||||
|       jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(targetPlatform); | ||||
|       jest.spyOn(Input, 'buildName', 'get').mockReturnValue(targetPlatform); | ||||
|       jest.spyOn(Input, 'androidAppBundle', 'get').mockReturnValue(true); | ||||
|       expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ buildFile: `${targetPlatform}.aab` }), | ||||
|       ); | ||||
|     }); | ||||
|     test.each` | ||||
|       targetPlatform                        | androidSymbolType | ||||
|       ${Platform.types.Android}             | ${'none'} | ||||
|       ${Platform.types.Android}             | ${'public'} | ||||
|       ${Platform.types.Android}             | ${'debugging'} | ||||
|       ${Platform.types.StandaloneWindows}   | ${'none'} | ||||
|       ${Platform.types.StandaloneWindows64} | ${'none'} | ||||
|     `(
 | ||||
|       'androidSymbolType is set to $androidSymbolType when targetPlatform is $targetPlatform and input targetSymbolType is $androidSymbolType', | ||||
|       async ({ targetPlatform, androidSymbolType }) => { | ||||
|         jest.spyOn(Input, 'targetPlatform', 'get').mockReturnValue(targetPlatform); | ||||
|         jest.spyOn(Input, 'androidSymbolType', 'get').mockReturnValue(androidSymbolType); | ||||
|         jest.spyOn(Input, 'buildName', 'get').mockReturnValue(targetPlatform); | ||||
|         await expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidSymbolType })); | ||||
|       }, | ||||
|     ); | ||||
| 
 | ||||
|     it('returns the build method', async () => { | ||||
|       const mockValue = 'Namespace.ClassName.BuildMethod'; | ||||
|       jest.spyOn(Input, 'buildMethod', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildMethod: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ buildMethod: mockValue })); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the android keystore name', async () => { | ||||
|       const mockValue = 'keystore.keystore'; | ||||
|       jest.spyOn(Input, 'androidKeystoreName', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidKeystoreName: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ androidKeystoreName: mockValue }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the android keystore base64-encoded content', async () => { | ||||
|       const mockValue = 'secret'; | ||||
|       jest.spyOn(Input, 'androidKeystoreBase64', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidKeystoreBase64: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ androidKeystoreBase64: mockValue }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the android keystore pass', async () => { | ||||
|       const mockValue = 'secret'; | ||||
|       jest.spyOn(Input, 'androidKeystorePass', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidKeystorePass: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ androidKeystorePass: mockValue }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the android keyalias name', async () => { | ||||
|       const mockValue = 'secret'; | ||||
|       jest.spyOn(Input, 'androidKeyaliasName', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidKeyaliasName: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ androidKeyaliasName: mockValue }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the android keyalias pass', async () => { | ||||
|       const mockValue = 'secret'; | ||||
|       jest.spyOn(Input, 'androidKeyaliasPass', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ androidKeyaliasPass: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ androidKeyaliasPass: mockValue }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the android target sdk version', async () => { | ||||
|       const mockValue = 'AndroidApiLevelAuto'; | ||||
|       jest.spyOn(Input, 'androidTargetSdkVersion', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual( | ||||
|       await expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ androidTargetSdkVersion: mockValue }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the unity licensing server address', async () => { | ||||
|       const mockValue = 'http://example.com'; | ||||
|       jest.spyOn(Input, 'unityLicensingServer', 'get').mockReturnValue(mockValue); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual( | ||||
|         expect.objectContaining({ unityLicensingServer: mockValue }), | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|     it('throws error when no unity license provider provided', async () => { | ||||
|       delete process.env.UNITY_LICENSE; // Need to delete this as it is set for every test currently
 | ||||
|       await expect(BuildParameters.create()).rejects.toThrowError(); | ||||
|     }); | ||||
| 
 | ||||
|     it('return serial when no license server is provided', async () => { | ||||
|       const mockValue = '123'; | ||||
|       delete process.env.UNITY_LICENSE; // Need to delete this as it is set for every test currently
 | ||||
|       process.env.UNITY_SERIAL = mockValue; | ||||
|       await expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ unitySerial: mockValue })); | ||||
|       delete process.env.UNITY_SERIAL; | ||||
|     }); | ||||
| 
 | ||||
|     it('returns the custom parameters', async () => { | ||||
|       const mockValue = '-profile SomeProfile -someBoolean -someValue exampleValue'; | ||||
|       jest.spyOn(Input, 'customParameters', 'get').mockReturnValue(mockValue); | ||||
|       expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ customParameters: mockValue })); | ||||
|       await expect(BuildParameters.create()).resolves.toEqual(expect.objectContaining({ customParameters: mockValue })); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|  |  | |||
|  | @ -1,25 +1,39 @@ | |||
| import { customAlphabet } from 'nanoid'; | ||||
| import * as core from '@actions/core'; | ||||
| import AndroidVersioning from './android-versioning'; | ||||
| import CloudRunnerConstants from './cloud-runner/services/cloud-runner-constants'; | ||||
| import CloudRunnerNamespace from './cloud-runner/services/cloud-runner-namespace'; | ||||
| import CloudRunnerConstants from './cloud-runner/options/cloud-runner-constants'; | ||||
| import CloudRunnerBuildGuid from './cloud-runner/options/cloud-runner-guid'; | ||||
| import Input from './input'; | ||||
| import Platform from './platform'; | ||||
| import UnityVersioning from './unity-versioning'; | ||||
| import Versioning from './versioning'; | ||||
| import { GitRepoReader } from './input-readers/git-repo'; | ||||
| import { GithubCliReader } from './input-readers/github-cli'; | ||||
| import { Cli } from './cli/cli'; | ||||
| import GitHub from './github'; | ||||
| import CloudRunnerOptions from './cloud-runner/options/cloud-runner-options'; | ||||
| import CloudRunner from './cloud-runner/cloud-runner'; | ||||
| import * as core from '@actions/core'; | ||||
| 
 | ||||
| class BuildParameters { | ||||
|   // eslint-disable-next-line no-undef
 | ||||
|   [key: string]: any; | ||||
| 
 | ||||
|   public editorVersion!: string; | ||||
|   public customImage!: string; | ||||
|   public unitySerial!: string; | ||||
|   public runnerTempPath: string | undefined; | ||||
|   public unityLicensingServer!: string; | ||||
|   public skipActivation!: string; | ||||
|   public runnerTempPath!: string; | ||||
|   public targetPlatform!: string; | ||||
|   public projectPath!: string; | ||||
|   public buildProfile!: string; | ||||
|   public buildName!: string; | ||||
|   public buildPath!: string; | ||||
|   public buildFile!: string; | ||||
|   public buildMethod!: string; | ||||
|   public buildVersion!: string; | ||||
|   public manualExit!: boolean; | ||||
|   public enableGpu!: boolean; | ||||
|   public androidVersionCode!: string; | ||||
|   public androidKeystoreName!: string; | ||||
|   public androidKeystoreBase64!: string; | ||||
|  | @ -28,23 +42,35 @@ class BuildParameters { | |||
|   public androidKeyaliasPass!: string; | ||||
|   public androidTargetSdkVersion!: string; | ||||
|   public androidSdkManagerParameters!: string; | ||||
|   public androidExportType!: string; | ||||
|   public androidSymbolType!: string; | ||||
|   public dockerCpuLimit!: string; | ||||
|   public dockerMemoryLimit!: string; | ||||
|   public dockerIsolationMode!: string; | ||||
|   public containerRegistryRepository!: string; | ||||
|   public containerRegistryImageVersion!: string; | ||||
| 
 | ||||
|   public customParameters!: string; | ||||
|   public sshAgent!: string; | ||||
|   public cloudRunnerCluster!: string; | ||||
|   public awsBaseStackName!: string; | ||||
|   public sshPublicKeysDirectoryPath!: string; | ||||
|   public providerStrategy!: string; | ||||
|   public gitPrivateToken!: string; | ||||
|   public remoteBuildCluster!: string; | ||||
|   public awsStackName!: string; | ||||
|   public kubeConfig!: string; | ||||
|   public githubToken!: string; | ||||
|   public cloudRunnerMemory!: string; | ||||
|   public cloudRunnerCpu!: string; | ||||
|   public containerMemory!: string; | ||||
|   public containerCpu!: string; | ||||
|   public kubeVolumeSize!: string; | ||||
|   public kubeVolume!: string; | ||||
|   public kubeStorageClass!: string; | ||||
|   public runAsHostUser!: string; | ||||
|   public chownFilesTo!: string; | ||||
|   public commandHooks!: string; | ||||
|   public pullInputList!: string[]; | ||||
|   public inputPullCommand!: string; | ||||
|   public cacheKey!: string; | ||||
| 
 | ||||
|   public postBuildSteps!: string; | ||||
|   public preBuildSteps!: string; | ||||
|   public postBuildContainerHooks!: string; | ||||
|   public preBuildContainerHooks!: string; | ||||
|   public customJob!: string; | ||||
|   public runNumber!: string; | ||||
|   public branch!: string; | ||||
|  | @ -52,45 +78,89 @@ class BuildParameters { | |||
|   public gitSha!: string; | ||||
|   public logId!: string; | ||||
|   public buildGuid!: string; | ||||
|   public cloudRunnerBranch!: string; | ||||
|   public cloudRunnerDebug!: boolean | undefined; | ||||
|   public buildPlatform!: string | undefined; | ||||
|   public isCliMode!: boolean; | ||||
|   public maxRetainedWorkspaces!: number; | ||||
|   public useLargePackages!: boolean; | ||||
|   public useCompressionStrategy!: boolean; | ||||
|   public garbageMaxAge!: number; | ||||
|   public githubChecks!: boolean; | ||||
|   public asyncWorkflow!: boolean; | ||||
|   public githubCheckId!: string; | ||||
|   public finalHooks!: string[]; | ||||
|   public skipLfs!: boolean; | ||||
|   public skipCache!: boolean; | ||||
|   public cacheUnityInstallationOnMac!: boolean; | ||||
|   public unityHubVersionOnMac!: string; | ||||
|   public dockerWorkspacePath!: string; | ||||
| 
 | ||||
|   public static shouldUseRetainedWorkspaceMode(buildParameters: BuildParameters) { | ||||
|     return buildParameters.maxRetainedWorkspaces > 0 && CloudRunner.lockedWorkspace !== ``; | ||||
|   } | ||||
| 
 | ||||
|   static async create(): Promise<BuildParameters> { | ||||
|     const buildFile = this.parseBuildFile(Input.buildName, Input.targetPlatform, Input.androidAppBundle); | ||||
|     const buildFile = this.parseBuildFile(Input.buildName, Input.targetPlatform, Input.androidExportType); | ||||
|     const editorVersion = UnityVersioning.determineUnityVersion(Input.projectPath, Input.unityVersion); | ||||
|     const buildVersion = await Versioning.determineBuildVersion(Input.versioningStrategy, Input.specifiedVersion); | ||||
|     const androidVersionCode = AndroidVersioning.determineVersionCode(buildVersion, Input.androidVersionCode); | ||||
|     const androidSdkManagerParameters = AndroidVersioning.determineSdkManagerParameters(Input.androidTargetSdkVersion); | ||||
| 
 | ||||
|     // Todo - Don't use process.env directly, that's what the input model class is for.
 | ||||
|     // ---
 | ||||
|     let unitySerial = ''; | ||||
|     if (!process.env.UNITY_SERIAL) { | ||||
|       //No serial was present so it is a personal license that we need to convert
 | ||||
|       if (!process.env.UNITY_LICENSE) { | ||||
|         throw new Error(`Missing Unity License File and no Serial was found. If this
 | ||||
|                           is a personal license, make sure to follow the activation | ||||
|                           steps and set the UNITY_LICENSE GitHub secret or enter a Unity | ||||
|                           serial number inside the UNITY_SERIAL GitHub secret.`);
 | ||||
|     const androidSymbolExportType = Input.androidSymbolType; | ||||
|     if (Platform.isAndroid(Input.targetPlatform)) { | ||||
|       switch (androidSymbolExportType) { | ||||
|         case 'none': | ||||
|         case 'public': | ||||
|         case 'debugging': | ||||
|           break; | ||||
|         default: | ||||
|           throw new Error( | ||||
|             `Invalid androidSymbolType: ${Input.androidSymbolType}. Must be one of: none, public, debugging`, | ||||
|           ); | ||||
|       } | ||||
|       unitySerial = this.getSerialFromLicenseFile(process.env.UNITY_LICENSE); | ||||
|     } else { | ||||
|       unitySerial = process.env.UNITY_SERIAL!; | ||||
|     } | ||||
|     core.setSecret(unitySerial); | ||||
|     // ---
 | ||||
| 
 | ||||
|     let unitySerial = ''; | ||||
|     if (Input.unityLicensingServer === '') { | ||||
|       if (!Input.unitySerial && GitHub.githubInputEnabled) { | ||||
|         // No serial was present, so it is a personal license that we need to convert
 | ||||
|         if (!Input.unityLicense) { | ||||
|           throw new Error( | ||||
|             `Missing Unity License File and no Serial was found. If this
 | ||||
|                             is a personal license, make sure to follow the activation | ||||
|                             steps and set the UNITY_LICENSE GitHub secret or enter a Unity | ||||
|                             serial number inside the UNITY_SERIAL GitHub secret.`,
 | ||||
|           ); | ||||
|         } | ||||
|         unitySerial = this.getSerialFromLicenseFile(Input.unityLicense); | ||||
|       } else { | ||||
|         unitySerial = Input.unitySerial!; | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     if (unitySerial !== undefined && unitySerial.length === 27) { | ||||
|       core.setSecret(unitySerial); | ||||
|       core.setSecret(`${unitySerial.slice(0, -4)}XXXX`); | ||||
|     } | ||||
| 
 | ||||
|     return { | ||||
|       editorVersion, | ||||
|       customImage: Input.customImage, | ||||
|       unitySerial, | ||||
| 
 | ||||
|       runnerTempPath: process.env.RUNNER_TEMP, | ||||
|       unityLicensingServer: Input.unityLicensingServer, | ||||
|       skipActivation: Input.skipActivation, | ||||
|       runnerTempPath: Input.runnerTempPath, | ||||
|       targetPlatform: Input.targetPlatform, | ||||
|       projectPath: Input.projectPath, | ||||
|       buildProfile: Input.buildProfile, | ||||
|       buildName: Input.buildName, | ||||
|       buildPath: `${Input.buildsPath}/${Input.targetPlatform}`, | ||||
|       buildFile, | ||||
|       buildMethod: Input.buildMethod, | ||||
|       buildVersion, | ||||
|       manualExit: Input.manualExit, | ||||
|       enableGpu: Input.enableGpu, | ||||
|       androidVersionCode, | ||||
|       androidKeystoreName: Input.androidKeystoreName, | ||||
|       androidKeystoreBase64: Input.androidKeystoreBase64, | ||||
|  | @ -99,46 +169,84 @@ class BuildParameters { | |||
|       androidKeyaliasPass: Input.androidKeyaliasPass, | ||||
|       androidTargetSdkVersion: Input.androidTargetSdkVersion, | ||||
|       androidSdkManagerParameters, | ||||
|       androidExportType: Input.androidExportType, | ||||
|       androidSymbolType: androidSymbolExportType, | ||||
|       customParameters: Input.customParameters, | ||||
|       sshAgent: Input.sshAgent, | ||||
|       gitPrivateToken: await Input.gitPrivateToken(), | ||||
|       sshPublicKeysDirectoryPath: Input.sshPublicKeysDirectoryPath, | ||||
|       gitPrivateToken: Input.gitPrivateToken ?? (await GithubCliReader.GetGitHubAuthToken()), | ||||
|       runAsHostUser: Input.runAsHostUser, | ||||
|       chownFilesTo: Input.chownFilesTo, | ||||
|       cloudRunnerCluster: Input.cloudRunnerCluster, | ||||
|       awsBaseStackName: Input.awsBaseStackName, | ||||
|       kubeConfig: Input.kubeConfig, | ||||
|       githubToken: await Input.githubToken(), | ||||
|       cloudRunnerMemory: Input.cloudRunnerMemory, | ||||
|       cloudRunnerCpu: Input.cloudRunnerCpu, | ||||
|       kubeVolumeSize: Input.kubeVolumeSize, | ||||
|       kubeVolume: Input.kubeVolume, | ||||
|       postBuildSteps: Input.postBuildSteps, | ||||
|       preBuildSteps: Input.preBuildSteps, | ||||
|       customJob: Input.customJob, | ||||
|       dockerCpuLimit: Input.dockerCpuLimit, | ||||
|       dockerMemoryLimit: Input.dockerMemoryLimit, | ||||
|       dockerIsolationMode: Input.dockerIsolationMode, | ||||
|       containerRegistryRepository: Input.containerRegistryRepository, | ||||
|       containerRegistryImageVersion: Input.containerRegistryImageVersion, | ||||
|       providerStrategy: CloudRunnerOptions.providerStrategy, | ||||
|       buildPlatform: CloudRunnerOptions.buildPlatform, | ||||
|       kubeConfig: CloudRunnerOptions.kubeConfig, | ||||
|       containerMemory: CloudRunnerOptions.containerMemory, | ||||
|       containerCpu: CloudRunnerOptions.containerCpu, | ||||
|       kubeVolumeSize: CloudRunnerOptions.kubeVolumeSize, | ||||
|       kubeVolume: CloudRunnerOptions.kubeVolume, | ||||
|       postBuildContainerHooks: CloudRunnerOptions.postBuildContainerHooks, | ||||
|       preBuildContainerHooks: CloudRunnerOptions.preBuildContainerHooks, | ||||
|       customJob: CloudRunnerOptions.customJob, | ||||
|       runNumber: Input.runNumber, | ||||
|       branch: await Input.branch(), | ||||
|       // Todo - move this out of UserInput and into some class that determines additional information (as needed)
 | ||||
|       githubRepo: await Input.githubRepo(), | ||||
|       remoteBuildCluster: Input.cloudRunnerCluster, | ||||
|       awsStackName: Input.awsBaseStackName, | ||||
|       branch: Input.branch.replace('/head', '') || (await GitRepoReader.GetBranch()), | ||||
|       cloudRunnerBranch: CloudRunnerOptions.cloudRunnerBranch.split('/').reverse()[0], | ||||
|       cloudRunnerDebug: CloudRunnerOptions.cloudRunnerDebug, | ||||
|       githubRepo: (Input.githubRepo ?? (await GitRepoReader.GetRemote())) || 'game-ci/unity-builder', | ||||
|       isCliMode: Cli.isCliMode, | ||||
|       awsStackName: CloudRunnerOptions.awsStackName, | ||||
|       gitSha: Input.gitSha, | ||||
|       logId: customAlphabet(CloudRunnerConstants.alphabet, 9)(), | ||||
|       buildGuid: CloudRunnerNamespace.generateBuildName(Input.runNumber, Input.targetPlatform), | ||||
|       buildGuid: CloudRunnerBuildGuid.generateGuid(Input.runNumber, Input.targetPlatform), | ||||
|       commandHooks: CloudRunnerOptions.commandHooks, | ||||
|       inputPullCommand: CloudRunnerOptions.inputPullCommand, | ||||
|       pullInputList: CloudRunnerOptions.pullInputList, | ||||
|       kubeStorageClass: CloudRunnerOptions.kubeStorageClass, | ||||
|       cacheKey: CloudRunnerOptions.cacheKey, | ||||
|       maxRetainedWorkspaces: Number.parseInt(CloudRunnerOptions.maxRetainedWorkspaces), | ||||
|       useLargePackages: CloudRunnerOptions.useLargePackages, | ||||
|       useCompressionStrategy: CloudRunnerOptions.useCompressionStrategy, | ||||
|       garbageMaxAge: CloudRunnerOptions.garbageMaxAge, | ||||
|       githubChecks: CloudRunnerOptions.githubChecks, | ||||
|       asyncWorkflow: CloudRunnerOptions.asyncCloudRunner, | ||||
|       githubCheckId: CloudRunnerOptions.githubCheckId, | ||||
|       finalHooks: CloudRunnerOptions.finalHooks, | ||||
|       skipLfs: CloudRunnerOptions.skipLfs, | ||||
|       skipCache: CloudRunnerOptions.skipCache, | ||||
|       cacheUnityInstallationOnMac: Input.cacheUnityInstallationOnMac, | ||||
|       unityHubVersionOnMac: Input.unityHubVersionOnMac, | ||||
|       dockerWorkspacePath: Input.dockerWorkspacePath, | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   static parseBuildFile(filename, platform, androidAppBundle) { | ||||
|   static parseBuildFile(filename: string, platform: string, androidExportType: string): string { | ||||
|     if (Platform.isWindows(platform)) { | ||||
|       return `${filename}.exe`; | ||||
|     } | ||||
| 
 | ||||
|     if (Platform.isAndroid(platform)) { | ||||
|       return androidAppBundle ? `${filename}.aab` : `${filename}.apk`; | ||||
|       switch (androidExportType) { | ||||
|         case `androidPackage`: | ||||
|           return `${filename}.apk`; | ||||
|         case `androidAppBundle`: | ||||
|           return `${filename}.aab`; | ||||
|         case `androidStudioProject`: | ||||
|           return filename; | ||||
|         default: | ||||
|           throw new Error( | ||||
|             `Unknown Android Export Type: ${androidExportType}. Must be one of androidPackage for apk, androidAppBundle for aab, androidStudioProject for android project`, | ||||
|           ); | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     return filename; | ||||
|   } | ||||
| 
 | ||||
|   static getSerialFromLicenseFile(license) { | ||||
|   static getSerialFromLicenseFile(license: string) { | ||||
|     const startKey = `<DeveloperData Value="`; | ||||
|     const endKey = `"/>`; | ||||
|     const startIndex = license.indexOf(startKey) + startKey.length; | ||||
|  | @ -146,6 +254,7 @@ class BuildParameters { | |||
|       throw new Error(`License File was corrupted, unable to locate serial`); | ||||
|     } | ||||
|     const endIndex = license.indexOf(endKey, startIndex); | ||||
| 
 | ||||
|     // Slice off the first 4 characters as they are garbage values
 | ||||
|     return Buffer.from(license.slice(startIndex, endIndex), 'base64').toString('binary').slice(4); | ||||
|   } | ||||
|  |  | |||
|  | @ -1,5 +1,5 @@ | |||
| import * as core from '@actions/core'; | ||||
| import fs from 'fs'; | ||||
| import fs from 'node:fs'; | ||||
| import Action from './action'; | ||||
| import Project from './project'; | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,23 +0,0 @@ | |||
| const targets = new Array(); | ||||
| export function CliFunction(key: string, description: string) { | ||||
|   return function (target: any, propertyKey: string, descriptor: PropertyDescriptor) { | ||||
|     targets.push({ | ||||
|       target, | ||||
|       propertyKey, | ||||
|       descriptor, | ||||
|       key, | ||||
|       description, | ||||
|     }); | ||||
|   }; | ||||
| } | ||||
| export function GetCliFunctions(key) { | ||||
|   return targets.find((x) => x.key === key); | ||||
| } | ||||
| export function GetAllCliModes() { | ||||
|   return targets.map((x) => { | ||||
|     return { | ||||
|       key: x.key, | ||||
|       description: x.description, | ||||
|     }; | ||||
|   }); | ||||
| } | ||||
|  | @ -0,0 +1,45 @@ | |||
| export class CliFunctionsRepository { | ||||
|   private static targets: any[] = []; | ||||
|   public static PushCliFunction( | ||||
|     target: any, | ||||
|     propertyKey: string, | ||||
|     descriptor: PropertyDescriptor, | ||||
|     key: string, | ||||
|     description: string, | ||||
|   ) { | ||||
|     CliFunctionsRepository.targets.push({ | ||||
|       target, | ||||
|       propertyKey, | ||||
|       descriptor, | ||||
|       key, | ||||
|       description, | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   public static GetCliFunctions(key: any) { | ||||
|     const results = CliFunctionsRepository.targets.find((x) => x.key === key); | ||||
|     if (results === undefined || results.length === 0) { | ||||
|       throw new Error(`no CLI mode found for ${key}`); | ||||
|     } | ||||
| 
 | ||||
|     return results; | ||||
|   } | ||||
| 
 | ||||
|   public static GetAllCliModes() { | ||||
|     return CliFunctionsRepository.targets.map((x) => { | ||||
|       return { | ||||
|         key: x.key, | ||||
|         description: x.description, | ||||
|       }; | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   // eslint-disable-next-line no-unused-vars
 | ||||
|   public static PushCliFunctionSource(cliFunction: any) {} | ||||
| } | ||||
| 
 | ||||
| export function CliFunction(key: string, description: string) { | ||||
|   return (target: any, propertyKey: string, descriptor: PropertyDescriptor) => { | ||||
|     CliFunctionsRepository.PushCliFunction(target, propertyKey, descriptor, key, description); | ||||
|   }; | ||||
| } | ||||
|  | @ -2,87 +2,174 @@ import { Command } from 'commander-ts'; | |||
| import { BuildParameters, CloudRunner, ImageTag, Input } from '..'; | ||||
| import * as core from '@actions/core'; | ||||
| import { ActionYamlReader } from '../input-readers/action-yaml'; | ||||
| import CloudRunnerLogger from '../cloud-runner/services/cloud-runner-logger'; | ||||
| import { CliFunction, GetAllCliModes, GetCliFunctions } from './cli-decorator'; | ||||
| import { RemoteClientLogger } from './remote-client/remote-client-services/remote-client-logger'; | ||||
| import { CloudRunnerState } from '../cloud-runner/state/cloud-runner-state'; | ||||
| import { SetupCloudRunnerRepository } from './remote-client/setup-cloud-runner-repository'; | ||||
| import * as SDK from 'aws-sdk'; | ||||
| import CloudRunnerLogger from '../cloud-runner/services/core/cloud-runner-logger'; | ||||
| import CloudRunnerQueryOverride from '../cloud-runner/options/cloud-runner-query-override'; | ||||
| import { CliFunction, CliFunctionsRepository } from './cli-functions-repository'; | ||||
| import { Caching } from '../cloud-runner/remote-client/caching'; | ||||
| import { LfsHashing } from '../cloud-runner/services/utility/lfs-hashing'; | ||||
| import { RemoteClient } from '../cloud-runner/remote-client'; | ||||
| import CloudRunnerOptionsReader from '../cloud-runner/options/cloud-runner-options-reader'; | ||||
| import GitHub from '../github'; | ||||
| import { OptionValues } from 'commander'; | ||||
| import { InputKey } from '../input'; | ||||
| 
 | ||||
| export class CLI { | ||||
|   static async RunCli(options: any): Promise<void> { | ||||
|     Input.githubInputEnabled = false; | ||||
| 
 | ||||
|     const results = GetCliFunctions(options.mode); | ||||
| 
 | ||||
|     if (results === undefined || results.length === 0) { | ||||
|       throw new Error('no CLI mode found'); | ||||
| export class Cli { | ||||
|   public static options: OptionValues | undefined; | ||||
|   static get isCliMode() { | ||||
|     return Cli.options !== undefined && Cli.options.mode !== undefined && Cli.options.mode !== ''; | ||||
|   } | ||||
|   public static query(key: string, alternativeKey: string) { | ||||
|     if (Cli.options && Cli.options[key] !== undefined) { | ||||
|       return Cli.options[key]; | ||||
|     } | ||||
|     if (Cli.options && alternativeKey && Cli.options[alternativeKey] !== undefined) { | ||||
|       return Cli.options[alternativeKey]; | ||||
|     } | ||||
| 
 | ||||
|     CloudRunnerLogger.log(`Entrypoint: ${results.key}`); | ||||
| 
 | ||||
|     options.versioning = 'None'; | ||||
|     Input.cliOptions = options; | ||||
|     return await results.target[results.propertyKey](); | ||||
|   } | ||||
|   static isCliMode(options: any) { | ||||
|     return options.mode !== undefined && options.mode !== ''; | ||||
|     return; | ||||
|   } | ||||
| 
 | ||||
|   public static SetupCli() { | ||||
|   public static InitCliMode() { | ||||
|     CliFunctionsRepository.PushCliFunctionSource(RemoteClient); | ||||
|     CliFunctionsRepository.PushCliFunctionSource(Caching); | ||||
|     CliFunctionsRepository.PushCliFunctionSource(LfsHashing); | ||||
|     const program = new Command(); | ||||
|     program.version('0.0.1'); | ||||
|     const properties = Object.getOwnPropertyNames(Input); | ||||
|     core.info(`\n`); | ||||
|     core.info(`INPUT:`); | ||||
| 
 | ||||
|     const properties = CloudRunnerOptionsReader.GetProperties(); | ||||
|     const actionYamlReader: ActionYamlReader = new ActionYamlReader(); | ||||
|     for (const element of properties) { | ||||
|       program.option(`--${element} <${element}>`, actionYamlReader.GetActionYamlValue(element)); | ||||
|       if (Input[element] !== undefined && Input[element] !== '' && typeof Input[element] !== `function`) { | ||||
|         core.info(`${element} ${Input[element]}`); | ||||
|       } | ||||
|     } | ||||
|     core.info(`\n`); | ||||
|     program.option( | ||||
|       '-m, --mode <mode>', | ||||
|       GetAllCliModes() | ||||
|       CliFunctionsRepository.GetAllCliModes() | ||||
|         .map((x) => `${x.key} (${x.description})`) | ||||
|         .join(` | `), | ||||
|     ); | ||||
|     program.option('--populateOverride <populateOverride>', 'should use override query to pull input false by default'); | ||||
|     program.option('--cachePushFrom <cachePushFrom>', 'cache push from source folder'); | ||||
|     program.option('--cachePushTo <cachePushTo>', 'cache push to caching folder'); | ||||
|     program.option('--artifactName <artifactName>', 'caching artifact name'); | ||||
|     program.option('--select <select>', 'select a particular resource'); | ||||
|     program.option('--logFile <logFile>', 'output to log file (log stream only)'); | ||||
|     program.parse(process.argv); | ||||
|     Cli.options = program.opts(); | ||||
| 
 | ||||
|     return program.opts(); | ||||
|     return Cli.isCliMode; | ||||
|   } | ||||
| 
 | ||||
|   @CliFunction(`cli`, `runs a cloud runner build`) | ||||
|   static async RunCli(): Promise<void> { | ||||
|     GitHub.githubInputEnabled = false; | ||||
|     if (Cli.options!['populateOverride'] === `true`) { | ||||
|       await CloudRunnerQueryOverride.PopulateQueryOverrideInput(); | ||||
|     } | ||||
|     if (Cli.options!['logInput']) { | ||||
|       Cli.logInput(); | ||||
|     } | ||||
|     const results = CliFunctionsRepository.GetCliFunctions(Cli.options?.mode); | ||||
|     CloudRunnerLogger.log(`Entrypoint: ${results.key}`); | ||||
|     Cli.options!.versioning = 'None'; | ||||
| 
 | ||||
|     CloudRunner.buildParameters = await BuildParameters.create(); | ||||
|     CloudRunner.buildParameters.buildGuid = process.env.BUILD_GUID || ``; | ||||
|     CloudRunnerLogger.log(`Build Params:
 | ||||
|       ${JSON.stringify(CloudRunner.buildParameters, undefined, 4)} | ||||
|     `);
 | ||||
|     CloudRunner.lockedWorkspace = process.env.LOCKED_WORKSPACE || ``; | ||||
|     CloudRunnerLogger.log(`Locked Workspace: ${CloudRunner.lockedWorkspace}`); | ||||
|     await CloudRunner.setup(CloudRunner.buildParameters); | ||||
| 
 | ||||
|     return await results.target[results.propertyKey](Cli.options); | ||||
|   } | ||||
| 
 | ||||
|   @CliFunction(`print-input`, `prints all input`) | ||||
|   private static logInput() { | ||||
|     core.info(`\n`); | ||||
|     core.info(`INPUT:`); | ||||
|     const properties = CloudRunnerOptionsReader.GetProperties(); | ||||
|     for (const element of properties) { | ||||
|       if ( | ||||
|         element in Input && | ||||
|         Input[element as InputKey] !== undefined && | ||||
|         Input[element as InputKey] !== '' && | ||||
|         typeof Input[element as InputKey] !== `function` && | ||||
|         element !== 'length' && | ||||
|         element !== 'cliOptions' && | ||||
|         element !== 'prototype' | ||||
|       ) { | ||||
|         core.info(`${element} ${Input[element as InputKey]}`); | ||||
|       } | ||||
|     } | ||||
|     core.info(`\n`); | ||||
|   } | ||||
| 
 | ||||
|   @CliFunction(`cli-build`, `runs a cloud runner build`) | ||||
|   public static async CLIBuild(): Promise<string> { | ||||
|     const buildParameter = await BuildParameters.create(); | ||||
|     const baseImage = new ImageTag(buildParameter); | ||||
|     return await CloudRunner.run(buildParameter, baseImage.toString()); | ||||
| 
 | ||||
|     return (await CloudRunner.run(buildParameter, baseImage.toString())).BuildResults; | ||||
|   } | ||||
| 
 | ||||
|   @CliFunction(`remote-cli`, `sets up a repository, usually before a game-ci build`) | ||||
|   static async runRemoteClientJob() { | ||||
|     const buildParameter = JSON.parse(process.env.BUILD_PARAMETERS || '{}'); | ||||
|     RemoteClientLogger.log(`Build Params:
 | ||||
|       ${JSON.stringify(buildParameter, undefined, 4)} | ||||
|     `);
 | ||||
|     CloudRunnerState.setup(buildParameter); | ||||
|     await SetupCloudRunnerRepository.run(); | ||||
|   @CliFunction(`async-workflow`, `runs a cloud runner build`) | ||||
|   public static async asyncronousWorkflow(): Promise<string> { | ||||
|     const buildParameter = await BuildParameters.create(); | ||||
|     const baseImage = new ImageTag(buildParameter); | ||||
|     await CloudRunner.setup(buildParameter); | ||||
| 
 | ||||
|     return (await CloudRunner.run(buildParameter, baseImage.toString())).BuildResults; | ||||
|   } | ||||
| 
 | ||||
|   @CliFunction(`cach-push`, `push to cache`) | ||||
|   static async cachePush() {} | ||||
|   @CliFunction(`checks-update`, `runs a cloud runner build`) | ||||
|   public static async checksUpdate() { | ||||
|     const buildParameter = await BuildParameters.create(); | ||||
| 
 | ||||
|   @CliFunction(`cach-pull`, `pull from cache`) | ||||
|   static async cachePull() {} | ||||
|     await CloudRunner.setup(buildParameter); | ||||
|     const input = JSON.parse(process.env.CHECKS_UPDATE || ``); | ||||
|     core.info(`Checks Update ${process.env.CHECKS_UPDATE}`); | ||||
|     if (input.mode === `create`) { | ||||
|       throw new Error(`Not supported: only use update`); | ||||
|     } else if (input.mode === `update`) { | ||||
|       await GitHub.updateGitHubCheckRequest(input.data); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @CliFunction(`garbage-collect-aws`, `garbage collect aws`) | ||||
|   static async garbageCollectAws() { | ||||
|     process.env.AWS_REGION = Input.region; | ||||
|     const CF = new SDK.CloudFormation(); | ||||
|   @CliFunction(`garbage-collect`, `runs garbage collection`) | ||||
|   public static async GarbageCollect(): Promise<string> { | ||||
|     const buildParameter = await BuildParameters.create(); | ||||
| 
 | ||||
|     const stacks = await CF.listStacks().promise(); | ||||
|     CloudRunnerLogger.log(JSON.stringify(stacks, undefined, 4)); | ||||
|     await CloudRunner.setup(buildParameter); | ||||
| 
 | ||||
|     return await CloudRunner.Provider.garbageCollect(``, false, 0, false, false); | ||||
|   } | ||||
| 
 | ||||
|   @CliFunction(`list-resources`, `lists active resources`) | ||||
|   public static async ListResources(): Promise<string[]> { | ||||
|     const buildParameter = await BuildParameters.create(); | ||||
| 
 | ||||
|     await CloudRunner.setup(buildParameter); | ||||
|     const result = await CloudRunner.Provider.listResources(); | ||||
|     CloudRunnerLogger.log(JSON.stringify(result, undefined, 4)); | ||||
| 
 | ||||
|     return result.map((x) => x.Name); | ||||
|   } | ||||
| 
 | ||||
|   @CliFunction(`list-worfklow`, `lists running workflows`) | ||||
|   public static async ListWorfklow(): Promise<string[]> { | ||||
|     const buildParameter = await BuildParameters.create(); | ||||
| 
 | ||||
|     await CloudRunner.setup(buildParameter); | ||||
| 
 | ||||
|     return (await CloudRunner.Provider.listWorkflow()).map((x) => x.Name); | ||||
|   } | ||||
| 
 | ||||
|   @CliFunction(`watch`, `follows logs of a running workflow`) | ||||
|   public static async Watch(): Promise<string> { | ||||
|     const buildParameter = await BuildParameters.create(); | ||||
| 
 | ||||
|     await CloudRunner.setup(buildParameter); | ||||
| 
 | ||||
|     return await CloudRunner.Provider.watchWorkflow(); | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -1,117 +0,0 @@ | |||
| import { assert } from 'console'; | ||||
| import fs from 'fs'; | ||||
| import path from 'path'; | ||||
| import { Input } from '../../..'; | ||||
| import CloudRunnerLogger from '../../../cloud-runner/services/cloud-runner-logger'; | ||||
| import { CloudRunnerState } from '../../../cloud-runner/state/cloud-runner-state'; | ||||
| import { CloudRunnerSystem } from './cloud-runner-system'; | ||||
| import { LFSHashing } from './lfs-hashing'; | ||||
| import { RemoteClientLogger } from './remote-client-logger'; | ||||
| 
 | ||||
| export class Caching { | ||||
|   public static async PushToCache(cacheFolder: string, sourceFolder: string, cacheKey: string) { | ||||
|     const startPath = process.cwd(); | ||||
|     try { | ||||
|       if (!fs.existsSync(cacheFolder)) { | ||||
|         await CloudRunnerSystem.Run(`mkdir -p ${cacheFolder}`); | ||||
|       } | ||||
|       process.chdir(path.resolve(sourceFolder, '..')); | ||||
| 
 | ||||
|       if (Input.cloudRunnerTests) { | ||||
|         CloudRunnerLogger.log( | ||||
|           `Hashed cache folder ${await LFSHashing.hashAllFiles(sourceFolder)} ${sourceFolder} ${path.basename( | ||||
|             sourceFolder, | ||||
|           )}`,
 | ||||
|         ); | ||||
|       } | ||||
| 
 | ||||
|       if (Input.cloudRunnerTests) { | ||||
|         await CloudRunnerSystem.Run(`ls ${path.basename(sourceFolder)}`); | ||||
|       } | ||||
|       await CloudRunnerSystem.Run(`zip ${cacheKey}.zip ${path.basename(sourceFolder)}`); | ||||
|       assert(fs.existsSync(`${cacheKey}.zip`), 'cache zip exists'); | ||||
|       assert(fs.existsSync(path.basename(sourceFolder)), 'source folder exists'); | ||||
|       await CloudRunnerSystem.Run(`mv ${cacheKey}.zip ${cacheFolder}`); | ||||
|       RemoteClientLogger.log(`moved ${cacheKey}.zip to ${cacheFolder}`); | ||||
|       assert(fs.existsSync(`${path.join(cacheFolder, cacheKey)}.zip`), 'cache zip exists inside cache folder'); | ||||
| 
 | ||||
|       if (Input.cloudRunnerTests) { | ||||
|         await CloudRunnerSystem.Run(`ls ${cacheFolder}`); | ||||
|       } | ||||
|     } catch (error) { | ||||
|       process.chdir(`${startPath}`); | ||||
|       throw error; | ||||
|     } | ||||
|     process.chdir(`${startPath}`); | ||||
|   } | ||||
|   public static async PullFromCache(cacheFolder: string, destinationFolder: string, cacheKey: string = ``) { | ||||
|     const startPath = process.cwd(); | ||||
|     RemoteClientLogger.log(`Caching for ${path.basename(destinationFolder)}`); | ||||
|     try { | ||||
|       if (!fs.existsSync(cacheFolder)) { | ||||
|         fs.mkdirSync(cacheFolder); | ||||
|       } | ||||
| 
 | ||||
|       if (!fs.existsSync(destinationFolder)) { | ||||
|         fs.mkdirSync(destinationFolder); | ||||
|       } | ||||
| 
 | ||||
|       const latestInBranch = await (await CloudRunnerSystem.Run(`ls -t "${cacheFolder}" | grep .zip$ | head -1`)) | ||||
|         .replace(/\n/g, ``) | ||||
|         .replace('.zip', ''); | ||||
| 
 | ||||
|       process.chdir(cacheFolder); | ||||
| 
 | ||||
|       const cacheSelection = cacheKey !== `` && fs.existsSync(`${cacheKey}.zip`) ? cacheKey : latestInBranch; | ||||
|       await CloudRunnerLogger.log(`cache key ${cacheKey} selection ${cacheSelection}`); | ||||
| 
 | ||||
|       if (fs.existsSync(`${cacheSelection}.zip`)) { | ||||
|         const resultsFolder = `results${CloudRunnerState.buildParams.buildGuid}`; | ||||
|         await CloudRunnerSystem.Run(`mkdir -p ${resultsFolder}`); | ||||
|         if (Input.cloudRunnerTests) { | ||||
|           await CloudRunnerSystem.Run(`tree ${destinationFolder}`); | ||||
|         } | ||||
|         RemoteClientLogger.log(`cache item exists ${cacheFolder}/${cacheSelection}.zip`); | ||||
|         assert(`${fs.existsSync(destinationFolder)}`); | ||||
|         assert(`${fs.existsSync(`${cacheSelection}.zip`)}`); | ||||
|         const fullResultsFolder = path.join(cacheFolder, resultsFolder); | ||||
|         if (Input.cloudRunnerTests) { | ||||
|           await CloudRunnerSystem.Run(`tree ${cacheFolder}`); | ||||
|         } | ||||
|         await CloudRunnerSystem.Run(`unzip ${cacheSelection}.zip -d ${path.basename(resultsFolder)}`); | ||||
|         RemoteClientLogger.log(`cache item extracted to ${fullResultsFolder}`); | ||||
|         assert(`${fs.existsSync(fullResultsFolder)}`); | ||||
|         const destinationParentFolder = path.resolve(destinationFolder, '..'); | ||||
|         if (fs.existsSync(destinationFolder)) { | ||||
|           fs.rmSync(destinationFolder, { recursive: true, force: true }); | ||||
|         } | ||||
|         await CloudRunnerSystem.Run( | ||||
|           `mv "${fullResultsFolder}/${path.basename(destinationFolder)}" "${destinationParentFolder}"`, | ||||
|         ); | ||||
|         if (Input.cloudRunnerTests) { | ||||
|           await CloudRunnerSystem.Run(`tree ${destinationParentFolder}`); | ||||
|         } | ||||
|       } else { | ||||
|         RemoteClientLogger.logWarning(`cache item ${cacheKey} doesn't exist ${destinationFolder}`); | ||||
|         if (cacheSelection !== ``) { | ||||
|           if (Input.cloudRunnerTests) { | ||||
|             await CloudRunnerSystem.Run(`tree ${cacheFolder}`); | ||||
|           } | ||||
|           RemoteClientLogger.logWarning(`cache item ${cacheKey}.zip doesn't exist ${destinationFolder}`); | ||||
|           throw new Error(`Failed to get cache item, but cache hit was found: ${cacheSelection}`); | ||||
|         } | ||||
|       } | ||||
|     } catch (error) { | ||||
|       process.chdir(`${startPath}`); | ||||
|       throw error; | ||||
|     } | ||||
|     process.chdir(`${startPath}`); | ||||
|   } | ||||
| 
 | ||||
|   public static handleCachePurging() { | ||||
|     if (process.env.PURGE_REMOTE_BUILDER_CACHE !== undefined) { | ||||
|       RemoteClientLogger.log(`purging ${CloudRunnerState.purgeRemoteCaching}`); | ||||
|       fs.rmdirSync(CloudRunnerState.cacheFolder, { recursive: true }); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | @ -1,37 +0,0 @@ | |||
| import { exec } from 'child_process'; | ||||
| import { RemoteClientLogger } from './remote-client-logger'; | ||||
| 
 | ||||
| export class CloudRunnerSystem { | ||||
|   public static async Run(command: string, suppressError = false) { | ||||
|     for (const element of command.split(`\n`)) { | ||||
|       RemoteClientLogger.log(element); | ||||
|     } | ||||
|     return await new Promise<string>((promise) => { | ||||
|       let output = ''; | ||||
|       const child = exec(command, (error, stdout, stderr) => { | ||||
|         if (error && !suppressError) { | ||||
|           throw error; | ||||
|         } | ||||
|         if (stderr) { | ||||
|           const diagnosticOutput = `${stderr.toString()}`; | ||||
|           RemoteClientLogger.logCliDiagnostic(diagnosticOutput); | ||||
|           output += diagnosticOutput; | ||||
|           return; | ||||
|         } | ||||
|         const outputChunk = `${stdout}`; | ||||
|         output += outputChunk; | ||||
|       }); | ||||
|       child.on('close', function (code) { | ||||
|         RemoteClientLogger.log(`[Exit code ${code}]`); | ||||
|         if (code !== 0 && !suppressError) { | ||||
|           throw new Error(output); | ||||
|         } | ||||
|         const outputLines = output.split(`\n`); | ||||
|         for (const element of outputLines) { | ||||
|           RemoteClientLogger.log(element); | ||||
|         } | ||||
|         promise(output); | ||||
|       }); | ||||
|     }); | ||||
|   } | ||||
| } | ||||
|  | @ -1,42 +0,0 @@ | |||
| import path from 'path'; | ||||
| import { CloudRunnerState } from '../../../cloud-runner/state/cloud-runner-state'; | ||||
| import { CloudRunnerSystem } from './cloud-runner-system'; | ||||
| import fs from 'fs'; | ||||
| import { assert } from 'console'; | ||||
| import { Input } from '../../..'; | ||||
| import { RemoteClientLogger } from './remote-client-logger'; | ||||
| 
 | ||||
| export class LFSHashing { | ||||
|   public static async createLFSHashFiles() { | ||||
|     try { | ||||
|       await CloudRunnerSystem.Run(`git lfs ls-files -l | cut -d ' ' -f1 | sort > .lfs-assets-guid`); | ||||
|       await CloudRunnerSystem.Run(`md5sum .lfs-assets-guid > .lfs-assets-guid-sum`); | ||||
|       assert(fs.existsSync(`.lfs-assets-guid-sum`)); | ||||
|       assert(fs.existsSync(`.lfs-assets-guid`)); | ||||
|       const lfsHashes = { | ||||
|         lfsGuid: fs | ||||
|           .readFileSync(`${path.join(CloudRunnerState.repoPathFull, `.lfs-assets-guid`)}`, 'utf8') | ||||
|           .replace(/\n/g, ``), | ||||
|         lfsGuidSum: fs | ||||
|           .readFileSync(`${path.join(CloudRunnerState.repoPathFull, `.lfs-assets-guid-sum`)}`, 'utf8') | ||||
|           .replace(/\n/g, ``), | ||||
|       }; | ||||
|       if (Input.cloudRunnerTests) { | ||||
|         RemoteClientLogger.log(lfsHashes.lfsGuid); | ||||
|         RemoteClientLogger.log(lfsHashes.lfsGuidSum); | ||||
|       } | ||||
|       return lfsHashes; | ||||
|     } catch (error) { | ||||
|       throw error; | ||||
|     } | ||||
|   } | ||||
|   public static async hashAllFiles(folder: string) { | ||||
|     const startPath = process.cwd(); | ||||
|     process.chdir(folder); | ||||
|     const result = await (await CloudRunnerSystem.Run(`find -type f -exec md5sum "{}" + | sort | md5sum`)) | ||||
|       .replace(/\n/g, '') | ||||
|       .split(` `)[0]; | ||||
|     process.chdir(startPath); | ||||
|     return result; | ||||
|   } | ||||
| } | ||||
|  | @ -1,19 +0,0 @@ | |||
| import CloudRunnerLogger from '../../../cloud-runner/services/cloud-runner-logger'; | ||||
| 
 | ||||
| export class RemoteClientLogger { | ||||
|   public static log(message: string) { | ||||
|     CloudRunnerLogger.log(`[Client] ${message}`); | ||||
|   } | ||||
| 
 | ||||
|   public static logCliError(message: string) { | ||||
|     CloudRunnerLogger.log(`[Client][Error] ${message}`); | ||||
|   } | ||||
| 
 | ||||
|   public static logCliDiagnostic(message: string) { | ||||
|     CloudRunnerLogger.log(`[Client][Diagnostic] ${message}`); | ||||
|   } | ||||
| 
 | ||||
|   public static logWarning(message) { | ||||
|     CloudRunnerLogger.logWarning(message); | ||||
|   } | ||||
| } | ||||
|  | @ -1,81 +0,0 @@ | |||
| import fs from 'fs'; | ||||
| import { CloudRunnerState } from '../../cloud-runner/state/cloud-runner-state'; | ||||
| import { Caching } from './remote-client-services/caching'; | ||||
| import { LFSHashing } from './remote-client-services/lfs-hashing'; | ||||
| import { CloudRunnerSystem } from './remote-client-services/cloud-runner-system'; | ||||
| import { Input } from '../..'; | ||||
| import { RemoteClientLogger } from './remote-client-services/remote-client-logger'; | ||||
| import path from 'path'; | ||||
| import { assert } from 'console'; | ||||
| 
 | ||||
| export class SetupCloudRunnerRepository { | ||||
|   public static async run() { | ||||
|     try { | ||||
|       await CloudRunnerSystem.Run(`mkdir -p ${CloudRunnerState.buildPathFull}`); | ||||
|       await CloudRunnerSystem.Run(`mkdir -p ${CloudRunnerState.repoPathFull}`); | ||||
|       await CloudRunnerSystem.Run(`mkdir -p ${CloudRunnerState.cacheFolderFull}`); | ||||
| 
 | ||||
|       process.chdir(CloudRunnerState.repoPathFull); | ||||
|       if (Input.cloudRunnerTests) { | ||||
|         await CloudRunnerSystem.Run(`ls -lh`); | ||||
|         await CloudRunnerSystem.Run(`tree`); | ||||
|       } | ||||
|       await SetupCloudRunnerRepository.cloneRepoWithoutLFSFiles(); | ||||
|       if (Input.cloudRunnerTests) { | ||||
|         await CloudRunnerSystem.Run(`ls -lh`); | ||||
|         await CloudRunnerSystem.Run(`tree`); | ||||
|       } | ||||
|       const lfsHashes = await LFSHashing.createLFSHashFiles(); | ||||
|       if (fs.existsSync(CloudRunnerState.libraryFolderFull)) { | ||||
|         RemoteClientLogger.logWarning(`!Warning!: The Unity library was included in the git repository`); | ||||
|       } | ||||
|       await Caching.PullFromCache( | ||||
|         CloudRunnerState.lfsCacheFolderFull, | ||||
|         CloudRunnerState.lfsDirectoryFull, | ||||
|         `${lfsHashes.lfsGuid}`, | ||||
|       ); | ||||
|       await SetupCloudRunnerRepository.pullLatestLFS(); | ||||
|       await Caching.PushToCache( | ||||
|         CloudRunnerState.lfsCacheFolderFull, | ||||
|         CloudRunnerState.lfsDirectoryFull, | ||||
|         `${lfsHashes.lfsGuid}`, | ||||
|       ); | ||||
|       await Caching.PullFromCache(CloudRunnerState.libraryCacheFolderFull, CloudRunnerState.libraryFolderFull); | ||||
|       Caching.handleCachePurging(); | ||||
|     } catch (error) { | ||||
|       throw error; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   private static async cloneRepoWithoutLFSFiles() { | ||||
|     try { | ||||
|       process.chdir(`${CloudRunnerState.repoPathFull}`); | ||||
|       RemoteClientLogger.log(`Initializing source repository for cloning with caching of LFS files`); | ||||
|       await CloudRunnerSystem.Run(`git config --global advice.detachedHead false`); | ||||
|       RemoteClientLogger.log(`Cloning the repository being built:`); | ||||
|       await CloudRunnerSystem.Run(`git lfs install --skip-smudge`); | ||||
|       await CloudRunnerSystem.Run( | ||||
|         `git clone -b ${CloudRunnerState.branchName} ${CloudRunnerState.targetBuildRepoUrl} ${path.resolve( | ||||
|           `..`, | ||||
|           path.basename(CloudRunnerState.repoPathFull), | ||||
|         )}`,
 | ||||
|       ); | ||||
|       assert(fs.existsSync(`.git`)); | ||||
|       RemoteClientLogger.log(`${CloudRunnerState.buildParams.branch}`); | ||||
|       await CloudRunnerSystem.Run(`git checkout ${CloudRunnerState.buildParams.branch}`); | ||||
|       assert(fs.existsSync(path.join(`.git`, `lfs`)), 'LFS folder should not exist before caching'); | ||||
|       RemoteClientLogger.log(`Checked out ${process.env.GITHUB_SHA}`); | ||||
|     } catch (error) { | ||||
|       throw error; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   private static async pullLatestLFS() { | ||||
|     await CloudRunnerSystem.Run(`ls -lh ${CloudRunnerState.lfsDirectoryFull}/..`); | ||||
|     process.chdir(CloudRunnerState.repoPathFull); | ||||
|     await CloudRunnerSystem.Run(`git lfs pull`); | ||||
|     RemoteClientLogger.log(`pulled latest LFS files`); | ||||
|     assert(fs.existsSync(CloudRunnerState.lfsDirectoryFull)); | ||||
|     await CloudRunnerSystem.Run(`ls -lh ${CloudRunnerState.lfsDirectoryFull}/..`); | ||||
|   } | ||||
| } | ||||
|  | @ -1,16 +0,0 @@ | |||
| import CloudRunnerLogger from '../services/cloud-runner-logger'; | ||||
| import * as SDK from 'aws-sdk'; | ||||
| import * as core from '@actions/core'; | ||||
| import { Input } from '../..'; | ||||
| 
 | ||||
| export class AWSError { | ||||
|   static async handleStackCreationFailure(error: any, CF: SDK.CloudFormation, taskDefStackName: string) { | ||||
|     CloudRunnerLogger.log('aws error: '); | ||||
|     core.error(JSON.stringify(error, undefined, 4)); | ||||
|     if (Input.cloudRunnerTests) { | ||||
|       CloudRunnerLogger.log('Getting events and resources for task stack'); | ||||
|       const events = (await CF.describeStackEvents({ StackName: taskDefStackName }).promise()).StackEvents; | ||||
|       CloudRunnerLogger.log(JSON.stringify(events, undefined, 4)); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | @ -1,141 +0,0 @@ | |||
| import * as SDK from 'aws-sdk'; | ||||
| import CloudRunnerAWSTaskDef from './cloud-runner-aws-task-def'; | ||||
| import CloudRunnerSecret from '../services/cloud-runner-secret'; | ||||
| import { AWSTemplates } from './aws-templates'; | ||||
| import CloudRunnerLogger from '../services/cloud-runner-logger'; | ||||
| import { AWSError } from './aws-error'; | ||||
| 
 | ||||
| export class AWSJobStack { | ||||
|   private baseStackName: string; | ||||
|   constructor(baseStackName: string) { | ||||
|     this.baseStackName = baseStackName; | ||||
|   } | ||||
| 
 | ||||
|   public async setupCloudFormations( | ||||
|     CF: SDK.CloudFormation, | ||||
|     buildGuid: string, | ||||
|     image: string, | ||||
|     entrypoint: string[], | ||||
|     commands: string, | ||||
|     mountdir: string, | ||||
|     workingdir: string, | ||||
|     secrets: CloudRunnerSecret[], | ||||
|   ): Promise<CloudRunnerAWSTaskDef> { | ||||
|     const taskDefStackName = `${this.baseStackName}-${buildGuid}`; | ||||
|     let taskDefCloudFormation = AWSTemplates.readTaskCloudFormationTemplate(); | ||||
|     for (const secret of secrets) { | ||||
|       secret.ParameterKey = `${buildGuid.replace(/[^\dA-Za-z]/g, '')}${secret.ParameterKey.replace( | ||||
|         /[^\dA-Za-z]/g, | ||||
|         '', | ||||
|       )}`;
 | ||||
|       if (typeof secret.ParameterValue == 'number') { | ||||
|         secret.ParameterValue = `${secret.ParameterValue}`; | ||||
|       } | ||||
|       if (!secret.ParameterValue || secret.ParameterValue === '') { | ||||
|         secrets = secrets.filter((x) => x !== secret); | ||||
|         continue; | ||||
|       } | ||||
|       taskDefCloudFormation = AWSTemplates.insertAtTemplate( | ||||
|         taskDefCloudFormation, | ||||
|         'p1 - input', | ||||
|         AWSTemplates.getParameterTemplate(secret.ParameterKey), | ||||
|       ); | ||||
|       taskDefCloudFormation = AWSTemplates.insertAtTemplate( | ||||
|         taskDefCloudFormation, | ||||
|         'p2 - secret', | ||||
|         AWSTemplates.getSecretTemplate(`${secret.ParameterKey}`), | ||||
|       ); | ||||
|       taskDefCloudFormation = AWSTemplates.insertAtTemplate( | ||||
|         taskDefCloudFormation, | ||||
|         'p3 - container def', | ||||
|         AWSTemplates.getSecretDefinitionTemplate(secret.EnvironmentVariable, secret.ParameterKey), | ||||
|       ); | ||||
|     } | ||||
|     const secretsMappedToCloudFormationParameters = secrets.map((x) => { | ||||
|       return { ParameterKey: x.ParameterKey.replace(/[^\dA-Za-z]/g, ''), ParameterValue: x.ParameterValue }; | ||||
|     }); | ||||
|     const parameters = [ | ||||
|       { | ||||
|         ParameterKey: 'EnvironmentName', | ||||
|         ParameterValue: this.baseStackName, | ||||
|       }, | ||||
|       { | ||||
|         ParameterKey: 'ImageUrl', | ||||
|         ParameterValue: image, | ||||
|       }, | ||||
|       { | ||||
|         ParameterKey: 'ServiceName', | ||||
|         ParameterValue: taskDefStackName, | ||||
|       }, | ||||
|       { | ||||
|         ParameterKey: 'Command', | ||||
|         ParameterValue: 'echo "this template should be overwritten when running a task"', | ||||
|       }, | ||||
|       { | ||||
|         ParameterKey: 'EntryPoint', | ||||
|         ParameterValue: entrypoint.join(','), | ||||
|       }, | ||||
|       { | ||||
|         ParameterKey: 'WorkingDirectory', | ||||
|         ParameterValue: workingdir, | ||||
|       }, | ||||
|       { | ||||
|         ParameterKey: 'EFSMountDirectory', | ||||
|         ParameterValue: mountdir, | ||||
|       }, | ||||
|       ...secretsMappedToCloudFormationParameters, | ||||
|     ]; | ||||
| 
 | ||||
|     let previousStackExists = true; | ||||
|     while (previousStackExists) { | ||||
|       previousStackExists = false; | ||||
|       const stacks = await CF.listStacks().promise(); | ||||
|       if (!stacks.StackSummaries) { | ||||
|         throw new Error('Faild to get stacks'); | ||||
|       } | ||||
|       for (let index = 0; index < stacks.StackSummaries.length; index++) { | ||||
|         const element = stacks.StackSummaries[index]; | ||||
|         if (element.StackName === taskDefStackName && element.StackStatus !== 'DELETE_COMPLETE') { | ||||
|           previousStackExists = true; | ||||
|           CloudRunnerLogger.log(`Previous stack still exists: ${JSON.stringify(element)}`); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     try { | ||||
|       await CF.createStack({ | ||||
|         StackName: taskDefStackName, | ||||
|         TemplateBody: taskDefCloudFormation, | ||||
|         Capabilities: ['CAPABILITY_IAM'], | ||||
|         Parameters: parameters, | ||||
|       }).promise(); | ||||
|       CloudRunnerLogger.log('Creating cloud runner job'); | ||||
|       await CF.waitFor('stackCreateComplete', { StackName: taskDefStackName }).promise(); | ||||
|     } catch (error) { | ||||
|       await AWSError.handleStackCreationFailure( | ||||
|         error, | ||||
|         CF, | ||||
|         taskDefStackName, | ||||
|         //taskDefCloudFormation,
 | ||||
|         //parameters,
 | ||||
|         //secrets,
 | ||||
|       ); | ||||
|       throw error; | ||||
|     } | ||||
| 
 | ||||
|     const taskDefResources = ( | ||||
|       await CF.describeStackResources({ | ||||
|         StackName: taskDefStackName, | ||||
|       }).promise() | ||||
|     ).StackResources; | ||||
| 
 | ||||
|     const baseResources = (await CF.describeStackResources({ StackName: this.baseStackName }).promise()).StackResources; | ||||
| 
 | ||||
|     return { | ||||
|       taskDefStackName, | ||||
|       taskDefCloudFormation, | ||||
|       taskDefResources, | ||||
|       baseResources, | ||||
|     }; | ||||
|   } | ||||
| } | ||||
|  | @ -1,228 +0,0 @@ | |||
| import * as AWS from 'aws-sdk'; | ||||
| import CloudRunnerEnvironmentVariable from '../services/cloud-runner-environment-variable'; | ||||
| import * as core from '@actions/core'; | ||||
| import CloudRunnerAWSTaskDef from './cloud-runner-aws-task-def'; | ||||
| import * as zlib from 'zlib'; | ||||
| import CloudRunnerLogger from '../services/cloud-runner-logger'; | ||||
| import { Input } from '../..'; | ||||
| import { CloudRunnerState } from '../state/cloud-runner-state'; | ||||
| import { CloudRunnerStatics } from '../cloud-runner-statics'; | ||||
| import { CloudRunnerBuildCommandProcessor } from '../services/cloud-runner-build-command-process'; | ||||
| 
 | ||||
| class AWSTaskRunner { | ||||
|   static async runTask( | ||||
|     taskDef: CloudRunnerAWSTaskDef, | ||||
|     ECS: AWS.ECS, | ||||
|     CF: AWS.CloudFormation, | ||||
|     environment: CloudRunnerEnvironmentVariable[], | ||||
|     buildGuid: string, | ||||
|     commands: string, | ||||
|   ) { | ||||
|     const cluster = taskDef.baseResources?.find((x) => x.LogicalResourceId === 'ECSCluster')?.PhysicalResourceId || ''; | ||||
|     const taskDefinition = | ||||
|       taskDef.taskDefResources?.find((x) => x.LogicalResourceId === 'TaskDefinition')?.PhysicalResourceId || ''; | ||||
|     const SubnetOne = | ||||
|       taskDef.baseResources?.find((x) => x.LogicalResourceId === 'PublicSubnetOne')?.PhysicalResourceId || ''; | ||||
|     const SubnetTwo = | ||||
|       taskDef.baseResources?.find((x) => x.LogicalResourceId === 'PublicSubnetTwo')?.PhysicalResourceId || ''; | ||||
|     const ContainerSecurityGroup = | ||||
|       taskDef.baseResources?.find((x) => x.LogicalResourceId === 'ContainerSecurityGroup')?.PhysicalResourceId || ''; | ||||
|     const streamName = | ||||
|       taskDef.taskDefResources?.find((x) => x.LogicalResourceId === 'KinesisStream')?.PhysicalResourceId || ''; | ||||
| 
 | ||||
|     const task = await ECS.runTask({ | ||||
|       cluster, | ||||
|       taskDefinition, | ||||
|       platformVersion: '1.4.0', | ||||
|       overrides: { | ||||
|         containerOverrides: [ | ||||
|           { | ||||
|             name: taskDef.taskDefStackName, | ||||
|             environment, | ||||
|             command: ['-c', CloudRunnerBuildCommandProcessor.ProcessCommands(commands, CloudRunnerState.buildParams)], | ||||
|           }, | ||||
|         ], | ||||
|       }, | ||||
|       launchType: 'FARGATE', | ||||
|       networkConfiguration: { | ||||
|         awsvpcConfiguration: { | ||||
|           subnets: [SubnetOne, SubnetTwo], | ||||
|           assignPublicIp: 'ENABLED', | ||||
|           securityGroups: [ContainerSecurityGroup], | ||||
|         }, | ||||
|       }, | ||||
|     }).promise(); | ||||
| 
 | ||||
|     CloudRunnerLogger.log('Cloud runner job is starting'); | ||||
|     const taskArn = task.tasks?.[0].taskArn || ''; | ||||
| 
 | ||||
|     try { | ||||
|       await ECS.waitFor('tasksRunning', { tasks: [taskArn], cluster }).promise(); | ||||
|     } catch (error_) { | ||||
|       const error = error_ as Error; | ||||
|       await new Promise((resolve) => setTimeout(resolve, 3000)); | ||||
|       CloudRunnerLogger.log( | ||||
|         `Cloud runner job has ended ${ | ||||
|           (await AWSTaskRunner.describeTasks(ECS, cluster, taskArn)).containers?.[0].lastStatus | ||||
|         }`,
 | ||||
|       ); | ||||
| 
 | ||||
|       core.setFailed(error); | ||||
|       core.error(error); | ||||
|     } | ||||
|     CloudRunnerLogger.log(`Cloud runner job is running`); | ||||
| 
 | ||||
|     const output = await this.streamLogsUntilTaskStops(ECS, CF, taskDef, cluster, taskArn, streamName); | ||||
|     const exitCode = (await AWSTaskRunner.describeTasks(ECS, cluster, taskArn)).containers?.[0].exitCode; | ||||
|     CloudRunnerLogger.log(`Cloud runner job exit code ${exitCode}`); | ||||
|     if (exitCode !== 0 && exitCode !== undefined) { | ||||
|       core.error( | ||||
|         `job failed with exit code ${exitCode} ${JSON.stringify( | ||||
|           await ECS.describeTasks({ tasks: [taskArn], cluster }).promise(), | ||||
|           undefined, | ||||
|           4, | ||||
|         )}`,
 | ||||
|       ); | ||||
|       throw new Error(`job failed with exit code ${exitCode}`); | ||||
|     } else { | ||||
|       CloudRunnerLogger.log(`Cloud runner job has finished successfully`); | ||||
|       return output; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   static async describeTasks(ECS: AWS.ECS, clusterName: string, taskArn: string) { | ||||
|     const tasks = await ECS.describeTasks({ | ||||
|       cluster: clusterName, | ||||
|       tasks: [taskArn], | ||||
|     }).promise(); | ||||
|     if (tasks.tasks?.[0]) { | ||||
|       return tasks.tasks?.[0]; | ||||
|     } else { | ||||
|       throw new Error('No task found'); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   static async streamLogsUntilTaskStops( | ||||
|     ECS: AWS.ECS, | ||||
|     CF: AWS.CloudFormation, | ||||
|     taskDef: CloudRunnerAWSTaskDef, | ||||
|     clusterName: string, | ||||
|     taskArn: string, | ||||
|     kinesisStreamName: string, | ||||
|   ) { | ||||
|     const kinesis = new AWS.Kinesis(); | ||||
|     const stream = await AWSTaskRunner.getLogStream(kinesis, kinesisStreamName); | ||||
|     let iterator = await AWSTaskRunner.getLogIterator(kinesis, stream); | ||||
| 
 | ||||
|     CloudRunnerLogger.log( | ||||
|       `Cloud runner job status is ${(await AWSTaskRunner.describeTasks(ECS, clusterName, taskArn))?.lastStatus}`, | ||||
|     ); | ||||
| 
 | ||||
|     const logBaseUrl = `https://${Input.region}.console.aws.amazon.com/cloudwatch/home?region=${CF.config.region}#logsV2:log-groups/log-group/${taskDef.taskDefStackName}`; | ||||
|     CloudRunnerLogger.log(`You can also see the logs at AWS Cloud Watch: ${logBaseUrl}`); | ||||
|     let shouldReadLogs = true; | ||||
|     let timestamp: number = 0; | ||||
|     let output = ''; | ||||
|     while (shouldReadLogs) { | ||||
|       await new Promise((resolve) => setTimeout(resolve, 1500)); | ||||
|       const taskData = await AWSTaskRunner.describeTasks(ECS, clusterName, taskArn); | ||||
|       ({ timestamp, shouldReadLogs } = AWSTaskRunner.checkStreamingShouldContinue(taskData, timestamp, shouldReadLogs)); | ||||
|       ({ iterator, shouldReadLogs, output } = await AWSTaskRunner.handleLogStreamIteration( | ||||
|         kinesis, | ||||
|         iterator, | ||||
|         shouldReadLogs, | ||||
|         taskDef, | ||||
|         output, | ||||
|       )); | ||||
|     } | ||||
|     return output; | ||||
|   } | ||||
| 
 | ||||
|   private static async handleLogStreamIteration( | ||||
|     kinesis: AWS.Kinesis, | ||||
|     iterator: string, | ||||
|     shouldReadLogs: boolean, | ||||
|     taskDef: CloudRunnerAWSTaskDef, | ||||
|     output: string, | ||||
|   ) { | ||||
|     const records = await kinesis | ||||
|       .getRecords({ | ||||
|         ShardIterator: iterator, | ||||
|       }) | ||||
|       .promise(); | ||||
|     iterator = records.NextShardIterator || ''; | ||||
|     ({ shouldReadLogs, output } = AWSTaskRunner.logRecords(records, iterator, taskDef, shouldReadLogs, output)); | ||||
|     return { iterator, shouldReadLogs, output }; | ||||
|   } | ||||
| 
 | ||||
|   private static checkStreamingShouldContinue(taskData: AWS.ECS.Task, timestamp: number, shouldReadLogs: boolean) { | ||||
|     if (taskData?.lastStatus !== 'RUNNING') { | ||||
|       if (timestamp === 0) { | ||||
|         CloudRunnerLogger.log('## Cloud runner job stopped, streaming end of logs'); | ||||
|         timestamp = Date.now(); | ||||
|       } | ||||
|       if (timestamp !== 0 && Date.now() - timestamp > 30000) { | ||||
|         CloudRunnerLogger.log('## Cloud runner status is not RUNNING for 30 seconds, last query for logs'); | ||||
|         shouldReadLogs = false; | ||||
|       } | ||||
|       CloudRunnerLogger.log(`## Status of job: ${taskData.lastStatus}`); | ||||
|     } | ||||
|     return { timestamp, shouldReadLogs }; | ||||
|   } | ||||
| 
 | ||||
|   private static logRecords( | ||||
|     records, | ||||
|     iterator: string, | ||||
|     taskDef: CloudRunnerAWSTaskDef, | ||||
|     shouldReadLogs: boolean, | ||||
|     output: string, | ||||
|   ) { | ||||
|     if (records.Records.length > 0 && iterator) { | ||||
|       for (let index = 0; index < records.Records.length; index++) { | ||||
|         const json = JSON.parse( | ||||
|           zlib.gunzipSync(Buffer.from(records.Records[index].Data as string, 'base64')).toString('utf8'), | ||||
|         ); | ||||
|         if (json.messageType === 'DATA_MESSAGE') { | ||||
|           for (let logEventsIndex = 0; logEventsIndex < json.logEvents.length; logEventsIndex++) { | ||||
|             let message = json.logEvents[logEventsIndex].message; | ||||
|             if (json.logEvents[logEventsIndex].message.includes(`---${CloudRunnerState.buildParams.logId}`)) { | ||||
|               CloudRunnerLogger.log('End of log transmission received'); | ||||
|               shouldReadLogs = false; | ||||
|             } else if (message.includes('Rebuilding Library because the asset database could not be found!')) { | ||||
|               core.warning('LIBRARY NOT FOUND!'); | ||||
|             } | ||||
|             message = `[${CloudRunnerStatics.logPrefix}] ${message}`; | ||||
|             if (Input.cloudRunnerTests) { | ||||
|               output += message; | ||||
|             } | ||||
|             CloudRunnerLogger.log(message); | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     return { shouldReadLogs, output }; | ||||
|   } | ||||
| 
 | ||||
|   private static async getLogStream(kinesis: AWS.Kinesis, kinesisStreamName: string) { | ||||
|     return await kinesis | ||||
|       .describeStream({ | ||||
|         StreamName: kinesisStreamName, | ||||
|       }) | ||||
|       .promise(); | ||||
|   } | ||||
| 
 | ||||
|   private static async getLogIterator(kinesis: AWS.Kinesis, stream) { | ||||
|     return ( | ||||
|       ( | ||||
|         await kinesis | ||||
|           .getShardIterator({ | ||||
|             ShardIteratorType: 'TRIM_HORIZON', | ||||
|             StreamName: stream.StreamDescription.StreamName, | ||||
|             ShardId: stream.StreamDescription.Shards[0].ShardId, | ||||
|           }) | ||||
|           .promise() | ||||
|       ).ShardIterator || '' | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| export default AWSTaskRunner; | ||||
|  | @ -1,9 +0,0 @@ | |||
| import * as AWS from 'aws-sdk'; | ||||
| 
 | ||||
| class CloudRunnerAWSTaskDef { | ||||
|   public taskDefStackName!: string; | ||||
|   public taskDefCloudFormation!: string; | ||||
|   public taskDefResources: AWS.CloudFormation.StackResources | undefined; | ||||
|   public baseResources: AWS.CloudFormation.StackResources | undefined; | ||||
| } | ||||
| export default CloudRunnerAWSTaskDef; | ||||
|  | @ -1,98 +0,0 @@ | |||
| import * as SDK from 'aws-sdk'; | ||||
| import CloudRunnerSecret from '../services/cloud-runner-secret'; | ||||
| import CloudRunnerEnvironmentVariable from '../services/cloud-runner-environment-variable'; | ||||
| import CloudRunnerAWSTaskDef from './cloud-runner-aws-task-def'; | ||||
| import AWSTaskRunner from './aws-task-runner'; | ||||
| import { CloudRunnerProviderInterface } from '../services/cloud-runner-provider-interface'; | ||||
| import BuildParameters from '../../build-parameters'; | ||||
| import CloudRunnerLogger from '../services/cloud-runner-logger'; | ||||
| import { AWSJobStack } from './aws-job-stack'; | ||||
| import { AWSBaseStack } from './aws-base-stack'; | ||||
| import { Input } from '../..'; | ||||
| 
 | ||||
| class AWSBuildEnvironment implements CloudRunnerProviderInterface { | ||||
|   private baseStackName: string; | ||||
| 
 | ||||
|   constructor(buildParameters: BuildParameters) { | ||||
|     this.baseStackName = buildParameters.awsBaseStackName; | ||||
|   } | ||||
|   async cleanupSharedResources( | ||||
|     // eslint-disable-next-line no-unused-vars
 | ||||
|     buildGuid: string, | ||||
|     // eslint-disable-next-line no-unused-vars
 | ||||
|     buildParameters: BuildParameters, | ||||
|     // eslint-disable-next-line no-unused-vars
 | ||||
|     branchName: string, | ||||
|     // eslint-disable-next-line no-unused-vars
 | ||||
|     defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], | ||||
|   ) {} | ||||
|   async setupSharedResources( | ||||
|     // eslint-disable-next-line no-unused-vars
 | ||||
|     buildGuid: string, | ||||
|     // eslint-disable-next-line no-unused-vars
 | ||||
|     buildParameters: BuildParameters, | ||||
|     // eslint-disable-next-line no-unused-vars
 | ||||
|     branchName: string, | ||||
|     // eslint-disable-next-line no-unused-vars
 | ||||
|     defaultSecretsArray: { ParameterKey: string; EnvironmentVariable: string; ParameterValue: string }[], | ||||
|   ) {} | ||||
| 
 | ||||
|   async runTask( | ||||
|     buildGuid: string, | ||||
|     image: string, | ||||
|     commands: string, | ||||
|     mountdir: string, | ||||
|     workingdir: string, | ||||
|     environment: CloudRunnerEnvironmentVariable[], | ||||
|     secrets: CloudRunnerSecret[], | ||||
|   ): Promise<string> { | ||||
|     process.env.AWS_REGION = Input.region; | ||||
|     const ECS = new SDK.ECS(); | ||||
|     const CF = new SDK.CloudFormation(); | ||||
|     CloudRunnerLogger.log(`AWS Region: ${CF.config.region}`); | ||||
|     const entrypoint = ['/bin/sh']; | ||||
|     const startTimeMs = Date.now(); | ||||
| 
 | ||||
|     await new AWSBaseStack(this.baseStackName).setupBaseStack(CF); | ||||
|     const taskDef = await new AWSJobStack(this.baseStackName).setupCloudFormations( | ||||
|       CF, | ||||
|       buildGuid, | ||||
|       image, | ||||
|       entrypoint, | ||||
|       commands, | ||||
|       mountdir, | ||||
|       workingdir, | ||||
|       secrets, | ||||
|     ); | ||||
| 
 | ||||
|     let postRunTaskTimeMs; | ||||
|     let output = ''; | ||||
|     try { | ||||
|       const postSetupStacksTimeMs = Date.now(); | ||||
|       CloudRunnerLogger.log(`Setup job time: ${Math.floor((postSetupStacksTimeMs - startTimeMs) / 1000)}s`); | ||||
|       output = await AWSTaskRunner.runTask(taskDef, ECS, CF, environment, buildGuid, commands); | ||||
|       postRunTaskTimeMs = Date.now(); | ||||
|       CloudRunnerLogger.log(`Run job time: ${Math.floor((postRunTaskTimeMs - postSetupStacksTimeMs) / 1000)}s`); | ||||
|     } finally { | ||||
|       await this.cleanupResources(CF, taskDef); | ||||
|       const postCleanupTimeMs = Date.now(); | ||||
|       if (postRunTaskTimeMs !== undefined) | ||||
|         CloudRunnerLogger.log(`Cleanup job time: ${Math.floor((postCleanupTimeMs - postRunTaskTimeMs) / 1000)}s`); | ||||
|     } | ||||
|     return output; | ||||
|   } | ||||
| 
 | ||||
|   async cleanupResources(CF: SDK.CloudFormation, taskDef: CloudRunnerAWSTaskDef) { | ||||
|     CloudRunnerLogger.log('Cleanup starting'); | ||||
|     await CF.deleteStack({ | ||||
|       StackName: taskDef.taskDefStackName, | ||||
|     }).promise(); | ||||
| 
 | ||||
|     await CF.waitFor('stackDeleteComplete', { | ||||
|       StackName: taskDef.taskDefStackName, | ||||
|     }).promise(); | ||||
|     CloudRunnerLogger.log(`Deleted Stack: ${taskDef.taskDefStackName}`); | ||||
|     CloudRunnerLogger.log('Cleanup complete'); | ||||
|   } | ||||
| } | ||||
| export default AWSBuildEnvironment; | ||||
|  | @ -1,3 +0,0 @@ | |||
| export class CloudRunnerStatics { | ||||
|   public static readonly logPrefix = `Cloud-Runner-System`; | ||||
| } | ||||
|  | @ -1,50 +0,0 @@ | |||
| import { BuildParameters, ImageTag } from '..'; | ||||
| import CloudRunner from './cloud-runner'; | ||||
| import Input from '../input'; | ||||
| import { CloudRunnerStatics } from './cloud-runner-statics'; | ||||
| import { TaskParameterSerializer } from './services/task-parameter-serializer'; | ||||
| import UnityVersioning from '../unity-versioning'; | ||||
| 
 | ||||
| describe('Cloud Runner', () => { | ||||
|   it('responds', () => {}); | ||||
| }); | ||||
| describe('Cloud Runner', () => { | ||||
|   const testSecretName = 'testSecretName'; | ||||
|   const testSecretValue = 'testSecretValue'; | ||||
|   if (Input.cloudRunnerTests) { | ||||
|     it('All build parameters sent to cloud runner as env vars', async () => { | ||||
|       Input.cliOptions = { | ||||
|         versioning: 'None', | ||||
|         projectPath: 'test-project', | ||||
|         unityVersion: UnityVersioning.read('test-project'), | ||||
|         customJob: ` | ||||
|         - name: 'step 1' | ||||
|           image: 'alpine' | ||||
|           commands: 'printenv' | ||||
|           secrets: | ||||
|             - name: '${testSecretName}' | ||||
|               value: '${testSecretValue}' | ||||
|         `,
 | ||||
|       }; | ||||
|       Input.githubInputEnabled = false; | ||||
|       const buildParameter = await BuildParameters.create(); | ||||
|       const baseImage = new ImageTag(buildParameter); | ||||
|       const file = await CloudRunner.run(buildParameter, baseImage.toString()); | ||||
|       expect(file).toContain(JSON.stringify(buildParameter)); | ||||
|       expect(file).toContain(`${Input.ToEnvVarFormat(testSecretName)}=${testSecretValue}`); | ||||
|       const environmentVariables = TaskParameterSerializer.readBuildEnvironmentVariables(); | ||||
|       const newLinePurgedFile = file | ||||
|         .replace(/\s+/g, '') | ||||
|         .replace(new RegExp(`\\[${CloudRunnerStatics.logPrefix}\\]`, 'g'), ''); | ||||
|       for (const element of environmentVariables) { | ||||
|         if (element.value !== undefined && typeof element.value !== 'function') { | ||||
|           if (typeof element.value === `string`) { | ||||
|             element.value = element.value.replace(/\s+/g, ''); | ||||
|           } | ||||
|           expect(newLinePurgedFile).toContain(`${element.name}=${element.value}`); | ||||
|         } | ||||
|       } | ||||
|       Input.githubInputEnabled = true; | ||||
|     }, 1000000); | ||||
|   } | ||||
| }); | ||||
|  | @ -1,72 +1,187 @@ | |||
| import AWSBuildPlatform from './aws'; | ||||
| import { BuildParameters } from '..'; | ||||
| import { CloudRunnerState } from './state/cloud-runner-state'; | ||||
| import Kubernetes from './k8s'; | ||||
| import CloudRunnerLogger from './services/cloud-runner-logger'; | ||||
| import { CloudRunnerStepState } from './state/cloud-runner-step-state'; | ||||
| import AwsBuildPlatform from './providers/aws'; | ||||
| import { BuildParameters, Input } from '..'; | ||||
| import Kubernetes from './providers/k8s'; | ||||
| import CloudRunnerLogger from './services/core/cloud-runner-logger'; | ||||
| import { CloudRunnerStepParameters } from './options/cloud-runner-step-parameters'; | ||||
| import { WorkflowCompositionRoot } from './workflows/workflow-composition-root'; | ||||
| import { CloudRunnerError } from './error/cloud-runner-error'; | ||||
| import { TaskParameterSerializer } from './services/task-parameter-serializer'; | ||||
| import { TaskParameterSerializer } from './services/core/task-parameter-serializer'; | ||||
| import * as core from '@actions/core'; | ||||
| import CloudRunnerSecret from './options/cloud-runner-secret'; | ||||
| import { ProviderInterface } from './providers/provider-interface'; | ||||
| import CloudRunnerEnvironmentVariable from './options/cloud-runner-environment-variable'; | ||||
| import TestCloudRunner from './providers/test'; | ||||
| import LocalCloudRunner from './providers/local'; | ||||
| import LocalDockerCloudRunner from './providers/docker'; | ||||
| import GitHub from '../github'; | ||||
| import SharedWorkspaceLocking from './services/core/shared-workspace-locking'; | ||||
| import { FollowLogStreamService } from './services/core/follow-log-stream-service'; | ||||
| import CloudRunnerResult from './services/core/cloud-runner-result'; | ||||
| 
 | ||||
| class CloudRunner { | ||||
|   private static setup(buildParameters: BuildParameters) { | ||||
|   public static Provider: ProviderInterface; | ||||
|   public static buildParameters: BuildParameters; | ||||
|   private static defaultSecrets: CloudRunnerSecret[]; | ||||
|   private static cloudRunnerEnvironmentVariables: CloudRunnerEnvironmentVariable[]; | ||||
|   static lockedWorkspace: string = ``; | ||||
|   public static readonly retainedWorkspacePrefix: string = `retained-workspace`; | ||||
|   public static get isCloudRunnerEnvironment() { | ||||
|     return process.env[`GITHUB_ACTIONS`] !== `true`; | ||||
|   } | ||||
|   public static get isCloudRunnerAsyncEnvironment() { | ||||
|     return process.env[`ASYNC_WORKFLOW`] === `true`; | ||||
|   } | ||||
|   public static async setup(buildParameters: BuildParameters) { | ||||
|     CloudRunnerLogger.setup(); | ||||
|     CloudRunnerState.setup(buildParameters); | ||||
|     CloudRunner.setupBuildPlatform(); | ||||
|     const parameters = TaskParameterSerializer.readBuildEnvironmentVariables(); | ||||
|     for (const element of parameters) { | ||||
|       core.setOutput(element.name, element.value); | ||||
|     CloudRunnerLogger.log(`Setting up cloud runner`); | ||||
|     CloudRunner.buildParameters = buildParameters; | ||||
|     if (CloudRunner.buildParameters.githubCheckId === ``) { | ||||
|       CloudRunner.buildParameters.githubCheckId = await GitHub.createGitHubCheck(CloudRunner.buildParameters.buildGuid); | ||||
|     } | ||||
|     CloudRunner.setupSelectedBuildPlatform(); | ||||
|     CloudRunner.defaultSecrets = TaskParameterSerializer.readDefaultSecrets(); | ||||
|     CloudRunner.cloudRunnerEnvironmentVariables = | ||||
|       TaskParameterSerializer.createCloudRunnerEnvironmentVariables(buildParameters); | ||||
|     if (GitHub.githubInputEnabled) { | ||||
|       const buildParameterPropertyNames = Object.getOwnPropertyNames(buildParameters); | ||||
|       for (const element of CloudRunner.cloudRunnerEnvironmentVariables) { | ||||
|         // CloudRunnerLogger.log(`Cloud Runner output ${Input.ToEnvVarFormat(element.name)} = ${element.value}`);
 | ||||
|         core.setOutput(Input.ToEnvVarFormat(element.name), element.value); | ||||
|       } | ||||
|       for (const element of buildParameterPropertyNames) { | ||||
|         // CloudRunnerLogger.log(`Cloud Runner output ${Input.ToEnvVarFormat(element)} = ${buildParameters[element]}`);
 | ||||
|         core.setOutput(Input.ToEnvVarFormat(element), buildParameters[element]); | ||||
|       } | ||||
|       core.setOutput( | ||||
|         Input.ToEnvVarFormat(`buildArtifact`), | ||||
|         `build-${CloudRunner.buildParameters.buildGuid}.tar${ | ||||
|           CloudRunner.buildParameters.useCompressionStrategy ? '.lz4' : '' | ||||
|         }`,
 | ||||
|       ); | ||||
|     } | ||||
|     FollowLogStreamService.Reset(); | ||||
|   } | ||||
| 
 | ||||
|   private static setupBuildPlatform() { | ||||
|     switch (CloudRunnerState.buildParams.cloudRunnerCluster) { | ||||
|   private static setupSelectedBuildPlatform() { | ||||
|     CloudRunnerLogger.log(`Cloud Runner platform selected ${CloudRunner.buildParameters.providerStrategy}`); | ||||
|     switch (CloudRunner.buildParameters.providerStrategy) { | ||||
|       case 'k8s': | ||||
|         CloudRunnerLogger.log('Cloud Runner platform selected Kubernetes'); | ||||
|         CloudRunnerState.CloudRunnerProviderPlatform = new Kubernetes(CloudRunnerState.buildParams); | ||||
|         CloudRunner.Provider = new Kubernetes(CloudRunner.buildParameters); | ||||
|         break; | ||||
|       default: | ||||
|       case 'aws': | ||||
|         CloudRunnerLogger.log('Cloud Runner platform selected AWS'); | ||||
|         CloudRunnerState.CloudRunnerProviderPlatform = new AWSBuildPlatform(CloudRunnerState.buildParams); | ||||
|         CloudRunner.Provider = new AwsBuildPlatform(CloudRunner.buildParameters); | ||||
|         break; | ||||
|       case 'test': | ||||
|         CloudRunner.Provider = new TestCloudRunner(); | ||||
|         break; | ||||
|       case 'local-docker': | ||||
|         CloudRunner.Provider = new LocalDockerCloudRunner(); | ||||
|         break; | ||||
|       case 'local-system': | ||||
|         CloudRunner.Provider = new LocalCloudRunner(); | ||||
|         break; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   static async run(buildParameters: BuildParameters, baseImage: string) { | ||||
|     CloudRunner.setup(buildParameters); | ||||
|     if (baseImage.includes(`undefined`)) { | ||||
|       throw new Error(`baseImage is undefined`); | ||||
|     } | ||||
|     await CloudRunner.setup(buildParameters); | ||||
|     await CloudRunner.Provider.setupWorkflow( | ||||
|       CloudRunner.buildParameters.buildGuid, | ||||
|       CloudRunner.buildParameters, | ||||
|       CloudRunner.buildParameters.branch, | ||||
|       CloudRunner.defaultSecrets, | ||||
|     ); | ||||
|     try { | ||||
|       core.startGroup('Setup remote runner'); | ||||
|       await CloudRunnerState.CloudRunnerProviderPlatform.setupSharedResources( | ||||
|         CloudRunnerState.buildParams.buildGuid, | ||||
|         CloudRunnerState.buildParams, | ||||
|         CloudRunnerState.branchName, | ||||
|         CloudRunnerState.defaultSecrets, | ||||
|       ); | ||||
|       core.endGroup(); | ||||
|       if (buildParameters.maxRetainedWorkspaces > 0) { | ||||
|         CloudRunner.lockedWorkspace = SharedWorkspaceLocking.NewWorkspaceName(); | ||||
| 
 | ||||
|         const result = await SharedWorkspaceLocking.GetLockedWorkspace( | ||||
|           CloudRunner.lockedWorkspace, | ||||
|           CloudRunner.buildParameters.buildGuid, | ||||
|           CloudRunner.buildParameters, | ||||
|         ); | ||||
| 
 | ||||
|         if (result) { | ||||
|           CloudRunnerLogger.logLine(`Using retained workspace ${CloudRunner.lockedWorkspace}`); | ||||
|           CloudRunner.cloudRunnerEnvironmentVariables = [ | ||||
|             ...CloudRunner.cloudRunnerEnvironmentVariables, | ||||
|             { name: `LOCKED_WORKSPACE`, value: CloudRunner.lockedWorkspace }, | ||||
|           ]; | ||||
|         } else { | ||||
|           CloudRunnerLogger.log(`Max retained workspaces reached ${buildParameters.maxRetainedWorkspaces}`); | ||||
|           buildParameters.maxRetainedWorkspaces = 0; | ||||
|           CloudRunner.lockedWorkspace = ``; | ||||
|         } | ||||
|       } | ||||
|       await CloudRunner.updateStatusWithBuildParameters(); | ||||
|       const output = await new WorkflowCompositionRoot().run( | ||||
|         new CloudRunnerStepState( | ||||
|         new CloudRunnerStepParameters( | ||||
|           baseImage, | ||||
|           TaskParameterSerializer.readBuildEnvironmentVariables(), | ||||
|           CloudRunnerState.defaultSecrets, | ||||
|           CloudRunner.cloudRunnerEnvironmentVariables, | ||||
|           CloudRunner.defaultSecrets, | ||||
|         ), | ||||
|       ); | ||||
|       core.startGroup('Cleanup'); | ||||
|       await CloudRunnerState.CloudRunnerProviderPlatform.cleanupSharedResources( | ||||
|         CloudRunnerState.buildParams.buildGuid, | ||||
|         CloudRunnerState.buildParams, | ||||
|         CloudRunnerState.branchName, | ||||
|         CloudRunnerState.defaultSecrets, | ||||
|       await CloudRunner.Provider.cleanupWorkflow( | ||||
|         CloudRunner.buildParameters, | ||||
|         CloudRunner.buildParameters.branch, | ||||
|         CloudRunner.defaultSecrets, | ||||
|       ); | ||||
|       CloudRunnerLogger.log(`Cleanup complete`); | ||||
|       core.endGroup(); | ||||
|       return output; | ||||
|     } catch (error) { | ||||
|       core.endGroup(); | ||||
|       await CloudRunnerError.handleException(error); | ||||
|       if (!CloudRunner.buildParameters.isCliMode) core.endGroup(); | ||||
|       if (buildParameters.asyncWorkflow && this.isCloudRunnerEnvironment && this.isCloudRunnerAsyncEnvironment) { | ||||
|         await GitHub.updateGitHubCheck(CloudRunner.buildParameters.buildGuid, `success`, `success`, `completed`); | ||||
|       } | ||||
| 
 | ||||
|       if (BuildParameters.shouldUseRetainedWorkspaceMode(buildParameters)) { | ||||
|         const workspace = CloudRunner.lockedWorkspace || ``; | ||||
|         await SharedWorkspaceLocking.ReleaseWorkspace( | ||||
|           workspace, | ||||
|           CloudRunner.buildParameters.buildGuid, | ||||
|           CloudRunner.buildParameters, | ||||
|         ); | ||||
|         const isLocked = await SharedWorkspaceLocking.IsWorkspaceLocked(workspace, CloudRunner.buildParameters); | ||||
|         if (isLocked) { | ||||
|           throw new Error( | ||||
|             `still locked after releasing ${await SharedWorkspaceLocking.GetAllLocksForWorkspace( | ||||
|               workspace, | ||||
|               buildParameters, | ||||
|             )}`,
 | ||||
|           ); | ||||
|         } | ||||
|         CloudRunner.lockedWorkspace = ``; | ||||
|       } | ||||
| 
 | ||||
|       await GitHub.triggerWorkflowOnComplete(CloudRunner.buildParameters.finalHooks); | ||||
| 
 | ||||
|       if (buildParameters.constantGarbageCollection) { | ||||
|         CloudRunner.Provider.garbageCollect(``, true, buildParameters.garbageMaxAge, true, true); | ||||
|       } | ||||
| 
 | ||||
|       return new CloudRunnerResult(buildParameters, output, true, true, false); | ||||
|     } catch (error: any) { | ||||
|       CloudRunnerLogger.log(JSON.stringify(error, undefined, 4)); | ||||
|       await GitHub.updateGitHubCheck( | ||||
|         CloudRunner.buildParameters.buildGuid, | ||||
|         `Failed - Error ${error?.message || error}`, | ||||
|         `failure`, | ||||
|         `completed`, | ||||
|       ); | ||||
|       if (!CloudRunner.buildParameters.isCliMode) core.endGroup(); | ||||
|       await CloudRunnerError.handleException(error, CloudRunner.buildParameters, CloudRunner.defaultSecrets); | ||||
|       throw error; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   private static async updateStatusWithBuildParameters() { | ||||
|     const content = { ...CloudRunner.buildParameters }; | ||||
|     content.gitPrivateToken = ``; | ||||
|     content.unitySerial = ``; | ||||
|     content.unityEmail = ``; | ||||
|     content.unityPassword = ``; | ||||
|     const jsonContent = JSON.stringify(content, undefined, 4); | ||||
|     await GitHub.updateGitHubCheck(jsonContent, CloudRunner.buildParameters.buildGuid); | ||||
|   } | ||||
| } | ||||
| export default CloudRunner; | ||||
|  |  | |||
|  | @ -1,16 +1,15 @@ | |||
| import CloudRunnerLogger from '../services/cloud-runner-logger'; | ||||
| import { CloudRunnerState } from '../state/cloud-runner-state'; | ||||
| import CloudRunnerLogger from '../services/core/cloud-runner-logger'; | ||||
| import * as core from '@actions/core'; | ||||
| import CloudRunner from '../cloud-runner'; | ||||
| import CloudRunnerSecret from '../options/cloud-runner-secret'; | ||||
| import BuildParameters from '../../build-parameters'; | ||||
| 
 | ||||
| export class CloudRunnerError { | ||||
|   public static async handleException(error: unknown) { | ||||
|   public static async handleException(error: unknown, buildParameters: BuildParameters, secrets: CloudRunnerSecret[]) { | ||||
|     CloudRunnerLogger.error(JSON.stringify(error, undefined, 4)); | ||||
|     core.setFailed('Cloud Runner failed'); | ||||
|     await CloudRunnerState.CloudRunnerProviderPlatform.cleanupSharedResources( | ||||
|       CloudRunnerState.buildParams.buildGuid, | ||||
|       CloudRunnerState.buildParams, | ||||
|       CloudRunnerState.branchName, | ||||
|       CloudRunnerState.defaultSecrets, | ||||
|     ); | ||||
|     if (CloudRunner.Provider !== undefined) { | ||||
|       await CloudRunner.Provider.cleanupWorkflow(buildParameters, buildParameters.branch, secrets); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  |  | |||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
		Reference in New Issue