The Unmock Python clent
Project description
Unmock (Python SDK)
Public API mocking for Python.
Unmock can be used to test modules that perform requests to third-party APIs like Hubspot, SendGrid, Behance, and hundreds of other public APIs.
Unmock can also be used to mock these APIs in a development environment, i.e. an express server on a local machine or in a staging environment.
The Unmock Python package offers intuitive, hassle-free SDK to the Unmock service with minimal setup steps.
The ultimate goal of unmock is to provide a semantically and functionally adequate mock of the internet.
Unmock also provides access via other languages, all with similar interface. We have unmock-js already publicly available, and we are working on .Net, PHP and Java. We're open to more requests - just let us know!
Table of Contents
How does it work?
Unmock works by overriding Python's low-level HTTPConnection
's and
HTTPRequest
's functions, thereby capturing calls
made by popular packages such as requests
and urllib3
.
Unmock works out of the box for most APIs that it mocks and does not require any additional configuration. For APIs that it does not mock yet, or to tweak return values from the unmock service, you can consult the URLs printed to the command line by unmock.
We intend to offer Python2.7 support quite soon, along with other common
libraries such as aiohttp
, pycurl
, etc.
Install
$ pip install unmock
Usage
Tests
In your unit tests, you can invoke unmock in several ways:
- If you're using pytest for your tests, you can either use the unmock fixture (you don't even need to import unmock!) -
import pytest
import requests
def test_behance(unmock_local):
response = requests.get("https://www.behance.net/v2/projects/5456?api_key=u_n_m_o_c_k_200")
assert response.json().get("project").get("id") == 5456
... or you may want to use unmock for all your tests, in which case you
can simply use the --unmock
flag for pytest:
pytest tests --unmock
- You can control use unmock in a scoped manner using context managers:
# do stuff
with unmock.patch():
response = requests.get("https://www.example.com/")
# do stuff with mocked response
real_response = requests.get("https://www.example.com/") # won't be mocked
# You can also access the returned object to modify certain runtime behaviour:
with unmock.patch() as opts:
# can modify certain behaviour aspects via `opts` object now too
response = requests.get("https://www.example.com/")
- You can have fine grained control over unmock using the
init
andreset
methods, and modify theUnmockOptions
object during runtime:
import unmock
# do stuff
opts = unmock.init()
res1 = requests.get("https://www.example.com") # will be mocked
opts.save = True
res2 = requests.get("https://www.example.com") # will be mocked and response will be saved
unmock.reset()
res3 = requests.get("https://www.example.com") # will not be mocked
Unmock will then either serve JIT semantically functionally correct mocks from its database or an empty JSON object for unmocked APIs that can be filled in by the user. The address of these editable objects is printed to the command line during tests.
Development
After you create your flask, django, or own server, call
unmock_options = unmock.init()
unmock_options.ignore("story")
# equivalent to calling:
unmock.init(ignore="story")
This has the same effect as activating unmock in your tests.
It will intercept HTTP traffic and serve semantically and functionally
adequate mocks of the APIs in the unmock catalogue.
The main difference is the result of ignore("story")
passed to unmock
options, which tells the service to ignore the order of mocked requests.
Always use this option when the order of mocked data does not matter,
i.e. when you are in sandbox or development mode.
For users of the unmock.io service, this will
help unmock better organize your mocks in its web dashboard.
unmock.io
The URLs printed to the command line are hosted by unmock.io. You can consult the documentation about that service here.
Scoping
As a handy shortcut to initializing and reseting the capturing of API
calls, we also offer the use of context manager via unmock.patch()
.
patch
accepts as parameters anything that init
accepts.
Saving mocks
All mocks can be saved to a folder called .unmock
in your user's home
directory by adding a save
field to the unmock options object like so:
unmock_options = unmock.init(save=True)
You can also specify a specific location to save the directory:
unmock_options = unmock.init(save=True, path=".") # Saves in current path
Unmock refers to every mock by a unique hash. Individual mocks or groups of mocks can be saved by setting save to either a single hash or an array of hashes like so:
unmock_options = unmock.init(save=["ahash", "anotherhash", "yetanotherhash"])
Ignoring aspects of a mock
Sometimes, you would like for two mocks of slightly API calls to be
treated as equivalent by unmock. For example, you may want all GET
calls to the same path with different headers to be served the same
mock. To do this, use the ignore
field of the unmock options object.
You can do this while initializing unmock or afterwards (as shown before
with ignoring "story"
):
# Option A:
unmock_options = unmock.init()
unmock_options.ignore("headers", "story")
# Option B:
unmock.init(ignore=["headers", "story"])
The following fields may be ignored:
headers
: the headers of the requesthostname
: the hostname of the requestmethod
: the method of the request (ie GET, POST, PUT, DELETE). Note that this is case insensitive!path
: the path of the requeststory
: the story of the request, meaning its order in a series of requests
Ignore evaluates regular expressions, so you can also pass
"headers|path"
instead of ["headers", "path"]
. Furthermore, to
ignore nested headers, pass a dictionary such as
{"headers": "Authorization" }
, or to match against the value of a
header, {"headers": { Authorization: "Bearer *" }}
. When using the
ignore method on the UnmockOptions
object (returned from a call to init
),
you may pass either a list (*args
) or a dictionary (**kwargs
).
Adding a signature
Sometimes, it is useful to sign a mock with a unique signature. This is
useful, for example, when AB testing code that should serve two
different mocks for the same endpoint in otherwise similar conditions.
To do this, use the signature
field of the unmock options object:
unmock_options = unmock.init()
unmock_options.signature = "signature-for-this-particular-test"
# Equivalent to
unmock.init(signature="signature-for-this-particular-test")
Whitelisting API
If you do not want a particular API to be mocked, whitelist it.
unmock_options = unmock.init()
unmock_options.whitelist = ["api.hubspot.com", "api.typeform.com"]
# Equivalent to:
unmock.init(whitelist=["api.hubspot.com", "api.typeform.com"])
unmock.io tokens
If you are subscribed to the unmock.io service, you can pass your unmock token directly to the unmock object.
unmock.init(token="my-token")
At a certain point this becomes a bit tedious, (even if very readable),
at which point you will want to create a credentials file. See
unmock.io/docs for more information on
credential files.
Behind the scenes, we automatically create a credentials file for you,
for caching purposes. With this, subsequent calls to unmock.init()
will read the token from the credential files.
Contributing
Thanks for wanting to contribute! We will soon have a contributing page detaling how to contribute. Meanwhile, star this repository, open issues and ask for more features and support!
Please note that this project is released with a Contributor Code of Conduct. By participating in this project you agree to abide by its terms.
License
Copyright (c) 2018–2019 Meeshkan and other contributors.
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file unmock-0.1.2.tar.gz
.
File metadata
- Download URL: unmock-0.1.2.tar.gz
- Upload date:
- Size: 21.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/1.13.0 pkginfo/1.5.0.1 requests/2.21.0 setuptools/41.0.0 requests-toolbelt/0.8.0 tqdm/4.31.1 CPython/3.6.8
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 4fa24aabb2e24760cfc6ac38c49e3c4d65cd264e4b335c1247ddb6c1953d49a2 |
|
MD5 | 0d2a4b3e5ae21fa0f4ac1bc54fdcaa76 |
|
BLAKE2b-256 | f953ff8a1a623b71bce3c84a57b0fa3984789c393314795c239a6a2cbfe41e68 |
File details
Details for the file unmock-0.1.2-py2.py3-none-any.whl
.
File metadata
- Download URL: unmock-0.1.2-py2.py3-none-any.whl
- Upload date:
- Size: 20.9 kB
- Tags: Python 2, Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/1.13.0 pkginfo/1.5.0.1 requests/2.21.0 setuptools/41.0.0 requests-toolbelt/0.8.0 tqdm/4.31.1 CPython/3.6.8
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 976326c897e8df5eff792116aecfc3bd6e0f7a81d5d42beb84c7ce0917bf4ae5 |
|
MD5 | ddbc9b9802fedaa60a66af2d55ce6f80 |
|
BLAKE2b-256 | c6e48163d153d5724ee85436981d5ed132d2eb86904437c56b640731337f7f67 |