dengzedong
2024-10-10 b45bad33154fb97b76e6c54a86609d446f02ad21
提交 | 用户 | 时间
9ec4bd 1 <template>
2   <Dialog v-model="dialogVisible" :title="dialogTitle">
3     <el-form
4       class="-mb-15px"
5       :model="formData"
6       ref="formRef"
7       :inline="true"
8       :rules="formRules"
9       label-width="68px"
10       v-loading="formLoading"
11     >
12       <el-form-item style="width: 100%">
13         <el-divider content-position="left">模型信息</el-divider>
14       </el-form-item>
15       <el-form-item label="全类名" style="width: 90%" prop="className">
16         <el-input v-model="formData.className" placeholder=""/>
17       </el-form-item>
18       <el-form-item label="方法名" prop="methodName">
19         <el-select v-model="formData.methodName" @change="methodChange" style="width: 240px">
20           <el-option
21             v-for="item in methodList"
22             :key="item.id"
23             :label="item.methodName"
24             :value="item.methodName"
25           />
26         </el-select>
27       </el-form-item>
28       <el-divider content-position="left">模型参数信息</el-divider>
29       <el-row :gutter="20">
30         <el-col :span="2" style="margin-bottom: 10px;margin-left: 20px">
31           <el-button tag="a" href="/template/模型参数导入模板.xlsx" download="模型参数导入模板.xlsx" style="text-decoration: none;" type="primary" size="small" link>模板下载</el-button>
32         </el-col>
33         <el-col :span="2" style="margin-bottom: 10px;">
34           <el-upload
35             ref="uploadRef"
36             v-model:file-list="fileList"
37             :show-file-list="false"
38             :action="importUrl"
39             :auto-upload="true"
40             :disabled="formLoading"
41             :before-upload="beforeUpload"
42             :headers="uploadHeaders"
43             :on-error="submitFormError"
44             :on-success="submitFormSuccess"
45             accept=".xlsx"
46           >
47             <el-button type="primary" size="small" link>参数导入</el-button>
48           </el-upload>
49         </el-col>
50       </el-row>
51       <el-row v-for="(item,index) in formData.datas" :key="index" :gutter="20">
52         <el-col :span="20">
53           <el-form-item :label="'参数_' + (index)" required style="width: 100%">
54             <el-input v-model="formData.datas[index]" placeholder="" />
55           </el-form-item>
56         </el-col>
57       </el-row>
58       <el-row v-if="hasModel" :gutter="20">
59         <el-col :span="20">
60           <el-form-item label="model" required style="width: 100%">
61             <el-input v-model="formData.model" placeholder="" />
62           </el-form-item>
63         </el-col>
64       </el-row>
65       <el-divider content-position="left">模型设置信息</el-divider>
66       <el-row :gutter="20">
67         <el-col :span="4">
68           <el-button type="primary" size="small" @click="addRow()">新增</el-button>
69         </el-col>
70       </el-row>
71       <el-table :data="formData.modelSettings" border>
72         <el-table-column
73           prop=""
74           label="参数key"
75           align="center">
76           <template #default="scope">
77             <el-input size="small" v-model="scope.row.settingKey" maxlength="50" clearable />
78           </template>
79         </el-table-column>
80         <el-table-column
81           prop=""
82           label="参数value"
83           align="center">
84           <template #default="scope">
85             <el-input size="small" v-model="scope.row.settingValue" maxlength="50" clearable />
86           </template>
87         </el-table-column>
88         <el-table-column label="操作" fixed="right" header-align="center" align="center" width="100">
89           <template #default="scope">
90             <el-button
91               @click="deleteRow(scope.$index)"
92               key="danger"
93               type="danger"
94               link
95             >删除</el-button>
96           </template>
97         </el-table-column>
98       </el-table>
99       <el-divider content-position="left">模型运行结果</el-divider>
100       <el-input v-model="modelRunResult" placeholder="" rows="4" type="textarea" />
101       <div style="display: flex;flex-direction: row;justify-content: end;margin-top: 16px">
102         <el-button :loading="modelRunloading" type="primary" @click="modelRun()">运行</el-button>
103       </div>
104     </el-form>
105   </Dialog>
106 </template>
107 <script lang="ts" setup>
aff5c9 108   import * as MpkApi from '@/api/model/mpk/mpk'
9ec4bd 109   import {FormRules} from "element-plus";
110   import {getAccessToken, getTenantId} from "@/utils/auth";
111
112   const { t } = useI18n() // 国际化
113   const message = useMessage() // 消息弹窗
114
115   const dialogVisible = ref(false) // 弹窗的是否展示
116   const dialogTitle = ref('模型运行') // 弹窗的标题
117
118   const formData = reactive({
2bff3e 119     pyName: '',
9ec4bd 120     className: '',
121     methodName: '',
122     datas: [],
123     modelSettings: [],
124     model: undefined
125   })
126
127   // 模型方法下拉列表
128   const methodList = ref([])
129   const hasModel = ref(false)
130
131   /** 打开弹窗 */
132   const open = async (row) => {
133     dialogVisible.value = true
134     formData.className = row.pkgName + '.impl.' + row.pyName + 'Impl';
2bff3e 135     formData.pyName = row.pyName;
9ec4bd 136     const mpk = await MpkApi.getMpk(row.id)
137     methodList.value = mpk.modelMethods
138     formData.methodName = mpk.modelMethods[0].methodName
139     formData.datas = []
140     for (let i = 0 ; i < mpk.modelMethods[0].dataLength ; i++) {
141      formData.datas[i] = '[[]]'
142     }
143     hasModel.value = mpk.modelMethods[0].model === 1
144   }
145   defineExpose({ open }) // 提供 open 方法,用于打开弹窗
146
147   const formRules = reactive<FormRules>({
148     methodName: [
149       {required: true, message: '方法名不能为空', trigger: 'blur'}
150     ],
151     className: [
152       {required: true, message: '全类名不能为空', trigger: 'blur'}
153     ]
154   })
155
156   const addRow = function () {
157     formData.modelSettings.push({
158       settingKey: '',
159       settingValue: ''
160     })
161   }
162   const deleteRow = function (index) {
163     formData.modelSettings.splice(index, 1)
164   }
165   const methodChange = function (value) {
166     formData.datas = []
167     for (let i = 0 ; i < methodList.value.find(e => e.methodName === value)?.dataLength ; i++) {
168       formData.datas[i] = '[[]]'
169     }
170     hasModel.value = methodList.value.find(e => e.methodName === value)?.model === 1
171   }
172
173   const fileList = ref([]) // 文件列表
174   const importUrl =
175     import.meta.env.VITE_BASE_URL + import.meta.env.VITE_API_URL + '/model/mpk/api/import'
176   const formLoading = ref(false) // 表单的加载中
177   const uploadHeaders = ref() // 上传 Header 头
178   /** 上传错误提示 */
179   const submitFormError = (): void => {
180     message.error('导入失败,请检查导入文件!')
181     formLoading.value = false
182   }
183   const submitFormSuccess = (response: any) => {
184     if (response.code !== 0) {
185       message.error(response.msg)
186       formLoading.value = false
187       return
188     }
189     const datas = response.data;
190     for (let i=0;i<formData.datas.length;i++) {
191       formData.datas[i] = datas[i]
192     }
193     message.success('导入成功')
194     formLoading.value = false
195   }
196   const beforeUpload = function (file) {
197     // 提交请求
198     uploadHeaders.value = {
199       Authorization: 'Bearer ' + getAccessToken(),
200       'tenant-id': getTenantId()
201     }
202     formLoading.value = true
203     return true;
204   }
205
206   // 模型运行结果
207   const modelRunResult = ref('')
208   // 模型运行loading
209   const modelRunloading = ref(false)
210   // 表单 Ref
211   const formRef = ref()
212   // 运行
213   const modelRun = async () => {
214 // 校验表单
215     if (!formRef) return
216     const valid = await formRef.value.validate()
217     if (!valid) return
218     // 提交请求
219     modelRunloading.value = true
220     try {
221       const data = {
222         ...formData
223       }
224
225       //处理modelSettings
226       let settingsPredict = {};
227       data.modelSettings.forEach(e => {
228         settingsPredict[e.settingKey] = e.settingValue;
229       })
230       data.modelSettings = settingsPredict
231       data.hasModel = hasModel.value
232       if (data.hasModel && data.model) {
233         data.model = {model_path:data.model}
234       }else {
235         data.model = undefined
236       }
237
238       modelRunResult.value = await MpkApi.modelRun(data)
239       modelRunloading.value = false
240       message.success('运行成功')
241     } finally {
242       modelRunloading.value = false
243     }
244   }
245 </script>