mirror of
https://github.com/opendatalab/MinerU.git
synced 2026-03-27 11:08:32 +07:00
Compare commits
9 Commits
release-2.
...
release-2.
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
bd80ce2ea2 | ||
|
|
86690409ab | ||
|
|
62d1ef184e | ||
|
|
6d9380323b | ||
|
|
56f25a4e90 | ||
|
|
f85f53d805 | ||
|
|
f7ee044bf3 | ||
|
|
715ccbb08e | ||
|
|
03f8e91889 |
@@ -43,6 +43,8 @@
|
||||
</div>
|
||||
|
||||
# Changelog
|
||||
- 2025/07/24 2.1.5 Released
|
||||
- `sglang` 0.4.9 version adaptation, synchronously upgrading the dockerfile base image to sglang 0.4.9.post3
|
||||
- 2025/07/23 2.1.4 Released
|
||||
- Bug Fixes
|
||||
- Fixed the issue of excessive memory consumption during the `MFR` step in the `pipeline` backend under certain scenarios #2771
|
||||
|
||||
@@ -43,6 +43,8 @@
|
||||
</div>
|
||||
|
||||
# 更新记录
|
||||
- 2025/07/24 2.1.5发布
|
||||
- `sglang` 0.4.9 版本适配,同步升级dockerfile基础镜像为sglang 0.4.9.post3
|
||||
- 2025/07/23 2.1.4发布
|
||||
- bug修复
|
||||
- 修复`pipeline`后端中`MFR`步骤在某些情况下显存消耗过大的问题 #2771
|
||||
|
||||
@@ -1,5 +1,7 @@
|
||||
# Use the official sglang image
|
||||
FROM lmsysorg/sglang:v0.4.8.post1-cu126
|
||||
FROM lmsysorg/sglang:v0.4.9.post3-cu126
|
||||
# For blackwell GPU, use the following line instead:
|
||||
# FROM lmsysorg/sglang:v0.4.9.post3-cu128-b200
|
||||
|
||||
# Install libgl for opencv support & Noto fonts for Chinese characters
|
||||
RUN apt-get update && \
|
||||
|
||||
@@ -1,5 +1,7 @@
|
||||
# Use the official sglang image
|
||||
FROM lmsysorg/sglang:v0.4.8.post1-cu126
|
||||
FROM lmsysorg/sglang:v0.4.9.post3-cu126
|
||||
# For blackwell GPU, use the following line instead:
|
||||
# FROM lmsysorg/sglang:v0.4.9.post3-cu128-b200
|
||||
|
||||
# Install libgl for opencv support & Noto fonts for Chinese characters
|
||||
RUN apt-get update && \
|
||||
|
||||
@@ -10,8 +10,8 @@ docker build -t mineru-sglang:latest -f Dockerfile .
|
||||
```
|
||||
|
||||
> [!TIP]
|
||||
> The [Dockerfile](https://github.com/opendatalab/MinerU/blob/master/docker/global/Dockerfile) uses `lmsysorg/sglang:v0.4.8.post1-cu126` as the base image by default, supporting Turing/Ampere/Ada Lovelace/Hopper platforms.
|
||||
> If you are using the newer `Blackwell` platform, please modify the base image to `lmsysorg/sglang:v0.4.8.post1-cu128-b200` before executing the build operation.
|
||||
> The [Dockerfile](https://github.com/opendatalab/MinerU/blob/master/docker/global/Dockerfile) uses `lmsysorg/sglang:v0.4.9.post3-cu126` as the base image by default, supporting Turing/Ampere/Ada Lovelace/Hopper platforms.
|
||||
> If you are using the newer `Blackwell` platform, please modify the base image to `lmsysorg/sglang:v0.4.9.post3-cu128-b200` before executing the build operation.
|
||||
|
||||
## Docker Description
|
||||
|
||||
|
||||
@@ -10,8 +10,8 @@ docker build -t mineru-sglang:latest -f Dockerfile .
|
||||
```
|
||||
|
||||
> [!TIP]
|
||||
> [Dockerfile](https://github.com/opendatalab/MinerU/blob/master/docker/china/Dockerfile)默认使用`lmsysorg/sglang:v0.4.8.post1-cu126`作为基础镜像,支持Turing/Ampere/Ada Lovelace/Hopper平台,
|
||||
> 如您使用较新的`Blackwell`平台,请将基础镜像修改为`lmsysorg/sglang:v0.4.8.post1-cu128-b200` 再执行build操作。
|
||||
> [Dockerfile](https://github.com/opendatalab/MinerU/blob/master/docker/china/Dockerfile)默认使用`lmsysorg/sglang:v0.4.9.post3-cu126`作为基础镜像,支持Turing/Ampere/Ada Lovelace/Hopper平台,
|
||||
> 如您使用较新的`Blackwell`平台,请将基础镜像修改为`lmsysorg/sglang:v0.4.9.post3-cu128-b200` 再执行build操作。
|
||||
|
||||
## Docker说明
|
||||
|
||||
|
||||
@@ -1,16 +1,9 @@
|
||||
from sglang.srt.configs.model_config import multimodal_model_archs
|
||||
from sglang.srt.models.registry import ModelRegistry
|
||||
|
||||
try:
|
||||
# sglang==0.4.5.post3
|
||||
from sglang.srt.managers.multimodal_processor import (
|
||||
PROCESSOR_MAPPING as PROCESSOR_MAPPING,
|
||||
)
|
||||
except ImportError:
|
||||
# sglang==0.4.4.post1
|
||||
from sglang.srt.managers.image_processor import (
|
||||
IMAGE_PROCESSOR_MAPPING as PROCESSOR_MAPPING,
|
||||
)
|
||||
from sglang.srt.managers.multimodal_processor import (
|
||||
PROCESSOR_MAPPING as PROCESSOR_MAPPING,
|
||||
)
|
||||
|
||||
from .. import vlm_hf_model as _
|
||||
from .image_processor import Mineru2ImageProcessor
|
||||
|
||||
@@ -5,21 +5,22 @@ from typing import List, Optional, Union
|
||||
|
||||
import numpy as np
|
||||
|
||||
try:
|
||||
# sglang==0.4.5.post3
|
||||
from sglang.version import __version__ as sglang_version
|
||||
from packaging import version
|
||||
if version.parse(sglang_version) >= version.parse("0.4.9"):
|
||||
# sglang >= 0.4.9
|
||||
from sglang.srt.multimodal.processors.base_processor import (
|
||||
BaseMultimodalProcessor as BaseProcessor,
|
||||
)
|
||||
from sglang.srt.multimodal.mm_utils import divide_to_patches, expand2square, select_best_resolution
|
||||
else:
|
||||
# 0.4.7 <= sglang < 0.4.9
|
||||
from sglang.srt.managers.multimodal_processors.base_processor import (
|
||||
BaseMultimodalProcessor as BaseProcessor,
|
||||
)
|
||||
from sglang.srt.mm_utils import divide_to_patches, expand2square, select_best_resolution
|
||||
|
||||
get_global_processor = None
|
||||
except ImportError:
|
||||
# sglang==0.4.4.post1
|
||||
from sglang.srt.managers.image_processors.base_image_processor import (
|
||||
BaseImageProcessor as BaseProcessor,
|
||||
get_global_processor,
|
||||
)
|
||||
|
||||
from sglang.srt.mm_utils import divide_to_patches, expand2square, select_best_resolution
|
||||
get_global_processor = None
|
||||
from sglang.srt.utils import load_image, logger
|
||||
from sglang.utils import get_exception_traceback
|
||||
|
||||
@@ -123,64 +124,6 @@ class Mineru2ImageProcessor(BaseProcessor):
|
||||
image_processor,
|
||||
)
|
||||
|
||||
# sglang==0.4.4.post1
|
||||
async def process_images_async(
|
||||
self,
|
||||
image_data: List[Union[str, bytes]],
|
||||
input_text,
|
||||
request_obj,
|
||||
*args,
|
||||
**kwargs,
|
||||
):
|
||||
if not image_data:
|
||||
return None
|
||||
|
||||
modalities = request_obj.modalities or ["image"]
|
||||
aspect_ratio = getattr(self.hf_config, "image_aspect_ratio", "")
|
||||
|
||||
grid_pinpoints = (
|
||||
self.hf_config.image_grid_pinpoints
|
||||
if hasattr(self.hf_config, "image_grid_pinpoints") and "anyres" in aspect_ratio
|
||||
else None
|
||||
)
|
||||
|
||||
if isinstance(image_data, str):
|
||||
image_data = [image_data]
|
||||
|
||||
if isinstance(image_data, list) and len(image_data) > 0:
|
||||
if "multi-images" in modalities or "video" in modalities:
|
||||
# Multiple images
|
||||
aspect_ratio = "pad" # LLaVA OneVision Handling: more than one image --> interleaved image mode or video mode. We do not use anyres
|
||||
pixel_values, image_hashes, image_sizes = [], [], []
|
||||
res = []
|
||||
for img_data in image_data:
|
||||
res.append(self._process_single_image(img_data, aspect_ratio, grid_pinpoints))
|
||||
res = await asyncio.gather(*res)
|
||||
for pixel_v, image_h, image_s in res:
|
||||
pixel_values.append(pixel_v)
|
||||
image_hashes.append(image_h)
|
||||
image_sizes.append(image_s)
|
||||
|
||||
if isinstance(pixel_values[0], np.ndarray):
|
||||
pixel_values = np.stack(pixel_values, axis=0)
|
||||
else:
|
||||
# A single image
|
||||
pixel_values, image_hash, image_size = await self._process_single_image(
|
||||
image_data[0], aspect_ratio, grid_pinpoints
|
||||
)
|
||||
image_hashes = [image_hash]
|
||||
image_sizes = [image_size]
|
||||
else:
|
||||
raise ValueError(f"Invalid image data: {image_data}")
|
||||
|
||||
return {
|
||||
"pixel_values": pixel_values,
|
||||
"image_hashes": image_hashes,
|
||||
"image_sizes": image_sizes,
|
||||
"modalities": request_obj.modalities or ["image"],
|
||||
}
|
||||
|
||||
# sglang==0.4.5.post3
|
||||
async def process_mm_data_async(
|
||||
self,
|
||||
image_data: List[Union[str, bytes]],
|
||||
@@ -191,11 +134,50 @@ class Mineru2ImageProcessor(BaseProcessor):
|
||||
):
|
||||
from sglang.srt.managers.schedule_batch import Modality, MultimodalDataItem
|
||||
|
||||
result = await self.process_images_async(image_data, input_text, request_obj, *args, **kwargs)
|
||||
|
||||
if result is None:
|
||||
if not image_data:
|
||||
return None
|
||||
|
||||
modalities = request_obj.modalities or ["image"]
|
||||
aspect_ratio = getattr(self.hf_config, "image_aspect_ratio", None)
|
||||
grid_pinpoints = (
|
||||
self.hf_config.image_grid_pinpoints
|
||||
if hasattr(self.hf_config, "image_grid_pinpoints")
|
||||
and "anyres" in aspect_ratio
|
||||
else None
|
||||
)
|
||||
|
||||
if isinstance(image_data, str):
|
||||
image_data = [image_data]
|
||||
|
||||
if isinstance(image_data, list) and len(image_data) > 0:
|
||||
if "multi-images" in modalities or "video" in modalities:
|
||||
# Multiple images
|
||||
aspect_ratio = "pad" # LLaVA OneVision Handling: more than one image --> interleaved image mode or video mode. We do not use anyres
|
||||
pixel_values, data_hashes, image_sizes = [], [], []
|
||||
res = []
|
||||
for img_data in image_data:
|
||||
res.append(
|
||||
self._process_single_image(
|
||||
img_data, aspect_ratio, grid_pinpoints
|
||||
)
|
||||
)
|
||||
|
||||
res = await asyncio.gather(*res)
|
||||
for pixel_v, image_h, image_s in res:
|
||||
pixel_values.append(pixel_v)
|
||||
data_hashes.append(image_h)
|
||||
image_sizes.append(image_s)
|
||||
|
||||
if isinstance(pixel_values[0], np.ndarray):
|
||||
pixel_values = np.stack(pixel_values, axis=0)
|
||||
else:
|
||||
# A single image
|
||||
pixel_values, image_hash, image_size = await self._process_single_image(
|
||||
image_data[0], aspect_ratio, grid_pinpoints
|
||||
)
|
||||
image_sizes = [image_size]
|
||||
else:
|
||||
raise ValueError(f"Invalid image data: {image_data}")
|
||||
modality = Modality.IMAGE
|
||||
if isinstance(request_obj.modalities, list):
|
||||
if request_obj.modalities[0] == "multi-images":
|
||||
@@ -203,15 +185,29 @@ class Mineru2ImageProcessor(BaseProcessor):
|
||||
elif request_obj.modalities[0] == "video":
|
||||
modality = Modality.VIDEO
|
||||
|
||||
return {
|
||||
"mm_items": [
|
||||
MultimodalDataItem(
|
||||
pixel_values=result["pixel_values"],
|
||||
image_sizes=result["image_sizes"],
|
||||
modality=modality,
|
||||
)
|
||||
],
|
||||
}
|
||||
|
||||
if version.parse(sglang_version) >= version.parse("0.4.9.post3"):
|
||||
# sglang >= 0.4.9.post3
|
||||
return {
|
||||
"mm_items": [
|
||||
MultimodalDataItem(
|
||||
feature=pixel_values,
|
||||
model_specific_data={
|
||||
"image_sizes": image_sizes,
|
||||
},
|
||||
modality=modality,
|
||||
)
|
||||
],
|
||||
}
|
||||
else:
|
||||
# 0.4.7 <= sglang <= 0.4.9.post2
|
||||
return {
|
||||
"mm_items": [
|
||||
MultimodalDataItem(
|
||||
pixel_values=pixel_values,
|
||||
image_sizes=image_sizes,
|
||||
modality=modality,
|
||||
)
|
||||
],
|
||||
}
|
||||
|
||||
ImageProcessorMapping = {Mineru2QwenForCausalLM: Mineru2ImageProcessor}
|
||||
|
||||
@@ -5,9 +5,20 @@ from typing import Iterable, List, Optional, Tuple
|
||||
import numpy as np
|
||||
import torch
|
||||
from sglang.srt.layers.quantization.base_config import QuantizationConfig
|
||||
from sglang.srt.mm_utils import (
|
||||
get_anyres_image_grid_shape, # unpad_image, unpad_image_shape
|
||||
)
|
||||
|
||||
from sglang.version import __version__ as sglang_version
|
||||
from packaging import version
|
||||
if version.parse(sglang_version) >= version.parse("0.4.9"):
|
||||
# sglang >= 0.4.9
|
||||
from sglang.srt.multimodal.mm_utils import (
|
||||
get_anyres_image_grid_shape,
|
||||
)
|
||||
else:
|
||||
# 0.4.7 <= sglang < 0.4.9
|
||||
from sglang.srt.mm_utils import (
|
||||
get_anyres_image_grid_shape,
|
||||
)
|
||||
|
||||
from sglang.srt.model_executor.forward_batch_info import ForwardBatch
|
||||
from sglang.srt.model_loader.weight_utils import default_weight_loader
|
||||
from sglang.srt.models.qwen2 import Qwen2ForCausalLM
|
||||
@@ -111,14 +122,9 @@ class Mineru2QwenForCausalLM(nn.Module):
|
||||
raise ValueError(f"Unexpected select feature: {self.select_feature}")
|
||||
|
||||
def pad_input_ids(self, input_ids: List[int], image_inputs):
|
||||
if hasattr(image_inputs, "mm_items"): # MultimodalInputs
|
||||
# sglang==0.4.5.post3
|
||||
image_sizes = flatten_nested_list([item.image_sizes for item in image_inputs.mm_items])
|
||||
pad_values = [item.pad_value for item in image_inputs.mm_items]
|
||||
else: # ImageInputs
|
||||
# sglang==0.4.4.post1
|
||||
image_sizes = image_inputs.image_sizes
|
||||
pad_values = image_inputs.pad_values
|
||||
|
||||
image_sizes = flatten_nested_list([item.image_sizes for item in image_inputs.mm_items])
|
||||
pad_values = [item.pad_value for item in image_inputs.mm_items]
|
||||
|
||||
# hardcode for spatial_unpad + anyres
|
||||
# if image_inputs.modalities is not None and (
|
||||
@@ -196,14 +202,8 @@ class Mineru2QwenForCausalLM(nn.Module):
|
||||
positions: torch.Tensor,
|
||||
forward_batch: ForwardBatch,
|
||||
) -> torch.Tensor:
|
||||
if hasattr(forward_batch, "mm_inputs"):
|
||||
# sglang==0.4.5.post3
|
||||
image_inputs = forward_batch.mm_inputs
|
||||
is_sglang_mm_inputs = True
|
||||
else:
|
||||
# sglang==0.4.4.post1
|
||||
image_inputs = forward_batch.image_inputs
|
||||
is_sglang_mm_inputs = False
|
||||
|
||||
image_inputs = forward_batch.mm_inputs
|
||||
|
||||
if image_inputs is None:
|
||||
image_inputs = []
|
||||
@@ -223,12 +223,7 @@ class Mineru2QwenForCausalLM(nn.Module):
|
||||
max_image_offset = []
|
||||
for im in image_inputs:
|
||||
if im:
|
||||
if hasattr(im, "mm_items"):
|
||||
# sglang==0.4.5.post3
|
||||
modalities_list.extend([downgrade_modality(item.modality) for item in im.mm_items])
|
||||
elif im.modalities is not None:
|
||||
# sglang==0.4.4.post1
|
||||
modalities_list.extend(im.modalities)
|
||||
modalities_list.extend([downgrade_modality(item.modality) for item in im.mm_items])
|
||||
if im and im.image_offsets:
|
||||
max_image_offset.append(np.max(np.array(im.image_offsets) + np.array(im.image_pad_len)))
|
||||
else:
|
||||
@@ -240,8 +235,18 @@ class Mineru2QwenForCausalLM(nn.Module):
|
||||
if need_vision.any():
|
||||
bs = forward_batch.batch_size
|
||||
|
||||
if is_sglang_mm_inputs:
|
||||
# sglang==0.4.5.post3
|
||||
if version.parse(sglang_version) >= version.parse("0.4.9.post3"):
|
||||
# sglang >= 0.4.9.post3
|
||||
pixel_values = flatten_nested_list(
|
||||
[[item.feature for item in image_inputs[i].mm_items] for i in range(bs) if need_vision[i]]
|
||||
) # image_inputs[batch_idx].mm_items[item_idx].pixel_values is Tensor
|
||||
image_sizes = [
|
||||
flatten_nested_list([item.model_specific_data["image_sizes"] for item in image_inputs[i].mm_items])
|
||||
for i in range(bs)
|
||||
if need_vision[i]
|
||||
] # image_inputs[batch_idx].mm_items[item_idx].image_sizes should be tuple, but is list of tuple for now.
|
||||
else:
|
||||
# 0.4.7 <= sglang <= 0.4.9.post2
|
||||
pixel_values = flatten_nested_list(
|
||||
[[item.pixel_values for item in image_inputs[i].mm_items] for i in range(bs) if need_vision[i]]
|
||||
) # image_inputs[batch_idx].mm_items[item_idx].pixel_values is Tensor
|
||||
@@ -250,10 +255,6 @@ class Mineru2QwenForCausalLM(nn.Module):
|
||||
for i in range(bs)
|
||||
if need_vision[i]
|
||||
] # image_inputs[batch_idx].mm_items[item_idx].image_sizes should be tuple, but is list of tuple for now.
|
||||
else:
|
||||
# sglang==0.4.4.post1
|
||||
pixel_values = [image_inputs[i].pixel_values for i in range(bs) if need_vision[i]]
|
||||
image_sizes = [image_inputs[i].image_sizes for i in range(bs) if need_vision[i]]
|
||||
|
||||
########## Encode Image ########
|
||||
|
||||
|
||||
@@ -1 +1 @@
|
||||
__version__ = "2.1.3"
|
||||
__version__ = "2.1.4"
|
||||
|
||||
@@ -53,7 +53,7 @@ vlm = [
|
||||
"pydantic",
|
||||
]
|
||||
sglang = [
|
||||
"sglang[all]>=0.4.8,<0.4.9",
|
||||
"sglang[all]>=0.4.7,<0.4.10",
|
||||
]
|
||||
pipeline = [
|
||||
"matplotlib>=3.10,<4",
|
||||
|
||||
Reference in New Issue
Block a user