boss 3 dagar sedan
förälder
incheckning
08819b6a06

+ 81 - 0
src/api/system/smsPort.js

@@ -0,0 +1,81 @@
+import request from '@/utils/request'
+
+// ========== 端口池 ==========
+export function listPort(query) {
+  return request({ url: '/admin/smsPort/port/list', method: 'get', params: query })
+}
+
+export function listPortByApi(apiId) {
+  return request({ url: '/admin/smsPort/port/listByApi/' + apiId, method: 'get' })
+}
+
+export function addPort(data) {
+  return request({ url: '/admin/smsPort/port', method: 'post', data })
+}
+
+export function updatePort(data) {
+  return request({ url: '/admin/smsPort/port', method: 'put', data })
+}
+
+export function delPort(portId) {
+  return request({ url: '/admin/smsPort/port/' + portId, method: 'delete' })
+}
+
+// ========== 端口分配 ==========
+export function listAssign(query) {
+  return request({ url: '/admin/smsPort/assign/list', method: 'get', params: query })
+}
+
+export function addAssign(data) {
+  return request({ url: '/admin/smsPort/assign', method: 'post', data })
+}
+
+export function updateAssign(data) {
+  return request({ url: '/admin/smsPort/assign', method: 'put', data })
+}
+
+export function delAssign(id) {
+  return request({ url: '/admin/smsPort/assign/' + id, method: 'delete' })
+}
+
+// ========== 手机卡管理 ==========
+export function listCard(query) {
+  return request({ url: '/admin/smsPort/card/list', method: 'get', params: query })
+}
+
+export function getCard(cardId) {
+  return request({ url: '/admin/smsPort/card/' + cardId, method: 'get' })
+}
+
+export function addCard(data) {
+  return request({ url: '/admin/smsPort/card', method: 'post', data })
+}
+
+export function updateCard(data) {
+  return request({ url: '/admin/smsPort/card', method: 'put', data })
+}
+
+export function delCard(cardId) {
+  return request({ url: '/admin/smsPort/card/' + cardId, method: 'delete' })
+}
+
+// ========== 中间件 ==========
+export function listMiddleware(query) {
+  return request({ url: '/admin/smsPort/middleware/list', method: 'get', params: query })
+}
+
+export function getMiddlewareByApi(apiId) {
+  return request({ url: '/admin/smsPort/middleware/byApi/' + apiId, method: 'get' })
+}
+
+export function addMiddleware(data) {
+  return request({ url: '/admin/smsPort/middleware', method: 'post', data })
+}
+
+export function updateMiddleware(data) {
+  return request({ url: '/admin/smsPort/middleware', method: 'put', data })
+}
+
+export function delMiddleware(id) {
+  return request({ url: '/admin/smsPort/middleware/' + id, method: 'delete' })
+}

+ 171 - 19
src/views/admin/smsApi/index.vue

@@ -4,7 +4,7 @@
       <el-form-item label="接口名称" prop="apiName">
         <el-input v-model="queryParams.apiName" placeholder="请输入接口名称" clearable @keyup.enter.native="handleQuery" />
       </el-form-item>
-      <el-form-item label="短信类型" prop="smsType">
+      <el-form-item label="发送类型" prop="smsType">
         <el-select v-model="queryParams.smsType" placeholder="请选择" clearable>
           <el-option v-for="item in smsTypeOptions" :key="item.value" :label="item.label" :value="item.value" />
         </el-select>
@@ -13,6 +13,7 @@
         <el-select v-model="queryParams.provider" placeholder="请选择" clearable>
           <el-option label="润方" value="rf" />
           <el-option label="德华" value="dh" />
+          <el-option label="手机卡" value="card" />
         </el-select>
       </el-form-item>
       <el-form-item>
@@ -27,10 +28,47 @@
       </el-col>
     </el-row>
 
-    <el-table v-loading="loading" :data="apiList" border size="small">
+    <el-table v-loading="loading" :data="apiList" border size="small" @expand-change="handleExpand">
+      <el-table-column type="expand">
+        <template slot-scope="props">
+          <div style="padding:10px 20px">
+            <el-row :gutter="10" class="mb8">
+              <el-col :span="1.5">
+                <el-button type="primary" plain icon="el-icon-plus" size="mini" @click="handleAddPort(props.row)">新增端口</el-button>
+              </el-col>
+            </el-row>
+            <el-table :data="portMap[props.row.apiId] || []" border size="mini" v-loading="portLoading">
+              <el-table-column label="端口ID" align="center" prop="portId" width="70" />
+              <el-table-column label="端口名称" align="center" prop="portName" min-width="120" />
+              <el-table-column label="端口号" align="center" prop="portNo" min-width="100">
+                <template slot-scope="scope">
+                  <span v-if="props.row.provider === 'card'">{{ scope.row.portNo }}</span>
+                  <span v-else>{{ scope.row.portNo }}</span>
+                </template>
+              </el-table-column>
+              <el-table-column label="独立账户" align="center" prop="account" min-width="100" show-overflow-tooltip />
+              <el-table-column label="独立签名" align="center" prop="sign" width="80" />
+              <el-table-column v-if="props.row.provider === 'card'" label="卡槽" align="center" prop="slotIndex" width="60">
+                <template slot-scope="scope">{{ scope.row.slotIndex || '-' }}</template>
+              </el-table-column>
+              <el-table-column label="状态" align="center" prop="status" width="70">
+                <template slot-scope="scope">
+                  <el-tag :type="scope.row.status === 1 ? 'success' : 'danger'" size="mini">{{ scope.row.status === 1 ? '正常' : '禁用' }}</el-tag>
+                </template>
+              </el-table-column>
+              <el-table-column label="操作" align="center" width="120">
+                <template slot-scope="scope">
+                  <el-button size="mini" type="text" @click="handleEditPort(scope.row, props.row)">修改</el-button>
+                  <el-button size="mini" type="text" style="color:#f56c6c" @click="handleDelPort(scope.row)">删除</el-button>
+                </template>
+              </el-table-column>
+            </el-table>
+          </div>
+        </template>
+      </el-table-column>
       <el-table-column label="ID" align="center" prop="apiId" width="60" />
       <el-table-column label="接口名称" align="center" prop="apiName" min-width="140" />
