275 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			ReStructuredText
		
	
	
	
	
	
			
		
		
	
	
			275 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			ReStructuredText
		
	
	
	
	
	
| Request Data
 | |
| ============
 | |
| 
 | |
| When an endpoint receives a HTTP request, the route function is passed a
 | |
| `Request` object.
 | |
| 
 | |
| The following variables are accessible as properties on `Request` objects:
 | |
| 
 | |
| - `json` (any) - JSON body
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     from sanic.response import json
 | |
| 
 | |
|     @app.route("/json")
 | |
|     def post_json(request):
 | |
|       return json({ "received": True, "message": request.json })
 | |
| 
 | |
| 
 | |
| - `args` (dict) - Query string variables. A query string is the section of a
 | |
|   URL that resembles ``?key1=value1&key2=value2``.
 | |
| If that URL were to be parsed, the `args` dictionary would look like `{'key1': ['value1'], 'key2': ['value2']}`.
 | |
| The request's `query_string` variable holds the unparsed string value. Property is providing the default parsing
 | |
| strategy. If you would like to change it look to the section below (`Changing the default parsing rules of the queryset`).
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     from sanic.response import json
 | |
| 
 | |
|     @app.route("/query_string")
 | |
|     def query_string(request):
 | |
|       return json({ "parsed": True, "args": request.args, "url": request.url, "query_string": request.query_string })
 | |
| 
 | |
| - `query_args` (list) - On many cases you would need to access the url arguments in
 | |
|   a less packed form. `query_args` is the list of `(key, value)` tuples.
 | |
| Property is providing the default parsing strategy. If you would like to change it look to the section below
 | |
| (`Changing the default parsing rules of the queryset`). For the same previous URL queryset `?key1=value1&key2=value2`,
 | |
| the `query_args` list would look like `[('key1', 'value1'), ('key2', 'value2')]`. And in case of the multiple params
 | |
| with the same key like `?key1=value1&key2=value2&key1=value3` the `query_args` list would look like
 | |
| `[('key1', 'value1'), ('key2', 'value2'), ('key1', 'value3')]`.
 | |
| 
 | |
| The difference between Request.args and Request.query_args for the queryset `?key1=value1&key2=value2&key1=value3`
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     from sanic import Sanic
 | |
|     from sanic.response import json
 | |
| 
 | |
|     app = Sanic(__name__)
 | |
| 
 | |
| 
 | |
|     @app.route("/test_request_args")
 | |
|     async def test_request_args(request):
 | |
|       return json({
 | |
|           "parsed": True,
 | |
|           "url": request.url,
 | |
|           "query_string": request.query_string,
 | |
|           "args": request.args,
 | |
|           "raw_args": request.raw_args,
 | |
|           "query_args": request.query_args,
 | |
|       })
 | |
| 
 | |
|     if __name__ == '__main__':
 | |
|       app.run(host="0.0.0.0", port=8000)
 | |
| 
 | |
|   Output
 | |
| 
 | |
| .. code-block:: json
 | |
| 
 | |
|     {
 | |
|         "parsed":true,
 | |
|         "url":"http:\/\/0.0.0.0:8000\/test_request_args?key1=value1&key2=value2&key1=value3",
 | |
|         "query_string":"key1=value1&key2=value2&key1=value3",
 | |
|         "args":{"key1":["value1","value3"],"key2":["value2"]},
 | |
|         "raw_args":{"key1":"value1","key2":"value2"},
 | |
|         "query_args":[["key1","value1"],["key2","value2"],["key1","value3"]]
 | |
|     }
 | |
| 
 | |
| - `raw_args` contains only the first entry of `key1`. Will be deprecated in the future versions.
 | |
| 
 | |
| - `files` (dictionary of `File` objects) - List of files that have a name, body, and type
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     from sanic.response import json
 | |
| 
 | |
|     @app.route("/files")
 | |
|     def post_json(request):
 | |
|       test_file = request.files.get('test')
 | |
| 
 | |
