Skip to content

Commit

Permalink
Preliminary pagination support API
Browse files Browse the repository at this point in the history
  • Loading branch information
yakky committed Dec 30, 2017
1 parent 956de29 commit c893208
Show file tree
Hide file tree
Showing 4 changed files with 51 additions and 12 deletions.
21 changes: 21 additions & 0 deletions docs/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -169,6 +169,27 @@ You can also specify filters
tasks = api.tasks.list(project=1)
By defaults list returns all objects, eventually getting the
paginated results behind the scenes.

If you want to disable pagination support and get all the
objects at once use the `all` argument

.. code:: python
tasks = api.tasks.list(all=True)
.. warning:: be aware that if the unpaginated results may exceed
the data the parser can handle and may result in an error.

You can also explicitly control the pagination by passing `lazy_pagination`
argument and the page number

.. code:: python
tasks_page_1 = api.tasks.list(lazy_pagination) # Will only return page 1
tasks_page_2 = api.tasks.list(lazy_pagination, page=2) # Will return page 2
Attach a file
~~~~~~~~~~~~~

Expand Down
18 changes: 15 additions & 3 deletions taiga/models/base.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
from copy import copy

import dateutil.parser
import re
import six
Expand Down Expand Up @@ -34,11 +36,21 @@ def __init__(self, requester):

class ListResource(Resource):

def list(self, **queryparams):
def list(self, lazy_pagination=False, all=True, **queryparams):
result = self.requester.get(
self.instance.endpoint, query=queryparams
self.instance.endpoint, query=queryparams, paginate=all
)
objects = self.parse_list(result.json())
objects = SearchableList()
objects.extend(self.parse_list(result.json()))
page = 2 if result.headers.get('X-Pagination-Next', False) else None
while page and not lazy_pagination:
pageparams = queryparams.copy()
pageparams['page'] = page
result = self.requester.get(
self.instance.endpoint, query=pageparams,
)
objects.extend(self.parse_list(result.json()))
page = page + 1 if result.headers.get('X-Pagination-Next', False) else None
return objects

def get(self, resource_id):
Expand Down
20 changes: 13 additions & 7 deletions taiga/requestmaker.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
from requests.packages.urllib3.exceptions import InsecureRequestWarning


def _disable_pagination():
def _requests_compatible_true():
if LooseVersion(requests.__version__) >= LooseVersion('2.11.0'):
return 'True'
else:
Expand Down Expand Up @@ -61,12 +61,15 @@ def __init__(self,
api_path, host,
token,
token_type='Bearer',
tls_verify=True):
tls_verify=True,
enable_pagination=True
):
self.api_path = api_path
self.host = host
self.token = token
self.token_type = token_type
self.tls_verify = tls_verify
self.enable_pagination = enable_pagination
self._cache = RequestCache()
if not self.tls_verify:
requests.packages.urllib3.disable_warnings(InsecureRequestWarning)
Expand All @@ -78,12 +81,15 @@ def cache(self):
def is_bad_response(self, response):
return 400 <= response.status_code <= 500

def headers(self):
def headers(self, paginate=True):
headers = {
'Content-type': 'application/json',
'Authorization': '{0} {1}'.format(self.token_type, self.token),
'x-disable-pagination': _disable_pagination()
}
if self.enable_pagination and paginate:
headers['x-lazy-pagination'] = _requests_compatible_true()
else:
headers['x-disable-pagination'] = _requests_compatible_true()
return headers

def urljoin(self, *parts):
Expand All @@ -96,7 +102,7 @@ def get_full_url(self, uri, query={}, **parameters):
)
return full_url

def get(self, uri, query={}, cache=False, **parameters):
def get(self, uri, query={}, cache=False, paginate=True, **parameters):
try:
full_url = self.urljoin(
self.host, self.api_path,
Expand All @@ -114,7 +120,7 @@ def get(self, uri, query={}, cache=False, **parameters):
if not result:
result = requests.get(
full_url,
headers=self.headers(),
headers=self.headers(paginate),
params=query,
verify=self.tls_verify
)
Expand All @@ -137,7 +143,7 @@ def post(self, uri, payload=None, query={}, files={}, **parameters):
if files:
headers = {
'Authorization': '{0} {1}'.format(self.token_type, self.token),
'x-disable-pagination': _disable_pagination()
'x-disable-pagination': _requests_compatible_true()
}
data = payload
else:
Expand Down
4 changes: 2 additions & 2 deletions tests/test_requestmaker.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
from taiga.requestmaker import RequestMaker, _disable_pagination
from taiga.requestmaker import RequestMaker, _requests_compatible_true
import taiga.exceptions
import requests
import unittest
Expand Down Expand Up @@ -33,7 +33,7 @@ def test_call_requests_post_with_files(self, requests_post):
data=None, params={},
headers={
'Authorization': 'Bearer f4k3',
'x-disable-pagination': _disable_pagination()
'x-disable-pagination': _requests_compatible_true()
}
)

Expand Down

0 comments on commit c893208

Please sign in to comment.