123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183 |
- #!/usr/bin/env python3
- #
- # Opts.py, an options parsing library for Python.
- # Copyright (C)2013-2023 The MMGen Project <mmgen@tuta.io>
- #
- # This program is free software: you can redistribute it and/or modify
- # it under the terms of the GNU General Public License as published by
- # the Free Software Foundation, either version 3 of the License, or
- # (at your option) any later version.
- #
- # This program is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- # GNU General Public License for more details.
- #
- # You should have received a copy of the GNU General Public License
- # along with this program. If not, see <http://www.gnu.org/licenses/>.
- """
- share.Opts: Generic options parsing
- """
- import sys,re
- from collections import namedtuple
- pat = re.compile(r'^-([a-zA-Z0-9-]), --([a-zA-Z0-9-]{2,64})(=| )(.+)')
- def make_usage_str(prog_name,caller,data):
- lines = [data.strip()] if type(data) == str else data
- indent,col1_w = {
- 'help': (2,len(prog_name)+1),
- 'user': (0,len('USAGE:')),
- }[caller]
- def gen():
- ulbl = 'USAGE:'
- for line in lines:
- yield f'{ulbl:{col1_w}} {prog_name} {line}'
- ulbl = ''
- return ('\n'+(' '*indent)).join(gen())
- def usage(opts_data):
- print(make_usage_str(
- prog_name = opts_data['prog_name'],
- caller = 'user',
- data = opts_data['text'].get('usage2') or opts_data['text']['usage'] ))
- sys.exit(1)
- def print_help(*args):
- print(make_help(*args))
- sys.exit(0)
- def make_help(cfg,proto,opts_data,opt_filter):
- def parse_lines(text):
- filtered = False
- for line in text.strip().splitlines():
- m = pat.match(line)
- if m:
- filtered = bool(opt_filter and m[1] not in opt_filter)
- if not filtered:
- yield fs.format( ('-'+m[1]+',','')[m[1]=='-'], m[2], m[4] )
- elif not filtered:
- yield line
- opts_type,fs = ('options','{:<3} --{} {}') if cfg.help else ('long_options','{} --{} {}')
- t = opts_data['text']
- c = opts_data['code']
- nl = '\n '
- pn = opts_data['prog_name']
- from mmgen.help import help_notes_func
- def help_notes(k):
- return help_notes_func(proto,cfg,k)
- def help_mod(modname):
- import importlib
- return importlib.import_module('mmgen.help.'+modname).help(proto,cfg)
- def gen_arg_tuple(func,text):
- d = {
- 'proto': proto,
- 'help_notes': help_notes,
- 'help_mod': help_mod,
- 'cfg': cfg,
- }
- for arg in func.__code__.co_varnames:
- yield d[arg] if arg in d else text
- def gen_text():
- yield ' {} {}'.format(pn.upper()+':',t['desc'].strip())
- yield make_usage_str(pn,'help',t.get('usage2') or t['usage'])
- yield opts_type.upper().replace('_',' ') + ':'
- # process code for options
- opts_text = nl.join(parse_lines(t[opts_type]))
- if opts_type in c:
- arg_tuple = tuple(gen_arg_tuple(c[opts_type],opts_text))
- yield c[opts_type](*arg_tuple)
- else:
- yield opts_text
- # process code for notes
- if opts_type == 'options' and 'notes' in t:
- notes_text = t['notes']
- if 'notes' in c:
- arg_tuple = tuple(gen_arg_tuple(c['notes'],notes_text))
- notes_text = c['notes'](*arg_tuple)
- for line in notes_text.splitlines():
- yield line
- return nl.join(gen_text()) + '\n'
- def process_uopts(opts_data,short_opts,long_opts):
- import os,getopt
- opts_data['prog_name'] = os.path.basename(sys.argv[0])
- try:
- cl_uopts,uargs = getopt.getopt(sys.argv[1:],''.join(short_opts),long_opts)
- except getopt.GetoptError as e:
- print(e.args[0])
- sys.exit(1)
- def get_uopts():
- for uopt,uparm in cl_uopts:
- if uopt.startswith('--'):
- lo = uopt[2:]
- if lo in long_opts:
- yield (lo.replace('-','_'), True)
- else: # lo+'=' in long_opts
- yield (lo.replace('-','_'), uparm)
- else: # uopt.startswith('-')
- so = uopt[1]
- if so in short_opts:
- yield (long_opts[short_opts.index(so)].replace('-','_'), True)
- else: # so+':' in short_opts
- yield (long_opts[short_opts.index(so+':')][:-1].replace('-','_'), uparm)
- uopts = dict(get_uopts())
- if 'sets' in opts_data:
- for a_opt,a_val,b_opt,b_val in opts_data['sets']:
- if a_opt in uopts:
- u_val = uopts[a_opt]
- if (u_val and a_val == bool) or u_val == a_val:
- if b_opt in uopts and uopts[b_opt] != b_val:
- sys.stderr.write(
- 'Option conflict:'
- + '\n --{}={}, with'.format(b_opt.replace('_','-'),uopts[b_opt])
- + '\n --{}={}\n'.format(a_opt.replace('_','-'),uopts[a_opt]) )
- sys.exit(1)
- else:
- uopts[b_opt] = b_val
- return uopts,uargs
- def parse_opts(opts_data,opt_filter=None,parse_only=False):
- short_opts,long_opts,filtered_opts = [],[],[]
- def parse_lines(opts_type):
- for line in opts_data['text'][opts_type].strip().splitlines():
- m = pat.match(line)
- if m:
- if opt_filter and m[1] not in opt_filter:
- filtered_opts.append(m[2])
- else:
- if opts_type == 'options':
- short_opts.append(m[1] + ('',':')[m[3] == '='])
- long_opts.append(m[2] + ('','=')[m[3] == '='])
- parse_lines('options')
- if 'long_options' in opts_data['text']:
- parse_lines('long_options')
- uopts,uargs = process_uopts(opts_data,short_opts,long_opts)
- return namedtuple('parsed_cmd_opts',['user_opts','cmd_args','opts','filtered_opts'])(
- uopts, # dict
- uargs, # list, callers can pop
- tuple(o.replace('-','_').rstrip('=') for o in long_opts),
- tuple(o.replace('-','_') for o in filtered_opts),
- )
|