2019-03-24 13:09:10 +00:00
|
|
|
from __future__ import print_function
|
|
|
|
|
2016-10-07 18:15:05 +00:00
|
|
|
import json
|
|
|
|
|
2020-11-29 20:48:26 +00:00
|
|
|
try:
|
|
|
|
from allmydata.scripts.types_ import SubCommands
|
|
|
|
except ImportError:
|
|
|
|
pass
|
|
|
|
|
2016-10-07 18:15:05 +00:00
|
|
|
from twisted.python import usage
|
|
|
|
from twisted.internet import defer, reactor
|
|
|
|
|
|
|
|
from wormhole import wormhole
|
|
|
|
|
|
|
|
from allmydata.util.encodingutil import argv_to_abspath
|
|
|
|
from allmydata.scripts.common import get_default_nodedir, get_introducer_furl
|
2020-11-16 20:02:51 +00:00
|
|
|
from allmydata.node import read_config
|
2016-10-07 18:15:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
class InviteOptions(usage.Options):
|
|
|
|
synopsis = "[options] <nickname>"
|
|
|
|
description = "Create a client-only Tahoe-LAFS node (no storage server)."
|
|
|
|
|
|
|
|
optParameters = [
|
|
|
|
("shares-needed", None, None, "How many shares are needed to reconstruct files from this node"),
|
|
|
|
("shares-happy", None, None, "Distinct storage servers new node will upload shares to"),
|
|
|
|
("shares-total", None, None, "Total number of shares new node will upload"),
|
|
|
|
]
|
|
|
|
|
|
|
|
def parseArgs(self, *args):
|
|
|
|
if len(args) != 1:
|
|
|
|
raise usage.UsageError(
|
|
|
|
"Provide a single argument: the new node's nickname"
|
|
|
|
)
|
|
|
|
self['nick'] = args[0].strip()
|
|
|
|
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def _send_config_via_wormhole(options, config):
|
|
|
|
out = options.stdout
|
|
|
|
err = options.stderr
|
|
|
|
relay_url = options.parent['wormhole-server']
|
2019-03-24 13:09:10 +00:00
|
|
|
print("Connecting to '{}'...".format(relay_url), file=out)
|
2016-10-07 18:15:05 +00:00
|
|
|
wh = wormhole.create(
|
|
|
|
appid=options.parent['wormhole-invite-appid'],
|
|
|
|
relay_url=relay_url,
|
|
|
|
reactor=reactor,
|
|
|
|
)
|
|
|
|
yield wh.get_welcome()
|
2019-03-24 13:09:10 +00:00
|
|
|
print("Connected to wormhole server", file=out)
|
2016-10-07 18:15:05 +00:00
|
|
|
|
|
|
|
# must call allocate_code before get_code will ever succeed
|
|
|
|
wh.allocate_code()
|
|
|
|
code = yield wh.get_code()
|
2019-03-24 13:09:10 +00:00
|
|
|
print("Invite Code for client: {}".format(code), file=out)
|
2016-10-07 18:15:05 +00:00
|
|
|
|
|
|
|
wh.send_message(json.dumps({
|
|
|
|
u"abilities": {
|
|
|
|
u"server-v1": {},
|
|
|
|
}
|
|
|
|
}))
|
|
|
|
|
|
|
|
client_intro = yield wh.get_message()
|
2019-03-24 13:09:10 +00:00
|
|
|
print(" received client introduction", file=out)
|
2016-10-07 18:15:05 +00:00
|
|
|
client_intro = json.loads(client_intro)
|
|
|
|
if not u'abilities' in client_intro:
|
2019-03-24 13:09:10 +00:00
|
|
|
print("No 'abilities' from client", file=err)
|
2016-10-07 18:15:05 +00:00
|
|
|
defer.returnValue(1)
|
|
|
|
if not u'client-v1' in client_intro[u'abilities']:
|
2019-03-24 13:09:10 +00:00
|
|
|
print("No 'client-v1' in abilities from client", file=err)
|
2016-10-07 18:15:05 +00:00
|
|
|
defer.returnValue(1)
|
|
|
|
|
2019-03-24 13:09:10 +00:00
|
|
|
print(" transmitting configuration", file=out)
|
2016-10-07 18:15:05 +00:00
|
|
|
wh.send_message(json.dumps(config))
|
|
|
|
yield wh.close()
|
|
|
|
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def invite(options):
|
|
|
|
if options.parent['node-directory']:
|
|
|
|
basedir = argv_to_abspath(options.parent['node-directory'])
|
|
|
|
else:
|
|
|
|
basedir = get_default_nodedir()
|
2020-11-16 20:02:51 +00:00
|
|
|
config = read_config(basedir, u"")
|
2016-10-07 18:15:05 +00:00
|
|
|
out = options.stdout
|
|
|
|
err = options.stderr
|
|
|
|
|
|
|
|
try:
|
|
|
|
introducer_furl = get_introducer_furl(basedir, config)
|
|
|
|
except Exception as e:
|
2019-03-24 13:09:10 +00:00
|
|
|
print("Can't find introducer FURL for node '{}': {}".format(basedir, str(e)), file=err)
|
2016-10-07 18:15:05 +00:00
|
|
|
raise SystemExit(1)
|
|
|
|
|
|
|
|
nick = options['nick']
|
|
|
|
|
|
|
|
remote_config = {
|
|
|
|
"shares-needed": options["shares-needed"] or config.get('client', 'shares.needed'),
|
|
|
|
"shares-total": options["shares-total"] or config.get('client', 'shares.total'),
|
|
|
|
"shares-happy": options["shares-happy"] or config.get('client', 'shares.happy'),
|
|
|
|
"nickname": nick,
|
|
|
|
"introducer": introducer_furl,
|
|
|
|
}
|
|
|
|
|
|
|
|
yield _send_config_via_wormhole(options, remote_config)
|
2019-03-24 13:09:10 +00:00
|
|
|
print("Completed successfully", file=out)
|
2016-10-07 18:15:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
subCommands = [
|
|
|
|
("invite", None, InviteOptions,
|
|
|
|
"Invite a new node to this grid"),
|
2020-11-29 20:48:26 +00:00
|
|
|
] # type: SubCommands
|
2016-10-07 18:15:05 +00:00
|
|
|
|
|
|
|
dispatch = {
|
|
|
|
"invite": invite,
|
|
|
|
}
|