using System; using System.Collections.Concurrent; namespace ZeroLevel.Services.Shedulling { /// /// Simple scheduler for periodic and one-time scheduled tasks /// internal class ShedullerImpl : ISheduller { private readonly DateTimeSheduller _sheduller; public ShedullerImpl() { _sheduller = new DateTimeSheduller(); } #region One time events public long RemindAfter(TimeSpan timespan, Action callback) { return _sheduller.Push(timespan, callback); } public long RemindAt(DateTime date, Action callback) { return _sheduller.Push(date, callback); } #endregion One time events #region Repitable behaviour private readonly ConcurrentDictionary _repitableActions = new ConcurrentDictionary(); /// /// Performs an action once a period, while the period is recalculated according to the transferred function at each re-creation of the task. /// /// Function to calculate the next period /// Task ID public long RemindEveryNonlinearPeriod(Func nextEventPeriodCalcFunction, Action callback, bool breakWherError = false) { return RemindEveryNonlinearPeriod(nextEventPeriodCalcFunction, nextEventPeriodCalcFunction, callback, breakWherError); } /// /// Performs an action once a period, while the period is recalculated according to the transferred function at each re-creation of the task. /// /// The function to calculate the period to the first execution /// The function for calculating the period until subsequent performances /// Task ID public long RemindEveryNonlinearPeriod(Func firstEventPeriodCalcFunction, Func nextEventPeriodCalcFunction, Action callback, bool breakWherError = false) { var obj = new ExpiredObject { ExpirationDate = DateTime.Now.AddMilliseconds(firstEventPeriodCalcFunction().TotalMilliseconds) }; _repitableActions.TryAdd(obj.Key, obj); obj.Callback = (key) => { try { callback(key); } catch (Exception ex) { Log.SystemError(ex, $"[Sheduller] Fault call task '{key}' handler"); if (breakWherError) return; } ExpiredObject repObj; if (_repitableActions.TryGetValue(obj.Key, out repObj)) { _sheduller.Push(repObj.Reset(DateTime.Now.AddMilliseconds(nextEventPeriodCalcFunction().TotalMilliseconds))); } }; return _sheduller.Push(obj); } /// /// Performs an action once per period, while the date is recalculated from the function transferred each time the task is recreated. /// /// The function to calculate the next date /// Action /// Task ID public long RemindEveryNonlinearDate(Func nextEventDateCalcFunction, Action callback, bool breakWherError = false) { return RemindEveryNonlinearDate(nextEventDateCalcFunction, nextEventDateCalcFunction, callback, breakWherError); } public long RemindEveryNonlinearDate(DateTime firstTime, Func nextEventDateCalcFunction, Action callback, bool breakWherError = false) { var obj = new ExpiredObject { ExpirationDate = firstTime }; _repitableActions.TryAdd(obj.Key, obj); obj.Callback = (key) => { try { callback(key); } catch (Exception ex) { Log.SystemError(ex, $"[Sheduller] Fault call task '{key}' handler"); if (breakWherError) return; } ExpiredObject repObj; if (_repitableActions.TryGetValue(obj.Key, out repObj)) { var nextDate = nextEventDateCalcFunction(obj.ExpirationDate); if (DateTime.Compare(nextDate, DateTime.MinValue) == 0) { Remove(repObj.Key); } else { _sheduller.Push(repObj.Reset(nextDate)); } } }; return _sheduller.Push(obj); } /// /// Performs an action once per period, while the date is recalculated from the function transferred each time the task is recreated. /// /// The function to calculate the first run date /// The function to calculate the next date /// Action /// Task ID public long RemindEveryNonlinearDate(Func firstEventDateCalcFunction, Func nextEventDateCalcFunction, Action callback, bool breakWherError = false) { var obj = new ExpiredObject { ExpirationDate = firstEventDateCalcFunction(DateTime.Now) }; _repitableActions.TryAdd(obj.Key, obj); obj.Callback = (key) => { try { callback(key); } catch (Exception ex) { Log.SystemError(ex, $"[Sheduller] Fault call task '{key}' handler"); if (breakWherError) return; } ExpiredObject repObj; if (_repitableActions.TryGetValue(obj.Key, out repObj)) { var nextDate = nextEventDateCalcFunction(obj.ExpirationDate); if (DateTime.Compare(nextDate, DateTime.MinValue) == 0) { Remove(repObj.Key); } else { _sheduller.Push(repObj.Reset(nextDate)); } } }; return _sheduller.Push(obj); } /// /// Performs an action once in a specified period /// /// Period /// Action /// Task ID public long RemindEvery(TimeSpan timespan, Action callback, bool breakWherError = false) { return RemindEvery(timespan, timespan, callback, breakWherError); } /// /// Performs an action once in a specified period /// /// Period to first run /// Period /// Action /// Task ID public long RemindEvery(TimeSpan first, TimeSpan next, Action callback, bool breakWherError = false) { var obj = new ExpiredObject { ExpirationDate = DateTime.Now.AddMilliseconds(first.TotalMilliseconds) }; _repitableActions.TryAdd(obj.Key, obj); obj.Callback = (key) => { try { callback(key); } catch (Exception ex) { Log.SystemError(ex, $"[Sheduller] Fault call task '{key}' handler"); if (breakWherError) return; } ExpiredObject repObj; if (_repitableActions.TryGetValue(obj.Key, out repObj)) { _sheduller.Push(repObj.Reset(DateTime.Now.AddMilliseconds(next.TotalMilliseconds))); } }; return _sheduller.Push(obj); } public long RemindWhile(TimeSpan period, Func callback, Action continueWith = null, bool breakWherError = false) { var obj = new ExpiredObject { ExpirationDate = DateTime.Now.AddMilliseconds(period.TotalMilliseconds) }; _repitableActions.TryAdd(obj.Key, obj); obj.Callback = (key) => { bool success = false; try { success = callback(key); } catch (Exception ex) { Log.SystemError(ex, $"[Sheduller] Fault call task '{key}' handler"); if (breakWherError) return; } if (success) { Remove(obj.Key); if (continueWith != null) continueWith(); } else { ExpiredObject repObj; if (_repitableActions.TryGetValue(obj.Key, out repObj)) { _sheduller.Push(repObj.Reset(DateTime.Now.AddMilliseconds(period.TotalMilliseconds))); } } }; return _sheduller.Push(obj); } #endregion Repitable behaviour #region Sheduller control public void Pause() { _sheduller.Pause(); } public void Resume() { _sheduller.Resume(); } public void Clean() { _sheduller.Clean(); } public bool Remove(long key) { var success = _sheduller.Remove(key); if (_repitableActions.ContainsKey(key)) { ExpiredObject rem; return _repitableActions.TryRemove(key, out rem); } return success; } #endregion Sheduller control #region IDisposable public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } protected virtual void Dispose(bool disposing) { if (disposing) { _sheduller.Dispose(); } } #endregion IDisposable } }