mirror of
https://github.com/zerotier/ZeroTierOne.git
synced 2025-01-18 02:40:13 +00:00
Clean old netconf-service from attic.
This commit is contained in:
parent
fbb990f8a3
commit
e184aa4cb4
1
.gitignore
vendored
1
.gitignore
vendored
@ -25,6 +25,7 @@
|
||||
*.obj
|
||||
*.tlog
|
||||
*.pid
|
||||
*.pkg
|
||||
/*.deb
|
||||
/*.rpm
|
||||
/build-*
|
||||
|
@ -1,51 +0,0 @@
|
||||
# ZeroTier One Network Configuration Service
|
||||
|
||||
## What is it?
|
||||
|
||||
It's the thing that controls virtual networks. It's a completely separate subsystem from supernodes; see the [Technical FAQ](https://github.com/zerotier/ZeroTierOne/wiki/Technical-FAQ) for more information on that.
|
||||
|
||||
ZeroTier's 16-digit / 64-bit network IDs are actually two numbers packed together into one. The most significant 40 bits / first 10 digits of a network ID are the ZeroTier address of the *network configuration master* responsible for issuing network configurations and certificates to members of the network. The least significant 24 bits / last 6 digits are an arbitrary 24-bit number used to identify this network on its given master.
|
||||
|
||||
When a ZeroTier node joins a network or updates its network configuration, it queries the network configuration master and receives a response containing either an error or a dictionary with that node's membership information and (if it's a private network) membership certificate.
|
||||
|
||||
Network configuration masters can go offline without affecting communication on the network, since they're only needed when it's necessary to issue new configurations. (Certificates are a more involved topic that's beyond the scope of this document, but suffice to say that the same applies there.) They can also be made fault tolerant by mirroring their identities and databases to a backup server that can take over if the main server dies.
|
||||
|
||||
Networks managed through the *zerotier.com* site are managed by network configuration masters run by ZeroTier networks, but if you're willing to do a bit of manual system administration you can set up and run your own.
|
||||
|
||||
## Installation
|
||||
|
||||
The first step is to choose a node to act as a netconf master. The netconf master subprocess is only supported on Unix-like platforms and has only been tested so far on Linux, but will probably work on Mac as well. But Windows builds don't support service subprocesses so the master cannot be a Windows node.
|
||||
|
||||
At the moment, netconf masters *cannot join their own networks*. We recommend using a node that you don't intend to make a member of the networks it administrates. Or, as an alternative, you can run ZeroTier One on a different port and with a different home folder. This is what we do, and on our masters we run it separately on startup with a script that executes "sudo /var/lib/zerotier-one/zerotier-one -p9994 /var/lib/zerotier-one-netconf-master &". This runs the service on UDP port 9994 and uses "/var/lib/zerotier-one-netconf-master" as its home folder. *Two or more instances of ZeroTier One can co-exist on the same machine as long as only one of them tries to actually join networks.*
|
||||
|
||||
Before setting up, three prerequisites must be installed: [Node](http://nodejs.org), npm (node's package manager), and Redis (a NoSQL database). On most Linux distributions there are packages for all these, so installing them should be easy. On CentOS/RedHat based distributions they are in the [EPEL](https://fedoraproject.org/wiki/EPEL) repository, so after installing or enabling EPEL you can just type "sudo yum install nodejs npm redis". Then start the Redis service with "sudo service redis start" and enable it on boot with "sudo chkconfig redis on". (Commands may differ on other Linux distributions.) Verify that it's running by typing "redis-cli" and testing whether it connects.
|
||||
|
||||
Once prerequisites are installed, follow these instructions:
|
||||
|
||||
1) Go to the ZeroTier home (for the node you plan to designate as master) and create a subfolder called "services.d".
|
||||
|
||||
2) Copy or symlink "netconf-master" from this repository into "services.d".
|
||||
|
||||
3) In the "services.d" folder, create a symlink called "netconf.service" that points to "netconf-master/netconf.service". Also check to make sure that "netconf-master/netconf.service" is executable. This is what the ZeroTier One service will execute to launch the subprocess.
|
||||
|
||||
4) In "services.d/netconf-master" type "npm install" to install NodeJS package dependencies.
|
||||
|
||||
5) Edit the initdb.js and change the network ID and other settings of the network you want to create. Remember how network IDs are made; the *first 10 digits* of your network's ID *must be the ZeroTier address of your netconf master node*. If it doesn't have one yet, try starting the ZeroTier service briefly and then killing it and it will generate *identity.public* and *identity.secret* automatically. The address is the first hex field in either of these files.
|
||||
|
||||
6) Run "node initdb.js" to initialize your Redis database. (If you want to use a Redis database other than 0, such as to avoid polluting another Redis database on the same machine, edit "config.js" and change the Redis database index there before initializing the database or running the service.)
|
||||
|
||||
7) Start or restart the ZeroTier One service. Check its "node.log" and your system process list to ensure that it successfully started the netconf server slave process.
|
||||
|
||||
## Using Your Own Networks
|
||||
|
||||
Once the netconf master service is running, you can test it by simply joining the network ID of the network you created from any ZeroTier One node anywhere in the world. If everything is set up correctly, it should just work. Your netconf service is controlling this network.
|
||||
|
||||
## Private Networks and Certificates
|
||||
|
||||
If you set the "private" flag to "1" and designated your network as private, members must be authorized. Each time a member attempts to join a network, a member record is created in Redis. These will have Redis names like "zt1:network:################:member:##########:~". Open this hash in any Redis database editor (or use hgetall and hset in redis-cli) and set "authorized" to "1" for whichever members you wish to authorize.
|
||||
|
||||
## What About the Web UI?
|
||||
|
||||
At the moment, the web-based administration interface used on zerotier.com is the only part of the system that is not open source. Right now it's all tied in with our payment processor, and our model right now is to charge for the convenience of using it.
|
||||
|
||||
We *might* open source it in the future, but in the meantime anyone with a bit of developer expertise should be able to write a few scripts to list networks, change network parameters, authorize members on private networks, etc. The Redis database schema is documented in *redis-schema.md*, and only the stuff under "zt1:network:..." is actually used by the netconf master service. The stuff under "zt1:user:..." is used by our web UI and doesn't need to be present for the netconf service to operate.
|
@ -1,3 +0,0 @@
|
||||
exports.redisDb = 0; // live
|
||||
//exports.redisDb = 1; // test
|
||||
//exports.redisDb = 2; // dev
|
@ -1,73 +0,0 @@
|
||||
/*
|
||||
* Populates a new Redis database with data, which can be edited below.
|
||||
*/
|
||||
|
||||
var INIT_DATA = {
|
||||
// Must be present in any database
|
||||
"zt1": 1,
|
||||
|
||||
/* The network ID here must be set to the ZeroTier address of your netconf
|
||||
* master (the node where netconf-master will be running) plus an arbitrary
|
||||
* 24-bit network ID. This will create the full 16-digit network ID of the
|
||||
* network you will join. This must be in the object name and in the "id"
|
||||
* field within the object itself. */
|
||||
"zt1:network:31443a1a0a111111:~": {
|
||||
"id": "31443a1a0a111111", // netconf master ZT address + 24-bit ID
|
||||
"name": "zerotier-testnet", // short name, no spaces or special chars
|
||||
"desc": "Test Network", // description
|
||||
"infrastructure": 0, // unused by netconf-master
|
||||
"private": 0, // set to '1' to require member approval
|
||||
"creationTime": 0, // unused by netconf-master
|
||||
"owner": "", // unused by netconf-master
|
||||
"etherTypes": "0800,0806", // hex ethernet frame types allowed
|
||||
"enableBroadcast": 1, // set to '1' to enable ff:ff:ff:ff:ff:ff
|
||||
"v4AssignMode": "zt", // 'zt' to assign, 'none' to let OS do it
|
||||
"v4AssignPool": "192.168.123.0/24", // IPv4 net block / netmask bits
|
||||
"v6AssignMode": "none" // 'zt' to assign, 'none' to let OS do it
|
||||
}
|
||||
};
|
||||
|
||||
var config = require('./config.js');
|
||||
|
||||
var async = require('async');
|
||||
var redis = require('redis');
|
||||
var DB = redis.createClient();
|
||||
DB.on("error",function(err) { console.error('redis query error: '+err); });
|
||||
DB.select(config.redisDb,function() {});
|
||||
|
||||
DB.get("zt1",function(err,value) {
|
||||
if ((value)&&(!err)) {
|
||||
console.log("Redis database #"+config.redisDb+" appears to already contain data; flush it first!");
|
||||
return process.exit(0);
|
||||
}
|
||||
|
||||
async.eachSeries(Object.keys(INIT_DATA),function(key,next) {
|
||||
var value = INIT_DATA[key];
|
||||
if (typeof value === 'object') {
|
||||
console.log(key);
|
||||
async.eachSeries(Object.keys(value),function(hkey,next2) {
|
||||
var hvalue = value[hkey];
|
||||
if (hvalue === true)
|
||||
hvalue = 1;
|
||||
if (hvalue === false)
|
||||
hvalue = 0;
|
||||
if (typeof hvalue !== 'string')
|
||||
hvalue = hvalue.toString();
|
||||
console.log('\t'+hkey+': '+hvalue);
|
||||
DB.hset(key,hkey,hvalue,next2);
|
||||
},next);
|
||||
} else if ((typeof value !== 'undefined')&&(value !== null)) {
|
||||
if (value === true)
|
||||
value = 1;
|
||||
if (value === false)
|
||||
value = 0;
|
||||
if (typeof value !== 'string')
|
||||
value = value.toString();
|
||||
console.log(key+': '+value);
|
||||
DB.set(key,value,next);
|
||||
} else return next(null);
|
||||
},function(err) {
|
||||
console.log('Done!');
|
||||
return process.exit(0);
|
||||
});
|
||||
});
|
@ -1,624 +0,0 @@
|
||||
//
|
||||
// ZeroTier One - Network Virtualization Everywhere
|
||||
// Copyright (C) 2011-2015 ZeroTier, Inc.
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// This program is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
//
|
||||
// --
|
||||
//
|
||||
// ZeroTier may be used and distributed under the terms of the GPLv3, which
|
||||
// are available at: http://www.gnu.org/licenses/gpl-3.0.html
|
||||
//
|
||||
// If you would like to embed ZeroTier into a commercial application or
|
||||
// redistribute it in a modified binary form, please contact ZeroTier Networks
|
||||
// LLC. Start here: http://www.zerotier.com/
|
||||
//
|
||||
|
||||
var config = require('./config.js');
|
||||
|
||||
// Fields in netconf response dictionary
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_ALLOWED_ETHERNET_TYPES = "et";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_NETWORK_ID = "nwid";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_TIMESTAMP = "ts";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_ISSUED_TO = "id";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_MULTICAST_LIMIT = "ml";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_MULTICAST_RATES = "mr";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_PRIVATE = "p";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_NAME = "n";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_DESC = "d";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_IPV4_STATIC = "v4s";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_IPV6_STATIC = "v6s";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_CERTIFICATE_OF_MEMBERSHIP = "com";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_ENABLE_BROADCAST = "eb";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_ALLOW_PASSIVE_BRIDGING = "pb";
|
||||
var ZT_NETWORKCONFIG_DICT_KEY_ACTIVE_BRIDGES = "ab";
|
||||
|
||||
// Path to zerotier-idtool binary, invoked to enerate certificates of membership
|
||||
var ZEROTIER_IDTOOL = '/usr/local/bin/zerotier-idtool';
|
||||
|
||||
// From Constants.hpp in node/
|
||||
var ZT_NETWORK_AUTOCONF_DELAY = 60000;
|
||||
var ZT_NETWORK_CERTIFICATE_TTL_WINDOW = (ZT_NETWORK_AUTOCONF_DELAY * 4);
|
||||
|
||||
// Connect to redis, assuming database 0 and no auth (for now)
|
||||
var async = require('async');
|
||||
var redis = require('redis');
|
||||
var DB = redis.createClient();
|
||||
DB.on("error",function(err) { console.error('redis query error: '+err); });
|
||||
DB.select(config.redisDb,function() {});
|
||||
|
||||
// Global variables -- these are initialized on startup or netconf-init message
|
||||
var netconfSigningIdentity = null; // identity of netconf master, with private key portion
|
||||
|
||||
// spawn() function to launch sub-processes
|
||||
var spawn = require('child_process').spawn;
|
||||
|
||||
// Returns true for fields that are "true" according to ZT redis schema
|
||||
function ztDbTrue(v) { return ((v === '1')||(v === 'true')||(v > 0)); }
|
||||
|
||||
//
|
||||
// ZeroTier One Dictionary -- encoding-compatible with Dictionary in C++ code base
|
||||
//
|
||||
|
||||
function Dictionary(fromStr)
|
||||
{
|
||||
var self = this;
|
||||
|
||||
this.data = {};
|
||||
|
||||
this._esc = function(data) {
|
||||
var es = '';
|
||||
for(var i=0;i<data.length;++i) {
|
||||
var c = data.charAt(i);
|
||||
switch(c) {
|
||||
case '\0': es += '\\0'; break;
|
||||
case '\r': es += '\\r'; break;
|
||||
case '\n': es += '\\n'; break;
|
||||
case '\\': es += '\\\\'; break;
|
||||
case '=': es += '\\='; break;
|
||||
default: es += c; break;
|
||||
}
|
||||
}
|
||||
return es;
|
||||
};
|
||||
this._unesc = function(s) {
|
||||
if (typeof s !== 'string')
|
||||
return '';
|
||||
var uns = '';
|
||||
var escapeState = false;
|
||||
for(var i=0;i<s.length;++i) {
|
||||
var c = s.charAt(i);
|
||||
if (escapeState) {
|
||||
escapeState = false;
|
||||
switch(c) {
|
||||
case '0': uns += '\0'; break;
|
||||
case 'r': uns += '\r'; break;
|
||||
case 'n': uns += '\n'; break;
|
||||
default: uns += c; break;
|
||||
}
|
||||
} else{
|
||||
if ((c !== '\r')&&(c !== '\n')&&(c !== '\0')) {
|
||||
if (c === '\\')
|
||||
escapeState = true;
|
||||
else uns += c;
|
||||
}
|
||||
}
|
||||
}
|
||||
return uns;
|
||||
};
|
||||
|
||||
this.toString = function() {
|
||||
var str = '';
|
||||
|
||||
for(var key in self.data) {
|
||||
str += self._esc(key);
|
||||
str += '=';
|
||||
var value = self.data[key];
|
||||
if (value)
|
||||
str += self._esc(value.toString());
|
||||
str += '\n';
|
||||
}
|
||||
|
||||
return str;
|
||||
};
|
||||
|
||||
this.fromString = function(str) {
|
||||
self.data = {};
|
||||
if (typeof str !== 'string')
|
||||
return self;
|
||||
|
||||
var lines = str.split('\n');
|
||||
for(var l=0;l<lines.length;++l) {
|
||||
var escapeState = false;
|
||||
var eqAt = 0;
|
||||
for(;eqAt<lines[l].length;++eqAt) {
|
||||
var c = lines[l].charAt(eqAt);
|
||||
if (escapeState)
|
||||
escapeState = false;
|
||||
else if (c === '\\')
|
||||
escapeState = true;
|
||||
else if (c === '=')
|
||||
break;
|
||||
}
|
||||
|
||||
var k = self._unesc(lines[l].substr(0,eqAt));
|
||||
++eqAt;
|
||||
if ((k)&&(k.length > 0))
|
||||
self.data[k] = self._unesc((eqAt < lines[l].length) ? lines[l].substr(eqAt) : '');
|
||||
}
|
||||
|
||||
return self;
|
||||
};
|
||||
|
||||
if ((typeof fromStr === 'string')&&(fromStr.length > 0))
|
||||
self.fromString(fromStr);
|
||||
};
|
||||
|
||||
//
|
||||
// Identity implementation using zerotier-idtool as subprocess to do actual crypto work
|
||||
//
|
||||
|
||||
function Identity(idstr)
|
||||
{
|
||||
var self = this;
|
||||
|
||||
this.str = '';
|
||||
this.fields = [];
|
||||
|
||||
this.toString = function() {
|
||||
return self.str;
|
||||
};
|
||||
|
||||
this.address = function() {
|
||||
return ((self.fields.length > 0) ? self.fields[0] : '0000000000');
|
||||
};
|
||||
|
||||
this.fromString = function(str) {
|
||||
self.str = '';
|
||||
self.fields = [];
|
||||
if (typeof str !== 'string')
|
||||
return;
|
||||
for(var i=0;i<str.length;++i) {
|
||||
if ("0123456789abcdef:".indexOf(str.charAt(i)) < 0)
|
||||
return; // invalid character in identity
|
||||
}
|
||||
var fields = str.split(':');
|
||||
if ((fields.length < 3)||(fields[0].length !== 10)||(fields[1] !== '0'))
|
||||
return;
|
||||
self.str = str;
|
||||
self.fields = fields;
|
||||
};
|
||||
|
||||
this.isValid = function() {
|
||||
return (! ((self.fields.length < 3)||(self.fields[0].length !== 10)||(self.fields[1] !== '0')) );
|
||||
};
|
||||
|
||||
this.hasPrivate = function() {
|
||||
return ((self.isValid())&&(self.fields.length >= 4));
|
||||
};
|
||||
|
||||
if (typeof idstr === 'string')
|
||||
self.fromString(idstr);
|
||||
};
|
||||
|
||||
//
|
||||
// Invokes zerotier-idtool to generate certificates for private networks
|
||||
//
|
||||
|
||||
function generateCertificateOfMembership(nwid,peerAddress,callback)
|
||||
{
|
||||
// The first fields of these COM tuples come from
|
||||
// CertificateOfMembership.hpp's enum of required
|
||||
// certificate default fields.
|
||||
var comTimestamp = '0,' + Date.now().toString(16) + ',' + ZT_NETWORK_CERTIFICATE_TTL_WINDOW.toString(16);
|
||||
var comNwid = '1,' + nwid + ',0';
|
||||
var comIssuedTo = '2,' + peerAddress + ',ffffffffffffffff';
|
||||
|
||||
var cert = '';
|
||||
var certErr = '';
|
||||
|
||||
var idtool = spawn(ZEROTIER_IDTOOL,[ 'mkcom',netconfSigningIdentity,comTimestamp,comNwid,comIssuedTo ]);
|
||||
idtool.stdout.on('data',function(data) {
|
||||
cert += data;
|
||||
});
|
||||
idtool.stderr.on('data',function(data) {
|
||||
certErr += data;
|
||||
});
|
||||
idtool.on('close',function(exitCode) {
|
||||
if (certErr.length > 0)
|
||||
console.error('zerotier-idtool stderr returned: '+certErr);
|
||||
return callback((cert.length > 0) ? cert : null,exitCode);
|
||||
});
|
||||
}
|
||||
|
||||
//
|
||||
// Message handler for messages over ZeroTier One service bus
|
||||
//
|
||||
|
||||
function doNetconfInit(message)
|
||||
{
|
||||
netconfSigningIdentity = new Identity(message.data['netconfId']);
|
||||
if (!netconfSigningIdentity.hasPrivate()) {
|
||||
netconfSigningIdentity = null;
|
||||
console.error('got invalid netconf signing identity in netconf-init');
|
||||
} // else console.error('got netconf-init, running! id: '+netconfSigningIdentity.address());
|
||||
}
|
||||
|
||||
function doNetconfRequest(message)
|
||||
{
|
||||
if ((netconfSigningIdentity === null)||(!netconfSigningIdentity.hasPrivate())) {
|
||||
console.error('got netconf-request before netconf-init, ignored');
|
||||
return;
|
||||
}
|
||||
|
||||
var peerId = new Identity(message.data['peerId']);
|
||||
var nwid = message.data['nwid'];
|
||||
var requestId = message.data['requestId'];
|
||||
if ((!peerId)||(!peerId.isValid())||(!nwid)||(nwid.length !== 16)||(!requestId)) {
|
||||
console.error('missing one or more required fields in netconf-request');
|
||||
return;
|
||||
}
|
||||
|
||||
var networkKey = 'zt1:network:'+nwid+':~';
|
||||
var memberKey = 'zt1:network:'+nwid+':member:'+peerId.address()+':~';
|
||||
var ipAssignmentsKey = 'zt1:network:'+nwid+':ipAssignments';
|
||||
|
||||
var network = null;
|
||||
var member = null;
|
||||
|
||||
var authorized = false;
|
||||
|
||||
var v4NeedAssign = false;
|
||||
var v6NeedAssign = false;
|
||||
var v4Assignments = [];
|
||||
var v6Assignments = [];
|
||||
var ipAssignments = []; // both v4 and v6
|
||||
var activeBridges = '';
|
||||
|
||||
async.series([function(next) {
|
||||
|
||||
// network lookup
|
||||
DB.hgetall(networkKey,function(err,obj) {
|
||||
if ((!err)&&(obj)&&(obj.id === nwid))
|
||||
network = obj;
|
||||
return next(null);
|
||||
});
|
||||
|
||||
},function(next) {
|
||||
|
||||
// member lookup
|
||||
if (!network)
|
||||
return next(null);
|
||||
|
||||
DB.hgetall(memberKey,function(err,obj) {
|
||||
if (err)
|
||||
return next(err);
|
||||
|
||||
if (obj) {
|
||||
// Update existing member record with new last seen time, etc.
|
||||
member = obj;
|
||||
authorized = ((!ztDbTrue(network['private'])) || ztDbTrue(member['authorized']));
|
||||
var updatedFields = {
|
||||
'lastSeen': Date.now(),
|
||||
'authorized': authorized ? '1' : '0' // reset authorized to unhide in UI, since UI uses -1 to hide
|
||||
};
|
||||
if (!('identity' in member))
|
||||
updatedFields['identity'] = peerId.toString();
|
||||
if (!('firstSeen' in member))
|
||||
updatedFields['firstSeen'] = Date.now();
|
||||
if (message.data['from'])
|
||||
updatedFields['lastAt'] = message.data['from'];
|
||||
if (message.data['clientVersion'])
|
||||
updatedFields['clientVersion'] = message.data['clientVersion'];
|
||||
if (message.data['clientOs'])
|
||||
updatedFields['clientOs'] = message.data['clientOs'];
|
||||
DB.hmset(memberKey,updatedFields,next);
|
||||
} else {
|
||||
// Add member record to network for newly seen peer
|
||||
authorized = ztDbTrue(network['private']) ? false : true; // public networks authorize everyone by default
|
||||
var now = Date.now().toString();
|
||||
member = {
|
||||
'id': peerId.address(),
|
||||
'nwid': nwid,
|
||||
'authorized': authorized ? '1' : '0',
|
||||
'identity': peerId.toString(),
|
||||
'firstSeen': now,
|
||||
'lastSeen': now
|
||||
};
|
||||
if (message.data['from'])
|
||||
member['lastAt'] = message.data['from'];
|
||||
if (message.data['clientVersion'])
|
||||
member['clientVersion'] = message.data['clientVersion'];
|
||||
if (message.data['clientOs'])
|
||||
member['clientOs'] = message.data['clientOs'];
|
||||
DB.hmset(memberKey,member,next);
|
||||
}
|
||||
});
|
||||
|
||||
},function(next) {
|
||||
|
||||
// Figure out which IP address auto-assignments we need to look up or make
|
||||
if ((!network)||(!authorized))
|
||||
return next(null);
|
||||
|
||||
v4NeedAssign = (network['v4AssignMode'] === 'zt');
|
||||
v6NeedAssign = (network['v6AssignMode'] === 'zt');
|
||||
|
||||
var ipacsv = member['ipAssignments'];
|
||||
if (ipacsv) {
|
||||
var ipa = ipacsv.split(',');
|
||||
for(var i=0;i<ipa.length;++i) {
|
||||
if (ipa[i]) {
|
||||
ipAssignments.push(ipa[i]);
|
||||
if ((ipa[i].indexOf('.') > 0)&&(v4NeedAssign))
|
||||
v4Assignments.push(ipa[i]);
|
||||
else if ((ipa[i].indexOf(':') > 0)&&(v6NeedAssign))
|
||||
v6Assignments.push(ipa[i]);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return next(null);
|
||||
|
||||
},function(next) {
|
||||
|
||||
// assign IPv4 if needed
|
||||
if ((!network)||(!authorized)||(!v4NeedAssign)||(v4Assignments.length > 0))
|
||||
return next(null);
|
||||
|
||||
var peerAddress = peerId.address();
|
||||
|
||||
var ipnetwork = 0;
|
||||
var netmask = 0;
|
||||
var netmaskBits = 0;
|
||||
var v4pool = network['v4AssignPool']; // technically csv but only one netblock currently supported
|
||||
if (v4pool) {
|
||||
var v4poolSplit = v4pool.split('/');
|
||||
if (v4poolSplit.length === 2) {
|
||||
var networkSplit = v4poolSplit[0].split('.');
|
||||
if (networkSplit.length === 4) {
|
||||
ipnetwork |= (parseInt(networkSplit[0],10) << 24) & 0xff000000;
|
||||
ipnetwork |= (parseInt(networkSplit[1],10) << 16) & 0x00ff0000;
|
||||
ipnetwork |= (parseInt(networkSplit[2],10) << 8) & 0x0000ff00;
|
||||
ipnetwork |= parseInt(networkSplit[3],10) & 0x000000ff;
|
||||
netmaskBits = parseInt(v4poolSplit[1],10);
|
||||
if (netmaskBits > 32)
|
||||
netmaskBits = 32; // sanity check
|
||||
for(var i=0;i<netmaskBits;++i)
|
||||
netmask |= (0x80000000 >> i);
|
||||
netmask &= 0xffffffff;
|
||||
}
|
||||
}
|
||||
}
|
||||
if ((ipnetwork === 0)||(netmask === 0xffffffff))
|
||||
return next(null);
|
||||
var invmask = netmask ^ 0xffffffff;
|
||||
|
||||
var abcd = 0;
|
||||
var ipAssignmentAttempts = 0;
|
||||
|
||||
async.whilst(
|
||||
function() { return ((v4Assignments.length === 0)&&(ipAssignmentAttempts < 1000)); },
|
||||
function(next2) {
|
||||
++ipAssignmentAttempts;
|
||||
|
||||
// Generate or increment IP address source bits
|
||||
if (abcd === 0) {
|
||||
var a = parseInt(peerAddress.substr(2,2),16) & 0xff;
|
||||
var b = parseInt(peerAddress.substr(4,2),16) & 0xff;
|
||||
var c = parseInt(peerAddress.substr(6,2),16) & 0xff;
|
||||
var d = parseInt(peerAddress.substr(8,2),16) & 0xff;
|
||||
abcd = (a << 24) | (b << 16) | (c << 8) | d;
|
||||
} else ++abcd;
|
||||
if ((abcd & 0xff) === 0)
|
||||
abcd |= 1;
|
||||
abcd &= 0xffffffff;
|
||||
|
||||
// Derive an IP to test and generate assignment ip/bits string
|
||||
var ip = (abcd & invmask) | (ipnetwork & netmask);
|
||||
var assignment = ((ip >> 24) & 0xff).toString(10) + '.' + ((ip >> 16) & 0xff).toString(10) + '.' + ((ip >> 8) & 0xff).toString(10) + '.' + (ip & 0xff).toString(10) + '/' + netmaskBits.toString(10);
|
||||
|
||||
// Check :ipAssignments to see if this IP is already taken
|
||||
DB.hget(ipAssignmentsKey,assignment,function(err,value) {
|
||||
if (err)
|
||||
return next2(err);
|
||||
|
||||
// IP is already taken, try again via async.whilst()
|
||||
if ((value)&&(value !== peerAddress))
|
||||
return next2(null); // if someone's already got this IP, keep looking
|
||||
|
||||
v4Assignments.push(assignment);
|
||||
ipAssignments.push(assignment);
|
||||
|
||||
// Save assignment to :ipAssignments hash
|
||||
DB.hset(ipAssignmentsKey,assignment,peerAddress,function(err) {
|
||||
if (err)
|
||||
return next2(err);
|
||||
|
||||
// Save updated CSV list of assignments to member record
|
||||
var ipacsv = ipAssignments.join(',');
|
||||
member['ipAssignments'] = ipacsv;
|
||||
DB.hset(memberKey,'ipAssignments',ipacsv,next2);
|
||||
});
|
||||
});
|
||||
},
|
||||
next
|
||||
);
|
||||
|
||||
},function(next) {
|
||||
|
||||
// assign IPv6 if needed -- TODO
|
||||
if ((!network)||(!authorized)||(!v6NeedAssign)||(v6Assignments.length > 0))
|
||||
return next(null);
|
||||
|
||||
return next(null);
|
||||
|
||||
},function(next) {
|
||||
|
||||
// Get active bridges
|
||||
if ((!network)||(!authorized))
|
||||
return next(null);
|
||||
|
||||
DB.keys('zt1:network:'+nwid+':member:*:~',function(err,keys) {
|
||||
if (keys) {
|
||||
async.eachSeries(keys,function(key,nextKey) {
|
||||
DB.hgetall(key,function(err,abr) {
|
||||
if ( (abr) &&
|
||||
(abr.id) &&
|
||||
(abr.id.length === 10) &&
|
||||
( (!ztDbTrue(network['private'])) || ztDbTrue(abr['authorized']) ) &&
|
||||
(ztDbTrue(abr['activeBridge'])) ) {
|
||||
if (activeBridges.length)
|
||||
activeBridges += ',';
|
||||
activeBridges += abr.id;
|
||||
}
|
||||
return nextKey(null);
|
||||
});
|
||||
},next);
|
||||
} else return next(null);
|
||||
});
|
||||
|
||||
}],function(err) {
|
||||
|
||||
if (err) {
|
||||
console.error('error answering netconf-request for '+peerId.address()+': '+err);
|
||||
return;
|
||||
}
|
||||
|
||||
var response = new Dictionary();
|
||||
response.data['peer'] = peerId.address();
|
||||
response.data['nwid'] = nwid;
|
||||
response.data['type'] = 'netconf-response';
|
||||
response.data['requestId'] = requestId;
|
||||
|
||||
if ((network)&&(authorized)) {
|
||||
var certificateOfMembership = null;
|
||||
var privateNetwork = ztDbTrue(network['private']);
|
||||
|
||||
async.series([function(next) {
|
||||
|
||||
// Generate certificate of membership if necessary
|
||||
if (privateNetwork) {
|
||||
generateCertificateOfMembership(nwid,peerId.address(),function(cert,exitCode) {
|
||||
if (cert) {
|
||||
certificateOfMembership = cert;
|
||||
return next(null);
|
||||
} else return next(new Error('zerotier-idtool returned '+exitCode));
|
||||
});
|
||||
} else return next(null);
|
||||
|
||||
}],function(err) {
|
||||
|
||||
// Send response to parent process
|
||||
if (err) {
|
||||
console.error('unable to generate certificate for peer '+peerId.address()+' on network '+nwid+': '+err);
|
||||
response.data['error'] = 'ACCESS_DENIED'; // unable to generate certificate
|
||||
} else {
|
||||
var netconf = new Dictionary();
|
||||
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_ALLOWED_ETHERNET_TYPES] = network['etherTypes'];
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_NETWORK_ID] = nwid;
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_TIMESTAMP] = Date.now().toString(16);
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_ISSUED_TO] = peerId.address();
|
||||
if (network['multicastLimit'])
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_MULTICAST_LIMIT] = network['multicastLimit'];
|
||||
if (network['multicastRates']) {
|
||||
var ratesD = new Dictionary();
|
||||
var ratesJ = JSON.parse(network['multicastRates']);
|
||||
for(var k in ratesJ) {
|
||||
if ((k)&&(ratesJ[k]))
|
||||
ratesD.data[k] = ratesJ[k];
|
||||
}
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_MULTICAST_RATES] = ratesD.toString();
|
||||
}
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_PRIVATE] = privateNetwork ? '1' : '0';
|
||||
if (network['name'])
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_NAME] = network['name'];
|
||||
if (network['desc'])
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_DESC] = network['desc'];
|
||||
if ((v4NeedAssign)&&(v4Assignments.length > 0))
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_IPV4_STATIC] = v4Assignments.join(',');
|
||||
if ((v6NeedAssign)&&(v6Assignments.length > 0))
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_IPV6_STATIC] = v6Assignments.join(',');
|
||||
if (certificateOfMembership !== null)
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_CERTIFICATE_OF_MEMBERSHIP] = certificateOfMembership;
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_ENABLE_BROADCAST] = ztDbTrue(network['enableBroadcast']) ? '1' : '0';
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_ALLOW_PASSIVE_BRIDGING] = ztDbTrue(network['allowPassiveBridging']) ? '1' : '0';
|
||||
if ((activeBridges)&&(activeBridges.length > 0))
|
||||
netconf.data[ZT_NETWORKCONFIG_DICT_KEY_ACTIVE_BRIDGES] = activeBridges; // comma-delimited list
|
||||
|
||||
response.data['netconf'] = netconf.toString();
|
||||
}
|
||||
|
||||
process.stdout.write(response.toString()+'\n');
|
||||
|
||||
});
|
||||
|
||||
} else {
|
||||
|
||||
// Peer not authorized to join network or network not found (right now we always send ACCESS_DENIED)
|
||||
response.data['error'] = 'ACCESS_DENIED';
|
||||
process.stdout.write(response.toString()+'\n');
|
||||
|
||||
}
|
||||
|
||||
});
|
||||
}
|
||||
|
||||
function handleMessage(dictStr)
|
||||
{
|
||||
var message = new Dictionary(dictStr);
|
||||
if (!('type' in message.data)) {
|
||||
console.error('ignored message without request type field');
|
||||
return;
|
||||
} else if (message.data['type'] === 'netconf-init') {
|
||||
doNetconfInit(message);
|
||||
} else if (message.data['type'] === 'netconf-request') {
|
||||
doNetconfRequest(message);
|
||||
} else {
|
||||
console.error('ignored unrecognized message type: '+message.data['type']);
|
||||
}
|
||||
};
|
||||
|
||||
//
|
||||
// Read stream of double-CR-terminated dictionaries from stdin until close/EOF
|
||||
//
|
||||
|
||||
var stdinReadBuffer = '';
|
||||
|
||||
process.stdin.on('readable',function() {
|
||||
var chunk = process.stdin.read();
|
||||
if (chunk)
|
||||
stdinReadBuffer += chunk;
|
||||
for(;;) {
|
||||
var boundary = stdinReadBuffer.indexOf('\n\n');
|
||||
if (boundary >= 0) {
|
||||
handleMessage(stdinReadBuffer.substr(0,boundary + 1));
|
||||
stdinReadBuffer = stdinReadBuffer.substr(boundary + 2);
|
||||
} else break;
|
||||
}
|
||||
});
|
||||
process.stdin.on('end',function() {
|
||||
process.exit(0);
|
||||
});
|
||||
process.stdin.on('close',function() {
|
||||
process.exit(0);
|
||||
});
|
||||
process.stdin.on('error',function() {
|
||||
process.exit(0);
|
||||
});
|
||||
|
||||
// Tell ZeroTier One that the service is running, solicit netconf-init
|
||||
process.stdout.write('type=ready\n\n');
|
||||
|
@ -1,13 +0,0 @@
|
||||
#!/bin/bash
|
||||
|
||||
export PATH=/bin:/usr/bin:/usr/local/bin
|
||||
|
||||
# We will start in ZT_HOME
|
||||
|
||||
if [ ! -d ./services.d/netconf-service ]; then
|
||||
echo 'cannot find netconf-service subfolder to launch subprocess' >&2
|
||||
exit 1
|
||||
fi
|
||||
|
||||
cd services.d/netconf-service
|
||||
exec node netconf-master.js
|
@ -1,15 +0,0 @@
|
||||
{
|
||||
"name": "zt1-netconf-service",
|
||||
"version": "0.0.0",
|
||||
"description": "Worker in charge of issuing network configuration from ZeroTier One netconf masters",
|
||||
"main": "netconf-master.js",
|
||||
"scripts": {
|
||||
"test": "echo \"Error: no test specified\" && exit 1"
|
||||
},
|
||||
"author": "ZeroTier Networks LLC",
|
||||
"license": "GPL",
|
||||
"dependencies": {
|
||||
"redis": "~0.10.3",
|
||||
"async": "~0.9.0"
|
||||
}
|
||||
}
|
@ -1,111 +0,0 @@
|
||||
# ZeroTier One Redis Database Schema
|
||||
|
||||
## Notes
|
||||
|
||||
- : is used as the key namespace separator as per de-facto Redis standard.
|
||||
- A top-level record may have a :~ child containing a hash. This is the root hash and contains any simple key=value properties of the record.
|
||||
- Booleans: any value other than "1" or "true" is false.
|
||||
- Timestamps are in milliseconds since the epoch and are stored as base-10 integers.
|
||||
- IPv4 addresees: stored in standard dot notation e.g. 1.2.3.4
|
||||
- IPv6 addresses: :'s are optional and addresses must be stored *without* shortening, e.g. with :0000: instead of ::. It must be possible to strip :'s from the address and get 128 bits of straight hex.
|
||||
- Hexadecimal: all hex values must be lower case
|
||||
- 16-digit network IDs and 10-digit addresses are left zero-padded to 16 and 10 digits respectively, as they are everywhere else in the ZT1 universe.
|
||||
|
||||
Note: right now KEYS globbing is used in a few places in the code to search for stuff. In the future we'll want to add SET/ZSET index fields for these to optimize, but that won't become an issue until there are at least millions of records. (Millions of users will give us lots of "good problems to have." :)
|
||||
|
||||
## Field attribute flags used in this documentation (not in database)
|
||||
|
||||
- ! required
|
||||
- M mutable (user-editable via API)
|
||||
- R read-only (at least from API/user perspective)
|
||||
- H hidden (not returned by API queries)
|
||||
|
||||
## Base Configuration
|
||||
|
||||
### zt1 (value: 1)
|
||||
|
||||
If this key is not present with a value of 1, the API server code will auto-init the DB with initial data containing stuff like the Earth network and default users. This is not used by netconf-service.
|
||||
|
||||
## Users (ZeroTier Networks only)
|
||||
|
||||
This record type is only of interest to ZeroTier Networks itself. It holds user records, billing information, subscriptions, etc. Netconf masters do not use these records so you don't need to worry about this if you are trying to run your own.
|
||||
|
||||
### zt1:user:\<auth\>:\<authUserId\>:~
|
||||
|
||||
Note: users are referred to elsewhere in the database by their compound key \<auth\>:\<authUserId\> as stored here in the id field.
|
||||
|
||||
- !R id :: must be auth:authUserId
|
||||
- !R auth :: authentication type e.g. 'google' or 'local'
|
||||
- !R authUserId :: user ID under auth schema, like an e-mail address or a Google profile ID.
|
||||
- M email :: user's email address
|
||||
- R confirmed :: is e-mail confirmed?
|
||||
- R lastLogin :: timestamp of last login
|
||||
- R creationTime: :: timestamp of account creation
|
||||
- M displayName :: usually First Last, defaults to e-mail address for 'local' auth and whatever the OpenID API says for third party auth such as Google.
|
||||
- M defaultCard :: ID of default credit card (actual card objects are stored by Stripe, not in this database)
|
||||
- M ui :: arbitrary field that can be used by the UI to store stuff
|
||||
- R stripeCustomerId :: customer ID for Stripe credit card service if the user has cards on file (we don't store cards, we let Stripe do that)
|
||||
|
||||
## Networks
|
||||
|
||||
Network records are used by the netconf master to issue network configuration information to peers on the ZT1 network itself. Here is where you should look if you want to play with running your own!
|
||||
|
||||
### zt1:network:\<nwid\>:~
|
||||
|
||||
Each network has a network record indexed by its 64-bit network ID in lower-case hexadecimal. Unless otherwise indicated all integer values are in hexadecimal.
|
||||
|
||||
- !R id :: must be \<nwid\>
|
||||
- !M name :: network's globally unique short name, which can contain only characters valid in an e-mail address. It's the job of the code that populates this DB to ensure that this is globally unique.
|
||||
- R owner :: id of user who owns this network (not used by netconf master, only for web UI and web API)
|
||||
- R billingUser :: user paying for premium subscriptions (also unused by netconf-master)
|
||||
- R billingUserConfirmed :: if true, billingUser has confirmed and authorized billing
|
||||
- M desc :: a longer network description
|
||||
- R infrastructure :: if true, network can't be deleted through API or web UI
|
||||
- M private :: if true, network requires authentication
|
||||
- R creationTime :: timestamp of network creation
|
||||
- M etherTypes :: comma-delimited list of integers indicating Ethernet types permitted on network
|
||||
- M enableBroadcast :: if true, ff:ff:ff:ff:ff:ff is enabled network-wide
|
||||
- M v4AssignMode :: 'none' (or null/empty/etc.), 'zt', 'dhcp'
|
||||
- M v4AssignPool :: network/bits from which to assign IPs
|
||||
- M v6AssignMode :: 'none' (or null/empty/etc.), 'zt', 'v6native', 'dhcp6'
|
||||
- M v6AssignPool :: network/bits from which to assign IPs
|
||||
- M allowPassiveBridging :: if true, allow passive bridging
|
||||
- M multicastLimit :: maximum number of recipients to receive a multicast on this network
|
||||
- M multicastRates :: packed JSON containing multicast rates (see below)
|
||||
- M subscriptions :: comma-delimited list of subscriptions for this network
|
||||
- M ui :: arbitrary field that can be used by the UI to store stuff
|
||||
|
||||
Multicast rates are encoded as a JSON document. Each key is a multicast group in "MAC/ADI" format (e.g. *ff:ff:ff:ff:ff:ff/0*), and each value is a comma-delimited tuple of hex integer values: preload, max balance, and rate of accrual in bytes per second. An entry for *0* (or *0/0* or *00:00:00:00:00:00/0*) indicates the default setting for all unspecified multicast groups. Setting a rate limit like *ffffffff,ffffffff,ffffffff* as default will effectively turn off rate limits.
|
||||
|
||||
### zt1:network:\<nwid\>:member:\<address\>:~
|
||||
|
||||
For private networks, each member of the network must have a record that indicates whether it is allowed to communicate. The address is the 10-digit lower-case hexadecimal ZeroTier address.
|
||||
|
||||
The netconf-master will automatically add any peer that even attempts to request a netconf / certificate. These are added with authorized set to false. The hideInSlushpile field is used in the UI to allow network admins to hide unknown/bogus join attempts that they don't want to keep seeing.
|
||||
|
||||
- !R id :: must be \<address\>
|
||||
- !R nwid :: must be \<nwid\>
|
||||
- M authorized :: true if node is authorized and will be issued valid certificates and network configurations
|
||||
- M activeBridge :: true if node is an active bridge
|
||||
- M name :: name of system
|
||||
- M notes :: annotation field
|
||||
- R authorizedBy :: user ID of user who authorized membership
|
||||
- R authorizedAt :: timestamp of authorization
|
||||
- R identity :: string-serialized full public node identity as last seen by netconf-master
|
||||
- R firstSeen :: timestamp node first tried to authorize to this net
|
||||
- R lastSeen :: timestamp node last tried to authorize to this net
|
||||
- R lastAt :: real network address from which node was last seen
|
||||
- R clientVersion :: software version last seen by netconf-master
|
||||
- R clientOs :: operating system last seen by netconf-master
|
||||
- R ipAssignments :: comma-delimited list of IP address assignments (see below)
|
||||
- M ui :: string-serialized JSON blob for use by the user interface, ignored by netconf-master
|
||||
|
||||
The name, notes, authorizedBy, and authorizedAt fields are only for use by the administration UI. The netconf-master does not care about this. The identity, firstSeen, lastSeen, lastAt, version, and os fields are populated by netconf-master.
|
||||
|
||||
The ipAssignments field is re-generated whenever the zt1:network:\<nwid\>:ipAssignments hash is modified for this member. Both the API code and the netconf-master code must keep this in sync. This field is read-only for API users; the ipAssignments part of the API must be used to modify member IP address assignments.
|
||||
|
||||
### zt1:network:\<nwid\>:ipAssignments
|
||||
|
||||
This is a hash mapping IP/netmask bits fields to 10-digit ZeroTier addresses of network members. IPv4 fields contain dots, e.g. "10.2.3.4/24" or "29.1.1.1/7". IPv6 fields contain colons. Note that IPv6 IP abbreviations must *not* be used; use \:0000\: instead of \:\: for zero parts of addresses. This is to simplify parser code and canonicalize for rapid search. All hex digits must be lower-case.
|
||||
|
||||
This is only used if the network's IPv4 and/or IPv6 auto-assign mode is 'zt' for ZeroTier assignment. The netconf-master will auto-populate by choosing unused IPs, and it can be edited via the API as well.
|
Loading…
Reference in New Issue
Block a user