pdf_image_tools.py 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. import os
  2. from pathlib import Path
  3. from typing import Tuple
  4. import io
  5. # from app.common.s3 import get_s3_client
  6. from magic_pdf.libs.commons import fitz
  7. from loguru import logger
  8. from magic_pdf.libs.commons import parse_bucket_key, join_path
  9. from magic_pdf.libs.hash_utils import compute_sha256
  10. def cut_image(bbox: Tuple, page_num: int, page: fitz.Page, save_parent_path: str, s3_return_path=None, img_s3_client=None, upload_switch=True):
  11. """
  12. 从第page_num页的page中,根据bbox进行裁剪出一张jpg图片,返回图片路径
  13. save_path:需要同时支持s3和本地, 图片存放在save_path下,文件名是: {page_num}_{bbox[0]}_{bbox[1]}_{bbox[2]}_{bbox[3]}.jpg , bbox内数字取整。
  14. """
  15. # 拼接文件名
  16. filename = f"{page_num}_{int(bbox[0])}_{int(bbox[1])}_{int(bbox[2])}_{int(bbox[3])}.jpg"
  17. # 老版本返回不带bucket的路径
  18. s3_img_path = join_path(s3_return_path, filename) if s3_return_path is not None else None
  19. # 新版本生成s3的平铺路径
  20. s3_img_hash256_path = f"{compute_sha256(s3_img_path)}.jpg"
  21. # 打印图片文件名
  22. # print(f"Saved {image_save_path}")
  23. #检查坐标
  24. # x_check = int(bbox[2]) - int(bbox[0])
  25. # y_check = int(bbox[3]) - int(bbox[1])
  26. # if x_check <= 0 or y_check <= 0:
  27. #
  28. # if image_save_path.startswith("s3://"):
  29. # logger.exception(f"传入图片坐标有误,x1<x0或y1<y0,{s3_img_path}")
  30. # return s3_img_path
  31. # else:
  32. # logger.exception(f"传入图片坐标有误,x1<x0或y1<y0,{image_save_path}")
  33. # return image_save_path
  34. # 将坐标转换为fitz.Rect对象
  35. rect = fitz.Rect(*bbox)
  36. # 配置缩放倍数为3倍
  37. zoom = fitz.Matrix(3, 3)
  38. # 截取图片
  39. pix = page.get_pixmap(clip=rect, matrix=zoom)
  40. if save_parent_path.startswith("s3://"):
  41. if not upload_switch:
  42. pass
  43. else:
  44. """图片保存到s3"""
  45. # 从save_parent_path获取bucket_name
  46. bucket_name, bucket_key = parse_bucket_key(save_parent_path)
  47. # 平铺路径赋值给bucket_key
  48. bucket_key = s3_img_hash256_path
  49. # 将字节流上传到s3
  50. byte_data = pix.tobytes(output='jpeg', jpg_quality=95)
  51. file_obj = io.BytesIO(byte_data)
  52. if img_s3_client is not None:
  53. img_s3_client.upload_fileobj(file_obj, bucket_name, bucket_key)
  54. # 每个图片上传任务都创建一个新的client
  55. # img_s3_client_once = get_s3_client(image_save_path)
  56. # img_s3_client_once.upload_fileobj(file_obj, bucket_name, bucket_key)
  57. else:
  58. logger.exception("must input img_s3_client")
  59. # return s3_img_path # 早期版本要求返回不带bucket的路径
  60. s3_image_save_path = f"s3://{bucket_name}/{s3_img_hash256_path}" # 新版本返回平铺的s3路径
  61. return s3_image_save_path
  62. else:
  63. # 保存图片到本地
  64. # 先检查一下image_save_path的父目录是否存在,如果不存在,就创建
  65. local_image_save_path = join_path(save_parent_path, filename)
  66. parent_dir = os.path.dirname(local_image_save_path)
  67. if not os.path.exists(parent_dir):
  68. os.makedirs(parent_dir)
  69. pix.save(local_image_save_path, jpg_quality=95)
  70. # 为了直接能在markdown里看,这里把地址改为相对于mardown的地址
  71. pth = Path(local_image_save_path)
  72. local_image_save_path = f"{pth.parent.name}/{pth.name}"
  73. return local_image_save_path
  74. def save_images_by_bboxes(book_name: str, page_num: int, page: fitz.Page, save_path: str,
  75. image_bboxes: list, images_overlap_backup:list, table_bboxes: list, equation_inline_bboxes: list,
  76. equation_interline_bboxes: list, img_s3_client) -> dict:
  77. """
  78. 返回一个dict, key为bbox, 值是图片地址
  79. """
  80. image_info = []
  81. image_backup_info = []
  82. table_info = []
  83. inline_eq_info = []
  84. interline_eq_info = []
  85. # 图片的保存路径组成是这样的: {s3_or_local_path}/{book_name}/{images|tables|equations}/{page_num}_{bbox[0]}_{bbox[1]}_{bbox[2]}_{bbox[3]}.jpg
  86. s3_return_image_path = join_path(book_name, "images")
  87. image_save_path = join_path(save_path, s3_return_image_path)
  88. s3_return_table_path = join_path(book_name, "tables")
  89. table_save_path = join_path(save_path, s3_return_table_path)
  90. s3_return_equations_inline_path = join_path(book_name, "equations_inline")
  91. equation_inline_save_path = join_path(save_path, s3_return_equations_inline_path)
  92. s3_return_equation_interline_path = join_path(book_name, "equation_interline")
  93. equation_interline_save_path = join_path(save_path, s3_return_equation_interline_path)
  94. for bbox in image_bboxes:
  95. if any([bbox[0]>=bbox[2], bbox[1]>=bbox[3]]):
  96. logger.warning(f"image_bboxes: 错误的box, {bbox}")
  97. continue
  98. image_path = cut_image(bbox, page_num, page, image_save_path, s3_return_image_path, img_s3_client)
  99. image_info.append({"bbox": bbox, "image_path": image_path})
  100. for bbox in images_overlap_backup:
  101. if any([bbox[0]>=bbox[2], bbox[1]>=bbox[3]]):
  102. logger.warning(f"images_overlap_backup: 错误的box, {bbox}")
  103. continue
  104. image_path = cut_image(bbox, page_num, page, image_save_path, s3_return_image_path, img_s3_client)
  105. image_backup_info.append({"bbox": bbox, "image_path": image_path})
  106. for bbox in table_bboxes:
  107. if any([bbox[0]>=bbox[2], bbox[1]>=bbox[3]]):
  108. logger.warning(f"table_bboxes: 错误的box, {bbox}")
  109. continue
  110. image_path = cut_image(bbox, page_num, page, table_save_path, s3_return_table_path, img_s3_client)
  111. table_info.append({"bbox": bbox, "image_path": image_path})
  112. for bbox in equation_inline_bboxes:
  113. if any([bbox[0]>=bbox[2], bbox[1]>=bbox[3]]):
  114. logger.warning(f"equation_inline_bboxes: 错误的box, {bbox}")
  115. continue
  116. image_path = cut_image(bbox[:4], page_num, page, equation_inline_save_path, s3_return_equations_inline_path, img_s3_client, upload_switch=False)
  117. inline_eq_info.append({'bbox':bbox[:4], "image_path":image_path, "latex_text":bbox[4]})
  118. for bbox in equation_interline_bboxes:
  119. if any([bbox[0]>=bbox[2], bbox[1]>=bbox[3]]):
  120. logger.warning(f"equation_interline_bboxes: 错误的box, {bbox}")
  121. continue
  122. image_path = cut_image(bbox[:4], page_num, page, equation_interline_save_path, s3_return_equation_interline_path, img_s3_client, upload_switch=False)
  123. interline_eq_info.append({"bbox":bbox[:4], "image_path":image_path, "latex_text":bbox[4]})
  124. return image_info, image_backup_info, table_info, inline_eq_info, interline_eq_info