Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/polytypic/fastener

Functional Zipper for manipulating JSON
https://github.com/polytypic/fastener

cursor functional immutable json query transform zipper

Last synced: 2 months ago
JSON representation

Functional Zipper for manipulating JSON

Awesome Lists containing this project

README

        

# [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#) [Fastener](#fastener) · [![GitHub stars](https://img.shields.io/github/stars/polytypic/fastener.svg?style=social)](https://github.com/polytypic/fastener) [![npm](https://img.shields.io/npm/dm/fastener.svg)](https://www.npmjs.com/package/fastener)

[Zippers](https://www.st.cs.uni-saarland.de/edu/seminare/2005/advanced-fp/docs/huet-zipper.pdf)
are a powerful abstraction for implementing arbitrary queries and transforms on
immutable data structures and for step-by-step navigation and modification of
data structures. This library implements a simple zipper designed for
manipulating JSON data.

[![npm version](https://badge.fury.io/js/fastener.svg)](http://badge.fury.io/js/fastener)
[![Bower version](https://badge.fury.io/bo/fastener.svg)](https://badge.fury.io/bo/fastener)
[![Build Status](https://travis-ci.org/polytypic/fastener.svg?branch=master)](https://travis-ci.org/polytypic/fastener)
[![Code Coverage](https://img.shields.io/codecov/c/github/polytypic/fastener/master.svg)](https://codecov.io/github/polytypic/fastener?branch=master)
[![](https://david-dm.org/polytypic/fastener.svg)](https://david-dm.org/polytypic/fastener) [![](https://david-dm.org/polytypic/fastener/dev-status.svg)](https://david-dm.org/polytypic/fastener?type=dev)

## [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#contents) [Contents](#contents)

* [Tutorial](#tutorial)
* [Reference](#reference)
* [Introduction and Elimination](#introduction-and-elimination)
* [`F.toZipper(json) ~> zipper`](#F-toZipper "F.toZipper: JSON -> Zipper")
* [`F.fromZipper(zipper) ~> json`](#F-fromZipper "F.fromZipper: Zipper -> JSON")
* [Focus](#focus)
* [`F.get(zipper) ~> json`](#F-get "F.get: Zipper -> JSON")
* [`F.modify(json => json, zipper) ~> zipper`](#F-modify "F.modify: (JSON -> JSON) -> Zipper -> Zipper")
* [`F.set(json, zipper) ~> zipper`](#F-set "F.set: JSON -> Zipper -> Zipper")
* [Movement](#movement)
* [Parent-Child movement](#parent-child-movement)
* [`F.downHead(zipper) ~> maybeZipper`](#F-downHead "F.downHead: Zipper -> Maybe Zipper")
* [`F.downLast(zipper) ~> maybeZipper`](#F-downLast "F.downLast: Zipper -> Maybe Zipper")
* [`F.downTo(key, zipper) ~> maybeZipper`](#F-downTo "F.downTo: (String|Number) -> Zipper -> Maybe Zipper")
* [`F.keyOf(zipper) ~> maybeKey`](#F-keyOf "F.keyOf: Zipper -> Maybe (String|Number)")
* [`F.up(zipper) ~> maybeZipper`](#F-up "F.up: Zipper -> Maybe Zipper")
* [Path movement](#path-movement)
* [`F.downPath([...keys], zipper) ~> maybeZipper`](#F-downPath "F.downPath: [String|Number] -> Zipper -> Maybe Zipper")
* [`F.pathOf(zipper) ~> [...keys]`](#F-pathOf "F.pathOf: Zipper -> [String|Number]")
* [Sibling movement](#sibling-movement)
* [`F.head(zipper) ~> maybeZipper`](#F-head "F.head: Zipper -> Maybe Zipper")
* [`F.last(zipper) ~> maybeZipper`](#F-last "F.last: Zipper -> Maybe Zipper")
* [`F.left(zipper) ~> maybeZipper`](#F-left "F.left: Zipper -> Maybe Zipper")
* [`F.right(zipper) ~> maybeZipper`](#F-right "F.right: Zipper -> Maybe Zipper")
* [Queries](#queries)
* [`F.queryMove(zipper => maybeZipper, value, zipper => value, zipper) ~> value`](#F-queryMove "F.queryMove: (Zipper -> Maybe Zipper) -> a -> (Zipper -> a) -> Zipper -> a")
* [Transforms](#transforms)
* [`F.transformMove(move, zipper => zipper, zipper) ~> zipper`](#F-transformMove "F.transformMove: (F.downHead|F.downLast|F.downTo(key)|F.left|F.right|F.up) -> (Zipper -> Zipper) -> Zipper -> Zipper")
* [`F.everywhere(json => json, zipper) ~> zipper`](#F-everywhere "F.everywhere: (JSON -> JSON) -> Zipper -> Zipper")
* [Related Work](#related-work)

## [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#tutorial) [Tutorial](#tutorial)

Playing with zippers in a REPL can be very instructive. First we require the
libraries

```jsx
import * as F from "fastener"
import * as R from "ramda"
```

and define a little helper using
[`reduce`](http://ramdajs.com/0.21.0/docs/#reduce) to perform a sequence of
operations on a value:

```js
const seq = (x, ...fs) => R.reduce((x, f) => f(x), x, fs)
```

Let's work with the following simple JSON object:

```js
const data = {contents: [{language: "en", text: "Title"},
{language: "sv", text: "Rubrik"}]}
```

First we just create a zipper using [`F.toZipper`](#F-toZipper):

```js
seq(F.toZipper(data))
// { focus: { contents: [ [Object], [Object] ] } }
```

As can be seen, **_the zipper is just a simple JSON object_** and the `focus` is
the `data` object that we gave to [`F.toZipper`](#F-toZipper). As long the data
structure being manipulated is JSON, you can serialize and deserialize zippers
as JSON. However, it is recommended that you use the zipper combinators to
operate on zippers rather than rely on their exact format.

Let's then move into the `contents` property of the object using
[`F.downTo`](#F-downTo):

```js
seq(F.toZipper(data),
F.downTo('contents'))
// { left: null,
// focus:
// [ { language: 'en', text: 'Title' },
// { language: 'sv', text: 'Rubrik' } ],
// key: 'contents',
// right: null }
```

As seen above, the `focus` now has the `contents` array. We can use
[`F.get`](#F-get) to extract the value under focus:

```js
seq(F.toZipper(data),
F.downTo('contents'),
F.get)
// [ { language: 'en', text: 'Title' },
// { language: 'sv', text: 'Rubrik' } ]
```

Then we move into the first element of `contents` using
[`F.downHead`](#F-downHead):

```js
seq(F.toZipper(data),
F.downTo('contents'),
F.downHead)
// { left: null,
// focus: { language: 'en', text: 'Title' },
// key: 0,
// right: [ null, { language: 'sv', text: 'Rubrik' } ],
// up: { left: null, key: 'contents', right: null } }
```

And continue into the first property of that which happens to be the `language`:

```js
seq(F.toZipper(data),
F.downTo('contents'),
F.downHead,
F.downHead)
// { left: null,
// focus: 'en',
// key: 'language',
// right: [ null, 'Title', 'text' ],
// up:
// { left: null,
// key: 0,
// right: [ null, [Object] ],
// up: { left: null, key: 'contents', right: null } } }
```

And to the next property, `title`, using [`F.right`](#F-right):

```js
seq(F.toZipper(data),
F.downTo('contents'),
F.downHead,
F.downHead,
F.right)
// { left: [ null, 'en', 'language' ],
// focus: 'Title',
// key: 'text',
// right: null,
// up:
// { left: null,
// key: 0,
// right: [ null, [Object] ],
// up: { left: null, key: 'contents', right: null } } }
```

Let's then use [`F.modify`](#F-modify) to modify the `title`:

```js
seq(F.toZipper(data),
F.downTo('contents'),
F.downHead,
F.downHead,
F.right,
F.modify(t => "The " + t))
// { left: [ null, 'en', 'language' ],
// focus: 'The Title',
// key: 'text',
// right: null,
// up:
// { left: null,
// key: 0,
// right: [ null, [Object] ],
// up: { left: null, key: 'contents', right: null } } }
```

When we now move outwards using [`F.up`](#F-up) we can see the changed title
become part of the data:

```js
seq(F.toZipper(data),
F.downTo('contents'),
F.downHead,
F.downHead,
F.right,
F.modify(t => "The " + t),
F.up)
// { left: null,
// key: 0,
// right: [ null, { language: 'sv', text: 'Rubrik' } ],
// up: { left: null, key: 'contents', right: null },
// focus: { language: 'en', text: 'The Title' } }
```

We can also just move back to the root and get the updated data structure using
[`F.fromZipper`](#F-fromZipper):

```js
seq(F.toZipper(data),
F.downTo('contents'),
F.downHead,
F.downHead,
F.right,
F.modify(t => "The " + t),
F.fromZipper)
// { contents:
// [ { language: 'en', text: 'The Title' },
// { language: 'sv', text: 'Rubrik' } ] }
```

The above hopefully helped to understand how zippers work. However, it is
important to realize that one typically does not use zipper combinators to
create such a specific sequence of operations. One rather uses the zipper
combinators to create new combinators that perform more complex operations
directly.

Let's first define a zipper combinator that, given a zipper focused on an array,
tries to focus on an element inside the array that satisfies a given predicate:

```js
const find = R.curry((p, z) => F.downTo(R.findIndex(p, F.get(z)), z))
```

Like all the basic zipper movement combinators, [`F.downTo`](#F-downTo) is a
*partial function* that returns `undefined` in case the index is out of bounds.
Let's define a simple function to compose partial functions:

```js
const pipePartial = (...fs) => z => {
for (let i=0; z !== undefined && i pipePartial(
F.downTo('contents'),
find(R.whereEq({language})),
F.downTo('text'))
```

Now we can say:

```js
seq(data,
F.toZipper,
textIn("en"),
F.modify(x => 'The ' + x),
F.fromZipper)
// { contents:
// [ { language: 'en', text: 'The Title' },
// { language: 'sv', text: 'Rubrik' } ] }
```

Of course, this just scratches the surface. Zippers are powerful enough to
implement arbitrary transforms on data structures. This can also make them more
difficult to compose and reason about than more limited approaches such as
[lenses](https://github.com/calmm-js/partial.lenses).

## [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#reference) [Reference](#reference)

The zipper combinators are available as named exports. Typically one just
imports the library as:

```jsx
import * as F from "fastener"
```

In the following examples we will make use of the function

```jsx
const seq = (x, ...fs) => R.reduce((x, f) => f(x), x, fs)
```

written using [`reduce`](http://ramdajs.com/0.21.0/docs/#reduce) that allows one
to express a sequence of operations to perform starting from a given value.

### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#introduction-and-elimination) [Introduction and Elimination](#introduction-and-elimination)

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-toZipper) [`F.toZipper(json) ~> zipper`](#F-toZipper "F.toZipper: JSON -> Zipper")

`F.toZipper(json)` creates a new zipper that is focused on the root of the given
JSON object.

For example:

```js
seq(F.toZipper([1,2,3]),
F.downHead,
F.modify(x => x + 1),
F.fromZipper)
// [ 2, 2, 3 ]
```

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-fromZipper) [`F.fromZipper(zipper) ~> json`](#F-fromZipper "F.fromZipper: Zipper -> JSON")

`F.fromZipper(zipper)` extracts the modified JSON object from the given zipper.

For example:

```js
seq(F.toZipper([1,2,3]),
F.downHead,
F.modify(x => x + 1),
F.fromZipper)
// [ 2, 2, 3 ]
```

### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#focus) [Focus](#focus)

Focus combinators allow one to inspect and modify the element that a zipper is
focused on.

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-get) [`F.get(zipper) ~> json`](#F-get "F.get: Zipper -> JSON")

`F.get(zipper)` returns the element that the zipper is focused on.

For example:

```js
seq(F.toZipper(1), F.get)
// 1
```
```js
seq(F.toZipper(["a","b","c"]),
F.downTo(2),
F.get)
// 'c'
```

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-modify) [`F.modify(json => json, zipper) ~> zipper`](#F-modify "F.modify: (JSON -> JSON) -> Zipper -> Zipper")

`F.modify(fn, zipper)` is equivalent to `F.set(fn(F.get(zipper)), zipper)` and
replaces the element that the zipper is focused on with the value returned by
the given function for the element.

For example:

```js
seq(F.toZipper(["a","b","c"]),
F.downTo(2),
F.modify(x => x + x),
F.fromZipper)
// [ 'a', 'b', 'cc' ]
```

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-set) [`F.set(json, zipper) ~> zipper`](#F-set "F.set: JSON -> Zipper -> Zipper")

`F.set(json, zipper)` replaces the element that the zipper is focused on with
the given value.

For example:

```js
seq(F.toZipper(["a","b","c"]),
F.downTo(1),
F.set('lol'),
F.fromZipper)
// [ 'a', 'lol', 'c' ]
```

### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#movement) [Movement](#movement)

Movement combinators can be applied to any zipper, but they return `undefined`
in case of illegal moves.

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#parent-child-movement) [Parent-Child movement](#parent-child-movement)

Parent-Child movement is moving the focus between a parent object or array and a
child element of said parent.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-downHead) [`F.downHead(zipper) ~> maybeZipper`](#F-downHead "F.downHead: Zipper -> Maybe Zipper")

`F.downHead(zipper)` moves the focus to the leftmost element of the object or
array that the zipper is focused on.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-downLast) [`F.downLast(zipper) ~> maybeZipper`](#F-downLast "F.downLast: Zipper -> Maybe Zipper")

`F.downLast(zipper)` moves the focus to the rightmost element of the object or
array that the zipper is focused on.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-downTo) [`F.downTo(key, zipper) ~> maybeZipper`](#F-downTo "F.downTo: (String|Number) -> Zipper -> Maybe Zipper")

`F.downTo(key, zipper)` moves the focus to the specified object property or
array index of the object or array that the zipper is focused on.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-keyOf) [`F.keyOf(zipper) ~> maybeKey`](#F-keyOf "F.keyOf: Zipper -> Maybe (String|Number)")

`F.keyOf(zipper)` returns the object property name or the array index that the
zipper is currently focused on.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-up) [`F.up(zipper) ~> maybeZipper`](#F-up "F.up: Zipper -> Maybe Zipper")

`F.up(zipper)` moves the focus from an array element or object property to the
containing array or object.

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#path-movement) [Path movement](#path-movement)

Path movement is moving the focus along a path from a parent object or array to
a nested child element.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-downPath) [`F.downPath([...keys], zipper) ~> maybeZipper`](#F-downPath "F.downPath: [String|Number] -> Zipper -> Maybe Zipper")

`F.downPath(path, zipper)` moves the focus along the specified path of keys.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-pathOf) [`F.pathOf(zipper) ~> [...keys]`](#F-pathOf "F.pathOf: Zipper -> [String|Number]")

`F.pathOf(zipper)` returns the path from the root to the current element focused
on by the zipper.

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#sibling-movement) [Sibling movement](#sibling-movement)

Sibling movement is moving the focus between the elements of an array or an object.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-head) [`F.head(zipper) ~> maybeZipper`](#F-head "F.head: Zipper -> Maybe Zipper")

`F.head(zipper)` moves the focus to the leftmost sibling of the current focus.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-last) [`F.last(zipper) ~> maybeZipper`](#F-last "F.last: Zipper -> Maybe Zipper")

`F.last(zipper)` moves the focus to the rightmost sibling of the current focus.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-left) [`F.left(zipper) ~> maybeZipper`](#F-left "F.left: Zipper -> Maybe Zipper")

`F.left(zipper)` moves the focus to the element on the left of the current focus.

##### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-right) [`F.right(zipper) ~> maybeZipper`](#F-right "F.right: Zipper -> Maybe Zipper")

`F.right(zipper)` moves the focus to the element on the right of the current focus.

### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#queries) [Queries](#queries)

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-queryMove) [`F.queryMove(zipper => maybeZipper, value, zipper => value, zipper) ~> value`](#F-queryMove "F.queryMove: (Zipper -> Maybe Zipper) -> a -> (Zipper -> a) -> Zipper -> a")

`F.queryMove(move, default, fn, zipper)` applies the given function `fn` to the
zipper focused on after the given movement and returns the result unless the
move was illegal in which case the given default value is returned instead.

For example:

```js
seq(F.toZipper({x: 1}),
F.queryMove(F.downTo('y'), false, () => true))
// false
```
```js
seq(F.toZipper({y: 1}),
F.queryMove(F.downTo('y'), false, () => true))
// true
```

### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#transforms) [Transforms](#transforms)

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-transformMove) [`F.transformMove(move, zipper => zipper, zipper) ~> zipper`](#F-transformMove "F.transformMove: (F.downHead|F.downLast|F.downTo(key)|F.left|F.right|F.up) -> (Zipper -> Zipper) -> Zipper -> Zipper")

`F.transformMove(move, fn, zipper)` applies the given function to the zipper
focused on after the given movement. The movement `move` must be one of
[`F.downHead`](#F-downHead), [`F.downLast`](#F-downLast),
[`F.downTo(key)`](#F-downTo), [`F.left`](#F-left), [`F.right`](#F-right), or
[`F.up`](#F-up). The function `fn` must the return a zipper focused on the same
element that it was given. Then the focus is moved back to the element that the
zipper was originally focused on. Nothing is done in case of an illegal move.

For example:

```js
seq(F.toZipper({y: 1}),
F.transformMove(F.downTo('y'), F.modify(x => x + 1)),
F.fromZipper)
// { y: 2 }
```
```js
seq(F.toZipper({x: 1}),
F.transformMove(F.downTo('y'), F.modify(x => x + 1)),
F.fromZipper)
// { x: 1 }
```

#### [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#F-everywhere) [`F.everywhere(json => json, zipper) ~> zipper`](#F-everywhere "F.everywhere: (JSON -> JSON) -> Zipper -> Zipper")

`F.everywhere(fn, zipper)` performs a transform of the focused element by
modifying each possible focus of the element with a bottom-up traversal.

For example:

```js
seq(F.toZipper({foo: 1,
bar: [{lol: "bal", example: 2}]}),
F.everywhere(x => typeof x === "number" ? x + 1 : x),
F.fromZipper)
// { foo: 2, bar: [ { lol: 'bal', example: 3 } ] }
```

## [≡](#contents) [▶](https://polytypic.github.io/fastener/index.html#related-work) [Related Work](#related-work)

While the implementation is very different, the choice of combinators is based
on Michael D. Adams' paper
[Scrap Your Zippers](http://michaeldadams.org/papers/scrap_your_zippers/).