pdf_parse_by_ocr.py 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. import json
  2. import os
  3. import time
  4. from loguru import logger
  5. from magic_pdf.libs.draw_bbox import draw_layout_bbox, draw_text_bbox
  6. from magic_pdf.libs.commons import (
  7. read_file,
  8. join_path,
  9. fitz,
  10. get_img_s3_client,
  11. get_delta_time,
  12. get_docx_model_output,
  13. )
  14. from magic_pdf.libs.coordinate_transform import get_scale_ratio
  15. from magic_pdf.libs.safe_filename import sanitize_filename
  16. from magic_pdf.pre_proc.detect_footer_by_model import parse_footers
  17. from magic_pdf.pre_proc.detect_footnote import parse_footnotes_by_model
  18. from magic_pdf.pre_proc.detect_header import parse_headers
  19. from magic_pdf.pre_proc.detect_page_number import parse_pageNos
  20. from magic_pdf.pre_proc.ocr_cut_image import cut_image_and_table
  21. from magic_pdf.pre_proc.ocr_detect_layout import layout_detect
  22. from magic_pdf.pre_proc.ocr_dict_merge import (
  23. remove_overlaps_min_spans,
  24. merge_spans_to_line_by_layout,
  25. modify_y_axis
  26. )
  27. from magic_pdf.pre_proc.ocr_remove_spans import remove_spans_by_bboxes
  28. from magic_pdf.pre_proc.remove_bbox_overlap import remove_overlap_between_bbox
  29. def construct_page_component(blocks, layout_bboxes, page_id, page_w, page_h, layout_tree):
  30. return_dict = {
  31. 'preproc_blocks': blocks,
  32. 'layout_bboxes': layout_bboxes,
  33. 'page_idx': page_id,
  34. 'page_size': [page_w, page_h],
  35. '_layout_tree': layout_tree,
  36. }
  37. return return_dict
  38. def parse_pdf_by_ocr(
  39. pdf_path,
  40. s3_pdf_profile,
  41. pdf_model_output,
  42. save_path,
  43. book_name,
  44. pdf_model_profile=None,
  45. image_s3_config=None,
  46. start_page_id=0,
  47. end_page_id=None,
  48. debug_mode=False,
  49. ):
  50. pdf_bytes = read_file(pdf_path, s3_pdf_profile)
  51. save_tmp_path = os.path.join(os.path.dirname(__file__), "../..", "tmp", "unittest")
  52. book_name = sanitize_filename(book_name)
  53. md_bookname_save_path = ""
  54. if debug_mode:
  55. save_path = join_path(save_tmp_path, "md")
  56. pdf_local_path = join_path(save_tmp_path, "download-pdfs", book_name)
  57. if not os.path.exists(os.path.dirname(pdf_local_path)):
  58. # 如果目录不存在,创建它
  59. os.makedirs(os.path.dirname(pdf_local_path))
  60. md_bookname_save_path = join_path(save_tmp_path, "md", book_name)
  61. if not os.path.exists(md_bookname_save_path):
  62. # 如果目录不存在,创建它
  63. os.makedirs(md_bookname_save_path)
  64. with open(pdf_local_path + ".pdf", "wb") as pdf_file:
  65. pdf_file.write(pdf_bytes)
  66. pdf_docs = fitz.open("pdf", pdf_bytes)
  67. # 初始化空的pdf_info_dict
  68. pdf_info_dict = {}
  69. img_s3_client = get_img_s3_client(save_path, image_s3_config)
  70. start_time = time.time()
  71. remove_bboxes = []
  72. end_page_id = end_page_id if end_page_id else len(pdf_docs) - 1
  73. for page_id in range(start_page_id, end_page_id + 1):
  74. # 获取当前页的page对象
  75. page = pdf_docs[page_id]
  76. # 获取当前页的宽高
  77. page_w = page.rect.width
  78. page_h = page.rect.height
  79. if debug_mode:
  80. time_now = time.time()
  81. logger.info(
  82. f"page_id: {page_id}, last_page_cost_time: {get_delta_time(start_time)}"
  83. )
  84. start_time = time_now
  85. # 获取当前页的模型数据
  86. ocr_page_info = get_docx_model_output(
  87. pdf_model_output, pdf_model_profile, page_id
  88. )
  89. """从json中获取每页的页码、页眉、页脚的bbox"""
  90. page_no_bboxes = parse_pageNos(page_id, page, ocr_page_info)
  91. header_bboxes = parse_headers(page_id, page, ocr_page_info)
  92. footer_bboxes = parse_footers(page_id, page, ocr_page_info)
  93. footnote_bboxes = parse_footnotes_by_model(
  94. page_id, page, ocr_page_info, md_bookname_save_path, debug_mode=debug_mode
  95. )
  96. # 构建需要remove的bbox列表
  97. need_remove_spans_bboxes = []
  98. need_remove_spans_bboxes.extend(page_no_bboxes)
  99. need_remove_spans_bboxes.extend(header_bboxes)
  100. need_remove_spans_bboxes.extend(footer_bboxes)
  101. need_remove_spans_bboxes.extend(footnote_bboxes)
  102. layout_dets = ocr_page_info["layout_dets"]
  103. spans = []
  104. # 计算模型坐标和pymu坐标的缩放比例
  105. horizontal_scale_ratio, vertical_scale_ratio = get_scale_ratio(
  106. ocr_page_info, page
  107. )
  108. for layout_det in layout_dets:
  109. category_id = layout_det["category_id"]
  110. allow_category_id_list = [1, 7, 13, 14, 15]
  111. if category_id in allow_category_id_list:
  112. x0, y0, _, _, x1, y1, _, _ = layout_det["poly"]
  113. bbox = [
  114. int(x0 / horizontal_scale_ratio),
  115. int(y0 / vertical_scale_ratio),
  116. int(x1 / horizontal_scale_ratio),
  117. int(y1 / vertical_scale_ratio),
  118. ]
  119. """要删除的"""
  120. # 3: 'header', # 页眉
  121. # 4: 'page number', # 页码
  122. # 5: 'footnote', # 脚注
  123. # 6: 'footer', # 页脚
  124. """当成span拼接的"""
  125. # 1: 'image', # 图片
  126. # 7: 'table', # 表格
  127. # 13: 'inline_equation', # 行内公式
  128. # 14: 'displayed_equation', # 行间公式
  129. # 15: 'text', # ocr识别文本
  130. """layout信息"""
  131. # 11: 'full column', # 单栏
  132. # 12: 'sub column', # 多栏
  133. span = {
  134. "bbox": bbox,
  135. }
  136. if category_id == 1:
  137. span["type"] = "image"
  138. elif category_id == 7:
  139. span["type"] = "table"
  140. elif category_id == 13:
  141. span["content"] = layout_det["latex"]
  142. span["type"] = "inline_equation"
  143. elif category_id == 14:
  144. span["content"] = layout_det["latex"]
  145. span["type"] = "displayed_equation"
  146. elif category_id == 15:
  147. span["content"] = layout_det["text"]
  148. span["type"] = "text"
  149. # print(span)
  150. spans.append(span)
  151. else:
  152. continue
  153. # 删除重叠spans中较小的那些
  154. spans = remove_overlaps_min_spans(spans)
  155. # 对tpye=["displayed_equation", "image", "table"]进行额外处理,如果左边有字的话,将该span的bbox中y0调整低于文字的y0
  156. spans = modify_y_axis(spans)
  157. # 删除remove_span_block_bboxes中的bbox
  158. spans = remove_spans_by_bboxes(spans, need_remove_spans_bboxes)
  159. # 对image和table截图
  160. spans = cut_image_and_table(spans, page, page_id, book_name, save_path)
  161. # 行内公式调整, 高度调整至与同行文字高度一致(优先左侧, 其次右侧)
  162. # 模型识别错误的行间公式, type类型转换成行内公式
  163. # bbox去除粘连
  164. spans = remove_overlap_between_bbox(spans)
  165. # 对tpye=["displayed_equation", "image", "table"]进行额外处理,如果左边有字的话,将该span的bbox中y0调整至不高于文字的y0
  166. # 从ocr_page_info中解析layout信息(按自然阅读方向排序,并修复重叠和交错的bad case)
  167. layout_bboxes, layout_tree = layout_detect(ocr_page_info['subfield_dets'], page, ocr_page_info)
  168. # 将spans合并成line(在layout内,从上到下,从左到右)
  169. lines = merge_spans_to_line_by_layout(spans, layout_bboxes)
  170. # 目前不做block拼接,先做个结构,每个block中只有一个line,block的bbox就是line的bbox
  171. blocks = []
  172. for line in lines:
  173. blocks.append(
  174. {
  175. "bbox": line["bbox"],
  176. "lines": [line],
  177. }
  178. )
  179. # 构造pdf_info_dict
  180. page_info = construct_page_component(blocks, layout_bboxes, page_id, page_w, page_h, layout_tree)
  181. pdf_info_dict[f"page_{page_id}"] = page_info
  182. # 在测试时,保存调试信息
  183. if debug_mode:
  184. params_file_save_path = join_path(
  185. save_tmp_path, "md", book_name, "preproc_out.json"
  186. )
  187. with open(params_file_save_path, "w", encoding="utf-8") as f:
  188. json.dump(pdf_info_dict, f, ensure_ascii=False, indent=4)
  189. # drow_bbox
  190. draw_layout_bbox(pdf_info_dict, pdf_path, md_bookname_save_path)
  191. draw_text_bbox(pdf_info_dict, pdf_path, md_bookname_save_path)
  192. return pdf_info_dict