Skip to content

Latest commit

 

History

History
208 lines (144 loc) · 7.9 KB

README.md

File metadata and controls

208 lines (144 loc) · 7.9 KB

femtoweb

femtoweb is an asynchronous Python HTTP server and web application framework that exists primarily for use with Micropython in support of my various IOT projects (e.g. iome).

Branches and their compatibility

branch pairs well with
master Python 3.9
micropython Micropython 1.13

Run the Server

Executing serve.py is just like executing server.py but with filesystem endpoints attached.

python3.9 serve.py

If all goes well, when you point a web browser at localhost:8000 you'll see the text:

404 Not Found

Looks good to me 👍

You're seeing this because, by default, the root path (i.e. /) is not routed to anything. If you go over to localhost:8000/_fs you'll hit this endpoint defined in filesystem_endpoints.py that allows you to navigate the local filesystem.

To demonstrate adding a handler for the root path, add the following here in serve.py, and restart the server.

from server import route, _200, GET

@route('/', methods=(GET,))
async def home(request):
    return _200(body="Hello from femtoweb!")

Now when you surf over to http://localhost:8000/ you should see:

Hello from femtoweb!

The Features

Request Routing

Registering a function as the handler for requests to a certain URL path is accomplished using the route decorator.

Usage

@route(path_pattern, methods=('GET',), query_param_parser_map=None)
async def handler(request):
   ... fun stuff ...
   return Response()

Where:

  • path_pattern is a regular expression string that will be used to match against the request path. Note that if you do not specifying a leading ^ or trailing $, both will be automatically added for you.
  • methods is an iterable of one or more of DELETE, GET, POST, PUT
  • query_param_parser_map is an optional <param-name> -> <parser-func> map that will be used to parse request params. Any missing required or invalid params will result in a 400 - Bad Request response.
  • Response is a Response object

Example

This route simply returns the value of the text request query parameter.

@route('/echo', methods=(GET,), query_param_parser_map={
    'text': as_type(str)
})
async def echo(request, text):
    return _200(body=text)

Available Query Param Parser Functions

Order and Methods

The order in which you define routes matters.

For example, if you define:

@route('.*', methods=(GET,))
async def catchall(request):
    ...

@route('/', methods=(GET, POST))
async def home(request):
    ...

A GET to / can not reach the home handler because the catchall handler was defined first and its path regex will match everything. A POST to /, however, will reach home because catchall only supports GET.

You can also route the same pattern multple times but for different methods. For example, home could be split into two functions, one for GET and one for POST:

@route('/', methods=(GET,))
async def home_GET(request):
    ...

@route('/', methods=(POST,))
async def home_POST(request):
    ...

Request Handler Decorators

event_source

The event_source decorator initiates an EventSource connection and passes an event emitter function as an argument to the request handler. Note that the handler must return None instead of the normal Response object, otherwise the connection will be closed.

Example:

@route('/events', methods=(GET,))
@event_source
async def events(request, emitter):
    n = 0
    while True:
        await emitter(n)
        n += 1
        await asyncio.sleep(1)

This handler will emit an incremented number each second.

To receive and view these events in a web browser, open the dev console and enter:

new EventSource('/events').onmessage = event => console.log(event.data)

You should see the count printed to the console, i.e.

0
1
2
3
...

This example is implemented in serve.py.

json_response

The json_response decorator will automatically encode the response as JSON.

Example:

@route('/time', methods=(GET,))
@json_response
async def get_time(request):
   return _200(body={'currentTime': datetime.now().isoformat()})

This will make the /time endpoint respond with the body {"currentTime": "2019-10-16T20:49:22.543090"} and Content-Type: application/json.

This example is implemented in serve.py.

File Operations

filesystem_endpoints.py implements a /_fs endpoint that supports file operations.

GET Operations

Currently, a GET to:

  • a directory-type object path will respond with an HTML page comprising a list of links
  • a file-type object path will respond with the file itself, setting the response Content-Type as appropriate
In-browser File Editor

filesystem_views.py defines a super-simple TextFileEditor in-browser editor for plain text files that you can access by specifying the edit=1 URL arg, e.g.: http://localhost:8000/_fs/hello.txt?edit=1

Once you're done editing, you can click the Submit button or press CTRL-Enter to submit your changes, after which it will automatically redirect to the non-edit URL for the file.

Note that:

  • You can create new files by also specifying create=1, e.g. ``http://localhost:8000/_fs/newfile.txt?edit=1&create=1`
  • No validation is currently performed on the submitted data, so if youre editing a JSON file and you submit something that isn't valid JSON, you won't know until your application tries to read it, and probably crashes.

curl

You can use curl to manipulate the filesystem from the command line:

# Create or update a file
curl --upload-file file.txt `http://localhost:8000/_fs/file.txt

# Get a file
curl `http://localhost:8000/_fs/file.txt

# Delete a file
curl -X DELETE `http://localhost:8000/_fs/file.txt