|       file_parameters = {
 | |
|           'body': test_file.body,
 | |
|           'name': test_file.name,
 | |
|           'type': test_file.type,
 | |
|       }
 | |
| 
 | |
|       return json({ "received": True, "file_names": request.files.keys(), "test_file_parameters": file_parameters })
 | |
| 
 | |
| - `form` (dict) - Posted form variables.
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     from sanic.response import json
 | |
| 
 | |
|     @app.route("/form")
 | |
|     def post_json(request):
 | |
|       return json({ "received": True, "form_data": request.form, "test": request.form.get('test') })
 | |
| 
 | |
| - `body` (bytes) - Posted raw body. This property allows retrieval of the
 | |
|   request's raw data, regardless of content type.
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|   from sanic.response import text
 | |
| 
 | |
|   @app.route("/users", methods=["POST",])
 | |
|   def create_user(request):
 | |
|       return text("You are trying to create a user with the following POST: %s" % request.body)
 | |
| 
 | |
| 
 | |
| - `headers` (dict) - A case-insensitive dictionary that contains the request headers.
 | |
| 
 | |
| - `method` (str) - HTTP method of the request (ie `GET`, `POST`).
 | |
| 
 | |
| - `ip` (str) - IP address of the requester.
 | |
| 
 | |
| - `port` (str) - Port address of the requester.
 | |
| 
 | |
| - `socket` (tuple) - (IP, port) of the requester.
 | |
| 
 | |
| - `app` - a reference to the Sanic application object that is handling this request. This is useful when inside blueprints or other handlers in modules that do not have access to the global `app` object.
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|   from sanic.response import json
 | |
|   from sanic import Blueprint
 | |
| 
 | |
|   bp = Blueprint('my_blueprint')
 | |
| 
 | |
|   @bp.route('/')
 | |
|   async def bp_root(request):
 | |
|       if request.app.config['DEBUG']:
 | |
|           return json({'status': 'debug'})
 | |
|       else:
 | |
|           return json({'status': 'production'})
 | |
| 
 | |
| - `url`: The full URL of the request, ie: `http://localhost:8000/posts/1/?foo=bar`
 | |
| - `scheme`: The URL scheme associated with the request: 'http|https|ws|wss' or arbitrary value given by the headers.
 | |
| - `host`: The host associated with the request(which in the `Host` header): `localhost:8080`
 | |
| - `server_name`: The hostname of the server, without port number. the value is seeked in this order: `config.SERVER_NAME`, `x-forwarded-host` header, :func:`Request.host`
 | |
| - `server_port`: Like `server_name`. Seeked in this order: `x-forwarded-port` header, :func:`Request.host`, actual port used by the transport layer socket.
 | |
| - `path`: The path of the request: `/posts/1/`
 | |
| - `query_string`: The query string of the request: `foo=bar` or a blank string `''`
 | |
| - `uri_template`: Template for matching route handler: `/posts/<id>/`
 | |
| - `token`: The value of Authorization header: `Basic YWRtaW46YWRtaW4=`
 | |
| 
 | |
| - `url_for`: Just like `sanic.Sanic.url_for`, but automatically determine `scheme` and `netloc` base on the request. Since this method is aiming to generate correct schema & netloc, `_external` is implied.
 | |
| 
 | |
| 
 | |
| Changing the default parsing rules of the queryset
 | |
| --------------------------------------------------
 | |
| 
 | |
| The default parameters that are using internally in `args` and `query_args` properties to parse queryset:
 | |
| 
 | |
| - `keep_blank_values` (bool): `False` - flag indicating whether blank values in
 | |
|   percent-encoded queries should be treated as blank strings.
 | |
|   A true value indicates that blanks should be retained as blank
 | |
|   strings.  The default false value indicates that blank values
 | |
|   are to be ignored and treated as if they were  not included.
 | |
| - `strict_parsing` (bool): `False` - flag indicating what to do with parsing errors. If
 | |
