Skip to content

greens231/jsonfp

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

47 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

JSON-FP

In a decentralized computing environment, when applications are dealing with huge amount of data it's a better practice to pass programming codes to various machines than moving data. However, how can machines of various configurations understand each other? Also, the "moving code, least moving data" policy would arguably work better in the functional programming paradigm than the imperative one.

Those questions/issues lead to the idea of doing functional programming in JSON. If programs can be coded in JSON, they can be easily shipped around and understood by machines of vaious settings. Combining JSON and functional programming also makes security issues easier to track or manage.

JSON-FP is part of an attempt to make data freely and easily accessed, distributed, annotated, meshed, even re-emerged with new values. To achieve that, it's important to be able to ship codes to where data reside, and that's what JSON-FP is trying to achieve.

JSON-FP playground

Want to play with JSON-FP to see how it works? You can test your JSON-FP code online and find examples at JSON-FP playground.

What's new

  • Four new operators (head, tail, bucket and infix) are introduced to the runtime core. The bucket operator can be used to distribute input list into sub-list (buckets) based on the bucket conditions or predicates (v-0.2.2).

  • The infix operator is introduced to solve the problem that JSON-FP always work on input as a whole. Consider adding two numbers x and y. This could be tricky if you don't want to introduce any side effect to your JSON-FP expression. With the infix operator, the problem can be easily solved by doing: infix: ['$in.x', {add: '$in.y'}].

  • The concept of streaming data is introduced in v-0.2.1. For descriptions and examples, please check Using "Streams" As Data Generators.

  • Adding customized operators to the JSON-FP runtime has been standardized (v-0.1.1). Please check Developing and Installing Packages for details.

  • Added the formula operator for metaprogramming. Developers now can use the formula operator to define a JSON-FP formula and use the convert operator to apply a formula (v-0.1.0).

For details about what's new in the current release, please check the release note.

Install

npm install -g jsonfp

Contents

Getting started

An example project has been created to demonstrate how to write JSON-FP expressions. In addition to that, test files under the test directory is also a good place to start. Below are some examples you may want to check:

  • simple: very simple examples to help you getting familiar with JSON-FP.

  • stream: showing how to use streams as input to JSON-FP expressions. In particular, the example shows how to use the iterator stream to replace the for-loop statement.

  • quick sort: the classic quick sort algorithm implemented in JSON-FP.

  • object query: showing how to query a list of JSON objects with various conditions.

  • metaprogramming: showing how to do alpha-conversion in JSON-FP.

Run programs

Below is how you can run or evaluate a JSON-FP program:

var  jsonfp = require('jsonfp');
jsonfp.init();

// providing a context(ctx) to run 'program' with 'input'
jsonfp.apply(ctx, input, program);

// or simply
jsonfp.apply(input, program);

program should be a JSON-FP program and input can be any value. Context is a plain object to act as an additional data channel to a program.

Promise or callback

Built-in operators of the current implementation will do their jobs synchronously. However, if you add your own customized operators and they would do things asynchronously, those asynchronous operators should return a promise to JSON-FP. JSON-FP knows how to deal with promise.

Other than operators, JSON-FP supports both promise and callbacks to deal with those asynchronous situations. In other words, if you expect the computation of your JSON-FP expression will be done asynchroously, you can either use the promise style:

jsonfp.apply(input, expr).then(function(value) {
    // value is the result
});

or the callback style:

jsonfp.apply(input, expr, function(err, value) {
    // value is the result
});

JSON-FP expression

A JSON-FP expression is a JSON object with a single property. The property key is the "operator" which works on the input data while the property value specifies options to that operator. So a JSON-FP expression is as simple as:

{op: options}

The interesting part is that options can be yet another JSON-FP expression. A typical example would be the case of applying the "map" operator. Assuming we have a list of documents and we want to remove all properties but the title property for each document. Below is what you can do with JSON-FP:

{"map":
	{"pick": "title"}
}

By repeatedly substituting expression value with another JSON-FP expression, an expression as simple as {op: options} can turn into a really sophisticated program.

Expression input

When running a JSON-FP program, you have to provide the initial input data. After that, the data flow will become implicit. For example, when operators are chained, input data will be fed to the head of the chain and every succeeding operator will get its input from the output of its predecesor. Another example is the 'map' operator which will iterate through its input (could be an array or a collection) and feed each element as the input to its child expression. In other words, the parent expression will decide how to provide input to its child expressions without needing explicit specifications by application developers.

However, if you want to change the implicit data flow, you can use the 'chain' operator to achieve that. Below is an example:

{
    '->': [
        {name: 'David', age: 28},
        {getter: 'age'}
    ]
}

If you evaluate the above expression, the result will be 28. The first one of the chained expressions is becoming the input to the second expression. With that technique, you can change expression input as you wish.

Evaluation

Anything that you send as an expression to JSON-FP will be evaluated recursively until a scalar is found. For example:

var  expr = {name: 'David'},
	 result = jsonfp.apply('Jones', expr);
console.log( JSON.stringify(result) );

In the above example, JSON-FP will try to evalute {name: 'David'} but figure out there is nothing it should do. So the output will be exactly the same as the input.

However, if you do something like:

var  expr = {name: {add: ' Cooper'}},
	 result = jsonfp.apply('David', expr);
console.log( JSON.stringify(result) );

This time JSON-FP finds something to work on and {name: 'David Cooper'} will be the output.

Variables

