Opts.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. #!/usr/bin/env python3
  2. #
  3. # Opts.py, an options parsing library for Python.
  4. # Copyright (C)2013-2023 The MMGen Project <mmgen@tuta.io>
  5. #
  6. # This program is free software: you can redistribute it and/or modify
  7. # it under the terms of the GNU General Public License as published by
  8. # the Free Software Foundation, either version 3 of the License, or
  9. # (at your option) any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # You should have received a copy of the GNU General Public License
  17. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. """
  19. share.Opts: Generic options parsing
  20. """
  21. import sys,re
  22. from collections import namedtuple
  23. pat = re.compile(r'^-([a-zA-Z0-9-]), --([a-zA-Z0-9-]{2,64})(=| )(.+)')
  24. def make_usage_str(prog_name,caller,data):
  25. lines = [data.strip()] if type(data) == str else data
  26. indent,col1_w = {
  27. 'help': (2,len(prog_name)+1),
  28. 'user': (0,len('USAGE:')),
  29. }[caller]
  30. def gen():
  31. ulbl = 'USAGE:'
  32. for line in lines:
  33. yield f'{ulbl:{col1_w}} {prog_name} {line}'
  34. ulbl = ''
  35. return ('\n'+(' '*indent)).join(gen())
  36. def usage(opts_data):
  37. print(make_usage_str(
  38. prog_name = opts_data['prog_name'],
  39. caller = 'user',
  40. data = opts_data['text'].get('usage2') or opts_data['text']['usage'] ))
  41. sys.exit(1)
  42. def print_help(*args):
  43. print(make_help(*args))
  44. sys.exit(0)
  45. def make_help(cfg,proto,opts_data,opt_filter):
  46. def parse_lines(text):
  47. filtered = False
  48. for line in text.strip().splitlines():
  49. m = pat.match(line)
  50. if m:
  51. filtered = bool(opt_filter and m[1] not in opt_filter)
  52. if not filtered:
  53. yield fs.format( ('-'+m[1]+',','')[m[1]=='-'], m[2], m[4] )
  54. elif not filtered:
  55. yield line
  56. opts_type,fs = ('options','{:<3} --{} {}') if cfg.help else ('long_options','{} --{} {}')
  57. t = opts_data['text']
  58. c = opts_data['code']
  59. nl = '\n '
  60. pn = opts_data['prog_name']
  61. from mmgen.help import help_notes_func
  62. def help_notes(k):
  63. return help_notes_func(proto,cfg,k)
  64. def help_mod(modname):
  65. import importlib
  66. return importlib.import_module('mmgen.help.'+modname).help(proto,cfg)
  67. def gen_arg_tuple(func,text):
  68. d = {
  69. 'proto': proto,
  70. 'help_notes': help_notes,
  71. 'help_mod': help_mod,
  72. 'cfg': cfg,
  73. }
  74. for arg in func.__code__.co_varnames:
  75. yield d[arg] if arg in d else text
  76. def gen_text():
  77. yield ' {} {}'.format(pn.upper()+':',t['desc'].strip())
  78. yield make_usage_str(pn,'help',t.get('usage2') or t['usage'])
  79. yield opts_type.upper().replace('_',' ') + ':'
  80. # process code for options
  81. opts_text = nl.join(parse_lines(t[opts_type]))
  82. if opts_type in c:
  83. arg_tuple = tuple(gen_arg_tuple(c[opts_type],opts_text))
  84. yield c[opts_type](*arg_tuple)
  85. else:
  86. yield opts_text
  87. # process code for notes
  88. if opts_type == 'options' and 'notes' in t:
  89. notes_text = t['notes']
  90. if 'notes' in c:
  91. arg_tuple = tuple(gen_arg_tuple(c['notes'],notes_text))
  92. notes_text = c['notes'](*arg_tuple)
  93. for line in notes_text.splitlines():
  94. yield line
  95. return nl.join(gen_text()) + '\n'
  96. def process_uopts(opts_data,short_opts,long_opts):
  97. import os,getopt
  98. opts_data['prog_name'] = os.path.basename(sys.argv[0])
  99. try:
  100. cl_uopts,uargs = getopt.getopt(sys.argv[1:],''.join(short_opts),long_opts)
  101. except getopt.GetoptError as e:
  102. print(e.args[0])
  103. sys.exit(1)
  104. def get_uopts():
  105. for uopt,uparm in cl_uopts:
  106. if uopt.startswith('--'):
  107. lo = uopt[2:]
  108. if lo in long_opts:
  109. yield (lo.replace('-','_'), True)
  110. else: # lo+'=' in long_opts
  111. yield (lo.replace('-','_'), uparm)
  112. else: # uopt.startswith('-')
  113. so = uopt[1]
  114. if so in short_opts:
  115. yield (long_opts[short_opts.index(so)].replace('-','_'), True)
  116. else: # so+':' in short_opts
  117. yield (long_opts[short_opts.index(so+':')][:-1].replace('-','_'), uparm)
  118. uopts = dict(get_uopts())
  119. if 'sets' in opts_data:
  120. for a_opt,a_val,b_opt,b_val in opts_data['sets']:
  121. if a_opt in uopts:
  122. u_val = uopts[a_opt]
  123. if (u_val and a_val == bool) or u_val == a_val:
  124. if b_opt in uopts and uopts[b_opt] != b_val:
  125. sys.stderr.write(
  126. 'Option conflict:'
  127. + '\n --{}={}, with'.format(b_opt.replace('_','-'),uopts[b_opt])
  128. + '\n --{}={}\n'.format(a_opt.replace('_','-'),uopts[a_opt]) )
  129. sys.exit(1)
  130. else:
  131. uopts[b_opt] = b_val
  132. return uopts,uargs
  133. def parse_opts(opts_data,opt_filter=None,parse_only=False):
  134. short_opts,long_opts,filtered_opts = [],[],[]
  135. def parse_lines(opts_type):
  136. for line in opts_data['text'][opts_type].strip().splitlines():
  137. m = pat.match(line)
  138. if m:
  139. if opt_filter and m[1] not in opt_filter:
  140. filtered_opts.append(m[2])
  141. else:
  142. if opts_type == 'options':
  143. short_opts.append(m[1] + ('',':')[m[3] == '='])
  144. long_opts.append(m[2] + ('','=')[m[3] == '='])
  145. parse_lines('options')
  146. if 'long_options' in opts_data['text']:
  147. parse_lines('long_options')
  148. uopts,uargs = process_uopts(opts_data,short_opts,long_opts)
  149. return namedtuple('parsed_cmd_opts',['user_opts','cmd_args','opts','filtered_opts'])(
  150. uopts, # dict
  151. uargs, # list, callers can pop
  152. tuple(o.replace('-','_').rstrip('=') for o in long_opts),
  153. tuple(o.replace('-','_') for o in filtered_opts),
  154. )