#!/usr/bin/env python # TODO: # - -h text # - output to clay import os import sys import logging import json import requests import argparse import base64 logging.basicConfig( level=logging.WARNING, format='%(levelname)s %(funcName)s %(lineno)s - %(message)s', stream=sys.stderr, ) logging.debug(['sys.argv', sys.argv]) def preprocess_args(old_args): """Split out [] We use [] to delimit tuples. The following syntaxes are all equivalent: -- --open --open a b --close --open c d --close --close -- [ [ a b ] [ c d ] ] -- [ [a b] [c d] ] -- [[a b] [c d]] -- etc We don't allow [[a b][c d]]. The rule is that we accept zero or more [ at the beginning of a token and zero or more ] at the end of a token. In this function, we convert all legal syntaxes to as if they were entered as in the first example above. This allows them to be parsed by a relatively sane argparse system. """ if old_args == []: return [] if old_args[0][0] == '[': if len(old_args[0]) > 1: r = preprocess_args([old_args[0][1:]] + old_args[1:]) return ['--open'] + r else: return ['--open'] + preprocess_args(old_args[1:]) if old_args[0][-1] == ']': if len(old_args[0]) > 1: return preprocess_args([old_args[0][:-1]]) + \ ['--close'] + preprocess_args(old_args[1:]) else: return ['--close'] + preprocess_args(old_args[1:]) return [old_args[0]] + preprocess_args(old_args[1:]) args = preprocess_args(sys.argv[1:]) logging.debug(['preprocessed', args]) class sourceAction(argparse.Action): """Handle source flag. This is all the 'primitive' source flags -- no nesting, no tuple stuff, just one flag with one argument. Besides the normal argparse.Action arguments, we require the following named argument: -- which='foo'. Since all source flags use res.source, this specifies the key of the entry for this flag. """ def __init__(self, option_strings, dest, **kwargs): self.which = kwargs['which'] del kwargs['which'] logging.debug('args %s %s %s' % (option_strings, dest, kwargs)) super(sourceAction, self).__init__(option_strings, dest, **kwargs) def __call__(self, parser, res, new_value, option_string): logging.debug('%r %r' % (new_value, option_string)) logging.debug('source %s' % res.source) logging.debug('level %s' % res.level) if res.source is not None: def help(source, level): logging.debug('source %s' % source) logging.debug('level %s' % level) if not isinstance(source, list): raise ValueError('Already specified one source') elif level == 0: msg = 'Already specified a source %r %s' % (source, level) raise ValueError(msg) elif level == 1: return source + [self.construct_value(new_value)] else: return source[:-1] + [help(source[-1], level - 1)] res.source = help(res.source, res.level) else: res.source = \ self.construct_value(new_value) logging.debug(res.source) def construct_value(self, new_value): # hack to allow dojo commands like '-test' without them parsing # as bash command-line arguments if new_value[0:4] == '####': return self.construct_value(new_value[4:]) if new_value == '-': return self.construct_value(''.join(sys.stdin.readlines())) elif new_value[0:2] == '@@': with open(new_value[2:]) as f: content = f.readlines() return self.construct_value(''.join(content)) else: return {self.which: new_value} class importFileAction(argparse.Action): """Handles the import statement. The --import statement reads in a jammed noun file from the current working directory and stuffs it the base64 encoded version which gets passed into your Urbit. """ def __call__(self, parser, res, new_value, option_string): logging.debug('%r %r' % (new_value, option_string)) logging.debug('source %s' % res.source) logging.debug('level %s' % res.level) # We check to see if there's a "{new_value}.jam" file in the current # working directory. If there isn't, we error data = "" filename = new_value + ".jam" with open(filename, 'rb') as f: data = f.read() if data == "": raise ValueError('Failed to read jamfile') base_data = base64.b64encode(data) res.source = {"import": {"app": new_value, "base64-jam": base_data}} class importAllAction(argparse.Action): """Handles the import-all statement. The --import-all statement reads in a jammed noun file from the path passed in and stuffs it the base64 encoded version which gets passed into your Urbit. """ def __call__(self, parser, res, new_value, option_string): logging.debug('%r %r' % (new_value, option_string)) logging.debug('source %s' % res.source) logging.debug('level %s' % res.level) # We check to see if there's a "{new_value}" file in the current # working directory. If there isn't, we error data = "" filename = new_value with open(filename, 'rb') as f: data = f.read() if data == "": raise ValueError('Failed to read jamfile') base_data = base64.b64encode(data) res.source = {"import-all": {"base64-jam": base_data}} class transformerAction(argparse.Action): """Handle transformer flag. This is all the tranformer flags. Each flag takes one argument and transforms the previous source. Besides the normal argparse.Action arguments, we require the following named arguments: -- which='foo'. Since all source flags use res.source, this specifies the key of the entry for this flag. -- nesting='foo'. The key for the argument is 'foo'. """ def __init__(self, option_strings, dest, **kwargs): self.which = kwargs['which'] self.nesting = kwargs['nesting'] del kwargs['which'] del kwargs['nesting'] super(transformerAction, self).__init__(option_strings, dest, **kwargs) def __call__(self, parser, res, new_value, option_string): logging.debug('%r %r' % (new_value, option_string)) logging.debug('source %s' % res.source) logging.debug('level %s' % res.level) if res.source is None: raise ValueError('Need source before transformer') else: def help(source, level): logging.debug('source %s' % source) logging.debug('level %s' % level) if level == 0 or level is None: res = {self.nesting: new_value, "next": source} return {self.which: res} elif not isinstance(source, list): raise ValueError('Already specified one source') else: return source[:-1] + [help(source[-1], level - 1)] res.source = help(res.source, res.level) logging.debug(res.source) class openAction(argparse.Action): """Handle open tuple. Opens a source tuple. Can only exist in the same places as any other source. """ def __init__(self, option_strings, dest, **kwargs): super(openAction, self).__init__(option_strings, dest, **kwargs) def __call__(self, parser, res, new_value, option_string): if res.level is None: res.level = 0 logging.debug('source %s' % res.source) logging.debug('level %s' % res.level) if res.source is None: res.source = [] res.level = 1 return def help(source, level): if not isinstance(source, list): raise ValueError('Starting tuple after source is finished') if level == 1: return (source + [[]], level + 1) elif level > 1: rsource, rlevel = help(source[-1], level - 1) return (source[:-1] + [rsource], rlevel + 1) else: msg = 'opening strange level %r %s' % (source, level) raise ValueError(msg) res.source, res.level = help(res.source, res.level) class closeAction(argparse.Action): """Handle close tuple. Closes a source tuple. Can only exist when a tuple is already open. """ def __init__(self, option_strings, dest, **kwargs): super(closeAction, self).__init__(option_strings, dest, **kwargs) def __call__(self, parser, res, new_value, option_string): if res.level is None: raise ValueError('Ending tuple before starting one') logging.debug('level %s' % res.level) if res.source is None: raise ValueError('Ending tuple with empty source') def help(source, level): if not isinstance(source, list): raise ValueError('Ending tuple that isn\'t a tuple') if level == 1: return level - 1 elif level > 1: return help(source[-1], level - 1) + 1 else: msg = 'closing strange level %r %s' % (source, level) raise ValueError(msg) res.level = help(res.source, res.level) logging.debug('level %s' % res.level) class sinkAction(argparse.Action): """Handle sink flag. We expect only one sinkAction to ever be executed. We recommend using mutually_exclusive_group's. Besides the normal action flags, we require the following named argument: -- which='foo'. Since all sink flags use res.sink, this specifies the key of the entry for this flag. """ def __init__(self, option_strings, dest, **kwargs): self.which = kwargs['which'] del kwargs['which'] super(sinkAction, self).__init__(option_strings, dest, **kwargs) def __call__(self, parser, res, new_value, option_string): res.sink = self.construct_value(new_value) logging.debug(res.sink) def construct_value(self, new_value): if self.which == 'output-file': return {self.which: new_value[::-1].replace('.','/',1)[::-1]} elif self.which == 'output-pill': return {self.which: new_value[::-1].replace('.','/',1)[::-1]} else: return {self.which: new_value} class FullPaths(argparse.Action): """Expand user- and relative-paths""" def __call__(self, parser, namespace, values, option_string=None): if values != None: path = os.path.abspath(os.path.expanduser(values)) setattr(namespace, self.dest, path) def is_dir(dirname): """Checks if a path is an actual directory""" if not os.path.isdir(dirname): msg = "{0} is not a directory".format(dirname) raise argparse.ArgumentTypeError(msg) else: return dirname def get_args(): """Get CLI arguments and options""" parser = argparse.ArgumentParser(description="""do something""") parser.add_argument('alignments', help="The folder of alignments", action=FullPaths, type=is_dir) parser = argparse.ArgumentParser(description='headless urbit') parser.add_argument('pier', nargs='?', help='target urbit directory', action=FullPaths, type=is_dir) parser.add_argument('-d', '--dojo', which='dojo', metavar='command-line', help='run dojo command', action=sourceAction, dest='source') parser.add_argument('-D', '--data', which='data', metavar='text', help='literal text data', action=sourceAction) parser.add_argument('-c', '--clay', which='clay', metavar='clay-path', help='load data from clay', action=sourceAction) parser.add_argument('-u', '--url', which='url', metavar='url', help='pull data from url', action=sourceAction) parser.add_argument('-a', '--api', which='api', metavar='command', help='get data from api', action=sourceAction) parser.add_argument('-g', '--get-api', which='get-api', metavar='api:endpoint', help='get data from api endpoint', action=sourceAction) parser.add_argument('-l', '--listen-api', which='listen-api', metavar='api:event', help='listen to event from api', action=sourceAction) parser.add_argument('-e', '--export', which='export', metavar='app-name', help='exports the application state', action=sourceAction) parser.add_argument('-i', '--import', metavar='app-name', help='imports the application state', action=importFileAction) parser.add_argument('-E', '--export-all', const={'export-all': None}, help='exports data from all landscape apps', action='store_const', dest='source') parser.add_argument('-I', '--import-all', metavar='jam-file', help='imports data for all landscape apps', action=importAllAction) parser.add_argument('-m', '--mark', which='as', metavar='mark', help='transform a source to another mark', nesting='mark', action=transformerAction) parser.add_argument('-H', '--hoon', which='hoon', metavar='code', help='transform a source by hoon code', nesting='code', action=transformerAction) parser.add_argument('--open', nargs=0, help='start tuple', action=openAction, dest='level') parser.add_argument('--close', nargs=0, help='stop tuple', action=closeAction) parser.add_argument('--cancel', const={'cancel': None}, help='cancels active lens command', action='store_const', dest='source') sinks = parser.add_mutually_exclusive_group() sinks.add_argument('-s', '--stdout', const={'stdout': None}, default={'stdout': None}, action='store_const', dest='sink') sinks.add_argument('-f', '--output-file', which='output-file', metavar='path', action=sinkAction) sinks.add_argument('-P', '--output-pill', which='output-pill', metavar='path', action=sinkAction) sinks.add_argument('-C', '--output-clay', which='output-clay', metavar='clay-path', action=sinkAction) sinks.add_argument('-U', '--output-url', which='url', metavar='url', action=sinkAction) sinks.add_argument('-t', '--to-api', which='to-api', metavar='api-command', action=sinkAction) sinks.add_argument('-n', '--send-api', which='send-api', metavar='api:endpoint', action=sinkAction) sinks.add_argument('-x', '--command', which='command', metavar='command', action=sinkAction) sinks.add_argument('-p', '--app', which='app', metavar='app', action=sinkAction) args = parser.parse_args(args) if args.source is None: args.source = {"data": ''.join(sys.stdin)} payload = {"source": args.source, "sink": args.sink} logging.debug(['payload', json.dumps(payload)]) PORT = "" if args.pier is None: PORT = os.environ.get('LENS_PORT', 12321) if not os.environ.has_key('LENS_PORT'): logging.warn("No pier or port specified, looking on port " + str(PORT)) else: with open(os.path.join(args.pier, ".http.ports")) as ports: for line in ports: if -1 != line.find("loopback"): PORT = line.split()[0] logging.info("Found port %s" % PORT) break if not PORT: logging.error("Error reading port from .http.ports file") sys.exit(1) url = "http://localhost:%s" % PORT r = requests.post(url, data=json.dumps(payload)) if r.headers.get('content-type') == 'application/octet-stream': name = r.headers.get('content-disposition').split('filename=',1)[1][1:-1] with open(name, 'wb') as f: for block in r.iter_content(1024): f.write(block) elif r.text[0] == '"': print r.text[1:-1].encode('utf-8').decode('string_escape') elif r.text[0] == '{': # print r.text json_data = json.loads(r.text) logging.debug(json_data) with open(json_data['file'][:0:-1].replace('/','.',1)[::-1], 'w') as f: f.write(base64.b64decode(json_data['data'])) else: logging.warn("unrecognized response") print r.text