|
1 | | -from lightbug_api import ( |
2 | | - App, |
3 | | - BaseRequest, |
4 | | - FromReq, |
5 | | - Router, |
6 | | - HandlerResponse, |
7 | | - JSONType, |
8 | | -) |
9 | | -from lightbug_http import HTTPRequest, HTTPResponse, OK |
| 1 | +# lightbug_api showcase |
| 2 | +# ───────────────────────────────────────────────────────────────────────────── |
| 3 | +# Run: mojo lightbug.mojo |
| 4 | +# Test: curl http://localhost:8080/ |
| 5 | +# curl http://localhost:8080/items |
| 6 | +# curl http://localhost:8080/items/42 |
| 7 | +# curl http://localhost:8080/items/42?verbose=true |
| 8 | +# curl -X POST http://localhost:8080/items \ |
| 9 | +# -H 'Content-Type: application/json' \ |
| 10 | +# -d '{"name":"Widget","price":9.99}' |
| 11 | +# curl -X PUT http://localhost:8080/items/42 \ |
| 12 | +# -H 'Content-Type: application/json' \ |
| 13 | +# -d '{"name":"Updated","price":19.99}' |
| 14 | +# curl -X DELETE http://localhost:8080/items/42 |
| 15 | +# curl http://localhost:8080/v1/status |
| 16 | +# ───────────────────────────────────────────────────────────────────────────── |
10 | 17 |
|
| 18 | +from lightbug_api import App, Router, HandlerResponse |
| 19 | +from lightbug_api.context import Context |
| 20 | +from lightbug_api.response import Response |
| 21 | +from lightbug_http.http.json import Json, JsonSerializable, JsonDeserializable |
11 | 22 |
|
12 | | -fn printer(req: HTTPRequest) raises -> HandlerResponse: |
13 | | - print("Got a request on ", req.uri.path, " with method ", req.method) |
14 | | - return OK(req.body_raw) |
15 | 23 |
|
16 | | -fn hello(req: HTTPRequest) raises -> HandlerResponse: |
17 | | - return OK("Hello 🔥!") |
| 24 | +# ----------------------------------------------------------------- data types |
18 | 25 |
|
| 26 | +@fieldwise_init |
| 27 | +struct Item(JsonSerializable, Movable, Defaultable): |
| 28 | + """An item returned in API responses.""" |
19 | 29 |
|
20 | | -fn nested(req: HTTPRequest) raises -> HandlerResponse: |
21 | | - print("Handling route:", req.uri.path) |
22 | | - # Returning a string will get marshaled to a proper `OK` response |
23 | | - return req.uri.path |
| 30 | + var id: Int |
| 31 | + var name: String |
| 32 | + var price: Float64 |
24 | 33 |
|
| 34 | + fn __init__(out self): |
| 35 | + self.id = 0 |
| 36 | + self.name = "" |
| 37 | + self.price = 0.0 |
25 | 38 |
|
26 | | -struct Payload(FromReq): |
27 | | - var request: HTTPRequest |
28 | | - var json: JSONType |
29 | | - var a: Int |
30 | 39 |
|
31 | | - def __init__(out self, request: HTTPRequest, json: JSONType): |
32 | | - self.a = 1 |
33 | | - self.request = request.copy() |
34 | | - self.json = json.copy() |
| 40 | +@fieldwise_init |
| 41 | +struct CreateItemRequest(JsonDeserializable, Movable, Defaultable): |
| 42 | + """JSON body expected for POST /items.""" |
35 | 43 |
|
36 | | - def __init__(out self, *, copy: Self): |
37 | | - self.a = copy.a |
38 | | - self.request = copy.request.copy() |
39 | | - self.json = copy.json.copy() |
| 44 | + var name: String |
| 45 | + var price: Float64 |
40 | 46 |
|
41 | | - def __str__(self) -> String: |
42 | | - return String(self.a) |
| 47 | + fn __init__(out self): |
| 48 | + self.name = "" |
| 49 | + self.price = 0.0 |
43 | 50 |
|
44 | | - def from_request(mut self, req: HTTPRequest) raises -> Self: |
45 | | - self.a = 2 |
46 | | - return self.copy() |
47 | 51 |
|
| 52 | +@fieldwise_init |
| 53 | +struct StatusResponse(JsonSerializable, Movable, Defaultable): |
| 54 | + var status: String |
| 55 | + var version: String |
48 | 56 |
|
49 | | -fn custom_request_payload(req: HTTPRequest) raises -> HandlerResponse: |
50 | | - var payload = Payload(request=req, json=JSONType()) |
51 | | - payload = payload.from_request(req) |
52 | | - print(payload.a) |
| 57 | + fn __init__(out self): |
| 58 | + self.status = "" |
| 59 | + self.version = "" |
53 | 60 |
|
54 | | - # Returning a JSON as the response, this is a very limited placeholder for now |
55 | | - var json_response = JSONType() |
56 | | - json_response["a"] = String(payload.a) |
57 | | - return json_response^ |
58 | 61 |
|
| 62 | +# ------------------------------------------------------------------ handlers |
| 63 | + |
| 64 | +fn index(ctx: Context) raises -> HandlerResponse: |
| 65 | + """GET / — plain-text welcome message.""" |
| 66 | + return Response.text("Welcome to lightbug_api 🔥") |
| 67 | + |
| 68 | + |
| 69 | +fn list_items(ctx: Context) raises -> HandlerResponse: |
| 70 | + """GET /items — return a hard-coded list as JSON.""" |
| 71 | + # In a real app you'd query a database here. |
| 72 | + return Response.json(Item(1, "Widget", 9.99)) |
| 73 | + |
| 74 | + |
| 75 | +fn get_item(ctx: Context) raises -> HandlerResponse: |
| 76 | + """GET /items/{id} — return one item by ID.""" |
| 77 | + var id = ctx.path_param("id", "unknown") |
| 78 | + var verbose = ctx.query("verbose", "false") |
| 79 | + |
| 80 | + if verbose == "true": |
| 81 | + print("GET /items/", id, " (verbose mode)") |
| 82 | + |
| 83 | + # Demonstrate that a bare Json value also works as HandlerResponse. |
| 84 | + return Json(Item(42, String("Item ", id), 9.99)) |
| 85 | + |
| 86 | + |
| 87 | +fn create_item(ctx: Context) raises -> HandlerResponse: |
| 88 | + """POST /items — deserialize JSON body, return 201 Created.""" |
| 89 | + var body = ctx.json[CreateItemRequest]() |
| 90 | + var created = Item(100, body.name, body.price) |
| 91 | + return Response.created(created) |
| 92 | + |
| 93 | + |
| 94 | +fn update_item(ctx: Context) raises -> HandlerResponse: |
| 95 | + """PUT /items/{id} — update an item.""" |
| 96 | + var id = ctx.path_param("id", "0") |
| 97 | + var body = ctx.json[CreateItemRequest]() |
| 98 | + return Response.json(Item(42, body.name, body.price)) |
| 99 | + |
| 100 | + |
| 101 | +fn delete_item(ctx: Context) raises -> HandlerResponse: |
| 102 | + """DELETE /items/{id} — delete an item, return 204 No Content.""" |
| 103 | + var id = ctx.path_param("id", "0") |
| 104 | + print("Deleting item", id) |
| 105 | + return Response.no_content() |
| 106 | + |
| 107 | + |
| 108 | +fn health(ctx: Context) raises -> HandlerResponse: |
| 109 | + """GET /v1/status — health check mounted under the v1 sub-router.""" |
| 110 | + return Response.json(StatusResponse("ok", "1.0.0")) |
| 111 | + |
| 112 | + |
| 113 | +# --------------------------------------------------------------------- main |
59 | 114 |
|
60 | 115 | fn main() raises: |
61 | 116 | var app = App() |
62 | 117 |
|
63 | | - app.get("/", hello) |
64 | | - |
65 | | - app.get("custom/", custom_request_payload) |
| 118 | + # Root |
| 119 | + app.get("/", index) |
66 | 120 |
|
67 | | - app.post("/", printer) |
| 121 | + # Items resource — all HTTP verbs |
| 122 | + app.get("/items", list_items) |
| 123 | + app.get("/items/{id}", get_item) |
| 124 | + app.post("/items", create_item) |
| 125 | + app.put("/items/{id}", update_item) |
| 126 | + app.delete("/items/{id}", delete_item) |
68 | 127 |
|
69 | | - var nested_router = Router("nested") |
70 | | - nested_router.get(path="all/echo/", handler=nested) |
71 | | - app.add_router(nested_router^) |
| 128 | + # Sub-router mounted at /v1 |
| 129 | + var v1 = Router("v1") |
| 130 | + v1.get("status", health) |
| 131 | + app.add_router(v1^) |
72 | 132 |
|
73 | | - app.start_server() |
| 133 | + app.run() |
0 commit comments