Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/healeycodes/jar
🫙 A server-side web framework that deploys to Vercel.
https://github.com/healeycodes/jar
build-output-api serverless-functions vercel web-framework
Last synced: about 1 month ago
JSON representation
🫙 A server-side web framework that deploys to Vercel.
- Host: GitHub
- URL: https://github.com/healeycodes/jar
- Owner: healeycodes
- Created: 2023-02-18T14:41:00.000Z (almost 2 years ago)
- Default Branch: main
- Last Pushed: 2024-11-18T15:10:51.000Z (about 2 months ago)
- Last Synced: 2024-11-29T21:46:56.298Z (about 1 month ago)
- Topics: build-output-api, serverless-functions, vercel, web-framework
- Language: Python
- Homepage: https://jar-docs.vercel.app
- Size: 87.9 KB
- Stars: 33
- Watchers: 1
- Forks: 1
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
[![CI](https://github.com/healeycodes/jar/actions/workflows/ci.yml/badge.svg)](https://github.com/healeycodes/jar/actions/workflows/ci.yml)
# Jar 🫙
> My blog post: [My Own Python Web Framework](https://healeycodes.com/my-own-python-web-framework)
Jar is a toy Python web framework, implemented in about 200 lines of code (see [cli.py](https://github.com/healeycodes/jar/blob/main/framework/cli.py)).
I built it to explore some ideas around framework APIs. Please don't actually use it.
It deploys to Vercel via the [Build Output API](https://vercel.com/docs/build-output-api/v3).
It's called Jar because it has almost no features and you need to fill it up yourself!
## Docs
https://jar-docs.vercel.app (built with Jar obviously!)
## Features
Jar uses file-system routing.
Pages are Python files that render content. They're put in the `pages` directory.
- Build pages aka [Static Files](https://vercel.com/docs/build-output-api/v3#vercel-primitives/static-files)
- Fresh pages aka [Serverless Functions](https://vercel.com/docs/build-output-api/v3#vercel-primitives/serverless-functions)
- Regenerated pages aka [Prerender Functions](https://vercel.com/docs/build-output-api/v3#vercel-primitives/prerender-functions)Public files (like CSS and other media files) go in the `public` directory and are served from the root path.
Checkout the source for the [kitchen sink example](https://github.com/healeycodes/jar/tree/main/examples/kitchensink), or [the docs website](https://github.com/healeycodes/jar/tree/main/examples/docs).
A typical project is structured like this:
```text
project/
├─ pages/
│ ├─ index.py
├─ public/
│ ├─ favicon.ico
```### Build pages
Generated once at build time. Served as a static file.
```python
import timedef render(data):
return f"{data['text']} I was built at {data['time']}
", {}def data():
return {
"text": "Hello, World!",
"time": time.time(),
}
```### Fresh pages
Generated for each request. Similar to Server-Side Rendering (SSR).
```python
import json
import timedef render(data):
content = f"Fresh Page rendered at {data['time']}
"
content += f"{data['request']}
"
return content, {}def data(request):
return {
"time": time.time(),
"request": json.dumps({
"method": request.method,
"path": request.path,
"headers": request.headers,
"body": request.body
}, indent=4)
}def config():
return {
"fresh": {}
}
```### Regenerated Pages
Similar to Next.js's [Incremental Static Regeneration](https://nextjs.org/docs/basic-features/data-fetching/incremental-static-regeneration).
```python
import timedef render(data):
return f"Regenerated Page, last rendered at {data['time']}
", {}def data(request=None):
return {
"time": time.time(),
}def config():
return {
"regenerate": {
"every": 5
}
}
```## More on the API
The `data` and `config` functions are optional. The properties that `config` returns defines the type of page. The default page is a build page.
Build pages don't have access to a request object.
The response that `render` returns is a tuple of `body, meta` where `meta` can have `status_code: int` and/or `headers: dict` keys e.g. `{"status_code": 200, "headers": {"some":"header"}}`.
## More on Packages
If you are using packages (i.e. something you install with `pip`) you have to install them locally inside your project before building the project with the CLI.
e.g. with `pip` you can run `pip3 install -r requirements.txt --target .` at the project's root.
See how the docs website is deployed for an example of this ([deploy-docs.sh](https://github.com/healeycodes/jar/blob/main/deploy-docs.sh)).
## Tests
- `pip3 install -r framework/requirements.txt`
- `pytest`## Deploy Docs
`./deploy-docs.sh`
## Deploy Kitchen Sink
- `pip3 install -r framework/requirements.txt`
- `python3 framework/cli.py build examples/kitchensink`
- `cd build && vercel --prebuilt`