2020-05-25 13:50:55 +00:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
import argparse
|
2020-06-24 13:24:07 +00:00
|
|
|
import json
|
2020-05-25 13:50:55 +00:00
|
|
|
import sys
|
2020-06-24 13:24:07 +00:00
|
|
|
import os
|
2020-05-25 13:50:55 +00:00
|
|
|
|
|
|
|
from socket import AF_INET
|
|
|
|
|
2020-06-17 13:07:05 +00:00
|
|
|
import zmq
|
2020-05-25 13:50:55 +00:00
|
|
|
from pyroute2 import IPRoute
|
|
|
|
|
|
|
|
class LokinetRPC:
|
|
|
|
|
|
|
|
def __init__(self, url):
|
2020-06-17 13:07:05 +00:00
|
|
|
self._rpc_context = zmq.Context()
|
|
|
|
self._rpc_socket = self._rpc_context.socket(zmq.DEALER)
|
|
|
|
self._rpc_socket.setsockopt(zmq.CONNECT_TIMEOUT, 5000)
|
|
|
|
self._rpc_socket.setsockopt(zmq.HANDSHAKE_IVL, 5000)
|
|
|
|
self._rpc_socket.connect(url)
|
2020-05-25 13:50:55 +00:00
|
|
|
|
2020-06-24 13:24:07 +00:00
|
|
|
def rpc(self, method, data=None):
|
|
|
|
tag = os.urandom(16)
|
|
|
|
req = [method.encode(), tag]
|
|
|
|
if data is not None:
|
|
|
|
req.append(json.dumps(data).encode('utf-8'))
|
|
|
|
self._rpc_socket.send_multipart(req)
|
2020-06-17 13:07:05 +00:00
|
|
|
if not self._rpc_socket.poll(timeout=50):
|
|
|
|
return
|
|
|
|
reply = self._rpc_socket.recv_multipart()
|
2020-06-24 13:24:07 +00:00
|
|
|
if len(reply) >= 3 and reply[0:2] == [b'REPLY', tag]:
|
|
|
|
try:
|
|
|
|
return json.loads(reply[2].decode())
|
|
|
|
except:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def map_exit(self, addr, maprange, token=None):
|
|
|
|
"""
|
|
|
|
map an exit range to a loki address and use token for auth if desires
|
|
|
|
"""
|
|
|
|
if token is None:
|
|
|
|
data = self.rpc("llarp.exit", {'exit': addr, 'range': maprange})
|
|
|
|
else:
|
|
|
|
data = self.rpc("llarp.exit", {'exit': addr, 'token': token, 'range': maprange})
|
|
|
|
if data is None:
|
|
|
|
print("no response from rpc")
|
|
|
|
return False
|
|
|
|
if data['error'] is not None:
|
|
|
|
print('failed to map exit range: {}'.format(data['error']))
|
|
|
|
return False
|
|
|
|
print('mapped {} to {}: {}'.format(maprange, addr, data['result']))
|
|
|
|
return True
|
|
|
|
|
|
|
|
def unmap_exit(self, maprange):
|
|
|
|
"""
|
|
|
|
unmap an ip range from exit
|
|
|
|
"""
|
|
|
|
data = self.rpc("llarp.exit", {'range': maprange, 'unmap': True})
|
|
|
|
if data is None:
|
|
|
|
print("no response from rpc")
|
|
|
|
return False
|
|
|
|
if data['error'] is not None:
|
|
|
|
print('failed to unmap exit range: {}'.format(data['error']))
|
|
|
|
return False
|
|
|
|
print('unmapped {}: {}'.format(maprange, data['result']))
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_endpoint_ifname(self, endpoint):
|
|
|
|
"""
|
|
|
|
get the network interface name belongint to the endpoint called endpoint
|
|
|
|
"""
|
|
|
|
data = self.rpc("llarp.status")
|
|
|
|
if data is None or data['error'] is not None:
|
|
|
|
return
|
|
|
|
services = data['result']['services']
|
|
|
|
if endpoint not in services:
|
|
|
|
return
|
|
|
|
return services[endpoint]['ifname']
|
|
|
|
|
2020-05-25 13:50:55 +00:00
|
|
|
def get_first_hops(self):
|
2020-06-17 13:07:05 +00:00
|
|
|
data = self.rpc("llarp.status")
|
2020-06-24 13:24:07 +00:00
|
|
|
if data is None:
|
|
|
|
print("no response from rpc")
|
|
|
|
elif data['error'] is not None:
|
|
|
|
print('error detecting first hops: {}'.format(data['error']))
|
|
|
|
else:
|
|
|
|
for link in data['result']['links']['outbound']:
|
|
|
|
for session in link["sessions"]['established']:
|
|
|
|
yield session['remoteAddr']
|
2020-05-25 13:50:55 +00:00
|
|
|
|
2020-06-17 13:07:05 +00:00
|
|
|
def close(self):
|
|
|
|
self._rpc_socket.close(linger=0)
|
|
|
|
|
2020-05-25 13:50:55 +00:00
|
|
|
|
|
|
|
def main():
|
|
|
|
ap = argparse.ArgumentParser()
|
2020-06-24 13:24:07 +00:00
|
|
|
ap.add_argument("--rpc", type=str, default='tcp://127.0.0.1:1190', help='rpc endpoint url for local lokinet')
|
|
|
|
ap.add_argument("--exit", type=str, required=True, help='loki address of exit to use')
|
|
|
|
ap.add_argument('--token', type=str, default=None, help='optional auth token to use when talking to exit')
|
|
|
|
ap.add_argument('--range', type=str, default='0.0.0.0/0', help='optional ip range to map to exit')
|
|
|
|
ap.add_argument("--endpoint", type=str, default='default', help='lokinet endpoint name to use')
|
2020-05-25 13:50:55 +00:00
|
|
|
ap.add_argument("--up", action='store_const', dest='action', const='up')
|
|
|
|
ap.add_argument("--down", action='store_const', dest='action', const='down')
|
|
|
|
args = ap.parse_args()
|
2020-06-17 13:07:05 +00:00
|
|
|
rpc = LokinetRPC(args.rpc)
|
2020-06-24 13:24:07 +00:00
|
|
|
|
2020-05-25 13:50:55 +00:00
|
|
|
hops = dict()
|
|
|
|
for hop in rpc.get_first_hops():
|
|
|
|
ip = hop.split(':')[0]
|
|
|
|
hops[ip] = 0
|
2020-05-29 16:31:57 +00:00
|
|
|
if len(hops) == 0:
|
|
|
|
print("lokinet is not connected yet")
|
2020-06-24 13:24:07 +00:00
|
|
|
rpc.close()
|
|
|
|
return 1
|
|
|
|
ifname = rpc.get_endpoint_ifname(args.endpoint)
|
|
|
|
if ifname is None:
|
|
|
|
print("cannot determine lokinet network interface name")
|
|
|
|
rpc.close()
|
2020-05-29 16:31:57 +00:00
|
|
|
return 1
|
2020-05-25 13:50:55 +00:00
|
|
|
with IPRoute() as ip:
|
|
|
|
ip.bind()
|
2020-05-29 16:31:57 +00:00
|
|
|
try:
|
2020-06-24 13:24:07 +00:00
|
|
|
idx = ip.link_lookup(ifname=ifname)[0]
|
|
|
|
except Exception as ex:
|
|
|
|
print("cannot find {}: {}".format(ifname, ex))
|
|
|
|
rpc.close()
|
2020-05-29 16:31:57 +00:00
|
|
|
return 1
|
2020-06-24 13:24:07 +00:00
|
|
|
# set up exit after binding routing socket
|
|
|
|
if args.action == 'up':
|
|
|
|
if not rpc.map_exit(args.exit, args.range, args.token):
|
|
|
|
rpc.close()
|
|
|
|
print('cannot map exit')
|
|
|
|
return
|
|
|
|
elif args.action == 'down':
|
|
|
|
if not rpc.unmap_exit(args.range):
|
|
|
|
rpc.close()
|
|
|
|
print('cannot unmap exit')
|
|
|
|
return
|
|
|
|
rpc.close()
|
2020-05-25 13:50:55 +00:00
|
|
|
gateways = ip.get_default_routes(family=AF_INET)
|
|
|
|
gateway = None
|
|
|
|
for g in gateways:
|
|
|
|
useThisGateway = True
|
|
|
|
for name, val in g['attrs']:
|
|
|
|
if name == 'RTA_OIF' and val == idx:
|
|
|
|
useThisGateway = False
|
|
|
|
if not useThisGateway:
|
|
|
|
continue
|
|
|
|
for name, val in g['attrs']:
|
|
|
|
if name == 'RTA_GATEWAY':
|
|
|
|
gateway = val
|
|
|
|
if gateway:
|
|
|
|
for address in hops:
|
|
|
|
try:
|
|
|
|
if args.action == 'up':
|
|
|
|
ip.route("add", dst="{}/32".format(address), gateway=gateway)
|
|
|
|
elif args.action == 'down':
|
|
|
|
ip.route("del", dst="{}/32".format(address), gateway=gateway)
|
|
|
|
except:
|
|
|
|
pass
|
2020-05-29 16:31:57 +00:00
|
|
|
|
2020-05-25 13:50:55 +00:00
|
|
|
if args.action == 'up':
|
2020-05-29 16:31:57 +00:00
|
|
|
try:
|
|
|
|
ip.route('add', dst='0.0.0.0/0', oif=idx)
|
|
|
|
except:
|
|
|
|
print('failed to add default route')
|
|
|
|
return 1
|
2020-05-25 13:50:55 +00:00
|
|
|
elif args.action == 'down':
|
2020-05-29 16:31:57 +00:00
|
|
|
try:
|
|
|
|
ip.route('del', dst='0.0.0.0/0', oif=idx)
|
|
|
|
except:
|
|
|
|
print('failed to remove default route')
|
|
|
|
return 1
|
|
|
|
else:
|
|
|
|
print("could not find gateway")
|
|
|
|
return 1
|
|
|
|
return 0
|
2020-05-25 13:50:55 +00:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2020-05-29 16:31:57 +00:00
|
|
|
sys.exit(main())
|