REST API 规范 - GitHub 实现

背景

本文写作时间:2017 年 3 月 30 日;
GitHub 版本:V3;

一直在用前后端分离的方式开发,接口的规范一直在努力使用 REST API 规范,但遇到稍微复杂一点的接口就不知道怎么定义了,例如分页、要查询的字段、筛选条件和排序条件等。

目标

  • 根据 GitHub 接口系统学习 REST API 规范;
  • 解决当前遇到的问题;
  • 要查询的字段、筛选、排序条件;
  • 分页;
  • PATCH 和 PUT 的具体区别;
  • 创建成功和失败时的响应内容;
  • 登录和鉴权;

GitHub 接口

因为 GitHub 接口功能非常丰富,为了重点学习自己需要的,按照对于当前的重要程度来区别不同的接口功能。

概要

协议:HTTPS
根端点(Root Endpoint):https://api.github.com
格式:默认 JSON
字段:见如下示例,如果不存在用 null 而不是忽略掉

示例:

curl -i https://api.github.com/users/octocat/orgs

HTTP/1.1 200 OK
Server: nginx
Date: Fri, 12 Oct 2012 23:33:14 GMT
Content-Type: application/json; charset=utf-8
Connection: keep-alive
Status: 200 OK
ETag: "a00049ba79152d03380c34652f2cb612"
X-GitHub-Media-Type: github.v3
X-RateLimit-Limit: 5000
X-RateLimit-Remaining: 4987
X-RateLimit-Reset: 1350085394
Content-Length: 5
Cache-Control: max-age=0, private, must-revalidate
X-Content-Type-Options: nosniff

支持的 HTTP 动词:

  • HEAD
  • GET
  • POST
  • PATCH
  • PUT
  • DELETE

重要

认证

三种认证方式

基础认证
curl -u "username" https://api.github.com
OAuth2 认证(头部发送)
curl -H "Authorization: token OAUTH-TOKEN" https://api.github.com
OAuth2 认证(作为参数发送)
curl https://api.github.com/?access_token=OAUTH-TOKEN

登录失败

无效的认证信息回返回 401 错误:

curl -i https://api.github.com -u foo:bar
HTTP/1.1 401 Unauthorized
{
  "message": "Bad credentials",
  "documentation_url": "https://developer.github.com/v3"
}

在短期内一直错误的话,会返回 403 错误:

curl -i https://api.github.com -u valid_username:valid_password
HTTP/1.1 403 Forbidden
{
  "message": "Maximum number of login attempts exceeded. Please try again later.",
  "documentation_url": "https://developer.github.com/v3"
}

分页

API 自动对请求的元素分页,不同的 API 有不同的默认值,可以指定查询的最大长度,但对某些资源不起作用。

请求

  • 通过 per_page 设置每页的元素数量;
  • 通过 page 设置要查询的页码;
  • 如果两项都不设置,系统根据默认的 per_pagepage 值返回;

响应

测试的 API 端点:https://api.github.com/search/code?q=addClass+user:mozilla&page=14

响应如下:

HTTP/1.1 200 OK
Server: GitHub.com
Date: Thu, 30 Mar 2017 05:47:30 GMT
Content-Type: application/json; charset=utf-8
Content-Length: 161418
Status: 200 OK
X-RateLimit-Limit: 10
X-RateLimit-Remaining: 9
X-RateLimit-Reset: 1490852910
Cache-Control: no-cache
X-GitHub-Media-Type: github.v3; format=json
Link: <https://api.github.com/search/code?q=addClass+user%3Amozilla&page=15>; rel="next", <https://api.github.com/search/code?q=addClass+user%3Amozilla&page=34>; rel="last", <https://api.github.com/search/code?q=addClass+user%3Amozilla&page=1>; rel="first", <https://api.github.com/search/code?q=addClass+user%3Amozilla&page=13>; rel="prev"
Access-Control-Expose-Headers: ETag, Link, X-GitHub-OTP, X-RateLimit-Limit, X-RateLimit-Remaining, X-RateLimit-Reset, X-OAuth-Scopes, X-Accepted-OAuth-Scopes, X-Poll-Interval
Access-Control-Allow-Origin: *
Content-Security-Policy: default-src 'none'
Strict-Transport-Security: max-age=31536000; includeSubdomains; preload
X-Content-Type-Options: nosniff
X-Frame-Options: deny
X-XSS-Protection: 1; mode=block
Vary: Accept-Encoding
X-Served-By: 7b641bda7ec2ca7cd9df72d2578baf75
X-GitHub-Request-Id: CB87:1B3BB:E967A5:128F029:58DC9BF1

{
  "total_count": 2525,
  "incomplete_results": false,
  "items": [
    ...
  ]
}

响应头 Link 里面包含了相关页面的链接。响应体里面 total_count 表示总数量,items 是数组,里面是要查询的元素。

客户端错误

无效 JSON 会导致 400 错误

HTTP/1.1 400 Bad Request
Content-Length: 35

{"message":"Problems parsing JSON"}

错误的 JSON 格式会导致 400 错误

HTTP/1.1 400 Bad Request
Content-Length: 40

{"message":"Body should be a JSON object"}

无效的字段会导致 422 错误

HTTP/1.1 422 Unprocessable Entity
Content-Length: 149

{
  "message": "Validation Failed",
  "errors": [
    {
      "resource": "Issue",
      "field": "title",
      "code": "missing_field" // missing, missing_field, invalid, already_exists
    }
  ]
}

参考资料

一般

不重要

获取 API 支持的端点

目标:查询 API 支持的端点;

curl https://api.github.com

API 版本号

目标:允许客户调用指定版本的接口

Accept: application/vnd.github.v3+json

媒体类型

目标:可以根据 Accept 请求头返回对应的媒体类型;

HTTP 跳转

目标:合适的时候会使用 HTTP 跳转,状态码是 301/302/307;

条件查询

大部分响应回返回一个 ETag 头。很多响应也会返回一个 Last-Modified 头。你可以使用这些头的值来对那些资源使用 If-None-MatchIf-Modified-Since 做子查询。如果资源没有修改,返回 304。

参考资料

参考资料

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,646评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,969评论 25 707
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,796评论 6 342
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,608评论 18 399
  • 今天主要是結合昨晚bootstrap與Rails實作的合併,今天在做一此整個流程的整理, 從頭開始, 1.創一個n...
    小慷阅读 130评论 0 0