2014-10-05 16:25:31 +08:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
import argparse
|
2014-12-10 19:11:13 +08:00
|
|
|
import time
|
2014-12-28 18:56:26 +08:00
|
|
|
import asyncio
|
2014-10-23 18:48:03 +08:00
|
|
|
|
2014-12-10 13:04:18 +08:00
|
|
|
from prettytable import PrettyTable
|
|
|
|
|
2014-10-23 18:48:03 +08:00
|
|
|
from artiq.management.pc_rpc import Client
|
2014-12-28 18:56:26 +08:00
|
|
|
from artiq.management.sync_struct import Subscriber
|
|
|
|
from artiq.management.tools import clear_screen
|
2014-10-05 16:25:31 +08:00
|
|
|
|
|
|
|
|
|
|
|
def _get_args():
|
2014-12-29 12:48:14 +08:00
|
|
|
parser = argparse.ArgumentParser(description="ARTIQ CLI client")
|
2014-10-23 19:07:36 +08:00
|
|
|
parser.add_argument(
|
|
|
|
"-s", "--server", default="::1",
|
|
|
|
help="hostname or IP of the master to connect to")
|
|
|
|
parser.add_argument(
|
2014-12-28 18:56:26 +08:00
|
|
|
"--port", default=None, type=int,
|
2014-10-23 19:07:36 +08:00
|
|
|
help="TCP port to use to connect to the master")
|
2014-12-08 19:22:02 +08:00
|
|
|
|
|
|
|
subparsers = parser.add_subparsers(dest="action")
|
2014-12-10 13:04:18 +08:00
|
|
|
subparsers.required = True
|
2014-12-08 19:22:02 +08:00
|
|
|
|
2014-12-10 13:04:18 +08:00
|
|
|
parser_add = subparsers.add_parser("submit", help="submit an experiment")
|
2014-12-08 19:22:02 +08:00
|
|
|
parser_add.add_argument(
|
|
|
|
"-p", "--periodic", default=None, type=float,
|
|
|
|
help="run the experiment periodically every given number of seconds")
|
|
|
|
parser_add.add_argument(
|
|
|
|
"-t", "--timeout", default=None, type=float,
|
|
|
|
help="specify a timeout for the experiment to complete")
|
|
|
|
parser_add.add_argument("-f", "--function", default="run",
|
|
|
|
help="function to run")
|
|
|
|
parser_add.add_argument("-u", "--unit", default=None,
|
|
|
|
help="unit to run")
|
|
|
|
parser_add.add_argument("file", help="file containing the unit to run")
|
|
|
|
|
2014-12-10 13:04:18 +08:00
|
|
|
parser_cancel = subparsers.add_parser("cancel",
|
|
|
|
help="cancel an experiment")
|
|
|
|
parser_cancel.add_argument("-p", "--periodic", default=False,
|
|
|
|
action="store_true",
|
|
|
|
help="cancel a periodic experiment")
|
|
|
|
parser_cancel.add_argument("rid", type=int,
|
|
|
|
help="run identifier (RID/PRID)")
|
|
|
|
|
2014-12-29 18:44:50 +08:00
|
|
|
parser_show_queue = subparsers.add_parser(
|
|
|
|
"show-queue", help="show the experiment queue")
|
|
|
|
|
|
|
|
parser_show_periodic = subparsers.add_parser(
|
|
|
|
"show-periodic", help="show the periodic experiment table")
|
2014-12-10 13:04:18 +08:00
|
|
|
|
2014-10-05 16:25:31 +08:00
|
|
|
return parser.parse_args()
|
|
|
|
|
|
|
|
|
2014-12-10 13:04:18 +08:00
|
|
|
def _action_submit(remote, args):
|
|
|
|
run_params = {
|
|
|
|
"file": args.file,
|
|
|
|
"unit": args.unit,
|
|
|
|
"function": args.function
|
|
|
|
}
|
|
|
|
if args.periodic is None:
|
|
|
|
rid = remote.run_once(run_params, args.timeout)
|
|
|
|
print("RID: {}".format(rid))
|
|
|
|
else:
|
|
|
|
prid = remote.run_periodic(run_params, args.timeout,
|
|
|
|
args.periodic)
|
|
|
|
print("PRID: {}".format(prid))
|
|
|
|
|
|
|
|
|
|
|
|
def _action_cancel(remote, args):
|
|
|
|
if args.periodic:
|
|
|
|
remote.cancel_periodic(args.rid)
|
|
|
|
else:
|
|
|
|
remote.cancel_once(args.rid)
|
|
|
|
|
|
|
|
|
2014-12-28 18:56:26 +08:00
|
|
|
def _show_queue(queue):
|
|
|
|
clear_screen()
|
|
|
|
if queue:
|
2014-12-10 13:04:18 +08:00
|
|
|
table = PrettyTable(["RID", "File", "Unit", "Function", "Timeout"])
|
|
|
|
for rid, run_params, timeout in queue:
|
|
|
|
row = [rid, run_params["file"]]
|
|
|
|
for x in run_params["unit"], run_params["function"], timeout:
|
|
|
|
row.append("-" if x is None else x)
|
|
|
|
table.add_row(row)
|
|
|
|
print(table)
|
2014-12-10 19:11:13 +08:00
|
|
|
else:
|
|
|
|
print("Queue is empty")
|
2014-12-28 18:56:26 +08:00
|
|
|
|
|
|
|
|
|
|
|
def _show_periodic(periodic):
|
|
|
|
clear_screen()
|
2014-12-10 19:11:13 +08:00
|
|
|
if periodic:
|
|
|
|
table = PrettyTable(["Next run", "PRID", "File", "Unit", "Function",
|
|
|
|
"Timeout", "Period"])
|
2014-12-29 18:44:50 +08:00
|
|
|
sp = sorted(periodic.items(), key=lambda x: (x[1][0], x[0]))
|
2014-12-10 19:11:13 +08:00
|
|
|
for prid, (next_run, run_params, timeout, period) in sp:
|
|
|
|
row = [time.strftime("%m/%d %H:%M:%S", time.localtime(next_run)),
|
|
|
|
prid, run_params["file"]]
|
|
|
|
for x in run_params["unit"], run_params["function"], timeout:
|
|
|
|
row.append("-" if x is None else x)
|
|
|
|
row.append(period)
|
|
|
|
table.add_row(row)
|
|
|
|
print(table)
|
|
|
|
else:
|
|
|
|
print("No periodic schedule")
|
2014-12-10 13:04:18 +08:00
|
|
|
|
|
|
|
|
2014-12-28 18:56:26 +08:00
|
|
|
def _run_subscriber(host, port, subscriber):
|
2014-12-29 18:44:50 +08:00
|
|
|
if port is None:
|
|
|
|
port = 8887
|
2014-12-28 18:56:26 +08:00
|
|
|
loop = asyncio.get_event_loop()
|
2014-10-23 18:48:03 +08:00
|
|
|
try:
|
2014-12-28 18:56:26 +08:00
|
|
|
loop.run_until_complete(subscriber.connect(host, port))
|
|
|
|
try:
|
|
|
|
loop.run_until_complete(asyncio.wait_for(subscriber.receive_task,
|
|
|
|
None))
|
|
|
|
print("Connection to master lost")
|
|
|
|
finally:
|
|
|
|
loop.run_until_complete(subscriber.close())
|
2014-10-23 18:48:03 +08:00
|
|
|
finally:
|
2014-12-28 18:56:26 +08:00
|
|
|
loop.close()
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
args = _get_args()
|
|
|
|
if args.action == "show-queue":
|
|
|
|
queue = []
|
|
|
|
def init_queue(x):
|
|
|
|
queue[:] = x
|
|
|
|
return queue
|
2014-12-29 18:44:50 +08:00
|
|
|
subscriber = Subscriber("queue", init_queue,
|
|
|
|
lambda: _show_queue(queue))
|
|
|
|
_run_subscriber(args.server, args.port, subscriber)
|
|
|
|
elif args.action == "show-periodic":
|
|
|
|
periodic = dict()
|
|
|
|
def init_periodic(x):
|
|
|
|
periodic.clear()
|
|
|
|
periodic.update(x)
|
|
|
|
return periodic
|
|
|
|
subscriber = Subscriber("periodic", init_periodic,
|
|
|
|
lambda: _show_periodic(periodic))
|
|
|
|
_run_subscriber(args.server, args.port, subscriber)
|
2014-12-28 18:56:26 +08:00
|
|
|
else:
|
|
|
|
port = 8888 if args.port is None else args.port
|
2014-12-31 20:13:10 +08:00
|
|
|
remote = Client(args.server, port, "master_schedule")
|
2014-12-28 18:56:26 +08:00
|
|
|
try:
|
|
|
|
globals()["_action_" + args.action](remote, args)
|
|
|
|
finally:
|
|
|
|
remote.close_rpc()
|
2014-10-05 16:25:31 +08:00
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|