-      <el-table-column label="短信类型" align="center" prop="smsType" width="140">
+      <el-table-column label="发送类型" align="center" prop="smsType" width="140">
         <template slot-scope="scope">
           <el-tag :type="scope.row.smsType === 1 ? 'success' : scope.row.smsType === 2 ? 'warning' : 'danger'" size="small">
             {{ smsTypeFormat(scope.row.smsType) }}
@@ -38,7 +76,7 @@
         </template>
       </el-table-column>
       <el-table-column label="服务商" align="center" prop="provider" width="80">
-        <template slot-scope="scope">{{ scope.row.provider === 'rf' ? '润方' : '德华' }}</template>
+        <template slot-scope="scope">{{ providerFormat(scope.row.provider) }}</template>
       </el-table-column>
       <el-table-column label="账户名" align="center" prop="account" min-width="100" show-overflow-tooltip />
       <el-table-column label="签名" align="center" prop="sign" width="80" />
@@ -47,11 +85,6 @@
           <span style="color:#e6a23c;font-weight:bold">{{ scope.row.costPrice || '-' }}</span>
         </template>
       </el-table-column>
-      <el-table-column label="默认" align="center" prop="isDefault" width="60">
-        <template slot-scope="scope">
-          <el-tag v-if="scope.row.isDefault === 1" type="success" size="mini">是</el-tag>
-        </template>
-      </el-table-column>
       <el-table-column label="状态" align="center" prop="status" width="70">
         <template slot-scope="scope">
           <el-tag :type="scope.row.status === 1 ? 'success' : 'danger'" size="small">{{ scope.row.status === 1 ? '正常' : '禁用' }}</el-tag>
@@ -65,7 +98,7 @@
       </el-table-column>
     </el-table>
 
-    <!-- 新增/修改弹窗 -->
+    <!-- 新增/修改接口弹窗 -->
     <el-dialog :title="dialogTitle" :visible.sync="dialogVisible" width="680px" append-to-body>
       <el-form ref="form" :model="form" :rules="rules" label-width="120px" size="small">
         <el-row :gutter="20">
@@ -75,7 +108,7 @@
             </el-form-item>
           </el-col>
           <el-col :span="12">
-            <el-form-item label="短信类型" prop="smsType">
+            <el-form-item label="发送类型" prop="smsType">
               <el-select v-model="form.smsType" placeholder="请选择" style="width:100%">
                 <el-option v-for="item in smsTypeOptions" :key="item.value" :label="item.label" :value="item.value" />
               </el-select>
@@ -88,6 +121,7 @@
               <el-select v-model="form.provider" placeholder="请选择" style="width:100%">
                 <el-option label="润方" value="rf" />
                 <el-option label="德华" value="dh" />
+                <el-option label="手机卡" value="card" />
               </el-select>
             </el-form-item>
           </el-col>
@@ -101,19 +135,19 @@
         <el-row :gutter="20">
           <el-col :span="12">
             <el-form-item label="账户名" prop="account">
-              <el-input v-model="form.account" placeholder="请输入账户名" />
+              <el-input v-model="form.account" placeholder="接口级默认账户(端口可覆盖)" />
             </el-form-item>
           </el-col>
           <el-col :span="12">
             <el-form-item label="密码" prop="password">
-              <el-input v-model="form.password" placeholder="请输入密码" show-password />
+              <el-input v-model="form.password" placeholder="接口级默认密码" show-password />
             </el-form-item>
           </el-col>
         </el-row>
         <el-row :gutter="20">
           <el-col :span="12">
             <el-form-item label="短信签名">
-              <el-input v-model="form.sign" placeholder="请输入短信签名" />
+              <el-input v-model="form.sign" placeholder="接口级默认签名" />
             </el-form-item>
           </el-col>
           <el-col :span="12">
@@ -125,7 +159,7 @@
         <el-row v-if="form.provider === 'rf'" :gutter="20">
           <el-col :span="12">
             <el-form-item label="扩展码">
-              <el-input v-model="form.code" placeholder="润方专用" />
+              <el-input v-model="form.code" placeholder="润方专用(旧,端口优先)" />
             </el-form-item>
           </el-col>
         </el-row>
@@ -154,20 +188,73 @@
         <el-button @click="dialogVisible = false">取 消</el-button>
       </div>
     </el-dialog>
