Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/organix/tartjs
Tiny Actor Run-Time in JavaScript
https://github.com/organix/tartjs
Last synced: 3 months ago
JSON representation
Tiny Actor Run-Time in JavaScript
- Host: GitHub
- URL: https://github.com/organix/tartjs
- Owner: organix
- License: mit
- Created: 2013-11-28T00:25:00.000Z (about 11 years ago)
- Default Branch: master
- Last Pushed: 2016-12-12T19:32:36.000Z (about 8 years ago)
- Last Synced: 2024-04-14T10:48:31.932Z (10 months ago)
- Language: JavaScript
- Size: 85.9 KB
- Stars: 72
- Watchers: 9
- Forks: 5
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-ccamel - organix/tartjs - Tiny Actor Run-Time in JavaScript (JavaScript)
README
# tart
_Stability: 1 - [Experimental](https://github.com/tristanls/stability-index#stability-1---experimental)_
[![NPM version](https://badge.fury.io/js/tart.png)](http://npmjs.org/package/tart)
JavaScript implementation of [Tiny Actor Run-Time](https://github.com/organix/tart).
## Contributors
[@dalnefre](https://github.com/dalnefre), [@tristanls](https://github.com/tristanls)
## Overview
The goal of `tart` is to provide the smallest possible actor library in JavaScript that has the full power of a "pure" actor model of computation.
* [Modules](#modules)
* [Usage](#usage)
* [Tests](#tests)
* [Benchmarks](#benchmarks)
* [Documentation](#documentation)
* [Tweet](#tweet)
* [Minimal](#minimal)
* [Pluggable](#pluggable)
* [Sources](#sources)`tart` happens to fit into a tweet :D
function(){var c=function(b){var a=function(m){setImmediate(function(){x.behavior(m)})},x={self:a,behavior:b,sponsor:c};return a};return c}
## Modules
Here is a selection of modules from the authors of Tart:
* [tart-adapter](https://github.com/tristanls/tart-adapter) Adapter turning synchronous functions into asynchronous Tart actors.
* [tart-ansible](https://github.com/tristanls/tart-ansible) Enables remote message delivery for Tart actors.
* [tart-lambda](https://github.com/dalnefre/tart-lambda) A simple lambda-calculus evaluator.
* [tart-marshal](https://github.com/dalnefre/tart-marshal) Send messages between memory domains (JSON serialization).
* [tart-membrane](https://github.com/tristanls/tart-membrane) Tart membrane implementation (distributed confinement).
* [tart-revocable](https://github.com/tristanls/tart-revocable) Tart implementation of a revocable proxy.
* [tart-stepping](https://github.com/dalnefre/tart-stepping). --DEPRECATED-- superceded by [tart-tracing](https://github.com/tristanls/tart-tracing).
* [tart-stepping-sandbox](https://github.com/tristanls/tart-stepping-sandbox) Remotely controlled sandbox for controlling [tart-stepping](https://github.com/dalnefre/tart-stepping).
* [tart-tracing](https://github.com/tristanls/tart-tracing) Tart tracing control (maintains a full event history for debugging).
* [tart-transport-http](https://github.com/tristanls/tart-transport-http) HTTP transport implementation for Tart.
* [tart-transport-https](https://github.com/tristanls/tart-transport-https) HTTPS transport implementation for Tart.
* [tart-transport-tcp](https://github.com/tristanls/tart-transport-tcp) TCP transport implementation for Tart.
* [tart-transport-tls](https://github.com/tristanls/tart-transport-tls) TLS transport implementation for Tart.
* [tart-transport-udp](https://github.com/tristanls/tart-transport-udp) UDP transport implementation for Tart.
* [tart-vm](https://github.com/tristanls/tart-vm) Tart behavior sandbox.## Usage
To run the below example run:
npm run readme
```javascript
"use strict";var tart = require('../index.js');
var sponsor = tart.minimal();
// create an actor that has no state
var statelessActor = sponsor(function (message) {
console.log('got message', message);
});// create an actor with state
var statefulActorBeh = function (state) {
return function (message) {
console.log('got message', message);
console.log('actor state', state);
};
};var statefulActor = sponsor(statefulActorBeh({some: 'state'}));
// create an actor with state that changes behavior
var flipFlop = function (state) {
var firstBeh = function (message) {
console.log('firstBeh got message', message);
console.log('actor state', state);
this.behavior = secondBeh;
};
var secondBeh = function (message) {
console.log('secondBeh got message', message);
console.log('actor state', state);
this.behavior = firstBeh;
};
return firstBeh;
};var serialActor = sponsor(flipFlop({some: 'state'}));
// create an actor that creates a chain of actors
var chainActorBeh = function (count) {
return function (message) {
console.log('chain actor', count);
if (--count >= 0) {
var next = this.sponsor(chainActorBeh(count));
next(message);
}
};
};var chainActor = sponsor(chainActorBeh(10));
// send messages to the actors
statelessActor('some message');
statefulActor({some: 'other message'});
serialActor('first message');
serialActor('second message');
serialActor('third message');
serialActor('fourth message');
chainActor('go');
```## Tests
npm test
## Benchmarks
Benchmarks were run on the [Minimal](#minimal) implementation.
### Erlang Challenge
Erlang Challenge consists of creating a ring of M actors, sending N simple messages around the ring and increasing M until running out of resources.
The benchmark implements a modified version of the challenge by creating 100,000 actors and running 10 simple messages around the ring.
npm run erlangChallenge
### 100,000 actor ring
starting 100000 actor ring
sending 10 messages
..........
done
all times in NANOSECONDS
construction time:
336492833
loop times:
221762650
212925428
213672756
215595649
215465809
216958143
217696839
221343351
222385758
226308376
loop average:
218411475.9For rings of sizes larger than 4 Million you may need to expand memory available to V8. To do that, the following command will start the Erlang challenge with ~10GB of memory available:
node --max_old_space_size=10000 scripts/erlangChallenge.js
### 30,000,000 actor ring
30 Million actor ring benchmark took up about ~8.5 GB of memory.
starting 30000000 actor ring
sending 1 messages
.
done
all times in NANOSECONDS
construction time:
233098891645
loop times:
244265390009
loop average:
244265390009## Documentation
The [Minimal](#minimal) implementation is the implementation optimized for fastest execution time. In contrast, [Pluggable](#pluggable) implementation allows for total control of the runtime and execution semantics. Although the default behavior of [Pluggable](#pluggable) is the same as [Minimal](#minimal), it is somewhat slower due to extra overhead incurred by pluggability of control and observability mechanisms.
* [Tweet](#tweet)
* [Minimal](#minimal)
* [Pluggable](#pluggable)### Tweet
**Public API**
* [tart.tweet()](#tarttweet)
* [sponsor(behavior)](#sponsorbehavior)
* [actor(message)](#actormessage)### tart.tweet()
Creates a sponsor capability to create new actors with using the Tweetable implementation :D.
WARNING: If an exception is thrown during message processing the Tweetable run-time will crash. For fastest stable implementation use [Minimal](#minimal).
### sponsor(behavior)
Same as the core [Minimal](#minimal) implementation. _See: [sponsor(behavior)](#sponsorbehavior-1)_
### actor(message)
Same as the core [Minimal](#minimal) implementation. _See: [actor(message)](#actormessage-1)_
### Minimal
**Public API**
* [tart.minimal(\[options\])](#tartminimaloptions)
* [sponsor(behavior)](#sponsorbehavior-1)
* [actor(message)](#actormessage-1)### tart.minimal([options])
* `options`: _Object_ _(Default: `undefined`)_
* `fail`: _Function_ _(Default: `function (exception) {}`)_ `function (exception) {}` An optional handler to call if a sponsored actor behavior throws an exception.
* Return: _Function_ `function (behavior) {}` A capability to create new actors.Creates a sponsor capability to create new actors with.
```javascript
var tart = require('tart');
var sponsor = tart.minimal();var reportingSponsor = tart.minimal({
fail: function (exception) {
console.dir(exception);
}
});
```### sponsor(behavior)
* `behavior`: _Function_ `function (message) {}` Actor behavior to invoke every time an actor receives a message.
* Return: _Function_ `function (message) {}` Actor reference in form of a capability that can be invoked to send the actor a message.Creates a new actor and returns the actor reference in form of a capability to send that actor a message.
```javascript
var tart = require('tart');
var sponsor = tart.minimal();
var actor = sponsor(function (message) {
console.log('got message', message);
console.log(this.self);
console.log(this.behavior);
console.log(this.sponsor);
});
```When the `behavior` is invoked upon the receipt of a message, it's `this` will be bound with the following:
* `this.self`: _Function_ `function (message) {}` Reference to the actor that is executing the `behavior` (in form of a capability that can be invoked to send the actor a message).
* `this.behavior`: _Function_ `function (message) {}` The behavior of the actor. To change actor behavior (a "become" operation) assign a new function to this parameter.
* `this.sponsor`: _Function_ `function (behavior) {}` A capability to create new actors. To create a new actor call `this.sponsor(behavior)`.### actor(message)
* `message`: _Any_ Any message.
Asynchronously sends the `message` to the `actor`.
```javascript
var tart = require('tart');
var sponsor = tart.minimal();
var actor = sponsor(function behavior(message) {
console.log('got message', message);
});
actor('hello actor world');
```### Pluggable
**Public API**
* [tart.pluggable(\[options\])](#tartpluggableoptions)
* [sponsor(behavior)](#sponsorbehavior-2)
* [actor(message)](#actormessage-2)### tart.pluggable([options])
* `options`: _Object_ _(Default: undefined)_ Optional overrides.
* `constructConfig`: _Function_ _(Default: `function (options) {}`)_ `function (options) {}` Configuration creation function that is given `options`. It should return a capability `function (behavior) {}` to create new actors.
* `deliver`: _Function_ _(Default: `function (context, message, options) {}`)_ `function (context, message, options) {}` Deliver function that returns a function for `dispatch` to dispatch.
* `dispatch`: _Function_ _(Default: `setImmediate`)_ `function (deliver) {}` Dispatch function for dispatching `deliver` closures.
* `fail`: _Function_ _(Default: `function (exception) {}`)_ `function (exception) {}` An optional handler to call if a sponsored actor behavior throws an exception.
* `annotate`: _Function_ _(Default: `function (actor) { return actor; }`)_ `function (actor) {}` An optional method to wrap/modify newly-created actors.
* Return: _Function_ `function (behavior) {}` A capability to create new actors.Creates a sponsor capability to create new actors with and allows replacing parts of the implementation.
To run the below example run:
npm run pluggable
```javascript
var tart = require('tart');var dispatch = function (deliver) {
console.log('delivering a message');
deliver();
};var deliver = function deliver(context, message, options) {
console.log('delivering message', message, 'to context', context);
return function deliver() {
try {
context.behavior(message);
} catch (exception) {
console.log('got exception', exception);
}
};
};var constructConfig = function constructConfig(options) {
var config = function create(behavior) {
var actor = function send(message) {
options.dispatch(options.deliver(context, message, options));
};
actor = options.annotate(actor);
var context = {
self: actor,
behavior: behavior,
sponsor: config
};
console.log('created actor in context', context);
return actor;
};
return config;
};var annotate = (function (n) {
return function annotate(actor) {
var id = '@' + n++;
actor.toString = actor.inspect = function () {
return id;
};
return actor;
};
})(0);var sponsor = tart.pluggable({
constructConfig: constructConfig,
deliver: deliver,
dispatch: dispatch,
annotate: annotate
});var actor = sponsor(function (message) {
console.log('got message', message);
});actor('foo');
```### sponsor(behavior)
Same as the core [Minimal](#minimal) implementation. _See: [sponsor(behavior)](#sponsorbehavior-1)_
### actor(message)
Same as the core [Minimal](#minimal) implementation. _See: [actor(message)](#actormessage-1)_
## Releases
We follow semantic versioning policy ([semver.org](http://semver.org/)):
>Given a version number MAJOR.MINOR.PATCH, increment the:
>
>MAJOR version when you make incompatible API changes,
>MINOR version when you add functionality in a backwards-compatible manner, and
>PATCH version when you make backwards-compatible bug fixes.## Sources
* [Tiny Actor Run-Time](https://github.com/organix/tart)