2026-06-20 15:50:24 +08:00
|
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
"""
|
|
|
|
|
|
QGIS 专题图生成子进程入口。
|
|
|
|
|
|
|
|
|
|
|
|
由主进程 (Python 3.10) 通过 subprocess 调用,
|
|
|
|
|
|
运行在 QGIS 自带的 Python 3.12 环境中。
|
|
|
|
|
|
|
|
|
|
|
|
支持两种模式:
|
|
|
|
|
|
- 批量模式(推荐):单次启动 QgsApplication,顺序处理多个模板
|
|
|
|
|
|
输入: { "config": {...}, "models": [{...}, {...}, ...] }
|
|
|
|
|
|
- 单任务模式(兼容):
|
|
|
|
|
|
输入: { "config": {...}, "model": {...} }
|
|
|
|
|
|
|
|
|
|
|
|
输出 JSON (stdout):
|
|
|
|
|
|
批量: { "results": [{"name": "...", "output": "..."}, ...] }
|
|
|
|
|
|
单任务: { "name": "...", "output": "..." }
|
|
|
|
|
|
|
|
|
|
|
|
错误: stderr + exit code 1
|
|
|
|
|
|
"""
|
|
|
|
|
|
import json
|
|
|
|
|
|
import os
|
|
|
|
|
|
import sys
|
|
|
|
|
|
import time
|
|
|
|
|
|
|
|
|
|
|
|
# ============================================================
|
2026-06-21 12:47:43 +08:00
|
|
|
|
# 环境初始化(必须在任何 QGIS/Qt import 之前)
|
2026-06-20 15:50:24 +08:00
|
|
|
|
# ============================================================
|
|
|
|
|
|
|
|
|
|
|
|
QGIS_ROOT = os.environ.get("QGIS_ROOT", "D:/QGIS")
|
|
|
|
|
|
|
|
|
|
|
|
|
2026-06-21 12:47:43 +08:00
|
|
|
|
def _detect_qgis_app_dir():
|
|
|
|
|
|
"""自动检测 QGIS 应用目录(qgis-ltr 或 qgis)"""
|
|
|
|
|
|
for name in ("qgis-ltr", "qgis"):
|
|
|
|
|
|
d = os.path.join(QGIS_ROOT, "apps", name)
|
|
|
|
|
|
if os.path.isdir(d):
|
|
|
|
|
|
return d
|
|
|
|
|
|
raise RuntimeError(
|
|
|
|
|
|
f"未找到 QGIS 应用目录: {QGIS_ROOT}\\apps\\qgis-ltr 或 qgis"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
2026-06-20 15:50:24 +08:00
|
|
|
|
def _setup_python_path():
|
|
|
|
|
|
"""将项目根目录和 QGIS Python 路径加入 sys.path"""
|
|
|
|
|
|
project_root = os.path.dirname(
|
|
|
|
|
|
os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
|
|
|
|
|
|
)
|
|
|
|
|
|
if project_root not in sys.path:
|
|
|
|
|
|
sys.path.insert(0, project_root)
|
|
|
|
|
|
|
2026-06-21 12:47:43 +08:00
|
|
|
|
qgis_app_dir = _detect_qgis_app_dir()
|
|
|
|
|
|
qgis_python = os.path.join(qgis_app_dir, "python")
|
2026-06-20 15:50:24 +08:00
|
|
|
|
if os.path.isdir(qgis_python) and qgis_python not in sys.path:
|
|
|
|
|
|
sys.path.insert(0, qgis_python)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _setup_environment():
|
2026-06-21 12:47:43 +08:00
|
|
|
|
"""设置 QGIS 运行所需的环境变量和 DLL 搜索路径。
|
|
|
|
|
|
|
|
|
|
|
|
QGIS_PREFIX_PATH / QT_PLUGIN_PATH / GDAL_DATA / PROJ_DATA
|
|
|
|
|
|
已由主进程通过 build_qgis_subprocess_env() 设置,子进程不重复设。
|
|
|
|
|
|
这里注册 DLL 搜索目录并预加载核心 DLL 以确保正确加载顺序。
|
|
|
|
|
|
"""
|
2026-06-20 15:50:24 +08:00
|
|
|
|
os.environ["PYTHONUTF8"] = "1"
|
|
|
|
|
|
os.environ["GDAL_FILENAME_IS_UTF8"] = "YES"
|
|
|
|
|
|
os.environ["VSI_CACHE"] = "TRUE"
|
|
|
|
|
|
os.environ["VSI_CACHE_SIZE"] = "1000000"
|
|
|
|
|
|
|
|
|
|
|
|
if sys.platform == "win32":
|
|
|
|
|
|
import ctypes
|
2026-06-21 12:47:43 +08:00
|
|
|
|
qgis_app_dir = _detect_qgis_app_dir()
|
|
|
|
|
|
dll_dirs = [
|
2026-06-20 15:50:24 +08:00
|
|
|
|
os.path.join(qgis_app_dir, "bin"),
|
|
|
|
|
|
os.path.join(QGIS_ROOT, "apps", "Qt5", "bin"),
|
|
|
|
|
|
os.path.join(QGIS_ROOT, "apps", "gdal", "lib"),
|
|
|
|
|
|
]
|
2026-06-21 12:47:43 +08:00
|
|
|
|
for dll_dir in dll_dirs:
|
|
|
|
|
|
if os.path.isdir(dll_dir):
|
|
|
|
|
|
os.add_dll_directory(dll_dir)
|
|
|
|
|
|
|
|
|
|
|
|
# 预加载核心 DLL —— 强制从 QGIS 目录加载,防止系统 PATH 中同名 DLL 干扰
|
2026-06-20 15:50:24 +08:00
|
|
|
|
_preload_dlls = [
|
|
|
|
|
|
"qgis_core.dll", "qgispython.dll",
|
|
|
|
|
|
"Qt5Core.dll", "Qt5Gui.dll", "Qt5Widgets.dll",
|
|
|
|
|
|
"Qt5Network.dll", "Qt5Svg.dll", "Qt5Xml.dll",
|
|
|
|
|
|
"Qt5Concurrent.dll", "Qt5PrintSupport.dll",
|
|
|
|
|
|
]
|
2026-06-21 12:47:43 +08:00
|
|
|
|
for dll_dir in dll_dirs:
|
2026-06-20 15:50:24 +08:00
|
|
|
|
for dll_name in _preload_dlls:
|
|
|
|
|
|
dll_path = os.path.join(dll_dir, dll_name)
|
|
|
|
|
|
if os.path.isfile(dll_path):
|
|
|
|
|
|
try:
|
|
|
|
|
|
ctypes.WinDLL(dll_path)
|
|
|
|
|
|
except OSError:
|
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ============================================================
|
2026-06-21 12:47:43 +08:00
|
|
|
|
# 主逻辑
|
2026-06-20 15:50:24 +08:00
|
|
|
|
# ============================================================
|
|
|
|
|
|
|
|
|
|
|
|
def _init_qgis():
|
|
|
|
|
|
"""初始化 QgsApplication(只做一次)"""
|
|
|
|
|
|
from qgis.core import QgsApplication
|
|
|
|
|
|
|
2026-06-21 12:47:43 +08:00
|
|
|
|
qgis_app_dir = _detect_qgis_app_dir()
|
2026-06-20 15:50:24 +08:00
|
|
|
|
QgsApplication.setPrefixPath(qgis_app_dir, True)
|
|
|
|
|
|
qgs_app = QgsApplication([], False)
|
|
|
|
|
|
qgs_app.initQgis()
|
|
|
|
|
|
return qgs_app
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _process_single(service, model):
|
|
|
|
|
|
"""处理单个模板,返回结果 dict"""
|
|
|
|
|
|
name = service.generate(model)
|
|
|
|
|
|
return {"name": name, "output": model["outFile"]}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
|
|
t_start = time.time()
|
|
|
|
|
|
|
|
|
|
|
|
# 环境初始化
|
|
|
|
|
|
_setup_environment()
|
|
|
|
|
|
_setup_python_path()
|
|
|
|
|
|
|
|
|
|
|
|
# 读取请求 JSON
|
|
|
|
|
|
if len(sys.argv) > 1 and os.path.isfile(sys.argv[1]):
|
|
|
|
|
|
with open(sys.argv[1], "r", encoding="utf-8") as f:
|
|
|
|
|
|
request = json.load(f)
|
|
|
|
|
|
else:
|
|
|
|
|
|
request = json.load(sys.stdin)
|
|
|
|
|
|
|
|
|
|
|
|
config = request["config"]
|
|
|
|
|
|
|
|
|
|
|
|
# 兼容批量和单任务模式
|
|
|
|
|
|
models = request.get("models") or [request["model"]]
|
|
|
|
|
|
|
|
|
|
|
|
# 初始化 QgsApplication(只做一次)
|
|
|
|
|
|
qgs_app = _init_qgis()
|
|
|
|
|
|
|
2026-06-21 13:29:19 +08:00
|
|
|
|
# 从磁盘恢复已缓存的模板(跨进程加速)
|
|
|
|
|
|
from app.services.qgis.map_service import template_cache
|
|
|
|
|
|
cached_count = template_cache.load_persistent_cache()
|
|
|
|
|
|
if cached_count > 0:
|
|
|
|
|
|
print(f"[qgis_runner] 磁盘缓存命中: {cached_count} 个模板", file=sys.stderr)
|
|
|
|
|
|
|
2026-06-20 15:50:24 +08:00
|
|
|
|
try:
|
|
|
|
|
|
from app.services.qgis.map_service import MapService
|
|
|
|
|
|
|
|
|
|
|
|
service = MapService(config)
|
|
|
|
|
|
results = []
|
|
|
|
|
|
|
|
|
|
|
|
for i, model in enumerate(models):
|
|
|
|
|
|
t_model = time.time()
|
|
|
|
|
|
try:
|
|
|
|
|
|
result = _process_single(service, model)
|
|
|
|
|
|
results.append(result)
|
|
|
|
|
|
elapsed = time.time() - t_model
|
|
|
|
|
|
print(
|
|
|
|
|
|
f"[qgis_runner] [{i+1}/{len(models)}] 完成: {result['name']}, "
|
|
|
|
|
|
f"耗时 {elapsed:.1f}s",
|
|
|
|
|
|
file=sys.stderr,
|
|
|
|
|
|
)
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
elapsed = time.time() - t_model
|
|
|
|
|
|
error_msg = f"{e}"
|
|
|
|
|
|
print(
|
|
|
|
|
|
f"[qgis_runner] [{i+1}/{len(models)}] 失败: {model.get('name', '?')}, "
|
|
|
|
|
|
f"耗时 {elapsed:.1f}s — {error_msg}",
|
|
|
|
|
|
file=sys.stderr,
|
|
|
|
|
|
)
|
|
|
|
|
|
results.append({"name": model.get("name", ""), "output": "", "error": error_msg})
|
|
|
|
|
|
|
|
|
|
|
|
# 输出结果
|
|
|
|
|
|
if len(models) == 1 and not request.get("models"):
|
|
|
|
|
|
# 单任务模式兼容
|
|
|
|
|
|
json.dump(results[0], sys.stdout, ensure_ascii=False)
|
|
|
|
|
|
else:
|
|
|
|
|
|
json.dump({"results": results}, sys.stdout, ensure_ascii=False)
|
|
|
|
|
|
sys.stdout.flush()
|
|
|
|
|
|
|
|
|
|
|
|
total = time.time() - t_start
|
|
|
|
|
|
ok = sum(1 for r in results if not r.get("error"))
|
|
|
|
|
|
fail = len(results) - ok
|
|
|
|
|
|
print(
|
|
|
|
|
|
f"\n[qgis_runner] 批量完成: {ok}成功/{fail}失败, 总耗时 {total:.1f}s",
|
|
|
|
|
|
file=sys.stderr,
|
|
|
|
|
|
)
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
elapsed = time.time() - t_start
|
|
|
|
|
|
print(f"[qgis_runner] 致命错误 ({elapsed:.1f}s): {e}", file=sys.stderr)
|
|
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
finally:
|
|
|
|
|
|
qgs_app.exitQgis()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
|
|
main()
|