+
+    <!-- 新增/修改端口弹窗 -->
+    <el-dialog :title="portDialogTitle" :visible.sync="portDialogVisible" width="560px" append-to-body>
+      <el-form ref="portForm" :model="portForm" :rules="portRules" label-width="100px" size="small">
+        <el-form-item label="端口名称" prop="portName">
+          <el-input v-model="portForm.portName" placeholder="如:润方端口1 / XX医院卡" />
+        </el-form-item>
+        <el-form-item label="端口号" prop="portNo">
+          <el-input v-model="portForm.portNo" :placeholder="currentApiProvider === 'card' ? '手机号' : 'extno扩展码'" />
+        </el-form-item>
+        <el-row :gutter="20">
+          <el-col :span="12">
+            <el-form-item label="独立账户">
+              <el-input v-model="portForm.account" placeholder="覆盖接口级" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="12">
+            <el-form-item label="独立密码">
+              <el-input v-model="portForm.password" placeholder="覆盖接口级" show-password />
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <el-row :gutter="20">
+          <el-col :span="12">
+            <el-form-item label="独立签名">
+              <el-input v-model="portForm.sign" placeholder="覆盖接口级" />
+            </el-form-item>
+          </el-col>
+          <el-col v-if="currentApiProvider === 'card'" :span="12">
+            <el-form-item label="卡槽号">
+              <el-select v-model="portForm.slotIndex" style="width:100%">
+                <el-option :value="1" label="卡槽1" />
+                <el-option :value="2" label="卡槽2" />
+              </el-select>
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <el-form-item label="状态">
+          <el-switch v-model="portForm.status" :active-value="1" :inactive-value="0" active-text="正常" inactive-text="禁用" />
+        </el-form-item>
+      </el-form>
+      <div slot="footer">
+        <el-button type="primary" @click="submitPortForm">确 定</el-button>
+        <el-button @click="portDialogVisible = false">取 消</el-button>
+      </div>
+    </el-dialog>
   </div>
 </template>
 
 <script>
 import { listSmsApi, getSmsApi, addSmsApi, updateSmsApi, delSmsApi } from '@/api/system/smsApi'