|   false (the default), errors are silently ignored. If true,
 | |
|   errors raise a ValueError exception.
 | |
| - `encoding` and `errors` (str): 'utf-8' and 'replace' - specify how to decode percent-encoded sequences
 | |
|   into Unicode characters, as accepted by the bytes.decode() method.
 | |
| 
 | |
| If you would like to change that default parameters you could call `get_args` and `get_query_args` methods
 | |
| with the new values.
 | |
| 
 | |
| For the queryset `/?test1=value1&test2=&test3=value3`:
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     from sanic.response import json
 | |
| 
 | |
|     @app.route("/query_string")
 | |
|     def query_string(request):
 | |
|         args_with_blank_values = request.get_args(keep_blank_values=True)
 | |
|         return json({
 | |
|             "parsed": True,
 | |
|             "url": request.url,
 | |
|             "args_with_blank_values": args_with_blank_values,
 | |
|             "query_string": request.query_string
 | |
|         })
 | |
| 
 | |
| The output will be:
 | |
| 
 | |
| .. code-block:: JSON
 | |
| 
 | |
|     {
 | |
|         "parsed": true,
 | |
|         "url": "http:\/\/0.0.0.0:8000\/query_string?test1=value1&test2=&test3=value3",
 | |
|         "args_with_blank_values": {"test1": ["value1"], "test2": "", "test3": ["value3"]},
 | |
|         "query_string": "test1=value1&test2=&test3=value3"
 | |
|     }
 | |
| 
 | |
| Accessing values using `get` and `getlist`
 | |
| ------------------------------------------
 | |
| 
 | |
| The `request.args` returns a subclass of `dict` called `RequestParameters`. 
 | |
| The key difference when using this object is the distinction between the `get` and `getlist` methods.
 | |
| 
 | |
| - `get(key, default=None)` operates as normal, except that when the value of
 | |
|   the given key is a list, *only the first item is returned*.
 | |
| - `getlist(key, default=None)` operates as normal, *returning the entire list*.
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     from sanic.request import RequestParameters
 | |
| 
 | |
|     args = RequestParameters()
 | |
|     args['titles'] = ['Post 1', 'Post 2']
 | |
| 
 | |
|     args.get('titles') # => 'Post 1'
 | |
| 
 | |
|     args.getlist('titles') # => ['Post 1', 'Post 2']
 | |
| 
 | |
| .. code-block:: python
 | |
|     from sanic import Sanic
 | |
|     from sanic.response import json
 | |
| 
 | |
|     app = Sanic(name="example")
 | |
| 
 | |
|     @app.route("/")
 | |
|     def get_handler(request):
 | |
|         return json({
 | |
|             "p1": request.args.getlist("p1")
 | |
|         })
 | |
|         
 | |
| Accessing the handler name with the request.endpoint attribute
 | |
| --------------------------------------------------------------
 | |
| 
 | |
| The `request.endpoint` attribute holds the handler's name. For instance, the below
 | |
| route will return "hello".
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     from sanic.response import text
 | |
|     from sanic import Sanic
 | |
| 
 | |
|     app = Sanic()
 | |
| 
 | |
|     @app.get("/")
 | |
|     def hello(request):
 | |
|         return text(request.endpoint)
 | |
| 
 | |
| Or, with a blueprint it will be include both, separated by a period. For example, the below route would return foo.bar:
 | |
| 
 | |
| .. code-block:: python
 | |
| 
 | |
|     from sanic import Sanic
 | |
|     from sanic import Blueprint
 | |
|     from sanic.response import text
 | |
| 
 | |
| 
 | |
|     app = Sanic(__name__)
 | |
|     blueprint = Blueprint('foo')
 | |
| 
 | |
|     @blueprint.get('/')
 | |
|     async def bar(request):
 | |
|         return text(request.endpoint)
 | |
| 
 | |
|     app.blueprint(blueprint)
 | |
| 
 | |
|     app.run(host="0.0.0.0", port=8000, debug=True)
 | 
