| | |
| | | using Newtonsoft.Json; |
| | | using OfficeOpenXml.FormulaParsing.Excel.Functions.RefAndLookup; |
| | | using System; |
| | | using System.Collections.Concurrent; |
| | | using System.Collections.Generic; |
| | | using System.Linq; |
| | | using System.Text; |
| | |
| | | List<Dt_LocationInfo>? locationInfos = null; |
| | | |
| | | CleanupPreviousInvalidLocks(outboundOrderDetails); |
| | | // å¼å¯äºå¡ï¼ä½¿ç¨æ°æ®åºè¡çº§é |
| | | using (var transaction = _outboundOrderDetailService.Db.Ado.UseTran()) |
| | | { |
| | | try |
| | | { |
| | | // ä½¿ç¨æ²è§ééå®è®¢åæç» |
| | | var lockedOrderDetails = new List<Dt_OutboundOrderDetail>(); |
| | | foreach (var key in keys) |
| | | { |
| | | var detail = _outboundOrderDetailService.Db.Ado.SqlQuerySingle<Dt_OutboundOrderDetail>( |
| | | "SELECT * FROM Dt_OutboundOrderDetail WITH (UPDLOCK, ROWLOCK) WHERE Id = @Id", |
| | | new { Id = key }); |
| | | |
| | | if (detail != null) |
| | | { |
| | | lockedOrderDetails.Add(detail); |
| | | } |
| | | } |
| | | |
| | | if (!lockedOrderDetails.Any()) |
| | | { |
| | | throw new Exception("æªæ¾å°åºåºåæç»ä¿¡æ¯"); |
| | | } |
| | | (List<Dt_StockInfo>, List<Dt_OutboundOrderDetail>, List<Dt_OutStockLockInfo>, List<Dt_LocationInfo>) result = _outboundOrderDetailService.AssignStockOutbound(outboundOrderDetails); |
| | | if (result.Item1 != null && result.Item1.Count > 0) |
| | | { |
| | |
| | | orderDetails = result.Item2; |
| | | outStockLockInfos = result.Item3; |
| | | locationInfos = result.Item4; |
| | | transaction.CommitTran(); |
| | | } |
| | | else |
| | | { |
| | | transaction.RollbackTran(); |
| | | throw new Exception("æ åºå"); |
| | | } |
| | | } |
| | | catch (Exception) |
| | | { |
| | | transaction.RollbackTran(); |
| | | throw; |
| | | } |
| | | return (tasks, stockInfos, orderDetails, outStockLockInfos, locationInfos); |
| | | } |
| | | } |
| | | /// <summary> |
| | | /// æ¸
çä¹åçæ æéå®è®°å½ |
| | |
| | | return tasks; |
| | | } |
| | | |
| | | #region å
åé管çå¨ |
| | | private static readonly ConcurrentDictionary<string, SemaphoreSlim> _normalmaterialLocks = |
| | | new ConcurrentDictionary<string, SemaphoreSlim>(); |
| | | private static readonly ConcurrentDictionary<string, DateTime> _normallockLastUsed = |
| | | new ConcurrentDictionary<string, DateTime>(); |
| | | private static readonly object _normalcleanupLock = new object(); |
| | | private static DateTime _normallastCleanupTime = DateTime.MinValue; |
| | | |
| | | /// <summary> |
| | | /// è·åç©æçº§å
åé |
| | | /// </summary> |
| | | private SemaphoreSlim GetNormalMaterialSemaphore(string materialCode, string batchNo, string supplyCode) |
| | | { |
| | | // å建éé®ï¼ç©æ+æ¹æ¬¡+ä¾åºå |
| | | string lockKey = $"MaterialLock_{materialCode}_{batchNo}_{supplyCode}"; |
| | | |
| | | // æ¸
çé¿æ¶é´ä¸ç¨çéï¼æ¯å°æ¶æ¸
ç䏿¬¡ï¼ |
| | | var now = DateTime.Now; |
| | | if ((now - _normallastCleanupTime).TotalHours >= 1) |
| | | { |
| | | lock (_normalcleanupLock) |
| | | { |
| | | if ((now - _normallastCleanupTime).TotalHours >= 1) |
| | | { |
| | | var keysToRemove = _normallockLastUsed |
| | | .Where(kvp => (now - kvp.Value).TotalHours > 2) |
| | | .Select(kvp => kvp.Key) |
| | | .ToList(); |
| | | |
| | | foreach (var key in keysToRemove) |
| | | { |
| | | if (_normalmaterialLocks.TryRemove(key, out var _semaphore)) |
| | | { |
| | | _semaphore.Dispose(); |
| | | } |
| | | _normallockLastUsed.TryRemove(key, out _); |
| | | } |
| | | |
| | | _normallastCleanupTime = now; |
| | | } |
| | | } |
| | | } |
| | | |
| | | // è·åæå建信å·é |
| | | var semaphore = _normalmaterialLocks.GetOrAdd(lockKey, _ => new SemaphoreSlim(1, 1)); |
| | | _normallockLastUsed[lockKey] = now; |
| | | |
| | | return semaphore; |
| | | } |
| | | |
| | | /// <summary> |
| | | /// æ´æ°å
åéæåä½¿ç¨æ¶é´ |
| | | /// </summary> |
| | | private void UpdateNormalMaterialLockUsedTime(string materialCode, string batchNo, string supplyCode) |
| | | { |
| | | string lockKey = $"MaterialLock_{materialCode}_{batchNo}_{supplyCode}"; |
| | | _normallockLastUsed[lockKey] = DateTime.Now; |
| | | } |
| | | #endregion |
| | | /// <summary> |
| | | /// çæåºåºä»»å¡ |
| | | /// </summary> |
| | |
| | | List<Dt_OutboundOrderDetail> outboundOrderDetails = new List<Dt_OutboundOrderDetail>(); |
| | | List<Dt_OutStockLockInfo> outStockLockInfos = new List<Dt_OutStockLockInfo>(); |
| | | List<Dt_LocationInfo> locationInfos = new List<Dt_LocationInfo>(); |
| | | // å
è·åææè®¢åæç»ï¼ç¡®å®éè¦éå®çç©æ |
| | | var orderDetails = _outboundOrderDetailService.Repository.QueryData(x => keys.Contains(x.Id)); |
| | | if (orderDetails == null || orderDetails.Count == 0) |
| | | { |
| | | return WebResponseContent.Instance.Error("æªæ¾å°åºåºåæç»ä¿¡æ¯"); |
| | | } |
| | | |
| | | // è·åææéè¦éå®çç©æåç» |
| | | var materialGroups = orderDetails |
| | | .GroupBy(x => new { x.MaterielCode, x.BatchNo, x.SupplyCode }) |
| | | .Select(g => new |
| | | { |
| | | g.Key.MaterielCode, |
| | | g.Key.BatchNo, |
| | | g.Key.SupplyCode, |
| | | Count = g.Count() |
| | | }) |
| | | .ToList(); |
| | | |
| | | // æé¡ºåºè·åææç©æçå
åéï¼æç©æä»£ç æåºä»¥é¿å
æ»éï¼ |
| | | var semaphores = new List<SemaphoreSlim>(); |
| | | var acquiredLocks = new List<(string MaterialCode, string BatchNo, string SupplyCode)>(); |
| | | |
| | | try |
| | | { |
| | | foreach (var group in materialGroups.OrderBy(g => g.MaterielCode).ThenBy(g => g.BatchNo)) |
| | | { |
| | | var semaphore = GetMaterialSemaphore(group.MaterielCode, group.BatchNo, group.SupplyCode); |
| | | |
| | | // çå¾
è·åéï¼æå¤çå¾
30ç§ |
| | | bool lockAcquired = await semaphore.WaitAsync(TimeSpan.FromSeconds(30)); |
| | | |
| | | if (!lockAcquired) |
| | | { |
| | | // 妿è·åé失败ï¼éæ¾å·²è·åçææé |
| | | foreach (var acquiredSemaphore in semaphores) |
| | | { |
| | | acquiredSemaphore.Release(); |
| | | } |
| | | return WebResponseContent.Instance.Error($"ç©æ[{group.MaterielCode}]æ¹æ¬¡[{group.BatchNo}]åé
ç¹å¿ï¼è¯·ç¨åéè¯"); |
| | | } |
| | | |
| | | semaphores.Add(semaphore); |
| | | acquiredLocks.Add((group.MaterielCode, group.BatchNo, group.SupplyCode)); |
| | | } |
| | | |
| | | (List<Dt_Task>, List<Dt_StockInfo>?, List<Dt_OutboundOrderDetail>?, List<Dt_OutStockLockInfo>?, List<Dt_LocationInfo>?) result = OutboundTaskDataHandle(keys, outStation); |
| | | if (result.Item2 != null && result.Item2.Count > 0) |
| | |
| | | |
| | | WebResponseContent content = await GenerateOutboundTaskDataUpdateAsync(tasks, stockInfos, outboundOrderDetails, outStockLockInfos, locationInfos); |
| | | return content; |
| | | } |
| | | finally |
| | | { |
| | | // éæ¾ææå
åéå¹¶æ´æ°ä½¿ç¨æ¶é´ |
| | | foreach (var semaphore in semaphores) |
| | | { |
| | | semaphore.Release(); |
| | | } |
| | | |
| | | foreach (var lockInfo in acquiredLocks) |
| | | { |
| | | UpdateMaterialLockUsedTime(lockInfo.MaterialCode, lockInfo.BatchNo, lockInfo.SupplyCode); |
| | | } |
| | | } |
| | | } |
| | | catch (Exception ex) |
| | | { |
| | |
| | | |
| | | |
| | | #region åæ¹åé
åºå |
| | | #region å
åé管çå¨ |
| | | private static readonly ConcurrentDictionary<string, SemaphoreSlim> _materialLocks = |
| | | new ConcurrentDictionary<string, SemaphoreSlim>(); |
| | | private static readonly ConcurrentDictionary<string, DateTime> _lockLastUsed = |
| | | new ConcurrentDictionary<string, DateTime>(); |
| | | private static readonly object _cleanupLock = new object(); |
| | | private static DateTime _lastCleanupTime = DateTime.MinValue; |
| | | |
| | | /// <summary> |
| | | /// è·åç©æçº§å
åé |
| | | /// </summary> |
| | | private SemaphoreSlim GetMaterialSemaphore(string materialCode, string batchNo, string supplyCode) |
| | | { |
| | | // å建éé®ï¼ç©æ+æ¹æ¬¡+ä¾åºå |
| | | string lockKey = $"MaterialLock_{materialCode}_{batchNo}_{supplyCode}"; |
| | | |
| | | // æ¸
çé¿æ¶é´ä¸ç¨çéï¼æ¯å°æ¶æ¸
ç䏿¬¡ï¼ |
| | | var now = DateTime.Now; |
| | | if ((now - _lastCleanupTime).TotalHours >= 1) |
| | | { |
| | | lock (_cleanupLock) |
| | | { |
| | | if ((now - _lastCleanupTime).TotalHours >= 1) |
| | | { |
| | | var keysToRemove = _lockLastUsed |
| | | .Where(kvp => (now - kvp.Value).TotalHours > 2) |
| | | .Select(kvp => kvp.Key) |
| | | .ToList(); |
| | | |
| | | foreach (var key in keysToRemove) |
| | | { |
| | | if (_materialLocks.TryRemove(key, out var _semaphore)) |
| | | { |
| | | _semaphore.Dispose(); |
| | | } |
| | | _lockLastUsed.TryRemove(key, out _); |
| | | } |
| | | |
| | | _lastCleanupTime = now; |
| | | } |
| | | } |
| | | } |
| | | |
| | | // è·åæå建信å·é |
| | | var semaphore = _materialLocks.GetOrAdd(lockKey, _ => new SemaphoreSlim(1, 1)); |
| | | _lockLastUsed[lockKey] = now; |
| | | |
| | | return semaphore; |
| | | } |
| | | |
| | | /// <summary> |
| | | /// éæ¾å
åéå¹¶æ´æ°æåä½¿ç¨æ¶é´ |
| | | /// </summary> |
| | | private void UpdateMaterialLockUsedTime(string materialCode, string batchNo, string supplyCode) |
| | | { |
| | | string lockKey = $"MaterialLock_{materialCode}_{batchNo}_{supplyCode}"; |
| | | _lockLastUsed[lockKey] = DateTime.Now; |
| | | } |
| | | #endregion |
| | | /// <summary> |
| | | /// åæ¹åé
åºå |
| | | /// </summary> |
| | | public async Task<WebResponseContent> GenerateOutboundBatchTasksAsync(int orderDetailId, decimal batchQuantity, string outStation) |
| | | { |
| | | try |
| | | { |
| | | // å
è·å订åæç»ä¿¡æ¯ï¼ç¡®å®ç©æ |
| | | var orderDetail = await _outboundOrderDetailService.Db.Queryable<Dt_OutboundOrderDetail>() |
| | | .FirstAsync(x => x.Id == orderDetailId); |
| | | |
| | | if (orderDetail == null) |
| | | return WebResponseContent.Instance.Error("æªæ¾å°è®¢åæç»ä¿¡æ¯"); |
| | | |
| | | // è·åç©æçº§å
åé |
| | | var semaphore = GetMaterialSemaphore(orderDetailId + orderDetail.MaterielCode, orderDetail.BatchNo, orderDetail.SupplyCode); |
| | | |
| | | // çå¾
è·åå
åéï¼æå¤çå¾
30ç§ |
| | | bool memoryLockAcquired = await semaphore.WaitAsync(TimeSpan.FromSeconds(30)); |
| | | |
| | | if (!memoryLockAcquired) |
| | | return WebResponseContent.Instance.Error("ç³»ç»ç¹å¿ï¼è¯·ç¨åéè¯"); |
| | | |
| | | try |
| | | { |
| | | List<Dt_Task> tasks = new List<Dt_Task>(); |
| | |
| | | WebResponseContent content = await GenerateOutboundTaskDataUpdateAsync(tasks, stockInfos, outboundOrderDetails, outStockLockInfos, locationInfos); |
| | | return content; |
| | | } |
| | | finally |
| | | { |
| | | // éæ¾å
åé |
| | | semaphore.Release(); |
| | | UpdateMaterialLockUsedTime(orderDetail.MaterielCode, orderDetail.BatchNo, orderDetail.SupplyCode); |
| | | } |
| | | } |
| | | catch (Exception ex) |
| | | { |
| | | _unitOfWorkManage.RollbackTran(); |
| | |
| | | List<Dt_Task> tasks = new List<Dt_Task>(); |
| | | |
| | | // è·å订åæç» |
| | | var outboundOrderDetail = await _outboundOrderDetailService.Db.Queryable<Dt_OutboundOrderDetail>() |
| | | var outboundOrderDetail = await _outboundOrderDetailService.Db.Queryable<Dt_OutboundOrderDetail>().With("UPDLOCK, ROWLOCK") |
| | | .FirstAsync(x => x.Id == orderDetailId); |
| | | |
| | | if (outboundOrderDetail == null) |