后端重构: - flask_app.py 拆分为 db.py/helpers.py/routes.py/seed_data.py + Blueprint - 删除死代码 init_db/latest_followup,净减 240 行 - migrations 反向依赖消除 财务项目详情: - 新增任务管理 tab(月份/类型/数量/已执行/差额/单价/执行金额/未执行金额) - 新增项目编号、开始/结束时间、项目经理、合同服务费标准(5%-25%下拉) - 科普业务类型新增科普专访、患教会 - task_data JSON 存储任务列表 财务视图: - 只保留总视图和月度视图,去除确收/毛利和回款/应付视图 - 月度视图月份选择器 - 表格统一居中对齐 - 去除流程项目/流程金额卡片
100 lines
5.7 KiB
Python
100 lines
5.7 KiB
Python
"""migrations/columns.py — 加列迁移(老表补字段,幂等)"""
|
|
|
|
|
|
def _add_column_if_missing(conn, table, column, ddl):
|
|
"""检查列是否存在,不存在才加(幂等)"""
|
|
from db import _exec, mysql, logger
|
|
|
|
cur = conn.cursor(dictionary=True)
|
|
cur.execute(f"SHOW COLUMNS FROM {table} LIKE %s", (column,))
|
|
exists = cur.fetchone()
|
|
cur.close()
|
|
if not exists:
|
|
try:
|
|
_exec(conn, ddl)
|
|
print(f"[migrate] {table}.{column} 列已添加")
|
|
except mysql.connector.Error as e:
|
|
logger.debug(f"add column {table}.{column} skipped: {e}")
|
|
|
|
|
|
def migrate_add_columns():
|
|
"""为老表补齐后续新增的字段"""
|
|
from db import db
|
|
|
|
conn = db()
|
|
try:
|
|
# tenant 字段(多工作台支持)
|
|
for table in ["sales_leads", "follow_up_records", "business_proposals",
|
|
"operation_projects", "product_versions", "finance_records",
|
|
"project_tasks"]:
|
|
_add_column_if_missing(conn, table, "tenant",
|
|
f"ALTER TABLE {table} ADD COLUMN tenant VARCHAR(100) NOT NULL DEFAULT '科普·无界'")
|
|
|
|
# business_proposals 扩展字段
|
|
_add_column_if_missing(conn, "business_proposals", "proposal_type",
|
|
"ALTER TABLE business_proposals ADD COLUMN proposal_type VARCHAR(100) NOT NULL DEFAULT '业务方案'")
|
|
_add_column_if_missing(conn, "business_proposals", "notes",
|
|
"ALTER TABLE business_proposals ADD COLUMN notes VARCHAR(2000) NOT NULL DEFAULT ''")
|
|
|
|
# product_versions 扩展字段
|
|
_add_column_if_missing(conn, "product_versions", "priority",
|
|
"ALTER TABLE product_versions ADD COLUMN priority VARCHAR(10) NOT NULL DEFAULT 'P2'")
|
|
_add_column_if_missing(conn, "product_versions", "start_date",
|
|
"ALTER TABLE product_versions ADD COLUMN start_date VARCHAR(30) NOT NULL DEFAULT ''")
|
|
_add_column_if_missing(conn, "product_versions", "plan_date",
|
|
"ALTER TABLE product_versions ADD COLUMN plan_date VARCHAR(30) NOT NULL DEFAULT ''")
|
|
_add_column_if_missing(conn, "product_versions", "dev_done_date",
|
|
"ALTER TABLE product_versions ADD COLUMN dev_done_date VARCHAR(30) NOT NULL DEFAULT ''")
|
|
_add_column_if_missing(conn, "product_versions", "test_date",
|
|
"ALTER TABLE product_versions ADD COLUMN test_date VARCHAR(30) NOT NULL DEFAULT ''")
|
|
_add_column_if_missing(conn, "product_versions", "devs",
|
|
"ALTER TABLE product_versions ADD COLUMN devs VARCHAR(500) NOT NULL DEFAULT ''")
|
|
_add_column_if_missing(conn, "product_versions", "testers",
|
|
"ALTER TABLE product_versions ADD COLUMN testers VARCHAR(500) NOT NULL DEFAULT ''")
|
|
|
|
# project_tasks 扩展字段
|
|
_add_column_if_missing(conn, "project_tasks", "status",
|
|
"ALTER TABLE project_tasks ADD COLUMN status VARCHAR(50) NOT NULL DEFAULT '未开始'")
|
|
_add_column_if_missing(conn, "project_tasks", "sort_order",
|
|
"ALTER TABLE project_tasks ADD COLUMN sort_order INT NOT NULL DEFAULT 0")
|
|
_add_column_if_missing(conn, "project_tasks", "priority",
|
|
"ALTER TABLE project_tasks ADD COLUMN priority VARCHAR(10) NOT NULL DEFAULT 'P2'")
|
|
|
|
# project_finances 12 个月度预算字段(确收/毛利/回款/费用)
|
|
for m in ["01","02","03","04","05","06","07","08","09","10","11","12"]:
|
|
for field in ["rev", "gross", "payment", "cost"]:
|
|
col = f"{field}_2026_{m}"
|
|
_add_column_if_missing(conn, "project_finances", col,
|
|
f"ALTER TABLE project_finances ADD COLUMN {col} DOUBLE NOT NULL DEFAULT 0")
|
|
|
|
# project_finances 总视图字段:已回款 / 应付 / 已付
|
|
_add_column_if_missing(conn, "project_finances", "total_payment",
|
|
"ALTER TABLE project_finances ADD COLUMN total_payment DOUBLE NOT NULL DEFAULT 0")
|
|
_add_column_if_missing(conn, "project_finances", "total_cost",
|
|
"ALTER TABLE project_finances ADD COLUMN total_cost DOUBLE NOT NULL DEFAULT 0")
|
|
_add_column_if_missing(conn, "project_finances", "total_paid",
|
|
"ALTER TABLE project_finances ADD COLUMN total_paid DOUBLE NOT NULL DEFAULT 0")
|
|
|
|
# project_finances 项目基本信息扩展字段
|
|
_add_column_if_missing(conn, "project_finances", "project_code",
|
|
"ALTER TABLE project_finances ADD COLUMN project_code VARCHAR(50) NOT NULL DEFAULT ''")
|
|
_add_column_if_missing(conn, "project_finances", "start_date",
|
|
"ALTER TABLE project_finances ADD COLUMN start_date VARCHAR(30) NOT NULL DEFAULT ''")
|
|
_add_column_if_missing(conn, "project_finances", "end_date",
|
|
"ALTER TABLE project_finances ADD COLUMN end_date VARCHAR(30) NOT NULL DEFAULT ''")
|
|
_add_column_if_missing(conn, "project_finances", "task_type",
|
|
"ALTER TABLE project_finances ADD COLUMN task_type VARCHAR(100) NOT NULL DEFAULT ''")
|
|
_add_column_if_missing(conn, "project_finances", "task_count",
|
|
"ALTER TABLE project_finances ADD COLUMN task_count DOUBLE NOT NULL DEFAULT 0")
|
|
_add_column_if_missing(conn, "project_finances", "service_fee_standard",
|
|
"ALTER TABLE project_finances ADD COLUMN service_fee_standard DOUBLE NOT NULL DEFAULT 0")
|
|
_add_column_if_missing(conn, "project_finances", "project_manager",
|
|
"ALTER TABLE project_finances ADD COLUMN project_manager VARCHAR(100) NOT NULL DEFAULT ''")
|
|
_add_column_if_missing(conn, "project_finances", "task_data",
|
|
"ALTER TABLE project_finances ADD COLUMN task_data TEXT")
|
|
|
|
conn.commit()
|
|
print("[migrate] 加列迁移完成")
|
|
finally:
|
|
conn.close()
|