Giter Site home page Giter Site logo

camsaul / django-rest-params Goto Github PK

View Code? Open in Web Editor NEW
18.0 4.0 10.0 28 KB

Function decorator for Django REST Framework for specifying and constraining API parameters

License: BSD 3-Clause "New" or "Revised" License

Python 99.14% Makefile 0.86%

django-rest-params's Introduction

Django REST Params

Function decorator to specify and validate parameters for API calls. Invalid params will automatically return a useful error message; validated params are passed to the wrapped function as kwargs.

pip install django-rest-params==1.0.2

A Few Examples

Specify the types of parameters, and check that they are within acceptable ranges:

from django_rest_params.decorators import params

@api_view(['GET'])
@params(latitude=float, latitude__gte=-90.0, latitude__lte=90.0,
        longitude=float, longitude__gte=-180.0, longitude__lte=180.0)
def get_something(request, latitude, longitude):
    # If latitude/longitude are missing or out of range, user will get an error message.
    # If we get here, we know they're valid
    pass

Create optional params with default values. Django REST Params supports POST params as well:

@api_view(['POST'])
@params(offset=int, offset__default=0)
def paged_api_call(request, offset):
    # if offset isn't specified, default value is used
    pass

Django REST Params works with ViewSets as well as request functions.

class ShirtsViewSet(viewsets.GenericViewSet):

    @params(colors=('red','blue','green','yellow'), colors__many=True,
            colors__optional=True, colors__name='color_filter')
    def get_shirts(self, request, colors):
        # Handle API calls like these:
        # /shirts?color_filter=red          __name lets you use a function param name different from the API param name
        # /shirts?color_filter=yellow,blue  __many lets you pass multiple values
        # /shirts                           __optional will set colors to None if it isn't specified
        # /shirts?color_filter=black        ERROR! This will return an error stating black is invalid, and listing the valid options
        pass

Options

TYPE

Specify the type of a param:

latitude=float
valid options are:
  • int
  • float
  • bool (1/true are considered True, and 0/false False; this is not case-sensitive)
  • str/unicode
  • tuple/list/set/frozenset (which will be treated as a list of valid options)
  • a django Model subclass (in which case the param will be treated as a PK to that Model)

GT/LT/GTE/LTE

Automatically check that a param falls within a certain range. Valid for float, int, or Model PK, which all do numerical comparisons.

latitude__gte=-90.0
latitude__lte=90.0

LENGTH__LT/GT/LTE/GTE/EQ

Only valid for str params. Check the length of the str

description__length__lt=256
country_code__length__eq=2

OPTIONAL

latitude__optional=True # same as latitude__default=None

Default is False; if set to True, this param will be checked for validity (it will still return a 400 if it doesn't pass gte checks, for example), but will be passed to the wrapped function as None if it wasn't specified.

DEFAULT

sort_by=('messages_count', 'most_recent')
sort_by__default='messages_count'

Implies that this param is optional. Specify a default value for this param if it isn't specified.

NAME

By default, we'll look for a param with the same name as the kwargs, e.g.

user_id=User # User is a Django model. Look for user_id param, fetch corresponding User, pass to wrapped fn as user_id

But sometimes it makes more sense to call such a param 'user' locally, so you can do:

user=User, user__name='user_id' # look for user_id, assign to user

MANY

users=int # param 'users=1' is ok, 'users=1,2' is not
users__many=True # param 'users=1,2' will return tuple of (1, 2), 'users=1' will return (1)

Allow User to (optionally) specify params as CSV (GET) or Array (JSON POST) If many==True, the params will be returned as a tuple regardless of whether or not there was only one param

DEFERRED

user__deferred=True

By default, Django REST Params will retrieve an object like this:

User.objects.only('id').get(id=user_id) # all fields except for 'id' are deferred

Usually, this is preferrable, since we usually don't need to fetch the entire object from the DB, and it is significantly faster than doing so. By setting __deferred to False, Django REST Params will change the object retrieval call to this:

User.objects.get(id=user_id)  # All fields are fetched

FIELD

category = Category # by default, do Category.get(id=category)
category__field='name' # instead, do Category.get(name=category)

Applies to Django models only. By default, we treat the param as an ID; instead, you can treat it as something else, e.g. 'name'

METHOD

Valid methods for passing this param. Default is 'POST' for POST/PUT requests and GET for all others

user__method='GET' # GET only
user__method=('GET', 'POST') # allow either source

Extra Customization

You can tweak some behavior by setting adding a 'DJANGO_REST_PARAMS' dict to your Django settings module:

DJANGO_REST_PARAMS: {
    'TRUE_VALUES': ('1', 'true'),    # tuple of case-insensitive string values we'll accept as True for a param of type bool.
    'FALSE_VALUES': ('0', 'false'),  # string values that are considered false
}

Tests

Run the (fairly extensive) unit tests:

make test

Mock classes are used to simulate Django models / managers / Django REST Framework requests, so these tests don't actually need to run inside a Django app.

License

3-clause BSD.

django-rest-params's People

Contributors

camsaul avatar dbinoj avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar  avatar

django-rest-params's Issues

Integers in list

When I try to use integers in list, like page_size=(10, 20, 50, 100), django-rest-params compares it as string and throws an error. Is there any way to send a list/tuple of integers and specify that the param takes only integer? Right now TYPE of list assumes that it is of str type. The work around I'm using right now is page_size=("10", "20", "50", "100") defining the list as strings and type casting it in the view. It would be nice if we can specify the type of items in the list.

Python 3 support

Does this plugin support Python 3?
I'm getting this error:

  File "/home/dbinoj/.virtualenvs/py3/lib/python3.5/site-packages/django_rest_params/decorators.py", line 20, in params
    VALID_TYPES = int, float, str, unicode, bool
NameError: name 'unicode' is not defined

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    ๐Ÿ–– Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. ๐Ÿ“Š๐Ÿ“ˆ๐ŸŽ‰

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google โค๏ธ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.