pdf_parse_by_ocr.py 7.03 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
import os
import time

from loguru import logger

from magic_pdf.libs.commons import read_file, join_path, fitz, get_img_s3_client, get_delta_time, get_docx_model_output
from magic_pdf.libs.safe_filename import sanitize_filename
from magic_pdf.pre_proc.detect_footer_by_model import parse_footers
from magic_pdf.pre_proc.detect_footnote import parse_footnotes_by_model
from magic_pdf.pre_proc.detect_header import parse_headers
from magic_pdf.pre_proc.detect_page_number import parse_pageNos
12
13
from magic_pdf.pre_proc.ocr_detect_layout import layout_detect
from magic_pdf.pre_proc.ocr_dict_merge import merge_spans_to_line, remove_overlaps_min_spans
14
from magic_pdf.pre_proc.ocr_remove_spans import remove_spans_by_bboxes
赵小蒙's avatar
赵小蒙 committed
15
16


17
def construct_page_component(page_id, blocks, layout_bboxes):
赵小蒙's avatar
赵小蒙 committed
18
    return_dict = {
赵小蒙's avatar
赵小蒙 committed
19
20
        'preproc_blocks': blocks,
        'page_idx': page_id,
21
        'layout_bboxes': layout_bboxes,
赵小蒙's avatar
赵小蒙 committed
22
23
24
25
26
    }
    return return_dict


def parse_pdf_by_ocr(
27
28
29
30
31
32
    pdf_path,
    s3_pdf_profile,
    pdf_model_output,
    book_name,
    pdf_model_profile=None,
    image_s3_config=None,
赵小蒙's avatar
赵小蒙 committed
33
34
    start_page_id=0,
    end_page_id=None,
35
    debug_mode=False,
赵小蒙's avatar
赵小蒙 committed
36
):
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
    pdf_bytes = read_file(pdf_path, s3_pdf_profile)
    save_tmp_path = os.path.join(os.path.dirname(__file__), "../..", "tmp", "unittest")
    book_name = sanitize_filename(book_name)
    md_bookname_save_path = ""
    if debug_mode:
        save_path = join_path(save_tmp_path, "md")
        pdf_local_path = join_path(save_tmp_path, "download-pdfs", book_name)

        if not os.path.exists(os.path.dirname(pdf_local_path)):
            # 如果目录不存在,创建它
            os.makedirs(os.path.dirname(pdf_local_path))

        md_bookname_save_path = join_path(save_tmp_path, "md", book_name)
        if not os.path.exists(md_bookname_save_path):
            # 如果目录不存在,创建它
            os.makedirs(md_bookname_save_path)

        with open(pdf_local_path + ".pdf", "wb") as pdf_file:
            pdf_file.write(pdf_bytes)

赵小蒙's avatar
赵小蒙 committed
57

58
59
    pdf_docs = fitz.open("pdf", pdf_bytes)
    # 初始化空的pdf_info_dict
赵小蒙's avatar
赵小蒙 committed
60
    pdf_info_dict = {}
61
62
63
64
65
66
67
    img_s3_client = get_img_s3_client(save_path, image_s3_config)

    start_time = time.time()

    remove_bboxes = []

    end_page_id = end_page_id if end_page_id else len(pdf_docs) - 1
赵小蒙's avatar
赵小蒙 committed
68
    for page_id in range(start_page_id, end_page_id + 1):
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96

        # 获取当前页的page对象
        page = pdf_docs[page_id]

        if debug_mode:
            time_now = time.time()
            logger.info(f"page_id: {page_id}, last_page_cost_time: {get_delta_time(start_time)}")
            start_time = time_now

        # 获取当前页的模型数据
        ocr_page_info = get_docx_model_output(pdf_model_output, pdf_model_profile, page_id)

        """从json中获取每页的页码、页眉、页脚的bbox"""
        page_no_bboxes = parse_pageNos(page_id, page, ocr_page_info)
        header_bboxes = parse_headers(page_id, page, ocr_page_info)
        footer_bboxes = parse_footers(page_id, page, ocr_page_info)
        footnote_bboxes =  parse_footnotes_by_model(page_id, page, ocr_page_info, md_bookname_save_path, debug_mode=debug_mode)

        # 构建需要remove的bbox列表
        need_remove_spans_bboxes = []
        need_remove_spans_bboxes.extend(page_no_bboxes)
        need_remove_spans_bboxes.extend(header_bboxes)
        need_remove_spans_bboxes.extend(footer_bboxes)
        need_remove_spans_bboxes.extend(footnote_bboxes)
        remove_bboxes.append(need_remove_spans_bboxes)



赵小蒙's avatar
赵小蒙 committed
97
98
        layout_dets = ocr_page_info['layout_dets']
        spans = []
