using AutoMapper;
using DMS.Application.DTOs;
using DMS.Application.Interfaces;
using DMS.Application.Interfaces.Database;
using DMS.Core.Interfaces;
using DMS.Core.Models;
namespace DMS.Application.Services.Database;
///
/// 变量应用服务,负责处理变量相关的业务逻辑。
/// 实现 接口。
///
public class VariableAppService : IVariableAppService
{
private readonly IRepositoryManager _repoManager;
private readonly IMapper _mapper;
///
/// 构造函数,通过依赖注入获取仓储管理器和AutoMapper实例。
///
/// 仓储管理器实例。
/// AutoMapper 实例。
public VariableAppService(IRepositoryManager repoManager, IMapper mapper)
{
_repoManager = repoManager;
_mapper = mapper;
}
///
/// 异步根据ID获取变量数据传输对象。
///
/// 变量ID。
/// 变量数据传输对象。
public async Task GetVariableByIdAsync(int id)
{
var variable = await _repoManager.Variables.GetByIdAsync(id);
return _mapper.Map(variable);
}
///
/// 异步根据OPC UA NodeId获取变量数据传输对象。
///
/// OPC UA NodeId。
/// 变量数据传输对象。
public async Task GetVariableByOpcUaNodeIdAsync(string opcUaNodeId)
{
var variable = await _repoManager.Variables.GetByOpcUaNodeIdAsync(opcUaNodeId);
return variable == null ? null : _mapper.Map(variable);
}
///
/// 异步根据OPC UA NodeId列表获取变量数据传输对象列表。
///
/// OPC UA NodeId列表。
/// 变量数据传输对象列表。
public async Task> GetVariableByOpcUaNodeIdsAsync(List opcUaNodeIds)
{
var variables = await _repoManager.Variables.GetByOpcUaNodeIdsAsync(opcUaNodeIds);
return _mapper.Map>(variables);
}
///
/// 异步获取所有变量数据传输对象列表。
///
/// 变量数据传输对象列表。
public async Task> GetAllVariablesAsync()
{
var variables = await _repoManager.Variables.GetAllAsync();
return _mapper.Map>(variables);
}
///
/// 异步创建一个新变量(事务性操作)。
///
/// 要创建的变量数据传输对象。
/// 新创建的变量数据传输对象。
/// 如果创建变量时发生错误。
public async Task CreateVariableAsync(VariableDto variableDto)
{
try
{
await _repoManager.BeginTranAsync();
var variable = _mapper.Map(variableDto);
var addedVariable = await _repoManager.Variables.AddAsync(variable);
await _repoManager.CommitAsync();
return _mapper.Map(addedVariable);
}
catch (Exception ex)
{
await _repoManager.RollbackAsync();
throw new ApplicationException($"创建变量时发生错误,操作已回滚,错误信息:{ex.Message}", ex);
}
}
///
/// 异步更新一个已存在的变量(事务性操作)。
///
/// 要更新的变量数据传输对象。
/// 受影响的行数。
/// 如果找不到变量或更新变量时发生错误。
public async Task UpdateVariableAsync(VariableDto variableDto)
{
try
{
await _repoManager.BeginTranAsync();
var variable = await _repoManager.Variables.GetByIdAsync(variableDto.Id);
if (variable == null)
{
throw new ApplicationException($"Variable with ID {variableDto.Id} not found.");
}
_mapper.Map(variableDto, variable);
int res = await _repoManager.Variables.UpdateAsync(variable);
await _repoManager.CommitAsync();
return res;
}
catch (Exception ex)
{
await _repoManager.RollbackAsync();
throw new ApplicationException($"更新变量时发生错误,操作已回滚,错误信息:{ex.Message}", ex);
}
}
///
/// 异步批量更新变量(事务性操作)。
///
/// 要更新的变量数据传输对象列表。
/// 受影响的行数。
/// 如果更新变量时发生错误。
public async Task UpdateVariablesAsync(List variableDtos)
{
try
{
await _repoManager.BeginTranAsync();
int totalAffected = 0;
foreach (var variableDto in variableDtos)
{
var variable = await _repoManager.Variables.GetByIdAsync(variableDto.Id);
if (variable == null)
{
throw new ApplicationException($"Variable with ID {variableDto.Id} not found.");
}
_mapper.Map(variableDto, variable);
int res = await _repoManager.Variables.UpdateAsync(variable);
totalAffected += res;
}
await _repoManager.CommitAsync();
return totalAffected;
}
catch (Exception ex)
{
await _repoManager.RollbackAsync();
throw new ApplicationException($"批量更新变量时发生错误,操作已回滚,错误信息:{ex.Message}", ex);
}
}
///
/// 异步删除一个变量(事务性操作)。
///
/// 要删除变量的ID。
/// 如果删除成功则为 true,否则为 false。
/// 如果删除变量失败。
/// 如果删除变量时发生其他错误。
public async Task DeleteVariableAsync(int id)
{
try
{
await _repoManager.BeginTranAsync();
var delRes = await _repoManager.Variables.DeleteByIdAsync(id);
if (delRes == 0)
{
throw new InvalidOperationException($"删除变量失败:变量ID:{id},请检查变量Id是否存在");
}
await _repoManager.CommitAsync();
return true;
}
catch (Exception ex)
{
await _repoManager.RollbackAsync();
throw new ApplicationException($"删除变量时发生错误,操作已回滚,错误信息:{ex.Message}", ex);
}
}
///
/// 异步批量删除变量(事务性操作)。
///
/// 要删除的变量ID列表。
/// 如果删除成功则为 true,否则为 false。
/// 如果ID列表为空或null。
/// 如果删除变量时发生错误。
public async Task DeleteVariablesAsync(List ids)
{
if (ids == null || !ids.Any())
{
throw new ArgumentException("变量ID列表不能为空", nameof(ids));
}
try
{
await _repoManager.BeginTranAsync();
// 批量删除变量
var deletedCount = await _repoManager.Variables.DeleteByIdsAsync(ids);
// 检查是否所有变量都被成功删除
if (deletedCount != ids.Count)
{
throw new InvalidOperationException($"删除变量失败:请求删除 {ids.Count} 个变量,实际删除 {deletedCount} 个变量");
}
await _repoManager.CommitAsync();
return true;
}
catch (Exception ex)
{
await _repoManager.RollbackAsync();
throw new ApplicationException($"批量删除变量时发生错误,操作已回滚,错误信息:{ex.Message}", ex);
}
}
public async Task> BatchImportVariablesAsync(List variables)
{
try
{
var variableModels = _mapper.Map>(variables);
var addedVariables = await _repoManager.Variables.AddBatchAsync(variableModels);
return _mapper.Map>(addedVariables);
}
catch (Exception ex)
{
throw new ApplicationException($"批量导入变量时发生错误,错误信息:{ex.Message}", ex);
}
}
public async Task> FindExistingVariablesAsync(IEnumerable variablesToCheck)
{
if (variablesToCheck == null || !variablesToCheck.Any())
{
return new List();
}
var names = variablesToCheck.Select(v => v.Name).Where(n => !string.IsNullOrEmpty(n)).Distinct().ToList();
var s7Addresses = variablesToCheck.Select(v => v.S7Address).Where(a => !string.IsNullOrEmpty(a)).Distinct().ToList();
var opcUaNodeIds = variablesToCheck.Select(v => v.OpcUaNodeId).Where(id => !string.IsNullOrEmpty(id)).Distinct().ToList();
var allVariables = await _repoManager.Variables.GetAllAsync();
var existingVariablesFromDb = allVariables.Where(v =>
(names.Any() && !string.IsNullOrEmpty(v.Name) && names.Contains(v.Name)) ||
(s7Addresses.Any() && !string.IsNullOrEmpty(v.S7Address) && s7Addresses.Contains(v.S7Address)) ||
(opcUaNodeIds.Any() && !string.IsNullOrEmpty(v.OpcUaNodeId) && opcUaNodeIds.Contains(v.OpcUaNodeId)))
.ToList();
if (existingVariablesFromDb == null || !existingVariablesFromDb.Any())
{
return new List();
}
var existingNames = new HashSet(existingVariablesFromDb.Select(v => v.Name).Where(n => !string.IsNullOrEmpty(n)));
var existingS7Addresses = new HashSet(existingVariablesFromDb.Select(v => v.S7Address).Where(a => !string.IsNullOrEmpty(a)));
var existingOpcUaNodeIds = new HashSet(existingVariablesFromDb.Select(v => v.OpcUaNodeId).Where(id => !string.IsNullOrEmpty(id)));
var result = variablesToCheck.Where(v =>
(!string.IsNullOrEmpty(v.Name) && existingNames.Contains(v.Name)) ||
(!string.IsNullOrEmpty(v.S7Address) && existingS7Addresses.Contains(v.S7Address)) ||
(!string.IsNullOrEmpty(v.OpcUaNodeId) && existingOpcUaNodeIds.Contains(v.OpcUaNodeId)))
.ToList();
return result;
}
public async Task FindExistingVariableAsync(VariableDto variableToCheck)
{
if (variableToCheck == null)
{
return null;
}
// 创建一个包含单个元素的列表以便复用现有的逻辑
var variablesToCheck = new List { variableToCheck };
var existingVariables = await FindExistingVariablesAsync(variablesToCheck);
// 如果找到了匹配的变量,返回第一个(也是唯一一个)
return existingVariables.FirstOrDefault();
}
}