原文連接:https://ssshooter.com/2019-01...javascript
本文爲你們提供一種使用 GitHub API 更新 GitHub 倉庫的方法。一般咱們會使用 Git 客戶端 Commit 而後 Push 到 GitHub,但 GitHub 爲咱們提供了相關 API,在某些狀況下能夠直接經過 API 更新倉庫。html
在此以前有一些前置知識點,你須要瞭解 git 的儲存原理。碰巧最近掘金有一篇很火的文章有提到相關知識。java
使用 GitHub API 更新倉庫只需六步。雖然對於第一次接觸的新手來講可能會有點複雜,可是理清關係以後思路便會很清晰了。node
注意:訪問 POST 方法的接口必須帶 token,雖然 GET 能夠不帶,可是會限制訪問頻率,因此建議都帶上。相關文檔:https://developer.github.com/v3/#authenticationios
https://developer.github.com/v3/git/refs/#get-a-referencegit
GET https://api.github.com/repos/ssshooter/test/git/refs/heads/master
github
{ "ref": "refs/heads/master", "node_id": "MDM6UmVmMTY0Nzk4NDczOm1hc3Rlcg==", "url": "https://api.github.com/repos/ssshooter/test/git/refs/heads/master", "object": { "sha": "cda66de943082033f4b761639df77728d7bca4f0", "type": "commit", "url": "https://api.github.com/repos/ssshooter/test/git/commits/cda66de943082033f4b761639df77728d7bca4f0" } }
https://developer.github.com/v3/git/commits/#get-a-commitaxios
GET https://api.github.com/repos/ssshooter/test/git/commits/cda66de943082033f4b761639df77728d7bca4f0
api
{ "sha": "cda66de943082033f4b761639df77728d7bca4f0", "node_id": "MDY6Q29tbWl0MTY0Nzk4NDczOmNkYTY2ZGU5NDMwODIwMzNmNGI3NjE2MzlkZjc3NzI4ZDdiY2E0ZjA=", "url": "https://api.github.com/repos/ssshooter/test/git/commits/cda66de943082033f4b761639df77728d7bca4f0", "html_url": "https://github.com/ssshooter/test/commit/cda66de943082033f4b761639df77728d7bca4f0", "author": { "name": "DeJavuJo", "email": "ssshooterx@gmail.com", "date": "2019-01-09T06:02:07Z" }, "committer": { "name": "GitHub", "email": "noreply@github.com", "date": "2019-01-09T06:02:07Z" }, "tree": { "sha": "d7a57d28ace0db12773c7d70675f94a48da6421f", "url": "https://api.github.com/repos/ssshooter/test/git/trees/d7a57d28ace0db12773c7d70675f94a48da6421f" }, "message": "Create readme.md", "parents": [ ], "verification": { "verified": true, "reason": "valid", "signature": "-----BEGIN PGP SIGNATURE-----\n\nwsBcBAABCAAQBQJcNY5fCRBK7hj4Ov3rIwAAdHIIABh8I89hATqg1mSYtpx1aY\nJt/woDobMO7FGE5qXO0NNrCMqwF6mdPJOMMMZvVWF1ULTm8ZJ52GAh5xAnPMZEnQ\n8tTjj/Qc0eZCm5B1xO66rgx+7eKkeGUPJj5bX7Lf0i9Se70Ff0jaCdH94RgiSL2d\nclDLhNyM4u6AH//k7S3Ud1O6ezXd4+99381Xa331PDLhJttUAbRFRCThszbcxRUT\nPvyhPNvXB5ug4J+tAMaZJvZEaED0c1k2Yx1+TYkLvPjOlqXvqaDVpMZieluD5l+f\nu13NAoCnLfBe0Skak/8MxRDbMcJYNW78ll60HAa6jJtElD8Vkek8WoVAZ8p3iIE=\n=xI87\n-----END PGP SIGNATURE-----\n", "payload": "tree d7a57d28ace0db12773c7d70675f94a48da6421f\nauthor DeJavuJo 1547013727 +0800\ncommitter GitHub 1547013727 +0800\n\nCreate readme.md" } }
https://developer.github.com/v3/git/blobs/#create-a-blobssh
POST https://api.github.com/repos/ssshooter/test/git/blobs
{ "content": "Content of the blob", "encoding": "utf-8" }
{ "sha": "929246f65aab4d636cb229c790f966afc332c124", "url": "https://api.github.com/repos/ssshooter/test/git/blobs/929246f65aab4d636cb229c790f966afc332c124" }
https://developer.github.com/v3/git/trees/#create-a-tree
POST https://api.github.com/repos/ssshooter/test/git/trees
注意:tree.path 能夠寫深層目錄,如 deep/deep/newFile.md(前面不用寫斜槓)
{ "base_tree": "d7a57d28ace0db12773c7d70675f94a48da6421f", // commit tree 的 sha "tree": [ { "path": "apiCommitFile.md", // 文件路徑 "mode": "100644", // 類型,詳情看文檔 "type": "blob", "sha": "929246f65aab4d636cb229c790f966afc332c124" // 剛纔生成的 blob 的 sha } ] }
{ "sha": "2b3a65095ceb7cf4425f52d59ca5974d826cff80", "url": "https://api.github.com/repos/ssshooter/test/git/trees/2b3a65095ceb7cf4425f52d59ca5974d826cff80", "tree": [ { "path": "readme.md", "mode": "100644", "type": "blob", "sha": "b1b716105590454bfc4c0247f193a04088f39c7f", "size": 5, "url": "https://api.github.com/repos/ssshooter/test/git/blobs/b1b716105590454bfc4c0247f193a04088f39c7f" }, { "path": "tree", "mode": "040000", "type": "tree", "sha": "91d8d59147e395effaeacd01c9d8553b37cf77c5", "url": "https://api.github.com/repos/ssshooter/test/git/trees/91d8d59147e395effaeacd01c9d8553b37cf77c5" } ], "truncated": false }
https://developer.github.com/v3/git/commits/#create-a-commit
POST https://api.github.com/repos/ssshooter/test/git/commits
{ "message": "a Commit with GitHub api", "parents": [ "cda66de943082033f4b761639df77728d7bca4f0" // 上次 commit 的sha ], "tree": "2b3a65095ceb7cf4425f52d59ca5974d826cff80" }
{ "sha": "45c58a9358b67fc81e4034cb36c5196d791686ef", "node_id": "MDY6Q29tbWl0MTY0Nzk4NDczOjQ1YzU4YTkzNThiNjdmYzgxZTQwMzRjYjM2YzUxOTZkNzkxNjg2ZWY=", "url": "https://api.github.com/repos/ssshooter/test/git/commits/45c58a9358b67fc81e4034cb36c5196d791686ef", "html_url": "https://github.com/ssshooter/test/commit/45c58a9358b67fc81e4034cb36c5196d791686ef", "author": { "name": "ssshooter", "email": "ssshooterx@gmail.com", "date": "2019-01-09T10:24:10Z" }, "committer": { "name": "ssshooter", "email": "ssshooterx@gmail.com", "date": "2019-01-09T10:24:10Z" }, "tree": { "sha": "2b3a65095ceb7cf4425f52d59ca5974d826cff80", "url": "https://api.github.com/repos/ssshooter/test/git/trees/2b3a65095ceb7cf4425f52d59ca5974d826cff80" }, "message": "a Commit with GitHub api", "parents": [ { "sha": "cda66de943082033f4b761639df77728d7bca4f0", "url": "https://api.github.com/repos/ssshooter/test/git/commits/cda66de943082033f4b761639df77728d7bca4f0", "html_url": "https://github.com/ssshooter/test/commit/cda66de943082033f4b761639df77728d7bca4f0" } ], "verification": { "verified": false, "reason": "unsigned", "signature": null, "payload": null } }
https://developer.github.com/v3/git/refs/#update-a-reference
https://api.github.com/repos/ssshooter/test/git/refs/heads/master
{ "sha":"45c58a9358b67fc81e4034cb36c5196d791686ef", "force":true }
{ "ref": "refs/heads/master", "node_id": "MDM6UmVmMTY0Nzk4NDczOm1hc3Rlcg==", "url": "https://api.github.com/repos/ssshooter/test/git/refs/heads/master", "object": { "sha": "45c58a9358b67fc81e4034cb36c5196d791686ef", "type": "commit", "url": "https://api.github.com/repos/ssshooter/test/git/commits/45c58a9358b67fc81e4034cb36c5196d791686ef" } }
var updateGitHubRes = function(blob, path) { var commitSha var commitTreeSha return getRef() .then(({ data }) => { commitSha = data.object.sha return getCommit(commitSha) }) .then(({ data }) => { commitTreeSha = data.tree.sha return createBlob(blob) }) .then(({ data }) => { var blobSha = data.sha return createTree(commitTreeSha, path, blobSha) }) .then(({ data }) => { var treeSha = data.sha return createCommit(commitSha, treeSha) }) .then(({ data }) => { var newCommitSha = data.sha return updataRef(newCommitSha) }) .catch(err => { console.log(err) }) } var getRef = function() { return axios.get(`/${owner}/${repo}/git/refs/heads/master`) } var getCommit = function(commitSha) { return axios.get(`/${owner}/${repo}/git/commits/${commitSha}`) } var createBlob = function(content) { return axios.post(`/${owner}/${repo}/git/blobs`, { content, encoding: 'utf-8' }) } var createTree = function(base_tree, path, sha) { return axios.post(`/${owner}/${repo}/git/trees`, { base_tree, // commit tree 的 sha tree: [ { path, // 文件路徑 mode: '100644', // 類型,詳情看文檔 type: 'blob', sha // 剛纔生成的 blob 的 sha } ] }) } var createCommit = function(parentCommitSha, tree, message = 'update post') { return axios.post(`/${owner}/${repo}/git/commits`, { message, parents: [parentCommitSha],// 上次 commit 的sha tree }) } var updataRef = function(newCommitSha) { return axios.post(`/${owner}/${repo}/git/refs/heads/master`, { sha: newCommitSha, force: true }) }
https://int128.hatenablog.com/entry/2017/09/05/161641
http://www.javashuo.com/article/p-xixzbpdd-y.html
https://git-scm.com/book/zh/v2/Git-內部原理-Git-引用