Initial commit

This commit is contained in:
LilyRose2798 2024-03-21 22:05:04 +11:00
commit f07be54064
7 changed files with 399 additions and 0 deletions

23
.github/workflows/test.yml vendored Normal file
View file

@ -0,0 +1,23 @@
name: test
on:
push:
branches:
- master
- main
pull_request:
jobs:
test:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- uses: erlef/setup-beam@v1
with:
otp-version: "26.0.2"
gleam-version: "1.0.0"
rebar3-version: "3"
# elixir-version: "1.15.4"
- run: gleam deps download
- run: gleam test
- run: gleam format --check src test

4
.gitignore vendored Normal file
View file

@ -0,0 +1,4 @@
*.beam
*.ez
/build
erl_crash.dump

28
README.md Normal file
View file

@ -0,0 +1,28 @@
# jasper
[![Package Version](https://img.shields.io/hexpm/v/jasper)](https://hex.pm/packages/jasper)
[![Hex Docs](https://img.shields.io/badge/hex-docs-ffaff3)](https://hexdocs.pm/jasper/)
```sh
gleam add jasper
```
```gleam
import gleam/io
import jasper.{parse_json, query_json, String, Root, Key, Index}
pub fn main() {
let assert Ok(json) = parse_json("{ \"foo\": [1, true, \"hi\"] }")
let assert Ok(String(str)) = query_json(json, Root |> Key("foo") |> Index(2))
io.println(str)
}
```
Further documentation can be found at <https://hexdocs.pm/jasper>.
## Development
```sh
gleam run # Run the project
gleam test # Run the tests
gleam shell # Run an Erlang shell
```

13
gleam.toml Normal file
View file

@ -0,0 +1,13 @@
name = "jasper"
version = "1.0.0"
description = "Utilities for parsing and querying JSON data"
licences = ["AGPL-3.0-only"]
repository = { type = "github", user = "LilyRose2798", repo = "jasper" }
[dependencies]
gleam_stdlib = "~> 0.34 or ~> 1.0"
pears = "~> 0.3"
[dev-dependencies]
gleeunit = "~> 1.0"

13
manifest.toml Normal file
View file

@ -0,0 +1,13 @@
# This file was generated by Gleam
# You typically do not need to edit this file
packages = [
{ name = "gleam_stdlib", version = "0.36.0", build_tools = ["gleam"], requirements = [], otp_app = "gleam_stdlib", source = "hex", outer_checksum = "C0D14D807FEC6F8A08A7C9EF8DFDE6AE5C10E40E21325B2B29365965D82EB3D4" },
{ name = "gleeunit", version = "1.0.2", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "gleeunit", source = "hex", outer_checksum = "D364C87AFEB26BDB4FB8A5ABDE67D635DC9FA52D6AB68416044C35B096C6882D" },
{ name = "pears", version = "0.3.0", build_tools = ["gleam"], requirements = ["gleam_stdlib"], otp_app = "pears", source = "hex", outer_checksum = "F823EDF5C7F8606A0B7764C071B6EE8515FC341D6FC69902E81120633DF7E983" },
]
[requirements]
gleam_stdlib = { version = "~> 0.34 or ~> 1.0" }
gleeunit = { version = "~> 1.0" }
pears = { version = "~> 0.3"}

197
src/jasper.gleam Normal file
View file

@ -0,0 +1,197 @@
import gleam/float
import gleam/int
import gleam/string
import gleam/result
import gleam/option.{None, Some}
import gleam/dict.{type Dict}
import gleam/list
import pears.{type Parser}
import pears/chars.{type Char, digit, string}
import pears/combinators.{
alt, between, choice, eof, just, lazy, left, many0, many1, map, maybe, none_of,
one_of, pair, recognize, right, sep_by0, seq, to,
}
pub type JsonObject =
Dict(String, JsonValue)
pub type JsonArray =
List(JsonValue)
pub type JsonValue {
Object(JsonObject)
Array(JsonArray)
String(String)
Number(Float)
Boolean(Bool)
Null
}
fn whitespace0() -> Parser(Char, List(Char)) {
one_of([" ", "\n", "\r", "\t"])
|> many0()
}
fn value_parser() -> Parser(Char, JsonValue) {
let padded = fn(parser: Parser(_, a)) { left(parser, whitespace0()) }
let symbol = fn(s: String) { padded(string(s)) }
let hex_digit =
one_of([
"0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "a", "b", "c", "d", "e",
"f", "A", "B", "C", "D", "E", "F",
])
let unicode_escape_digits =
recognize(seq([hex_digit, hex_digit, hex_digit, hex_digit]))
let escape =
just("\\")
|> right(
choice([
just("\\"),
just("/"),
just("\""),
to(just("b"), "\u{0008}"),
to(just("f"), "\u{000C}"),
to(just("n"), "\n"),
to(just("r"), "\r"),
to(just("t"), "\t"),
map(right(just("u"), unicode_escape_digits), fn(value) {
let assert Ok(number) = int.base_parse(string.concat(value), 16)
let assert Ok(codepoint) = string.utf_codepoint(number)
string.from_utf_codepoints([codepoint])
}),
]),
)
let str =
none_of(["\""])
|> alt(escape)
|> many0()
|> map(string.concat)
|> between(just("\""), just("\""))
let value = lazy(value_parser)
let num =
maybe(just("-"))
|> pair(
alt(
to(just("0"), ["0"]),
recognize(pair(
one_of(["1", "2", "3", "4", "5", "6", "7", "8", "9"]),
many0(digit()),
)),
)
|> map(string.concat),
)
|> pair(maybe(
just(".")
|> right(many1(digit()))
|> map(string.concat),
))
|> pair(
recognize(maybe(
alt(just("e"), just("E"))
|> pair(maybe(one_of(["+", "-"])))
|> pair(many1(digit())),
))
|> map(string.concat),
)
|> map(fn(p) {
case p {
#(#(#(neg, ns), ds), ex) -> {
{
option.unwrap(neg, "") <> ns <> "." <> option.unwrap(ds, "0") <> ex
}
|> float.parse
|> result.unwrap(case neg {
Some(_) -> -1.7976931348623158e308
None -> 1.7976931348623158e308
})
|> Number
}
}
})
let bool =
alt(to(string("true"), Boolean(True)), to(string("false"), Boolean(False)))
let null = to(string("null"), Null)
let array =
sep_by0(value, symbol(","))
|> between(symbol("["), symbol("]"))
|> map(Array)
let obj =
str
|> left(symbol(":"))
|> pair(value)
|> sep_by0(symbol(","))
|> map(dict.from_list)
|> between(symbol("{"), symbol("}"))
|> map(Object)
choice([num, bool, null, map(str, String), array, obj])
|> padded()
}
fn json_parser() -> Parser(Char, JsonValue) {
value_parser()
|> between(whitespace0(), eof())
}
pub type JsonParseError {
UnexpectedToken(found: Char)
UnexpectedEndOfInput
}
pub fn parse_json(value: String) -> Result(JsonValue, JsonParseError) {
case json_parser()(chars.input(value)) {
Ok(pears.Parsed(_, j)) -> Ok(j)
Error(e) ->
Error(case e {
pears.UnexpectedToken(_, _, f) -> UnexpectedToken(f)
pears.UnexpectedEndOfInput(_, _) -> UnexpectedEndOfInput
})
}
}
pub type JsonQuery {
Root
Key(JsonQuery, key: String)
Index(JsonQuery, index: Int)
}
pub type JsonQueryError {
UnexpectedType(JsonValue)
MissingObjectKey(JsonValue, key: String)
IndexOutOfBounds(JsonValue, index: Int)
}
pub fn query_json(
json: JsonValue,
query: JsonQuery,
) -> Result(JsonValue, JsonQueryError) {
case query {
Root -> Ok(json)
Key(q, k) ->
case query_json(json, q) {
Ok(Object(o) as j) ->
dict.get(o, k)
|> result.replace_error(MissingObjectKey(j, k))
Ok(j) -> Error(UnexpectedType(j))
x -> x
}
Index(q, i) ->
case query_json(json, q) {
Ok(Array(a) as j) ->
list.at(a, i)
|> result.replace_error(IndexOutOfBounds(j, i))
Ok(j) -> Error(UnexpectedType(j))
x -> x
}
}
}

121
test/jasper_test.gleam Normal file
View file

@ -0,0 +1,121 @@
import gleeunit
import gleeunit/should
import gleam/dict
import jasper.{
type JsonValue, Array, Boolean, Index, IndexOutOfBounds, Key, MissingObjectKey,
Null, Number, Object, Root, String, UnexpectedType, parse_json, query_json,
}
pub fn main() {
gleeunit.main()
}
fn should_parse(json: String, result: JsonValue) {
json
|> parse_json
|> should.equal(Ok(result))
}
pub fn parse_numbers_test() {
should_parse("4.2", Number(4.2))
should_parse("42", Number(42.0))
}
pub fn parse_booleans_test() {
should_parse("true", Boolean(True))
should_parse("false", Boolean(False))
}
pub fn parse_null_test() {
should_parse("null", Null)
}
pub fn parse_strings_test() {
should_parse("\"hello\"", String("hello"))
}
pub fn parse_arrays_test() {
should_parse("[]", Array([]))
should_parse("[1, 2, 3]", Array([Number(1.0), Number(2.0), Number(3.0)]))
should_parse(
"[true, false, null]",
Array([Boolean(True), Boolean(False), Null]),
)
should_parse(
"[\"hello\", \"world\"]",
Array([String("hello"), String("world")]),
)
}
pub fn parse_objects_test() {
should_parse("{}", Object(dict.new()))
should_parse(
"{\"a\": 1, \"b\": 2}",
Object(dict.from_list([#("a", Number(1.0)), #("b", Number(2.0))])),
)
should_parse(
"{\"a\": true, \"b\": false, \"c\": null}",
Object(
dict.from_list([
#("a", Boolean(True)),
#("b", Boolean(False)),
#("c", Null),
]),
),
)
should_parse(
"{\"a\": \"hello\", \"b\": \"world\"}",
Object(dict.from_list([#("a", String("hello")), #("b", String("world"))])),
)
should_parse(
"{\"👋\": [1, 2, 3], \"b\": {\"c\": 4}}",
Object(
dict.from_list([
#("👋", Array([Number(1.0), Number(2.0), Number(3.0)])),
#("b", Object(dict.from_list([#("c", Number(4.0))]))),
]),
),
)
}
pub fn query_test() {
query_json(String("foo"), Root)
|> should.equal(Ok(String("foo")))
query_json(
String("foo"),
Root
|> Key("foo"),
)
|> should.equal(Error(UnexpectedType(String("foo"))))
query_json(
String("foo"),
Root
|> Index(2),
)
|> should.equal(Error(UnexpectedType(String("foo"))))
query_json(
Array([String("foo")]),
Root
|> Index(2),
)
|> should.equal(Error(IndexOutOfBounds(Array([String("foo")]), 2)))
query_json(
Object(dict.from_list([#("bar", Array([String("foo")]))])),
Root
|> Key("bar")
|> Index(2),
)
|> should.equal(Error(IndexOutOfBounds(Array([String("foo")]), 2)))
query_json(
Object(dict.from_list([#("bar", Array([String("foo")]))])),
Root
|> Key("foo")
|> Index(2),
)
|> should.equal(
Error(MissingObjectKey(
Object(dict.from_list([#("bar", Array([String("foo")]))])),
"foo",
)),
)
}