Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Retry-After support #579

Open
wants to merge 8 commits into
base: master
Choose a base branch
from
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions .travis.yml
Original file line number Diff line number Diff line change
@@ -14,6 +14,7 @@ matrix:
sudo: true

install:
- travis_retry pip install 'six>=1.12.0'
- travis_retry pip install tox

script:
90 changes: 89 additions & 1 deletion httpbin/core.py
Original file line number Diff line number Diff line change
@@ -89,7 +89,6 @@ def jsonify(*args, **kwargs):
app.add_template_global("HTTPBIN_TRACKING" in os.environ, name="tracking_enabled")

app.config["SWAGGER"] = {"title": "httpbin.org", "uiversion": 3}

template = {
"swagger": "2.0",
"info": {
@@ -135,13 +134,18 @@ def jsonify(*args, **kwargs):
{"name": "Cookies", "description": "Creates, reads and deletes Cookies"},
{"name": "Images", "description": "Returns different image formats"},
{"name": "Redirects", "description": "Returns different redirect responses"},
{
"name": "Rate limiting",
"description": "Test client-side rate limiting with 429 responses and optional Retry-After headers",
},
{
"name": "Anything",
"description": "Returns anything that is passed to request",
},
],
}


swagger_config = {
"headers": [],
"specs": [
@@ -777,6 +781,90 @@ def view_status_code(codes):
return status_code(code)


@app.route(
"/too-many-requests", methods=["GET", "POST", "PUT", "DELETE", "PATCH", "TRACE"]
)
def response_retry_after():
"""Return 429 status code with no Retry-After header
---
tags:
- Rate limiting
produces:
- text/plain
responses:
429:
description: Too Many Requests
"""
response = make_response()
response.status_code = 429

return response


@app.route(
"/retry-after/date/<seconds>", methods=["GET", "POST", "PUT", "DELETE", "PATCH", "TRACE"]
)
@app.route(
"/retry-after/<seconds>/date", methods=["GET", "POST", "PUT", "DELETE", "PATCH", "TRACE"]
)
def response_retry_after_date(seconds):
"""Return 429 status code with a Retry-After header per RFC 6585 in Date format
---
tags:
- Rate limiting
parameters:
- in: path
name: seconds
produces:
- text/plain
responses:
429:
description: Too Many Requests
"""
try:
retry_after = int(seconds)
except ValueError:
return Response("Invalid number of seconds", status=400)

response = make_response()
response.status_code = 429
response.headers['retry-after'] = http_date(time.time() + retry_after)

return response


@app.route(
"/retry-after/seconds/<seconds>", methods=["GET", "POST", "PUT", "DELETE", "PATCH", "TRACE"]
)
@app.route(
"/retry-after/<seconds>/seconds", methods=["GET", "POST", "PUT", "DELETE", "PATCH", "TRACE"]
)
def response_retry_after_seconds(seconds):
"""Return 429 status code with a Retry-After header per RFC 6585
---
tags:
- Rate limiting
parameters:
- in: path
name: seconds
produces:
- text/plain
responses:
429:
description: Too Many Requests
"""
try:
retry_after = int(seconds)
except ValueError:
return Response("Invalid number of seconds", status=400)

response = make_response()
response.status_code = 429
response.headers['retry-after'] = retry_after

return response


@app.route("/response-headers", methods=["GET", "POST"])
def response_headers():
"""Returns a set of response headers from the query string.
32 changes: 32 additions & 0 deletions test_httpbin.py
Original file line number Diff line number Diff line change
@@ -6,6 +6,7 @@
import contextlib
import six
import json
import time
from werkzeug.http import parse_dict_header
from hashlib import md5, sha256, sha512
from six import BytesIO
@@ -811,6 +812,37 @@ def test_parse_multi_value_header(self):
self.assertEqual(parse_multi_value_header('"xyzzy", "r2d2xxxx", "c3piozzzz"'), [ "xyzzy", "r2d2xxxx", "c3piozzzz" ])
self.assertEqual(parse_multi_value_header('W/"xyzzy", W/"r2d2xxxx", W/"c3piozzzz"'), [ "xyzzy", "r2d2xxxx", "c3piozzzz" ])
self.assertEqual(parse_multi_value_header('*'), [ "*" ])

def test_too_many_requests(self):
response = self.app.get('/too-many-requests')
self.assertEqual(response.status_code, 429)

def test_retry_after_seconds(self):
response = self.app.get('/retry-after/seconds/60')
self.assertEqual(response.status_code, 429)
self.assertEqual(response.headers.get('Retry-After'), '60')
response = self.app.get('/retry-after/60/seconds')
self.assertEqual(response.status_code, 429)
self.assertEqual(response.headers.get('Retry-After'), '60')

def test_retry_after_date(self):
response = self.app.get('/retry-after/date/60')
self.assertEqual(response.status_code, 429)
# difficult to test the actual response, but we can at least test that it is a parseable date
self.assertTrue(time.strptime(response.headers.get('Retry-After'), "%a, %d %b %Y %H:%M:%S GMT"))
response = self.app.get('/retry-after/60/date')
self.assertEqual(response.status_code, 429)
# difficult to test the actual response, but we can at least test that it is a parseable date
self.assertTrue(time.strptime(response.headers.get('Retry-After'), "%a, %d %b %Y %H:%M:%S GMT"))

def test_invalid_retry_after_seconds(self):
response = self.app.get('/retry-after/seconds/a')
self.assertEqual(response.status_code, 400)

def test_invalid_retry_after_date(self):
response = self.app.get('/retry-after/date/a')
self.assertEqual(response.status_code, 400)


if __name__ == '__main__':
unittest.main()
3 changes: 3 additions & 0 deletions tox.ini
Original file line number Diff line number Diff line change
@@ -2,6 +2,9 @@
envlist = py27,py36,py37

[testenv]
deps =
werkzeug==0.14.1
flask==1.0.2
commands=python test_httpbin.py

[testenv:release]