python_qmp_updater.py 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. #!/usr/bin/env python3
  2. #
  3. # Intended usage:
  4. #
  5. # git grep -l '\.qmp(' | xargs ./scripts/python_qmp_updater.py
  6. #
  7. import re
  8. import sys
  9. from typing import Optional
  10. start_reg = re.compile(r'^(?P<padding> *)(?P<res>\w+) = (?P<vm>.*).qmp\(',
  11. flags=re.MULTILINE)
  12. success_reg_templ = re.sub('\n *', '', r"""
  13. (\n*{padding}(?P<comment>\#.*$))?
  14. \n*{padding}
  15. (
  16. self.assert_qmp\({res},\ 'return',\ {{}}\)
  17. |
  18. assert\ {res}\['return'\]\ ==\ {{}}
  19. |
  20. assert\ {res}\ ==\ {{'return':\ {{}}}}
  21. |
  22. self.assertEqual\({res}\['return'\],\ {{}}\)
  23. )""")
  24. some_check_templ = re.sub('\n *', '', r"""
  25. (\n*{padding}(?P<comment>\#.*$))?
  26. \s*self.assert_qmp\({res},""")
  27. def tmatch(template: str, text: str,
  28. padding: str, res: str) -> Optional[re.Match[str]]:
  29. return re.match(template.format(padding=padding, res=res), text,
  30. flags=re.MULTILINE)
  31. def find_closing_brace(text: str, start: int) -> int:
  32. """
  33. Having '(' at text[start] search for pairing ')' and return its index.
  34. """
  35. assert text[start] == '('
  36. height = 1
  37. for i in range(start + 1, len(text)):
  38. if text[i] == '(':
  39. height += 1
  40. elif text[i] == ')':
  41. height -= 1
  42. if height == 0:
  43. return i
  44. raise ValueError
  45. def update(text: str) -> str:
  46. result = ''
  47. while True:
  48. m = start_reg.search(text)
  49. if m is None:
  50. result += text
  51. break
  52. result += text[:m.start()]
  53. args_ind = m.end()
  54. args_end = find_closing_brace(text, args_ind - 1)
  55. all_args = text[args_ind:args_end].split(',', 1)
  56. name = all_args[0]
  57. args = None if len(all_args) == 1 else all_args[1]
  58. unchanged_call = text[m.start():args_end+1]
  59. text = text[args_end+1:]
  60. padding, res, vm = m.group('padding', 'res', 'vm')
  61. m = tmatch(success_reg_templ, text, padding, res)
  62. if m is None:
  63. result += unchanged_call
  64. if ('query-' not in name and
  65. 'x-debug-block-dirty-bitmap-sha256' not in name and
  66. not tmatch(some_check_templ, text, padding, res)):
  67. print(unchanged_call + text[:200] + '...\n\n')
  68. continue
  69. if m.group('comment'):
  70. result += f'{padding}{m.group("comment")}\n'
  71. result += f'{padding}{vm}.cmd({name}'
  72. if args:
  73. result += ','
  74. if '\n' in args:
  75. m_args = re.search('(?P<pad> *).*$', args)
  76. assert m_args is not None
  77. cur_padding = len(m_args.group('pad'))
  78. expected = len(f'{padding}{res} = {vm}.qmp(')
  79. drop = len(f'{res} = ')
  80. if cur_padding == expected - 1:
  81. # tolerate this bad style
  82. drop -= 1
  83. elif cur_padding < expected - 1:
  84. # assume nothing to do
  85. drop = 0
  86. if drop:
  87. args = re.sub('\n' + ' ' * drop, '\n', args)
  88. result += args
  89. result += ')'
  90. text = text[m.end():]
  91. return result
  92. for fname in sys.argv[1:]:
  93. print(fname)
  94. with open(fname) as f:
  95. t = f.read()
  96. t = update(t)
  97. with open(fname, 'w') as f:
  98. f.write(t)