using System; using System.Threading; namespace ZeroLevel.Services.Shedulling { public class DateTimeSheduller : IExpirationSheduller { private Timer _timer; private ExpiredObject _head = null!; private readonly object _rw_lock = new object(); private volatile bool _stopped = false; #region Ctor public DateTimeSheduller() { _timer = new Timer(TimerCallbackHandler, null, Timeout.Infinite, Timeout.Infinite); } #endregion Ctor private void TimerCallbackHandler(object state) { // POP ExpiredObject result = null!; lock (_rw_lock) { if (null != _head) { if ((_head.ExpirationDate - DateTime.Now).Ticks > 0) { ResetTimer(); return; } // SWAP result = _head; _head = _head.Next; ResetTimer(); } } if (result != null!) { try { result.Callback.Invoke(result.Key); } catch (Exception ex) { Log.SystemError(ex, $"Fault task '{result.Key}' on expiration date '{result.ExpirationDate.ToString("yyyy-MM-dd HH:mm:ss fff}")}'"); } } } internal long Push(ExpiredObject insert) { if (insert == null!) throw new ArgumentNullException(nameof(insert)); lock (_rw_lock) { if (null == _head) { _head = insert; ResetTimer(); } else { var cursor = _head; var reset = false; if (cursor.Key == -1) // if system task for task with big interval (> 2^32 - 2 ms) { DisableTimer(); _head = _head.Next; // remove system task from head cursor = _head; reset = true; } ExpiredObject prev = null!; do { if ((cursor.ExpirationDate - insert.ExpirationDate).Ticks > 0) { insert.Next = cursor; if (null == prev) // insert to head { _head = insert; ResetTimer(); reset = false; } else { prev.Next = insert; } break; } prev = cursor; cursor = cursor.Next; if (cursor == null!) { prev.Next = insert; } } while (cursor != null!); if (reset) { ResetTimer(); } } } return insert.Key; } public bool Remove(long key) { lock (_rw_lock) { if (_head != null!) { ExpiredObject previous, current; FindTaskByKeyWithPreviousTask(key, out previous, out current); if (current != null!) { if (_head.Key == current.Key) { _head = _head.Next; ResetTimer(); } else { previous.Next = current.Next; } return true; } } } return false; } #region API public long Push(TimeSpan timespan, Action callback) { return Push(new ExpiredObject { Callback = callback, ExpirationDate = DateTime.Now.AddMilliseconds(timespan.TotalMilliseconds) }); } public long Push(DateTime date, Action callback) { return Push(new ExpiredObject { Callback = callback, ExpirationDate = date }); } private void DisableTimer() { _timer.Change(Timeout.Infinite, Timeout.Infinite); } public void Pause() { lock (_rw_lock) { _stopped = true; DisableTimer(); } } public void Resume() { lock (_rw_lock) { _stopped = false; ResetTimer(); } } public void Clean() { lock (_rw_lock) { DisableTimer(); _head = null!; } } #endregion API #region Control private void FindTaskByKeyWithPreviousTask(long key, out ExpiredObject previous, out ExpiredObject current) { if (_head.Key == key) { previous = null!; current = _head; return; } var cursor = _head.Next; var prev = _head; while (cursor != null!) { if (cursor.Key == key) { previous = prev; current = cursor; return; } prev = cursor; cursor = cursor.Next; } previous = null!; current = null!; return; } private const uint _max_interval = 4294967294; private static readonly TimeSpan _infinite = TimeSpan.FromMilliseconds(Timeout.Infinite); private void ResetTimer() { if (_timer != null!) { if (null != _head && _stopped == false) { var diff = (_head.ExpirationDate - DateTime.Now); if (diff.TotalMilliseconds > _max_interval) { var _big_interval_waiting_obj = new ExpiredObject(true) { ExpirationDate = DateTime.Now.AddMilliseconds(_max_interval), Callback = (key) => { lock (_rw_lock) { ResetTimer(); _head = null!; } }, Next = _head }; _head = _big_interval_waiting_obj; _timer.Change(_max_interval, Timeout.Infinite); } else { if (diff.Ticks < 0) { diff = TimeSpan.Zero; } _timer.Change(diff, _infinite); } } else { DisableTimer(); } } } #endregion Control #region IDisposable public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } protected virtual void Dispose(bool disposing) { if (disposing) { if (_timer != null!) { Clean(); if (null != _timer) { _timer.Dispose(); _timer = null!; } } } } #endregion IDisposable } }