99
100
101
102
103
104
105
106
107
108
109
110
111

        # 将模型坐标转换成pymu格式下的未缩放坐标
        DPI = 72  # use this resolution
        pix = page.get_pixmap(dpi=DPI)
        pageL = 0
        pageR = int(pix.w)
        pageU = 0
        pageD = int(pix.h)
        width_from_json = ocr_page_info['page_info']['width']
        height_from_json = ocr_page_info['page_info']['height']
        LR_scaleRatio = width_from_json / (pageR - pageL)
        UD_scaleRatio = height_from_json / (pageD - pageU)

赵小蒙's avatar
赵小蒙 committed
112
113
        for layout_det in layout_dets:
            category_id = layout_det['category_id']
赵小蒙's avatar
赵小蒙 committed
114
            allow_category_id_list = [1, 7, 13, 14, 15]
赵小蒙's avatar
赵小蒙 committed
115
116
            if category_id in allow_category_id_list:
                x0, y0, _, _, x1, y1, _, _ = layout_det['poly']
117
118
119
120
                x0 = x0 / LR_scaleRatio
                y0 = y0 / UD_scaleRatio
                x1 = x1 / LR_scaleRatio
                y1 = y1 / UD_scaleRatio
赵小蒙's avatar
赵小蒙 committed
121
                bbox = [int(x0), int(y0), int(x1), int(y1)]
赵小蒙's avatar
赵小蒙 committed
122
123
124
125
126
127
128
129
130
131
132
133
134
135
                '''要删除的'''
                #  3: 'header',      # 页眉
                #  4: 'page number', # 页码
                #  5: 'footnote',    # 脚注
                #  6: 'footer',      # 页脚
                '''当成span拼接的'''
                #  1: 'image', # 图片
                #  7: 'table',       # 表格
                #  13: 'inline_equation',     # 行内公式
                #  14: 'displayed_equation',      # 行间公式
                #  15: 'text',      # ocr识别文本
                '''layout信息'''
                #  11: 'full column',   # 单栏
                #  12: 'sub column',    # 多栏
赵小蒙's avatar
赵小蒙 committed
136
137
138
                span = {
                    'bbox': bbox,
                }
赵小蒙's avatar
赵小蒙 committed
139
140
                if category_id == 1:
                    span['type'] = 'image'
141

赵小蒙's avatar
赵小蒙 committed
142
143
                elif category_id == 7:
                    span['type'] = 'table'
144

赵小蒙's avatar
赵小蒙 committed
145
                elif category_id == 13:
赵小蒙's avatar
赵小蒙 committed
146
147
148
149
150
151
152
153
154
155
156
157
158
                    span['content'] = layout_det['latex']
                    span['type'] = 'inline_equation'
                elif category_id == 14:
                    span['content'] = layout_det['latex']
                    span['type'] = 'displayed_equation'
                elif category_id == 15:
                    span['content'] = layout_det['text']
                    span['type'] = 'text'
                # print(span)
                spans.append(span)
            else:
                continue

赵小蒙's avatar
赵小蒙 committed
159
160
161
        # 删除重叠spans中较小的那些
        spans = remove_overlaps_min_spans(spans)

162
163
164
        # 删除remove_span_block_bboxes中的bbox
        spans = remove_spans_by_bboxes(spans, need_remove_spans_bboxes)

赵小蒙's avatar
赵小蒙 committed
165
166
167
168
169
170
171
        # 对tpye=["displayed_equation", "image", "table"]进行额外处理,如果左边有字的话,将该span的bbox中y0调整低于文字的y0


        # 将spans合并成line(从上到下,从左到右)
        lines = merge_spans_to_line(spans)
        # logger.info(lines)

赵小蒙's avatar
赵小蒙 committed
172
173
174
175
176
177
178
179
        # 目前不做block拼接,先做个结构,每个block中只有一个line,block的bbox就是line的bbox
        blocks = []
        for line in lines:
            blocks.append({
                "bbox": line['bbox'],
                "lines": [line],
            })

180
181
182
        # 从ocr_page_info中解析layout信息(按自然阅读方向排序,并修复重叠和交错的bad case)
        layout_bboxes = layout_detect(ocr_page_info['subfield_dets'])

赵小蒙's avatar
赵小蒙 committed
183
        # 构造pdf_info_dict
184
        page_info = construct_page_component(page_id, blocks, layout_bboxes)
赵小蒙's avatar
赵小蒙 committed
185
186
        pdf_info_dict[f"page_{page_id}"] = page_info

187
    # logger.info(remove_bboxes)
赵小蒙's avatar
赵小蒙 committed
188
189
    return pdf_info_dict