2024-03-29 10:28:05 +00:00
|
|
|
|
# coding=utf-8
|
|
|
|
|
|
"""
|
|
|
|
|
|
@project: maxkb
|
|
|
|
|
|
@Author:虎
|
|
|
|
|
|
@file: text_split_handle.py
|
|
|
|
|
|
@date:2024/3/27 18:19
|
|
|
|
|
|
@desc:
|
|
|
|
|
|
"""
|
|
|
|
|
|
import re
|
|
|
|
|
|
from typing import List
|
|
|
|
|
|
|
2024-08-20 08:17:42 +00:00
|
|
|
|
import fitz
|
2024-08-15 07:45:16 +00:00
|
|
|
|
import os
|
|
|
|
|
|
import tempfile
|
2024-08-16 04:57:06 +00:00
|
|
|
|
import logging
|
2024-08-15 07:45:16 +00:00
|
|
|
|
from langchain_community.document_loaders import PyPDFLoader
|
2024-03-29 10:28:05 +00:00
|
|
|
|
|
|
|
|
|
|
from common.handle.base_split_handle import BaseSplitHandle
|
|
|
|
|
|
from common.util.split_model import SplitModel
|
|
|
|
|
|
|
2024-08-16 04:57:06 +00:00
|
|
|
|
import time
|
|
|
|
|
|
|
2024-04-10 06:16:56 +00:00
|
|
|
|
default_pattern_list = [re.compile('(?<=^)# .*|(?<=\\n)# .*'),
|
|
|
|
|
|
re.compile('(?<=\\n)(?<!#)## (?!#).*|(?<=^)(?<!#)## (?!#).*'),
|
|
|
|
|
|
re.compile("(?<=\\n)(?<!#)### (?!#).*|(?<=^)(?<!#)### (?!#).*"),
|
|
|
|
|
|
re.compile("(?<=\\n)(?<!#)#### (?!#).*|(?<=^)(?<!#)#### (?!#).*"),
|
|
|
|
|
|
re.compile("(?<=\\n)(?<!#)##### (?!#).*|(?<=^)(?<!#)##### (?!#).*"),
|
|
|
|
|
|
re.compile("(?<=\\n)(?<!#)###### (?!#).*|(?<=^)(?<!#)###### (?!#).*"),
|
|
|
|
|
|
re.compile("(?<!\n)\n\n+")]
|
2024-03-29 10:28:05 +00:00
|
|
|
|
|
2024-08-16 04:57:06 +00:00
|
|
|
|
max_kb = logging.getLogger("max_kb")
|
2024-03-29 10:28:05 +00:00
|
|
|
|
|
2024-08-22 02:39:11 +00:00
|
|
|
|
|
2024-03-29 10:28:05 +00:00
|
|
|
|
class PdfSplitHandle(BaseSplitHandle):
|
2024-08-15 07:45:16 +00:00
|
|
|
|
def handle(self, file, pattern_list: List, with_filter: bool, limit: int, get_buffer, save_image):
|
|
|
|
|
|
with tempfile.NamedTemporaryFile(delete=False) as temp_file:
|
|
|
|
|
|
# 将上传的文件保存到临时文件中
|
|
|
|
|
|
for chunk in file.chunks():
|
|
|
|
|
|
temp_file.write(chunk)
|
|
|
|
|
|
# 获取临时文件的路径
|
|
|
|
|
|
temp_file_path = temp_file.name
|
|
|
|
|
|
|
2024-08-20 08:17:42 +00:00
|
|
|
|
pdf_document = fitz.open(temp_file_path)
|
2024-03-29 10:28:05 +00:00
|
|
|
|
try:
|
2024-08-16 04:57:06 +00:00
|
|
|
|
content = ""
|
2024-08-20 08:17:42 +00:00
|
|
|
|
for page_num in range(len(pdf_document)):
|
2024-08-16 04:57:06 +00:00
|
|
|
|
start_time = time.time()
|
2024-08-20 08:17:42 +00:00
|
|
|
|
page = pdf_document.load_page(page_num)
|
|
|
|
|
|
text = page.get_text()
|
2024-08-16 02:36:27 +00:00
|
|
|
|
|
2024-08-16 04:57:06 +00:00
|
|
|
|
if text and text.strip(): # 如果页面中有文本内容
|
|
|
|
|
|
page_content = text
|
|
|
|
|
|
else:
|
|
|
|
|
|
try:
|
2024-08-20 08:17:42 +00:00
|
|
|
|
new_doc = fitz.open()
|
|
|
|
|
|
new_doc.insert_pdf(pdf_document, from_page=page_num, to_page=page_num)
|
|
|
|
|
|
page_num_pdf = tempfile.gettempdir() + f"/{file.name}_{page_num}.pdf"
|
|
|
|
|
|
new_doc.save(page_num_pdf)
|
|
|
|
|
|
new_doc.close()
|
|
|
|
|
|
|
|
|
|
|
|
loader = PyPDFLoader(page_num_pdf, extract_images=True)
|
2024-08-16 04:57:06 +00:00
|
|
|
|
page_content = "\n" + loader.load()[0].page_content
|
2024-08-22 02:39:11 +00:00
|
|
|
|
except NotImplementedError as e:
|
|
|
|
|
|
# 文件格式不支持,直接退出
|
|
|
|
|
|
raise e
|
|
|
|
|
|
except BaseException as e:
|
|
|
|
|
|
# 当页出错继续进行下一页,防止一个页面出错导致整个文件解析失败
|
|
|
|
|
|
max_kb.error(f"File: {file.name}, Page: {page_num + 1}, error: {e}")
|
|
|
|
|
|
continue
|
2024-08-16 04:57:06 +00:00
|
|
|
|
finally:
|
2024-08-20 08:17:42 +00:00
|
|
|
|
os.remove(page_num_pdf)
|
2024-08-16 04:57:06 +00:00
|
|
|
|
|
|
|
|
|
|
content += page_content
|
2024-08-16 02:36:27 +00:00
|
|
|
|
|
2024-08-16 04:57:06 +00:00
|
|
|
|
elapsed_time = time.time() - start_time
|
|
|
|
|
|
# todo 实现进度条代替下面的普通输出
|
|
|
|
|
|
max_kb.debug(f"File: {file.name}, Page: {page_num + 1}, Time : {elapsed_time: .3f}s, content-length: {len(page_content)}")
|
2024-03-29 10:28:05 +00:00
|
|
|
|
if pattern_list is not None and len(pattern_list) > 0:
|
|
|
|
|
|
split_model = SplitModel(pattern_list, with_filter, limit)
|
|
|
|
|
|
else:
|
|
|
|
|
|
split_model = SplitModel(default_pattern_list, with_filter=with_filter, limit=limit)
|
2024-08-16 04:57:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
2024-03-29 10:28:05 +00:00
|
|
|
|
except BaseException as e:
|
2024-08-20 14:03:58 +00:00
|
|
|
|
max_kb.error(f"File: {file.name}, error: {e}")
|
2024-03-29 10:28:05 +00:00
|
|
|
|
return {'name': file.name,
|
|
|
|
|
|
'content': []}
|
2024-08-15 07:45:16 +00:00
|
|
|
|
finally:
|
2024-08-20 08:17:42 +00:00
|
|
|
|
pdf_document.close()
|
2024-08-15 07:45:16 +00:00
|
|
|
|
# 处理完后可以删除临时文件
|
|
|
|
|
|
os.remove(temp_file_path)
|
|
|
|
|
|
|
2024-03-29 10:28:05 +00:00
|
|
|
|
return {'name': file.name,
|
|
|
|
|
|
'content': split_model.parse(content)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def support(self, file, get_buffer):
|
|
|
|
|
|
file_name: str = file.name.lower()
|
|
|
|
|
|
if file_name.endswith(".pdf"):
|
|
|
|
|
|
return True
|
|
|
|
|
|
return False
|