_fix_plugin_wemall_user_rebate(); // 为余额/积分表添加来源字段 $this->_fix_plugin_payment_balance_integral(); // 为用户关系表添加索引优化 $this->_fix_plugin_wemall_user_relation(); // 为订单表添加检查约束 $this->_fix_plugin_wemall_order(); } /** * 修复返佣记录表. */ private function _fix_plugin_wemall_user_rebate() { $table = $this->table('plugin_wemall_user_rebate'); // 添加订单商品项ID字段(用于精确追踪返佣) if (!$this->hasColumn('plugin_wemall_user_rebate', 'order_item_id')) { $table->addColumn('order_item_id', 'biginteger', ['limit' => 20, 'default' => 0, 'null' => true, 'comment' => '订单商品项ID']) ->update(); } // 添加返佣规则ID字段(用于追溯规则版本) if (!$this->hasColumn('plugin_wemall_user_rebate', 'rebate_rule_id')) { $table->addColumn('rebate_rule_id', 'biginteger', ['limit' => 20, 'default' => 0, 'null' => true, 'comment' => '返佣规则ID']) ->update(); } // 添加金额非负约束 $this->executeModifyWithCheck('plugin_wemall_user_rebate', 'amount', "DECIMAL(20,2) NOT NULL DEFAULT '0.00'", 'amount >= 0'); } /** * 修复余额/积分表. */ private function _fix_plugin_payment_balance_integral() { // 修复余额表 $table = $this->table('plugin_payment_balance'); if (!$this->hasColumn('plugin_payment_balance', 'source_type')) { $table->addColumn('source_type', 'string', ['limit' => 50, 'default' => '', 'null' => true, 'comment' => '资金来源类型']) ->addColumn('source_id', 'string', ['limit' => 50, 'default' => '', 'null' => true, 'comment' => '资金来源ID']) ->update(); } // 修复积分表 $table = $this->table('plugin_payment_integral'); if (!$this->hasColumn('plugin_payment_integral', 'source_type')) { $table->addColumn('source_type', 'string', ['limit' => 50, 'default' => '', 'null' => true, 'comment' => '积分来源类型']) ->addColumn('source_id', 'string', ['limit' => 50, 'default' => '', 'null' => true, 'comment' => '积分来源ID']) ->update(); } // 添加金额非负约束 $this->execute("ALTER TABLE `plugin_payment_balance` MODIFY `amount` DECIMAL(20,2) NOT NULL DEFAULT '0.00'"); $this->execute("ALTER TABLE `plugin_payment_integral` MODIFY `amount` DECIMAL(20,2) NOT NULL DEFAULT '0.00'"); } /** * 修复用户关系表索引. */ private function _fix_plugin_wemall_user_relation() { // 为path字段添加前缀索引(优化LIKE查询性能) $this->execute('ALTER TABLE `plugin_wemall_user_relation` ADD INDEX `idx_path_prefix` (`path`(20))'); // 为代理层级字段添加索引 $this->execute('ALTER TABLE `plugin_wemall_user_relation` ADD INDEX `idx_puid1` (`puid1`)'); $this->execute('ALTER TABLE `plugin_wemall_user_relation` ADD INDEX `idx_puid2` (`puid2`)'); $this->execute('ALTER TABLE `plugin_wemall_user_relation` ADD INDEX `idx_puid3` (`puid3`)'); } /** * 修复订单表约束 */ private function _fix_plugin_wemall_order() { // 添加金额字段的非负约束 $amount_fields = [ 'amount_cost', 'amount_real', 'amount_total', 'amount_goods', 'amount_profit', 'amount_reduct', 'amount_balance', 'amount_integral', 'amount_payment', 'amount_express', 'amount_discount', 'coupon_amount', 'allow_balance', 'allow_integral', 'ratio_integral', 'rebate_amount', 'reward_balance', 'reward_integral', 'payment_amount', ]; foreach ($amount_fields as $field) { $this->executeModifyWithCheck('plugin_wemall_order', $field, "DECIMAL(20,2) NOT NULL DEFAULT '0.00'", "{$field} >= 0"); } // 添加状态字段的枚举约束 $this->executeModifyWithCheck('plugin_wemall_order', 'status', 'TINYINT NOT NULL DEFAULT 1', 'status BETWEEN 0 AND 7'); $this->executeModifyWithCheck('plugin_wemall_order', 'refund_status', 'TINYINT NOT NULL DEFAULT 0', 'refund_status BETWEEN 0 AND 7'); $this->executeModifyWithCheck('plugin_wemall_order', 'payment_status', 'TINYINT NOT NULL DEFAULT 0', 'payment_status BETWEEN 0 AND 2'); $this->executeModifyWithCheck('plugin_wemall_order', 'delivery_type', 'TINYINT NOT NULL DEFAULT 0', 'delivery_type BETWEEN 0 AND 1'); } private function executeModifyWithCheck(string $table, string $field, string $definition, string $check): void { $sql = "ALTER TABLE `{$table}` MODIFY `{$field}` {$definition}"; if ($this->supportsCheckConstraint()) { $sql .= " CHECK ({$check})"; } $this->execute($sql); } private function supportsCheckConstraint(): bool { static $supports = null; if ($supports !== null) { return $supports; } $row = $this->getAdapter()->fetchRow('SELECT VERSION() AS version'); $raw = (string)($row['version'] ?? reset($row) ?: '0.0.0'); preg_match('/\\d+(?:\\.\\d+){1,2}/', $raw, $match); $version = $match[0] ?? '0.0.0'; if (stripos($raw, 'mariadb') !== false) { return $supports = version_compare($version, '10.2.1', '>='); } return $supports = version_compare($version, '8.0.16', '>='); } }