mirror of
https://github.com/roc-lang/roc.git
synced 2025-09-29 14:54:47 +00:00
84 lines
2.2 KiB
Text
84 lines
2.2 KiB
Text
app "http"
|
|
packages { pf: "https://github.com/roc-lang/basic-webserver/releases/download/0.1/dCL3KsovvV-8A5D_W_0X_abynkcRcoAngsgF0xtvQsk.tar.br" }
|
|
imports [
|
|
pf.Stdout,
|
|
pf.Stderr,
|
|
pf.Task.{ Task },
|
|
pf.Http.{ Request, Response },
|
|
pf.Utc,
|
|
pf.Env,
|
|
]
|
|
provides [main] to pf
|
|
|
|
main : Request -> Task Response []
|
|
main = \req ->
|
|
|
|
handleReq =
|
|
# Log the date, time, method, and url to stdout
|
|
{} <- logRequest req |> Task.await
|
|
|
|
# Read environment variable
|
|
url <- readUrlEnv "TARGET_URL" |> Task.await
|
|
|
|
# Fetch the Roc website
|
|
content <- fetchContent url |> Task.await
|
|
|
|
# Respond with the website content
|
|
respond 200 content
|
|
|
|
# Handle any application errors
|
|
handleReq |> Task.onErr handleErr
|
|
|
|
AppError : [
|
|
EnvURLNotFound,
|
|
HttpError Http.Error,
|
|
]
|
|
|
|
logRequest : Request -> Task {} AppError
|
|
logRequest = \req ->
|
|
dateTime <- Utc.now |> Task.map Utc.toIso8601Str |> Task.await
|
|
|
|
Stdout.line "\(dateTime) \(Http.methodToStr req.method) \(req.url)"
|
|
|
|
readUrlEnv : Str -> Task Str AppError
|
|
readUrlEnv = \target ->
|
|
Env.var target
|
|
|> Task.mapErr \_ -> EnvURLNotFound
|
|
|
|
fetchContent : Str -> Task Str AppError
|
|
fetchContent = \url ->
|
|
Http.getUtf8 url
|
|
|> Task.mapErr \err -> HttpError err
|
|
|
|
handleErr : AppError -> Task Response []
|
|
handleErr = \err ->
|
|
|
|
# Build error message
|
|
message =
|
|
when err is
|
|
EnvURLNotFound -> "TARGET_URL environment variable not set"
|
|
HttpError _ -> "Http error fetching content"
|
|
|
|
# Log error to stderr
|
|
{} <- Stderr.line "Internal Server Error: \(message)" |> Task.await
|
|
_ <- Stderr.flush |> Task.attempt
|
|
|
|
# Respond with Http 500 Error
|
|
Task.ok {
|
|
status: 500,
|
|
headers: [
|
|
{ name: "Content-Type", value: Str.toUtf8 "text/html; charset=utf-8" },
|
|
],
|
|
body: Str.toUtf8 "Error 500 Internal Server Error\n",
|
|
}
|
|
|
|
# Respond with the given status code and body
|
|
respond : U16, Str -> Task Response AppError
|
|
respond = \code, body ->
|
|
Task.ok {
|
|
status: code,
|
|
headers: [
|
|
{ name: "Content-Type", value: Str.toUtf8 "text/html; charset=utf-8" },
|
|
],
|
|
body: Str.toUtf8 body,
|
|
}
|