CoreProfileHandler.cs 53 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967
  1. using EVCB_OCPP.Domain;
  2. using EVCB_OCPP.Domain.Models.Database;
  3. using EVCB_OCPP.Packet.Features;
  4. using EVCB_OCPP.Packet.Messages;
  5. using EVCB_OCPP.Packet.Messages.Core;
  6. using EVCB_OCPP.Packet.Messages.SubTypes;
  7. using EVCB_OCPP.WSServer.Dto;
  8. using EVCB_OCPP.WSServer.Service;
  9. using Newtonsoft.Json;
  10. using NLog;
  11. using OCPPPackage.Profiles;
  12. using OCPPServer.Protocol;
  13. using System;
  14. using System.Collections.Generic;
  15. using System.Data.Entity;
  16. using System.Data.SqlClient;
  17. using System.Diagnostics;
  18. using System.Linq;
  19. using System.Threading.Tasks;
  20. namespace EVCB_OCPP.WSServer.Message
  21. {
  22. internal partial class ProfileHandler
  23. {
  24. static private ILogger logger = NLog.LogManager.GetCurrentClassLogger();
  25. async internal Task<MessageResult> ExecuteCoreRequest(Actions action, ClientData session, IRequest request)
  26. {
  27. Stopwatch watch = new Stopwatch();
  28. if (action == Actions.Heartbeat)
  29. {
  30. watch.Start();
  31. }
  32. MessageResult result = new MessageResult() { Success = false };
  33. try
  34. {
  35. switch (action)
  36. {
  37. case Actions.DataTransfer:
  38. {
  39. var confirm = new DataTransferConfirmation() { status = DataTransferStatus.Rejected };
  40. result.Message = confirm;
  41. result.Success = true;
  42. }
  43. break;
  44. case Actions.BootNotification:
  45. {
  46. BootNotificationRequest _request = request as BootNotificationRequest;
  47. int heartbeat_interval = GlobalConfig.GetHEARTBEAT_INTERVAL();
  48. using (var db = new MainDBContext())
  49. {
  50. var _machine = db.Machine.Where(x => x.ChargeBoxId == session.ChargeBoxId).FirstOrDefault();
  51. _machine.ChargeBoxSerialNumber = string.IsNullOrEmpty(_request.chargeBoxSerialNumber) ? string.Empty : _request.chargeBoxSerialNumber;
  52. _machine.ChargePointSerialNumber = string.IsNullOrEmpty(_request.chargePointSerialNumber) ? string.Empty : _request.chargePointSerialNumber;
  53. _machine.ChargePointModel = string.IsNullOrEmpty(_request.chargePointModel) ? string.Empty : _request.chargePointModel;
  54. _machine.ChargePointVendor = string.IsNullOrEmpty(_request.chargePointVendor) ? string.Empty : _request.chargePointVendor;
  55. _machine.FW_CurrentVersion = string.IsNullOrEmpty(_request.firmwareVersion) ? string.Empty : _request.firmwareVersion;
  56. _machine.Iccid = string.IsNullOrEmpty(_request.iccid) ? string.Empty : _request.iccid;
  57. _machine.Imsi = string.IsNullOrEmpty(_request.imsi) ? string.Empty : _request.imsi;
  58. _machine.MeterSerialNumber = string.IsNullOrEmpty(_request.meterSerialNumber) ? string.Empty : _request.meterSerialNumber;
  59. _machine.MeterType = string.IsNullOrEmpty(_request.meterType) ? string.Empty : _request.meterType;
  60. db.SaveChanges();
  61. logger.Error(string.Format("{0} ChargeBoxSerialNumber:{1} ", session.ChargeBoxId, _request.chargePointSerialNumber));
  62. var configVaule = db.MachineConfiguration.Where(x => x.ChargeBoxId == session.ChargeBoxId && x.ConfigureName == StandardConfiguration.HeartbeatInterval)
  63. .Select(x => x.ConfigureSetting).FirstOrDefault();
  64. if (configVaule != null)
  65. {
  66. int.TryParse(configVaule, out heartbeat_interval);
  67. }
  68. }
  69. var confirm = new BootNotificationConfirmation() { currentTime = DateTime.Now, interval = session.IsPending ? heartbeat_interval : 5, status = session.IsPending ? Packet.Messages.SubTypes.RegistrationStatus.Accepted : RegistrationStatus.Pending };
  70. session.IsPending = !session.IsPending;
  71. result.Message = confirm;
  72. result.Success = true;
  73. }
  74. break;
  75. case Actions.StatusNotification:
  76. {
  77. //只保留最新上報狀況
  78. StatusNotificationRequest _request = request as StatusNotificationRequest;
  79. int preStatus = 0;
  80. using (var db = new MainDBContext())
  81. {
  82. var _oldStatus = db.ConnectorStatus.Where(x => x.ChargeBoxId == session.ChargeBoxId
  83. && x.ConnectorId == _request.connectorId).AsNoTracking().FirstOrDefault();
  84. if (_oldStatus != null && (_request.status != (ChargePointStatus)_oldStatus.Status || _request.status == ChargePointStatus.Faulted))
  85. {
  86. preStatus = _oldStatus.Status;
  87. db.Configuration.AutoDetectChangesEnabled = false;
  88. db.Configuration.ValidateOnSaveEnabled = false;
  89. db.ConnectorStatus.Attach(_oldStatus);
  90. _oldStatus.CreatedOn = _request.timestamp.HasValue ? _request.timestamp.Value : DateTime.Now;
  91. _oldStatus.Status = (int)_request.status;
  92. _oldStatus.ChargePointErrorCodeId = (int)_request.errorCode;
  93. _oldStatus.ErrorInfo = string.IsNullOrEmpty(_request.info) ? string.Empty : _request.info;
  94. _oldStatus.VendorId = string.IsNullOrEmpty(_request.vendorId) ? string.Empty : _request.vendorId;
  95. _oldStatus.VendorErrorCode = string.IsNullOrEmpty(_request.vendorErrorCode) ? string.Empty : _request.vendorErrorCode;
  96. db.Entry(_oldStatus).Property(x => x.CreatedOn).IsModified = true;
  97. db.Entry(_oldStatus).Property(x => x.Status).IsModified = true;
  98. db.Entry(_oldStatus).Property(x => x.ChargePointErrorCodeId).IsModified = true;
  99. db.Entry(_oldStatus).Property(x => x.ErrorInfo).IsModified = true;
  100. db.Entry(_oldStatus).Property(x => x.VendorId).IsModified = true;
  101. db.Entry(_oldStatus).Property(x => x.VendorErrorCode).IsModified = true;
  102. }
  103. if (_oldStatus == null)
  104. {
  105. var _currentStatus = new Domain.Models.Database.ConnectorStatus()
  106. {
  107. ChargeBoxId = session.ChargeBoxId,
  108. ConnectorId = (byte)_request.connectorId,
  109. CreatedOn = _request.timestamp.HasValue ? _request.timestamp.Value : DateTime.Now,
  110. Status = (int)_request.status,
  111. ChargePointErrorCodeId = (int)_request.errorCode,
  112. ErrorInfo = string.IsNullOrEmpty(_request.info) ? string.Empty : _request.info,
  113. VendorId = string.IsNullOrEmpty(_request.vendorId) ? string.Empty : _request.vendorId,
  114. VendorErrorCode = string.IsNullOrEmpty(_request.vendorErrorCode) ? string.Empty : _request.vendorErrorCode,
  115. Id = Guid.NewGuid().ToString()
  116. };
  117. db.ConnectorStatus.Add(_currentStatus);
  118. }
  119. if (_request.status == Packet.Messages.SubTypes.ChargePointStatus.Faulted)
  120. {
  121. db.MachineError.Add(new MachineError()
  122. {
  123. ConnectorId = (byte)_request.connectorId,
  124. CreatedOn = _request.timestamp.HasValue ? _request.timestamp.Value : DateTime.Now,
  125. Status = (int)_request.status,
  126. ChargeBoxId = session.ChargeBoxId,
  127. ErrorCodeId = (int)_request.errorCode,
  128. ErrorInfo = string.IsNullOrEmpty(_request.info) ? string.Empty : _request.info,
  129. PreStatus = _oldStatus == null ? -1 : preStatus,
  130. VendorErrorCode = string.IsNullOrEmpty(_request.vendorErrorCode) ? string.Empty : _request.vendorErrorCode,
  131. VendorId = string.IsNullOrEmpty(_request.vendorId) ? string.Empty : _request.vendorId
  132. });
  133. }
  134. db.SaveChanges();
  135. }
  136. if (_request.status == Packet.Messages.SubTypes.ChargePointStatus.Faulted)
  137. {
  138. var businessService = BusinessServiceFactory.CreateBusinessService(session.CustomerId.ToString());
  139. var notification = businessService.NotifyFaultStatus(new ErrorDetails()
  140. {
  141. ChargeBoxId = session.ChargeBoxId,
  142. ConnectorId = _request.connectorId,
  143. ErrorCode = _request.errorCode,
  144. Info = string.IsNullOrEmpty(_request.info) ? string.Empty : _request.info,
  145. OCcuredOn = _request.timestamp ?? DateTime.Now,
  146. VendorErrorCode = string.IsNullOrEmpty(_request.vendorErrorCode) ? string.Empty : _request.vendorErrorCode,
  147. });
  148. }
  149. var confirm = new StatusNotificationConfirmation() { };
  150. result.Message = confirm;
  151. result.Success = true;
  152. }
  153. break;
  154. case Actions.Heartbeat:
  155. {
  156. //Task updateBeatTask = Task.Run(async () =>
  157. // {
  158. // using (var db = new MainDBContext())
  159. // {
  160. // var machine = new Machine() { Id = session.MachineId };
  161. // if (machine != null)
  162. // {
  163. // db.Configuration.AutoDetectChangesEnabled = false;
  164. // db.Configuration.ValidateOnSaveEnabled = false;
  165. // db.Machine.Attach(machine);
  166. // machine.HeartbeatUpdatedOn = DateTime.Now;
  167. // machine.ConnectionType = session.UriScheme.Equals("wss") ? 2 : 1;
  168. // db.Entry(machine).Property(x => x.HeartbeatUpdatedOn).IsModified = true;
  169. // db.Entry(machine).Property(x => x.ConnectionType).IsModified = true;
  170. // await db.SaveChangesAsync();
  171. // }
  172. // }
  173. // });
  174. var confirm = new HeartbeatConfirmation() { currentTime = DateTime.Now };
  175. result.Message = confirm;
  176. result.Success = true;
  177. }
  178. break;
  179. case Actions.MeterValues:
  180. {
  181. MeterValuesRequest _request = request as MeterValuesRequest;
  182. string feeText = session.FeeDescription.Split(';')[1].Replace(" Current Rate: $", "").Split(' ')[0];
  183. decimal fee = decimal.Parse(feeText);
  184. decimal energy_kwh = 0;
  185. if (_request.meterValue.Count > 0)
  186. {
  187. using (var db = new MeterValueDBContext())
  188. {
  189. foreach (var item in _request.meterValue)
  190. {
  191. foreach (var sampleVaule in item.sampledValue)
  192. {
  193. decimal value = Convert.ToDecimal(sampleVaule.value);
  194. if (sampleVaule.context == ReadingContext.Sample_Periodic && sampleVaule.measurand == Measurand.Energy_Active_Import_Interval)
  195. {
  196. energy_kwh = sampleVaule.unit == UnitOfMeasure.Wh ? Decimal.Divide(value, 1000) : value;
  197. }
  198. string sp = "[dbo].[uspInsertMeterValueRecord] @ChargeBoxId," +
  199. "@ConnectorId,@Value,@CreatedOn,@ContextId,@FormatId,@MeasurandId,@PhaseId,@LocationId,@UnitId,@TransactionId";
  200. List<SqlParameter> parameter = new List<SqlParameter>
  201. {
  202. new SqlParameter("ChargeBoxId",session.ChargeBoxId),
  203. new SqlParameter("ConnectorId", (byte)_request.connectorId),
  204. new SqlParameter("Value",value),
  205. new SqlParameter("CreatedOn",item.timestamp),
  206. new SqlParameter("ContextId",sampleVaule.context.HasValue ? (int)sampleVaule.context : 0),
  207. new SqlParameter("FormatId",sampleVaule.format.HasValue ? (int)sampleVaule.format : 0),
  208. new SqlParameter("MeasurandId",sampleVaule.measurand.HasValue ? (int)sampleVaule.measurand : 0),
  209. new SqlParameter("PhaseId",sampleVaule.phase.HasValue ? (int)sampleVaule.phase : 0),
  210. new SqlParameter("LocationId",sampleVaule.location.HasValue ? (int)sampleVaule.location : 0),
  211. new SqlParameter("UnitId",sampleVaule.unit.HasValue ? (int)sampleVaule.unit : 0),
  212. new SqlParameter("TransactionId",_request.transactionId.HasValue?_request.transactionId:-1),
  213. };
  214. db.Database.ExecuteSqlCommand(sp, parameter.ToArray());
  215. }
  216. }
  217. }
  218. }
  219. if (energy_kwh > 0)
  220. {
  221. try
  222. {
  223. // ************等Alston 可以使用在打開這個功能 ~ 電樁畫面上目前不會顯示
  224. //using (var db = new MainDBContext())
  225. //{
  226. // db.ServerMessage.Add(new ServerMessage()
  227. // {
  228. // ChargeBoxId = session.ChargeBoxId,
  229. // CreatedBy = "Server",
  230. // CreatedOn = DateTime.Now,
  231. // OutAction = Actions.DataTransfer.ToString(),
  232. // OutRequest = JsonConvert.SerializeObject(
  233. // new DataTransferRequest()
  234. // {
  235. // messageId = "RunningCost",
  236. // vendorId = "Phihong Technology",
  237. // data = JsonConvert.SerializeObject(new
  238. // {
  239. // txId = _request.transactionId,
  240. // description = string.Format("Connection Fee: $0.00 {0}; Session Fee: ${1} {0}; Occupancy Fee: " +
  241. // "$0.00 {0}; Total Cost: ${1} {0}; Account Balance: $155.05 {0}", session.Currency,
  242. // Decimal.Multiply(energy_kwh, fee))
  243. // })
  244. // },
  245. // new JsonSerializerSettings() { NullValueHandling = NullValueHandling.Ignore, Formatting = Formatting.None }),
  246. // SerialNo = Guid.NewGuid().ToString(),
  247. // InMessage = string.Empty
  248. // }); ;
  249. // db.SaveChanges();
  250. //}
  251. }
  252. catch (Exception ex)
  253. {
  254. Console.WriteLine(string.Format("{0} :{1}", session.ChargeBoxId + " RunningCost", ex.Message));
  255. }
  256. }
  257. var confirm = new MeterValuesConfirmation() { };
  258. result.Message = confirm;
  259. result.Success = true;
  260. }
  261. break;
  262. case Actions.StartTransaction:
  263. {
  264. StartTransactionRequest _request = request as StartTransactionRequest;
  265. int _transactionId = -1;
  266. var businessService = BusinessServiceFactory.CreateBusinessService(session.CustomerId.ToString());
  267. var _idTagInfo = _request.idTag == "Backend" ? new IdTagInfo() { expiryDate = DateTime.UtcNow.AddDays(1), status = AuthorizationStatus.Accepted } : await businessService.Authorize(session.ChargeBoxId, _request.idTag);
  268. //特例****飛宏客戶旗下的電樁,若遇到Portal沒回應的狀況 ~允許充電
  269. if (session.CustomerId.ToString().ToUpper() == "8456AED9-6DD9-4BF3-A94C-9F5DCB9506F7" && _idTagInfo.status == AuthorizationStatus.ConcurrentTx)
  270. {
  271. _idTagInfo = new IdTagInfo() { expiryDate = DateTime.UtcNow.AddDays(1), status = AuthorizationStatus.Accepted };
  272. }
  273. using (var db = new MainDBContext())
  274. {
  275. var _CustomerId = db.Machine.Where(x => x.ChargeBoxId == session.ChargeBoxId).Include(x => x.Customer).
  276. Select(x => x.CustomerId).FirstOrDefault();
  277. var _existedTx = db.TransactionRecord.Where(x => x.CustomerId == _CustomerId && x.ChargeBoxId == session.ChargeBoxId
  278. && x.ConnectorId == _request.connectorId && x.StartTime == _request.timestamp).Select(C => new { C.Id }).AsNoTracking().FirstOrDefault();
  279. TransactionRecord _newTransaction = new TransactionRecord();
  280. if (_existedTx == null)
  281. {
  282. _newTransaction = new TransactionRecord()
  283. {
  284. ChargeBoxId = session.ChargeBoxId,
  285. ConnectorId = (byte)_request.connectorId,
  286. CreatedOn = DateTime.Now,
  287. StartIdTag = _request.idTag,
  288. MeterStart = _request.meterStart,
  289. CustomerId = _CustomerId,
  290. StartTime = _request.timestamp,
  291. ReservationId = _request.reservationId.HasValue ? _request.reservationId.Value : 0,
  292. Fee = session.FeeDescription
  293. };
  294. db.TransactionRecord.Add(_newTransaction);
  295. db.SaveChanges();
  296. _transactionId = _newTransaction.Id;
  297. logger.Info("***************************************************** ");
  298. logger.Info(string.Format("{0} :TransactionId {1} ", session.ChargeBoxId, _newTransaction.Id));
  299. logger.Info("***************************************************** ");
  300. }
  301. else
  302. {
  303. _transactionId = _existedTx.Id;
  304. logger.Error("Duplication ***************************************************** " + _existedTx.Id);
  305. }
  306. }
  307. var confirm = new StartTransactionConfirmation()
  308. {
  309. idTagInfo = _idTagInfo,
  310. transactionId = _transactionId
  311. };
  312. result.Message = confirm;
  313. result.Success = true;
  314. }
  315. break;
  316. case Actions.StopTransaction:
  317. {
  318. StopTransactionRequest _request = request as StopTransactionRequest;
  319. List<TransactionDataRecord> _TransactionDatas = new List<TransactionDataRecord>();
  320. int _ConnectorId = 0;
  321. var businessService = BusinessServiceFactory.CreateBusinessService(session.CustomerId.ToString());
  322. //特例****飛宏客戶旗下的電樁,若遇到Portal沒回應的狀況 ~允許充電
  323. var _idTagInfo = string.IsNullOrEmpty(_request.idTag) ? null : (_request.idTag == "Backend" ? new IdTagInfo() { expiryDate = DateTime.UtcNow.AddDays(1), status = AuthorizationStatus.Accepted } : await businessService.Authorize(session.ChargeBoxId, _request.idTag));
  324. if (session.CustomerId.ToString().ToUpper() == "8456AED9-6DD9-4BF3-A94C-9F5DCB9506F7" && _idTagInfo != null && _idTagInfo.status == AuthorizationStatus.ConcurrentTx)
  325. {
  326. _idTagInfo = new IdTagInfo() { expiryDate = DateTime.UtcNow.AddDays(1), status = AuthorizationStatus.Accepted };
  327. }
  328. using (var db = new MainDBContext())
  329. {
  330. var transaction = db.TransactionRecord.Where(x => x.Id == _request.transactionId
  331. && x.ChargeBoxId == session.ChargeBoxId).FirstOrDefault();
  332. if (transaction != null)
  333. {
  334. decimal energy_kwh = decimal.Subtract(_request.meterStop, transaction.MeterStart);
  335. energy_kwh = decimal.Divide(energy_kwh, 1000);
  336. string feeText = transaction.Fee.Split(';')[1].Replace(" Current Rate: $", "").Split(' ')[0];
  337. decimal fee = decimal.Parse(feeText);
  338. decimal cost = Decimal.Multiply(energy_kwh, (decimal)fee);
  339. if (session.Currency == "USD" || session.Currency == "EUR")
  340. {
  341. //0.4867
  342. if ((int)(cost * 100) < cost * 100)
  343. {
  344. cost = Decimal.Add(cost, (decimal)0.01);//0.4967
  345. }
  346. cost = Decimal.Parse(cost.ToString("0.00"));
  347. }
  348. else
  349. {
  350. if ((int)(cost) < cost )
  351. {
  352. cost = Decimal.Add(cost, (decimal)1);
  353. }
  354. cost = Decimal.Parse(cost.ToString("0"));
  355. }
  356. string receipt = string.Format("Connection Fee: $0.00 {0}; Session Fee: ${1} {0}; Occupancy Fee: " +
  357. "$0.00 {0}; Total Cost: ${1} {0}; Account Balance: $4555 {0}", session.Currency,
  358. cost);
  359. _ConnectorId = transaction.ConnectorId;
  360. transaction.MeterStop = _request.meterStop;
  361. transaction.StopTime = _request.timestamp;
  362. transaction.StopReasonId = _request.reason.HasValue ? (int)_request.reason.Value : 0;
  363. transaction.StopIdTag = _request.idTag;
  364. transaction.Receipt = receipt;
  365. transaction.Cost = cost;
  366. if (_TransactionDatas.Count > 0)
  367. {
  368. _TransactionDatas.ForEach(x => x.ConnectorId = (byte)transaction.ConnectorId);
  369. }
  370. await db.SaveChangesAsync();
  371. var confirm = new StopTransactionConfirmation()
  372. {
  373. idTagInfo = _idTagInfo
  374. };
  375. if (energy_kwh > 0)
  376. {
  377. db.ServerMessage.Add(new ServerMessage()
  378. {
  379. ChargeBoxId = session.ChargeBoxId,
  380. CreatedBy = "Server",
  381. CreatedOn = DateTime.Now,
  382. OutAction = Actions.DataTransfer.ToString(),
  383. OutRequest = JsonConvert.SerializeObject(
  384. new DataTransferRequest()
  385. {
  386. messageId = "FinalCost",
  387. vendorId = "Phihong Technology",
  388. data = JsonConvert.SerializeObject(new
  389. {
  390. txId = _request.transactionId,
  391. description = string.Format("Connection Fee: $0.00 {0}; Session Fee: ${1} {0}; Occupancy Fee: " +
  392. "$0.00 {0}; Total Cost: ${1} {0}; Account Balance: $4555 {0}", session.Currency,
  393. energy_kwh, Decimal.Multiply(energy_kwh, (decimal)fee))
  394. })
  395. },
  396. new JsonSerializerSettings() { NullValueHandling = NullValueHandling.Ignore, Formatting = Formatting.None }),
  397. SerialNo = Guid.NewGuid().ToString(),
  398. InMessage = string.Empty
  399. }); ;
  400. db.SaveChanges();
  401. }
  402. result.Message = confirm;
  403. result.Success = true;
  404. }
  405. else
  406. {
  407. result.Exception = new Exception("Can't find transactionId " + _request.transactionId);
  408. }
  409. }
  410. if (_request.transactionData != null)
  411. {
  412. if (_request.transactionData.Count > 0)
  413. {
  414. using (var _meterDb = new MeterValueDBContext())
  415. {
  416. foreach (var item in _request.transactionData)
  417. {
  418. foreach (var sampleVaule in item.sampledValue)
  419. {
  420. decimal value = Convert.ToDecimal(sampleVaule.value);
  421. string sp = "[dbo].[uspInsertMeterValueRecord] @ChargeBoxId," +
  422. "@ConnectorId,@Value,@CreatedOn,@ContextId,@FormatId,@MeasurandId,@PhaseId,@LocationId,@UnitId,@TransactionId";
  423. List<SqlParameter> parameter = new List<SqlParameter>
  424. {
  425. new SqlParameter("ChargeBoxId",session.ChargeBoxId),
  426. new SqlParameter("ConnectorId", (byte)_ConnectorId),
  427. new SqlParameter("Value",value),
  428. new SqlParameter("CreatedOn",item.timestamp),
  429. new SqlParameter("ContextId",sampleVaule.context.HasValue ? (int)sampleVaule.context : 0),
  430. new SqlParameter("FormatId",sampleVaule.format.HasValue ? (int)sampleVaule.format : 0),
  431. new SqlParameter("MeasurandId",sampleVaule.measurand.HasValue ? (int)sampleVaule.measurand : 0),
  432. new SqlParameter("PhaseId",sampleVaule.phase.HasValue ? (int)sampleVaule.phase : 0),
  433. new SqlParameter("LocationId",sampleVaule.location.HasValue ? (int)sampleVaule.location : 0),
  434. new SqlParameter("UnitId",sampleVaule.unit.HasValue ? (int)sampleVaule.unit : 0),
  435. new SqlParameter("TransactionId",_request.transactionId),
  436. };
  437. _meterDb.Database.ExecuteSqlCommand(sp, parameter.ToArray());
  438. }
  439. }
  440. }
  441. }
  442. }
  443. }
  444. break;
  445. case Actions.Authorize:
  446. {
  447. AuthorizeRequest _request = request as AuthorizeRequest;
  448. //特例****飛宏客戶旗下的電樁,若遇到Portal沒回應的狀況 ~允許充電
  449. var businessService = BusinessServiceFactory.CreateBusinessService(session.CustomerId.ToString());
  450. var confirm = new AuthorizeConfirmation()
  451. {
  452. idTagInfo = _request.idTag == "Backend" ? new IdTagInfo() { expiryDate = DateTime.UtcNow.AddDays(1), status = AuthorizationStatus.Accepted } : await businessService.Authorize(session.ChargeBoxId, _request.idTag)
  453. };
  454. if (session.CustomerId.ToString().ToUpper() == "8456AED9-6DD9-4BF3-A94C-9F5DCB9506F7" && confirm.idTagInfo.status == AuthorizationStatus.ConcurrentTx)
  455. {
  456. confirm.idTagInfo = new IdTagInfo() { expiryDate = DateTime.UtcNow.AddDays(1), status = AuthorizationStatus.Accepted };
  457. }
  458. result.Message = confirm;
  459. result.Success = true;
  460. }
  461. break;
  462. default:
  463. {
  464. Console.WriteLine(string.Format("Not Implement {0} Logic(ExecuteCoreRequest)", request.GetType().ToString().Replace("OCPPPackage.Messages.Core.", "")));
  465. }
  466. break;
  467. }
  468. }
  469. catch (Exception ex)
  470. {
  471. logger.Fatal(string.Format("chargeBoxId:{0} {1}", session.ChargeBoxId, action));
  472. logger.Fatal(string.Format("Data {0}", request.ToString()));
  473. logger.Fatal(string.Format("Error {0}", ex.ToString()));
  474. result.Exception = ex;
  475. }
  476. if (action == Actions.Heartbeat)
  477. {
  478. watch.Stop();
  479. if (watch.ElapsedMilliseconds / 1000 > 3)
  480. {
  481. logger.Error("Processing Hearbeat costs " + watch.ElapsedMilliseconds / 1000 + " seconds");
  482. }
  483. }
  484. return result;
  485. }
  486. internal MessageResult ExecuteCoreConfirm(Actions action, ClientData session, IConfirmation confirm, string requestId)
  487. {
  488. MessageResult result = new MessageResult() { Success = true };
  489. switch (action)
  490. {
  491. case Actions.DataTransfer:
  492. {
  493. DataTransferConfirmation _confirm = confirm as DataTransferConfirmation;
  494. DataTransferRequest _request = _confirm.GetRequest() as DataTransferRequest;
  495. using (var db = new MainDBContext())
  496. {
  497. var operation = db.MachineOperateRecord.Where(x => x.SerialNo == requestId &&
  498. x.ChargeBoxId == session.ChargeBoxId && x.Status == 0).FirstOrDefault();
  499. if (operation != null)
  500. {
  501. operation.FinishedOn = DateTime.Now;
  502. operation.Status = 1;//電樁有回覆
  503. operation.EVSE_Status = (int)_confirm.status;
  504. operation.EVSE_Value = string.IsNullOrEmpty(_confirm.data) ? "" : _confirm.data;
  505. db.SaveChanges();
  506. }
  507. if (_request.messageId == "ID_FirmwareVersion")
  508. {
  509. var machine = new Machine() { Id = session.MachineId };
  510. if (machine != null)
  511. {
  512. db.Configuration.AutoDetectChangesEnabled = false;
  513. db.Configuration.ValidateOnSaveEnabled = false;
  514. db.Machine.Attach(machine);
  515. machine.BoardVersions = _confirm.data;
  516. db.Entry(machine).Property(x => x.BoardVersions).IsModified = true;
  517. db.SaveChanges();
  518. }
  519. }
  520. }
  521. }
  522. break;
  523. case Actions.ChangeAvailability:
  524. {
  525. ChangeAvailabilityConfirmation _confirm = confirm as ChangeAvailabilityConfirmation;
  526. ChangeAvailabilityRequest _request = _confirm.GetRequest() as ChangeAvailabilityRequest;
  527. using (var db = new MainDBContext())
  528. {
  529. var operation = db.MachineOperateRecord.Where(x => x.SerialNo == requestId &&
  530. x.ChargeBoxId == session.ChargeBoxId && x.Status == 0).FirstOrDefault();
  531. if (operation != null)
  532. {
  533. operation.FinishedOn = DateTime.Now;
  534. operation.Status = 1;//電樁有回覆
  535. operation.EVSE_Status = (int)_confirm.status;
  536. operation.EVSE_Value = _confirm.status.ToString();
  537. db.SaveChanges();
  538. }
  539. }
  540. }
  541. break;
  542. case Actions.ClearCache:
  543. {
  544. ClearCacheConfirmation _confirm = confirm as ClearCacheConfirmation;
  545. ClearCacheRequest _request = _confirm.GetRequest() as ClearCacheRequest;
  546. using (var db = new MainDBContext())
  547. {
  548. var operation = db.MachineOperateRecord.Where(x => x.SerialNo == requestId &&
  549. x.ChargeBoxId == session.ChargeBoxId && x.Status == 0).FirstOrDefault();
  550. if (operation != null)
  551. {
  552. operation.FinishedOn = DateTime.Now;
  553. operation.Status = 1;//電樁有回覆
  554. operation.EVSE_Status = (int)_confirm.status;
  555. operation.EVSE_Value = _confirm.status.ToString();
  556. db.SaveChanges();
  557. }
  558. }
  559. }
  560. break;
  561. case Actions.RemoteStartTransaction:
  562. {
  563. RemoteStartTransactionConfirmation _confirm = confirm as RemoteStartTransactionConfirmation;
  564. RemoteStartTransactionRequest _request = _confirm.GetRequest() as RemoteStartTransactionRequest;
  565. using (var db = new MainDBContext())
  566. {
  567. var operation = db.MachineOperateRecord.Where(x => x.SerialNo == requestId &&
  568. x.ChargeBoxId == session.ChargeBoxId && x.Status == 0).FirstOrDefault();
  569. if (operation != null)
  570. {
  571. operation.FinishedOn = DateTime.Now;
  572. operation.Status = 1;//電樁有回覆
  573. operation.EVSE_Status = (int)_confirm.status;
  574. operation.EVSE_Value = _confirm.status.ToString();
  575. db.SaveChanges();
  576. }
  577. }
  578. }
  579. break;
  580. case Actions.RemoteStopTransaction:
  581. {
  582. RemoteStopTransactionConfirmation _confirm = confirm as RemoteStopTransactionConfirmation;
  583. RemoteStopTransactionRequest _request = _confirm.GetRequest() as RemoteStopTransactionRequest;
  584. using (var db = new MainDBContext())
  585. {
  586. var operation = db.MachineOperateRecord.Where(x => x.SerialNo == requestId &&
  587. x.ChargeBoxId == session.ChargeBoxId && x.Status == 0).FirstOrDefault();
  588. if (operation != null)
  589. {
  590. operation.FinishedOn = DateTime.Now;
  591. operation.Status = 1;//電樁有回覆
  592. operation.EVSE_Status = (int)_confirm.status;
  593. operation.EVSE_Value = _confirm.status.ToString();
  594. db.SaveChanges();
  595. }
  596. }
  597. }
  598. break;
  599. case Actions.Reset:
  600. {
  601. ResetConfirmation _confirm = confirm as ResetConfirmation;
  602. ResetRequest _request = _confirm.GetRequest() as ResetRequest;
  603. using (var db = new MainDBContext())
  604. {
  605. var operation = db.MachineOperateRecord.Where(x => x.SerialNo == requestId &&
  606. x.ChargeBoxId == session.ChargeBoxId && x.Status == 0).FirstOrDefault();
  607. if (operation != null)
  608. {
  609. operation.FinishedOn = DateTime.Now;
  610. operation.Status = 1;//電樁有回覆
  611. operation.EVSE_Status = (int)_confirm.status;
  612. operation.EVSE_Value = _confirm.status.ToString();
  613. db.SaveChanges();
  614. }
  615. }
  616. }
  617. break;
  618. case Actions.ChangeConfiguration:
  619. {
  620. ChangeConfigurationConfirmation _confirm = confirm as ChangeConfigurationConfirmation;
  621. ChangeConfigurationRequest _request = _confirm.GetRequest() as ChangeConfigurationRequest;
  622. using (var db = new MainDBContext())
  623. {
  624. var operation = db.MachineOperateRecord.Where(x => x.SerialNo == requestId &&
  625. x.ChargeBoxId == session.ChargeBoxId && x.Status == 0).FirstOrDefault();
  626. if (operation != null)
  627. {
  628. operation.FinishedOn = DateTime.Now;
  629. operation.Status = 1;//電樁有回覆
  630. operation.EVSE_Status = (int)_confirm.status;
  631. operation.EVSE_Value = _confirm.status.ToString();
  632. }
  633. if (_confirm.status == Packet.Messages.SubTypes.ConfigurationStatus.Accepted || _confirm.status == Packet.Messages.SubTypes.ConfigurationStatus.RebootRequired)
  634. {
  635. var configure = db.MachineConfiguration.Where(x => x.ChargeBoxId == session.ChargeBoxId).ToList();
  636. var foundConfig = configure.Find(x => x.ConfigureName == _request.key);
  637. if (foundConfig != null)
  638. {
  639. foundConfig.ReadOnly = false;
  640. foundConfig.ConfigureSetting = _request.value;
  641. }
  642. else
  643. {
  644. db.MachineConfiguration.Add(new MachineConfiguration()
  645. {
  646. ChargeBoxId = session.ChargeBoxId,
  647. ConfigureName = _request.key,
  648. ReadOnly = false,
  649. ConfigureSetting = _request.value
  650. });
  651. }
  652. }
  653. db.SaveChanges();
  654. }
  655. }
  656. break;
  657. case Actions.GetConfiguration:
  658. {
  659. try
  660. {
  661. GetConfigurationConfirmation _confirm = confirm as GetConfigurationConfirmation;
  662. // GetConfigurationRequest _request = _confirm.GetRequest() as GetConfigurationRequest;
  663. using (var db = new MainDBContext())
  664. {
  665. var configure = db.MachineConfiguration.Where(x => x.ChargeBoxId == session.ChargeBoxId).ToList();
  666. if (_confirm.configurationKey != null)
  667. {
  668. foreach (var item in _confirm.configurationKey)
  669. {
  670. string oldValue = string.Empty;
  671. if (item.key == null)
  672. {
  673. Console.WriteLine("*********************");
  674. }
  675. var foundConfig = configure.Find(x => x.ConfigureName == item.key);
  676. if (foundConfig != null)
  677. {
  678. if (foundConfig.ConfigureName == null)
  679. {
  680. Console.WriteLine("*********************");
  681. }
  682. if (foundConfig.ConfigureName == "SecurityProfile")
  683. {
  684. oldValue = foundConfig.ConfigureSetting;
  685. }
  686. foundConfig.ReadOnly = item.IsReadOnly;
  687. foundConfig.ConfigureSetting = string.IsNullOrEmpty(item.value) ? string.Empty : item.value;
  688. }
  689. else
  690. {
  691. db.MachineConfiguration.Add(new MachineConfiguration()
  692. {
  693. ChargeBoxId = session.ChargeBoxId,
  694. ConfigureName = item.key,
  695. ReadOnly = item.IsReadOnly,
  696. ConfigureSetting = string.IsNullOrEmpty(item.value) ? string.Empty : item.value,
  697. Exists = true
  698. });
  699. }
  700. }
  701. }
  702. if (_confirm.unknownKey != null)
  703. {
  704. foreach (var item in _confirm.unknownKey)
  705. {
  706. var foundConfig = configure.Find(x => x.ConfigureName == item);
  707. if (foundConfig != null)
  708. {
  709. foundConfig.ReadOnly = true;
  710. foundConfig.ConfigureSetting = string.Empty;
  711. foundConfig.Exists = false;
  712. }
  713. else
  714. {
  715. db.MachineConfiguration.Add(new MachineConfiguration()
  716. {
  717. ChargeBoxId = session.ChargeBoxId,
  718. ConfigureName = item
  719. });
  720. }
  721. }
  722. }
  723. var operation = db.MachineOperateRecord.Where(x => x.SerialNo == requestId &&
  724. x.ChargeBoxId == session.ChargeBoxId && x.Status == 0).FirstOrDefault();
  725. if (operation != null)
  726. {
  727. operation.FinishedOn = DateTime.Now;
  728. operation.Status = 1;//電樁有回覆
  729. operation.EVSE_Status = 1;
  730. operation.EVSE_Value = JsonConvert.SerializeObject(_confirm.configurationKey, Formatting.None);
  731. }
  732. db.SaveChanges();
  733. }
  734. }
  735. catch (Exception ex)
  736. {
  737. logger.Error(ex.ToString());
  738. }
  739. }
  740. break;
  741. case Actions.UnlockConnector:
  742. {
  743. UnlockConnectorConfirmation _confirm = confirm as UnlockConnectorConfirmation;
  744. UnlockConnectorRequest _request = _confirm.GetRequest() as UnlockConnectorRequest;
  745. using (var db = new MainDBContext())
  746. {
  747. var operation = db.MachineOperateRecord.Where(x => x.SerialNo == requestId &&
  748. x.ChargeBoxId == session.ChargeBoxId && x.Status == 0).FirstOrDefault();
  749. if (operation != null)
  750. {
  751. operation.FinishedOn = DateTime.Now;
  752. operation.Status = 1;//電樁有回覆
  753. operation.EVSE_Status = (int)_confirm.status;
  754. operation.EVSE_Value = _confirm.status.ToString();
  755. db.SaveChanges();
  756. }
  757. }
  758. }
  759. break;
  760. default:
  761. {
  762. Console.WriteLine(string.Format("Not Implement {0} Logic(ExecuteCoreConfirm)", confirm.GetType().ToString().Replace("OCPPPackage.Messages.Core.", "")));
  763. }
  764. break;
  765. }
  766. return result;
  767. }
  768. internal MessageResult ReceivedCoreError(Actions action, string errorMsg, ClientData session, string requestId)
  769. {
  770. MessageResult result = new MessageResult() { Success = true };
  771. switch (action)
  772. {
  773. case Actions.ChangeAvailability:
  774. case Actions.ChangeConfiguration:
  775. case Actions.ClearCache:
  776. case Actions.RemoteStartTransaction:
  777. case Actions.RemoteStopTransaction:
  778. case Actions.Reset:
  779. case Actions.GetConfiguration:
  780. case Actions.UnlockConnector:
  781. case Actions.DataTransfer:
  782. {
  783. if (action == Actions.DataTransfer)
  784. {
  785. logger.Debug(string.Format("DataTransfer Error {0}: {1}", session.ChargeBoxId, requestId));
  786. }
  787. using (var db = new MainDBContext())
  788. {
  789. var operation = db.MachineOperateRecord.Where(x => x.SerialNo == requestId &&
  790. x.ChargeBoxId == session.ChargeBoxId && x.Status == 0).FirstOrDefault();
  791. if (operation != null)
  792. {
  793. operation.FinishedOn = DateTime.Now;
  794. operation.Status = 1;//電樁有回覆
  795. operation.EVSE_Status = (int)255;//錯誤
  796. operation.EVSE_Value = errorMsg;
  797. db.SaveChanges();
  798. }
  799. }
  800. }
  801. break;
  802. default:
  803. {
  804. Console.WriteLine(string.Format("Not Implement {0} Logic(ReceivedCoreError)", action));
  805. }
  806. break;
  807. }
  808. return result;
  809. }
  810. }
  811. }