anchor.yml 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261
  1. name: Anchor
  2. on:
  3. schedule:
  4. - cron: "0 0 * * *"
  5. push:
  6. branches:
  7. - main
  8. pull_request:
  9. types: [opened, synchronize, reopened]
  10. branches:
  11. - main
  12. env:
  13. MAX_JOBS: 64
  14. MIN_PROJECTS_PER_JOB: 4
  15. MIN_PROJECTS_FOR_MATRIX: 4
  16. jobs:
  17. changes:
  18. runs-on: ubuntu-latest
  19. permissions:
  20. pull-requests: read
  21. outputs:
  22. changed_projects: ${{ steps.analyze.outputs.changed_projects }}
  23. total_projects: ${{ steps.analyze.outputs.total_projects }}
  24. matrix: ${{ steps.matrix.outputs.matrix }}
  25. steps:
  26. - uses: actions/checkout@v4
  27. - uses: dorny/paths-filter@v3
  28. id: changes
  29. if: github.event_name == 'pull_request'
  30. with:
  31. list-files: shell
  32. filters: |
  33. anchor:
  34. - added|modified: '**/anchor/**'
  35. workflow:
  36. - added|modified: '.github/workflows/anchor.yml'
  37. - name: Analyze Changes
  38. id: analyze
  39. run: |
  40. # Generate ignore pattern, excluding comments
  41. ignore_pattern=$(grep -v '^#' .github/.ghaignore | grep -v '^$' | tr '\n' '|' | sed 's/|$//')
  42. echo "Ignore pattern: $ignore_pattern"
  43. function get_projects() {
  44. find . -type d -name "anchor" | grep -vE "$ignore_pattern" | sort
  45. }
  46. # Determine which projects to build and test
  47. if [[ "${{ github.event_name }}" == "push" || "${{ github.event_name }}" == "schedule" || "${{ steps.changes.outputs.workflow }}" == "true" ]]; then
  48. projects=$(get_projects)
  49. elif [[ "${{ steps.changes.outputs.anchor }}" == "true" ]]; then
  50. changed_files=(${{ steps.changes.outputs.anchor_files }})
  51. projects=$(for file in "${changed_files[@]}"; do dirname "${file}" | grep anchor | sed 's#/anchor/.*#/anchor#g'; done | grep -vE "$ignore_pattern" | sort -u)
  52. else
  53. projects=""
  54. fi
  55. # Output project information
  56. if [[ -n "$projects" ]]; then
  57. echo "Projects to build and test"
  58. echo "$projects"
  59. total_projects=$(echo "$projects" | wc -l)
  60. echo "Total projects: $total_projects"
  61. echo "total_projects=$total_projects" >> $GITHUB_OUTPUT
  62. echo "changed_projects=$(echo "$projects" | jq -R -s -c 'split("\n")[:-1]')" >> $GITHUB_OUTPUT
  63. else
  64. echo "No projects to build and test."
  65. echo "total_projects=0" >> $GITHUB_OUTPUT
  66. echo "changed_projects=[]" >> $GITHUB_OUTPUT
  67. fi
  68. - name: Generate matrix
  69. id: matrix
  70. run: |
  71. total_projects=${{ steps.analyze.outputs.total_projects }}
  72. max_jobs=${{ env.MAX_JOBS }}
  73. min_projects_per_job=${{ env.MIN_PROJECTS_PER_JOB }}
  74. min_projects_for_matrix=${{ env.MIN_PROJECTS_FOR_MATRIX }}
  75. # Generate matrix based on number of projects
  76. if [ "$total_projects" -lt "$min_projects_for_matrix" ]; then
  77. echo "matrix=[0]" >> $GITHUB_OUTPUT
  78. else
  79. projects_per_job=$(( (total_projects + max_jobs - 1) / max_jobs ))
  80. projects_per_job=$(( projects_per_job > min_projects_per_job ? projects_per_job : min_projects_per_job ))
  81. num_jobs=$(( (total_projects + projects_per_job - 1) / projects_per_job ))
  82. indices=$(seq 0 $(( num_jobs - 1 )))
  83. echo "matrix=[$(echo $indices | tr ' ' ',')]" >> $GITHUB_OUTPUT
  84. fi
  85. build-and-test:
  86. needs: changes
  87. if: needs.changes.outputs.total_projects != '0'
  88. runs-on: ubuntu-latest
  89. strategy:
  90. fail-fast: false
  91. matrix:
  92. index: ${{ fromJson(needs.changes.outputs.matrix) }}
  93. name: build-and-test-group-${{ matrix.index }}
  94. outputs:
  95. failed_projects: ${{ steps.set-failed.outputs.failed_projects }}
  96. steps:
  97. - uses: actions/checkout@v4
  98. - uses: heyAyushh/setup-anchor@v4.9
  99. with:
  100. anchor-version: 0.30.1
  101. solana-cli-version: stable
  102. node-version: 20.x
  103. use-avm: false
  104. - name: Display Versions and Install pnpm
  105. run: |
  106. cargo-build-sbf --version
  107. solana -V
  108. solana-keygen new --no-bip39-passphrase
  109. rustc -V
  110. anchor -V
  111. npm i -g pnpm
  112. - name: Build and Test
  113. env:
  114. TOTAL_PROJECTS: ${{ needs.changes.outputs.total_projects }}
  115. PROJECTS_PER_JOB: ${{ env.MIN_PROJECTS_PER_JOB }}
  116. run: |
  117. function build_and_test() {
  118. local project=$1
  119. echo "Building and Testing $project"
  120. cd "$project" || return 1
  121. # Check Cargo.lock version before build
  122. if [ -f "Cargo.lock" ]; then
  123. echo "Checking Cargo.lock version for $project"
  124. grep "^version = " Cargo.lock | head -n1
  125. else
  126. echo "No Cargo.lock found for $project"
  127. fi
  128. # Run anchor build
  129. if ! anchor build; then
  130. echo "::error::anchor build failed for $project"
  131. # Check Cargo.lock version after failed build
  132. if [ -f "Cargo.lock" ]; then
  133. echo "Cargo.lock version after failed build:"
  134. grep "^version = " Cargo.lock | head -n1
  135. fi
  136. echo "$project: anchor build failed" >> $GITHUB_WORKSPACE/failed_projects.txt
  137. rm -rf target
  138. cd - > /dev/null
  139. return 1
  140. fi
  141. # Install dependencies
  142. if ! pnpm install --frozen-lockfile; then
  143. echo "::error::pnpm install failed for $project"
  144. echo "$project: pnpm install failed" >> $GITHUB_WORKSPACE/failed_projects.txt
  145. cd - > /dev/null
  146. return 1
  147. fi
  148. # Run anchor test
  149. if ! anchor test; then
  150. echo "::error::anchor test failed for $project"
  151. echo "$project: anchor test failed" >> $GITHUB_WORKSPACE/failed_projects.txt
  152. rm -rf target node_modules
  153. cd - > /dev/null
  154. return 1
  155. fi
  156. echo "Build and tests succeeded for $project."
  157. rm -rf target node_modules
  158. cd - > /dev/null
  159. return 0
  160. }
  161. # Determine which projects to build in this job
  162. readarray -t all_projects < <(echo '${{ needs.changes.outputs.changed_projects }}' | jq -r '.[]?')
  163. start_index=$(( ${{ matrix.index }} * PROJECTS_PER_JOB ))
  164. end_index=$(( start_index + PROJECTS_PER_JOB ))
  165. end_index=$(( end_index > TOTAL_PROJECTS ? TOTAL_PROJECTS : end_index ))
  166. echo "Projects to build and test in this job"
  167. for i in $(seq $start_index $(( end_index - 1 ))); do
  168. echo "${all_projects[$i]}"
  169. done
  170. # Build and test projects
  171. failed=false
  172. failed_projects=()
  173. for i in $(seq $start_index $(( end_index - 1 ))); do
  174. echo "::group::Building and testing ${all_projects[$i]}"
  175. if ! build_and_test "${all_projects[$i]}"; then
  176. failed=true
  177. failed_projects+=("${all_projects[$i]}")
  178. fi
  179. echo "::endgroup::"
  180. done
  181. if [[ "$failed" == "true" ]]; then
  182. echo "::group::Failed projects"
  183. cat $GITHUB_WORKSPACE/failed_projects.txt
  184. echo "::endgroup::"
  185. echo "failed_projects=${failed_projects[@]}" >> $GITHUB_OUTPUT
  186. exit 1
  187. else
  188. echo "failed_projects=" >> $GITHUB_OUTPUT
  189. fi
  190. - name: Set failed projects output
  191. id: set-failed
  192. if: failure()
  193. run: |
  194. # Prepare failed projects list for output
  195. failed_projects=$(cat $GITHUB_WORKSPACE/failed_projects.txt | jq -R -s -c 'split("\n")[:-1]')
  196. echo "failed_projects=$failed_projects" >> $GITHUB_OUTPUT
  197. summary:
  198. needs: [changes, build-and-test]
  199. if: always()
  200. runs-on: ubuntu-latest
  201. steps:
  202. - uses: actions/checkout@v4
  203. - name: Create job summary
  204. run: |
  205. echo "## Anchor Workflow Summary" >> $GITHUB_STEP_SUMMARY
  206. echo "- Total projects: ${{ needs.changes.outputs.total_projects }}" >> $GITHUB_STEP_SUMMARY
  207. # List all processed projects
  208. echo "<details>" >> $GITHUB_STEP_SUMMARY
  209. echo "<summary>Projects processed (click to expand)</summary>" >> $GITHUB_STEP_SUMMARY
  210. echo "" >> $GITHUB_STEP_SUMMARY
  211. echo '${{ needs.changes.outputs.changed_projects }}' | jq -r '.[]' | while read project; do
  212. echo "- $project" >> $GITHUB_STEP_SUMMARY
  213. done
  214. echo "" >> $GITHUB_STEP_SUMMARY
  215. echo "</details>" >> $GITHUB_STEP_SUMMARY
  216. # Report build and test results
  217. if [[ "${{ needs.build-and-test.result }}" == "failure" ]]; then
  218. echo "## :x: Build or tests failed" >> $GITHUB_STEP_SUMMARY
  219. echo "<details>" >> $GITHUB_STEP_SUMMARY
  220. echo "<summary>Failed projects (click to expand)</summary>" >> $GITHUB_STEP_SUMMARY
  221. echo "" >> $GITHUB_STEP_SUMMARY
  222. failed_projects='${{ needs.build-and-test.outputs.failed_projects }}'
  223. if [[ -n "$failed_projects" ]]; then
  224. echo "$failed_projects" | jq -r '.[]' | while IFS=: read -r project failure_reason; do
  225. echo "- **$project**" >> $GITHUB_STEP_SUMMARY
  226. echo " - Failure reason: $failure_reason" >> $GITHUB_STEP_SUMMARY
  227. done
  228. else
  229. echo "No failed projects reported. This might indicate an unexpected error in the workflow." >> $GITHUB_STEP_SUMMARY
  230. fi
  231. echo "" >> $GITHUB_STEP_SUMMARY
  232. echo "</details>" >> $GITHUB_STEP_SUMMARY
  233. elif [[ "${{ needs.build-and-test.result }}" == "success" ]]; then
  234. echo "## :white_check_mark: All builds and tests passed" >> $GITHUB_STEP_SUMMARY
  235. else
  236. echo "## :warning: Build and test job was skipped or canceled" >> $GITHUB_STEP_SUMMARY
  237. fi