Scheduler.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. using System;
  2. using System.Collections.Concurrent;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Text;
  6. using System.Threading;
  7. using System.Threading.Tasks;
  8. using GSG.NET.Concurrent;
  9. using GSG.NET.Logging;
  10. using GSG.NET.Utils;
  11. using OHV.Common.Events;
  12. using OHV.Common.Model;
  13. using OHV.Common.Shareds;
  14. using OHV.SqliteDAL;
  15. using Prism.Events;
  16. using VehicleControlSystem.ControlLayer.Serial.BatteryTabos;
  17. using VehicleControlSystem.ControlLayer.Serial.DataModel;
  18. namespace VehicleControlSystem.Managers
  19. {
  20. public class Scheduler : IDisposable
  21. {
  22. static Logger logger = Logger.GetLogger();
  23. private List<Command> commands;
  24. public List<Command> CommandsList
  25. {
  26. get { return commands; }
  27. set { commands = value; }
  28. }
  29. Thread thread = null;
  30. bool isThreadAlive = false;
  31. IEventAggregator eventAggregator;
  32. AutoManager autoManager;
  33. SqliteManager sql = null;
  34. BMUManager bMUManager = null;
  35. public event Action<Command> OnMaualAddedCommand;
  36. public bool IsCreateDryRunCommand { get; set; } = false;
  37. public Scheduler( IEventAggregator ea, AutoManager autoManager, SqliteManager sqlite, BMUManager bMU )
  38. {
  39. this.CommandsList = new List<Command>();
  40. eventAggregator = ea;// CommonServiceLocator.ServiceLocator.Current.GetInstance<IEventAggregator>();
  41. eventAggregator.GetEvent<VCSMessagePubSubEvent>().Subscribe( OnReceivedMessageEvent );
  42. this.autoManager = autoManager;
  43. this.sql = sqlite;
  44. this.bMUManager = bMU;
  45. }
  46. private void OnReceivedMessageEvent( VCSMessageEventArgs obj )
  47. {
  48. if ( obj.Kind == VCSMessageEventArgs.eVCSMessageKind.ReqAddCommand )
  49. {
  50. //var vInof = sql.VehicleInfoDAL.GetInfo();
  51. //if ( vInof.VehicleState == OHV.Common.Shareds.eVehicleState.Move )
  52. // this.sql.CommandDAL.Clean();
  53. this.AddCommand( obj.Command );
  54. }
  55. //var m = new GUIMessageEventArgs() { MessageText = obj.MessageText, MessageKey = obj.MessageKey };
  56. //m.Command = ObjectCopyUtils.DeepClone(obj.Command);
  57. //GUIMessagePublish(m);
  58. }
  59. void GUIMessagePublish( GUIMessageEventArgs args )
  60. {
  61. eventAggregator.GetEvent<GUIMessagePubSubEvent>().Publish( args );
  62. }
  63. public void Init()
  64. {
  65. this.isThreadAlive = true;
  66. this.thread = ThreadUtils.Invoke( this.ExcuteCommand );
  67. }
  68. public bool AddCommand( Command cmd )
  69. {
  70. try
  71. {
  72. this.sql.CommandDAL.Insert( cmd );
  73. logger.I( $"[Command Added] - Target : {cmd.TargetID} / Type : {cmd.Type}" );
  74. }
  75. catch ( Exception ex )
  76. {
  77. logger.E( ex );
  78. return false;
  79. }
  80. return true;
  81. }
  82. bool RemoveCommand( Command cmd )
  83. {
  84. if ( !this.CommandsList.Remove( cmd ) )
  85. {
  86. logger.D( $"Scheduler - [Command] : RemoveCommand fail ID-{cmd.CommandID} / Result-{cmd.Result} " );
  87. return false;
  88. }
  89. logger.D( $"Scheduler - [Command] : RemoveCommand OK ID-{cmd.CommandID} / Result-{cmd.Result} " );
  90. return true;
  91. }
  92. void ExcuteCommand()
  93. {
  94. while ( isThreadAlive )
  95. {
  96. Thread.Sleep( 300 );
  97. try
  98. {
  99. var state = this.autoManager.OperationModeProperty;
  100. switch ( state )
  101. {
  102. case OHV.Common.Shareds.eOperatationMode.ManualMode:
  103. break;
  104. case OHV.Common.Shareds.eOperatationMode.AutoMode:
  105. {
  106. switch ( this.autoManager.AutoModeStateProperty )
  107. {
  108. case OHV.Common.Shareds.eAutoModeState.ErrorStop:
  109. break;
  110. case OHV.Common.Shareds.eAutoModeState.StartRun:
  111. break;
  112. case OHV.Common.Shareds.eAutoModeState.Run:
  113. Command cmd = sql.CommandDAL.GetAll().FirstOrDefault();
  114. if ( cmd == null )
  115. {
  116. this.DryRunTest();
  117. continue;
  118. }
  119. DoWork( cmd );
  120. break;
  121. case OHV.Common.Shareds.eAutoModeState.Stop:
  122. break;
  123. default:
  124. break;
  125. }
  126. }
  127. break;
  128. case OHV.Common.Shareds.eOperatationMode.InitialMode:
  129. break;
  130. default:
  131. break;
  132. }
  133. }
  134. catch ( Exception ex )
  135. {
  136. logger.E( ex );
  137. }
  138. }
  139. logger.D( "[Scheduler] - Command Thread Dispose" );
  140. }
  141. int copyTarget = 0;
  142. bool isSTK = false;
  143. void DryRunTest()
  144. {
  145. if ( !IsCreateDryRunCommand )
  146. return;
  147. if ( this.bMUManager.ReceivedDataDic[eDataKind.SOC].Value <= 25 )
  148. {
  149. //! var chargePoint = "2";
  150. // Command chageCmd = new Command() { TargetID = chargePoint, Type = eCommandType.Charging };
  151. // chageCmd.ByWho = eCommandByWho.LocalSystem;
  152. // this.AddCommand( chageCmd );
  153. this.IsCreateDryRunCommand = false;
  154. return;
  155. }
  156. //string targetPoint = string.Empty;
  157. //if ( isSTK )
  158. //{
  159. // targetPoint = RouteManager.Instance.Equipments.Where( x => x.Name.Equals( "CHARGING_02" ) ).SingleOrDefault().pointID.ToString();
  160. // isSTK = false;
  161. //}
  162. //else
  163. //{
  164. // targetPoint = RouteManager.Instance.Equipments.Where( x => x.Name.Equals( "STOKER_LIFT_IP01" ) ).SingleOrDefault().pointID.ToString();
  165. // isSTK = true;
  166. //}
  167. //Command cmd = new Command() { TargetID = targetPoint, Type = eCommandType.Move };
  168. //cmd.ByWho = eCommandByWho.LocalSystem;
  169. //this.AddCommand( cmd );
  170. //LockUtils.Wait( 2500 );
  171. var ram = NumUtils.Random( 0 , RouteManager.Instance.Equipments.Count - 1 );
  172. if ( ram == this.copyTarget )
  173. return;
  174. else
  175. {
  176. var target = RouteManager.Instance.Equipments[ ram ];
  177. this.copyTarget = ram;
  178. var targetPoint = target.pointID.ToString();
  179. Command cmd = new Command() { TargetID = targetPoint , Type = eCommandType.Move };
  180. cmd.ByWho = eCommandByWho.LocalSystem;
  181. this.AddCommand( cmd );
  182. LockUtils.Wait( 2500 );
  183. //if ( ( target.Name.Equals( "CHARGING_02" ) ) || ( target.Name.Equals( "STOKER_LIFT_IP01" ) ) ||
  184. // ( target.Name.Equals( "STOKER_LIFT_OP01" ) ) || ( target.Name.Equals( "WORK_STATION_IP01" ) ) ||
  185. // ( target.Name.Equals( "WORK_STATION_OP01" ) ) )
  186. //{
  187. // this.copyTarget = ram;
  188. // var targetPoint = target.pointID.ToString();
  189. // Command cmd = new Command() { TargetID = targetPoint, Type = eCommandType.Move };
  190. // cmd.ByWho = eCommandByWho.LocalSystem;
  191. // this.AddCommand( cmd );
  192. // LockUtils.Wait( 2000 );
  193. //}
  194. }
  195. }
  196. private void DoWork( Command cmd )
  197. {
  198. switch ( cmd.State )
  199. {
  200. case OHV.Common.Shareds.eCommandState.Queued:
  201. if ( sql.SubCmdDAL.Count <= 0 ) //
  202. {
  203. switch ( cmd.Type )
  204. {
  205. case OHV.Common.Shareds.eCommandType.Move:
  206. case OHV.Common.Shareds.eCommandType.MoveNLoad:
  207. case OHV.Common.Shareds.eCommandType.MoveNUnload:
  208. case OHV.Common.Shareds.eCommandType.Charging:
  209. case eCommandType.Cancel:
  210. CreateSubCommand( cmd, eSubCommandType.Move );
  211. break;
  212. default:
  213. break;
  214. }
  215. cmd.State = eCommandState.Processing;
  216. this.sql.CommandDAL.Update( cmd );
  217. if ( cmd.ByWho == OHV.Common.Shareds.eCommandByWho.LocalSystem || cmd.ByWho == OHV.Common.Shareds.eCommandByWho.LocalOverWrite )
  218. this.OnMaualAddedCommand?.BeginInvoke( cmd, null, null );
  219. }
  220. break;
  221. case OHV.Common.Shareds.eCommandState.Processing:
  222. {
  223. var subCmd = sql.SubCmdDAL.GetAll().FirstOrDefault();
  224. if ( subCmd == null ) // Move 명령을 수행 완료 했다고 판단.
  225. {
  226. if ( cmd.IsSecondCommanded )
  227. {
  228. cmd.State = eCommandState.Complete;
  229. sql.CommandDAL.Update( cmd );
  230. break;
  231. }
  232. if ( cmd.Type == OHV.Common.Shareds.eCommandType.MoveNLoad )
  233. {
  234. if ( this.CreateSubCommand( cmd, eSubCommandType.Load ) )
  235. cmd.IsSecondCommanded = true;
  236. sql.CommandDAL.Update( cmd );
  237. break;
  238. }
  239. else if ( cmd.Type == OHV.Common.Shareds.eCommandType.MoveNUnload )
  240. {
  241. if ( this.CreateSubCommand( cmd, eSubCommandType.Unload ) )
  242. cmd.IsSecondCommanded = true;
  243. sql.CommandDAL.Update( cmd );
  244. break;
  245. }
  246. else if ( cmd.Type == OHV.Common.Shareds.eCommandType.Charging )
  247. {
  248. if ( this.CreateSubCommand( cmd, eSubCommandType.Charge ) )
  249. cmd.IsSecondCommanded = true;
  250. sql.CommandDAL.Update( cmd );
  251. break;
  252. }
  253. else //Move Command complete
  254. {
  255. cmd.State = eCommandState.Complete;
  256. sql.CommandDAL.Update( cmd );
  257. break;
  258. }
  259. }
  260. else
  261. {
  262. if ( subCmd.CmdID.Equals( cmd.CommandID ) ) //진행중 판단
  263. break;
  264. }
  265. }
  266. break;
  267. case OHV.Common.Shareds.eCommandState.Complete:
  268. sql.CommandDAL.Delete( cmd.CommandID );
  269. logger.I( $"[Command Deleted] - Target : {cmd.TargetID} / Type : {cmd.Type}" );
  270. break;
  271. default:
  272. break;
  273. }
  274. }
  275. private void CommandCancel( Command cmd )
  276. {
  277. }
  278. private bool CreateSubCommand( Command cmd, eSubCommandType type )
  279. {
  280. //명령을 만들기 전에 Vehicle 의 상태를 확인하여 만들자.
  281. switch ( type )
  282. {
  283. case eSubCommandType.Move:
  284. break;
  285. case eSubCommandType.Load:
  286. break;
  287. case eSubCommandType.Unload:
  288. break;
  289. case eSubCommandType.Charge:
  290. break;
  291. case eSubCommandType.Cancel:
  292. break;
  293. default:
  294. break;
  295. }
  296. var sub = new SubCmd
  297. {
  298. CmdType = SubCmd.eCmdType.Auto,
  299. Type = type,
  300. TargetID = cmd.TargetID,
  301. CmdID = cmd.CommandID,
  302. TargetList = new List<string>( cmd.TargetList ),
  303. };
  304. sql.SubCmdDAL.Insert( sub );
  305. logger.I( $"{sub.ID} Sub Command Create - Type={sub.Type}, TargetID={sub.TargetID}, CommandID={sub.CmdID}" );
  306. return true;
  307. }
  308. #region IDisposable Support
  309. private bool disposedValue = false; // 중복 호출을 검색하려면
  310. protected virtual void Dispose( bool disposing )
  311. {
  312. if ( !disposedValue )
  313. {
  314. if ( disposing )
  315. {
  316. // TODO: 관리되는 상태(관리되는 개체)를 삭제합니다.
  317. this.isThreadAlive = false;
  318. if ( !this.thread.Join( 3000 ) )
  319. ThreadUtils.Kill( this.thread );
  320. }
  321. // TODO: 관리되지 않는 리소스(관리되지 않는 개체)를 해제하고 아래의 종료자를 재정의합니다.
  322. // TODO: 큰 필드를 null로 설정합니다.
  323. disposedValue = true;
  324. }
  325. }
  326. // TODO: 위의 Dispose(bool disposing)에 관리되지 않는 리소스를 해제하는 코드가 포함되어 있는 경우에만 종료자를 재정의합니다.
  327. // ~Scheduler()
  328. // {
  329. // // 이 코드를 변경하지 마세요. 위의 Dispose(bool disposing)에 정리 코드를 입력하세요.
  330. // Dispose(false);
  331. // }
  332. // 삭제 가능한 패턴을 올바르게 구현하기 위해 추가된 코드입니다.
  333. public void Dispose()
  334. {
  335. // 이 코드를 변경하지 마세요. 위의 Dispose(bool disposing)에 정리 코드를 입력하세요.
  336. Dispose( true );
  337. // TODO: 위의 종료자가 재정의된 경우 다음 코드 줄의 주석 처리를 제거합니다.
  338. // GC.SuppressFinalize(this);
  339. }
  340. #endregion
  341. }
  342. }