2020-02-17 19:09:49 +00:00
|
|
|
#!/usr/bin/python3
|
|
|
|
|
2020-03-22 15:40:18 +00:00
|
|
|
# SPDX-License-Identifier: LGPL-3.0-or-later
|
2020-02-17 19:09:49 +00:00
|
|
|
# Copyright (c) 2020 Daniel Thompson
|
|
|
|
|
|
|
|
import argparse
|
2020-04-20 19:09:33 +01:00
|
|
|
import io
|
2020-02-17 19:09:49 +00:00
|
|
|
import random
|
|
|
|
import os.path
|
|
|
|
import pexpect
|
|
|
|
import time
|
|
|
|
import string
|
|
|
|
import sys
|
|
|
|
|
2020-04-06 21:40:34 +01:00
|
|
|
def pbar(iterable, quiet=False):
|
|
|
|
step = 100 / len(iterable)
|
|
|
|
for i, v in enumerate(iterable):
|
|
|
|
if not quiet:
|
|
|
|
percent = round(step * i, 1)
|
|
|
|
bar = int(percent) // 2
|
2020-04-10 20:31:26 +01:00
|
|
|
print(f'[{"#"*bar}{"."*(50-bar)}] {percent}%', end='\r', flush=True)
|
2020-04-06 21:40:34 +01:00
|
|
|
yield v
|
|
|
|
if not quiet:
|
2020-04-10 20:31:26 +01:00
|
|
|
print(f'[{"#"*50}] 100% ')
|
2020-04-06 21:40:34 +01:00
|
|
|
|
2020-02-17 19:09:49 +00:00
|
|
|
def sync(c):
|
2020-03-22 12:37:19 +00:00
|
|
|
"""Stop the watch and synchronize with the command prompt.
|
|
|
|
|
|
|
|
Sending a random print ensure the final export (of the prompt)
|
|
|
|
does not accidentally match a previously issued prompt.
|
|
|
|
"""
|
2020-02-17 19:09:49 +00:00
|
|
|
tag = ''.join([random.choice(string.ascii_uppercase) for i in range(6)])
|
|
|
|
|
|
|
|
c.send('\x03')
|
|
|
|
c.expect('>>> ')
|
|
|
|
c.sendline(f'print("{tag[:3]}""{tag[3:]}")')
|
|
|
|
c.expect(tag)
|
|
|
|
c.expect('>>> ')
|
|
|
|
|
|
|
|
def unsync(c):
|
2020-03-22 12:37:19 +00:00
|
|
|
"""Set the watch running again.
|
|
|
|
|
|
|
|
There must be an expect (or a sleep) since if we kill the subordinate
|
|
|
|
process too early then the sendline will not have completed.
|
|
|
|
"""
|
|
|
|
c.sendline('wasp.system.run()')
|
|
|
|
c.expect('Watch is running, use Ctrl-C to stop')
|
|
|
|
c.send('\x18')
|
2020-02-17 19:09:49 +00:00
|
|
|
|
2020-04-17 17:18:27 +01:00
|
|
|
def paste(c, f, verbose=False, chunk=None):
|
2020-03-09 21:34:01 +00:00
|
|
|
docstring = False
|
2020-04-06 21:40:34 +01:00
|
|
|
|
|
|
|
tosend = []
|
|
|
|
|
2020-02-17 19:09:49 +00:00
|
|
|
for ln in f.readlines():
|
|
|
|
ln = ln.rstrip()
|
2020-03-09 21:34:01 +00:00
|
|
|
|
|
|
|
# This is a bit loose (definitely not PEP-257 compliant) but
|
|
|
|
# is enough for most code.
|
|
|
|
if ln.lstrip().startswith('"""'):
|
|
|
|
docstring = True
|
|
|
|
if docstring:
|
|
|
|
if ln.rstrip().endswith('"""'):
|
|
|
|
docstring = False
|
|
|
|
continue
|
|
|
|
|
|
|
|
if ln.lstrip().startswith('#'):
|
|
|
|
continue
|
|
|
|
|
2020-04-17 17:18:27 +01:00
|
|
|
if ln.strip() == '':
|
|
|
|
continue
|
|
|
|
|
2020-04-06 21:40:34 +01:00
|
|
|
tosend.append(ln)
|
2020-03-09 21:34:01 +00:00
|
|
|
|
2020-04-06 21:40:34 +01:00
|
|
|
for ln in pbar(tosend, verbose):
|
2020-04-17 17:18:27 +01:00
|
|
|
if chunk and ln.startswith('class'):
|
|
|
|
chunk()
|
|
|
|
|
|
|
|
c.sendline(ln)
|
|
|
|
c.expect('=== ')
|
2020-02-17 19:09:49 +00:00
|
|
|
|
2020-04-20 19:09:33 +01:00
|
|
|
def print_log(logfile):
|
|
|
|
lines = logfile.getvalue().split('\n')
|
|
|
|
lines = [ l.strip('\x04\r') for l in lines ]
|
|
|
|
|
|
|
|
output = [ l for l in lines if l and l != '>>> ' ]
|
|
|
|
if output:
|
|
|
|
print('~~~')
|
|
|
|
print('\n'.join(output))
|
|
|
|
print('~~~')
|
|
|
|
|
2020-02-17 19:09:49 +00:00
|
|
|
def handle_eval(c, cmd):
|
|
|
|
verbose = bool(c.logfile)
|
|
|
|
|
|
|
|
c.send('\x05')
|
|
|
|
c.expect('=== ')
|
|
|
|
c.sendline(cmd)
|
|
|
|
c.expect('=== ')
|
|
|
|
|
2020-04-20 19:09:33 +01:00
|
|
|
if not verbose:
|
|
|
|
c.logfile = io.StringIO()
|
2020-02-17 19:09:49 +00:00
|
|
|
c.send('\x04')
|
|
|
|
c.expect('>>> ')
|
|
|
|
if not verbose:
|
2020-04-20 19:09:33 +01:00
|
|
|
print_log(c.logfile)
|
|
|
|
c.logfile.close()
|
2020-02-17 19:09:49 +00:00
|
|
|
c.logfile = None
|
|
|
|
|
|
|
|
def handle_exec(c, fname):
|
|
|
|
verbose = bool(c.logfile)
|
|
|
|
|
2020-04-20 19:09:33 +01:00
|
|
|
log = io.StringIO()
|
|
|
|
|
2020-04-17 17:18:27 +01:00
|
|
|
def chunk():
|
2020-04-20 19:09:33 +01:00
|
|
|
if not verbose:
|
|
|
|
c.logfile = log
|
2020-04-17 17:18:27 +01:00
|
|
|
c.send('\x04')
|
|
|
|
c.expect('>>> ')
|
|
|
|
if not verbose:
|
|
|
|
c.logfile = None
|
|
|
|
c.send('\x05')
|
|
|
|
c.expect('=== ')
|
|
|
|
|
2020-02-17 19:09:49 +00:00
|
|
|
with open(fname) as f:
|
|
|
|
if not verbose:
|
2020-04-06 21:40:34 +01:00
|
|
|
print(f'Preparing to run {fname}:')
|
2020-02-17 19:09:49 +00:00
|
|
|
|
|
|
|
c.send('\x05')
|
|
|
|
c.expect('=== ')
|
|
|
|
|
2020-04-17 17:18:27 +01:00
|
|
|
paste(c, f, verbose, chunk)
|
2020-02-17 19:09:49 +00:00
|
|
|
|
2020-04-20 19:09:33 +01:00
|
|
|
if not verbose:
|
|
|
|
c.logfile = log
|
2020-02-17 19:09:49 +00:00
|
|
|
c.send('\x04')
|
|
|
|
c.expect('>>> ')
|
|
|
|
if not verbose:
|
|
|
|
c.logfile = None
|
|
|
|
|
2020-04-20 19:09:33 +01:00
|
|
|
print_log(log)
|
|
|
|
log.close()
|
|
|
|
|
2020-04-17 17:17:24 +01:00
|
|
|
def handle_reset(c):
|
|
|
|
c.send('\x05')
|
|
|
|
c.expect('=== ')
|
|
|
|
c.sendline('import machine')
|
|
|
|
c.sendline('machine.reset()')
|
|
|
|
c.expect('=== ')
|
|
|
|
c.send('\x04')
|
|
|
|
|
2020-02-17 19:09:49 +00:00
|
|
|
def handle_rtc(c):
|
|
|
|
# Wait for the clock to tick over to the next second
|
|
|
|
now = then = time.localtime()
|
|
|
|
while now[5] == then[5]:
|
|
|
|
now = time.localtime()
|
|
|
|
|
|
|
|
# Set the time
|
2020-02-19 19:32:06 +00:00
|
|
|
c.sendline(f'watch.rtc.set_localtime(({now[0]}, {now[1]}, {now[2]}, {now[3]}, {now[4]}, {now[5]}, {now[6]}, {now[7]}))')
|
2020-02-17 19:09:49 +00:00
|
|
|
c.expect('>>> ')
|
|
|
|
|
2020-05-09 14:21:39 +01:00
|
|
|
def check_rtc(c):
|
|
|
|
c.sendline('print(watch.rtc.get_localtime())')
|
|
|
|
c.expect('\(([0-9]+), ([0-9]+), ([0-9]+), ([0-9]+), ([0-9]+), ([0-9]+), ([0-9]+), ([0-9]+)\)')
|
|
|
|
t = time.localtime()
|
|
|
|
|
|
|
|
watch_hms = (int(c.match[4]), int(c.match[5]), int(c.match[6]))
|
|
|
|
watch_str = f'{watch_hms[0]:02d}:{watch_hms[1]:02d}:{watch_hms[2]:02d}'
|
|
|
|
host_hms = (t.tm_hour, t.tm_min, t.tm_sec)
|
|
|
|
host_str = f'{host_hms[0]:02d}:{host_hms[1]:02d}:{host_hms[2]:02d}'
|
|
|
|
delta = 3600 * (host_hms[0] - watch_hms[0]) + \
|
|
|
|
60 * (host_hms[1] - watch_hms[1]) + \
|
|
|
|
(host_hms[2] - watch_hms[2])
|
|
|
|
print(f"PC <-> watch: {watch_str} <-> {host_str} (delta {delta})")
|
|
|
|
|
|
|
|
c.expect('>>> ')
|
|
|
|
|
2020-02-17 19:09:49 +00:00
|
|
|
def handle_upload(c, fname):
|
|
|
|
verbose = bool(c.logfile)
|
|
|
|
|
2020-02-19 19:33:35 +00:00
|
|
|
c.sendline('from shell import upload')
|
|
|
|
c.expect('>>> ')
|
|
|
|
|
2020-02-17 19:09:49 +00:00
|
|
|
with open(fname) as f:
|
|
|
|
if not verbose:
|
2020-04-06 21:40:34 +01:00
|
|
|
print(f'Uploading {fname}:')
|
2020-02-17 19:09:49 +00:00
|
|
|
|
2020-02-19 19:33:35 +00:00
|
|
|
c.sendline(f'upload("{os.path.basename(fname)}")')
|
2020-02-17 19:09:49 +00:00
|
|
|
c.expect('=== ')
|
|
|
|
paste(c, f, verbose)
|
|
|
|
c.send('\x04')
|
|
|
|
|
|
|
|
c.expect('>>> ')
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
description='WASP command and control client')
|
|
|
|
parser.add_argument('--console', action='store_true',
|
|
|
|
help='Launch a REPL session')
|
2020-05-09 14:21:39 +01:00
|
|
|
parser.add_argument('--check-rtc', action='store_true',
|
|
|
|
help='Compare workstation and watch times')
|
2020-02-17 19:09:49 +00:00
|
|
|
parser.add_argument('--device',
|
|
|
|
help='Connect only to a specific named device')
|
|
|
|
parser.add_argument('--exec',
|
|
|
|
help='Execute the contents of a file')
|
|
|
|
parser.add_argument('--eval',
|
|
|
|
help='Execute the provided python string')
|
2020-04-17 17:17:24 +01:00
|
|
|
parser.add_argument('--reset', action='store_true',
|
|
|
|
help="Reboot the device (and don't stay in bootloader mode)")
|
2020-02-17 19:09:49 +00:00
|
|
|
parser.add_argument('--rtc', action='store_true',
|
|
|
|
help='Set the time on the WASP device')
|
|
|
|
parser.add_argument('--upload',
|
|
|
|
help='Copy the specified file to the WASP device')
|
|
|
|
parser.add_argument('--verbose', action='store_true',
|
|
|
|
help='Log interaction with the WASP device')
|
|
|
|
|
|
|
|
args = parser.parse_args()
|
|
|
|
device_name = args.device
|
|
|
|
|
|
|
|
pynus = os.path.dirname(sys.argv[0]) + '/pynus/pynus.py'
|
|
|
|
console = pexpect.spawn(pynus, encoding='UTF-8')
|
|
|
|
if args.verbose:
|
|
|
|
console.logfile = sys.stdout
|
|
|
|
|
|
|
|
console.expect('Connect')
|
|
|
|
console.expect('Exit console using Ctrl-X')
|
|
|
|
time.sleep(0.5)
|
|
|
|
sync(console)
|
|
|
|
|
2020-03-08 10:16:49 +00:00
|
|
|
if args.rtc:
|
|
|
|
handle_rtc(console)
|
2020-02-17 19:09:49 +00:00
|
|
|
|
2020-05-09 14:21:39 +01:00
|
|
|
if args.check_rtc:
|
|
|
|
check_rtc(console)
|
|
|
|
|
2020-02-17 19:09:49 +00:00
|
|
|
if args.exec:
|
|
|
|
handle_exec(console, args.exec)
|
|
|
|
|
2020-03-08 10:16:49 +00:00
|
|
|
if args.eval:
|
|
|
|
handle_eval(console, args.eval)
|
|
|
|
|
2020-02-17 19:09:49 +00:00
|
|
|
if args.upload:
|
|
|
|
handle_upload(console, args.upload)
|
|
|
|
|
2020-02-19 19:35:49 +00:00
|
|
|
if args.console:
|
|
|
|
console.close()
|
|
|
|
os.execl(pynus, pynus)
|
|
|
|
|
2020-04-17 17:17:24 +01:00
|
|
|
if args.reset:
|
|
|
|
handle_reset(console)
|
|
|
|
sys.exit(0)
|
|
|
|
|
2020-02-17 19:09:49 +00:00
|
|
|
unsync(console)
|