mpk
dengzedong
2024-09-12 9c91ad8087da1707b973173ebc6d7c3ac7d89195
提交 | 用户 | 时间
9c91ad 1 <template>
D 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>
108   import * as MpkApi from '@/api/mpk/mpk'
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({
119     className: '',
120     methodName: '',
121     datas: [],
122     modelSettings: [],
123     model: undefined
124   })
125
126   // 模型方法下拉列表
127   const methodList = ref([])
128   const hasModel = ref(false)
129
130   /** 打开弹窗 */
131   const open = async (row) => {
132     dialogVisible.value = true
133     formData.className = row.pkgName + '.impl.' + row.pyName + 'Impl';
134     const mpk = await MpkApi.getMpk(row.id)
135     methodList.value = mpk.modelMethods
136     formData.methodName = mpk.modelMethods[0].methodName
137     formData.datas = []
138     for (let i = 0 ; i < mpk.modelMethods[0].dataLength ; i++) {
139      formData.datas[i] = '[[]]'
140     }
141     hasModel.value = mpk.modelMethods[0].model === 1
142   }
143   defineExpose({ open }) // 提供 open 方法,用于打开弹窗
144
145   const formRules = reactive<FormRules>({
146     methodName: [
147       {required: true, message: '方法名不能为空', trigger: 'blur'}
148     ],
149     className: [
150       {required: true, message: '全类名不能为空', trigger: 'blur'}
151     ]
152   })
153
154   const addRow = function () {
155     formData.modelSettings.push({
156       settingKey: '',
157       settingValue: ''
158     })
159   }
160   const deleteRow = function (index) {
161     formData.modelSettings.splice(index, 1)
162   }
163   const methodChange = function (value) {
164     formData.datas = []
165     for (let i = 0 ; i < methodList.value.find(e => e.methodName === value)?.dataLength ; i++) {
166       formData.datas[i] = '[[]]'
167     }
168     hasModel.value = methodList.value.find(e => e.methodName === value)?.model === 1
169   }
170
171   const fileList = ref([]) // 文件列表
172   const importUrl =
173     import.meta.env.VITE_BASE_URL + import.meta.env.VITE_API_URL + '/model/mpk/api/import'
174   const formLoading = ref(false) // 表单的加载中
175   const uploadHeaders = ref() // 上传 Header 头
176   /** 上传错误提示 */
177   const submitFormError = (): void => {
178     message.error('导入失败,请检查导入文件!')
179     formLoading.value = false
180   }
181   const submitFormSuccess = (response: any) => {
182     if (response.code !== 0) {
183       message.error(response.msg)
184       formLoading.value = false
185       return
186     }
187     const datas = response.data;
188     for (let i=0;i<formData.datas.length;i++) {
189       formData.datas[i] = datas[i]
190     }
191     message.success('导入成功')
192     formLoading.value = false
193   }
194   const beforeUpload = function (file) {
195     // 提交请求
196     uploadHeaders.value = {
197       Authorization: 'Bearer ' + getAccessToken(),
198       'tenant-id': getTenantId()
199     }
200     formLoading.value = true
201     return true;
202   }
203
204   // 模型运行结果
205   const modelRunResult = ref('')
206   // 模型运行loading
207   const modelRunloading = ref(false)
208   // 表单 Ref
209   const formRef = ref()
210   // 运行
211   const modelRun = async () => {
212 // 校验表单
213     if (!formRef) return
214     const valid = await formRef.value.validate()
215     if (!valid) return
216     // 提交请求
217     modelRunloading.value = true
218     try {
219       const data = {
220         ...formData
221       }
222
223       //处理modelSettings
224       let settingsPredict = {};
225       data.modelSettings.forEach(e => {
226         settingsPredict[e.settingKey] = e.settingValue;
227       })
228       data.modelSettings = settingsPredict
229       data.hasModel = hasModel.value
230       if (data.hasModel && data.model) {
231         data.model = {model_path:data.model}
232       }else {
233         data.model = undefined
234       }
235
236       modelRunResult.value = await MpkApi.modelRun(data)
237       modelRunloading.value = false
238       message.success('运行成功')
239     } finally {
240       modelRunloading.value = false
241     }
242   }
243 </script>