export_thread.py 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. import os
  2. import pathlib
  3. import queue
  4. import subprocess
  5. import threading
  6. from exception import FilterException
  7. import dest_paths
  8. import export_task
  9. import svg
  10. import util
  11. import log
  12. class ExportThread:
  13. """
  14. A class representing and managing a single thread that executes
  15. exporting tasks from the export queue.
  16. """
  17. def __init__(self, queue, name, total, m, input_path, formats, path,
  18. renderer, license_enabled, cache):
  19. self.queue = queue
  20. self.name = name
  21. self.total = total
  22. self.m = m
  23. self.input_path = input_path
  24. self.formats = formats
  25. self.path = path
  26. self.cache = cache
  27. self.renderer = renderer
  28. self.license_enabled = license_enabled
  29. self.err = None
  30. # this essentially tells self.run() to stop running if it is True
  31. self.kill_flag = False
  32. # the actual thread part of this thread
  33. self.thread = threading.Thread(target=self.run)
  34. # start the thread part of this thread!
  35. self.thread.start()
  36. def kill(self):
  37. """
  38. Requests this thread to be teriminated by activating the self.kill_flag flag.
  39. (This effectively stops self.run() from running)
  40. """
  41. self.kill_flag = True
  42. def join(self):
  43. """
  44. Wait for this thread to finish and merge it.
  45. """
  46. self.thread.join()
  47. def export_emoji(self, emoji, emoji_svg, f, path, license):
  48. """
  49. Runs a single export batch.
  50. """
  51. final_path = dest_paths.format_path(path, emoji, f)
  52. # try to make the directory for this particular export batch.
  53. dest_paths.make_dir_structure_for_file(final_path)
  54. # svg format doesn't involve a resolution so it can go straight to export.
  55. if f == 'svg':
  56. export_task.to_svg(emoji_svg, final_path, self.name, license.get('svg'), self.license_enabled, optimise=False)
  57. elif f == 'svgo':
  58. export_task.to_svg(emoji_svg, final_path, self.name, license.get('svg'), self.license_enabled, optimise=True)
  59. else:
  60. # any format other than svg is a raster, therefore it needs
  61. # to have a number separated by a dash.
  62. raster_format = f.split("-")
  63. try:
  64. size = int(raster_format[1])
  65. except ValueError:
  66. self.err = Exception(f"""A format you gave ('{f}') isn't correct. All formats
  67. that aren't svg must have a number separated by a dash.
  68. (ie 'png-32', 'webp-128')""")
  69. # now the size has been retrieved, try image
  70. # conversion based on the format.
  71. if raster_format[0] == "png":
  72. export_task.to_raster(emoji_svg, final_path, self.renderer, "png", size, self.name)
  73. elif raster_format[0] == "pngc":
  74. export_task.to_raster(emoji_svg, final_path, self.renderer, "pngc", size, self.name)
  75. elif raster_format[0] == "webp":
  76. export_task.to_raster(emoji_svg, final_path, self.renderer, "webp", size, self.name)
  77. elif raster_format[0] == "flif":
  78. export_task.to_raster(emoji_svg, final_path, self.renderer, "flif", size, self.name)
  79. elif raster_format[0] == "avif":
  80. export_task.to_raster(emoji_svg, final_path, self.renderer, "avif", size, self.name)
  81. else:
  82. self.err = Exception(f"""A format you gave ('{f}') uses a file format
  83. ('{raster_format[0]}') that orxporter
  84. doesn't support.""")
  85. def run(self):
  86. """
  87. The process of getting and executing a single export task in
  88. the queue.
  89. This is what the actual thread part of this class is tasked
  90. with working on.
  91. """
  92. try:
  93. # basically: do stuff as long as it's not requested to
  94. # be killed by the class
  95. while not self.kill_flag:
  96. # try to get an item from the queue.
  97. # break the loop if nothing is left.
  98. try:
  99. i, emoji = self.queue.get_nowait()
  100. except queue.Empty:
  101. break
  102. # compose the file path of the emoji.
  103. dest_paths.format_path(self.path, emoji, 'svg')
  104. # check if the src attribute is in the emoji.
  105. # if so, make a proper path out of it.
  106. if 'src' not in emoji:
  107. raise ValueError('Missing src attribute')
  108. srcpath = os.path.join(self.m.homedir, self.input_path,
  109. emoji['src'])
  110. # load the SVG source file
  111. try:
  112. emoji_svg = open(srcpath, 'r').read()
  113. except Exception:
  114. raise ValueError('Could not load file: ' + srcpath)
  115. # convert colormaps (if applicable)
  116. if 'color' in emoji:
  117. pfrom, pto = util.get_color_palettes(emoji, self.m)
  118. emoji_svg = svg.translate_color(emoji_svg, pfrom, pto)
  119. # for each format in the emoji, export it as that
  120. for f in self.formats:
  121. final_path = dest_paths.format_path(self.path, emoji, f)
  122. cache_hit = False
  123. if self.cache:
  124. dest_paths.make_dir_structure_for_file(final_path)
  125. cache_hit = self.cache.load_from_cache(emoji, f,
  126. final_path)
  127. if not cache_hit:
  128. self.export_emoji(emoji, emoji_svg, f, self.path,
  129. self.m.license)
  130. if self.cache:
  131. self.cache.save_to_cache(emoji, f, final_path)
  132. # tell the progress bar that this task has been completed.
  133. log.export_task_count += 1
  134. except Exception as e:
  135. self.err = e