merge_dotsocr_paddleocr.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. """
  2. 合并 DotsOCR 和 PaddleOCR 的结果
  3. 主程序入口
  4. """
  5. import json
  6. import argparse
  7. from pathlib import Path
  8. import sys
  9. from pathlib import Path
  10. # 添加 ocr_platform 根目录到 Python 路径
  11. ocr_platform_root = Path(__file__).parents[2] # ocr_merger -> ocr_tools -> ocr_platform -> repository.git
  12. if str(ocr_platform_root) not in sys.path:
  13. sys.path.insert(0, str(ocr_platform_root))
  14. try:
  15. from ocr_tools.ocr_merger.dotsocr_merger import DotsOCRMerger
  16. except ImportError:
  17. from dotsocr_merger import DotsOCRMerger
  18. def merge_single_file(dotsocr_file: Path, paddle_file: Path, output_dir: Path,
  19. output_type: str, merger: DotsOCRMerger) -> bool:
  20. """
  21. 合并单个文件
  22. Args:
  23. dotsocr_file: DotsOCR JSON 文件路径
  24. paddle_file: PaddleOCR JSON 文件路径
  25. output_dir: 输出目录
  26. output_type: 输出格式
  27. merger: 合并器实例
  28. Returns:
  29. 是否成功
  30. """
  31. print(f"📄 处理: {dotsocr_file.name}")
  32. # 输出文件路径
  33. merged_md_path = output_dir / f"{dotsocr_file.stem}.md"
  34. merged_json_path = output_dir / f"{dotsocr_file.stem}.json"
  35. try:
  36. # ✅ 合并数据 (统一输出为MinerU格式)
  37. merged_data = merger.merge_table_with_bbox(
  38. str(dotsocr_file),
  39. str(paddle_file),
  40. data_format='mineru' # 强制使用MinerU格式
  41. )
  42. # ✅ 生成 Markdown (基于MinerU格式)
  43. if output_type in ['markdown', 'both']:
  44. markdown = merger.generate_enhanced_markdown(
  45. merged_data,
  46. str(merged_md_path),
  47. str(dotsocr_file),
  48. data_format='mineru' # 强制使用MinerU格式
  49. )
  50. # ✅ 保存 JSON (MinerU格式)
  51. if output_type in ['json', 'both']:
  52. with open(merged_json_path, 'w', encoding='utf-8') as f:
  53. json.dump(merged_data, f, ensure_ascii=False, indent=2)
  54. print(f" ✅ 合并完成 (MinerU格式)")
  55. print(f" 📊 共处理了 {len(merged_data)} 个对象")
  56. print(f" 💾 输出文件:")
  57. if output_type in ['markdown', 'both']:
  58. print(f" - {merged_md_path.name}")
  59. if output_type in ['json', 'both']:
  60. print(f" - {merged_json_path.name}")
  61. return True
  62. except Exception as e:
  63. print(f" ❌ 处理失败: {e}")
  64. import traceback
  65. traceback.print_exc()
  66. return False
  67. def merge_dotsocr_batch(dotsocr_dir: str, paddle_dir: str, output_dir: str,
  68. output_type: str = 'both',
  69. look_ahead_window: int = 10,
  70. similarity_threshold: int = 80):
  71. """
  72. 批量合并 DotsOCR 和 PaddleOCR 的结果
  73. Args:
  74. dotsocr_dir: DotsOCR 结果目录
  75. paddle_dir: PaddleOCR 结果目录
  76. output_dir: 输出目录
  77. output_type: 输出格式
  78. look_ahead_window: 向前查找窗口大小
  79. similarity_threshold: 相似度阈值
  80. """
  81. dotsocr_path = Path(dotsocr_dir)
  82. paddle_path = Path(paddle_dir)
  83. output_path = Path(output_dir)
  84. output_path.mkdir(parents=True, exist_ok=True)
  85. merger = DotsOCRMerger(look_ahead_window, similarity_threshold)
  86. # 查找所有 DotsOCR 的 JSON 文件
  87. dotsocr_files = list(dotsocr_path.glob('*_page_*[0-9].json'))
  88. dotsocr_files.sort()
  89. print(f"\n🔍 找到 {len(dotsocr_files)} 个 DotsOCR 文件")
  90. print(f"📂 DotsOCR 目录: {dotsocr_dir}")
  91. print(f"📂 PaddleOCR 目录: {paddle_dir}")
  92. print(f"📂 输出目录: {output_dir}")
  93. print(f"⚙️ 查找窗口: {look_ahead_window}")
  94. print(f"⚙️ 相似度阈值: {similarity_threshold}%\n")
  95. success_count = 0
  96. failed_count = 0
  97. for dotsocr_file in dotsocr_files:
  98. # 查找对应的 PaddleOCR 文件
  99. paddle_file = paddle_path / dotsocr_file.name
  100. if not paddle_file.exists():
  101. print(f"⚠️ 跳过: 未找到对应的 PaddleOCR 文件: {paddle_file.name}\n")
  102. failed_count += 1
  103. continue
  104. if merge_single_file(dotsocr_file, paddle_file, output_path, output_type, merger):
  105. success_count += 1
  106. else:
  107. failed_count += 1
  108. print()
  109. print("=" * 60)
  110. print(f"✅ 处理完成!")
  111. print(f"📊 统计信息:")
  112. print(f" - 总文件数: {len(dotsocr_files)}")
  113. print(f" - 成功: {success_count}")
  114. print(f" - 失败: {failed_count}")
  115. print("=" * 60)
  116. def main():
  117. """主函数"""
  118. parser = argparse.ArgumentParser(
  119. description='合并 DotsOCR 和 PaddleOCR 的识别结果,统一输出为MinerU格式',
  120. formatter_class=argparse.RawDescriptionHelpFormatter,
  121. epilog="""
  122. 示例用法:
  123. 1. 批量处理整个目录:
  124. python merge_dotsocr_paddleocr.py \\
  125. --dotsocr-dir /path/to/dotsocr/results \\
  126. --paddle-dir /path/to/paddle/results \\
  127. --output-dir /path/to/output
  128. 2. 处理单个文件:
  129. python merge_dotsocr_paddleocr.py \\
  130. --dotsocr-file /path/to/file_page_001.json \\
  131. --paddle-file /path/to/file_page_001.json \\
  132. --output-dir /path/to/output
  133. 3. 自定义参数:
  134. python merge_dotsocr_paddleocr.py \\
  135. --dotsocr-dir /path/to/dotsocr \\
  136. --paddle-dir /path/to/paddle \\
  137. --output-dir /path/to/output \\
  138. --window 15 \\
  139. --threshold 85
  140. 输出格式说明:
  141. - JSON: 统一的MinerU格式JSON文件
  142. - Markdown: 基于MinerU格式生成的Markdown文件
  143. """
  144. )
  145. # 文件/目录参数
  146. file_group = parser.add_argument_group('文件参数')
  147. file_group.add_argument(
  148. '--dotsocr-file',
  149. type=str,
  150. help='DotsOCR 输出的 JSON 文件路径(单文件模式)'
  151. )
  152. file_group.add_argument(
  153. '--paddle-file',
  154. type=str,
  155. help='PaddleOCR 输出的 JSON 文件路径(单文件模式)'
  156. )
  157. dir_group = parser.add_argument_group('目录参数')
  158. dir_group.add_argument(
  159. '--dotsocr-dir',
  160. type=str,
  161. help='DotsOCR 结果目录(批量模式)'
  162. )
  163. dir_group.add_argument(
  164. '--paddle-dir',
  165. type=str,
  166. help='PaddleOCR 结果目录(批量模式)'
  167. )
  168. # 输出参数
  169. output_group = parser.add_argument_group('输出参数')
  170. output_group.add_argument(
  171. '-o', '--output-dir',
  172. type=str,
  173. required=True,
  174. help='输出目录(必需)'
  175. )
  176. output_group.add_argument(
  177. '-f', '--output-type',
  178. choices=['json', 'markdown', 'both'],
  179. default='both',
  180. help='输出格式'
  181. )
  182. # 算法参数
  183. algo_group = parser.add_argument_group('算法参数')
  184. algo_group.add_argument(
  185. '-w', '--window',
  186. type=int,
  187. default=15,
  188. help='向前查找的窗口大小(默认: 15)'
  189. )
  190. algo_group.add_argument(
  191. '-t', '--threshold',
  192. type=int,
  193. default=80,
  194. help='文本相似度阈值(0-100,默认: 80)'
  195. )
  196. args = parser.parse_args()
  197. output_type = args.output_type.lower()
  198. # 验证参数
  199. if args.dotsocr_file and args.paddle_file:
  200. # 单文件模式
  201. dotsocr_file = Path(args.dotsocr_file)
  202. paddle_file = Path(args.paddle_file)
  203. output_dir = Path(args.output_dir)
  204. if not dotsocr_file.exists():
  205. print(f"❌ 错误: DotsOCR 文件不存在: {dotsocr_file}")
  206. return
  207. if not paddle_file.exists():
  208. print(f"❌ 错误: PaddleOCR 文件不存在: {paddle_file}")
  209. return
  210. output_dir.mkdir(parents=True, exist_ok=True)
  211. print("\n🔧 单文件处理模式")
  212. print(f"📄 DotsOCR 文件: {dotsocr_file}")
  213. print(f"📄 PaddleOCR 文件: {paddle_file}")
  214. print(f"📂 输出目录: {output_dir}\n")
  215. merger = DotsOCRMerger(
  216. look_ahead_window=args.window,
  217. similarity_threshold=args.threshold
  218. )
  219. success = merge_single_file(dotsocr_file, paddle_file, output_dir, output_type, merger)
  220. if success:
  221. print("\n✅ 处理完成!")
  222. else:
  223. print("\n❌ 处理失败!")
  224. elif args.dotsocr_dir and args.paddle_dir:
  225. # 批量模式
  226. if not Path(args.dotsocr_dir).exists():
  227. print(f"❌ 错误: DotsOCR 目录不存在: {args.dotsocr_dir}")
  228. return
  229. if not Path(args.paddle_dir).exists():
  230. print(f"❌ 错误: PaddleOCR 目录不存在: {args.paddle_dir}")
  231. return
  232. print("\n🔧 批量处理模式")
  233. merge_dotsocr_batch(
  234. args.dotsocr_dir,
  235. args.paddle_dir,
  236. args.output_dir,
  237. output_type=output_type,
  238. look_ahead_window=args.window,
  239. similarity_threshold=args.threshold
  240. )
  241. else:
  242. parser.print_help()
  243. print("\n❌ 错误: 请指定单文件模式或批量模式的参数")
  244. if __name__ == "__main__":
  245. print("🚀 启动 DotsOCR + PaddleOCR 合并程序 (统一输出MinerU格式)...")
  246. import sys
  247. if len(sys.argv) == 1:
  248. # 默认配置
  249. default_config = {
  250. "dotsocr-file": "/Users/zhch158/workspace/data/流水分析/A用户_单元格扫描流水/dotsocr_vllm_results/A用户_单元格扫描流水_page_002.json",
  251. "paddle-file": "/Users/zhch158/workspace/data/流水分析/A用户_单元格扫描流水/ppstructurev3_client_results/A用户_单元格扫描流水_page_002.json",
  252. "output-dir": "/Users/zhch158/workspace/data/流水分析/A用户_单元格扫描流水/dotsocr_vllm_results_cell_bbox",
  253. "output-type": "both",
  254. "window": "15",
  255. "threshold": "85"
  256. }
  257. print("ℹ️ 未提供命令行参数,使用默认配置运行...")
  258. print("⚙️ 默认参数:")
  259. for key, value in default_config.items():
  260. print(f" --{key}: {value}")
  261. sys.argv = [sys.argv[0]]
  262. for key, value in default_config.items():
  263. sys.argv.extend([f"--{key}", str(value)])
  264. sys.exit(main())