ai-engineering-init 1.7.0 → 1.10.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude/agents/bug-analyzer.md +103 -0
- package/.claude/agents/code-reviewer.md +115 -5
- package/.claude/agents/image-reader.md +154 -0
- package/.claude/agents/loki-runner.md +80 -0
- package/.claude/agents/mysql-runner.md +81 -0
- package/.claude/agents/requirements-analyzer.md +162 -0
- package/.claude/agents/task-fetcher.md +75 -0
- package/.claude/commands/dev.md +29 -0
- package/.claude/commands/next.md +31 -1
- package/.claude/commands/progress.md +23 -1
- package/.claude/hooks/skill-forced-eval.js +46 -62
- package/.claude/settings.json +10 -1
- package/.claude/skills/api-development/SKILL.md +179 -130
- package/.claude/skills/architecture-design/SKILL.md +102 -212
- package/.claude/skills/backend-annotations/SKILL.md +166 -220
- package/.claude/skills/bug-detective/SKILL.md +225 -186
- package/.claude/skills/code-patterns/SKILL.md +127 -244
- package/.claude/skills/collaborating-with-codex/SKILL.md +96 -113
- package/.claude/skills/crud-development/SKILL.md +226 -307
- package/.claude/skills/data-permission/SKILL.md +131 -202
- package/.claude/skills/database-ops/SKILL.md +158 -355
- package/.claude/skills/error-handler/SKILL.md +224 -285
- package/.claude/skills/file-oss-management/SKILL.md +174 -169
- package/.claude/skills/git-workflow/SKILL.md +123 -341
- package/.claude/skills/json-serialization/SKILL.md +121 -137
- package/.claude/skills/performance-doctor/SKILL.md +83 -89
- package/.claude/skills/redis-cache/SKILL.md +134 -185
- package/.claude/skills/scheduled-jobs/SKILL.md +187 -224
- package/.claude/skills/security-guard/SKILL.md +168 -276
- package/.claude/skills/sms-mail/SKILL.md +266 -228
- package/.claude/skills/social-login/SKILL.md +257 -195
- package/.claude/skills/tenant-management/SKILL.md +172 -188
- package/.claude/skills/utils-toolkit/SKILL.md +214 -222
- package/.claude/skills/websocket-sse/SKILL.md +251 -172
- package/.claude/skills/workflow-engine/SKILL.md +178 -250
- package/.codex/skills/api-development/SKILL.md +179 -130
- package/.codex/skills/architecture-design/SKILL.md +102 -212
- package/.codex/skills/backend-annotations/SKILL.md +166 -220
- package/.codex/skills/bug-detective/SKILL.md +225 -186
- package/.codex/skills/code-patterns/SKILL.md +127 -244
- package/.codex/skills/collaborating-with-codex/SKILL.md +96 -113
- package/.codex/skills/crud-development/SKILL.md +226 -307
- package/.codex/skills/data-permission/SKILL.md +131 -202
- package/.codex/skills/database-ops/SKILL.md +158 -355
- package/.codex/skills/dev/SKILL.md +476 -131
- package/.codex/skills/error-handler/SKILL.md +224 -285
- package/.codex/skills/file-oss-management/SKILL.md +174 -169
- package/.codex/skills/git-workflow/SKILL.md +123 -341
- package/.codex/skills/json-serialization/SKILL.md +121 -137
- package/.codex/skills/next/SKILL.md +186 -42
- package/.codex/skills/performance-doctor/SKILL.md +83 -89
- package/.codex/skills/progress/SKILL.md +147 -76
- package/.codex/skills/redis-cache/SKILL.md +134 -185
- package/.codex/skills/scheduled-jobs/SKILL.md +187 -224
- package/.codex/skills/security-guard/SKILL.md +168 -276
- package/.codex/skills/sms-mail/SKILL.md +266 -228
- package/.codex/skills/social-login/SKILL.md +257 -195
- package/.codex/skills/tenant-management/SKILL.md +172 -188
- package/.codex/skills/utils-toolkit/SKILL.md +214 -222
- package/.codex/skills/websocket-sse/SKILL.md +251 -172
- package/.codex/skills/workflow-engine/SKILL.md +178 -250
- package/.cursor/agents/bug-analyzer.md +102 -0
- package/.cursor/agents/code-reviewer.md +80 -97
- package/.cursor/agents/image-reader.md +154 -0
- package/.cursor/agents/loki-runner.md +80 -0
- package/.cursor/agents/mysql-runner.md +81 -0
- package/.cursor/agents/project-manager.md +1 -1
- package/.cursor/agents/requirements-analyzer.md +141 -0
- package/.cursor/agents/task-fetcher.md +75 -0
- package/.cursor/hooks/cursor-skill-eval.js +66 -6
- package/.cursor/skills/api-development/SKILL.md +179 -130
- package/.cursor/skills/architecture-design/SKILL.md +102 -212
- package/.cursor/skills/backend-annotations/SKILL.md +166 -220
- package/.cursor/skills/bug-detective/SKILL.md +225 -186
- package/.cursor/skills/code-patterns/SKILL.md +127 -244
- package/.cursor/skills/collaborating-with-codex/SKILL.md +96 -113
- package/.cursor/skills/crud-development/SKILL.md +226 -307
- package/.cursor/skills/data-permission/SKILL.md +131 -202
- package/.cursor/skills/database-ops/SKILL.md +158 -355
- package/.cursor/skills/error-handler/SKILL.md +224 -285
- package/.cursor/skills/file-oss-management/SKILL.md +174 -169
- package/.cursor/skills/git-workflow/SKILL.md +123 -341
- package/.cursor/skills/json-serialization/SKILL.md +121 -137
- package/.cursor/skills/performance-doctor/SKILL.md +83 -89
- package/.cursor/skills/redis-cache/SKILL.md +134 -185
- package/.cursor/skills/scheduled-jobs/SKILL.md +187 -224
- package/.cursor/skills/security-guard/SKILL.md +168 -276
- package/.cursor/skills/sms-mail/SKILL.md +266 -228
- package/.cursor/skills/social-login/SKILL.md +257 -195
- package/.cursor/skills/tenant-management/SKILL.md +172 -188
- package/.cursor/skills/utils-toolkit/SKILL.md +214 -222
- package/.cursor/skills/websocket-sse/SKILL.md +251 -172
- package/.cursor/skills/workflow-engine/SKILL.md +178 -250
- package/AGENTS.md +117 -540
- package/CLAUDE.md +105 -117
- package/README.md +37 -6
- package/bin/index.js +5 -1
- package/package.json +1 -1
- package/src/skills/api-development/SKILL.md +179 -130
- package/src/skills/architecture-design/SKILL.md +102 -212
- package/src/skills/backend-annotations/SKILL.md +166 -220
- package/src/skills/bug-detective/SKILL.md +225 -186
- package/src/skills/code-patterns/SKILL.md +127 -244
- package/src/skills/collaborating-with-codex/SKILL.md +96 -113
- package/src/skills/crud-development/SKILL.md +226 -307
- package/src/skills/data-permission/SKILL.md +131 -202
- package/src/skills/database-ops/SKILL.md +158 -355
- package/src/skills/error-handler/SKILL.md +224 -285
- package/src/skills/file-oss-management/SKILL.md +174 -169
- package/src/skills/git-workflow/SKILL.md +123 -341
- package/src/skills/json-serialization/SKILL.md +121 -137
- package/src/skills/performance-doctor/SKILL.md +83 -89
- package/src/skills/redis-cache/SKILL.md +134 -185
- package/src/skills/scheduled-jobs/SKILL.md +187 -224
- package/src/skills/security-guard/SKILL.md +168 -276
- package/src/skills/sms-mail/SKILL.md +266 -228
- package/src/skills/social-login/SKILL.md +257 -195
- package/src/skills/tenant-management/SKILL.md +172 -188
- package/src/skills/utils-toolkit/SKILL.md +214 -222
- package/src/skills/websocket-sse/SKILL.md +251 -172
- package/src/skills/workflow-engine/SKILL.md +178 -250
- package/.claude/skills/skill-creator/LICENSE.txt +0 -202
- package/.claude/skills/skill-creator/SKILL.md +0 -479
- package/.claude/skills/skill-creator/agents/analyzer.md +0 -274
- package/.claude/skills/skill-creator/agents/comparator.md +0 -202
- package/.claude/skills/skill-creator/agents/grader.md +0 -223
- package/.claude/skills/skill-creator/assets/eval_review.html +0 -146
- package/.claude/skills/skill-creator/eval-viewer/generate_review.py +0 -471
- package/.claude/skills/skill-creator/eval-viewer/viewer.html +0 -1325
- package/.claude/skills/skill-creator/references/schemas.md +0 -430
- package/.claude/skills/skill-creator/scripts/__init__.py +0 -0
- package/.claude/skills/skill-creator/scripts/aggregate_benchmark.py +0 -401
- package/.claude/skills/skill-creator/scripts/generate_report.py +0 -326
- package/.claude/skills/skill-creator/scripts/improve_description.py +0 -248
- package/.claude/skills/skill-creator/scripts/package_skill.py +0 -136
- package/.claude/skills/skill-creator/scripts/quick_validate.py +0 -103
- package/.claude/skills/skill-creator/scripts/run_eval.py +0 -310
- package/.claude/skills/skill-creator/scripts/run_loop.py +0 -332
- package/.claude/skills/skill-creator/scripts/utils.py +0 -47
|
@@ -1,187 +1,532 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
当需要从零开始开发新功能、完整开发流程时自动使用此 Skill。
|
|
1
|
+
# /dev - 开发新功能
|
|
2
|
+
|
|
3
|
+
智能开发向导,专为 leniu-tengyun-core(云食堂)四层架构设计。包含表结构设计、代码生成全流程。
|
|
5
4
|
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
5
|
+
## 适用场景
|
|
6
|
+
|
|
7
|
+
- ✅ 从零开始开发新业务功能
|
|
8
|
+
- ✅ 需要数据库表设计
|
|
9
|
+
- ✅ 需要完整的四层代码(Controller → Business → Service → Mapper)
|
|
11
10
|
|
|
12
|
-
触发词:开发功能、dev、新功能、功能开发、从零开发、完整开发、开发新模块
|
|
13
11
|
---
|
|
14
12
|
|
|
15
|
-
|
|
13
|
+
## 执行流程
|
|
14
|
+
|
|
15
|
+
### 第零步:复杂度快速评估(30秒内完成)
|
|
16
|
+
|
|
17
|
+
在开始开发前,快速判断功能复杂度:
|
|
18
|
+
|
|
19
|
+
| 维度 | 轻量(直接开发) | 中等 | 复杂(建议 OpenSpec) |
|
|
20
|
+
|------|-----------------|------|---------------------|
|
|
21
|
+
| **表数量** | 1 张表 | 2-3 张表 | 4+ 张表 |
|
|
22
|
+
| **跨模块** | 单模块内 | 偶尔跨模块 | 多模块协作 |
|
|
23
|
+
| **业务流程** | 简单 CRUD | 状态流转 | 多步骤审批/复杂流程 |
|
|
24
|
+
| **接口数量** | ≤5 个 | 6-10 个 | >10 个 |
|
|
25
|
+
|
|
26
|
+
**评估结果处理**:
|
|
16
27
|
|
|
17
|
-
|
|
28
|
+
- **轻量/中等** → 继续本流程(`/dev`),直接进入第一步
|
|
29
|
+
- **复杂** → 提示用户:
|
|
30
|
+
```
|
|
31
|
+
此功能涉及 [X 张表 / 多模块协作 / 复杂流程],建议先用 OpenSpec 进行需求拆解:
|
|
32
|
+
- 快速拆解:/opsx:ff [功能名](一键生成 proposal → specs → design → tasks)
|
|
33
|
+
- 逐步拆解:/opsx:new [功能名](交互式逐步创建)
|
|
18
34
|
|
|
19
|
-
|
|
20
|
-
- ✅ **纯后端专注**:无前端,专注后端 CRUD 代码生成
|
|
21
|
-
- ✅ **包名适配**:`org.dromara.*`
|
|
22
|
-
- ✅ **智能推断**:模块 → 表前缀 → 包名 → 图标自动识别
|
|
23
|
-
- ✅ **全自动配置**:代码生成器配置完整(gen_table + gen_table_column)
|
|
24
|
-
- ✅ **菜单权限**:自动生成完整的菜单和权限配置
|
|
35
|
+
拆解完成后,可按 tasks.md 中的任务逐个使用 /dev 开发。
|
|
25
36
|
|
|
26
|
-
|
|
37
|
+
如果确定直接开发,请回复"继续"。
|
|
38
|
+
```
|
|
39
|
+
|
|
40
|
+
> 注意:这是建议而非强制。用户回复"继续"后直接进入第一步。
|
|
41
|
+
|
|
42
|
+
---
|
|
27
43
|
|
|
28
44
|
### 第一步:需求确认
|
|
29
45
|
|
|
46
|
+
询问用户:
|
|
47
|
+
|
|
30
48
|
```
|
|
31
49
|
请告诉我要开发的功能:
|
|
32
50
|
|
|
33
|
-
1.
|
|
34
|
-
2.
|
|
51
|
+
1. 功能名称?(如:员工考勤管理、菜品分类管理)
|
|
52
|
+
2. 所属模块?(sys-canteen / sys-kitchen / sys-drp / sys-common)
|
|
53
|
+
3. 端类型?(web管理端 / mobile移动端 / android设备端 / open开放接口)
|
|
35
54
|
```
|
|
36
55
|
|
|
37
|
-
|
|
56
|
+
根据所属模块确定包名:
|
|
38
57
|
|
|
39
|
-
| 模块 |
|
|
40
|
-
|
|
41
|
-
|
|
|
42
|
-
|
|
|
43
|
-
|
|
|
58
|
+
| 模块 | 包名前缀 | 路由前缀 |
|
|
59
|
+
|------|---------|---------|
|
|
60
|
+
| sys-canteen | `net.xnzn.core.canteen` | `/api/v2/web/canteen` |
|
|
61
|
+
| sys-kitchen | `net.xnzn.core.kitchen` | `/api/v2/web/kitchen` |
|
|
62
|
+
| sys-drp | `net.xnzn.core.drp` | `/api/v2/web/drp` |
|
|
63
|
+
| sys-common | `net.xnzn.core.common` | `/api/v2/web/common` |
|
|
44
64
|
|
|
45
65
|
---
|
|
46
66
|
|
|
47
|
-
###
|
|
48
|
-
|
|
49
|
-
**⚠️ 重要**:检查功能是否已存在,避免重复开发
|
|
67
|
+
### 第二步:功能重复检查(强制执行)
|
|
50
68
|
|
|
51
69
|
```bash
|
|
52
|
-
#
|
|
53
|
-
Grep pattern: "[功能名]
|
|
54
|
-
Grep pattern: "[功能名]Controller" path: ruoyi-modules/ output_mode: files_with_matches
|
|
55
|
-
|
|
56
|
-
# 2. 检查数据库表
|
|
57
|
-
SHOW TABLES LIKE '[表前缀]%';
|
|
70
|
+
# 检查是否已有相同 Controller
|
|
71
|
+
Grep pattern: "[功能名]Controller" path: [模块目录]/src/main/java output_mode: files_with_matches
|
|
58
72
|
|
|
59
|
-
#
|
|
60
|
-
|
|
73
|
+
# 检查是否已有相同 Service
|
|
74
|
+
Grep pattern: "[功能名]Service" path: [模块目录]/src/main/java output_mode: files_with_matches
|
|
61
75
|
```
|
|
62
76
|
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
### 第三步:数据库现状分析(自动执行)
|
|
66
|
-
|
|
67
|
-
从 `ruoyi-admin/src/main/resources/application-dev.yml` 动态读取数据库配置。
|
|
77
|
+
- ✅ 未存在 → 继续开发
|
|
78
|
+
- ⚠️ 已存在 → 停止,提示修改现有代码
|
|
68
79
|
|
|
69
80
|
---
|
|
70
81
|
|
|
71
|
-
###
|
|
82
|
+
### 第三步:数据库表设计
|
|
72
83
|
|
|
73
|
-
####
|
|
84
|
+
#### 3.1 智能字段命名推断
|
|
74
85
|
|
|
75
|
-
| 字段后缀 |
|
|
76
|
-
|
|
77
|
-
| `xxx_name`
|
|
78
|
-
| `
|
|
79
|
-
| `
|
|
80
|
-
| `
|
|
81
|
-
| `
|
|
82
|
-
| `is_xxx` |
|
|
83
|
-
| `xxx_amount` / `xxx_price` | 金额 | numberInput | EQ |
|
|
84
|
-
| `xxx_time` / `xxx_date` | 时间 | datetime | BETWEEN |
|
|
86
|
+
| 字段后缀 | Java 类型 | 查询方式 | 说明 |
|
|
87
|
+
|---------|---------|---------|------|
|
|
88
|
+
| `xxx_name` / `xxx_title` | String | LIKE | 名称/标题(模糊搜索)|
|
|
89
|
+
| `xxx_type` / `xxx_status` | Integer | EQ | 类型/状态(精确匹配)|
|
|
90
|
+
| `xxx_time` / `xxx_date` | LocalDateTime | BETWEEN | 时间范围查询 |
|
|
91
|
+
| `xxx_amount` / `xxx_price` | Long | EQ | 金额(分为单位)|
|
|
92
|
+
| `remark` | String | - | 备注 |
|
|
93
|
+
| `is_xxx` | Integer | EQ | 布尔标志(0/1)|
|
|
85
94
|
|
|
86
|
-
####
|
|
95
|
+
#### 3.2 建表 SQL 模板(leniu 规范)
|
|
87
96
|
|
|
88
97
|
```sql
|
|
89
|
-
CREATE TABLE [
|
|
90
|
-
id
|
|
91
|
-
tenant_id VARCHAR(20) DEFAULT '000000' COMMENT '租户ID',
|
|
98
|
+
CREATE TABLE `[表名]` (
|
|
99
|
+
`id` BIGINT NOT NULL COMMENT '主键(雪花ID)',
|
|
92
100
|
|
|
93
101
|
-- 业务字段
|
|
94
|
-
xxx_name
|
|
95
|
-
xxx_type
|
|
96
|
-
status
|
|
97
|
-
|
|
98
|
-
--
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
102
|
+
`xxx_name` VARCHAR(100) NOT NULL COMMENT '名称',
|
|
103
|
+
`xxx_type` TINYINT DEFAULT 1 COMMENT '类型(1-xxx,2-xxx)',
|
|
104
|
+
`status` TINYINT DEFAULT 1 COMMENT '状态(0停用 1启用)',
|
|
105
|
+
|
|
106
|
+
-- 审计字段(注意:leniu 规范,不是 createBy/createTime)
|
|
107
|
+
`crby` VARCHAR(64) DEFAULT NULL COMMENT '创建人',
|
|
108
|
+
`crtime` DATETIME DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
|
|
109
|
+
`upby` VARCHAR(64) DEFAULT NULL COMMENT '更新人',
|
|
110
|
+
`uptime` DATETIME DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',
|
|
111
|
+
|
|
112
|
+
-- 逻辑删除(注意:leniu 规范 2=正常,1=删除,与 RuoYi 相反)
|
|
113
|
+
`del_flag` TINYINT DEFAULT 2 COMMENT '删除标识(1删除 2正常)',
|
|
114
|
+
|
|
115
|
+
PRIMARY KEY (`id`),
|
|
116
|
+
KEY `idx_status` (`status`),
|
|
117
|
+
KEY `idx_crtime` (`crtime`),
|
|
118
|
+
KEY `idx_del_flag` (`del_flag`)
|
|
119
|
+
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='[功能说明]表';
|
|
108
120
|
```
|
|
109
121
|
|
|
122
|
+
**⚠️ 重要规范**:
|
|
123
|
+
- `id` 使用雪花ID(`Id.next()`),不使用 `AUTO_INCREMENT`
|
|
124
|
+
- `del_flag`:`2=正常,1=删除`(与 RuoYi 的 `0=正常` 相反)
|
|
125
|
+
- 无需 `tenant_id` 字段(双库物理隔离)
|
|
126
|
+
- 无需 `create_dept` 字段(leniu 不使用)
|
|
127
|
+
|
|
110
128
|
---
|
|
111
129
|
|
|
112
|
-
###
|
|
130
|
+
### 第四步:生成方案确认(仅此一次)
|
|
113
131
|
|
|
114
132
|
```markdown
|
|
115
|
-
##
|
|
133
|
+
## 代码生成方案
|
|
116
134
|
|
|
117
135
|
### 基本配置
|
|
118
|
-
-
|
|
119
|
-
- **模块**:
|
|
120
|
-
- **表名**:
|
|
121
|
-
- **Java
|
|
122
|
-
- **包名**:
|
|
123
|
-
-
|
|
124
|
-
|
|
125
|
-
###
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
136
|
+
- **功能名称**:XXX 管理
|
|
137
|
+
- **模块**:sys-canteen
|
|
138
|
+
- **表名**:canteen_xxx
|
|
139
|
+
- **Java 类名前缀**:Xxx
|
|
140
|
+
- **包名**:net.xnzn.core.canteen.xxx
|
|
141
|
+
- **接口路由**:/api/v2/web/canteen/xxx
|
|
142
|
+
|
|
143
|
+
### 文件清单(四层架构)
|
|
144
|
+
| 层 | 文件 | 说明 |
|
|
145
|
+
|----|------|------|
|
|
146
|
+
| Controller | `web/controller/XxxWebController.java` | Web 端接口 |
|
|
147
|
+
| Business | `web/business/impl/XxxWebBusiness.java` | 业务编排 |
|
|
148
|
+
| Service | `common/service/impl/XxxService.java` | 单表 CRUD |
|
|
149
|
+
| Mapper | `common/mapper/XxxMapper.java` | ORM 映射 |
|
|
150
|
+
| Entity | `common/model/Xxx.java` | 实体类 |
|
|
151
|
+
| DTO | `web/dto/XxxDTO.java` | 请求参数 |
|
|
152
|
+
| VO | `web/vo/XxxVO.java` | 返回对象 |
|
|
153
|
+
|
|
154
|
+
**确认开始生成?**(回复"确认"或"开始")
|
|
131
155
|
```
|
|
132
156
|
|
|
133
157
|
---
|
|
134
158
|
|
|
135
|
-
###
|
|
159
|
+
### 第五步:自动生成代码
|
|
160
|
+
|
|
161
|
+
用户确认后,按以下顺序生成所有文件:
|
|
162
|
+
|
|
163
|
+
#### 5.1 Entity 实体类
|
|
164
|
+
|
|
165
|
+
```java
|
|
166
|
+
package net.xnzn.core.[模块].[功能].common.model;
|
|
167
|
+
|
|
168
|
+
import com.baomidou.mybatisplus.annotation.*;
|
|
169
|
+
import io.swagger.annotations.ApiModel;
|
|
170
|
+
import io.swagger.annotations.ApiModelProperty;
|
|
171
|
+
import lombok.Data;
|
|
172
|
+
import lombok.experimental.Accessors;
|
|
173
|
+
import java.io.Serializable;
|
|
174
|
+
import java.time.LocalDateTime;
|
|
175
|
+
|
|
176
|
+
/**
|
|
177
|
+
* [功能名称] 实体
|
|
178
|
+
*
|
|
179
|
+
* @author [作者]
|
|
180
|
+
* @date [日期]
|
|
181
|
+
*/
|
|
182
|
+
@Data
|
|
183
|
+
@Accessors(chain = true)
|
|
184
|
+
@TableName("[表名]")
|
|
185
|
+
@ApiModel("[功能名称]")
|
|
186
|
+
public class [实体名] implements Serializable {
|
|
187
|
+
|
|
188
|
+
@ApiModelProperty("主键ID")
|
|
189
|
+
@TableId(value = "id", type = IdType.INPUT)
|
|
190
|
+
private Long id;
|
|
191
|
+
|
|
192
|
+
@ApiModelProperty("名称")
|
|
193
|
+
@TableField("xxx_name")
|
|
194
|
+
private String xxxName;
|
|
195
|
+
|
|
196
|
+
// 审计字段
|
|
197
|
+
@ApiModelProperty("删除标识(1删除,2正常)")
|
|
198
|
+
@TableField("del_flag")
|
|
199
|
+
private Integer delFlag;
|
|
200
|
+
|
|
201
|
+
@ApiModelProperty("创建人")
|
|
202
|
+
@TableField(value = "crby", fill = FieldFill.INSERT)
|
|
203
|
+
private String crby;
|
|
204
|
+
|
|
205
|
+
@ApiModelProperty("创建时间")
|
|
206
|
+
@TableField(value = "crtime", fill = FieldFill.INSERT)
|
|
207
|
+
private LocalDateTime crtime;
|
|
208
|
+
|
|
209
|
+
@ApiModelProperty("更新人")
|
|
210
|
+
@TableField(value = "upby", fill = FieldFill.INSERT_UPDATE)
|
|
211
|
+
private String upby;
|
|
212
|
+
|
|
213
|
+
@ApiModelProperty("更新时间")
|
|
214
|
+
@TableField(value = "uptime", fill = FieldFill.INSERT_UPDATE)
|
|
215
|
+
private LocalDateTime uptime;
|
|
216
|
+
}
|
|
217
|
+
```
|
|
218
|
+
|
|
219
|
+
#### 5.2 DTO 请求参数
|
|
220
|
+
|
|
221
|
+
```java
|
|
222
|
+
package net.xnzn.core.[模块].[功能].web.dto;
|
|
223
|
+
|
|
224
|
+
import io.swagger.annotations.ApiModel;
|
|
225
|
+
import io.swagger.annotations.ApiModelProperty;
|
|
226
|
+
import lombok.Data;
|
|
227
|
+
import jakarta.validation.constraints.*;
|
|
228
|
+
import java.io.Serializable;
|
|
229
|
+
|
|
230
|
+
@Data
|
|
231
|
+
@ApiModel("[功能名称] 请求参数")
|
|
232
|
+
public class [实体名]DTO implements Serializable {
|
|
233
|
+
|
|
234
|
+
@ApiModelProperty("主键ID(修改时必填)")
|
|
235
|
+
@NotNull(message = "ID不能为空", groups = {UpdateGroup.class})
|
|
236
|
+
private Long id;
|
|
237
|
+
|
|
238
|
+
@ApiModelProperty("名称")
|
|
239
|
+
@NotBlank(message = "名称不能为空", groups = {InsertGroup.class, UpdateGroup.class})
|
|
240
|
+
@Size(max = 100, message = "名称不能超过100个字符")
|
|
241
|
+
private String xxxName;
|
|
242
|
+
|
|
243
|
+
// 分页参数(查询接口使用)
|
|
244
|
+
@ApiModelProperty("页码")
|
|
245
|
+
private Integer pageNum = 1;
|
|
246
|
+
|
|
247
|
+
@ApiModelProperty("每页条数")
|
|
248
|
+
private Integer pageSize = 10;
|
|
249
|
+
}
|
|
250
|
+
```
|
|
251
|
+
|
|
252
|
+
#### 5.3 VO 返回对象
|
|
136
253
|
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
254
|
+
```java
|
|
255
|
+
package net.xnzn.core.[模块].[功能].web.vo;
|
|
256
|
+
|
|
257
|
+
import com.fasterxml.jackson.annotation.JsonFormat;
|
|
258
|
+
import io.swagger.annotations.ApiModel;
|
|
259
|
+
import io.swagger.annotations.ApiModelProperty;
|
|
260
|
+
import lombok.Data;
|
|
261
|
+
import java.io.Serializable;
|
|
262
|
+
import java.time.LocalDateTime;
|
|
263
|
+
|
|
264
|
+
@Data
|
|
265
|
+
@ApiModel("[功能名称] 返回对象")
|
|
266
|
+
public class [实体名]VO implements Serializable {
|
|
267
|
+
|
|
268
|
+
@ApiModelProperty("主键ID")
|
|
269
|
+
private Long id;
|
|
270
|
+
|
|
271
|
+
@ApiModelProperty("名称")
|
|
272
|
+
private String xxxName;
|
|
273
|
+
|
|
274
|
+
@ApiModelProperty("创建时间")
|
|
275
|
+
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
|
|
276
|
+
private LocalDateTime crtime;
|
|
277
|
+
}
|
|
278
|
+
```
|
|
279
|
+
|
|
280
|
+
#### 5.4 Mapper 接口(与 XML 同目录)
|
|
281
|
+
|
|
282
|
+
```java
|
|
283
|
+
package net.xnzn.core.[模块].[功能].common.mapper;
|
|
284
|
+
|
|
285
|
+
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
|
|
286
|
+
import net.xnzn.core.[模块].[功能].common.model.[实体名];
|
|
287
|
+
import org.apache.ibatis.annotations.Mapper;
|
|
288
|
+
|
|
289
|
+
/**
|
|
290
|
+
* [功能名称] Mapper
|
|
291
|
+
*/
|
|
292
|
+
@Mapper
|
|
293
|
+
public interface [实体名]Mapper extends BaseMapper<[实体名]> {
|
|
294
|
+
}
|
|
295
|
+
```
|
|
296
|
+
|
|
297
|
+
#### 5.5 Service 实现
|
|
298
|
+
|
|
299
|
+
```java
|
|
300
|
+
package net.xnzn.core.[模块].[功能].common.service.impl;
|
|
301
|
+
|
|
302
|
+
import cn.hutool.core.bean.BeanUtil;
|
|
303
|
+
import cn.hutool.core.util.ObjectUtil;
|
|
304
|
+
import cn.hutool.core.util.StrUtil;
|
|
305
|
+
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
|
306
|
+
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
|
|
307
|
+
import com.pig4cloud.pigx.common.core.exception.LeException;
|
|
308
|
+
import lombok.extern.slf4j.Slf4j;
|
|
309
|
+
import net.xnzn.core.[模块].[功能].web.dto.[实体名]DTO;
|
|
310
|
+
import net.xnzn.core.[模块].[功能].common.mapper.[实体名]Mapper;
|
|
311
|
+
import net.xnzn.core.[模块].[功能].common.model.[实体名];
|
|
312
|
+
import net.xnzn.core.[模块].[功能].web.vo.[实体名]VO;
|
|
313
|
+
import net.xnzn.framework.id.Id;
|
|
314
|
+
import org.springframework.stereotype.Service;
|
|
315
|
+
import org.springframework.transaction.annotation.Transactional;
|
|
316
|
+
|
|
317
|
+
import javax.annotation.Resource;
|
|
318
|
+
import java.util.Collections;
|
|
319
|
+
import java.util.List;
|
|
320
|
+
import java.util.Optional;
|
|
321
|
+
|
|
322
|
+
@Slf4j
|
|
323
|
+
@Service
|
|
324
|
+
public class [实体名]Service {
|
|
325
|
+
|
|
326
|
+
@Resource
|
|
327
|
+
private [实体名]Mapper [实体名小写]Mapper;
|
|
328
|
+
|
|
329
|
+
@Transactional(rollbackFor = Exception.class)
|
|
330
|
+
public Long add([实体名]DTO dto) {
|
|
331
|
+
[实体名] entity = BeanUtil.copyProperties(dto, [实体名].class);
|
|
332
|
+
entity.setId(Id.next());
|
|
333
|
+
entity.setDelFlag(2); // 2=正常
|
|
334
|
+
[实体名小写]Mapper.insert(entity);
|
|
335
|
+
return entity.getId();
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
@Transactional(rollbackFor = Exception.class)
|
|
339
|
+
public void update([实体名]DTO dto) {
|
|
340
|
+
Optional.ofNullable([实体名小写]Mapper.selectById(dto.getId()))
|
|
341
|
+
.orElseThrow(() -> new LeException("记录不存在"));
|
|
342
|
+
[实体名] entity = BeanUtil.copyProperties(dto, [实体名].class);
|
|
343
|
+
[实体名小写]Mapper.updateById(entity);
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
@Transactional(rollbackFor = Exception.class)
|
|
347
|
+
public void delete(Long id) {
|
|
348
|
+
Optional.ofNullable([实体名小写]Mapper.selectById(id))
|
|
349
|
+
.orElseThrow(() -> new LeException("记录不存在"));
|
|
350
|
+
[实体名] entity = new [实体名]().setId(id).setDelFlag(1); // 1=删除
|
|
351
|
+
[实体名小写]Mapper.updateById(entity);
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
public [实体名]VO getById(Long id) {
|
|
355
|
+
[实体名] entity = Optional.ofNullable([实体名小写]Mapper.selectById(id))
|
|
356
|
+
.orElseThrow(() -> new LeException("记录不存在"));
|
|
357
|
+
return BeanUtil.copyProperties(entity, [实体名]VO.class);
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
public List<[实体名]VO> list([实体名]DTO dto) {
|
|
361
|
+
LambdaQueryWrapper<[实体名]> wrapper = buildWrapper(dto);
|
|
362
|
+
List<[实体名]> list = [实体名小写]Mapper.selectList(wrapper);
|
|
363
|
+
if (list == null || list.isEmpty()) {
|
|
364
|
+
return Collections.emptyList();
|
|
365
|
+
}
|
|
366
|
+
return BeanUtil.copyToList(list, [实体名]VO.class);
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
private LambdaQueryWrapper<[实体名]> buildWrapper([实体名]DTO dto) {
|
|
370
|
+
LambdaQueryWrapper<[实体名]> wrapper = Wrappers.lambdaQuery();
|
|
371
|
+
wrapper.eq([实体名]::getDelFlag, 2); // 只查正常数据
|
|
372
|
+
if (StrUtil.isNotBlank(dto.getXxxName())) {
|
|
373
|
+
wrapper.like([实体名]::getXxxName, dto.getXxxName());
|
|
374
|
+
}
|
|
375
|
+
wrapper.orderByDesc([实体名]::getCrtime);
|
|
376
|
+
return wrapper;
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
```
|
|
380
|
+
|
|
381
|
+
#### 5.6 Business 业务层
|
|
382
|
+
|
|
383
|
+
```java
|
|
384
|
+
package net.xnzn.core.[模块].[功能].web.business.impl;
|
|
385
|
+
|
|
386
|
+
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
|
|
387
|
+
import com.github.pagehelper.page.PageMethod;
|
|
388
|
+
import net.xnzn.core.[模块].[功能].web.dto.[实体名]DTO;
|
|
389
|
+
import net.xnzn.core.[模块].[功能].common.service.impl.[实体名]Service;
|
|
390
|
+
import net.xnzn.core.[模块].[功能].web.vo.[实体名]VO;
|
|
391
|
+
import lombok.extern.slf4j.Slf4j;
|
|
392
|
+
import org.springframework.stereotype.Service;
|
|
393
|
+
|
|
394
|
+
import javax.annotation.Resource;
|
|
395
|
+
import java.util.List;
|
|
396
|
+
|
|
397
|
+
@Slf4j
|
|
398
|
+
@Service
|
|
399
|
+
public class [实体名]WebBusiness {
|
|
400
|
+
|
|
401
|
+
@Resource
|
|
402
|
+
private [实体名]Service [实体名小写]Service;
|
|
403
|
+
|
|
404
|
+
public Long add([实体名]DTO dto) {
|
|
405
|
+
return [实体名小写]Service.add(dto);
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
public void update([实体名]DTO dto) {
|
|
409
|
+
[实体名小写]Service.update(dto);
|
|
410
|
+
}
|
|
411
|
+
|
|
412
|
+
public void delete(Long id) {
|
|
413
|
+
[实体名小写]Service.delete(id);
|
|
414
|
+
}
|
|
415
|
+
|
|
416
|
+
public [实体名]VO getById(Long id) {
|
|
417
|
+
return [实体名小写]Service.getById(id);
|
|
418
|
+
}
|
|
419
|
+
|
|
420
|
+
public List<[实体名]VO> page([实体名]DTO dto) {
|
|
421
|
+
PageMethod.startPage(dto.getPageNum(), dto.getPageSize());
|
|
422
|
+
return [实体名小写]Service.list(dto);
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
```
|
|
426
|
+
|
|
427
|
+
#### 5.7 Controller 接口层
|
|
428
|
+
|
|
429
|
+
```java
|
|
430
|
+
package net.xnzn.core.[模块].[功能].web.controller;
|
|
431
|
+
|
|
432
|
+
import com.pig4cloud.pigx.common.core.util.LeRequest;
|
|
433
|
+
import io.swagger.annotations.Api;
|
|
434
|
+
import io.swagger.annotations.ApiOperation;
|
|
435
|
+
import lombok.extern.slf4j.Slf4j;
|
|
436
|
+
import net.xnzn.core.[模块].[功能].web.business.impl.[实体名]WebBusiness;
|
|
437
|
+
import net.xnzn.core.[模块].[功能].web.dto.[实体名]DTO;
|
|
438
|
+
import net.xnzn.core.[模块].[功能].web.vo.[实体名]VO;
|
|
439
|
+
import net.xnzn.framework.secure.filter.annotation.RequiresAuthentication;
|
|
440
|
+
import org.springframework.validation.annotation.Validated;
|
|
441
|
+
import org.springframework.web.bind.annotation.*;
|
|
442
|
+
|
|
443
|
+
import javax.annotation.Resource;
|
|
444
|
+
import java.util.List;
|
|
445
|
+
|
|
446
|
+
@Slf4j
|
|
447
|
+
@RestController
|
|
448
|
+
@RequestMapping("/api/v2/web/[模块]/[功能]")
|
|
449
|
+
@Api(tags = "[功能名称]管理")
|
|
450
|
+
public class [实体名]WebController {
|
|
451
|
+
|
|
452
|
+
@Resource
|
|
453
|
+
private [实体名]WebBusiness [实体名小写]Business;
|
|
454
|
+
|
|
455
|
+
@PostMapping("/add")
|
|
456
|
+
@ApiOperation("新增")
|
|
457
|
+
@RequiresAuthentication
|
|
458
|
+
public Long add(@Validated(InsertGroup.class) @RequestBody LeRequest<[实体名]DTO> request) {
|
|
459
|
+
return [实体名小写]Business.add(request.getContent());
|
|
460
|
+
}
|
|
461
|
+
|
|
462
|
+
@PostMapping("/update")
|
|
463
|
+
@ApiOperation("修改")
|
|
464
|
+
@RequiresAuthentication
|
|
465
|
+
public void update(@Validated(UpdateGroup.class) @RequestBody LeRequest<[实体名]DTO> request) {
|
|
466
|
+
[实体名小写]Business.update(request.getContent());
|
|
467
|
+
}
|
|
468
|
+
|
|
469
|
+
@PostMapping("/delete")
|
|
470
|
+
@ApiOperation("删除")
|
|
471
|
+
@RequiresAuthentication
|
|
472
|
+
public void delete(@RequestBody LeRequest<Long> request) {
|
|
473
|
+
[实体名小写]Business.delete(request.getContent());
|
|
474
|
+
}
|
|
475
|
+
|
|
476
|
+
@GetMapping("/get/{id}")
|
|
477
|
+
@ApiOperation("获取详情")
|
|
478
|
+
@RequiresAuthentication
|
|
479
|
+
public [实体名]VO getById(@PathVariable Long id) {
|
|
480
|
+
return [实体名小写]Business.getById(id);
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
@PostMapping("/page")
|
|
484
|
+
@ApiOperation("分页查询")
|
|
485
|
+
@RequiresAuthentication
|
|
486
|
+
public List<[实体名]VO> page(@RequestBody LeRequest<[实体名]DTO> request) {
|
|
487
|
+
return [实体名小写]Business.page(request.getContent());
|
|
488
|
+
}
|
|
489
|
+
}
|
|
490
|
+
```
|
|
141
491
|
|
|
142
492
|
---
|
|
143
493
|
|
|
144
|
-
###
|
|
494
|
+
### 第六步:完成报告
|
|
145
495
|
|
|
146
496
|
```markdown
|
|
147
|
-
##
|
|
148
|
-
|
|
149
|
-
###
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
├── controller/business/AdController.java
|
|
168
|
-
├── domain/Ad.java
|
|
169
|
-
├── domain/bo/AdBo.java
|
|
170
|
-
├── domain/vo/AdVo.java
|
|
171
|
-
├── mapper/AdMapper.java
|
|
172
|
-
├── service/IAdService.java
|
|
173
|
-
└── service/impl/AdServiceImpl.java
|
|
174
|
-
\`\`\`
|
|
497
|
+
## 代码生成完成!
|
|
498
|
+
|
|
499
|
+
### 已生成文件(7 个)
|
|
500
|
+
|
|
501
|
+
| 层 | 文件 |
|
|
502
|
+
|----|------|
|
|
503
|
+
| Controller | web/controller/[实体名]WebController.java |
|
|
504
|
+
| Business | web/business/impl/[实体名]WebBusiness.java |
|
|
505
|
+
| Service | common/service/impl/[实体名]Service.java |
|
|
506
|
+
| Mapper | common/mapper/[实体名]Mapper.java |
|
|
507
|
+
| Entity | common/model/[实体名].java |
|
|
508
|
+
| DTO | web/dto/[实体名]DTO.java |
|
|
509
|
+
| VO | web/vo/[实体名]VO.java |
|
|
510
|
+
|
|
511
|
+
### 下一步
|
|
512
|
+
|
|
513
|
+
1. 执行建表 SQL
|
|
514
|
+
2. 根据业务需求完善 buildWrapper 查询条件
|
|
515
|
+
3. 如有复杂业务逻辑,在 Business 层添加编排
|
|
516
|
+
4. 使用 /check 检查代码规范
|
|
175
517
|
```
|
|
176
518
|
|
|
177
519
|
---
|
|
178
520
|
|
|
179
|
-
##
|
|
180
|
-
|
|
181
|
-
1. ✅
|
|
182
|
-
2. ✅
|
|
183
|
-
3. ✅
|
|
184
|
-
4. ✅ **
|
|
185
|
-
5. ✅
|
|
186
|
-
6. ✅
|
|
187
|
-
7. ✅
|
|
521
|
+
## 执行规则
|
|
522
|
+
|
|
523
|
+
1. ✅ **包名**:必须是 `net.xnzn.core.[模块].*`
|
|
524
|
+
2. ✅ **四层架构**:Controller → Business → Service → Mapper
|
|
525
|
+
3. ✅ **审计字段**:crby/crtime/upby/uptime(不是 createBy/createTime)
|
|
526
|
+
4. ✅ **del_flag**:`2=正常,1=删除`(不是 0=正常)
|
|
527
|
+
5. ✅ **无 tenant_id**:双库物理隔离,Entity 不含此字段
|
|
528
|
+
6. ✅ **LeRequest<T>**:POST 请求体统一封装
|
|
529
|
+
7. ✅ **BeanUtil**:对象转换用 `BeanUtil.copyProperties()`(不用 MapstructUtils)
|
|
530
|
+
8. ✅ **LeException**:异常用 `LeException`(不用 ServiceException)
|
|
531
|
+
9. ✅ **Id.next()**:主键用雪花 ID(不用 AUTO_INCREMENT)
|
|
532
|
+
10. ✅ **Mapper XML**:与 Java 文件放同一目录(不在 resources/mapper/)
|