Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/sdesalas/botbrains
Robot brain tools to automate your nodebot
https://github.com/sdesalas/botbrains
3d arduino artificial-learning neural-network nodebots nodejs robot webgl
Last synced: 3 months ago
JSON representation
Robot brain tools to automate your nodebot
- Host: GitHub
- URL: https://github.com/sdesalas/botbrains
- Owner: sdesalas
- License: gpl-3.0
- Created: 2017-05-28T07:24:15.000Z (over 7 years ago)
- Default Branch: master
- Last Pushed: 2021-04-15T14:33:02.000Z (over 3 years ago)
- Last Synced: 2024-05-09T12:24:27.605Z (8 months ago)
- Topics: 3d, arduino, artificial-learning, neural-network, nodebots, nodejs, robot, webgl
- Language: JavaScript
- Homepage:
- Size: 4.63 MB
- Stars: 11
- Watchers: 2
- Forks: 2
- Open Issues: 5
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
[![Build Status](https://travis-ci.org/sdesalas/botbrains.svg?branch=master)](https://travis-ci.org/sdesalas/botbrains)
# BotBrains
BotBrains is a set of artificial learning tools to automate an [Arduino](http://arduino.org)-based robot.
Its been built as part of an educational workshop on artificial learning specifically for [International Nodebots Day](https://www.eventbrite.com.au/e/international-nodebots-day-melbourne-2017-tickets-34845310261).
This material here is very basic and aimed more at communicating the core concept of a neural network through practice than dealing with all the theoretical stuff that is available out there.
To read more about how this library came to be see the article on [Asynchronous Neural Networks](http://desalasworks.com/article/asynchronous-neural-networks-in-javascript/) on my blog.
## Interact with your robot brains in 3D.
The key aspect of BotBrains is the ability to watch signals travel across your robot's neural network in 3D, and train it with positive and negative reinforcement.
![brain-3d.png](brain-3d.gif)
## Quick start
You need [NodeJS](https://nodejs.org/en/download/) installed, version 6 or above.
```sh
$ mkdir my-bot && cd my-bot
$ npm install botbrains
$ cd node_modules/botbrains
$ npm start
```The above will perform a quick test with a random visualization. For a full test you need to rig up a robot.
## Proper setup
You should be adding this to an existing robot project such as [johnny-five](http://johnny-five.io/).
Here is a longer example:
```sh
$ mkdir my-bot && cd my-bot
$ npm install johnny-five
$ npm install botbrains
```### robot.js
```js
const five = require('johnny-five');
const botbrains = require('botbrains');const board = new five.Board({port: process.argv[2] || '' });
board.on('ready', () => {
const network = new botbrains.NeuralNetwork(32);
// PROXIMITY SENSOR INPUT (pin A6)
const sensor = new five.Sensor({ pin: 'A6', freq: 200 });
sensor.on('change', () => network.input('Proximity')(sensor.value / 1024));// MOTOR OUTPUT (pins D6-D8)
const left_motor = new five.Motor({ pins: { pwm: 6, dir: 7, }, invertPWM: true, });
const right_motor = new five.Motor({ pins: { pwm: 9, dir: 8, }, invertPWM: true, });// Output binding can be reasonably random.
// It doesn't matter what gets mapped to what
// since the robot will learn to coordinate itself
// using positive and negative feedback.network.output('Wheel (L)')
.on('data', (power) => { // between 0 and 1
const speed = Math.floor(power * 255);
if (power > 0.25) left_motor.forward(speed);
else left_motor.stop();
});network.output('Wheel (R)')
.on('data', (power) => { // between 0 and 1
const speed = Math.floor(power * 255);
if (power > 0.25) right_motor.forward(speed);
else right_motor.stop();
});// DISPLAY VIA LOCAHOST (http.Server)
const server = botbrains.Toolkit.visualise(network);
const address = server.address();console.log('Bot brain ready for interaction. Please open http://localhost:' + address.port);
});
```Then run it!
```sh
$ node robot.js
```
## API :: NeuralNetworkNeuralNetwork is a class in the botbrains module and can be loaded in the following ways:
```js
import { NeuralNetwork } from 'botbrains'; // ES6const NeuralNetwork = require('botbrains').NeuralNetwork; // Node, CommonJS
const NeuralNetwork = (window || this).botbrains.NeuralNetwork; // Browser, in global context
```### new NeuralNetwork(size, opts)
Generates a neural network.
- **`size`**: The number of neurons in the neural network
- **`opts`**: A map of settings for the network
- `.shape`: Choice of 'drum', 'tube', 'ring', 'ball', 'sausage', 'snake', or any other in [NetworkShaper.js](src/NetworkShaper.js). Defaults to 'drum'.
- `.shaper`: The [shaper function](#shaper-function) used for giving shape to the network. If available will ignore `.shape`.
- `.connectionsPerNeuron`: Average synapses per neuron. Defaults to `4`.
- `.signalFireThreshold`: Threshold (between 0 and 1) needed to trigger onward neurons. Defaults to `0.3`.
- `.signalSpeed`: Speed in neurons per second. Defaults to `20`.
- `.startingWeight`: Starting weight per synapse. Defaults to `signalFireThreshold(/3 x connectionsPerNeuron)`.
- `.neuronRecovery`: Neuron recovery as fraction of `signalSpeed`. Defaults to `1/5`.
- `.learningRate`: Max increase/decrease to connection strength when learning. Defaults to `0.05`.
- `.learningPeriod`: Milliseconds in the past on which learning applies. Defaults to `20 * 1000` ms.For example, to create a network of 100 neurons using all default options:
```js
let network = new NeuralNetwork(100);
```To create a ring-shaped network of 100 neurons with double the speed and learning rate.
```js
let network = new NeuralNetwork(1000, { shape: 'ring', signalSpeed: 40, learningRate: 0.3 });
```If a `String` is passed in as the `opts` parameter, its interpreted as the network shape.
```js
let network = new NeuralNetwork(100, 'ring');
```If a `Function` is passed as the `opts` parameter, its interpreted as the [shaper function](#shaper-function), see examples in [NetworkShaper.js](src/NetworkShaper.js).
```js
let network = new NeuralNetwork(100, (count) => Math.floor(Math.random() * count));
```### network.input(label [, neurons=1])
Creates an input into the network.
- **`label`**: A label used to visually identify the neurons involved.
- **`neurons`**: Optional `Number` of neurons involved, or array of numbers (`Number[]`) defining the network nodes that are involved in the input.Returns:
- **`Function(signal 0-1)`**: A function that accepts incoming signals (a float between 0 and 1).
Usage:
```js
var input1 = network.input('Microphone') // 1 x Neuron assigned automatically
sensor.on('data', data => input1(data / 1024));var input2 = network.input('LightSensor (L)', 3) // How many neurons? => 3
sensor.on('data', data => input2(data / 1024));var input3 = network.input('LightSensor (R)', [10,11,12]) // Which neurons? => 10, 11 & 12
sensor.on('data', data => input3(data / 1024));
```### network.output(label [, neurons=1])
Returns an [EventEmitter](https://nodejs.org/api/events.html#events_class_eventemitter) with 2 events: `data` and `change`.
- **`label`**: A label used to visually identify the neurons involved.
- **`neurons`**: Optional `Number` of neurons involved, or array of numbers (`Number[]`) defining the network nodes that are involved in the input.**Event: `data`**:
Fires whenever there is data to output.
The event handler function will receive the following arguments.
- **`data`**: A numeric floating point value between 0 and 1, containing the strength of the outgoing signal.
```js
network.output('Motor (L)') // 1 x Neuron assigned automatically
.on('data', (pwr) => console.log(`Power: ${pwr}.`));network.output('Motor (R)', 4) // How many neurons? => 3
.on('data', (pwr) => console.log(`Power: ${pwr}.`));network.output('Buzzer', [6,7,8]) // Which neurons? => 6, 7 & 8
.on('data', (pwr) => console.log(`Power: ${pwr}.`));
```**Event: `change`**:
An event that fires whenever there is a change in the outgoing data.
The event handler function will receive the following arguments.
- **`data`**: A numeric floating point value between 0 and 1, containing the strength of the outgoing signal.
- **`last`**: The previous value output.
- **`diff`**: The difference between `last` and `data`.```js
network.output('Buzzer')
.on('change', function(data, last, diff) {
console.log(`Buzzer output is: ${data}. Previous output was ${last}. Difference is ${diff}`);
});
```### network.join(network, at, surfaceArea)
Merges a network into another network, or into itself.
```js
const network = new NeuralNetwork(100, 'ball');
network.join(new NeuralNetwork(100, 'ring'), 0.5, 0.1); // join on the middle, using 10% of nodes.
network.join(network); // join on itself
```### Shaper Function
A shaper is a function that determines the shape of the network by returning the onward connections made by each neuron.
For example, if a neuron is connected to other neurons at random, the final shape of the network will be a ball. If its connected to nearby neurons the shape will be more of a snake or cylinder. If neurons close to the end are linked to neurons at the beginning, the end product will be more of a ring or a doughnut.
The shaper function is executed *once for every synapse in the network*. If there are 10 nodes, and 4 synapses per node, it will fire 40 times to determine the onward neuron in each of those synapses.
![shaper.png](shaper.png)
A shaper function has the following inputs:
- **`count`**: The total number of nodes in the network. In other words, in a network of 10 neurons, this will be `10`. Useful for linking up the end of the network back to its beginning or for discarding links outside the network.
- **`index`**: The position of the originating neuron inside the `nodes` array. In a network of 10 nodes, the first node is `0`, the last node is `9`, so a value of `9` means the last neuron in the network.
- **`connectionCount`**: A neuron has several connections (synapses) originating from it. For example if the neuron has 4 connections, this will be 4.
- **`connectionIndex`**: The position in the array of the connection currently being made. In other words, if there are 4 connections (synapses) in the originating neuron, the shaper function will execute 4 times for it, with a `connectionIndex` value of `0` to `3` accordingly.And returns:
- **`target`**: The destination position of an onward neuron (for connecting to it). In a network of 10 nodes, when we are mapping a neuron `index` of `0` (the first node) and `connectionIndex` of `0` (the first synapse), a `target` of `9` means that connection is made towards the last neuron.
Bear in mind that since the shaper function will be executed *multiple times per neuron* to determine the onward neuron for each connection. You will want these to connect to different onward neurons. This can be done either choosing an onward `target` at random, or by using the `synapseIndex` argument to calculate the `target` neuron in a deterministic manner.
Here is an example of simple shaper function:
```js
// Random ball shape
new NeuralNetwork(100, (count, index) => Math.floor(Math.random() * count));
```Another more involved example:
```js
// Ring shape
const network = new NeuralNetwork(100, ring);function ring(count, index) {
const width = count / 12;
const forwardBias = Math.ceil(width * Math.random());
const target = index + forwardBias;
if (target < count) {
return target;
}
return target - count; // link to beginning
}
```There are more examples in [NetworkShaper.js](src/NetworkShaper.js).
**Why does shape matter?**
The Neural Network model used for [botbrains](https://www.npmjs.com/package/botbrains) is [asynchronous](https://en.wikipedia.org/wiki/Asynchrony_(computer_programming)). Signal propagate across the network in the same manner as they would in an animal brain, one neuron at a time. Different shapes matter because they create resonance and oscillation patterns that are important for producing particular outputs to inputs in a time-dependent manner.