195 lines
7.9 KiB
C#
195 lines
7.9 KiB
C#
/*
|
||
*所有关于gateway_nodes类的业务代码应在此处编写
|
||
*可使用repository.调用常用方法,获取EF/Dapper等信息
|
||
*如果需要事务请使用repository.DbContextBeginTransaction
|
||
*也可使用DBServerProvider.手动获取数据库相关信息
|
||
*用户信息、权限、角色等使用UserContext.Current操作
|
||
*gateway_nodesService对增、删、改查、导入、导出、审核业务代码扩展参照ServiceFunFilter
|
||
*/
|
||
using VolPro.Core.BaseProvider;
|
||
using VolPro.Core.Extensions.AutofacManager;
|
||
using VolPro.Entity.DomainModels;
|
||
using System.Linq;
|
||
using VolPro.Core.Utilities;
|
||
using System.Linq.Expressions;
|
||
using VolPro.Core.Extensions;
|
||
using Microsoft.EntityFrameworkCore;
|
||
using Microsoft.Extensions.DependencyInjection;
|
||
using Microsoft.AspNetCore.Http;
|
||
using Warehouse.IRepositories;
|
||
using System;
|
||
using System.Collections.Generic;
|
||
using System.Threading.Tasks;
|
||
using System.Text.Json;
|
||
|
||
namespace Warehouse.Services
|
||
{
|
||
/// <summary>
|
||
/// gateway_nodes 业务逻辑(partial)。注册/心跳/设备同步。
|
||
/// </summary>
|
||
public partial class gateway_nodesService
|
||
{
|
||
private readonly IHttpContextAccessor _httpContextAccessor;
|
||
private readonly Igateway_nodesRepository _repository;
|
||
|
||
[ActivatorUtilitiesConstructor]
|
||
public gateway_nodesService(
|
||
Igateway_nodesRepository dbRepository,
|
||
IHttpContextAccessor httpContextAccessor
|
||
)
|
||
: base(dbRepository)
|
||
{
|
||
_httpContextAccessor = httpContextAccessor;
|
||
_repository = dbRepository;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 网关注册(Upsert)。
|
||
/// NodeCode 匹配则更新适配器类型/地址/在线状态;
|
||
/// NodeCode 不匹配且 Token 验证通过则插入新记录。
|
||
/// </summary>
|
||
[Obsolete("由 A1 API Controller 自动调用,不建议手动调用")]
|
||
public async Task<gateway_nodes> RegisterNodeAsync(string nodeCode, string token, string adapterTypes, string baseUrl)
|
||
{
|
||
var existingList = await _repository.FindAsIQueryable(x => x.NodeCode == nodeCode).ToListAsync();
|
||
var existing = existingList.FirstOrDefault();
|
||
|
||
gateway_nodes entity;
|
||
if (existing != null)
|
||
{
|
||
if (existing.NodeToken != token)
|
||
throw new UnauthorizedAccessException("NodeToken 不匹配");
|
||
|
||
existing.AdapterTypes = adapterTypes;
|
||
existing.BaseUrl = baseUrl;
|
||
existing.IsOnline = "在线";
|
||
existing.LastHeartbeat = DateTime.Now;
|
||
_repository.DbContext.Updateable(existing).ExecuteCommand();
|
||
entity = existing;
|
||
}
|
||
else
|
||
{
|
||
entity = new gateway_nodes
|
||
{
|
||
NodeCode = nodeCode,
|
||
NodeName = nodeCode,
|
||
NodeToken = token,
|
||
AdapterTypes = adapterTypes,
|
||
BaseUrl = baseUrl,
|
||
IsOnline = "在线",
|
||
Enable = "启用",
|
||
LastHeartbeat = DateTime.Now,
|
||
CreateDate = DateTime.Now
|
||
};
|
||
_repository.DbContext.Insertable(entity).ExecuteCommand();
|
||
}
|
||
return entity;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 心跳更新。更新 LastHeartbeat 并标记在线。
|
||
/// </summary>
|
||
[Obsolete("由 A2 API Controller 自动调用,不建议手动调用")]
|
||
public async Task UpdateHeartbeatAsync(string nodeCode, string token)
|
||
{
|
||
var entityList = await _repository.FindAsIQueryable(x => x.NodeCode == nodeCode && x.NodeToken == token).ToListAsync();
|
||
var entity = entityList.FirstOrDefault();
|
||
if (entity == null)
|
||
throw new UnauthorizedAccessException("认证失败:NodeCode 或 Token 无效");
|
||
|
||
entity.IsOnline = "在线";
|
||
entity.LastHeartbeat = DateTime.Now;
|
||
_repository.DbContext.Updateable(entity).ExecuteCommand();
|
||
}
|
||
|
||
/// <summary>
|
||
/// 设备数据同步。按字段分治原则写入 base_device:
|
||
/// 首次入库写全量,后续仅更新网关字段。
|
||
/// parentSourceId 解析为 ParentDeviceId。
|
||
/// </summary>
|
||
[Obsolete("由 A3 API Controller 自动调用,不建议手动调用")]
|
||
public async Task<(int added, int updated)> SyncDevicesAsync(int gatewayNodeId, List<SyncDeviceItem> devices)
|
||
{
|
||
var db = _repository.DbContext;
|
||
|
||
var adapterCodes = devices.Select(d => d.AdapterCode).Distinct().ToList();
|
||
var existingIds = db.Queryable<base_device>()
|
||
.Where(x => x.NodeId == gatewayNodeId && adapterCodes.Contains(x.AdapterCode))
|
||
.ToList()
|
||
.ToDictionary(x => (x.AdapterCode, x.SourceId), x => x.DeviceId);
|
||
|
||
int added = 0, updated = 0;
|
||
foreach (var d in devices)
|
||
{
|
||
var key = (d.AdapterCode, d.SourceId);
|
||
existingIds.TryGetValue(key, out var existingId);
|
||
bool isNew = existingId == 0;
|
||
|
||
int? parentDeviceId = null;
|
||
if (!string.IsNullOrEmpty(d.ParentSourceId))
|
||
{
|
||
existingIds.TryGetValue((d.AdapterCode, d.ParentSourceId), out var pid);
|
||
if (pid > 0) parentDeviceId = pid;
|
||
}
|
||
|
||
if (isNew)
|
||
{
|
||
var entity = new base_device
|
||
{
|
||
DeviceName = d.Name ?? $"DEV_{d.SourceId}",
|
||
AdapterCode = d.AdapterCode,
|
||
SourceId = d.SourceId,
|
||
DeviceCategory = d.Category,
|
||
DeviceGroup = d.Group,
|
||
NodeId = gatewayNodeId,
|
||
IsParent = d.IsParent ? "是" : "否",
|
||
ParentDeviceId = parentDeviceId,
|
||
IsOnline = d.IsOnline ? "在线" : "离线",
|
||
IpAddress = d.IpAddress,
|
||
Port = d.Port,
|
||
ExtraData = d.ExtraDataJson,
|
||
Enable = "启用",
|
||
LastSyncTime = DateTime.Now,
|
||
CreateDate = DateTime.Now
|
||
};
|
||
db.Insertable(entity).ExecuteCommand();
|
||
added++;
|
||
}
|
||
else
|
||
{
|
||
var entity = db.Queryable<base_device>().InSingle(existingId);
|
||
if (entity != null)
|
||
{
|
||
entity.IsOnline = d.IsOnline ? "在线" : "离线";
|
||
entity.IsParent = d.IsParent ? "是" : "否";
|
||
entity.ParentDeviceId = parentDeviceId ?? entity.ParentDeviceId;
|
||
entity.IpAddress = d.IpAddress;
|
||
entity.Port = d.Port;
|
||
entity.ExtraData = d.ExtraDataJson ?? entity.ExtraData;
|
||
entity.LastSyncTime = DateTime.Now;
|
||
db.Updateable(entity).ExecuteCommand();
|
||
updated++;
|
||
}
|
||
}
|
||
}
|
||
return (added, updated);
|
||
}
|
||
}
|
||
|
||
/// <summary>网关同步设备条目</summary>
|
||
public class SyncDeviceItem
|
||
{
|
||
public string AdapterCode { get; set; } = "";
|
||
public string SourceId { get; set; } = "";
|
||
public string? Name { get; set; }
|
||
public string? Category { get; set; }
|
||
public string? Group { get; set; }
|
||
public bool IsParent { get; set; }
|
||
public string? ParentSourceId { get; set; }
|
||
public bool IsOnline { get; set; }
|
||
public string? IpAddress { get; set; }
|
||
public int? Port { get; set; }
|
||
public string? ExtraDataJson { get; set; }
|
||
}
|
||
}
|