pdf_image_tools.py 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  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, return_path, imageWriter, 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])}"
  17. # 老版本返回不带bucket的路径
  18. img_path = join_path(return_path, filename) if return_path is not None else None
  19. # 新版本生成平铺路径
  20. img_hash256_path = f"{compute_sha256(img_path)}.jpg"
  21. # 将坐标转换为fitz.Rect对象
  22. rect = fitz.Rect(*bbox)
  23. # 配置缩放倍数为3倍
  24. zoom = fitz.Matrix(3, 3)
  25. # 截取图片
  26. pix = page.get_pixmap(clip=rect, matrix=zoom)
  27. byte_data = pix.tobytes(output='jpeg', jpg_quality=95)
  28. imageWriter.write(data=byte_data, path=img_hash256_path, mode="binary")
  29. return img_hash256_path
  30. def save_images_by_bboxes(book_name: str, page_num: int, page: fitz.Page, save_path: str,
  31. image_bboxes: list, images_overlap_backup:list, table_bboxes: list, equation_inline_bboxes: list,
  32. equation_interline_bboxes: list, img_s3_client) -> dict:
  33. """
  34. 返回一个dict, key为bbox, 值是图片地址
  35. """
  36. image_info = []
  37. image_backup_info = []
  38. table_info = []
  39. inline_eq_info = []
  40. interline_eq_info = []
  41. # 图片的保存路径组成是这样的: {s3_or_local_path}/{book_name}/{images|tables|equations}/{page_num}_{bbox[0]}_{bbox[1]}_{bbox[2]}_{bbox[3]}.jpg
  42. s3_return_image_path = join_path(book_name, "images")
  43. image_save_path = join_path(save_path, s3_return_image_path)
  44. s3_return_table_path = join_path(book_name, "tables")
  45. table_save_path = join_path(save_path, s3_return_table_path)
  46. s3_return_equations_inline_path = join_path(book_name, "equations_inline")
  47. equation_inline_save_path = join_path(save_path, s3_return_equations_inline_path)
  48. s3_return_equation_interline_path = join_path(book_name, "equation_interline")
  49. equation_interline_save_path = join_path(save_path, s3_return_equation_interline_path)
  50. for bbox in image_bboxes:
  51. if any([bbox[0]>=bbox[2], bbox[1]>=bbox[3]]):
  52. logger.warning(f"image_bboxes: 错误的box, {bbox}")
  53. continue
  54. image_path = cut_image(bbox, page_num, page, image_save_path, s3_return_image_path, img_s3_client)
  55. image_info.append({"bbox": bbox, "image_path": image_path})
  56. for bbox in images_overlap_backup:
  57. if any([bbox[0]>=bbox[2], bbox[1]>=bbox[3]]):
  58. logger.warning(f"images_overlap_backup: 错误的box, {bbox}")
  59. continue
  60. image_path = cut_image(bbox, page_num, page, image_save_path, s3_return_image_path, img_s3_client)
  61. image_backup_info.append({"bbox": bbox, "image_path": image_path})
  62. for bbox in table_bboxes:
  63. if any([bbox[0]>=bbox[2], bbox[1]>=bbox[3]]):
  64. logger.warning(f"table_bboxes: 错误的box, {bbox}")
  65. continue
  66. image_path = cut_image(bbox, page_num, page, table_save_path, s3_return_table_path, img_s3_client)
  67. table_info.append({"bbox": bbox, "image_path": image_path})
  68. for bbox in equation_inline_bboxes:
  69. if any([bbox[0]>=bbox[2], bbox[1]>=bbox[3]]):
  70. logger.warning(f"equation_inline_bboxes: 错误的box, {bbox}")
  71. continue
  72. image_path = cut_image(bbox[:4], page_num, page, equation_inline_save_path, s3_return_equations_inline_path, img_s3_client, upload_switch=False)
  73. inline_eq_info.append({'bbox':bbox[:4], "image_path":image_path, "latex_text":bbox[4]})
  74. for bbox in equation_interline_bboxes:
  75. if any([bbox[0]>=bbox[2], bbox[1]>=bbox[3]]):
  76. logger.warning(f"equation_interline_bboxes: 错误的box, {bbox}")
  77. continue
  78. image_path = cut_image(bbox[:4], page_num, page, equation_interline_save_path, s3_return_equation_interline_path, img_s3_client, upload_switch=False)
  79. interline_eq_info.append({"bbox":bbox[:4], "image_path":image_path, "latex_text":bbox[4]})
  80. return image_info, image_backup_info, table_info, inline_eq_info, interline_eq_info