Github Script Repos and Collaborators List Sync


在 2018 年時為了公用帳號有一個 Github repostiories & collaborators 列表的需求,還特別在 GAE 上用 Golang 寫了一個服務並啟用 cron job
cron.yaml
|
|
每一個小時用 Golang 呼叫 Github Graphql 取回來所有 Repo 列表,最後透過 Spreadsheet API 寫到特定的 Spreadsheet 中
- 需要開一個 Google Platform Platfrom 專案
- 部署 GAE 應用程式
- Golang 呼叫 Github Graphql 取資料
- Spreadsheet API 回寫資料
一這波操作下來是沒花到什麼錢啦,就是麻煩了一些,本來這個小東西跑的也好好的,最近最近人員的異動導至需要重新 review 大家的權限,Github 在 2019/11/3 正式推出了 Github Action,自己滿多 Repo 也早就在使用了,所以想說上述的需撾用 Github Action 重新實作會更來的容易些,事實上真的很方更,每一次 Action 的執行時間都不到 30 秒,就是快
在重構的時候也發現了許多非常棒的工具,也因此特別寫一篇文章來記錄一下
GitHub GraphQL API: Github Graphql Playground
grpchql nektos/act: Run your GitHub Actions locally 🚀: 不得不提這一個工具,可以在本地端開發測試 Github Action,本地就可以開發就是方便
(pic https://github.com/nektos/act)act demo actions/github-script: Write workflows scripting the GitHub API in JavaScript: 在寫 Github Action workflows scripts 時可以直接用 Javascript 呼到 Github API,包好了直接用方便
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
jobs: readme: runs-on: ubuntu-latest steps: - run: npm install tablemark - uses: actions/github-script@v4 id: repository_collaborators with: github-token: ${{ secrets.REPO_TOKEN }} script: | const query = `query($endCursor: String = null) { viewer { repositories(first: 100, after: $endCursor, affiliations: [OWNER], orderBy: {field: CREATED_AT, direction: DESC}) { totalCount pageInfo { endCursor hasNextPage } nodes { name createdAt url } } } }`; const request = { owner: context.repo.owner, repo: context.repo.repo } const {data:collaboratorsData} = (await github.repos.getContent({ ...request, path: 'collaborators.txt' }));
github.repos.getContent
Github API 直接封裝起來用一波qeek-dev/github-project-sync: sync github projects & collaborators: Octokit plugin adding one method for all of api.github.com REST API endpoints
這個網站提供了 API endpoints 的文件可以查詢
EX:
1 2 3 4 5
octokit.rest.repos.getContent({ owner, repo, path, });
.github/workflows/cron.yml
name: sync repository collaborators | |
on: | |
push: | |
branches: | |
- master | |
paths-ignore: | |
- '**.md' | |
schedule: # Run workflow automatically | |
- cron: '0 * * * *' # Runs every hour, on the hour | |
workflow_dispatch: # Run workflow manually (without waiting for the cron to be called), through the Github Actions Workflow page directly | |
jobs: | |
readme: | |
runs-on: ubuntu-latest | |
steps: | |
- run: npm install tablemark | |
- uses: actions/github-script@v4 | |
id: repository_collaborators | |
with: | |
github-token: ${{ secrets.REPO_TOKEN }} | |
script: | | |
const query = `query($endCursor: String = null) { | |
viewer { | |
repositories(first: 100, after: $endCursor, affiliations: [OWNER], orderBy: {field: CREATED_AT, direction: DESC}) { | |
totalCount | |
pageInfo { | |
endCursor | |
hasNextPage | |
} | |
nodes { | |
name | |
isPrivate | |
isFork | |
createdAt | |
collaborators { | |
edges { | |
permission | |
} | |
nodes { | |
login | |
} | |
} | |
url | |
} | |
} | |
} | |
}`; | |
const request = { | |
owner: context.repo.owner, | |
repo: context.repo.repo | |
} | |
let res = [] | |
let endCursor = null | |
let hasNextPage = false | |
do { | |
let { viewer: { repositories: { pageInfo, nodes } } } = await github.graphql(query, {endCursor}) | |
res = res.concat(nodes) | |
endCursor = pageInfo.endCursor | |
hasNextPage = pageInfo.hasNextPage | |
} while (hasNextPage) | |
const content = res.map((repo, i)=>{ | |
let collaborators = [] | |
for( let i = 0; i < repo.collaborators.edges.length; i++ ){ | |
if (repo.collaborators.nodes[i].login != context.repo.owner){ | |
collaborators.push(`[@${repo.collaborators.nodes[i].login}](https://github.com/${repo.collaborators.nodes[i].login}):${repo.collaborators.edges[i].permission}`) | |
} | |
} | |
return { | |
'-': i+1, | |
name: `[${repo.name}](${repo.url})`, | |
isPrivate: repo.isPrivate ? `✅` : `❌`, | |
isFork: repo.isFork ? `✅` : `❌`, | |
createdAt: repo.createdAt, | |
collaborators: collaborators.join('<br>'), | |
} | |
}) | |
const tablemark = require('tablemark') | |
let md = tablemark(content) | |
const result = `# Auto Sync github projects & collaborators | |
${md} | |
` | |
const {data:readmeData} = (await github.repos.getContent({ | |
...request, | |
path: 'README.md' | |
})); | |
if (Buffer.from(readmeData.content, 'base64').compare(Buffer.from(result)) === 0) return | |
await github.repos.createOrUpdateFileContents({ | |
...request, | |
path: 'README.md', | |
message: `Updated with the latest github repository & collaborators`, | |
content: Buffer.from(result).toString('base64'), | |
sha: readmeData.sha | |
}) |
基本的流程如下
- Github Graphql 取回所有的 repos & collaborators, 需要
github-token: ${{ secrets.REPO_TOKEN }}
- 準備 md 所需的資料
- 使用
tablemark
library 將 Array 轉換為 markdown if (Buffer.from(data.content, 'base64').compare(Buffer.from(result)) === 0) return
,比對新舊README.md
檔案是否一致,不一樣才需要更新Buffer.from(data.content, 'base64').toString()
,如果需要取出github.repos.getContent
中的值需要進行轉 base64 轉換
心得
如果專案放在 Github 上,Github action 真的可以作滿多事情的。GitHub Marketplace 也有非常多的資源可以用,不知道怎麼寫的話可以去援尋原始碼,真的推薦可以使用