demo.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. # Copyright (c) Opendatalab. All rights reserved.
  2. import copy
  3. import json
  4. import os
  5. from pathlib import Path
  6. from loguru import logger
  7. from mineru.cli.common import convert_pdf_bytes_to_bytes_by_pypdfium2, prepare_env, read_fn
  8. from mineru.data.data_reader_writer import FileBasedDataWriter
  9. from mineru.utils.draw_bbox import draw_layout_bbox, draw_span_bbox
  10. from mineru.utils.enum_class import MakeMode
  11. from mineru.backend.vlm.vlm_analyze import doc_analyze as vlm_doc_analyze
  12. from mineru.backend.pipeline.pipeline_analyze import doc_analyze as pipeline_doc_analyze
  13. from mineru.backend.pipeline.pipeline_middle_json_mkcontent import union_make as pipeline_union_make
  14. from mineru.backend.pipeline.model_json_to_middle_json import result_to_middle_json as pipeline_result_to_middle_json
  15. from mineru.backend.vlm.vlm_middle_json_mkcontent import union_make as vlm_union_make
  16. from mineru.utils.guess_suffix_or_lang import guess_suffix_by_path
  17. def do_parse(
  18. output_dir, # Output directory for storing parsing results
  19. pdf_file_names: list[str], # List of PDF file names to be parsed
  20. pdf_bytes_list: list[bytes], # List of PDF bytes to be parsed
  21. p_lang_list: list[str], # List of languages for each PDF, default is 'ch' (Chinese)
  22. backend="pipeline", # The backend for parsing PDF, default is 'pipeline'
  23. parse_method="auto", # The method for parsing PDF, default is 'auto'
  24. formula_enable=True, # Enable formula parsing
  25. table_enable=True, # Enable table parsing
  26. server_url=None, # Server URL for vlm-http-client backend
  27. f_draw_layout_bbox=True, # Whether to draw layout bounding boxes
  28. f_draw_span_bbox=True, # Whether to draw span bounding boxes
  29. f_dump_md=True, # Whether to dump markdown files
  30. f_dump_middle_json=True, # Whether to dump middle JSON files
  31. f_dump_model_output=True, # Whether to dump model output files
  32. f_dump_orig_pdf=True, # Whether to dump original PDF files
  33. f_dump_content_list=True, # Whether to dump content list files
  34. f_make_md_mode=MakeMode.MM_MD, # The mode for making markdown content, default is MM_MD
  35. start_page_id=0, # Start page ID for parsing, default is 0
  36. end_page_id=None, # End page ID for parsing, default is None (parse all pages until the end of the document)
  37. ):
  38. if backend == "pipeline":
  39. for idx, pdf_bytes in enumerate(pdf_bytes_list):
  40. new_pdf_bytes = convert_pdf_bytes_to_bytes_by_pypdfium2(pdf_bytes, start_page_id, end_page_id)
  41. pdf_bytes_list[idx] = new_pdf_bytes
  42. infer_results, all_image_lists, all_pdf_docs, lang_list, ocr_enabled_list = pipeline_doc_analyze(pdf_bytes_list, p_lang_list, parse_method=parse_method, formula_enable=formula_enable,table_enable=table_enable)
  43. for idx, model_list in enumerate(infer_results):
  44. model_json = copy.deepcopy(model_list)
  45. pdf_file_name = pdf_file_names[idx]
  46. local_image_dir, local_md_dir = prepare_env(output_dir, pdf_file_name, parse_method)
  47. image_writer, md_writer = FileBasedDataWriter(local_image_dir), FileBasedDataWriter(local_md_dir)
  48. images_list = all_image_lists[idx]
  49. pdf_doc = all_pdf_docs[idx]
  50. _lang = lang_list[idx]
  51. _ocr_enable = ocr_enabled_list[idx]
  52. middle_json = pipeline_result_to_middle_json(model_list, images_list, pdf_doc, image_writer, _lang, _ocr_enable, formula_enable)
  53. pdf_info = middle_json["pdf_info"]
  54. pdf_bytes = pdf_bytes_list[idx]
  55. _process_output(
  56. pdf_info, pdf_bytes, pdf_file_name, local_md_dir, local_image_dir,
  57. md_writer, f_draw_layout_bbox, f_draw_span_bbox, f_dump_orig_pdf,
  58. f_dump_md, f_dump_content_list, f_dump_middle_json, f_dump_model_output,
  59. f_make_md_mode, middle_json, model_json, is_pipeline=True
  60. )
  61. else:
  62. if backend.startswith("vlm-"):
  63. backend = backend[4:]
  64. f_draw_span_bbox = False
  65. parse_method = "vlm"
  66. for idx, pdf_bytes in enumerate(pdf_bytes_list):
  67. pdf_file_name = pdf_file_names[idx]
  68. pdf_bytes = convert_pdf_bytes_to_bytes_by_pypdfium2(pdf_bytes, start_page_id, end_page_id)
  69. local_image_dir, local_md_dir = prepare_env(output_dir, pdf_file_name, parse_method)
  70. image_writer, md_writer = FileBasedDataWriter(local_image_dir), FileBasedDataWriter(local_md_dir)
  71. middle_json, infer_result = vlm_doc_analyze(pdf_bytes, image_writer=image_writer, backend=backend, server_url=server_url)
  72. pdf_info = middle_json["pdf_info"]
  73. _process_output(
  74. pdf_info, pdf_bytes, pdf_file_name, local_md_dir, local_image_dir,
  75. md_writer, f_draw_layout_bbox, f_draw_span_bbox, f_dump_orig_pdf,
  76. f_dump_md, f_dump_content_list, f_dump_middle_json, f_dump_model_output,
  77. f_make_md_mode, middle_json, infer_result, is_pipeline=False
  78. )
  79. def _process_output(
  80. pdf_info,
  81. pdf_bytes,
  82. pdf_file_name,
  83. local_md_dir,
  84. local_image_dir,
  85. md_writer,
  86. f_draw_layout_bbox,
  87. f_draw_span_bbox,
  88. f_dump_orig_pdf,
  89. f_dump_md,
  90. f_dump_content_list,
  91. f_dump_middle_json,
  92. f_dump_model_output,
  93. f_make_md_mode,
  94. middle_json,
  95. model_output=None,
  96. is_pipeline=True
  97. ):
  98. """处理输出文件"""
  99. if f_draw_layout_bbox:
  100. draw_layout_bbox(pdf_info, pdf_bytes, local_md_dir, f"{pdf_file_name}_layout.pdf")
  101. if f_draw_span_bbox:
  102. draw_span_bbox(pdf_info, pdf_bytes, local_md_dir, f"{pdf_file_name}_span.pdf")
  103. if f_dump_orig_pdf:
  104. md_writer.write(
  105. f"{pdf_file_name}_origin.pdf",
  106. pdf_bytes,
  107. )
  108. image_dir = str(os.path.basename(local_image_dir))
  109. if f_dump_md:
  110. make_func = pipeline_union_make if is_pipeline else vlm_union_make
  111. md_content_str = make_func(pdf_info, f_make_md_mode, image_dir)
  112. md_writer.write_string(
  113. f"{pdf_file_name}.md",
  114. md_content_str,
  115. )
  116. if f_dump_content_list:
  117. make_func = pipeline_union_make if is_pipeline else vlm_union_make
  118. content_list = make_func(pdf_info, MakeMode.CONTENT_LIST, image_dir)
  119. md_writer.write_string(
  120. f"{pdf_file_name}_content_list.json",
  121. json.dumps(content_list, ensure_ascii=False, indent=4),
  122. )
  123. if f_dump_middle_json:
  124. md_writer.write_string(
  125. f"{pdf_file_name}_middle.json",
  126. json.dumps(middle_json, ensure_ascii=False, indent=4),
  127. )
  128. if f_dump_model_output:
  129. md_writer.write_string(
  130. f"{pdf_file_name}_model.json",
  131. json.dumps(model_output, ensure_ascii=False, indent=4),
  132. )
  133. logger.info(f"local output dir is {local_md_dir}")
  134. def parse_doc(
  135. path_list: list[Path],
  136. output_dir,
  137. lang="ch",
  138. backend="pipeline",
  139. method="auto",
  140. server_url=None,
  141. start_page_id=0,
  142. end_page_id=None
  143. ):
  144. """
  145. Parameter description:
  146. path_list: List of document paths to be parsed, can be PDF or image files.
  147. output_dir: Output directory for storing parsing results.
  148. lang: Language option, default is 'ch', optional values include['ch', 'ch_server', 'ch_lite', 'en', 'korean', 'japan', 'chinese_cht', 'ta', 'te', 'ka']。
  149. Input the languages in the pdf (if known) to improve OCR accuracy. Optional.
  150. Adapted only for the case where the backend is set to "pipeline"
  151. backend: the backend for parsing pdf:
  152. pipeline: More general.
  153. vlm-transformers: More general.
  154. vlm-vllm-engine: Faster(engine).
  155. vlm-http-client: Faster(client).
  156. without method specified, pipeline will be used by default.
  157. method: the method for parsing pdf:
  158. auto: Automatically determine the method based on the file type.
  159. txt: Use text extraction method.
  160. ocr: Use OCR method for image-based PDFs.
  161. Without method specified, 'auto' will be used by default.
  162. Adapted only for the case where the backend is set to "pipeline".
  163. server_url: When the backend is `http-client`, you need to specify the server_url, for example:`http://127.0.0.1:30000`
  164. start_page_id: Start page ID for parsing, default is 0
  165. end_page_id: End page ID for parsing, default is None (parse all pages until the end of the document)
  166. """
  167. try:
  168. file_name_list = []
  169. pdf_bytes_list = []
  170. lang_list = []
  171. for path in path_list:
  172. file_name = str(Path(path).stem)
  173. pdf_bytes = read_fn(path)
  174. file_name_list.append(file_name)
  175. pdf_bytes_list.append(pdf_bytes)
  176. lang_list.append(lang)
  177. do_parse(
  178. output_dir=output_dir,
  179. pdf_file_names=file_name_list,
  180. pdf_bytes_list=pdf_bytes_list,
  181. p_lang_list=lang_list,
  182. backend=backend,
  183. parse_method=method,
  184. server_url=server_url,
  185. start_page_id=start_page_id,
  186. end_page_id=end_page_id
  187. )
  188. except Exception as e:
  189. logger.exception(e)
  190. if __name__ == '__main__':
  191. # args
  192. __dir__ = os.path.dirname(os.path.abspath(__file__))
  193. pdf_files_dir = os.path.join(__dir__, "pdfs")
  194. output_dir = os.path.join(__dir__, "output")
  195. pdf_suffixes = ["pdf"]
  196. image_suffixes = ["png", "jpeg", "jp2", "webp", "gif", "bmp", "jpg"]
  197. doc_path_list = []
  198. for doc_path in Path(pdf_files_dir).glob('*'):
  199. if guess_suffix_by_path(doc_path) in pdf_suffixes + image_suffixes:
  200. doc_path_list.append(doc_path)
  201. """如果您由于网络问题无法下载模型,可以设置环境变量MINERU_MODEL_SOURCE为modelscope使用免代理仓库下载模型"""
  202. # os.environ['MINERU_MODEL_SOURCE'] = "modelscope"
  203. """Use pipeline mode if your environment does not support VLM"""
  204. parse_doc(doc_path_list, output_dir, backend="pipeline")
  205. """To enable VLM mode, change the backend to 'vlm-xxx'"""
  206. # parse_doc(doc_path_list, output_dir, backend="vlm-transformers") # more general.
  207. # parse_doc(doc_path_list, output_dir, backend="vlm-vllm-engine") # faster(engine).
  208. # parse_doc(doc_path_list, output_dir, backend="vlm-http-client", server_url="http://127.0.0.1:30000") # faster(client).