+import { listPortByApi, addPort, updatePort, delPort } from '@/api/system/smsPort'
 
 export default {
   name: 'AdminSmsApi',
   data() {
     return {
       loading: false,
+      portLoading: false,
       apiList: [],
+      portMap: {},
       dialogVisible: false,
       dialogTitle: '',
+      portDialogVisible: false,
+      portDialogTitle: '',
+      currentApiId: null,
+      currentApiProvider: '',
       smsTypeOptions: [
         { value: 1, label: '行业验证码通知短信' },
         { value: 2, label: '营销短信' },
@@ -177,11 +264,14 @@ export default {
       form: {},
       rules: {
         apiName: [{ required: true, message: '接口名称不能为空', trigger: 'blur' }],
-        smsType: [{ required: true, message: '请选择短信类型', trigger: 'change' }],
+        smsType: [{ required: true, message: '请选择发送类型', trigger: 'change' }],
         provider: [{ required: true, message: '请选择服务商', trigger: 'change' }],
-        account: [{ required: true, message: '账户名不能为空', trigger: 'blur' }],
-        password: [{ required: true, message: '密码不能为空', trigger: 'blur' }],
         costPrice: [{ required: true, message: '成本价不能为空', trigger: 'blur' }]
+      },
+      portForm: {},
+      portRules: {
+        portName: [{ required: true, message: '端口名称不能为空', trigger: 'blur' }],
+        portNo: [{ required: true, message: '端口号不能为空', trigger: 'blur' }]
       }
     }
   },
@@ -193,6 +283,10 @@ export default {
       const map = { 1: '行业验证码通知', 2: '营销短信', 3: '5G消息' }
       return map[type] || '未知'
     },
+    providerFormat(p) {
+      const map = { rf: '润方', dh: '德华', card: '手机卡' }
+      return map[p] || p
+    },
     getList() {
       this.loading = true
       listSmsApi(this.queryParams).then(res => {
@@ -243,7 +337,7 @@ export default {
       })
     },
     handleDelete(row) {
-      this.$confirm('确认删除接口"' + row.apiName + '"?删除后租户绑定也将同步解除。', '提示', {
+      this.$confirm('确认删除接口"' + row.apiName + '"?删除后端口池和租户绑定也将同步解除。', '提示', {
         confirmButtonText: '确定', cancelButtonText: '取消', type: 'warning'
       }).then(() => {
         delSmsApi(row.apiId).then(() => {
@@ -251,6 +345,64 @@ export default {
           this.getList()
         })
       }).catch(() => {})
+    },
+    // ========== 端口池 ==========
+    handleExpand(row, expandedRows) {
+      if (this.portMap[row.apiId]) return
+      this.loadPorts(row.apiId)
+    },
+    loadPorts(apiId) {
+      this.portLoading = true
+      listPortByApi(apiId).then(res => {
+        this.$set(this.portMap, apiId, res.data || [])
+      }).finally(() => { this.portLoading = false })
+    },
+    resetPortForm() {
+      this.portForm = { portId: undefined, apiId: null, portName: '', portNo: '', account: '', password: '', sign: '', slotIndex: null, status: 1 }
+    },
+    handleAddPort(apiRow) {
+      this.resetPortForm()
+      this.currentApiId = apiRow.apiId
+      this.currentApiProvider = apiRow.provider
+      this.portForm.apiId = apiRow.apiId
+      this.portDialogTitle = '新增端口 - ' + apiRow.apiName
+      this.portDialogVisible = true
+    },
+    handleEditPort(portRow, apiRow) {
+      this.resetPortForm()
+      this.currentApiId = apiRow.apiId
+      this.currentApiProvider = apiRow.provider
+      this.portForm = { ...portRow }
+      this.portDialogTitle = '修改端口'
+      this.portDialogVisible = true
+    },
+    submitPortForm() {
+      this.$refs.portForm.validate(valid => {
+        if (!valid) return
+        if (this.portForm.portId) {
+          updatePort(this.portForm).then(() => {
+            this.$message.success('修改成功')
+            this.portDialogVisible = false
+            this.loadPorts(this.currentApiId)
+          })
+        } else {
+          addPort(this.portForm).then(() => {
+            this.$message.success('新增成功')
+            this.portDialogVisible = false
+            this.loadPorts(this.currentApiId)
+          })
+        }
+      })
+    },
+    handleDelPort(row) {
+      this.$confirm('确认删除端口"' + row.portName + '"?', '提示', {
+        confirmButtonText: '确定', cancelButtonText: '取消', type: 'warning'
+      }).then(() => {
+        delPort(row.portId).then(() => {
+          this.$message.success('删除成功')
+          this.loadPorts(row.apiId)
+        })
+      }).catch(() => {})
     }
   }
 }

+ 43 - 13
src/views/admin/smsApiTenant/index.vue

@@ -1,8 +1,8 @@
 <template>
   <div class="app-container">
     <el-form :model="queryParams" ref="queryForm" :inline="true" size="small">
-      <el-form-item label="租户" prop="companyId">
-        <el-select v-model="queryParams.companyId" placeholder="选择租户" clearable filterable>
+      <el-form-item label="租户" prop="tenantId">
+        <el-select v-model="queryParams.tenantId" placeholder="选择租户" clearable filterable>
           <el-option v-for="c in companyList" :key="c.companyId" :label="c.companyName" :value="c.companyId" />
         </el-select>
       </el-form-item>
@@ -35,7 +35,12 @@
         </template>
       </el-table-column>
       <el-table-column label="服务商" align="center" prop="provider" width="80">
-        <template slot-scope="scope">{{ scope.row.provider === 'rf' ? '润方' : '德华' }}</template>
+        <template slot-scope="scope">{{ providerFormat(scope.row.provider) }}</template>
+      </el-table-column>
+      <el-table-column label="优先级" align="center" prop="priority" width="70">
+        <template slot-scope="scope">
+          <el-tag type="warning" size="mini">{{ scope.row.priority || 1 }}</el-tag>
+        </template>
       </el-table-column>
       <el-table-column label="成本价" align="center" prop="costPrice" width="100">
         <template slot-scope="scope">
@@ -52,6 +57,11 @@
           <span style="color:#67c23a;font-weight:bold">{{ calcProfit(scope.row) }}</span>
         </template>
       </el-table-column>
+      <el-table-column label="手动选择" align="center" prop="allowManual" width="80">
+        <template slot-scope="scope">
+          <el-tag :type="scope.row.allowManual === 1 ? 'success' : 'info'" size="mini">{{ scope.row.allowManual === 1 ? '允许' : '否' }}</el-tag>
+        </template>
+      </el-table-column>
       <el-table-column label="状态" align="center" prop="status" width="70">
         <template slot-scope="scope">
           <el-tag :type="scope.row.status === 1 ? 'success' : 'danger'" size="small">{{ scope.row.status === 1 ? '启用' : '禁用' }}</el-tag>
@@ -73,15 +83,23 @@
             <el-option v-for="api in apiOptions" :key="api.apiId" :label="api.apiName + ' (' + smsTypeFormat(api.smsType) + ')'" :value="api.apiId" />
           </el-select>
         </el-form-item>
-        <el-form-item label="选择租户" prop="companyId">
-          <el-select v-model="addForm.companyId" placeholder="请选择租户" filterable style="width:100%">
-            <el-option v-for="c in companyList" :key="c.companyId" :label="c.companyName" :value="c.companyId" />
+        <el-form-item label="选择租户" prop="tenantId">
+          <el-select v-model="addForm.tenantId" placeholder="请选择租户" filterable style="width:100%">
+            <el-option v-for="c in companyList" :key="c.tenantId" :label="c.companyName" :value="c.tenantId" />
           </el-select>
         </el-form-item>
         <el-form-item label="租户售价" prop="price">
           <el-input-number v-model="addForm.price" :precision="4" :step="0.001" :min="0" style="width:100%" />
           <span style="color:#909399;font-size:12px;margin-left:8px">元/条</span>
         </el-form-item>
+        <el-form-item label="优先级" prop="priority">
+          <el-input-number v-model="addForm.priority" :min="1" :max="99" style="width:100%" />
+          <span style="color:#909399;font-size:12px;margin-left:8px">1最高,同类型多接口时按优先级选择</span>
+        </el-form-item>
+        <el-form-item label="允许手动选择">
+          <el-switch v-model="addForm.allowManual" :active-value="1" :inactive-value="0" active-text="允许" inactive-text="否" />
+          <span style="color:#909399;font-size:12px;margin-left:8px">是否允许销售手动选择此接口发送</span>
+        </el-form-item>
         <el-form-item v-if="selectedApiCost" label="">
           <span style="color:#909399;font-size:12px">该接口成本价: {{ selectedApiCost }} 元/条,建议售价 >= 成本价</span>
         </el-form-item>
@@ -108,6 +126,12 @@
           <el-input-number v-model="editForm.price" :precision="4" :step="0.001" :min="0" style="width:100%" />
           <span style="color:#909399;font-size:12px;margin-left:8px">元/条</span>
         </el-form-item>
+        <el-form-item label="优先级">
+          <el-input-number v-model="editForm.priority" :min="1" :max="99" style="width:100%" />
+        </el-form-item>
+        <el-form-item label="允许手动选择">
+          <el-switch v-model="editForm.allowManual" :active-value="1" :inactive-value="0" active-text="允许" inactive-text="否" />
+        </el-form-item>
         <el-form-item label="状态">
           <el-switch v-model="editForm.status" :active-value="1" :inactive-value="0" active-text="启用" inactive-text="禁用" />
         </el-form-item>
@@ -137,14 +161,14 @@ export default {
         { value: 2, label: '营销短信' },
         { value: 3, label: '5G消息' }
       ],
-      queryParams: { companyId: undefined, smsType: undefined },
+      queryParams: { tenantId: undefined, smsType: undefined },
       addDialogVisible: false,
       editDialogVisible: false,
-      addForm: { apiId: undefined, companyId: undefined, price: 0 },
-      editForm: { id: undefined, price: 0, status: 1, companyName: '', apiName: '', costPrice: 0 },
+      addForm: { apiId: undefined, tenantId: undefined, price: 0, priority: 1, allowManual: 0 },
+      editForm: { id: undefined, price: 0, priority: 1, allowManual: 0, status: 1, companyName: '', apiName: '', costPrice: 0 },
       addRules: {
         apiId: [{ required: true, message: '请选择接口', trigger: 'change' }],
-        companyId: [{ required: true, message: '请选择租户', trigger: 'change' }],
+        tenantId: [{ required: true, message: '请选择租户', trigger: 'change' }],
         price: [{ required: true, message: '请填写售价', trigger: 'blur' }]
       }
     }
@@ -165,6 +189,10 @@ export default {
       const map = { 1: '行业验证码通知', 2: '营销短信', 3: '5G消息' }
       return map[type] || '未知'
     },
+    providerFormat(p) {
+      const map = { rf: '润方', dh: '德华', card: '手机卡' }
+      return map[p] || p
+    },
     calcProfit(row) {
       if (row.price != null && row.costPrice != null) {
         return (row.price - row.costPrice).toFixed(4)
@@ -196,11 +224,11 @@ export default {
     },
     handleQuery() { this.getList() },
     resetQuery() {
-      this.queryParams = { companyId: undefined, smsType: undefined }
+      this.queryParams = { tenantId: undefined, smsType: undefined }
       this.handleQuery()
     },
     handleAdd() {
-      this.addForm = { apiId: undefined, companyId: undefined, price: 0 }
+      this.addForm = { apiId: undefined, tenantId: undefined, price: 0, priority: 1, allowManual: 0 }
       this.addDialogVisible = true
     },
     submitAdd() {
@@ -217,6 +245,8 @@ export default {
       this.editForm = {
         id: row.id,
         price: row.price,
+        priority: row.priority || 1,
+        allowManual: row.allowManual || 0,
         status: row.status,
         companyName: row.companyName,
         apiName: row.apiName,
@@ -225,7 +255,7 @@ export default {
       this.editDialogVisible = true
     },
     submitEdit() {
-      updateSmsApiTenant({ id: this.editForm.id, price: this.editForm.price, status: this.editForm.status }).then(() => {
+      updateSmsApiTenant({ id: this.editForm.id, price: this.editForm.price, priority: this.editForm.priority, allowManual: this.editForm.allowManual, status: this.editForm.status }).then(() => {
         this.$message.success('修改成功')
         this.editDialogVisible = false
         this.getList()

+ 437 - 0
src/views/admin/smsCard/index.vue

@@ -0,0 +1,437 @@
+<template>
+  <div class="app-container">
+    <el-form :model="queryParams" ref="queryForm" :inline="true" size="small">
+      <el-form-item label="租户" prop="tenantId">
+        <el-select v-model="queryParams.tenantId" placeholder="选择租户" clearable filterable>
+          <el-option v-for="c in companyList" :key="c.companyId" :label="c.companyName" :value="c.companyId" />
+        </el-select>
+      </el-form-item>
+      <el-form-item label="状态" prop="status">
+        <el-select v-model="queryParams.status" placeholder="请选择" clearable>
+          <el-option label="在线" :value="1" />
+          <el-option label="离线" :value="0" />
+          <el-option label="禁用" :value="2" />
+        </el-select>
+      </el-form-item>
+      <el-form-item>
+        <el-button type="primary" icon="el-icon-search" size="mini" @click="handleQuery">搜索</el-button>
+        <el-button icon="el-icon-refresh" size="mini" @click="resetQuery">重置</el-button>
+      </el-form-item>
+    </el-form>
+
+    <el-row :gutter="10" class="mb8">
+      <el-col :span="1.5">
+        <el-button type="primary" plain icon="el-icon-plus" size="mini" @click="handleAdd">新增手机卡</el-button>
+      </el-col>
+    </el-row>
+
+    <el-table v-loading="loading" :data="cardList" border size="small">
+      <el-table-column label="ID" align="center" prop="cardId" width="60" />
+      <el-table-column label="IMEI" align="center" prop="imei" width="150" />
+      <el-table-column label="设备名称" align="center" prop="deviceName" min-width="100" />
+      <el-table-column label="租户" align="center" prop="tenantName" min-width="120" show-overflow-tooltip />
+      <el-table-column label="卡槽1手机号" align="center" prop="phone1" width="130" />
+      <el-table-column label="卡槽2手机号" align="center" prop="phone2" width="130" />
+      <el-table-column label="关联端口" align="center" prop="portName" min-width="100" />
+      <el-table-column label="状态" align="center" prop="status" width="70">
+        <template slot-scope="scope">
+          <el-tag :type="scope.row.status === 1 ? 'success' : scope.row.status === 0 ? 'danger' : 'info'" size="small">
+            {{ scope.row.status === 1 ? '在线' : scope.row.status === 0 ? '离线' : '禁用' }}
+          </el-tag>
+        </template>
+      </el-table-column>
+      <!-- 短信相关列 -->
+      <el-table-column label="今日已发" align="center" width="80">
+        <template slot-scope="scope">
+          <span :class="{'text-danger': scope.row.smsSentToday >= scope.row.smsDailyLimit}">{{ scope.row.smsSentToday || 0 }}</span>
+        </template>
+      </el-table-column>
+      <el-table-column label="每小时限制" align="center" prop="smsHourlyLimit" width="90" />
+      <el-table-column label="每日限制" align="center" prop="smsDailyLimit" width="80" />
+      <el-table-column label="短信余额" align="center" width="80">
+        <template slot-scope="scope">
+          <span :class="{'text-danger': scope.row.smsBalance <= 0}">{{ scope.row.smsBalance || 0 }}</span>
+        </template>
+      </el-table-column>
+      <!-- 通话相关列 -->
+      <el-table-column label="今日已拨" align="center" prop="callSentToday" width="80" />
+      <el-table-column label="拨打间隔(秒)" align="center" prop="callIntervalSeconds" width="100" />
+      <el-table-column label="通话余额(分)" align="center" prop="callMinutesBalance" width="100" />
+      <el-table-column label="话费余额(元)" align="center" prop="phoneBillBalance" width="100" />
+      <el-table-column label="呼转" align="center" width="60">
+        <template slot-scope="scope">
+          <el-tag v-if="scope.row.allowCallForward === 1" type="success" size="mini">是</el-tag>
+          <el-tag v-else type="info" size="mini">否</el-tag>
+        </template>
+      </el-table-column>
+      <el-table-column label="最后心跳" align="center" prop="lastHeartbeat" width="160" />
+      <el-table-column label="操作" align="center" width="160" fixed="right">
+        <template slot-scope="scope">
+          <el-button size="mini" type="text" @click="handleUpdate(scope.row)">编辑</el-button>
+          <el-button size="mini" type="text" style="color:#f56c6c" @click="handleDelete(scope.row)">删除</el-button>
+        </template>
+      </el-table-column>
+    </el-table>
+
+    <!-- 新增/修改弹窗 -->
+    <el-dialog :title="dialogTitle" :visible.sync="dialogVisible" width="680px" append-to-body>
+      <el-form ref="form" :model="form" :rules="rules" label-width="120px" size="small">
+        <el-row :gutter="20">
+          <el-col :span="12">
+            <el-form-item label="IMEI" prop="imei">
+              <el-input v-model="form.imei" placeholder="手机IMEI号" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="12">
+            <el-form-item label="设备名称">
+              <el-input v-model="form.deviceName" placeholder="如:红米Note12" />
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <el-row :gutter="20">
+          <el-col :span="12">
+            <el-form-item label="所属租户" prop="tenantId">
+              <el-select v-model="form.tenantId" placeholder="选择租户" filterable style="width:100%">
+                <el-option v-for="c in companyList" :key="c.companyId" :label="c.companyName" :value="c.companyId" />
+              </el-select>
+            </el-form-item>
+          </el-col>
+          <el-col :span="12">
+            <el-form-item label="关联端口">
+              <el-input v-model="form.portId" placeholder="端口ID(可选)" />
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <el-row :gutter="20">
+          <el-col :span="12">
+            <el-form-item label="卡槽1手机号">
+              <el-input v-model="form.phone1" placeholder="手机号" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="12">
+            <el-form-item label="卡槽2手机号">
+              <el-input v-model="form.phone2" placeholder="双卡时填写" />
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <el-form-item label="状态">
+          <el-select v-model="form.status" style="width:100%">
+            <el-option :value="0" label="离线" />
+            <el-option :value="1" label="在线" />
+            <el-option :value="2" label="禁用" />
+          </el-select>
+        </el-form-item>
+
+        <!-- 短信限制设置 -->
+        <el-divider content-position="left">短信发送限制</el-divider>
+        <el-row :gutter="20">
+          <el-col :span="8">
+            <el-form-item label="每小时限制" prop="smsHourlyLimit">
+              <el-input-number v-model="form.smsHourlyLimit" :min="0" :max="9999" style="width:100%" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="8">
+            <el-form-item label="每日限制" prop="smsDailyLimit">
+              <el-input-number v-model="form.smsDailyLimit" :min="0" :max="99999" style="width:100%" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="8">
+            <el-form-item label="短信余额(条)" prop="smsBalance">
+              <el-input-number v-model="form.smsBalance" :min="0" :max="999999" style="width:100%" />
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <el-row :gutter="20">
+          <el-col :span="8">
+            <el-form-item label="今日已发">
+              <el-input-number v-model="form.smsSentToday" :min="0" style="width:100%" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="8">
+            <el-form-item label="当前小时已发">
+              <el-input-number v-model="form.smsSentHour" :min="0" style="width:100%" />
+            </el-form-item>
+          </el-col>
+        </el-row>
+
+        <!-- 通话限制设置 -->
+        <el-divider content-position="left">通话限制</el-divider>
+        <el-row :gutter="20">
+          <el-col :span="8">
+            <el-form-item label="拨打间隔(秒)" prop="callIntervalSeconds">
+              <el-input-number v-model="form.callIntervalSeconds" :min="0" :max="3600" style="width:100%" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="8">
+            <el-form-item label="今日已拨">
+              <el-input-number v-model="form.callSentToday" :min="0" style="width:100%" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="8">
+            <el-form-item label="通话余额(分)">
+              <el-input-number v-model="form.callMinutesBalance" :min="0" :precision="2" :step="1" style="width:100%" />
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <el-row :gutter="20">
+          <el-col :span="8">
+            <el-form-item label="话费余额(元)">
+              <el-input-number v-model="form.phoneBillBalance" :min="0" :precision="4" :step="1" style="width:100%" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="8">
+            <el-form-item label="允许呼转">
+              <el-switch v-model="form.allowCallForward" :active-value="1" :inactive-value="0" active-text="是" inactive-text="否" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="8">
+            <el-form-item label="呼转手机号">
+              <el-input v-model="form.forwardPhone" placeholder="呼转目标手机号" :disabled="form.allowCallForward !== 1" />
+            </el-form-item>
+          </el-col>
+        </el-row>
+
+        <el-form-item label="备注">
+          <el-input v-model="form.remark" type="textarea" :rows="2" />
+        </el-form-item>
+      </el-form>
+      <div slot="footer">
+        <el-button type="primary" @click="submitForm">确 定</el-button>
+        <el-button @click="dialogVisible = false">取 消</el-button>
+      </div>
+    </el-dialog>
+
+    <!-- 中间件配置区域 -->
+    <el-divider content-position="left">手机卡发送中间件配置</el-divider>
+    <el-row :gutter="10" class="mb8">
+      <el-col :span="1.5">
+        <el-button type="primary" plain icon="el-icon-plus" size="mini" @click="handleAddMw">新增中间件</el-button>
+      </el-col>
+    </el-row>
+    <el-table :data="mwList" border size="small" v-loading="mwLoading">
+      <el-table-column label="ID" align="center" prop="id" width="60" />
+      <el-table-column label="接口" align="center" prop="apiName" min-width="140" />
+      <el-table-column label="名称" align="center" prop="middlewareName" min-width="120" />
+      <el-table-column label="回调URL" align="center" prop="callbackUrl" min-width="200" show-overflow-tooltip />
+      <el-table-column label="心跳URL" align="center" prop="heartbeatUrl" min-width="180" show-overflow-tooltip />
+      <el-table-column label="重试次数" align="center" prop="maxRetry" width="80" />
+      <el-table-column label="超时(秒)" align="center" prop="timeoutSeconds" width="80" />
+      <el-table-column label="状态" align="center" prop="status" width="70">
+        <template slot-scope="scope">
+          <el-tag :type="scope.row.status === 1 ? 'success' : 'danger'" size="mini">{{ scope.row.status === 1 ? '正常' : '禁用' }}</el-tag>
+        </template>
+      </el-table-column>
+      <el-table-column label="操作" align="center" width="120">
+        <template slot-scope="scope">
+          <el-button size="mini" type="text" @click="handleEditMw(scope.row)">编辑</el-button>
+          <el-button size="mini" type="text" style="color:#f56c6c" @click="handleDelMw(scope.row)">删除</el-button>
+        </template>
+      </el-table-column>
+    </el-table>
+
+    <!-- 中间件弹窗 -->
+    <el-dialog :title="mwDialogTitle" :visible.sync="mwDialogVisible" width="560px" append-to-body>
+      <el-form ref="mwForm" :model="mwForm" :rules="mwRules" label-width="100px" size="small">
+        <el-form-item label="关联接口" prop="apiId">
+          <el-select v-model="mwForm.apiId" placeholder="选择手机卡类型接口" style="width:100%">
+            <el-option v-for="api in cardApiList" :key="api.apiId" :label="api.apiName" :value="api.apiId" />
+          </el-select>
+        </el-form-item>
+        <el-form-item label="名称" prop="middlewareName">
+          <el-input v-model="mwForm.middlewareName" />
+        </el-form-item>
+        <el-form-item label="回调URL" prop="callbackUrl">
+          <el-input v-model="mwForm.callbackUrl" placeholder="下发发送任务的URL" />
+        </el-form-item>
+        <el-form-item label="心跳URL">
+          <el-input v-model="mwForm.heartbeatUrl" placeholder="心跳上报URL" />
+        </el-form-item>
+        <el-form-item label="鉴权Token">
+          <el-input v-model="mwForm.authToken" />
+        </el-form-item>
+        <el-row :gutter="20">
+          <el-col :span="12">
+            <el-form-item label="重试次数">
+              <el-input-number v-model="mwForm.maxRetry" :min="0" :max="10" style="width:100%" />
+            </el-form-item>
+          </el-col>
+          <el-col :span="12">
+            <el-form-item label="超时(秒)">
+              <el-input-number v-model="mwForm.timeoutSeconds" :min="5" :max="300" style="width:100%" />
+            </el-form-item>
+          </el-col>
+        </el-row>
+        <el-form-item label="状态">
+          <el-switch v-model="mwForm.status" :active-value="1" :inactive-value="0" active-text="正常" inactive-text="禁用" />
+        </el-form-item>
+      </el-form>
+      <div slot="footer">
+        <el-button type="primary" @click="submitMwForm">确 定</el-button>
+        <el-button @click="mwDialogVisible = false">取 消</el-button>
+      </div>
+    </el-dialog>
+  </div>
+</template>
+
+<script>
+import { listCard, addCard, updateCard, delCard, listMiddleware, addMiddleware, updateMiddleware, delMiddleware } from '@/api/system/smsPort'
+import { listSmsApi } from '@/api/system/smsApi'
+import { listAllCompanies } from '@/api/admin/sysCompany'
+
+export default {
+  name: 'AdminSmsCard',
+  data() {
+    return {
+      loading: false,
+      mwLoading: false,
+      cardList: [],
+      mwList: [],
+      companyList: [],
+      cardApiList: [],
+      dialogVisible: false,
+      dialogTitle: '',
+      mwDialogVisible: false,
+      mwDialogTitle: '',
+      queryParams: { tenantId: undefined, status: undefined },
+      form: {},
+      rules: {
+        imei: [{ required: true, message: 'IMEI不能为空', trigger: 'blur' }],
+        tenantId: [{ required: true, message: '请选择租户', trigger: 'change' }]
+      },
+      mwForm: {},
+      mwRules: {
+        apiId: [{ required: true, message: '请选择接口', trigger: 'change' }],
+        middlewareName: [{ required: true, message: '名称不能为空', trigger: 'blur' }],
+        callbackUrl: [{ required: true, message: '回调URL不能为空', trigger: 'blur' }]
+      }
+    }
+  },
+  created() {
+    this.getList()
+    this.loadMwList()
+    this.loadCompanyList()
+    this.loadCardApiList()
+  },
+  methods: {
+    getList() {
+      this.loading = true
+      listCard(this.queryParams).then(res => {
+        this.cardList = res.data || []
+      }).finally(() => { this.loading = false })
+    },
+    loadMwList() {
+      this.mwLoading = true
+      listMiddleware({}).then(res => {
+        this.mwList = res.data || []
+      }).finally(() => { this.mwLoading = false })
+    },
+    loadCompanyList() {
+      listAllCompanies({ pageNum: 1, pageSize: 1000 }).then(res => {
+        this.companyList = res.rows || res.data || []
+      })
+    },
+    loadCardApiList() {
+      listSmsApi({ provider: 'card', status: 1 }).then(res => {
+        this.cardApiList = res.data || []
+      })
+    },
+    handleQuery() { this.getList() },
+    resetQuery() {
+      this.queryParams = { tenantId: undefined, status: undefined }
+      this.handleQuery()
+    },
+    resetForm() {
+      this.form = {
+        cardId: undefined, imei: '', deviceName: '', tenantId: undefined, portId: undefined,
+        phone1: '', phone2: '', simCount: 1, status: 0,
+        smsHourlyLimit: 100, smsDailyLimit: 500, smsBalance: 0,
+        smsSentToday: 0, smsSentHour: 0,
+        callSentToday: 0, callIntervalSeconds: 30,
+        callMinutesBalance: 0, phoneBillBalance: 0,
+        allowCallForward: 0, forwardPhone: '',
+        remark: ''
+      }
+    },
+    handleAdd() {
+      this.resetForm()
+      this.dialogTitle = '新增手机卡'
+      this.dialogVisible = true
+    },
+    handleUpdate(row) {
+      this.resetForm()
+      this.form = { ...row }
+      this.dialogTitle = '修改手机卡'
+      this.dialogVisible = true
+    },
+    submitForm() {
+      this.$refs.form.validate(valid => {
+        if (!valid) return
+        if (this.form.cardId) {
+          updateCard(this.form).then(() => {
+            this.$message.success('修改成功')
+            this.dialogVisible = false
+            this.getList()
+          })
+        } else {
+          this.form.simCount = (this.form.phone2 && this.form.phone2.length > 0) ? 2 : 1
+          addCard(this.form).then(() => {
+            this.$message.success('新增成功')
+            this.dialogVisible = false
+            this.getList()
+          })
+        }
+      })
+    },
+    handleDelete(row) {
+      this.$confirm('确认删除此手机卡记录?', '提示', { type: 'warning' }).then(() => {
+        delCard(row.cardId).then(() => { this.$message.success('删除成功'); this.getList() })
+      }).catch(() => {})
+    },
+    // ========== 中间件 ==========
+    resetMwForm() {
+      this.mwForm = { id: undefined, apiId: undefined, middlewareName: '', callbackUrl: '', heartbeatUrl: '', authToken: '', maxRetry: 3, timeoutSeconds: 30, status: 1 }
+    },
+    handleAddMw() {
+      this.resetMwForm()
+      this.mwDialogTitle = '新增中间件'
+      this.mwDialogVisible = true
+    },
+    handleEditMw(row) {
+      this.resetMwForm()
+      this.mwForm = { ...row }
+      this.mwDialogTitle = '修改中间件'
+      this.mwDialogVisible = true
+    },
+    submitMwForm() {
+      this.$refs.mwForm.validate(valid => {
+        if (!valid) return
+        if (this.mwForm.id) {
+          updateMiddleware(this.mwForm).then(() => {
+            this.$message.success('修改成功')
+            this.mwDialogVisible = false
+            this.loadMwList()
+          })
+        } else {
+          addMiddleware(this.mwForm).then(() => {
+            this.$message.success('新增成功')
+            this.mwDialogVisible = false
+            this.loadMwList()
+          })
+        }
+      })
+    },
+    handleDelMw(row) {
+      this.$confirm('确认删除此中间件配置?', '提示', { type: 'warning' }).then(() => {
+        delMiddleware(row.id).then(() => { this.$message.success('删除成功'); this.loadMwList() })
+      }).catch(() => {})
+    }
+  }
+}
+</script>
+
+<style scoped>
+.text-danger {
+  color: #f56c6c;
+  font-weight: bold;
+}
+</style>