YSON implementation for JavaScript
YSON is similar to JSON. Main benefits:
$ npx jsr add @j0code/yson
$ npm i @j0code/yson
import YSON from "@j0code/yson" // same as JSON let s = YSON.stringify(someObject) let o = YSON.parse(s) // additional let f = await YSON.load("./file.yson") // load file with fetch and YSON.parse() it let t = YSON.parse(s, [YourClass1, YourClass2]) // allows parsing your own classes (see Types)
import YSON from "https://j0code.github.io/yson/main.js"
or host it yourself!
Note: for proper type-checking, include the following in your tsconfig.json
:
"paths": { "https://j0code.github.io/yson/*": ["./node_modules/@j0code/yson/src/*"] }
5 6.5 .33 1.5e-2 true false null "Hello World" #abcdef
In JavaScript, hexadecimal integers are represented as BigInts.
Strings only allow double quotes (""
). Single quotes (''
) are not supported.
The difference to JSON is that you don't need double quotes (""
) around keys in most cases.
If your keys include any characters other than a-zA-Z0-9_-
, you will need to use a string.
{ a: 3, s: "a string", b: false "this works": true }
[1, 2, 3, 4, "Hello World", false, 42]
{ a: Date "2022-06-06T11:59:41.108Z", b: URL "https://github.com/j0code/node-yson/", c: Map { key1: value1, key2: value2 }, d: Set [value1, value2], e: YourClass { x: 5 y: 3 }, f: YourClass [ 1, 2, 3 ] }
See also: Custom Types
YSON.parse(raw, types, options)
ParseOptions
, currently nothing (optional)Parses your raw YSON string and returns a YSONValue
.
See more on the types
parameter: Custom Types
const data = YSON.parse(`{ message: "ok", status: 200 }`)
const raw = YSON.stringify(data, options)
,
, :
and type names)[1, 2]
-> [ 1, 2 ]
))Stringifies data by representing it in YSON format.
const raw = YSON.stringify({ message: "ok", status: 200 }, { space: "\t" })
const raw = YSON.stringify(source, types)
Uses fetch()
to load the UTF-8 encoded file specified in source
and then parses its contents.
const data = YSON.load("./data.yson")
By default, types are automatically emitted.
They are taken from instance.constructor.name
where instance
is an Instance of some arbitrary class.
For parsing, the YSON Parser actually needs some more information.
You need to provide an Object that maps type names to YSONReviver
functions or the constructor of your class if it has a static method .fromYSON()
which is a YSONReviver
. Their purpose is to validate the input and/or turn the raw data into a more useful object (e.g. your class).
import { YSONReviver } from "@j0code/yson" export default class Rectangle { // the type annotation is technically not necessary but helps to avoid mistakes and enables code completion static fromYSON: YSONReviver<Rectangle> = x => { if (typeof x != "object" || x instanceof Array) return // reject String and Array types if (!("width" in x)) throw new Error("Rectangles need to have a width") if (!("height" in x)) throw new Error("Rectangles need to have a height") return new Rectangle(x.width, x.height) } readonly width: number readonly height: number constructor(width: number, height: number) { this.width = width this.height = height } get area() { return this.width = this.height } toYSON() { return { width: this.width, height: this.height } } } const square1 = new Rectangle(20, 20) const square2 = new Rectangle(10, 10) const data = YSON.stringify([ square1, square2 ]) // [Rectangle{width:20,height:20},Rectangle{width:10,height:10}]
export function parseVector(vec: unknown) { if (!(vec instanceof Array)) return // reject String and Object types if (typeof vec[0] != "number") return if (typeof vec[1] != "number") return return { ...vec, x: vec[0], y: vec[0], toYSON: () => [vec[0], vec[1]] } } const vec = YSON.parse("Vector [3, 4]", { Vector: parseVector } ) // => { '0': 3, '1': 4, x: 3, y: 4, toYSON: [Function: toYSON] }
This example shows how you could both validate inputs and provide a more user-friendly interface (vec.x instead of vec[0]).
Caution: stringifying will lose type information since this is not a class
{ window: { position: Vector [0, 500], dimensions: Rectangle { width: 300, height: 200 } } }
import { YSON } from "@j0code/yson" import Rectangle from "./Rectangle.js" import { parseVector } from "./Vector.js" const brokenConfig = YSON.load("./config.yson") // { window: { position: [ 0, 500 ], dimensions: { width: 300, height: 200 } } } const config = YSON.load("./config.yson", { Rectangle, Vector: parseVector }) // { window: { position: { '0': 1, '1': 4, x: 1, y: 1, toYSON: [Function: toYSON] }, dimensions: Rectangle { width: 300, height: 200 } } }
Add Package
deno add jsr:@j0code/yson
Import symbol
import * as yson from "@j0code/yson";
Import directly with a jsr specifier
import * as yson from "jsr:@j0code/yson";
Add Package
pnpm i jsr:@j0code/yson
pnpm dlx jsr add @j0code/yson
Import symbol
import * as yson from "@j0code/yson";
Add Package
yarn add jsr:@j0code/yson
yarn dlx jsr add @j0code/yson
Import symbol
import * as yson from "@j0code/yson";
Add Package
npx jsr add @j0code/yson
Import symbol
import * as yson from "@j0code/yson";
Add Package
bunx jsr add @j0code/yson
Import symbol
import * as yson from "@j0code/yson";