Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/deckerego/ampule

A tiny HTTP server made for CircuitPython WiFi devices (like the ESP32)
https://github.com/deckerego/ampule

circuitpython circuitpython-library esp32-s2 feather-huzzah featherwing http-server pixelwing

Last synced: 3 months ago
JSON representation

A tiny HTTP server made for CircuitPython WiFi devices (like the ESP32)

Awesome Lists containing this project

README

        

# ampule

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](https://github.com/deckerego/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](https://bottlepy.org/docs/dev/index.html),
Adafruit's [CircuitPython WSGI](https://github.com/adafruit/Adafruit_CircuitPython_WSGI)
library, Adafruit's [ESP32 SPI WSGI Server](https://github.com/adafruit/Adafruit_CircuitPython_ESP32SPI/blob/main/adafruit_esp32spi/adafruit_esp32spi_wsgiserver.py),
and Adafruit's [CircuitPython Requests library](https://github.com/adafruit/Adafruit_CircuitPython_Requests).

## Usage

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.

### "Hello World" Example

The following example is a simple, working HTTP server that accepts an
HTTP GET request at `/hello/world` and responds with "Hi There!".

```python
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)
socket.setblocking(True)
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 in blocking mode, 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!"

### Variables in the URL Path

Route paths can also contain variables, as in:

```python
@ampule.route("/hello/")
def light_set(request, name):
return (200, {}, "Hi there %s!" % name)
```

### Query Parameters

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!":

```python
@ampule.route("/hello/world")
def light_set(request):
name = request.params["name"]
return (200, {}, "Hi there %s!" % name)
```

### Specifying HTTP Method on Routes

You can explicitly specify the HTTP method on a route. If it is omitted,
the match defaults to 'GET'.

```python
@ampule.route("/hello/world", method='POST')
def light_set(request):
name = request.body
return (200, {}, "Hi there %s!" % name)
```

### Specifying Headers

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:

```python
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}')
```

## Building ampule

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.

### Building MicroPython libraries

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:

```sh
mpy-cross ampule.py
```

To build a MicroPython compiled library.

### Running tests

Tests require `pytest` to be installed along with a local version of ampule, as in:

```sh
pip3 install -e .
```

Once installed pytest can be invoked with default arguments, as in:

```sh
pytest
```