- 增强3个CRUD JSON配置:过滤器、子表关联、下拉选择alters - 修复9个API dspy文件:移除违规import,改用init.py函数直接调用 - 新增2个搜索API:get_search_sub_reseller_id、get_search_agreement_id - 自动生成分销商编号(SD-YYYYMMDD-NNNN)和协议编号(DA-YYYYMMDD-NNNN) - 级联删除:删除分销商时级联删除协议及明细,删除协议时级联删除明细 - 更新load_path.py注册新API路径
282 lines
12 KiB
Python
Executable File
282 lines
12 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
"""
|
|
supplychain 模块 RBAC 权限管理脚本
|
|
|
|
使用方法:
|
|
cd ~/repos/sage
|
|
./py3/bin/python ~/repos/supplychain/scripts/load_path.py
|
|
|
|
此脚本注册 supplychain 模块的所有 RBAC 权限路径。
|
|
每次代码变更如有新 path 出现,需同步更新本脚本的 paths 列表。
|
|
|
|
路径分类:
|
|
- any: 静态资源/菜单,无需登录
|
|
- logined: 需要认证的页面和 API
|
|
- reseller.operator: 运营角色 — 供应商、供销合同管理
|
|
- reseller.sale: 销售角色 — 二级分销商、分销协议管理
|
|
"""
|
|
|
|
import subprocess
|
|
import os
|
|
import sys
|
|
|
|
# 查找 Sage 根目录
|
|
def find_sage_root():
|
|
candidates = [
|
|
os.path.expanduser("~/repos/sage"),
|
|
os.path.expanduser("~/sage"),
|
|
os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))),
|
|
]
|
|
for c in candidates:
|
|
if os.path.isdir(os.path.join(c, "py3")) and os.path.isdir(os.path.join(c, "wwwroot")):
|
|
return c
|
|
return None
|
|
|
|
SAGE_ROOT = find_sage_root()
|
|
if not SAGE_ROOT:
|
|
print("ERROR: Cannot find Sage root directory")
|
|
sys.exit(1)
|
|
|
|
SET_PERM = os.path.join(SAGE_ROOT, "py3", "bin", "python")
|
|
SET_PERM_SCRIPT = os.path.join(SAGE_ROOT, "set_role_perm.py")
|
|
|
|
if not os.path.exists(SET_PERM_SCRIPT):
|
|
print(f"ERROR: set_role_perm.py not found at {SET_PERM_SCRIPT}")
|
|
sys.exit(1)
|
|
|
|
# ============================================================
|
|
# 权限路径定义
|
|
# ============================================================
|
|
|
|
# any — 无需登录,静态资源/CRUD 别名目录
|
|
PATHS_ANY = [
|
|
"/supplychain/menu.ui",
|
|
"/supplychain/suppliers_list",
|
|
"/supplychain/supply_contracts_list",
|
|
"/supplychain/supply_contract_items_list",
|
|
"/supplychain/sub_distributors_list",
|
|
"/supplychain/distribution_agreements_list",
|
|
"/supplychain/distribution_agreement_items_list",
|
|
"/supplychain/supplychain_accounting_list",
|
|
"/supplychain/platform_supply_relations_list",
|
|
"/supplychain/platform_supply_products_list",
|
|
"/supplychain/product_supplier_mapping_list",
|
|
]
|
|
|
|
# logined — 需要认证的页面和 API
|
|
PATHS_LOGINED = [
|
|
# 模块入口
|
|
"/supplychain",
|
|
"/supplychain/index.ui",
|
|
# 功能页面
|
|
"/supplychain/suppliers.ui",
|
|
"/supplychain/supply_contracts.ui",
|
|
"/supplychain/sub_distributors.ui",
|
|
"/supplychain/distribution_agreements.ui",
|
|
"/supplychain/accounting.ui",
|
|
# CRUD 列表页
|
|
"/supplychain/suppliers_list/index.ui",
|
|
"/supplychain/supply_contracts_list/index.ui",
|
|
"/supplychain/supply_contract_items_list/index.ui",
|
|
"/supplychain/sub_distributors_list/index.ui",
|
|
"/supplychain/distribution_agreements_list/index.ui",
|
|
"/supplychain/distribution_agreement_items_list/index.ui",
|
|
"/supplychain/supplychain_accounting_list/index.ui",
|
|
# CRUD API — suppliers
|
|
"/supplychain/api/suppliers_create.dspy",
|
|
"/supplychain/api/suppliers_update.dspy",
|
|
"/supplychain/api/suppliers_delete.dspy",
|
|
# CRUD API — supply_contracts
|
|
"/supplychain/api/supply_contracts_create.dspy",
|
|
"/supplychain/api/supply_contracts_update.dspy",
|
|
"/supplychain/api/supply_contracts_delete.dspy",
|
|
# CRUD API — supply_contract_items
|
|
"/supplychain/api/supply_contract_items_create.dspy",
|
|
"/supplychain/api/supply_contract_items_update.dspy",
|
|
"/supplychain/api/supply_contract_items_delete.dspy",
|
|
# Search API — filter dropdowns
|
|
"/supplychain/api/get_search_supplier_id.dspy",
|
|
"/supplychain/api/get_search_contract_id.dspy",
|
|
"/supplychain/api/get_search_prodtypeid.dspy",
|
|
"/supplychain/api/get_search_productid.dspy",
|
|
"/supplychain/api/get_search_sub_reseller_id.dspy",
|
|
"/supplychain/api/get_search_agreement_id.dspy",
|
|
# CRUD API — sub_distributors
|
|
"/supplychain/api/sub_distributors_create.dspy",
|
|
"/supplychain/api/sub_distributors_update.dspy",
|
|
"/supplychain/api/sub_distributors_delete.dspy",
|
|
# CRUD API — distribution_agreements
|
|
"/supplychain/api/distribution_agreements_create.dspy",
|
|
"/supplychain/api/distribution_agreements_update.dspy",
|
|
"/supplychain/api/distribution_agreements_delete.dspy",
|
|
# CRUD API — distribution_agreement_items
|
|
"/supplychain/api/distribution_agreement_items_create.dspy",
|
|
"/supplychain/api/distribution_agreement_items_update.dspy",
|
|
"/supplychain/api/distribution_agreement_items_delete.dspy",
|
|
# CRUD API — supplychain_accounting
|
|
"/supplychain/api/supplychain_accounting_create.dspy",
|
|
"/supplychain/api/supplychain_accounting_update.dspy",
|
|
"/supplychain/api/supplychain_accounting_delete.dspy",
|
|
# 业务 API
|
|
"/supplychain/api/calculate_accounting.dspy",
|
|
"/supplychain/api/query_supply_discount.dspy",
|
|
"/supplychain/api/query_dist_discount.dspy",
|
|
# 平台供销关系
|
|
"/supplychain/platform_supply_relations_list/index.ui",
|
|
"/supplychain/platform_supply_relations_list/get_platform_supply_relations.dspy",
|
|
"/supplychain/platform_supply_relations_list/add_platform_supply_relations.dspy",
|
|
"/supplychain/platform_supply_relations_list/update_platform_supply_relations.dspy",
|
|
"/supplychain/platform_supply_relations_list/delete_platform_supply_relations.dspy",
|
|
"/supplychain/platform_supply_products_list/index.ui",
|
|
"/supplychain/platform_supply_products_list/get_platform_supply_products.dspy",
|
|
"/supplychain/platform_supply_products_list/add_platform_supply_products.dspy",
|
|
"/supplychain/platform_supply_products_list/update_platform_supply_products.dspy",
|
|
"/supplychain/platform_supply_products_list/delete_platform_supply_products.dspy",
|
|
"/supplychain/product_supplier_mapping_list/index.ui",
|
|
"/supplychain/product_supplier_mapping_list/get_product_supplier_mapping.dspy",
|
|
"/supplychain/product_supplier_mapping_list/add_product_supplier_mapping.dspy",
|
|
"/supplychain/product_supplier_mapping_list/update_product_supplier_mapping.dspy",
|
|
"/supplychain/product_supplier_mapping_list/delete_product_supplier_mapping.dspy",
|
|
# CRUD API — platform_supply_relations
|
|
"/supplychain/api/platform_supply_relations_create.dspy",
|
|
"/supplychain/api/platform_supply_relations_update.dspy",
|
|
"/supplychain/api/platform_supply_relations_delete.dspy",
|
|
# CRUD API — platform_supply_products
|
|
"/supplychain/api/platform_supply_products_create.dspy",
|
|
"/supplychain/api/platform_supply_products_update.dspy",
|
|
"/supplychain/api/platform_supply_products_delete.dspy",
|
|
# CRUD API — product_supplier_mapping
|
|
"/supplychain/api/product_supplier_mapping_create.dspy",
|
|
"/supplychain/api/product_supplier_mapping_update.dspy",
|
|
"/supplychain/api/product_supplier_mapping_delete.dspy",
|
|
# 平台供应链业务 API
|
|
"/supplychain/api/query_platform_suppliers.dspy",
|
|
"/supplychain/api/query_platform_products.dspy",
|
|
"/supplychain/api/import_supplier_product.dspy",
|
|
]
|
|
|
|
# 角色专属权限
|
|
PATHS_OPERATOR = [
|
|
"/supplychain/suppliers.ui",
|
|
"/supplychain/suppliers_list",
|
|
"/supplychain/suppliers_list/index.ui",
|
|
"/supplychain/api/suppliers_create.dspy",
|
|
"/supplychain/api/suppliers_update.dspy",
|
|
"/supplychain/api/suppliers_delete.dspy",
|
|
"/supplychain/supply_contracts.ui",
|
|
"/supplychain/supply_contracts_list",
|
|
"/supplychain/supply_contracts_list/index.ui",
|
|
"/supplychain/api/supply_contracts_create.dspy",
|
|
"/supplychain/api/supply_contracts_update.dspy",
|
|
"/supplychain/api/supply_contracts_delete.dspy",
|
|
"/supplychain/supply_contract_items_list",
|
|
"/supplychain/supply_contract_items_list/index.ui",
|
|
"/supplychain/api/supply_contract_items_create.dspy",
|
|
"/supplychain/api/supply_contract_items_update.dspy",
|
|
"/supplychain/api/supply_contract_items_delete.dspy",
|
|
# 平台供销关系管理
|
|
"/supplychain/platform_supply_relations_list",
|
|
"/supplychain/platform_supply_relations_list/index.ui",
|
|
"/supplychain/platform_supply_relations_list/get_platform_supply_relations.dspy",
|
|
"/supplychain/platform_supply_relations_list/add_platform_supply_relations.dspy",
|
|
"/supplychain/platform_supply_relations_list/update_platform_supply_relations.dspy",
|
|
"/supplychain/platform_supply_relations_list/delete_platform_supply_relations.dspy",
|
|
"/supplychain/api/platform_supply_relations_create.dspy",
|
|
"/supplychain/api/platform_supply_relations_update.dspy",
|
|
"/supplychain/api/platform_supply_relations_delete.dspy",
|
|
"/supplychain/platform_supply_products_list",
|
|
"/supplychain/platform_supply_products_list/index.ui",
|
|
"/supplychain/platform_supply_products_list/get_platform_supply_products.dspy",
|
|
"/supplychain/platform_supply_products_list/add_platform_supply_products.dspy",
|
|
"/supplychain/platform_supply_products_list/update_platform_supply_products.dspy",
|
|
"/supplychain/platform_supply_products_list/delete_platform_supply_products.dspy",
|
|
"/supplychain/api/platform_supply_products_create.dspy",
|
|
"/supplychain/api/platform_supply_products_update.dspy",
|
|
"/supplychain/api/platform_supply_products_delete.dspy",
|
|
"/supplychain/product_supplier_mapping_list",
|
|
"/supplychain/product_supplier_mapping_list/index.ui",
|
|
"/supplychain/product_supplier_mapping_list/get_product_supplier_mapping.dspy",
|
|
"/supplychain/product_supplier_mapping_list/add_product_supplier_mapping.dspy",
|
|
"/supplychain/product_supplier_mapping_list/update_product_supplier_mapping.dspy",
|
|
"/supplychain/product_supplier_mapping_list/delete_product_supplier_mapping.dspy",
|
|
"/supplychain/api/product_supplier_mapping_create.dspy",
|
|
"/supplychain/api/product_supplier_mapping_update.dspy",
|
|
"/supplychain/api/product_supplier_mapping_delete.dspy",
|
|
"/supplychain/api/query_platform_suppliers.dspy",
|
|
"/supplychain/api/query_platform_products.dspy",
|
|
"/supplychain/api/import_supplier_product.dspy",
|
|
]
|
|
|
|
PATHS_SALE = [
|
|
"/supplychain/sub_distributors.ui",
|
|
"/supplychain/sub_distributors_list",
|
|
"/supplychain/sub_distributors_list/index.ui",
|
|
"/supplychain/api/sub_distributors_create.dspy",
|
|
"/supplychain/api/sub_distributors_update.dspy",
|
|
"/supplychain/api/sub_distributors_delete.dspy",
|
|
"/supplychain/distribution_agreements.ui",
|
|
"/supplychain/distribution_agreements_list",
|
|
"/supplychain/distribution_agreements_list/index.ui",
|
|
"/supplychain/api/distribution_agreements_create.dspy",
|
|
"/supplychain/api/distribution_agreements_update.dspy",
|
|
"/supplychain/api/distribution_agreements_delete.dspy",
|
|
"/supplychain/distribution_agreement_items_list",
|
|
"/supplychain/distribution_agreement_items_list/index.ui",
|
|
"/supplychain/api/distribution_agreement_items_create.dspy",
|
|
"/supplychain/api/distribution_agreement_items_update.dspy",
|
|
"/supplychain/api/distribution_agreement_items_delete.dspy",
|
|
]
|
|
|
|
# ============================================================
|
|
# 执行注册
|
|
# ============================================================
|
|
|
|
def run_set_perm(role, path, verbose=True):
|
|
"""Register a single permission path."""
|
|
cmd = [SET_PERM, SET_PERM_SCRIPT, role, path]
|
|
result = subprocess.run(cmd, capture_output=True, text=True)
|
|
if verbose:
|
|
output = result.stdout.strip()
|
|
if output:
|
|
print(f" {role}: {path} -> {output}")
|
|
return result.returncode == 0
|
|
|
|
|
|
def register_role_paths(role, paths):
|
|
"""Register all paths for a role."""
|
|
count = 0
|
|
for path in paths:
|
|
if run_set_perm(role, path, verbose=False):
|
|
count += 1
|
|
print(f" {role}: {count}/{len(paths)} paths registered")
|
|
return count
|
|
|
|
|
|
def main():
|
|
print(f"Sage root: {SAGE_ROOT}")
|
|
print(f"set_role_perm.py: {SET_PERM_SCRIPT}")
|
|
print()
|
|
|
|
total = 0
|
|
|
|
print("[1/4] Registering 'any' role paths...")
|
|
total += register_role_paths("any", PATHS_ANY)
|
|
|
|
print("[2/4] Registering 'logined' role paths...")
|
|
total += register_role_paths("logined", PATHS_LOGINED)
|
|
|
|
print("[3/4] Registering 'reseller.operator' role paths...")
|
|
total += register_role_paths("reseller.operator", PATHS_OPERATOR)
|
|
|
|
print("[4/4] Registering 'reseller.sale' role paths...")
|
|
total += register_role_paths("reseller.sale", PATHS_SALE)
|
|
|
|
print()
|
|
print(f"Done. Total {total} permission entries registered.")
|
|
print()
|
|
print("NOTE: Restart Sage after permission changes to reload RBAC cache.")
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|