A tiny HTTP server made for CircuitPython WiFi devices (like the ESP32).
Note that ampule is in alpha and right now for use by tally_circuitpy. Feel free to use it, but know that there are tons of things not yet implemented.
ampule gathers inspiration from Bottle: Python Web Framework, Adafruit's CircuitPython WSGI library, Adafruit's ESP32 SPI WSGI Server, and Adafruit's CircuitPython Requests library.
Route definitions in ampule are expressed through annotations that define the path and optionally the method to be matched against for incoming HTTP requests. ampule will pass along the request and any path elements that operate as arguments, and expects the HTTP status code, headers, and body to be returned as a tuple.
The following example is a simple, working HTTP server that accepts an
HTTP GET request at /hello/world
and responds with "Hi There!".
import ampule, socketpool, wifi
@ampule.route("/hello/world")
def light_set(request):
return (200, {}, 'Hi There!'})
try:
from secrets import secrets
except ImportError:
print("WiFi secrets not found in secrets.py")
raise
try:
print("Connecting to %s..." % secrets["ssid"])
print("MAC: ", [hex(i) for i in wifi.radio.mac_address])
wifi.radio.connect(secrets["ssid"], secrets["password"])
except:
print("Error connecting to WiFi")
raise
pool = socketpool.SocketPool(wifi.radio)
socket = pool.socket()
socket.bind(['0.0.0.0', 80])
socket.listen(1)
print("Connected to %s, IPv4 Addr: " % secrets["ssid"], wifi.radio.ipv4_address)
while True:
ampule.listen(socket)
The majority of this code is CircuitPython boilerplate that connects to a WiFi network, listens on port 80, and connects ampule to the open socket.
The line @ampule.route("/hello/world")
registers the following function for
the path specified, and responds with HTTP 200, no headers, and a response body
of "Hi There!"
Route paths can also contain variables, as in:
@ampule.route("/hello/<name>")
def light_set(request, name):
return (200, {}, "Hi there %s!" % name)
Query parameters are passed along with the request. If a URL ends with
/hello/world?name=Bob
then the following route will return
"Hi there Bob!":
@ampule.route("/hello/world")
def light_set(request):
name = request.params["name"]
return (200, {}, "Hi there %s!" % name)
You can explicitly specify the HTTP method on a route. If it is omitted, the match defaults to 'GET'.
@ampule.route("/hello/world", method='POST')
def light_set(request):
name = request.body
return (200, {}, "Hi there %s!" % name)
Headers are returned as part of the returned tuple in the route handler. As an example, returning JSON content and permitting cross-origin access would be:
headers = {
"Content-Type": "application/json; charset=UTF-8",
"Access-Control-Allow-Origin": '*',
"Access-Control-Allow-Methods": 'GET, POST',
"Access-Control-Allow-Headers": 'Origin, Accept, Content-Type, X-Requested-With, X-CSRF-Token'
}
@ampule.route("/hello/world")
def light_set(request):
return (200, headers, '{"hello": true}')
ampule can be added as a .py file into CIRCUITPY projects, or it can be
used as a MicroPython compiled library and added to the lib/
folder
of a CIRCUITPY project.
After installing the mpy-cross
compiler from
https://learn.adafruit.com/creating-and-sharing-a-circuitpython-library?view=all#mpy-2982472-11
you can execute:
mpy-cross ampule.py
To build a MicroPython compiled library.
Tests require pytest
to be installed along with a local version of ampule, as in:
pip3 install -e .
Once installed pytest can be invoked with default arguments, as in:
pytest