5 changed files with 159 additions and 16 deletions
@ -0,0 +1,112 @@ |
|||
import re |
|||
|
|||
PARTS_RE = re.compile('(\<|\[)((?:\w+|\:|\||\.\.\.| (?:[0-9]+))+)(?:\>|\])') |
|||
|
|||
TYPE_MAP = { |
|||
'str': str, |
|||
'int': int, |
|||
'float': float, |
|||
'snowflake': int, |
|||
} |
|||
|
|||
|
|||
class ArgumentError(Exception): |
|||
pass |
|||
|
|||
|
|||
class Argument(object): |
|||
def __init__(self, raw): |
|||
self.name = None |
|||
self.count = 1 |
|||
self.required = False |
|||
self.types = None |
|||
self.parse(raw) |
|||
|
|||
@property |
|||
def true_count(self): |
|||
return self.count or 1 |
|||
|
|||
def convert(self, obj): |
|||
for typ in self.types: |
|||
typ = TYPE_MAP.get(typ) |
|||
try: |
|||
return typ(obj) |
|||
except Exception as e: |
|||
continue |
|||
raise e |
|||
|
|||
def parse(self, raw): |
|||
prefix, part = raw |
|||
|
|||
if prefix == '<': |
|||
self.required = True |
|||
else: |
|||
self.required = False |
|||
|
|||
if part.endswith('...'): |
|||
part = part[:-3] |
|||
self.count = 0 |
|||
elif ' ' in part: |
|||
split = part.split(' ', 1) |
|||
part, self.count = split[0], int(split[1]) |
|||
|
|||
if ':' in part: |
|||
part, typeinfo = part.split(':') |
|||
self.types = typeinfo.split('|') |
|||
|
|||
self.name = part.strip() |
|||
|
|||
|
|||
class ArgumentSet(object): |
|||
def __init__(self, args=None): |
|||
self.args = args or [] |
|||
|
|||
def append(self, arg): |
|||
if self.args and not self.args[-1].required and arg.required: |
|||
raise Exception('Required argument cannot come after an optional argument') |
|||
|
|||
if self.args and not self.args[-1].count: |
|||
raise Exception('No arguments can come after a catch-all') |
|||
|
|||
self.args.append(arg) |
|||
|
|||
def parse(self, rawargs): |
|||
parsed = [] |
|||
|
|||
for index, arg in enumerate(self.args): |
|||
if not arg.required and index + arg.true_count <= len(rawargs): |
|||
continue |
|||
|
|||
raw = rawargs[index:index + arg.true_count] |
|||
|
|||
if arg.types: |
|||
for idx, r in enumerate(raw): |
|||
try: |
|||
raw[idx] = arg.convert(r) |
|||
except: |
|||
raise ArgumentError('cannot convert `{}` to `{}`'.format( |
|||
r, ', '.join(arg.types) |
|||
)) |
|||
|
|||
parsed.append(raw) |
|||
|
|||
return parsed |
|||
|
|||
@property |
|||
def length(self): |
|||
return len(self.args) |
|||
|
|||
@property |
|||
def required_length(self): |
|||
return sum([i.true_count for i in self.args if i.required]) |
|||
|
|||
|
|||
def parse_arguments(line): |
|||
args = ArgumentSet() |
|||
|
|||
data = PARTS_RE.findall(line) |
|||
if len(data): |
|||
for item in data: |
|||
args.append(Argument(item)) |
|||
|
|||
return args |
Loading…
Reference in new issue