Jay 1.0.0
See the version list below for details.
dotnet add package Jay --version 1.0.0
NuGet\Install-Package Jay -Version 1.0.0
<PackageReference Include="Jay" Version="1.0.0" />
paket add Jay --version 1.0.0
#r "nuget: Jay, 1.0.0"
// Install Jay as a Cake Addin #addin nuget:?package=Jay&version=1.0.0 // Install Jay as a Cake Tool #tool nuget:?package=Jay&version=1.0.0
Jay
JSON in F# is tough and confusing. This library aims to make it much simpler.
Getting Started
This will be completed once released on NuGet. Stay tuned.
An Example
Let's consider a stripped down Tweet object.
Note: the
user
andentities
properties have been removed for clarity.
{
"created_at": "Wed Oct 10 20:19:24 +0000 2018",
"id": 1050118621198921728,
"id_str": "1050118621198921728",
"text": "To make room for more expression, we will now count all emojis as equal—including those with gender and skin t… https://t.co/MkGjXf9aXm",
}
In order to work with this in our F# program, we'll first need to create a record type.
type Tweet =
{
CreatedAt : DateTimeOffset
Id : int64
IdStr : string
Text : string
}
Next we'll define a static type extension called FromJson
to consume our JSON and return a Tweet
record.
type Tweet =
{
CreatedAt : DateTimeOffset
Id : int64
IdStr : string
Text : string
}
static member FromJson (json : Json) =
{
CreatedAt = json?created_at.AsDateTimeOffset()
Id = json?id.AsInt64()
IdStr = json?idStr.AsString()
Text = json?text.AsString()
}
let tweetJson = ... // JSON from above
let tweet =
tweetJson
|> Json.parse
|> Tweet.FromJson
Finally, we'll create another static type extension ToJson
to convert our record back into JSON represented as an abstract syntax tree.
type Tweet =
{
CreatedAt : DateTimeOffset
Id : int64
IdStr : string
Text : string
}
static member FromJson (json : Json) =
{
CreatedAt = json?created_at.AsDateTimeOffset()
Id = json?id.AsInt64()
IdStr = json?idStr.AsString()
Text = json?text.AsString()
}
static member ToJson (tweet : Tweet) =
Json.Object
[|
"created_at", Json.String (tweet.CreatedAt.ToString())
"id", Json.Number (float tweet.Id)
"id_str", Json.String tweet.IdStr
"text", Json.String tweet.Text
|]
let tweetJson = ... // JSON from above
let tweet =
tweetJson
|> Json.parse
|> Tweet.FromJson
let json =
tweet
|> Tweet.ToJson
|> Json.serialize
And that's it! Not so painful was it?
A little background
If you're a newcomer to F#, it can be confusing how to handle JSON since there is no commonly accepted approach. Especially if you're coming from C# where you'd normally rely on a reflection-based deserializer. Not requiring you to do anything other than define a class for the JSON to be deserialized into.
Of course F# has JSON type providers, which effectively amount to the same result. Except they generate the types for you based on sampling the actual JSON you're intending to consume. If you're doing quick-and-dirty exploratory work, type providers are immensely useful.
F# being so terse, it actually turns out that it is incredible practical to map your own JSON (as well as any other IO entry points - see Donald). At first this sounds crazy. But if you consider that these IO boundaries are common places for faults. It likely behoves you to be explicit in those places.
The aim of this library was to take the core JSON parser from the amazing FSharp.Data project, and modernize/simplify it's API. The hopes of this effort is to make JSON more approachable and easier to reason about for newcomers to the language.
Find a bug?
There's an issue for that.
License
Built with ♥ by Pim Brouwers in Toronto, ON. Licensed under Apache License 2.0.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. 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. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
.NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.0
- FSharp.Core (>= 4.7.0)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.