The Server
Every request to your site begins in server/src/Main.gren
.
If you generated your project with prettynice init
, it will look something like this:
module Main exposing (main)
import Gen.Components.Counterimport Node exposing (Environment)import Prettyniceimport Prettynice.Request exposing (Request)import Prettynice.Response as Response exposing (Response)import Task exposing (Task)import Transmutable.Html as H exposing (Html)import Transmutable.Html.Attributes as A
main : Prettynice.Program Model {}main = Prettynice.defineProgram { init = init , router = router }
type alias Model = {}
init : Environment -> Prettynice.Init Model {}init env = Prettynice.startProgram { env = env , host = "localhost" , port_ = 3000 , model = {} }
router : Model -> Request -> Response msg -> Task Never (Response msg)router model request response = when request.path is
[] -> response |> viewHome |> Task.succeed
[ "hello", name ] -> response |> Response.asText ("Hello, " ++ name) |> Task.succeed
_ -> response |> Response.setStatus 404 |> Response.asText "not found" |> Task.succeed
viewHome : Response msg -> Response msgviewHome = Response.asHtml { title = "My Website" , head = [ linkStylesheet ] , body = [ H.h1 [] [ H.text "Welcome to my website!" ] , H.p [] [ H.text "I made this counter just for you." ] , Gen.Components.Counter.init { start = 0 } ] }
linkStylesheet : Html msglinkStylesheet = H.link [ A.rel "stylesheet", A.href "/styles.css" ]
If you aren’t familiar with Gren, it would be a good idea to read through the book ↗ first. The rest of this guide will assume basic familiarity with Gren and focus on the parts specific to Prettynice.
Imports
module Main exposing (main)
import Gen.Components.Counterimport Node exposing (Environment)import Prettyniceimport Prettynice.Request exposing (Request)import Prettynice.Response as Response exposing (Response)import Task exposing (Task)import Transmutable.Html as H exposing (Html)import Transmutable.Html.Attributes as A
We start with the usual module definition ↗ followed by imports ↗ for the modules we’ll need for our server.
We’re aliasing the Html
module as H
and the Html.Attributes
module as A
make them easier to write, since they will be used often.
The rest of these docs will be using those aliases, but you can call them whatever you want in your own code.
See the imports section of the Gren book ↗
for more information on importing and aliasing modules.
Defining your Program
We start with a main
function that defines a Program
↗.
using defineProgram
↗:
main : Prettynice.Program Model {}main = Prettynice.defineProgram { init = init , router = router }
This will look similar to the main
definition of a
Gren node program ↗.
That’s because a Prettynice server is a Gren node program!
But instead of update
and subscription
functions, it takes a router
.
The router is a function dedicated to handling HTTP requests (we’ll cover it in detail later),
and for most situations, that’s all you’ll need for a Prettynice server.
But if you do want update
and subscriptions
, take a look at
defineFullProgram
↗,
which gives you the full power of a Gren node program, and the
server-side-tea example ↗
to see it in action.
Defining your Model
Next we define a type alias ↗ for our model:
type alias Model = {}
The model is used to hold server-side, runtime state. That’s not something we need for this example, so we’re leaving it empty. See the server-side-tea ↗ and websockets ↗ examples for apps that use this feature.
Starting your Program
Your init
function should call startProgram
↗.
This is where you define the host and port the server will listen on:
init : Environment -> Prettynice.Init Model {}init env = Prettynice.startProgram { env = env , host = "localhost" , port_ = 3000 , model = {} }
Here we’re starting a server that will be available on http://localhost:3000 ↗. This is also where we define our initial model value. But again, we’re not using that here, so we set it to an empty record.
For a more complex program, we might need to initialize some subsystems ↗ here and save subsystem permissions to our model so they can be used in the router. See the bytes example ↗ for an app that does that.
Routing Requests
The heart of your server is the router function. Every request to your site will call this function, with the model you defined in init, a request ↗ holding information about the request, and a response ↗ that you can use to respond to the request:
router : Model -> Request -> Response msg -> Task Never (Response msg)router model request response = when request.path is [] -> response |> viewHome |> Task.succeed
[ "hello", name ] -> response |> Response.asText ("Hello, " ++ name) |> Task.succeed
_ -> response |> Response.setStatus 404 |> Response.asText "not found" |> Task.succeed
To route the request, we’re pattern matching ↗
and destructuring ↗ the request’s path.
request.path
is an array of strings. It holds the path portion of the URL split on /
.
For example, requesting http://localhost:3000/bowties/are/cool
would have a request.path
of ["bowties", "are", "cool"]
.
Requesting the root path would have an empty array ([]
).
For the root path ([]
) we are deferring to a viewHome
function that we’ll cover later.
For paths like /hello/mary
or /hello/sam
we capture the second part of the path in a name
variable,
and respond with plain text.
For all other paths (the _
catchall) we respond with a 404 error.
You can see that the router returns a Task
:
router : Model -> Request -> Response msg -> Task Never (Response msg)
A task ↗
is how Gren describes an asynchronous operation that can possibly fail.
This includes things like hitting the filesystem or an HTTP API.
Prettynice expects a task from the router so you can do things like this as part of your request/response handling.
Since we don’t need anything like that for this example, we just use
Task.succeed
↗.
Tasks describe both the error type that is possible, and the type of value that the task resolves to if there is no error.
The router’s return type is Task Never (Response msg)
which says an error should never be possible, and the router always
resolves to a response. This means if you are using tasks that can fail, you’ll need to handle those errors so it still resolves to a response (e.g. a 500 error response).
See the database example ↗
for an app that does this.
The Response
type has an associated msg
type variable ↗.
This would be relevant if we were using Prettynice.defineFullProgram
↗,
in which case we’d have a proper Msg
type and an update
function, but we don’t need that for this example,
so you can safely ignore it.
See Prettynice.Response
↗
for all the things you can do with a response.
Responding with HTML
For the homepage, we’re using Response.asHtml ↗ to render HTML:
viewHome : Response msg -> Response msgviewHome = Response.asHtml { title = "My Website" , head = [ linkStylesheet ] , body = [ H.h1 [] [ H.text "Welcome to my website!" ] , H.p [] [ H.text "I made this counter just for you." ] , Gen.Components.Counter.init { start = 0 } ] }
Because prettynice needs some control over the <head>
and <body>
tags, you can’t specify them directly,
but you have some fields to inject what you need into the page:
title
: Sets the<title>
tag. Here we’re rendering<title>My Website</title>
head
: An array of HTML that will be injected into the<head>
tag. Here we’re deferring to alinkStylesheet
function to link to a stylesheet. That will be covered in a later section. In a real app you might also want to specify some meta tags ↗ here.body
: An array of HTML to make up the<body>
of your page. Here we’re rendering a header, a paragraph tag, and a component that we will cover in the next section.
Initialize Components
The viewHome
function is referencing a component module:
, Gen.Components.Counter.init { start = 0 }
Where did this come from?
When you ran prettynice init
, it created an example component at client/src/Components/Counter.gren
.
Take a look at that file. It works just like a normal Browser.element
↗,
but it has a special Props
record.
When you run build
, prettynice looks at all the .gren
files in your
client/src/Components
directory and generates corresponding modules that will be available in your server code at
Gen.Components.[ComponentName]
. The module will have an init
function that takes an argument matching your Props
definition
and returns the HTML and Javascript needed to run your component in the browser.
This means you can create mini browser apps ↗ that can be embedded in your server-rendered HTML, loaded and ready with server-side data instead of fetching it client-side.
See the client-side-components example ↗ or the skip ahead to learn more about client-side components.
Assets and Styling
In the linkStylesheet
function we’re linking to /styles.css
:
linkStylesheet : Html msglinkStylesheet = H.link [ A.rel "stylesheet", A.href "/styles.css" ]
styles.css
was created by prettynice init
and placed in the public/
directory.
Any files in public/
will be accessible at the root url path.
So you can put other things there like images, fonts, etc. and reference them in your HTML anywhere you need them.