diff --git a/.gitattributes b/.gitattributes
new file mode 100644
index 00000000..9ed58410
--- /dev/null
+++ b/.gitattributes
@@ -0,0 +1,5 @@
+*.js linguist-vendored
+*.mjs linguist-vendored
+*.html linguist-documentation
+*.css linguist-vendored
+*.scss linguist-vendored
\ No newline at end of file
diff --git a/.github/ISSUE_TEMPLATE/bug_report.yml b/.github/ISSUE_TEMPLATE/bug_report.yml
index 4f161baa..13bbeef1 100644
--- a/.github/ISSUE_TEMPLATE/bug_report.yml
+++ b/.github/ISSUE_TEMPLATE/bug_report.yml
@@ -78,9 +78,9 @@ body:
#multiple: false
options:
-
- - "0.6.x"
- "0.7.x"
- "0.8.x"
+ - "0.9.x"
validations:
required: true
diff --git a/.readthedocs.yaml b/.readthedocs.yaml
index 974d040f..09a7b6b1 100644
--- a/.readthedocs.yaml
+++ b/.readthedocs.yaml
@@ -10,7 +10,7 @@ formats:
python:
install:
- - requirements: docs/zh_cn/requirements.txt
+ - requirements: next_docs/zh_cn/requirements.txt
sphinx:
- configuration: docs/zh_cn/conf.py
+ configuration: next_docs/zh_cn/conf.py
diff --git a/README.md b/README.md
index cdd869e4..c2e4b11d 100644
--- a/README.md
+++ b/README.md
@@ -17,7 +17,7 @@
[![OpenDataLab](https://img.shields.io/badge/Demo_on_OpenDataLab-blue?logo=&labelColor=white)](https://opendatalab.com/OpenSourceTools/Extractor/PDF)
[![HuggingFace](https://img.shields.io/badge/Demo_on_HuggingFace-yellow.svg?logo=&labelColor=white)](https://huggingface.co/spaces/opendatalab/MinerU)
[![ModelScope](https://img.shields.io/badge/Demo_on_ModelScope-purple?logo=&labelColor=white)](https://www.modelscope.cn/studios/OpenDataLab/MinerU)
-[![Colab](https://colab.research.google.com/assets/colab-badge.svg)](https://colab.research.google.com/gist/papayalove/b5f4913389e7ff9883c6b687de156e78/mineru_demo.ipynb)
+[![Colab](https://colab.research.google.com/assets/colab-badge.svg)](https://colab.research.google.com/gist/myhloli/3b3a00a4a0a61577b6c30f989092d20d/mineru_demo.ipynb)
[![Paper](https://img.shields.io/badge/Paper-arXiv-green)](https://arxiv.org/abs/2409.18839)
@@ -42,6 +42,7 @@
# Changelog
+- 2024/11/06 0.9.1 released. Integrated the [StructTable-InternVL2-1B](https://huggingface.co/U4R/StructTable-InternVL2-1B) model for table recognition functionality.
- 2024/10/31 0.9.0 released. This is a major new version with extensive code refactoring, addressing numerous issues, improving performance, reducing hardware requirements, and enhancing usability:
- Refactored the sorting module code to use [layoutreader](https://github.com/ppaanngggg/layoutreader) for reading order sorting, ensuring high accuracy in various layouts.
- Refactored the paragraph concatenation module to achieve good results in cross-column, cross-page, cross-figure, and cross-table scenarios.
diff --git a/README_ja-JP.md b/README_ja-JP.md
index 185db18d..b6351759 100644
--- a/README_ja-JP.md
+++ b/README_ja-JP.md
@@ -18,6 +18,9 @@
+
+ NOTE: このドキュメントはすでに古くなっています。最新版のドキュメントを参照してください。
+
[English](README.md) | [简体中文](README_zh-CN.md) | [日本語](README_ja-JP.md)
diff --git a/README_zh-CN.md b/README_zh-CN.md
index 783f8963..cb264480 100644
--- a/README_zh-CN.md
+++ b/README_zh-CN.md
@@ -17,7 +17,7 @@
[![OpenDataLab](https://img.shields.io/badge/Demo_on_OpenDataLab-blue?logo=&labelColor=white)](https://opendatalab.com/OpenSourceTools/Extractor/PDF)
[![ModelScope](https://img.shields.io/badge/Demo_on_ModelScope-purple?logo=&labelColor=white)](https://www.modelscope.cn/studios/OpenDataLab/MinerU)
[![HuggingFace](https://img.shields.io/badge/Demo_on_HuggingFace-yellow.svg?logo=&labelColor=white)](https://huggingface.co/spaces/opendatalab/MinerU)
-[![Colab](https://colab.research.google.com/assets/colab-badge.svg)](https://colab.research.google.com/gist/papayalove/b5f4913389e7ff9883c6b687de156e78/mineru_demo.ipynb)
+[![Colab](https://colab.research.google.com/assets/colab-badge.svg)](https://colab.research.google.com/gist/myhloli/3b3a00a4a0a61577b6c30f989092d20d/mineru_demo.ipynb)
[![Paper](https://img.shields.io/badge/Paper-arXiv-green)](https://arxiv.org/abs/2409.18839)
@@ -43,6 +43,7 @@
# 更新记录
+- 2024/11/06 0.9.1发布,为表格识别功能接入了[StructTable-InternVL2-1B](https://huggingface.co/U4R/StructTable-InternVL2-1B)模型
- 2024/10/31 0.9.0发布,这是我们进行了大量代码重构的全新版本,解决了众多问题,提升了性能,降低了硬件需求,并提供了更丰富的易用性:
- 重构排序模块代码,使用 [layoutreader](https://github.com/ppaanngggg/layoutreader) 进行阅读顺序排序,确保在各种排版下都能实现极高准确率
- 重构段落拼接模块,在跨栏、跨页、跨图、跨表情况下均能实现良好的段落拼接效果
diff --git a/docs/FAQ_en_us.md b/docs/FAQ_en_us.md
index 21ca861b..ce700ad8 100644
--- a/docs/FAQ_en_us.md
+++ b/docs/FAQ_en_us.md
@@ -57,3 +57,10 @@ pip install paddlepaddle-gpu==3.0.0b1 -i https://www.paddlepaddle.org.cn/package
```
Reference: https://github.com/opendatalab/MinerU/issues/558
+
+### 7. On some Linux servers, the program immediately reports an error `Illegal instruction (core dumped)`
+
+This might be because the server's CPU does not support the AVX/AVX2 instruction set, or the CPU itself supports it but has been disabled by the system administrator. You can try contacting the system administrator to remove the restriction or change to a different server.
+
+References: https://github.com/opendatalab/MinerU/issues/591 , https://github.com/opendatalab/MinerU/issues/736
+
diff --git a/docs/FAQ_zh_cn.md b/docs/FAQ_zh_cn.md
index 667abfa8..4fd8b692 100644
--- a/docs/FAQ_zh_cn.md
+++ b/docs/FAQ_zh_cn.md
@@ -59,3 +59,9 @@ pip install paddlepaddle-gpu==3.0.0b1 -i https://www.paddlepaddle.org.cn/package
```
参考:https://github.com/opendatalab/MinerU/issues/558
+
+### 7.在部分Linux服务器上,程序一运行就报错 `非法指令 (核心已转储)` 或 `Illegal instruction (core dumped)`
+
+可能是因为服务器CPU不支持AVX/AVX2指令集,或cpu本身支持但被运维禁用了,可以尝试联系运维解除限制或更换服务器。
+
+参考:https://github.com/opendatalab/MinerU/issues/591 , https://github.com/opendatalab/MinerU/issues/736
\ No newline at end of file
diff --git a/docs/README_Ubuntu_CUDA_Acceleration_en_US.md b/docs/README_Ubuntu_CUDA_Acceleration_en_US.md
index 45abe97b..2567a28f 100644
--- a/docs/README_Ubuntu_CUDA_Acceleration_en_US.md
+++ b/docs/README_Ubuntu_CUDA_Acceleration_en_US.md
@@ -93,7 +93,7 @@ Download a sample file from the repository and test it.
```sh
wget https://github.com/opendatalab/MinerU/raw/master/demo/small_ocr.pdf
-magic-pdf -p small_ocr.pdf
+magic-pdf -p small_ocr.pdf -o ./output
```
### 9. Test CUDA Acceleration
@@ -108,7 +108,7 @@ If your graphics card has at least **8GB** of VRAM, follow these steps to test C
```
2. Test CUDA acceleration with the following command:
```sh
- magic-pdf -p small_ocr.pdf
+ magic-pdf -p small_ocr.pdf -o ./output
```
### 10. Enable CUDA Acceleration for OCR
@@ -119,5 +119,5 @@ If your graphics card has at least **8GB** of VRAM, follow these steps to test C
```
2. Test OCR acceleration with the following command:
```sh
- magic-pdf -p small_ocr.pdf
+ magic-pdf -p small_ocr.pdf -o ./output
```
diff --git a/docs/README_Ubuntu_CUDA_Acceleration_zh_CN.md b/docs/README_Ubuntu_CUDA_Acceleration_zh_CN.md
index e685051a..46883fa3 100644
--- a/docs/README_Ubuntu_CUDA_Acceleration_zh_CN.md
+++ b/docs/README_Ubuntu_CUDA_Acceleration_zh_CN.md
@@ -11,7 +11,6 @@ nvidia-smi
注意:`CUDA Version` 显示的版本号应 >= 12.1,如显示的版本号小于12.1,请升级驱动
```plaintext
-```
+---------------------------------------------------------------------------------------+
| NVIDIA-SMI 537.34 Driver Version: 537.34 CUDA Version: 12.2 |
|-----------------------------------------+----------------------+----------------------+
@@ -93,7 +92,7 @@ pip install -U magic-pdf[full] --extra-index-url https://wheels.myhloli.com -i h
```bash
wget https://gitee.com/myhloli/MinerU/raw/master/demo/small_ocr.pdf
-magic-pdf -p small_ocr.pdf
+magic-pdf -p small_ocr.pdf -o ./output
```
## 9. 测试CUDA加速
@@ -111,7 +110,7 @@ magic-pdf -p small_ocr.pdf
**2.运行以下命令测试cuda加速效果**
```bash
-magic-pdf -p small_ocr.pdf
+magic-pdf -p small_ocr.pdf -o ./output
```
> 提示:CUDA加速是否生效可以根据log中输出的各个阶段cost耗时来简单判断,通常情况下,`layout detection cost` 和 `mfr time` 应提速10倍以上。
@@ -127,7 +126,7 @@ python -m pip install paddlepaddle-gpu==3.0.0b1 -i https://www.paddlepaddle.org.
**2.运行以下命令测试ocr加速效果**
```bash
-magic-pdf -p small_ocr.pdf
+magic-pdf -p small_ocr.pdf -o ./output
```
> 提示:CUDA加速是否生效可以根据log中输出的各个阶段cost耗时来简单判断,通常情况下,`ocr cost`应提速10倍以上。
diff --git a/docs/README_Windows_CUDA_Acceleration_en_US.md b/docs/README_Windows_CUDA_Acceleration_en_US.md
index 5a911895..d0725285 100644
--- a/docs/README_Windows_CUDA_Acceleration_en_US.md
+++ b/docs/README_Windows_CUDA_Acceleration_en_US.md
@@ -53,7 +53,7 @@ Download a sample file from the repository and test it.
```powershell
wget https://github.com/opendatalab/MinerU/raw/master/demo/small_ocr.pdf -O small_ocr.pdf
- magic-pdf -p small_ocr.pdf
+ magic-pdf -p small_ocr.pdf -o ./output
```
### 8. Test CUDA Acceleration
@@ -86,7 +86,7 @@ If your graphics card has at least 8GB of VRAM, follow these steps to test CUDA-
3. **Run the following command to test CUDA acceleration**:
```
- magic-pdf -p small_ocr.pdf
+ magic-pdf -p small_ocr.pdf -o ./output
```
### 9. Enable CUDA Acceleration for OCR
@@ -97,5 +97,5 @@ If your graphics card has at least 8GB of VRAM, follow these steps to test CUDA-
```
2. **Run the following command to test OCR acceleration**:
```
- magic-pdf -p small_ocr.pdf
+ magic-pdf -p small_ocr.pdf -o ./output
```
diff --git a/docs/README_Windows_CUDA_Acceleration_zh_CN.md b/docs/README_Windows_CUDA_Acceleration_zh_CN.md
index 8ac72787..1737ed67 100644
--- a/docs/README_Windows_CUDA_Acceleration_zh_CN.md
+++ b/docs/README_Windows_CUDA_Acceleration_zh_CN.md
@@ -55,7 +55,7 @@ pip install -U magic-pdf[full] --extra-index-url https://wheels.myhloli.com -i h
```powershell
wget https://github.com/opendatalab/MinerU/raw/master/demo/small_ocr.pdf -O small_ocr.pdf
- magic-pdf -p small_ocr.pdf
+ magic-pdf -p small_ocr.pdf -o ./output
```
## 8. 测试CUDA加速
@@ -87,7 +87,7 @@ pip install --force-reinstall torch==2.3.1 torchvision==0.18.1 --index-url https
**3.运行以下命令测试cuda加速效果**
```bash
-magic-pdf -p small_ocr.pdf
+magic-pdf -p small_ocr.pdf -o ./output
```
> 提示:CUDA加速是否生效可以根据log中输出的各个阶段的耗时来简单判断,通常情况下,`layout detection time` 和 `mfr time` 应提速10倍以上。
@@ -103,7 +103,7 @@ pip install paddlepaddle-gpu==2.6.1
**2.运行以下命令测试ocr加速效果**
```bash
-magic-pdf -p small_ocr.pdf
+magic-pdf -p small_ocr.pdf -o ./output
```
> 提示:CUDA加速是否生效可以根据log中输出的各个阶段cost耗时来简单判断,通常情况下,`ocr time`应提速10倍以上。
diff --git a/magic_pdf/data/data_reader_writer/multi_bucket_s3.py b/magic_pdf/data/data_reader_writer/multi_bucket_s3.py
index 4f6347b3..f54cbd24 100644
--- a/magic_pdf/data/data_reader_writer/multi_bucket_s3.py
+++ b/magic_pdf/data/data_reader_writer/multi_bucket_s3.py
@@ -1,3 +1,4 @@
+import os
from magic_pdf.config.exceptions import InvalidConfig, InvalidParams
from magic_pdf.data.data_reader_writer.base import DataReader, DataWriter
from magic_pdf.data.io.s3 import S3Reader, S3Writer
@@ -7,30 +8,34 @@
class MultiS3Mixin:
- def __init__(self, default_bucket: str, s3_configs: list[S3Config]):
+ def __init__(self, default_prefix: str, s3_configs: list[S3Config]):
"""Initialized with multiple s3 configs.
Args:
- default_bucket (str): the default bucket name of the relative path
+ default_prefix (str): the default prefix of the relative path. for example, {some_bucket}/{some_prefix} or {some_bucket}
s3_configs (list[S3Config]): list of s3 configs, the bucket_name must be unique in the list.
Raises:
- InvalidConfig: default bucket config not in s3_configs
- InvalidConfig: bucket name not unique in s3_configs
- InvalidConfig: default bucket must be provided
+ InvalidConfig: default bucket config not in s3_configs.
+ InvalidConfig: bucket name not unique in s3_configs.
+ InvalidConfig: default bucket must be provided.
"""
- if len(default_bucket) == 0:
- raise InvalidConfig('default_bucket must be provided')
+ if len(default_prefix) == 0:
+ raise InvalidConfig('default_prefix must be provided')
+
+ arr = default_prefix.strip("/").split("/")
+ self.default_bucket = arr[0]
+ self.default_prefix = "/".join(arr[1:])
found_default_bucket_config = False
for conf in s3_configs:
- if conf.bucket_name == default_bucket:
+ if conf.bucket_name == self.default_bucket:
found_default_bucket_config = True
break
if not found_default_bucket_config:
raise InvalidConfig(
- f'default_bucket: {default_bucket} config must be provided in s3_configs: {s3_configs}'
+ f'default_bucket: {self.default_bucket} config must be provided in s3_configs: {s3_configs}'
)
uniq_bucket = set([conf.bucket_name for conf in s3_configs])
@@ -39,7 +44,6 @@ def __init__(self, default_bucket: str, s3_configs: list[S3Config]):
f'the bucket_name in s3_configs: {s3_configs} must be unique'
)
- self.default_bucket = default_bucket
self.s3_configs = s3_configs
self._s3_clients_h: dict = {}
@@ -47,14 +51,14 @@ def __init__(self, default_bucket: str, s3_configs: list[S3Config]):
class MultiBucketS3DataReader(DataReader, MultiS3Mixin):
def read(self, path: str) -> bytes:
"""Read the path from s3, select diffect bucket client for each request
- based on the path, also support range read.
+ based on the bucket, also support range read.
Args:
- path (str): the s3 path of file, the path must be in the format of s3://bucket_name/path?offset,limit
- for example: s3://bucket_name/path?0,100
+ path (str): the s3 path of file, the path must be in the format of s3://bucket_name/path?offset,limit.
+ for example: s3://bucket_name/path?0,100.
Returns:
- bytes: the content of s3 file
+ bytes: the content of s3 file.
"""
may_range_params = parse_s3_range_params(path)
if may_range_params is None or 2 != len(may_range_params):
@@ -84,21 +88,22 @@ def __get_s3_client(self, bucket_name: str):
def read_at(self, path: str, offset: int = 0, limit: int = -1) -> bytes:
"""Read the file with offset and limit, select diffect bucket client
- for each request based on the path.
+ for each request based on the bucket.
Args:
- path (str): the file path
+ path (str): the file path.
offset (int, optional): the number of bytes skipped. Defaults to 0.
limit (int, optional): the number of bytes want to read. Defaults to -1 which means infinite.
Returns:
- bytes: the file content
+ bytes: the file content.
"""
if path.startswith('s3://'):
bucket_name, path = parse_s3path(path)
s3_reader = self.__get_s3_client(bucket_name)
else:
s3_reader = self.__get_s3_client(self.default_bucket)
+ path = os.path.join(self.default_prefix, path)
return s3_reader.read_at(path, offset, limit)
@@ -123,15 +128,16 @@ def __get_s3_client(self, bucket_name: str):
def write(self, path: str, data: bytes) -> None:
"""Write file with data, also select diffect bucket client for each
- request based on the path.
+ request based on the bucket.
Args:
path (str): the path of file, if the path is relative path, it will be joined with parent_dir.
- data (bytes): the data want to write
+ data (bytes): the data want to write.
"""
if path.startswith('s3://'):
bucket_name, path = parse_s3path(path)
s3_writer = self.__get_s3_client(bucket_name)
else:
s3_writer = self.__get_s3_client(self.default_bucket)
+ path = os.path.join(self.default_prefix, path)
return s3_writer.write(path, data)
diff --git a/magic_pdf/data/data_reader_writer/s3.py b/magic_pdf/data/data_reader_writer/s3.py
index b6f27a27..34ec43b7 100644
--- a/magic_pdf/data/data_reader_writer/s3.py
+++ b/magic_pdf/data/data_reader_writer/s3.py
@@ -6,6 +6,7 @@
class S3DataReader(MultiBucketS3DataReader):
def __init__(
self,
+ default_prefix_without_bucket: str,
bucket: str,
ak: str,
sk: str,
@@ -15,6 +16,7 @@ def __init__(
"""s3 reader client.
Args:
+ default_prefix_without_bucket: prefix that not contains bucket
bucket (str): bucket name
ak (str): access key
sk (str): secret key
@@ -23,7 +25,7 @@ def __init__(
refer to https://boto3.amazonaws.com/v1/documentation/api/1.9.42/guide/s3.html
"""
super().__init__(
- bucket,
+ f'{bucket}/{default_prefix_without_bucket}',
[
S3Config(
bucket_name=bucket,
@@ -39,6 +41,7 @@ def __init__(
class S3DataWriter(MultiBucketS3DataWriter):
def __init__(
self,
+ default_prefix_without_bucket: str,
bucket: str,
ak: str,
sk: str,
@@ -48,6 +51,7 @@ def __init__(
"""s3 writer client.
Args:
+ default_prefix_without_bucket: prefix that not contains bucket
bucket (str): bucket name
ak (str): access key
sk (str): secret key
@@ -56,7 +60,7 @@ def __init__(
refer to https://boto3.amazonaws.com/v1/documentation/api/1.9.42/guide/s3.html
"""
super().__init__(
- bucket,
+ f'{bucket}/{default_prefix_without_bucket}',
[
S3Config(
bucket_name=bucket,
diff --git a/magic_pdf/data/io/__init__.py b/magic_pdf/data/io/__init__.py
index e69de29b..badf1df0 100644
--- a/magic_pdf/data/io/__init__.py
+++ b/magic_pdf/data/io/__init__.py
@@ -0,0 +1,6 @@
+
+from magic_pdf.data.io.base import IOReader, IOWriter # noqa: F401
+from magic_pdf.data.io.http import HttpReader, HttpWriter # noqa: F401
+from magic_pdf.data.io.s3 import S3Reader, S3Writer # noqa: F401
+
+__all__ = ['IOReader', 'IOWriter', 'HttpReader', 'HttpWriter', 'S3Reader', 'S3Writer']
\ No newline at end of file
diff --git a/magic_pdf/data/io/base.py b/magic_pdf/data/io/base.py
index c02c2ed4..3c163d1f 100644
--- a/magic_pdf/data/io/base.py
+++ b/magic_pdf/data/io/base.py
@@ -29,7 +29,7 @@ def read_at(self, path: str, offset: int = 0, limit: int = -1) -> bytes:
pass
-class IOWriter:
+class IOWriter(ABC):
@abstractmethod
def write(self, path: str, data: bytes) -> None:
diff --git a/magic_pdf/data/schemas.py b/magic_pdf/data/schemas.py
index 3adb6760..c2efb46a 100644
--- a/magic_pdf/data/schemas.py
+++ b/magic_pdf/data/schemas.py
@@ -3,6 +3,8 @@
class S3Config(BaseModel):
+ """S3 config
+ """
bucket_name: str = Field(description='s3 bucket name', min_length=1)
access_key: str = Field(description='s3 access key', min_length=1)
secret_key: str = Field(description='s3 secret key', min_length=1)
@@ -11,5 +13,7 @@ class S3Config(BaseModel):
class PageInfo(BaseModel):
+ """The width and height of page
+ """
w: float = Field(description='the width of page')
h: float = Field(description='the height of page')
diff --git a/magic_pdf/dict2md/ocr_mkcontent.py b/magic_pdf/dict2md/ocr_mkcontent.py
index 49837441..2e2ce76e 100644
--- a/magic_pdf/dict2md/ocr_mkcontent.py
+++ b/magic_pdf/dict2md/ocr_mkcontent.py
@@ -119,6 +119,16 @@ def detect_language(text):
return 'empty'
+# 连写字符拆分
+def __replace_ligatures(text: str):
+ text = re.sub(r'fi', 'fi', text) # 替换 fi 连写符
+ text = re.sub(r'fl', 'fl', text) # 替换 fl 连写符
+ text = re.sub(r'ff', 'ff', text) # 替换 ff 连写符
+ text = re.sub(r'ffi', 'ffi', text) # 替换 ffi 连写符
+ text = re.sub(r'ffl', 'ffl', text) # 替换 ffl 连写符
+ return text
+
+
def merge_para_with_text(para_block):
para_text = ''
for i, line in enumerate(para_block['lines']):
@@ -141,22 +151,34 @@ def merge_para_with_text(para_block):
if span_type == ContentType.Text:
content = ocr_escape_special_markdown_char(span['content'])
elif span_type == ContentType.InlineEquation:
- content = f" ${span['content']}$ "
+ content = f"${span['content']}$"
elif span_type == ContentType.InterlineEquation:
content = f"\n$$\n{span['content']}\n$$\n"
+ content = content.strip()
if content != '':
langs = ['zh', 'ja', 'ko']
if line_lang in langs: # 遇到一些一个字一个span的文档,这种单字语言判断不准,需要用整行文本判断
- para_text += content # 中文/日语/韩文语境下,content间不需要空格分隔
- elif line_lang == 'en':
- # 如果是前一行带有-连字符,那么末尾不应该加空格
- if __is_hyphen_at_line_end(content):
- para_text += content[:-1]
- else:
- para_text += content + ' '
+ if span_type in [ContentType.Text, ContentType.InterlineEquation]:
+ para_text += content # 中文/日语/韩文语境下,content间不需要空格分隔
+ elif span_type == ContentType.InlineEquation:
+ para_text += f" {content} "
else:
- para_text += content + ' ' # 西方文本语境下 content间需要空格分隔
+ if span_type in [ContentType.Text, ContentType.InlineEquation]:
+ # 如果是前一行带有-连字符,那么末尾不应该加空格
+ if __is_hyphen_at_line_end(content):
+ para_text += content[:-1]
+ elif len(content) == 1 and content not in ['A', 'I', 'a', 'i']:
+ para_text += content
+ else: # 西方文本语境下 content间需要空格分隔
+ para_text += f"{content} "
+ elif span_type == ContentType.InterlineEquation:
+ para_text += content
+ else:
+ continue
+ # 连写字符拆分
+ para_text = __replace_ligatures(para_text)
+
return para_text
diff --git a/magic_pdf/model/pdf_extract_kit.py b/magic_pdf/model/pdf_extract_kit.py
index d6f07550..3366b7de 100644
--- a/magic_pdf/model/pdf_extract_kit.py
+++ b/magic_pdf/model/pdf_extract_kit.py
@@ -38,15 +38,13 @@
from magic_pdf.model.pek_sub_modules.layoutlmv3.model_init import Layoutlmv3_Predictor
from magic_pdf.model.pek_sub_modules.post_process import latex_rm_whitespace
from magic_pdf.model.pek_sub_modules.self_modify import ModifiedPaddleOCR
-# from magic_pdf.model.pek_sub_modules.structeqtable.StructTableModel import StructTableModel
+from magic_pdf.model.pek_sub_modules.structeqtable.StructTableModel import StructTableModel
from magic_pdf.model.ppTableModel import ppTableModel
def table_model_init(table_model_type, model_path, max_time, _device_='cpu'):
if table_model_type == MODEL_NAME.STRUCT_EQTABLE:
- # table_model = StructTableModel(model_path, max_time=max_time, device=_device_)
- logger.error("StructEqTable is under upgrade, the current version does not support it.")
- exit(1)
+ table_model = StructTableModel(model_path, max_time=max_time)
elif table_model_type == MODEL_NAME.TABLE_MASTER:
config = {
"model_dir": model_path,
@@ -463,7 +461,9 @@ def __call__(self, image):
html_code = None
if self.table_model_name == MODEL_NAME.STRUCT_EQTABLE:
with torch.no_grad():
- latex_code = self.table_model.image2latex(new_image)[0]
+ table_result = self.table_model.predict(new_image, "html")
+ if len(table_result) > 0:
+ html_code = table_result[0]
else:
html_code = self.table_model.img2html(new_image)
@@ -474,14 +474,17 @@ def __call__(self, image):
# 判断是否返回正常
if latex_code:
- expected_ending = latex_code.strip().endswith('end{tabular}') or latex_code.strip().endswith(
- 'end{table}')
+ expected_ending = latex_code.strip().endswith('end{tabular}') or latex_code.strip().endswith('end{table}')
if expected_ending:
res["latex"] = latex_code
else:
logger.warning(f"table recognition processing fails, not found expected LaTeX table end")
elif html_code:
- res["html"] = html_code
+ expected_ending = html_code.strip().endswith('