It's possible to refer to variables in a JSON-FP expression. They are expressed as a string with a leading '$' sign. For example, '$title' and '$in.id' are all valid variables. The syntax of JSON-FP variables is defined as such because we do not want to break the JSON syntax.

Input to an JSON-FP expression can be referred as '$in'. If input is a plain object with a 'title' property, you can refe to that title property as '$in.title'.

Besides input, you can put all other variables in the context variable. Below is an exmaple showing you how to use context variables:

var  expr = {name: {add: '$firstName'}},
	 ctx = {firstName: 'David '},
	 result = jsonfp.apply(ctx, 'Jones', expr);
	 
// Below should print out 'David Jones'
console.log( JSON.stringify(result) );

Note that in the above example we provide a context variable (ctx) to supply the first name to the expression. The result will be printed out on console as {name: 'David Jones'}.

Setting variables

A JSON-FP expression can unfold itself as a series (or a tree of series) of expressions. Input to the expression will water fall through or be transformed by those series of expressions. Sometimes we need to keep the intermediate results and recall them when necessary. To do so, the intermediate results should be able to be saved.

You ca save the intermediate results in the context variable like the following:

{
    $name: {getter: 'name'}
}

That will pick up the name property of input and store it into the context variable. The saved value will be visible to successive expressions. For example:

var expr = 
{eval:
    {
        $name: {getter: 'name'},
        $hobby: {getter: 'hobby'},
        response: {
            "->" : [
                ['$name', ' likes ', '$hobby'],
			    {reduce: 'add'}
            ]
        }
    }
};

will save the name and hobby of a person to the '$name' and '$hobby' variables respectively. However, if you do:

var expr = 
{eval:
    {
        response: {
            "->" : [
                ['$name', ' likes ', '$hobby'],
			    {reduce: 'add'}
            ]
        },
        $name: {getter: 'name'},
        $hobby: {getter: 'hobby'}
    }
};

You'll not get the expected result because of the reversed execution sequence.

Also note that, variables will not show up in results. The above example if done correctly will generate the result as (assuming input is {name: 'David', hobby: 'meditation'})

{
    response: 'David likes meditation'
}

rather than

{
    $name: 'David',
    $hobby: 'meditation',
    response: 'David likes meditation'
}

You won't have '$name' and '$hobby' as part of the return value.

Operators

What operators are available in a JSON-FP runtime will decide its capabilities, and that can be fully customized. Customizing the set of supported operators is a very important feature because it allows a server (or any JSON-FP runtime) to gauge what capacity it's willing to offer.

The current implementation comes with more than 30 operators. To view the list and usage of these built-in operators, please refre to this page for details.

If you need some functions not supported by the built-in operators, you can simply add your own! Below is an example:

var  jsonfp = require('jsonfp');

jsonfp.addMethod('average', function(input, x) {
    return  (input + x) / 2;
});

JSON-FP by default will evaluate the expression option before feeding the option to an operator. If you do not want JSON-FP to evaluate the expression option, you could do the following:

jsonfp.addMethod('getter', {op: doGetter, defOption: true});

function doGetter(input, expr) {
    return  input[expr];
};

If you specify the defOption property as true when adding methods to the JSON-FP runtime, that will stop JSON-FP from automaticaly evaluating the option.

API

jsonfp.init(options)

Before you evaluate any JSON-FP expressions, you should call jsonfp.init() to preload the built-in operators. You can also preload just part of the built-in operators by specifying the needed operators in the options parameter. For example:

jsonfp.init(['arithmetic', 'arrays', 'collections']);

The above example does not load "comparators".

jsonfp.apply(ctx, input, expr, cb)

Evaluates a JSON-FP expression and returns the result. If any JSON-FP expression is performed asynchronously, the return value will be a promise. If you prefer the callback style, you can put the callback function as the 4th parameter to the function call. cb(err, value) is an optional callback function which will take an error object (if any) and the result as the second parameter. Note that if invoked with a callback function, jsonfp.apply() will no longer return a value.

Parameter ctx is a context variable, input will be fed to the expression, and expr is the JSON-FP expression to be evaluated. The ctx parameter is optional.

jsonfp.isExpression(expr)

Checks to see if expr is a evaluable JSON-FP expression.

jsonfp.addMethod(name, func)

Adds an customized operator to the JSON-FP runtime. name is the operator name, and func can be a Javascript function or a plain object with the following properties:

  • op: a Javascript function to carry out the operator functions.
  • defOption: if true, the expression option will not be evaluated before the expression is evaluated. For operators such as "map" or "filter" which will treat the option as a JSON-FP expression, this property should be set to true.

jsonfp.removeMethod(name)

Removes an operator from a JSON-FP runtime.

Customizing JSON-FP

One of JSON-FP's great features is the ability to extend the language by adding customized operators. Since v0.1.1, a recommended way to develop and install customized operators have been introduced. With that, third party functions can be grouped and installed as packages, and you can freely customize the JSONF-FP runtime as you need. For details, you can check "Developing and Installing Packages" for details.

Metaprogramming

JSON-FP is homoiconic. That is you can manipulate JSON-FP programs the exact same way as you manipulate JSON objects. Becuase of that, using JSON-FP to do metaprogramming could be very easy.

The latest release (0.1.0) supports two metaprogramming related operators: convert and formula. The formula operator helps developers to define expression formula which can be reused and served as building blocks for more complicated JSON-FP programs. The convert operator can be used to apply those formula defined by the formula operator. To know more about how to use these two operators, you may want to check this example.

There are some simple examples in testLamda.js if you're interested.

About

Functional programming in JSON

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • JavaScript 100.0%