Inertial.Giraffe
0.0.1
dotnet add package Inertial.Giraffe --version 0.0.1
NuGet\Install-Package Inertial.Giraffe -Version 0.0.1
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Inertial.Giraffe" Version="0.0.1" />
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Inertial.Giraffe --version 0.0.1
The NuGet Team does not provide support for this client. Please contact its maintainers for support.
#r "nuget: Inertial.Giraffe, 0.0.1"
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install Inertial.Giraffe as a Cake Addin #addin nuget:?package=Inertial.Giraffe&version=0.0.1 // Install Inertial.Giraffe as a Cake Tool #tool nuget:?package=Inertial.Giraffe&version=0.0.1
The NuGet Team does not provide support for this client. Please contact its maintainers for support.
Inertial.Giraffe
What is this?
You're looking at an attempt (by an FSharp novice) to bring InertiaJS to the FSharp ecosystem with a few additional bells and whistles.
This library is a Giraffe server plugin that reads the Http headers of incoming requests from a Sutil app using the complementary Inertial.Client library and determines how to respond accordingly.
In Detail
To make this work you need to do the following:
- Create a Giraffe app like normal
- Install the Inertial.Giraffe package from nuget
- Create a Netstandard2.0 library project in the same solution with the server-side app that sits "above" both the Server app and the Client app in the same solution. Make sure both server and client reference this project. In this library (let's call it the "Common" library for demonstration purposes) you define the app domain with top-level "Props" and "Shared" types that the server and client will both reference. Props are types describing data available for a given "page". Shared is a type describing data that gets made available to all components all of the time -- think details of the signed-in user or flash messages. The top-level types also need functions (or as below static methods) that decode themselves from JSON. I chose to use Thoth.Json for this task because it has reliable "auto" decoders and for more complex scenarios allows for very intuitive composition. The decoder function for the top-level Props DU should take in a string name that is pattern matched to determine which child decoder is needed (see below for an example implementation):
open Thoth.Json // some helper functions for our decoders module Helpers = let resultDecoder<'T> (decoder: Decoder<'T>) = let decoder: Decoder<Result<'T,string>> = let decodeOK = Decode.field "Ok" decoder |> Decode.map Ok let decodeError = Decode.field "Error" (Decode.string) |> Decode.map Error Decode.oneOf [ decodeOK ; decodeError ] decoder let asyncChoice2Decoder<'T> (placeholder : Async<Result<'T,string>>) (decoder: Decoder<'T>) = let decoder = let decodeChoice1 = Decode.field "Choice1Of2" (emptyDecoder |> Decode.andThen (fun _ -> Decode.succeed (Choice1Of2 placeholder) )) let decodeChoice2 = Decode.field "Choice2Of2" (resultDecoder decoder) |> Decode.map Choice2Of2 Decode.oneOf [ decodeChoice1 ; decodeChoice2 ] decoder type Widget { name: string description: string } static member decoder = Decode.Auto.generateDecoder<Widget>() type User = { email : string username : string } static member decoder = Decode.Auto.generateDecoder<User option>() static member encoder userOpt = Encode.Auto.generateEncoder<User option>() type IndexPage = { widgets : Widget list asyncWidgets : Choice<Async<Result<Widget list,string>>,Result<Widget list,string>>> } static member decoder = Decode.object (fun get -> { widgets = get.Required.Field "widgets" (Decode.list Widget.decoder) asyncWidgets = get.Required.Field "asyncWidgets" (Helpers.asyncChoice2Decoder (async { return Ok <| [] }) // this is a placeholder that has the same type signature Widget.decoder) }) type Props = | Index of IndexPage static member index = nameof Props.Index static member decoder (name: string) : Decoder<Props option> = // note that this decoder is a function that takes a string matching the component name // and returns a decoder that has been mapped back to Option<Props> match name with | name when name = Props.index -> IndexPage.decoder |> Decode.map Index |> Decode.map Some | notFound -> failwith $"Could not find matching decoder for component named: {notFound}" type Shared = { user : User option flashMessage = string option } let extra = Extra.empty |> Extra.withCustom User.encoder User.decoder Decode.Auto.generateDecoder<Shared option>(extra=extra)
- Create a Sutil app and the Inertial.Client package see here
- In your Giraffe startup, add the Inertia Service to your IServiceCollection. The services.AddInertia extension function has the following type signature:
services.AddInertia<'Props,'Shared,'SSE>(
svc : IServiceCollection,
jsPath: string,
cssPath: string,
resolver: 'Props -> string,
shareFn: HttpContext -> Task<'Shared>,
sseInit:'SSE)
- 'Props and 'Shared are defined from Props and Shared types in the "Common" library and 'SSE generic type refers to the server sent messsage type that is observed -- most often just some JSON in string form
- The jsPath and cssPath arguments refer to the path of the js app bundle and cssPath bundle in the wwwroot folder of the web app. This should be customized to your specific scenario.
- The resolver function takes in a 'Props and returns the string name of the associated component. In my example below I attached an instance method to the 'Props type that pattern matches and returns the name:
// based on this type definition in your "Common" library project
type Props =
| Index of Index
member x.name =
match x with
| Index _ -> Props.index
static member index = nameof Props.Index
// you can use the following as your "resolver" argument:
let resolverFn = fun prop -> prop.name
- The shareFunction takes the HttpContext and returns an instance of the 'Shared type wrapped in Task. The idea here is that this function runs each time the app loads a component and it pulls in data that should be included in each view, think signed in user data or flash messages. This is an example implementation assuming the use of Asp.Identity:
// assuming the following Shared type in "Common" project
type Shared =
{
user: User option
}
and User =
{
email : string
id : string
}
open Inertial.Giraffe
module SharedHandler =
let shareFn (ctx:HttpContext) =
task {
if ctx.User.Identity.IsAuthenticated then
let userIdClaim = ctx.User.Claims |> Seq.tryFind (fun x -> x.Type = ClaimTypes.NameIdentifier)
match userIdClaim with
| Some claim ->
// lookup user using identity manager
let userManager = ctx.GetService<UserManager<IdentityUser>>()
let! user = userManager.FindByIdAsync claim.Value
// data gets shared across current request/response
return { user = Some { email = user.Email; id = claim.Value } }
| None ->
return { user = None }
else
return { user = None }
}
- The sseInit argument is the starting message. If you 'SSE type is string this could be an empty "". If you are using a more complex JSON string to pass messages it might be some initialized record type converted to JSON like below:
open Fable.Remoting.Json
open Newtonsoft.Json
// assuming you define a fableConverter
let fableConverter = FableJsonConverter() :> JsonConverter
// if you have a custom type
type InertialSSEEvent =
{
title : string
connectionId: string option
predicates : Predicates
firedOn : System.DateTime
}
member x.toJson() = JsonConvert.SerializeObject(x,converters=[|fableConverter|])
static member empty () =
{ title = ""; connectionId = None; predicates = { predicates = [||]; propsToEval = EvalAllProps } ; firedOn = System.DateTime.UtcNow }.toJson()
// then your sseInit would be
let sseInit = InertialSSEEvent.empty()
- Putting it all together you can add Inertia to the app:
// include this line to your startup when adding services to you IServiceCollection
services.AddInertia<Props,Shared,string>("/js/App.js","/css/style.css",(_.name),SharedHandler.shareFn,InertialSSEEvent.empty()) |> ignore
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net8.0 is compatible. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
-
net8.0
- Fable.Remoting.Json (>= 2.25.0)
- FSharp.Control.Reactive (>= 5.0.5)
- FSharp.Core (>= 8.0.401)
- Giraffe (>= 7.0.2)
- Microsoft.FSharpLu.Json (>= 0.11.7)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
0.0.1 | 83 | 12/9/2024 |