Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/soundtrackyourbrand/forma
Typespec based parsing of JSON-like data for Elixir
https://github.com/soundtrackyourbrand/forma
elixir json parsing typespec
Last synced: about 2 months ago
JSON representation
Typespec based parsing of JSON-like data for Elixir
- Host: GitHub
- URL: https://github.com/soundtrackyourbrand/forma
- Owner: soundtrackyourbrand
- License: mit
- Created: 2017-10-17T21:55:58.000Z (over 7 years ago)
- Default Branch: master
- Last Pushed: 2019-11-29T16:35:24.000Z (about 5 years ago)
- Last Synced: 2024-12-06T13:32:23.972Z (about 2 months ago)
- Topics: elixir, json, parsing, typespec
- Language: Elixir
- Size: 29.3 KB
- Stars: 27
- Watchers: 11
- Forks: 1
- Open Issues: 3
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
**forma**
_[swe] verb. /fạ̊r:ma/_
to adjust; adapt[![Build Status](https://travis-ci.org/soundtrackyourbrand/forma.svg?branch=master)](https://travis-ci.org/soundtrackyourbrand/forma)
# forma
Applies typespecs to JSON-like data.
This module can parse JSON-like data (such as maps with key strings)
into a more structured form by trying to map it to conform to a
module's typespec.This can generally be useful when interfacing with external data
sources that provide you data as JSON or MessagePack, but that you
wish to transform into either proper structs or richer data types
without a native JSON representation (such as dates or sets) in
your application.It is heavily inspired by Go's way of dealing with JSON data.
```elixir
defmodule User do
defstruct [:id, :name, :age, :gender]@type t :: %__MODULE__{
id: String.t,
name: String.t,
age: non_neg_integer(),
gender: :male | :female | :other | :prefer_not_to_say
}
endForma.parse(%{"id" => "1", "name" => "Fredrik", "age" => 30, "gender" => "male"}, User)
# => {:ok, %User{id: "1", name: "Fredrik", age: 30, gender: :male}}
```Forma tries to figure out how to translate its input to a typespec. However, not all
types have natural representations in JSON, for example dates, or don't want to expose
their internals (opaque types).If you're in control of the module defining the type, you can implement the `__forma__/2`
function to handle parsing input to your desired type```elixir
defmodule App.Date do
@opaque t :: Date# first argument is the type to be parsed in this module
def __forma__(:t, input) do
case Date.from_iso8601(input) do
{:ok, date} -> date
{:error, reason} -> raise reason
end
end
end
```If you're not in control of the module, you can pass a parser along as an optional
argument,```elixir
defmodule LogRow do
defstruct [:log, :timestamp]type t :: %__MODULE__{
log: String.t,
timestamp: NaiveDateTime.t
}
enddate = fn input ->
case NaiveDateTime.from_iso8601(input) do
{:ok, datetime} -> datetime
{:error, err} -> raise err
end
end
parsers = %{{NaiveDateTime, :t} => date}
Forma.parse(%{"log" => "An error occurred", "timestamp" => "2015-01-23 23:50:07"}, LogRow, parsers)
```The number of arguments to the parser functions depends on if the type is parameterized
or not (`MapSet.t` vs `MapSet.t(integer)`).## Installation
If [available in Hex](https://hex.pm/docs/publish), the package can be installed
by adding `forma` to your list of dependencies in `mix.exs`:```elixir
def deps do
[
{:forma, "~> 0.7.1"}
]
end
```Documentation can be generated with [ExDoc](https://github.com/elixir-lang/ex_doc)
and published on [HexDocs](https://hexdocs.pm). Once published, the docs can
be found at [https://hexdocs.pm/forma](https://hexdocs.pm/forma).