merge_paddleocr_vl_paddleocr.py 10 KB

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