Oxygen is a micro-framework built on top of the HTTP.jl library. Breathe easy knowing you can quickly spin up a web server with abstractions you're already familiar with.
pkg> add Oxygen
Create a web-server with very few lines of code
using Oxygen
using HTTP
@get "/greet" function(req::HTTP.Request)
return "hello world!"
end
# start the web server
serve()
Request handlers are just functions, which means there are many valid ways to express them
using Oxygen
@get "/greet" function()
"hello world!"
end
@get "/saluer" () -> begin
"Bonjour le monde!"
end
@get "/saludar" () -> "¡Hola Mundo!"
@get "/salutare" f() = "ciao mondo!"
@get "/emoji" 😎() = "please don't name functions like this"
# start the web server
serve()
Path parameters are declared with braces and are passed directly to your request handler.
using Oxygen
# use path params without type definitions (defaults to Strings)
@get "/add/{a}/{b}" function(req, a, b)
return parse(Float64, a) + parse(Float64, b)
end
# use path params with type definitions (they are automatically converted)
@get "/multiply/{a}/{b}" function(req, a::Float64, b::Float64)
return a * b
end
# The order of the parameters doesn't matter (just the name matters)
@get "/subtract/{a}/{b}" function(req, b::Int64, a::Int64)
return a - b
end
# start the web server
serve()
Use the queryparams()
function to extract and parse parameters from the url
using Oxygen
using HTTP
@get "/query" function(req::HTTP.Request)
# extract & return the query params from the request object
return queryparams(req)
end
# start the web server
serve()
You can interpolate variables directly into the paths, which makes dynamically registering routes a breeze
(Thanks to @anandijain for the idea)
using Oxygen
operations = Dict("add" => +, "multiply" => *)
for (pathname, operator) in operations
@get "/$pathname/{a}/{b}" function (req, a::Float64, b::Float64)
return operator(a, b)
end
end
# start the web server
serve()
All objects are automatically deserialized into JSON using the JSON3 library
using Oxygen
using HTTP
@get "/data" function(req::HTTP.Request)
return Dict("message" => "hello!", "value" => 99.3)
end
# start the web server
serve()
Oxygen provides some out-of-the-box serialization & deserialization for most objects but requires the use of StructTypes when converting structs
using Oxygen
using HTTP
using StructTypes
struct Animal
id::Int
type::String
name::String
end
# Add a supporting struct type definition so JSON3 can serialize & deserialize automatically
StructTypes.StructType(::Type{Animal}) = StructTypes.Struct()
@get "/get" function(req::HTTP.Request)
# serialize struct into JSON automatically (because we used StructTypes)
return Animal(1, "cat", "whiskers")
end
@post "/echo" function(req::HTTP.Request)
# deserialize JSON from the request body into an Animal struct
animal = json(req, Animal)
# serialize struct back into JSON automatically (because we used StructTypes)
return animal
end
# start the web server
serve()
For scenarios where you need to handle higher amounts of traffic, you can run Oxygen in a multithreaded mode. In order to utilize this mode, julia must have more than 1 thread to work with. You can start a julia session with 4 threads using the command below
julia --threads 4
serveparallel(queuesize=1024)
Starts the webserver in streaming mode and spawns n - 1 worker
threads. The queuesize
parameter sets how many requests can be scheduled within the queue (a julia Channel)
before they start getting dropped. Each worker thread pops requests off the queue and handles them asynchronously within each thread.
using Oxygen
using StructTypes
using Base.Threads
# Make the Atomic struct serializable
StructTypes.StructType(::Type{Atomic{Int64}}) = StructTypes.Struct()
x = Atomic{Int64}(0);
@get "/show" function()
return x
end
@get "/increment" function()
atomic_add!(x, 1)
return x
end
# start the web server in parallel mode
serveparallel()
You can integrate swagger into your api using some of the other great tools from the julia ecosystem.
using Oxygen
using SwagUI
using SwaggerMarkdown
@swagger """
/divide/{a}/{b}:
get:
description: Return the value of a / b
parameters:
- name: a
in: path
type: number
required: true
- name: b
in: path
type: number
required: true
responses:
'200':
description: Successfully returned a number
"""
@get "/divide/{a}/{b}" function (req, a::Float64, b::Float64)
return a / b
end
# title and version are required
info = Dict("title" => "My Demo Api", "version" => "1.0.0")
openApi = OpenAPI("2.0", info)
swagger_document = build(openApi)
swagger_html = render_swagger(swagger_document)
# setup endpoint to serve swagger documentation
@get "/swagger" function()
return html(swagger_html)
end
# start the web server
serve()
You can mount static files using this handy macro which recursively searches a folder for files and mounts everything. All files are loaded into memory on startup.
using Oxygen
# mount all files inside the "content" folder under the "/static" path
@staticfiles "content" "static"
# start the web server
serve()
Similar to @staticfiles, this macro mounts each path and re-reads the file for each request. This means that any changes to the files after the server has started will be displayed.
using Oxygen
# mount all files inside the "content" folder under the "/dynamic" path
@dynamicfiles "content" "dynamic"
# start the web server
serve()
@get(path, func)
Parameter | Type | Description |
---|---|---|
path |
string |
Required. The route to register |
func |
function |
Required. The request handler for this route |
Used to register a function to a specific endpoint to handle that corresponding type of request
@route(methods, path, func)
Parameter | Type | Description |
---|---|---|
methods |
array |
Required. The types of HTTP requests to register to this route |
path |
string |
Required. The route to register |
func |
function |
Required. The request handler for this route |
Low-level macro that allows a route to be handle multiple request types
@staticfiles(folder, mount)
Parameter | Type | Description |
---|---|---|
folder |
string |
Required. The folder to serve files from |
mountdir |
string |
The root endpoint to mount files under (default is "static") |
Serve all static files within a folder. This function recursively searches a directory and mounts all files under the mount directory using their relative paths.
html(content, status, headers)
Parameter | Type | Description |
---|---|---|
content |
string |
Required. The string to be returned as HTML |
status |
integer |
The HTTP response code (default is 200) |
headers |
dict |
The headers for the HTTP response (default has content-type header set to "text/html; charset=utf-8") |
Helper function to designate when content should be returned as HTML
queryparams(request)
Parameter | Type | Description |
---|---|---|
req |
HTTP.Request |
Required. The HTTP request object |
Returns the query parameters from a request as a Dict()
text(request)
Parameter | Type | Description |
---|---|---|
req |
HTTP.Request |
Required. The HTTP request object |
Returns the body of a request as a string
binary(request)
Parameter | Type | Description |
---|---|---|
req |
HTTP.Request |
Required. The HTTP request object |
Returns the body of a request as a binary file (returns a vector of UInt8
s)
json(request, classtype)
Parameter | Type | Description |
---|---|---|
req |
HTTP.Request |
Required. The HTTP request object |
classtype |
struct |
A struct to deserialize a JSON object into |
Deserialize the body of a request into a julia struct