manifest.py 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. import json
  2. import os
  3. import parse
  4. import util
  5. class Manifest:
  6. def __init__(self, homedir='.', filename=None):
  7. self.homedir = homedir
  8. self.classes = {}
  9. self.colormaps = {}
  10. self.defines = {}
  11. self.emoji = []
  12. self.palettes = {}
  13. self.shortcodes = {}
  14. self.codepoints = {}
  15. self.license = {}
  16. if filename is not None:
  17. self.include(filename)
  18. def add_emoji(self, emoji):
  19. self.emoji.append(emoji)
  20. if 'code' in emoji:
  21. if emoji['code'] in self.shortcodes:
  22. raise ValueError('Shortcode already in use: ' + emoji['code'])
  23. self.shortcodes[emoji['code']] = emoji
  24. if 'unicode' in emoji and '!' not in emoji['unicode']:
  25. if emoji['unicode'] in self.codepoints:
  26. raise ValueError('Codepoint already in use: ' +
  27. util.uni_to_hex_hash(emoji['unicode']))
  28. self.codepoints[emoji['unicode']] = emoji
  29. def compile_emoji(self, kwargs, color=None):
  30. res = dict(kwargs)
  31. if not color and 'color' in res:
  32. del res['color']
  33. elif color:
  34. if color not in self.colormaps:
  35. raise ValueError('Undefined colormap: ' + color)
  36. res['color'] = color
  37. for k, v in res.items():
  38. if '%c' in v:
  39. if not color:
  40. raise ValueError('%c without colormap')
  41. try:
  42. res[k] = v.replace('%c', self.colormaps[color]['code'])
  43. except KeyError:
  44. raise ValueError('Shortcode not defined for colormap: ' +
  45. color)
  46. if '%C' in v:
  47. if not color:
  48. raise ValueError('%C without colormap')
  49. try:
  50. subst = self.colormaps[color]['code']
  51. except KeyError:
  52. raise ValueError('Shortcode not defined for colormap: ' +
  53. color)
  54. if subst:
  55. subst = '_' + subst
  56. res[k] = v.replace('%C', subst)
  57. if '%u' in v:
  58. if not color:
  59. raise ValueError('%u without colormap')
  60. try:
  61. res[k] = v.replace('%u', self.colormaps[color]['unicode'])
  62. except KeyError:
  63. raise ValueError('Codepoint not defined for colormap: ' +
  64. color)
  65. if '%U' in v:
  66. if not color:
  67. raise ValueError('%U without colormap')
  68. try:
  69. color_code = self.colormaps[color]['unicode']
  70. except KeyError:
  71. raise ValueError('Codepoint not defined for colormap: ' +
  72. color)
  73. if color_code:
  74. res[k] = v.replace('%U', '#200D ' + color_code)
  75. else:
  76. res[k] = v.replace('%U', '')
  77. idx = 0
  78. while idx < len(v):
  79. idx = v.find('%(', idx)
  80. if idx == -1:
  81. break
  82. end = v.find(')', idx+2)
  83. if end == -1:
  84. raise ValueError('No matching parenthesis')
  85. prop = v[idx+2:end]
  86. if prop not in res:
  87. raise ValueError('Undefined property: ' + prop)
  88. res[k] = v[:idx] + res[prop] + v[end+1:]
  89. idx += 1
  90. v = res[k]
  91. if 'unicode' in res:
  92. if '!' in res['unicode']:
  93. res['unicode'] = '!'
  94. else:
  95. unistr = []
  96. for char in res['unicode'].split():
  97. try:
  98. if char[0] == '#':
  99. unistr.append(int(char[1:], 16))
  100. else:
  101. unistr.append(int(char))
  102. except ValueError:
  103. raise ValueError('Expected a number: ' + char)
  104. res['unicode'] = tuple(unistr)
  105. return res
  106. def exec_class(self, args, kwargs):
  107. if not args:
  108. raise ValueError('Missing id')
  109. if args[0] in self.classes:
  110. raise ValueError('Already defined: ' + args[0])
  111. if 'class' in kwargs:
  112. raise ValueError('Illegal recursion in class definition')
  113. res = {}
  114. for parent in args[1:]:
  115. if parent not in self.classes:
  116. raise ValueError('Parent class is undefined: ' + parent)
  117. res.update(self.classes[parent])
  118. res.update(kwargs)
  119. self.classes[args[0]] = res
  120. def exec_colormap(self, args, kwargs):
  121. if not args:
  122. raise ValueError('Missing id')
  123. if len(args) > 1:
  124. raise ValueError('Multiple ids')
  125. if args[0] in self.colormaps:
  126. raise ValueError('Already defined: ' + args[0])
  127. if 'src' not in kwargs:
  128. raise ValueError('Missing src')
  129. if 'dst' not in kwargs:
  130. raise ValueError('Missing dst')
  131. if kwargs['src'] not in self.palettes:
  132. raise ValueError('Undefined source palette: ' + kwargs['src'])
  133. if kwargs['dst'] not in self.palettes:
  134. raise ValueError('Undefined target palette: ' + kwargs['dst'])
  135. self.colormaps[args[0]] = kwargs
  136. def exec_define(self, args, kwargs):
  137. if kwargs:
  138. raise ValueError('kwargs not allowed in define expression')
  139. if len(args) < 2:
  140. raise ValueError('Missing argument')
  141. if args[0] in self.defines:
  142. raise ValueError('Already defined: ' + args[0])
  143. self.defines[args[0]] = ' '.join(args[1:])
  144. def exec_emoji(self, args, kwargs):
  145. emoji_args = {}
  146. for c in kwargs.get('class', '').split():
  147. if c not in self.classes:
  148. raise ValueError('Undefined class: ' + c)
  149. emoji_args.update(self.classes[c])
  150. emoji_args.update(kwargs)
  151. if 'src' not in emoji_args:
  152. raise ValueError('Missing src')
  153. if 'color' in emoji_args:
  154. for color in emoji_args['color'].split():
  155. self.add_emoji(self.compile_emoji(emoji_args, color))
  156. else:
  157. self.add_emoji(self.compile_emoji(emoji_args))
  158. def exec_include(self, args, kwargs):
  159. if not args:
  160. raise Exception('Missing filename')
  161. if len(args) > 1:
  162. raise ValueError('Multiple filenames')
  163. self.include(args[0])
  164. def exec_license(self, args, kwargs):
  165. for k, v in kwargs.items():
  166. path = os.path.join(self.homedir, v)
  167. if k == 'svg':
  168. try:
  169. self.license['svg'] = open(path, 'r').read()
  170. except OSError:
  171. raise Exception('Failed to load license file: ' + path)
  172. elif k == 'png':
  173. try:
  174. self.license['png'] = json.load(open(path, 'r'))
  175. except OSError:
  176. raise Exception('Failed to load license file: ' + path)
  177. except ValueError:
  178. raise ValueError('Failed to parse JSON in file: ' + path)
  179. def exec_palette(self, args, kwargs):
  180. if not args:
  181. raise ValueError('Missing id')
  182. if len(args) > 1:
  183. raise ValueError('Multiple ids')
  184. if args[0] in self.palettes:
  185. raise ValueError('Already defined: ' + args[0])
  186. self.palettes[args[0]] = kwargs
  187. def exec_expr(self, expr):
  188. final_expr = parse.subst_consts(expr, self.defines)
  189. try:
  190. head, args, kwargs = parse.parse_expr(final_expr)
  191. except Exception:
  192. raise ValueError('Syntax error')
  193. if head is None:
  194. return
  195. elif head == 'class':
  196. self.exec_class(args, kwargs)
  197. elif head == 'colormap':
  198. self.exec_colormap(args, kwargs)
  199. elif head == 'define':
  200. self.exec_define(args, kwargs)
  201. elif head == 'emoji':
  202. self.exec_emoji(args, kwargs)
  203. elif head == 'include':
  204. self.exec_include(args, kwargs)
  205. elif head == 'license':
  206. self.exec_license(args, kwargs)
  207. elif head == 'palette':
  208. self.exec_palette(args, kwargs)
  209. else:
  210. raise ValueError('Unknown expression type: ' + head)
  211. def include(self, filename):
  212. try:
  213. m_file = open(os.path.join(self.homedir, filename), 'r')
  214. except OSError:
  215. raise Exception('Could not open manifest file: ' + filename)
  216. for expr, line_num in parse.exps(m_file):
  217. try:
  218. self.exec_expr(expr)
  219. except Exception as e:
  220. raise Exception(f'In manifest file `{filename}` at line '
  221. f'{line_num}:\n'
  222. f'`{expr.strip()}`\n'
  223. f'Error: {e}')
  224. m_file.close()