This commit is contained in:
李志强 2026-06-22 11:24:08 +08:00
parent 4e1b30b3cc
commit a4af179c16
14 changed files with 612 additions and 36 deletions

View File

@ -16,44 +16,91 @@ type ApiCursorEquipmentController struct {
beego.Controller
}
// cursorIpInfo 对应 ip-api.com 返回的 JSON 结构
type cursorIpInfo struct {
Status string `json:"status"`
Country string `json:"country"`
CountryCode string `json:"countryCode"`
Region string `json:"region"`
RegionName string `json:"regionName"`
City string `json:"city"`
Zip string `json:"zip"`
Lat float64 `json:"lat"`
Lon float64 `json:"lon"`
Timezone string `json:"timezone"`
ISP string `json:"isp"`
Org string `json:"org"`
As string `json:"as"`
Query string `json:"query"`
}
type cursorEquipmentReportPayload struct {
DeviceInfo string `json:"deviceInfo"`
DeviceInfoSnake string `json:"device_info"`
MachineCode string `json:"machineCode"`
MachineCodeSnake string `json:"machine_code"`
Status *int8 `json:"status"`
System string `json:"system"`
Version string `json:"version"`
BindAccount string `json:"bindAccount"`
BindAccountSnake string `json:"bind_account"`
OwnerUserID *uint64 `json:"ownerUserId"`
OwnerUserIDSnake *uint64 `json:"owner_user_id"`
OwnerUserName string `json:"ownerUserName"`
OwnerUserNameSnake string `json:"owner_user_name"`
ActivationTime string `json:"activationTime"`
ActivationTimeSnake string `json:"activation_time"`
ExpireTime string `json:"expireTime"`
ExpireTimeSnake string `json:"expire_time"`
Remark string `json:"remark"`
DeviceInfo string `json:"deviceInfo"`
DeviceInfoSnake string `json:"device_info"`
MachineCode string `json:"machineCode"`
MachineCodeSnake string `json:"machine_code"`
Status *int8 `json:"status"`
System string `json:"system"`
Version string `json:"version"`
BindAccount string `json:"bindAccount"`
BindAccountSnake string `json:"bind_account"`
OwnerUserID *uint64 `json:"ownerUserId"`
OwnerUserIDSnake *uint64 `json:"owner_user_id"`
OwnerUserName string `json:"ownerUserName"`
OwnerUserNameSnake string `json:"owner_user_name"`
ActivationTime string `json:"activationTime"`
ActivationTimeSnake string `json:"activation_time"`
ExpireTime string `json:"expireTime"`
ExpireTimeSnake string `json:"expire_time"`
Remark string `json:"remark"`
IpInfo *cursorIpInfo `json:"ipInfo"`
}
type cursorEquipmentActivatePayload struct {
Code string `json:"code"`
ActivationCode string `json:"activationCode"`
ActivationCodeSnake string `json:"activation_code"`
DeviceInfo string `json:"deviceInfo"`
DeviceInfoSnake string `json:"device_info"`
MachineCode string `json:"machineCode"`
MachineCodeSnake string `json:"machine_code"`
System string `json:"system"`
Version string `json:"version"`
BindAccount string `json:"bindAccount"`
BindAccountSnake string `json:"bind_account"`
OwnerUserID *uint64 `json:"ownerUserId"`
OwnerUserIDSnake *uint64 `json:"owner_user_id"`
OwnerUserName string `json:"ownerUserName"`
OwnerUserNameSnake string `json:"owner_user_name"`
Remark string `json:"remark"`
Code string `json:"code"`
ActivationCode string `json:"activationCode"`
ActivationCodeSnake string `json:"activation_code"`
DeviceInfo string `json:"deviceInfo"`
DeviceInfoSnake string `json:"device_info"`
MachineCode string `json:"machineCode"`
MachineCodeSnake string `json:"machine_code"`
System string `json:"system"`
Version string `json:"version"`
BindAccount string `json:"bindAccount"`
BindAccountSnake string `json:"bind_account"`
OwnerUserID *uint64 `json:"ownerUserId"`
OwnerUserIDSnake *uint64 `json:"owner_user_id"`
OwnerUserName string `json:"ownerUserName"`
OwnerUserNameSnake string `json:"owner_user_name"`
Remark string `json:"remark"`
IpInfo *cursorIpInfo `json:"ipInfo"`
}
// cursorSaveIpLog 将 ipInfo 写入设备 IP 日志表(异步,失败不影响主流程)
func cursorSaveIpLog(equipmentID uint64, machineCode, source string, ip *cursorIpInfo) {
if ip == nil {
return
}
log := &models.PlatformCursorEquipmentIpLog{
EquipmentID: equipmentID,
MachineCode: machineCode,
Source: source,
Status: ip.Status,
Country: ip.Country,
CountryCode: ip.CountryCode,
Region: ip.Region,
RegionName: ip.RegionName,
City: ip.City,
Zip: ip.Zip,
Lat: ip.Lat,
Lon: ip.Lon,
Timezone: ip.Timezone,
ISP: ip.ISP,
Org: ip.Org,
AsInfo: ip.As,
Query: ip.Query,
}
_, _ = models.Orm.Insert(log)
}
func cursorFirstNonEmpty(values ...string) string {
@ -293,6 +340,9 @@ func (c *ApiCursorEquipmentController) Report() {
row.UpdateTime = &now
}
// 记录 IP 日志
cursorSaveIpLog(row.ID, machineCode, "report", p.IpInfo)
// 查询该设备最新激活码,补全激活时间和到期时间
var retActivationTime interface{} = row.ActivationTime
var retExpireTime interface{} = row.ExpireTime
@ -569,6 +619,9 @@ func (c *ApiCursorEquipmentController) ActivateByCode() {
}
rollback = false
// 记录 IP 日志
cursorSaveIpLog(device.ID, machineCode, "activateByCode", p.IpInfo)
c.jsonResult(200, "success", map[string]interface{}{
"activated": true,
"reused": false,

View File

@ -143,6 +143,40 @@ func (c *PlatformCursorEquipmentController) rowToMap(row *models.PlatformCursorE
lastExtractedAt = latestExtract.ExtractedTime
}
// 查询该设备最近一条 IP 日志
var lastLoginIp interface{}
var lastLoginIpInfo interface{}
var latestIpLog models.PlatformCursorEquipmentIpLog
ipCond := orm.NewCondition().
AndCond(orm.NewCondition().
Or("equipment_id", row.ID).
Or("machine_code", row.MachineCode))
if err := models.Orm.QueryTable(new(models.PlatformCursorEquipmentIpLog)).
SetCond(ipCond).
OrderBy("-id").
One(&latestIpLog); err == nil {
lastLoginIp = latestIpLog.Query
lastLoginIpInfo = map[string]interface{}{
"id": latestIpLog.ID,
"source": latestIpLog.Source,
"status": latestIpLog.Status,
"country": latestIpLog.Country,
"countryCode": latestIpLog.CountryCode,
"region": latestIpLog.Region,
"regionName": latestIpLog.RegionName,
"city": latestIpLog.City,
"zip": latestIpLog.Zip,
"lat": latestIpLog.Lat,
"lon": latestIpLog.Lon,
"timezone": latestIpLog.Timezone,
"isp": latestIpLog.ISP,
"org": latestIpLog.Org,
"asInfo": latestIpLog.AsInfo,
"query": latestIpLog.Query,
"createTime": latestIpLog.CreateTime,
}
}
return map[string]interface{}{
"id": row.ID,
"deviceInfo": row.DeviceInfo,
@ -164,6 +198,8 @@ func (c *PlatformCursorEquipmentController) rowToMap(row *models.PlatformCursorE
"activationCount": activationCount,
"extractCount": extractCount,
"lastExtractedAt": lastExtractedAt,
"lastLoginIp": lastLoginIp,
"lastLoginIpInfo": lastLoginIpInfo,
"remark": row.Remark,
"createTime": row.CreateTime,
"updateTime": row.UpdateTime,
@ -679,3 +715,85 @@ func (c *PlatformCursorEquipmentController) ExtractRecords() {
"pageSize": pageSize,
})
}
// IpLogs GET /platform/cursor/equipment/ipLogs?equipmentId=1
func (c *PlatformCursorEquipmentController) IpLogs() {
if _, err := c.platformClaims(); err != nil {
c.jsonErr(401, 401, err.Error())
return
}
equipmentID, _ := c.GetUint64("equipmentId")
if equipmentID == 0 {
equipmentID, _ = c.GetUint64("id")
}
if equipmentID == 0 {
c.jsonErr(400, 400, "缺少设备ID")
return
}
var equipment models.PlatformCursorEquipment
if err := models.Orm.QueryTable(new(models.PlatformCursorEquipment)).
Filter("id", equipmentID).
Filter("delete_time__isnull", true).
One(&equipment); err != nil {
c.jsonErr(404, 404, "设备不存在")
return
}
page, _ := c.GetInt("page", 1)
pageSize, _ := c.GetInt("pageSize", 20)
if page < 1 {
page = 1
}
if pageSize < 1 {
pageSize = 20
}
if pageSize > 200 {
pageSize = 200
}
ipCond := orm.NewCondition().
AndCond(orm.NewCondition().
Or("equipment_id", equipment.ID).
Or("machine_code", equipment.MachineCode))
qs := models.Orm.QueryTable(new(models.PlatformCursorEquipmentIpLog)).SetCond(ipCond)
total, _ := qs.Count()
var rows []models.PlatformCursorEquipmentIpLog
if _, err := qs.OrderBy("-id").Limit(pageSize, (page-1)*pageSize).All(&rows); err != nil {
c.jsonErr(500, 500, "获取 IP 日志失败: "+err.Error())
return
}
list := make([]map[string]interface{}, 0, len(rows))
for _, row := range rows {
list = append(list, map[string]interface{}{
"id": row.ID,
"source": row.Source,
"status": row.Status,
"country": row.Country,
"countryCode": row.CountryCode,
"region": row.Region,
"regionName": row.RegionName,
"city": row.City,
"zip": row.Zip,
"lat": row.Lat,
"lon": row.Lon,
"timezone": row.Timezone,
"isp": row.ISP,
"org": row.Org,
"asInfo": row.AsInfo,
"query": row.Query,
"createTime": row.CreateTime,
})
}
c.ok(map[string]interface{}{
"list": list,
"total": total,
"page": page,
"pageSize": pageSize,
})
}

View File

@ -58,6 +58,7 @@ func Init(_ string) {
new(SystemSoftwareUpgrade),
new(PlatformCursorEquipment),
new(PlatformCursorActivationCode),
new(PlatformCursorEquipmentIpLog),
new(PlatformAccountPoolKiro),
new(PlatformAccountPoolWindsurf),
new(PlatformAccountPoolCursor),

View File

@ -0,0 +1,31 @@
package models
import "time"
// PlatformCursorEquipmentIpLog 设备 IP 登录日志 yz_platform_cursor_equipment_ip_log
// 每次客户端调用 report / activateByCode 时,若携带 ipInfo 则写入一条记录。
type PlatformCursorEquipmentIpLog struct {
ID uint64 `orm:"column(id);pk;auto" json:"id"`
EquipmentID uint64 `orm:"column(equipment_id);default(0)" json:"equipmentId"`
MachineCode string `orm:"column(machine_code);size(128)" json:"machineCode"`
Source string `orm:"column(source);size(32)" json:"source"`
Status string `orm:"column(status);size(32)" json:"status"`
Country string `orm:"column(country);size(64)" json:"country"`
CountryCode string `orm:"column(country_code);size(8)" json:"countryCode"`
Region string `orm:"column(region);size(16)" json:"region"`
RegionName string `orm:"column(region_name);size(128)" json:"regionName"`
City string `orm:"column(city);size(128)" json:"city"`
Zip string `orm:"column(zip);size(32)" json:"zip"`
Lat float64 `orm:"column(lat)" json:"lat"`
Lon float64 `orm:"column(lon)" json:"lon"`
Timezone string `orm:"column(timezone);size(64)" json:"timezone"`
ISP string `orm:"column(isp);size(255)" json:"isp"`
Org string `orm:"column(org);size(255)" json:"org"`
AsInfo string `orm:"column(as_info);size(255)" json:"asInfo"`
Query string `orm:"column(query);size(64)" json:"query"`
CreateTime time.Time `orm:"column(create_time);auto_now_add;type(datetime)" json:"createTime"`
}
func (m *PlatformCursorEquipmentIpLog) TableName() string {
return "yz_platform_cursor_equipment_ip_log"
}

View File

@ -0,0 +1,29 @@
package models
// GetPlatformSettingValue 从 yz_system_tenant_setting_items 表中读取平台级别tid=0的配置值。
// key 不存在或读取失败时返回空字符串。
func GetPlatformSettingValue(key string) string {
type row struct {
SettingValue string
}
var r row
// tid=0 表示平台全局配置,与租户无关
err := Orm.Raw(
"SELECT IFNULL(setting_value, '') AS setting_value FROM yz_system_tenant_setting_items WHERE tid = 0 AND setting_key = ? AND delete_time IS NULL LIMIT 1",
key,
).QueryRow(&r)
if err != nil {
return ""
}
return r.SettingValue
}
// SetPlatformSettingValue 写入或更新平台级别tid=0的配置值。
func SetPlatformSettingValue(key, value string) error {
_, err := Orm.Raw(`
INSERT INTO yz_system_tenant_setting_items (tid, setting_key, setting_value, create_time, update_time)
VALUES (0, ?, ?, NOW(), NOW())
ON DUPLICATE KEY UPDATE setting_value = VALUES(setting_value), update_time = NOW(), delete_time = NULL
`, key, value).Exec()
return err
}

View File

@ -171,6 +171,7 @@ func Register() {
beego.Router("/platform/cursor/equipment/activate", &controllers.PlatformCursorEquipmentController{}, "post:Activate")
beego.Router("/platform/cursor/equipment/activationRecords", &controllers.PlatformCursorEquipmentController{}, "get:ActivationRecords")
beego.Router("/platform/cursor/equipment/extractRecords", &controllers.PlatformCursorEquipmentController{}, "get:ExtractRecords")
beego.Router("/platform/cursor/equipment/ipLogs", &controllers.PlatformCursorEquipmentController{}, "get:IpLogs")
// Cursor 激活码管理yz_platform_cursor_activation_code
beego.Router("/platform/cursor/activationcode/list", &controllers.PlatformCursorActivationCodeController{}, "get:List")

View File

@ -18,7 +18,6 @@ declare module 'vue' {
ElAvatar: typeof import('element-plus/es')['ElAvatar']
ElBacktop: typeof import('element-plus/es')['ElBacktop']
ElButton: typeof import('element-plus/es')['ElButton']
ElButtonGroup: typeof import('element-plus/es')['ElButtonGroup']
ElCard: typeof import('element-plus/es')['ElCard']
ElCascader: typeof import('element-plus/es')['ElCascader']
ElCheckbox: typeof import('element-plus/es')['ElCheckbox']
@ -40,7 +39,6 @@ declare module 'vue' {
ElFormItem: typeof import('element-plus/es')['ElFormItem']
ElHeader: typeof import('element-plus/es')['ElHeader']
ElIcon: typeof import('element-plus/es')['ElIcon']
ElImage: typeof import('element-plus/es')['ElImage']
ElInput: typeof import('element-plus/es')['ElInput']
ElInputNumber: typeof import('element-plus/es')['ElInputNumber']
ElLink: typeof import('element-plus/es')['ElLink']

View File

@ -63,3 +63,11 @@ export function getCursorEquipmentExtractRecords(params) {
params,
});
}
export function getCursorEquipmentIpLogs(params) {
return request({
url: `${baseUrl}/ipLogs`,
method: 'get',
params,
});
}

View File

@ -88,3 +88,11 @@ export function getCursorEquipmentExtractRecords(params: Record<string, any>) {
params,
});
}
export function getCursorEquipmentIpLogs(params: Record<string, any>) {
return request({
url: `${baseUrl}/ipLogs`,
method: 'get',
params,
});
}

View File

@ -42,6 +42,12 @@ function copyText(text: unknown, label = '内容') {
ElMessage.success(`${label}已复制`);
});
}
function sourceLabel(source: string) {
if (source === 'report') return '心跳上报';
if (source === 'activateByCode') return '激活码激活';
return source || '-';
}
</script>
<template>
@ -111,6 +117,59 @@ function copyText(text: unknown, label = '内容') {
</el-descriptions-item>
</el-descriptions>
<!-- IP 信息区块 -->
<template v-if="row.raw?.lastLoginIpInfo || row.lastLoginIpInfo">
<el-divider content-position="left">最后登录 IP 详情</el-divider>
<el-descriptions :column="2" border size="small">
<el-descriptions-item label="出口 IP">
<span class="ip-text">{{ display((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.query) }}</span>
<el-button
v-if="(row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.query"
link type="primary"
@click="copyText((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.query, 'IP 地址')"
>复制</el-button>
</el-descriptions-item>
<el-descriptions-item label="来源">
{{ sourceLabel((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.source) }}
</el-descriptions-item>
<el-descriptions-item label="国家">
{{ display((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.country) }}
<span v-if="(row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.countryCode" class="country-code">
{{ (row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.countryCode }}
</span>
</el-descriptions-item>
<el-descriptions-item label="省/州">
{{ display((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.regionName) }}
</el-descriptions-item>
<el-descriptions-item label="城市">
{{ display((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.city) }}
</el-descriptions-item>
<el-descriptions-item label="时区">
{{ display((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.timezone) }}
</el-descriptions-item>
<el-descriptions-item label="ISP 运营商" :span="2">
{{ display((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.isp) }}
</el-descriptions-item>
<el-descriptions-item label="组织" :span="2">
{{ display((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.org) }}
</el-descriptions-item>
<el-descriptions-item label="AS 信息" :span="2">
{{ display((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.asInfo) }}
</el-descriptions-item>
<el-descriptions-item label="经纬度">
{{ (row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.lat }},
{{ (row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.lon }}
</el-descriptions-item>
<el-descriptions-item label="上报时间">
{{ display((row.raw?.lastLoginIpInfo || row.lastLoginIpInfo)?.createTime) }}
</el-descriptions-item>
</el-descriptions>
</template>
<template v-else>
<el-divider content-position="left">最后登录 IP 详情</el-divider>
<div class="no-ip-tip">暂无 IP 记录客户端未上报 ipInfo</div>
</template>
<template #footer>
<el-button @click="emit('update:modelValue', false)">关闭</el-button>
</template>
@ -132,6 +191,24 @@ function copyText(text: unknown, label = '内容') {
line-height: 1.6;
}
.ip-text {
font-family: monospace;
font-weight: 600;
margin-right: 4px;
}
.country-code {
color: #909399;
font-size: 12px;
}
.no-ip-tip {
color: #909399;
font-size: 13px;
padding: 12px 0;
text-align: center;
}
:deep(.equipment-detail-dialog) {
max-width: calc(100vw - 24px);
}

View File

@ -0,0 +1,151 @@
<script lang="ts" setup>
import { ElMessage } from 'element-plus';
const props = defineProps({
modelValue: { type: Boolean, default: false },
row: { type: Object, default: null },
loading: { type: Boolean, default: false },
records: { type: Array as () => Record<string, any>[], default: () => [] },
total: { type: Number, default: 0 },
page: { type: Number, default: 1 },
pageSize: { type: Number, default: 20 },
});
const emit = defineEmits([
'update:modelValue',
'update:page',
'update:page-size',
'refresh',
]);
function handleOpen() {
emit('refresh');
}
function formatTime(value: any) {
if (!value) return '-';
const d = new Date(value);
if (Number.isNaN(d.getTime())) return String(value);
const p = (v: number) => String(v).padStart(2, '0');
return `${d.getFullYear()}-${p(d.getMonth() + 1)}-${p(d.getDate())} ${p(d.getHours())}:${p(d.getMinutes())}:${p(d.getSeconds())}`;
}
function sourceLabel(source: string) {
if (source === 'report') return '心跳上报';
if (source === 'activateByCode') return '激活码激活';
return source || '-';
}
function copyText(text: unknown, label = '内容') {
const value = String(text || '').trim();
if (!value) {
ElMessage.warning(`暂无${label}可复制`);
return;
}
navigator.clipboard.writeText(value).then(() => {
ElMessage.success(`${label}已复制`);
});
}
</script>
<template>
<el-dialog
class="ip-logs-dialog"
:model-value="modelValue"
title="IP 登录日志"
width="900px"
@update:model-value="(v: boolean) => emit('update:modelValue', v)"
@open="handleOpen"
>
<div class="dialog-desc" v-if="row">
<span>设备{{ row.machineCode || row.id || '-' }}</span>
</div>
<el-table v-loading="loading" :data="records" border stripe size="small" style="width: 100%">
<el-table-column label="IP 地址" width="140" prop="query">
<template #default="{ row: r }">
<span class="ip-text">{{ r.query || '-' }}</span>
<el-button v-if="r.query" link size="small" type="primary" @click="copyText(r.query, 'IP')">复制</el-button>
</template>
</el-table-column>
<el-table-column label="来源" width="110" align="center">
<template #default="{ row: r }">
<el-tag size="small" :type="r.source === 'activateByCode' ? 'warning' : 'info'">
{{ sourceLabel(r.source) }}
</el-tag>
</template>
</el-table-column>
<el-table-column label="国家 / 地区" min-width="140">
<template #default="{ row: r }">
<span>{{ r.country || '-' }}</span>
<span v-if="r.countryCode" class="country-code">{{ r.countryCode }}</span>
</template>
</el-table-column>
<el-table-column label="城市" width="110" prop="city">
<template #default="{ row: r }">{{ r.city || '-' }}</template>
</el-table-column>
<el-table-column label="ISP 运营商" min-width="200" show-overflow-tooltip prop="isp">
<template #default="{ row: r }">{{ r.isp || '-' }}</template>
</el-table-column>
<el-table-column label="时区" width="140" show-overflow-tooltip prop="timezone">
<template #default="{ row: r }">{{ r.timezone || '-' }}</template>
</el-table-column>
<el-table-column label="上报时间" width="170">
<template #default="{ row: r }">{{ formatTime(r.createTime) }}</template>
</el-table-column>
</el-table>
<el-pagination
class="pager"
:current-page="page"
:page-size="pageSize"
background
layout="total, sizes, prev, pager, next"
:page-sizes="[20, 50, 100]"
:total="total"
@current-change="(v: number) => emit('update:page', v)"
@size-change="(v: number) => emit('update:page-size', v)"
/>
<template #footer>
<el-button @click="emit('update:modelValue', false)">关闭</el-button>
<el-button type="primary" :loading="loading" @click="emit('refresh')">刷新</el-button>
</template>
</el-dialog>
</template>
<style scoped lang="less">
.dialog-desc {
margin-bottom: 12px;
color: #606266;
font-size: 13px;
}
.ip-text {
font-family: monospace;
font-weight: 600;
margin-right: 4px;
}
.country-code {
color: #909399;
font-size: 12px;
}
.pager {
display: flex;
justify-content: flex-end;
margin-top: 12px;
}
:deep(.ip-logs-dialog) {
max-width: calc(100vw - 24px);
}
@media (max-width: 768px) {
:deep(.ip-logs-dialog) {
width: calc(100vw - 24px) !important;
margin: 0 auto;
}
}
</style>

View File

@ -6,6 +6,7 @@ import EditDialog from './components/edit.vue';
import DeleteDialog from './components/delete.vue';
import ActivationRecords from './components/activationRecords.vue';
import ExtractRecords from './components/extractRecords.vue';
import IpLogs from './components/ipLogs.vue';
import {
activateCursorEquipment,
addCursorEquipment,
@ -13,6 +14,7 @@ import {
getCursorEquipmentActivationRecords,
getCursorEquipmentDetail,
getCursorEquipmentExtractRecords,
getCursorEquipmentIpLogs,
getCursorEquipmentList,
updateCursorEquipment,
} from '../../../api/cursorEquipment';
@ -59,6 +61,16 @@ const extractState = reactive({
pageSize: 20,
});
const ipLogsState = reactive({
loading: false,
records: [] as EquipmentRow[],
total: 0,
page: 1,
pageSize: 20,
});
const ipLogsVisible = ref(false);
const statusOptions = [
{ label: '未激活', value: 0 },
{ label: '已激活', value: 1 },
@ -122,6 +134,13 @@ watch(
},
);
watch(
() => [ipLogsState.page, ipLogsState.pageSize],
() => {
if (ipLogsVisible.value) fetchIpLogs();
},
);
function pick(raw: any, ...keys: string[]) {
for (const key of keys) {
if (raw?.[key] !== undefined && raw?.[key] !== null) return raw[key];
@ -167,6 +186,8 @@ function normalizeRow(raw: any): EquipmentRow {
lastExtractedAt: formatTime(pick(raw, 'last_extracted_at', 'lastExtractedAt', 'extracted_at')),
expiredAt: formatTime(pick(raw, 'expiredAt', 'expired_at', 'expireTime', 'expire_time')),
createdAt: formatTime(pick(raw, 'create_time', 'created_at', 'createdAt', 'CreatedAt')),
lastLoginIp: pick(raw, 'lastLoginIp', 'last_login_ip') || '',
lastLoginIpInfo: raw?.lastLoginIpInfo ?? null,
remark: pick(raw, 'remark', 'Remark'),
raw,
};
@ -343,6 +364,14 @@ function openExtractRecords(row: EquipmentRow) {
extractVisible.value = true;
}
function openIpLogs(row: EquipmentRow) {
currentRow.value = row;
ipLogsState.page = 1;
ipLogsState.records = [];
ipLogsState.total = 0;
ipLogsVisible.value = true;
}
async function fetchActivationRecords() {
if (!currentRow.value?.id) return;
activationState.loading = true;
@ -385,6 +414,27 @@ async function fetchExtractRecords() {
}
}
async function fetchIpLogs() {
if (!currentRow.value?.id) return;
ipLogsState.loading = true;
try {
const res = await getCursorEquipmentIpLogs({
equipmentId: currentRow.value.id,
page: ipLogsState.page,
pageSize: ipLogsState.pageSize,
});
if (res?.code !== 200) {
ElMessage.error(res?.msg || '获取 IP 日志失败');
return;
}
const list = Array.isArray(res?.data?.list) ? res.data.list : Array.isArray(res?.data) ? res.data : [];
ipLogsState.records = list;
ipLogsState.total = Number(res?.data?.total || list.length || 0);
} finally {
ipLogsState.loading = false;
}
}
function updateDeviceType() {
isMobile.value = window.innerWidth <= 768;
}
@ -483,6 +533,11 @@ onUnmounted(() => {
<el-table-column prop="lastActivatedAt" label="最后激活" width="180">
<template #default="{ row }">{{ row.lastActivatedAt || '-' }}</template>
</el-table-column>
<el-table-column label="最后登录 IP" width="140" align="center">
<template #default="{ row }">
<span class="ip-col-text">{{ row.lastLoginIp || '-' }}</span>
</template>
</el-table-column>
<el-table-column prop="expiredAt" label="过期时间" width="180">
<template #default="{ row }">{{ row.expiredAt || '-' }}</template>
</el-table-column>
@ -494,6 +549,7 @@ onUnmounted(() => {
<el-button v-if="row.status !== 'active'" link type="success" @click="handleActivate(row)">激活</el-button>
<el-button link type="info" @click="openActivationRecords(row)">激活记录</el-button>
<el-button link type="info" @click="openExtractRecords(row)">提取记录</el-button>
<el-button link type="primary" @click="openIpLogs(row)">IP日志</el-button>
<el-button link type="danger" @click="openDelete(row)">删除</el-button>
</template>
</el-table-column>
@ -549,6 +605,17 @@ onUnmounted(() => {
:total="extractState.total"
@refresh="fetchExtractRecords"
/>
<IpLogs
v-model="ipLogsVisible"
v-model:page="ipLogsState.page"
v-model:page-size="ipLogsState.pageSize"
:row="currentRow || undefined"
:loading="ipLogsState.loading"
:records="ipLogsState.records"
:total="ipLogsState.total"
@refresh="fetchIpLogs"
/>
</div>
</template>
@ -657,6 +724,12 @@ onUnmounted(() => {
font-size: 12px;
}
.ip-col-text {
font-family: monospace;
font-size: 12px;
color: #303133;
}
.pager {
display: flex;
justify-content: flex-end;

View File

@ -0,0 +1,28 @@
-- 设备 IP 登录日志表
-- 每次客户端调用 report / activateByCode 接口时,若携带 ipInfo则向本表插入一条记录
CREATE TABLE IF NOT EXISTS `yz_platform_cursor_equipment_ip_log` (
`id` BIGINT UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '主键',
`equipment_id` BIGINT UNSIGNED NOT NULL DEFAULT 0 COMMENT '关联 yz_platform_cursor_equipment.id0 表示设备尚未创建时的记录',
`machine_code` VARCHAR(128) NOT NULL DEFAULT '' COMMENT '机器码(冗余,便于无 equipment_id 时查询)',
`source` VARCHAR(32) NOT NULL DEFAULT '' COMMENT '来源report / activateByCode',
-- ip-api.com 返回的原始字段
`status` VARCHAR(32) NOT NULL DEFAULT '' COMMENT 'ip-api status',
`country` VARCHAR(64) NOT NULL DEFAULT '' COMMENT '国家',
`country_code` VARCHAR(8) NOT NULL DEFAULT '' COMMENT '国家代码',
`region` VARCHAR(16) NOT NULL DEFAULT '' COMMENT '省/州代码',
`region_name` VARCHAR(128) NOT NULL DEFAULT '' COMMENT '省/州名称',
`city` VARCHAR(128) NOT NULL DEFAULT '' COMMENT '城市',
`zip` VARCHAR(32) NOT NULL DEFAULT '' COMMENT '邮编',
`lat` DOUBLE NOT NULL DEFAULT 0 COMMENT '纬度',
`lon` DOUBLE NOT NULL DEFAULT 0 COMMENT '经度',
`timezone` VARCHAR(64) NOT NULL DEFAULT '' COMMENT '时区',
`isp` VARCHAR(255) NOT NULL DEFAULT '' COMMENT 'ISP 运营商',
`org` VARCHAR(255) NOT NULL DEFAULT '' COMMENT '组织',
`as_info` VARCHAR(255) NOT NULL DEFAULT '' COMMENT 'AS 信息',
`query` VARCHAR(64) NOT NULL DEFAULT '' COMMENT '出口 IP 地址',
`create_time` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '记录时间',
PRIMARY KEY (`id`),
KEY `idx_equipment_id` (`equipment_id`),
KEY `idx_machine_code` (`machine_code`),
KEY `idx_create_time` (`create_time`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='设备 IP 登录日志';