Skip to content

Files

Latest commit

 

History

History
201 lines (148 loc) · 6.95 KB

README.rst

File metadata and controls

201 lines (148 loc) · 6.95 KB

Logo

Main workflow status Documentation home Current pyPI version Supported on python versions Package license Any color you want

Project Description

This extension provides an easy and safe way to add dev-only shortcuts to routes in your flask application.

The main beneficiaries are microservices that need to be tested regularly in conjunction with their clients. If you need to assert working communication and basic integration in a sufficiently complex ecosystem, clients that can not freely chose how their requests are formed gain a lot from being able to receive predictable responses. By skipping over the details of how the microservice is implemented, which bugs and minor changes it experiences over time, testing basic API compatibility gets a lot more manageable.

In a test-engineering context, it would be considered a "fake".

What is a Shortcut?

In the context of this package, a shortcut is a condition-and-response pair. The response is anything that a view function can return, and the condition depends on one of the three possible mapping contexts.

In the first context, only the response is passed as the shortcut, and the condition is assumed to always be true, effectively replacing the route to always just return the given response. Showcased with the view foo in the usage section.

In the second context, a dictionary that maps strings to responses is passed as the shortcut. The strings need to be deserializeable as json, and the first one that can be fully matched as a substructure into the request body will see its condition as fulfilled, returning its associated response. If none of them are valid sub-matches, the original view function will run. Showcased with the view bar in the usage section.

In the third context, either a single function or a list of functions is passed as the shortcut. The functions can run any code whatsoever and will be executed one after the other as long as they return None, which means that their condition is not fulfilled. As soon as one of them returns something different, it is passed on as the response. If all of them return None, the original view function is executed. Showcased with the view baz in the usage section.

Usage

You can add shortcuts to your view functions either individually with decorators, or in a single swoop once all routes have been defined. Both ways are functionally equivalent.

Applying Shortcuts

With decorators:

from flask import Flask
from flask_shortcut import Shortcut

app = Flask(__name__)
short = Shortcut(app)

@app.route('/foo', methods=['GET'])
@short.cut(('short_foo', 200))
def foo():
    return 'foo'

@app.route('/bar', methods=['POST'])
@short.cut({
    '{"name": "TestUser"}': ('short_bar', 200),
    '{"name": "UserTest"}': ('longer_bar', 200),
})
def bar():
    return 'bar'

@app.route('/baz', methods=['POST'])
@short.cut(lambda: ("json_baz", 200) if "json" in request.mimetype else None)
def baz():
    return 'baz'

With a wire call

from flask import Flask
from flask_shortcut import Shortcut

app = Flask(__name__)

@app.route('/foo', methods=['GET'])
def foo():
    return 'foo'

@app.route('/bar', methods=['POST'])
def bar():
    return 'bar'

@app.route('/baz', methods=['POST'])
def baz():
    return 'baz'

Shortcut(app).wire(
    {
         '/foo': ('short_foo', 200),
         '/bar': {
             '{"name": "TestUser"}': ('short_bar', 200),
             '{"name": "UserTest"}': ('longer_bar', 200),
         }
         '/baz': lambda: ("json_baz", 200) if "json" in request.mimetype else None
    }
)

What it looks like

To showcase how the shortcuts are supposed to work, here is the result of a couple of requests sent against the server from the example above if it were run with FLASK_ENV=test flask run:

>>> from request import get, post
>>> get('http://127.0.0.1:5000/foo').text
'short_foo'  # the only response this route will give
>>> post('http://127.0.0.1:5000/bar', json={"name": "me"}).text
'bar'  # no shortcut match -> the original logic was executed
>>> post('http://127.0.0.1:5000/bar', json={"name": "TestUser"}).text
'short_bar'  # shortcut match
>>> post('http://127.0.0.1:5000/bar', json={"name": "UserTest", "job": None}).text
'longer_bar'  # shortcut only needs to be contained for a match
>>> post('http://127.0.0.1:5000/baz').text
'baz'  # no shortcut match -> the function returned None
>>> post('http://127.0.0.1:5000/baz', json={"name": "me"}).text
'json_baz'  # shortcut matched -> the function returned a valid response

One focus of this package is that a production deployment would remain as ignorant as possible about the existence of shortcuts. While the shortcut object is still created, it only delegates the view functions and no shortcut code has any chance of being executed, or showing up in stack traces.

Configuration

Shortcuts will only be applied when FLASK_ENV is set to something different from its default setting, production. You can extend that list through the SHORTCUT_EXCLUSIONS config setting, either by adding it to your app's config before creating any Shortcut objects, or preferably by setting up the whole config with a file.

Possible values for it are all environments that you want to block other than production separated by commas. For example staging,master will block the envs production, staging, and master from receiving shortcuts.