Нема описа

Utility.cs 49KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021
  1. using Newtonsoft.Json;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Data.SQLite;
  5. using System.Globalization;
  6. using System.IO;
  7. using System.Linq;
  8. using System.Management;
  9. using System.Net;
  10. using System.Runtime.InteropServices;
  11. using System.Security.Cryptography;
  12. using System.Text;
  13. using System.Windows.Forms;
  14. namespace BulkPrinting
  15. {
  16. public class Utility
  17. {
  18. public const int WM_USER = 0x400;
  19. [DllImport("user32.dll", SetLastError = true)]
  20. public static extern bool PostMessage(HandleRef hWnd, uint Msg, IntPtr wParam, IntPtr lParam);
  21. public static string GetHDDSerial()
  22. {
  23. ManagementObjectSearcher managementObjectSearcher = new ManagementObjectSearcher("SELECT * FROM Win32_PhysicalMedia");
  24. string result;
  25. ManagementObjectCollection.ManagementObjectEnumerator enumerator = null;
  26. try
  27. {
  28. enumerator = managementObjectSearcher.Get().GetEnumerator();
  29. while (enumerator.MoveNext())
  30. {
  31. string text = ((ManagementObject)enumerator.Current)["SerialNumber"].ToString();
  32. text = text.Replace(" ", "").Replace(".", "");
  33. if (text != "")
  34. {
  35. result = text;
  36. return result.Length > 15 ? result.Substring(result.Length - 15, 15) : result;
  37. }
  38. }
  39. }
  40. finally
  41. {
  42. if (enumerator != null)
  43. {
  44. ((IDisposable)enumerator).Dispose();
  45. }
  46. }
  47. result = "SERIAL ERROR";
  48. return result.Length > 15 ? result.Substring(result.Length - 15, 15) : result;
  49. }
  50. public static DBHelper OpenDBConnection()
  51. {
  52. return new DBHelper(String.Format("Data Source={0};", Globals.MaxDBFilePath), Globals.SessionDatabasePassword);
  53. }
  54. public static bool Login(LoginData UserLoginData, bool Offline, bool RememberMe) {
  55. string MaxDBPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), Configuration.MaxDataPathName);
  56. string PostData = JsonConvert.SerializeObject(UserLoginData);
  57. byte[] data = Encoding.ASCII.GetBytes(PostData);
  58. HttpWebRequest request = WebRequest.Create(Configuration.ServerBaseURL + "/api/login") as HttpWebRequest;
  59. request.ServerCertificateValidationCallback = delegate { return true; };
  60. request.Method = "POST";
  61. request.ContentType = "application/json";
  62. request.ContentLength = data.Length;
  63. request.Accept = "application/json";
  64. try
  65. {
  66. using (var stream = request.GetRequestStream())
  67. {
  68. stream.Write(data, 0, data.Length);
  69. }
  70. string responseString;
  71. using (var response = (HttpWebResponse)request.GetResponse())
  72. {
  73. responseString = new StreamReader(response.GetResponseStream()).ReadToEnd();
  74. }
  75. Globals.SessionData = JsonConvert.DeserializeObject<OKResponse>(responseString);
  76. if (RememberMe == true)
  77. {
  78. SaveSetting("Username", UserLoginData.Username);
  79. SaveSetting("UserID", UserLoginData.UserId.ToString());
  80. }
  81. else {
  82. SaveSetting("Username", "");
  83. SaveSetting("UserID", "");
  84. }
  85. SaveSetting("VendorID", UserLoginData.VendorId.ToString());
  86. Globals.SessionEncryptedDatabasePassword = Globals.SessionData.Credentials.Payload.EncryptedDatabasePassword;
  87. Globals.SessionSalt = Globals.SessionData.Credentials.Salt;
  88. Globals.SessionIterations = Globals.SessionData.Credentials.Iterations;
  89. return true;
  90. }
  91. catch (WebException ex)
  92. {
  93. if (ex.Response != null)
  94. {
  95. var response = ex.Response;
  96. var stream = response.GetResponseStream();
  97. var reader = new StreamReader(stream);
  98. var message = reader.ReadToEnd();
  99. MaxException MaxError = JsonConvert.DeserializeObject<MaxException>(message);
  100. if (MaxError.Code != null)
  101. {
  102. MessageBox.Show("Error " + MaxError.Code.ToString() + ": " + MaxError.Error, "Login Failed", MessageBoxButtons.OK, MessageBoxIcon.Error);
  103. }
  104. else
  105. {
  106. MessageBox.Show("Login Failed. Please try again.", "Login Failed", MessageBoxButtons.OK, MessageBoxIcon.Error);
  107. }
  108. }
  109. else
  110. {
  111. MessageBox.Show("The server did not return a response to the login request. Please try again later.", "Login Failed", MessageBoxButtons.OK, MessageBoxIcon.Error);
  112. }
  113. return false;
  114. }
  115. }
  116. public static bool RESTRequest<R>(ref R Result, string RESTPath)
  117. {
  118. return RESTRequest<bool?, R>(null, ref Result, RESTPath);
  119. }
  120. public static bool RESTRequest<T, R>(T POSTData, ref R Result, string RESTPath)
  121. {
  122. HttpWebRequest request = WebRequest.Create(Configuration.ServerBaseURL + RESTPath) as HttpWebRequest;
  123. request.AutomaticDecompression = DecompressionMethods.Deflate | DecompressionMethods.GZip;
  124. request.ServerCertificateValidationCallback = delegate { return true; };
  125. request.Headers.Add("Authorization", "Bearer " + Globals.SessionData.AccessToken);
  126. request.ContentType = "application/json";
  127. request.Accept = "application/json";
  128. try
  129. {
  130. if (POSTData != null)
  131. {
  132. string PostData = JsonConvert.SerializeObject(POSTData);
  133. byte[] data = Encoding.ASCII.GetBytes(PostData);
  134. request.ContentLength = data.Length;
  135. request.Method = "POST";
  136. using (var stream = request.GetRequestStream())
  137. {
  138. stream.Write(data, 0, data.Length);
  139. }
  140. }
  141. else {
  142. request.Method = "GET";
  143. }
  144. string responseString;
  145. using (var response = (HttpWebResponse)request.GetResponse())
  146. {
  147. responseString = new StreamReader(response.GetResponseStream()).ReadToEnd();
  148. }
  149. Result = JsonConvert.DeserializeObject<R>(responseString);
  150. return true;
  151. }
  152. catch (WebException ex)
  153. {
  154. if (ex != null && ex.Response != null)
  155. {
  156. var response = ex.Response;
  157. HttpStatusCode? status = (response as HttpWebResponse)?.StatusCode;
  158. if (status.HasValue && (status == HttpStatusCode.NotFound))
  159. {
  160. Result = default(R);
  161. }
  162. else
  163. {
  164. var stream = response.GetResponseStream();
  165. var reader = new StreamReader(stream);
  166. var message = reader.ReadToEnd();
  167. MaxException MaxError = JsonConvert.DeserializeObject<MaxException>(message);
  168. if (MaxError.Code != null)
  169. {
  170. MessageBox.Show("Error " + MaxError.Code.ToString() + ": " + MaxError.Error, "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
  171. }
  172. }
  173. }
  174. return false;
  175. }
  176. }
  177. private static bool SaveSetting(string Key, string Value)
  178. {
  179. SavedSettings StoredValues = LoadSavedSettings();
  180. switch (Key)
  181. {
  182. case "Username":
  183. StoredValues.Username = Value;
  184. break;
  185. case "UserID":
  186. if (Value == "")
  187. StoredValues.UserId = 0;
  188. else
  189. StoredValues.UserId = int.Parse(Value);
  190. break;
  191. case "VendorID":
  192. if (Value == "")
  193. StoredValues.VendorId = 0;
  194. else
  195. StoredValues.VendorId = int.Parse(Value);
  196. break;
  197. default:
  198. return false;
  199. }
  200. string MaxAppDataPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), Configuration.MaxDataPathName);
  201. string MaxSettingsFilePath = Path.Combine(MaxAppDataPath, "maxsettings.dat");
  202. File.WriteAllText(MaxSettingsFilePath, JsonConvert.SerializeObject(StoredValues));
  203. return true;
  204. }
  205. public static string LoadSetting(string Key)
  206. {
  207. SavedSettings StoredValues = LoadSavedSettings();
  208. string ReturnVal = "";
  209. switch (Key)
  210. {
  211. case "Username":
  212. ReturnVal = StoredValues.Username;
  213. break;
  214. case "UserID":
  215. ReturnVal = StoredValues.UserId.ToString();
  216. break;
  217. case "VendorID":
  218. ReturnVal = StoredValues.VendorId.ToString();
  219. break;
  220. default:
  221. return "";
  222. }
  223. if (ReturnVal == "0")
  224. return "";
  225. return ReturnVal;
  226. }
  227. private static SavedSettings LoadSavedSettings() {
  228. string MaxAppDataPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), Configuration.MaxDataPathName);
  229. string MaxSettingsFilePath = Path.Combine(MaxAppDataPath, "maxsettings.dat");
  230. SavedSettings StoredValues = new SavedSettings();
  231. if (!Directory.Exists(MaxAppDataPath))
  232. {
  233. Directory.CreateDirectory(MaxAppDataPath);
  234. }
  235. if (File.Exists(MaxSettingsFilePath)) {
  236. StoredValues = JsonConvert.DeserializeObject<SavedSettings>(File.ReadAllText(MaxSettingsFilePath));
  237. }
  238. else {
  239. File.WriteAllText(MaxSettingsFilePath, JsonConvert.SerializeObject(StoredValues));
  240. }
  241. return StoredValues;
  242. }
  243. public static byte[] Transform(ICryptoTransform transform, byte[] input)
  244. {
  245. using (var memoryStream = new MemoryStream())
  246. using (var cryptoStream = new CryptoStream(memoryStream, transform, CryptoStreamMode.Write))
  247. {
  248. cryptoStream.Write(input, 0, input.Length);
  249. cryptoStream.FlushFinalBlock();
  250. return memoryStream.ToArray();
  251. }
  252. }
  253. public static byte[] AesDecryptBytes(byte[] cipherText, byte[] key)
  254. {
  255. using (var aes = Aes.Create())
  256. using (var decryptor = aes.CreateDecryptor(key, new byte[16]))
  257. {
  258. return Transform(decryptor, cipherText);
  259. }
  260. }
  261. public static string AesDecryptString(byte[] cipherText, byte[] key)
  262. {
  263. return Encoding.ASCII.GetString(AesDecryptBytes(cipherText, key));
  264. }
  265. public static string TripleDESDecrypt(string cipherText, TripleDES des)
  266. {
  267. using (var decryptor = des.CreateDecryptor(des.Key, des.IV))
  268. {
  269. return Encoding.UTF8.GetString(Transform(decryptor, Convert.FromBase64String(cipherText)));
  270. }
  271. }
  272. public static string TripleDESDecrypt(string cipherText, byte[] key)
  273. {
  274. using (var des = TripleDES.Create())
  275. {
  276. des.Key = key;
  277. des.IV = new byte[8];
  278. return TripleDESDecrypt(cipherText, des);
  279. }
  280. }
  281. public static Batch GetBatch(int BatchId) {
  282. Batch RequestedBatch = new Batch();
  283. bool OrderResult = RESTRequest<Batch>(ref RequestedBatch, String.Format("/api/batches/{0}", BatchId));
  284. return RequestedBatch;
  285. }
  286. public static string GetNextInternalReference()
  287. {
  288. InternalReferenceResponse InternalReferenceRequest = new InternalReferenceResponse();
  289. bool OrderResult = RESTRequest<InternalReferenceResponse>(ref InternalReferenceRequest, "/api/vendors/nextinternalref");
  290. return InternalReferenceRequest.InternalReference;
  291. }
  292. public static void SaveBatch(DBHelper db, Batch batch)
  293. {
  294. lock (db.WriteLock)
  295. {
  296. using (var trans = db.BeginTransaction())
  297. {
  298. using (var command = db.CreateCommand(trans))
  299. {
  300. command.Parameters.AddWithValue("@Id", batch.Id);
  301. command.Parameters.AddWithValue("@OrderDate", batch.OrderDate);
  302. command.Parameters.AddWithValue("@OrderGuid", batch.OrderGuid);
  303. command.Parameters.AddWithValue("@OrderReference", batch.OrderReference);
  304. command.Parameters.AddWithValue("@NetworkId", batch.NetworkId);
  305. command.Parameters.AddWithValue("@NetworkName", batch.NetworkName);
  306. command.Parameters.AddWithValue("@ProductId", batch.ProductId);
  307. command.Parameters.AddWithValue("@ProductDescription", batch.ProductDescription);
  308. command.Parameters.AddWithValue("@VoucherType", batch.VoucherType);
  309. command.Parameters.AddWithValue("@FaceValue", batch.FaceValue);
  310. command.Parameters.AddWithValue("@DiscountPercentage", batch.DiscountPercentage);
  311. command.Parameters.AddWithValue("@RequestedQuantity", batch.RequestedQuantity);
  312. command.Parameters.AddWithValue("@DeliveredQuantity", batch.DeliveredQuantity);
  313. command.Parameters.AddWithValue("@Cost", batch.Cost);
  314. command.Parameters.AddWithValue("@ReadyForDownload", batch.ReadyForDownload);
  315. command.Parameters.AddWithValue("@InternalReference", batch.InternalReference);
  316. // We leave the value of 'Downloaded' unchanged here.
  317. command.CommandText =
  318. "UPDATE Batch SET OrderDate=@OrderDate, OrderGuid=@OrderGuid, OrderReference=@OrderReference, NetworkId=@NetworkId, " +
  319. "NetworkName=@NetworkName, ProductId=@ProductId, ProductDescription=@ProductDescription, VoucherType=@VoucherType, " +
  320. "FaceValue=@FaceValue, DiscountPercentage=@DiscountPercentage, RequestedQuantity=@RequestedQuantity, " +
  321. "DeliveredQuantity=@DeliveredQuantity, Cost=@Cost, ReadyForDownload=@ReadyForDownload, InternalReference=@InternalReference " +
  322. "WHERE Id=@Id";
  323. if (command.ExecuteNonQuery() == 0)
  324. {
  325. // 'Downloaded' is always 0 for new batches.
  326. command.CommandText =
  327. "INSERT INTO Batch (Id, OrderDate, OrderGuid, OrderReference, NetworkId, NetworkName, ProductId, ProductDescription, VoucherType, FaceValue, DiscountPercentage, RequestedQuantity, DeliveredQuantity, Cost, ReadyForDownload, InternalReference, Downloaded) " +
  328. "VALUES (@Id,@OrderDate,@OrderGuid,@OrderReference,@NetworkId,@NetworkName,@ProductId,@ProductDescription,@VoucherType,@FaceValue,@DiscountPercentage,@RequestedQuantity,@DeliveredQuantity,@Cost,@ReadyForDownload,@InternalReference,0)";
  329. command.ExecuteNonQuery();
  330. }
  331. }
  332. trans.Commit();
  333. }
  334. }
  335. }
  336. public static string GetSavedParameter(DBHelper db, string key)
  337. {
  338. return (string)db.ExecuteScalar("SELECT Value FROM Parameters WHERE Key=@Key",
  339. new SQLiteParameter("@Key", key));
  340. }
  341. public static int GetSavedParameterAsInt(DBHelper db, string key)
  342. {
  343. return Convert.ToInt32(GetSavedParameter(db, key));
  344. }
  345. public static bool GetSavedParameterAsBoolean(DBHelper db, string key)
  346. {
  347. return Convert.ToBoolean(GetSavedParameter(db, key));
  348. }
  349. public static bool UpdateSavedParameter(DBHelper db, string key, object value)
  350. {
  351. return db.ExecuteNonQuery(
  352. "UPDATE Parameters SET Value=@Value WHERE Key=@Key",
  353. new SQLiteParameter("@Key", key),
  354. new SQLiteParameter("@Value", Convert.ToString(value))) > 0;
  355. }
  356. public static void PrintVouchers(DBHelper db, int BatchId, int StartSeqNo, int EndSeqNo)
  357. {
  358. int VoucherCount = 0;
  359. int RowCount = 0;
  360. int PageCount = (int) Math.Ceiling(((decimal)StartSeqNo)/20);
  361. int TotalCount = 0;
  362. string SerialNumberTrimmed;
  363. bool IsReprint;
  364. var PrinterInitString = new StringBuilder();
  365. PrinterInitString.Append(Printer.INITIALISE_PRINTER).Append(Printer.EMPHASISE_ON).Append(Printer.UNIDIRECTIONAL_OFF).Append(Printer.CHARPITCHELITE);
  366. int initJobID = Globals.MaxPrinter.Open("Printer_Init");
  367. if (initJobID == 0) return;
  368. Globals.MaxPrinter.Print(PrinterInitString.ToString());
  369. Globals.MaxPrinter.Close();
  370. List<EventLog> LogEvents = new List<EventLog>();
  371. IList<PrintVoucher> VoucherRow = new List<PrintVoucher>();
  372. using (var Command = db.CreateCommand("SELECT DISTINCT v.Id,v.SequenceNumber,v.Serial,v.EncryptedPIN,v.BatchId,b.ProductDescription,l.VoucherId From Voucher v LEFT JOIN Batch b on v.BatchId = b.Id LEFT JOIN Logs l ON v.Id = l.VoucherId AND l.EventType = @eventtype WHERE v.BatchId=@batch_id AND v.SequenceNumber BETWEEN @seqstartno AND @seqendno ORDER BY v.SequenceNumber",
  373. new SQLiteParameter("@batch_id", BatchId),
  374. new SQLiteParameter("@eventtype", VendorEvent.VendorEventType.PrintVoucher),
  375. new SQLiteParameter("@seqstartno", StartSeqNo),
  376. new SQLiteParameter("@seqendno", EndSeqNo)))
  377. {
  378. using (SQLiteDataReader read = Command.ExecuteReader())
  379. {
  380. int JobID = Globals.MaxPrinter.Open("Vouchers");
  381. if (JobID == 0) return;
  382. while (read.Read())
  383. {
  384. VoucherCount++;
  385. PrintVoucher IndividualVoucher = new PrintVoucher();
  386. IndividualVoucher.SequenceNumber = (int)read["SequenceNumber"];
  387. IndividualVoucher.BatchId = (int)read["BatchId"];
  388. IndividualVoucher.Serial = (string)read["Serial"];
  389. IndividualVoucher.VoucherId = (int)read["Id"];
  390. IndividualVoucher.Description = (string)read["ProductDescription"];
  391. IsReprint = !read.IsDBNull(6);
  392. if (IsReprint)
  393. {
  394. IndividualVoucher.Description = "*" + IndividualVoucher.Description;
  395. }
  396. IndividualVoucher.DecryptedPIN = TripleDESDecrypt((string)read["EncryptedPIN"], Globals.SessionVoucherKey);
  397. VoucherRow.Add(IndividualVoucher);
  398. if (VoucherCount >= 5 || TotalCount == (EndSeqNo - StartSeqNo))
  399. {
  400. RowCount++;
  401. string PrintRow = "\r\n\n\n ";
  402. for (int Column = 0; Column < VoucherRow.Count(); Column++)
  403. {
  404. IndividualVoucher = VoucherRow[Column];
  405. PrintRow += (Column == 2 ? " " : "") + (" " + IndividualVoucher.DecryptedPIN).PadRight(30, ' ');
  406. }
  407. PrintRow = PrintRow.TrimEnd() + "\r\n\n ";
  408. for (int Column = 0; Column < VoucherRow.Count(); Column++)
  409. {
  410. IndividualVoucher = VoucherRow[Column];
  411. SerialNumberTrimmed = IndividualVoucher.Serial;
  412. if (SerialNumberTrimmed.Length > 24)
  413. {
  414. SerialNumberTrimmed = SerialNumberTrimmed.Substring(0, 24);
  415. }
  416. PrintRow += (Column == 2 ? " " : "") + SerialNumberTrimmed.PadRight(30, ' ');
  417. }
  418. PrintRow = PrintRow.TrimEnd() + "\r\n ";
  419. for (int Column = 0; Column < VoucherRow.Count(); Column++)
  420. {
  421. IndividualVoucher = VoucherRow[Column];
  422. int CurrentPage = (int)Math.Ceiling(((decimal)IndividualVoucher.SequenceNumber) / 20);
  423. PrintRow += (Column == 2 ? " " : "") + String.Format("{0}/{1}/{2}", IndividualVoucher.BatchId, IndividualVoucher.SequenceNumber, PageCount).PadRight(30, ' ');
  424. }
  425. PrintRow = PrintRow.TrimEnd() + "\r\n ";
  426. for (int Column = 0; Column < VoucherRow.Count(); Column++)
  427. {
  428. IndividualVoucher = VoucherRow[Column];
  429. PrintRow += (Column == 2 ? " " : "") + IndividualVoucher.Description.PadRight(30, ' ');
  430. }
  431. PrintRow = PrintRow.TrimEnd() + "\r\n\n\n\n\n\n\n\n\n\n\n";
  432. Globals.MaxPrinter.Print(PrintRow);
  433. foreach (PrintVoucher PrintedVoucher in VoucherRow)
  434. {
  435. var ExportEvent = new EventLog();
  436. ExportEvent.EventType = VendorEvent.VendorEventType.PrintVoucher;
  437. ExportEvent.VoucherId = PrintedVoucher.VoucherId;
  438. ExportEvent.Retry = IsReprint;
  439. LogEvents.Add(ExportEvent);
  440. }
  441. VoucherRow = new List<PrintVoucher>();
  442. VoucherCount = 0;
  443. if (RowCount >= 4)
  444. {
  445. //Globals.MaxPrinter.NewPage();
  446. PageCount++;
  447. RowCount = 0;
  448. //Globals.MaxPrinter.GetJobInfo(JobID);
  449. }
  450. }
  451. TotalCount++;
  452. }
  453. Globals.MaxPrinter.Close();
  454. }
  455. }
  456. LogBulkEvents(db, LogEvents);
  457. }
  458. public enum UserPermissions {
  459. CanPrintOnline,
  460. CanReprintOnline,
  461. CanPrintOffline,
  462. CanReprintOffline,
  463. BulkExport,
  464. BulkOrder,
  465. BulkViewPins,
  466. BulkReExport
  467. }
  468. public static bool CheckUserAccess(UserPermissions Permission) {
  469. if (Globals.SessionData.Credentials.Payload.User.Level == (int)UserLevel.Administrator) {
  470. return true;
  471. }
  472. switch (Permission) {
  473. case UserPermissions.CanPrintOnline:
  474. return Globals.SessionData.Credentials.Payload.User.CanPrintOnline;
  475. case UserPermissions.CanReprintOnline:
  476. return Globals.SessionData.Credentials.Payload.User.CanReprintOnline;
  477. case UserPermissions.CanPrintOffline:
  478. return Globals.SessionData.Credentials.Payload.User.CanPrintOffline;
  479. case UserPermissions.CanReprintOffline:
  480. return Globals.SessionData.Credentials.Payload.User.CanReprintOffline;
  481. case UserPermissions.BulkViewPins:
  482. return Globals.SessionData.Credentials.Payload.User.BulkViewPins;
  483. case UserPermissions.BulkOrder:
  484. return Globals.SessionData.Credentials.Payload.User.BulkOrder;
  485. case UserPermissions.BulkExport:
  486. return Globals.SessionData.Credentials.Payload.User.BulkExport;
  487. case UserPermissions.BulkReExport:
  488. return Globals.SessionData.Credentials.Payload.User.BulkReExport;
  489. default:
  490. return false;
  491. }
  492. }
  493. public static void Logout() {
  494. if (Globals.SessionData != null)
  495. {
  496. Globals.DB.ExecuteNonQuery("DELETE FROM SessionData"); //Destroy stored session data
  497. string SessionDataJson = JsonConvert.SerializeObject(Globals.SessionData);
  498. Globals.DB.ExecuteNonQuery(
  499. "INSERT INTO SessionData (Key,Value) VALUES (@key,@value)",
  500. new SQLiteParameter("@key", "SessionDataJson"),
  501. new SQLiteParameter("@value", SessionDataJson));
  502. LogEvent(Globals.DB, VendorEvent.VendorEventType.Logout);
  503. if (Globals.LogUploader != null)
  504. Globals.LogUploader.Cancel();
  505. if (Globals.LogDownloader != null)
  506. Globals.LogDownloader.Cancel();
  507. if (Globals.LogUploader != null)
  508. {
  509. Globals.LogUploader.Join();
  510. Globals.LogUploader = null;
  511. }
  512. if (Globals.LogDownloader != null)
  513. {
  514. Globals.LogDownloader.Join();
  515. Globals.LogDownloader = null;
  516. }
  517. Globals.DB.Close();
  518. Globals.DB.Dispose();
  519. Globals.DB = null;
  520. Globals.SessionData = null;
  521. Globals.SessionDatabasePassword = null;
  522. Globals.SessionVoucherKey = null;
  523. Globals.SessionMode = SessionModes.Invalid;
  524. Globals.ProductCatalogue = null;
  525. Globals.MaxPrinter.Close();
  526. UserLoginForm LoginForm = (UserLoginForm)Application.OpenForms["UserLoginForm"];
  527. LoginForm.Show();
  528. }
  529. }
  530. public static int GetLastSyncedLogID() {
  531. VendorEventsMetaData MetaData = new VendorEventsMetaData();
  532. bool MetaDataResult = RESTRequest<VendorEventsMetaData>(ref MetaData, "/api/vendorevents/meta");
  533. if (MetaData.LastVendorEventRemoteId == null) {
  534. return 0;
  535. }
  536. return (int)MetaData.LastVendorEventRemoteId;
  537. }
  538. public static void CheckLogSynchronisation(DBHelper db)
  539. {
  540. if (!GetSavedParameterAsBoolean(db, "LoggingInitialised"))
  541. {
  542. var lastSyncedLogId = GetLastSyncedLogID();
  543. // If we already have logs then there is no need to synchronise.
  544. // NOTE: logs are synchronised in reverse from SyncBackwardsFromLogId downwards
  545. if ((long)db.ExecuteScalar("SELECT COUNT(*) FROM Logs") == 0)
  546. {
  547. UpdateSavedParameter(db, "SyncBackwardsFromLogId", lastSyncedLogId);
  548. }
  549. // If there are logs on the server then ensure that the log autoincrement value is set to the last
  550. // log ID received by the server so new local logs don't clash with existing log entries on the server.
  551. if (lastSyncedLogId != 0)
  552. {
  553. lock (db.WriteLock)
  554. {
  555. using (var trans = db.BeginTransaction())
  556. {
  557. using (var command = db.CreateCommand(trans, new SQLiteParameter("@Id", lastSyncedLogId)))
  558. {
  559. command.CommandText = "UPDATE sqlite_sequence SET seq=CASE WHEN IFNULL(seq, 0)<@Id THEN @Id ELSE seq END WHERE name='Logs'";
  560. if (command.ExecuteNonQuery() == 0)
  561. {
  562. command.CommandText = "INSERT INTO sqlite_sequence (name, seq) VALUES ('Logs', @id)";
  563. command.ExecuteNonQuery();
  564. }
  565. }
  566. trans.Commit();
  567. }
  568. }
  569. }
  570. UpdateSavedParameter(db, "LoggingInitialised", true);
  571. }
  572. }
  573. public static void TriggerLogDownload()
  574. {
  575. if (Globals.LogDownloader == null)
  576. {
  577. Globals.LogDownloader = new LogDownloader(Globals.DB);
  578. Globals.LogDownloader.Start();
  579. }
  580. }
  581. public static void TriggerLogUpload()
  582. {
  583. if (Globals.LogUploader == null)
  584. {
  585. Globals.LogUploader = new LogUploader(Globals.DB);
  586. Globals.LogUploader.Start();
  587. }
  588. else
  589. {
  590. Globals.LogUploader.TriggerUpload();
  591. }
  592. }
  593. public static void LogBulkEvents(DBHelper db, List<EventLog> EventLogs)
  594. {
  595. string Sql = "INSERT INTO Logs (UserId, VoucherId, EventDate, EventType, Retry) VALUES (@userid, @voucherid, @eventdate, @eventtype, @retry)";
  596. lock (db.WriteLock)
  597. {
  598. using (var trans = db.BeginTransaction())
  599. {
  600. using (var Command = db.CreateCommand(Sql, trans))
  601. {
  602. foreach (var EventLog in EventLogs)
  603. {
  604. Command.Parameters.Clear();
  605. Command.Parameters.AddWithValue("@userid", Globals.SessionData.Credentials.Payload.User.Id);
  606. Command.Parameters.AddWithValue("@voucherid", EventLog.VoucherId);
  607. Command.Parameters.AddWithValue("@eventdate", DateTime.UtcNow);
  608. Command.Parameters.AddWithValue("@eventtype", EventLog.EventType);
  609. Command.Parameters.AddWithValue("@retry", EventLog.Retry);
  610. Command.ExecuteNonQuery();
  611. }
  612. }
  613. trans.Commit();
  614. }
  615. }
  616. if (Globals.SessionMode == SessionModes.Online)
  617. {
  618. TriggerLogUpload();
  619. }
  620. }
  621. public static void LogEvent(DBHelper db, VendorEvent.VendorEventType EventType, int? VoucherId = null, bool Retry = false)
  622. {
  623. db.ExecuteNonQuery(
  624. "INSERT INTO Logs (UserId, VoucherId, EventDate, EventType, Retry) " +
  625. "VALUES (@userid, @voucherid, @eventdate, @eventtype, @retry)",
  626. new SQLiteParameter("@userid", Globals.SessionData.Credentials.Payload.User.Id),
  627. new SQLiteParameter("@voucherid", VoucherId),
  628. new SQLiteParameter("@eventdate", DateTime.UtcNow),
  629. new SQLiteParameter("@eventtype", EventType),
  630. new SQLiteParameter("@retry", Retry));
  631. if (Globals.SessionMode == SessionModes.Online)
  632. {
  633. TriggerLogUpload();
  634. }
  635. }
  636. private static void SaveCurrentUserUsage(DBHelper db) {
  637. db.ExecuteNonQuery(
  638. "DELETE FROM AccessControlTracking WHERE UserID = @userid",
  639. new SQLiteParameter("@userid", Globals.SessionData.Credentials.Payload.User.Id));
  640. db.ExecuteNonQuery(
  641. "INSERT INTO AccessControlTracking (UserID, Date, Permission, CurrentUsage) VALUES " +
  642. "(@userid, datetime('now'), 'OfflinePrint', @offlineprintvalue)," +
  643. "(@userid, datetime('now'), 'OfflineReprint', @offlinereprintvalue)," +
  644. "(@userid, datetime('now'), 'OnlinePrint', @onlineprintvalue)," +
  645. "(@userid, datetime('now'), 'OnlineReprint', @onlinereprintvalue)," +
  646. "(@userid, datetime('now'), 'BulkOrder', @bulkorder)," +
  647. "(@userid, datetime('now'), 'BulkExport', @bulkexport)",
  648. new SQLiteParameter("@userid", Globals.SessionData.Credentials.Payload.User.Id),
  649. new SQLiteParameter("@offlineprintvalue", Globals.UserCurrentUsage.OfflinePrintValue),
  650. new SQLiteParameter("@offlinereprintvalue", Globals.UserCurrentUsage.OfflineReprintValue),
  651. new SQLiteParameter("@onlineprintvalue", Globals.UserCurrentUsage.OnlinePrintValue),
  652. new SQLiteParameter("@onlinereprintvalue", Globals.UserCurrentUsage.OnlineReprintValue),
  653. new SQLiteParameter("@bulkorder", Globals.UserCurrentUsage.BulkOrderValue),
  654. new SQLiteParameter("@bulkexport", Globals.UserCurrentUsage.BulkExportValue));
  655. }
  656. public static void AddUserUsage(DBHelper db, UserLimits.UserLimitTypes UserLimitType, decimal Value) {
  657. DateTime ServerDate = Globals.SessionData.Credentials.Payload.Date;
  658. DateTime Today = DateTime.Now;
  659. if (ServerDate.Date != Today.Date) //prevent system time tampering
  660. {
  661. MessageBox.Show("Date mismatch detected. Logging out.", "Server Date Mismatch", MessageBoxButtons.OK, MessageBoxIcon.Error);
  662. Logout();
  663. return;
  664. }
  665. switch (UserLimitType) {
  666. case UserLimits.UserLimitTypes.OfflinePrint:
  667. Globals.UserCurrentUsage.OfflinePrintValue += Value;
  668. break;
  669. case UserLimits.UserLimitTypes.OfflineReprint:
  670. Globals.UserCurrentUsage.OfflineReprintValue += Value;
  671. break;
  672. case UserLimits.UserLimitTypes.OnlinePrint:
  673. Globals.UserCurrentUsage.OnlinePrintValue += Value;
  674. break;
  675. case UserLimits.UserLimitTypes.OnlineReprint:
  676. Globals.UserCurrentUsage.OnlineReprintValue += Value;
  677. break;
  678. case UserLimits.UserLimitTypes.BulkExport:
  679. Globals.UserCurrentUsage.BulkExportValue += Value;
  680. break;
  681. case UserLimits.UserLimitTypes.BulkOrder:
  682. Globals.UserCurrentUsage.BulkOrderValue += Value;
  683. break;
  684. }
  685. //Immediately write to db to avoid cheating
  686. SaveCurrentUserUsage(db);
  687. }
  688. public static Boolean IsValueWithinRemainingUserLimit(UserLimits.UserLimitTypes UserLimitType, decimal Value) {
  689. decimal RemainingUserLimit = CheckRemainingUserLimit(UserLimitType);
  690. if (RemainingUserLimit == -1 || RemainingUserLimit >= Value) {
  691. return true;
  692. }
  693. return false;
  694. }
  695. public static decimal CheckRemainingUserLimit(UserLimits.UserLimitTypes UserLimitType) {
  696. if (Globals.SessionData.Credentials.Payload.User.Level == (int)UserLevel.Administrator)
  697. {
  698. return -1; //Signifies unlimited
  699. }
  700. switch (UserLimitType)
  701. {
  702. case UserLimits.UserLimitTypes.OfflinePrint:
  703. if (Globals.SessionData.Credentials.Payload.User.CanPrintOffline)
  704. {
  705. if (Globals.SessionData.Credentials.Payload.User.OfflinePrintValue > 0) //Unlimited from server is signified by 0, but remaining limit at 0 means limit reached, so change unlimited to -1 for internal use
  706. {
  707. return Globals.SessionData.Credentials.Payload.User.OfflinePrintValue - Globals.UserCurrentUsage.OfflinePrintValue;
  708. }
  709. else
  710. {
  711. return -1; //Signifies unlimited
  712. }
  713. }
  714. else
  715. {
  716. return 0;
  717. }
  718. case UserLimits.UserLimitTypes.OfflineReprint:
  719. if (Globals.SessionData.Credentials.Payload.User.CanReprintOffline)
  720. {
  721. if (Globals.SessionData.Credentials.Payload.User.OfflineReprintValue > 0) //Unlimited from server is signified by 0, but remaining limit at 0 means limit reached, so change unlimited to -1 for internal use
  722. {
  723. return Globals.SessionData.Credentials.Payload.User.OfflineReprintValue - Globals.UserCurrentUsage.OfflineReprintValue;
  724. }
  725. else
  726. {
  727. return -1; //Signifies unlimited
  728. }
  729. }
  730. else
  731. {
  732. return 0;
  733. }
  734. case UserLimits.UserLimitTypes.OnlinePrint:
  735. if (Globals.SessionData.Credentials.Payload.User.CanPrintOnline)
  736. {
  737. if (Globals.SessionData.Credentials.Payload.User.OnlinePrintValue > 0) //Unlimited from server is signified by 0, but remaining limit at 0 means limit reached, so change unlimited to -1 for internal use
  738. {
  739. return Globals.SessionData.Credentials.Payload.User.OnlinePrintValue - Globals.UserCurrentUsage.OnlinePrintValue;
  740. }
  741. else
  742. {
  743. return -1; //Signifies unlimited
  744. }
  745. }
  746. else
  747. {
  748. return 0;
  749. }
  750. case UserLimits.UserLimitTypes.OnlineReprint:
  751. if (Globals.SessionData.Credentials.Payload.User.CanReprintOnline)
  752. {
  753. if (Globals.SessionData.Credentials.Payload.User.OnlineReprintValue > 0) //Unlimited from server is signified by 0, but remaining limit at 0 means limit reached, so change unlimited to -1 for internal use
  754. {
  755. return Globals.SessionData.Credentials.Payload.User.OnlineReprintValue - Globals.UserCurrentUsage.OnlineReprintValue;
  756. }
  757. else
  758. {
  759. return -1; //Signifies unlimited
  760. }
  761. }
  762. else
  763. {
  764. return 0;
  765. }
  766. case UserLimits.UserLimitTypes.BulkExport:
  767. if (Globals.SessionData.Credentials.Payload.User.BulkExport)
  768. {
  769. if (Globals.SessionData.Credentials.Payload.User.BulkExportMaxValue > 0) //Unlimited from server is signified by 0, but remaining limit at 0 means limit reached, so change unlimited to -1 for internal use
  770. {
  771. return Globals.SessionData.Credentials.Payload.User.BulkExportMaxValue - Globals.UserCurrentUsage.BulkExportValue;
  772. }
  773. else
  774. {
  775. return -1; //Signifies unlimited
  776. }
  777. }
  778. else
  779. {
  780. return 0;
  781. }
  782. case UserLimits.UserLimitTypes.BulkOrder:
  783. if (Globals.SessionData.Credentials.Payload.User.BulkOrder)
  784. {
  785. if (Globals.SessionData.Credentials.Payload.User.BulkOrderMaxValue > 0) //Unlimited from server is signified by 0, but remaining limit at 0 means limit reached, so change unlimited to -1 for internal use
  786. {
  787. return Globals.SessionData.Credentials.Payload.User.BulkOrderMaxValue - Globals.UserCurrentUsage.BulkOrderValue;
  788. }
  789. else
  790. {
  791. return -1; //Signifies unlimited
  792. }
  793. }
  794. else
  795. {
  796. return 0;
  797. }
  798. }
  799. return 0;
  800. }
  801. private static int GetVoucherPrintCountFromLogs(DBHelper db, int VoucherId)
  802. {
  803. return int.Parse(db.ExecuteScalar(
  804. "SELECT COUNT(*) FROM Logs WHERE VoucherId=@voucherid AND EventType=@eventtype",
  805. new SQLiteParameter("@voucherid", VoucherId),
  806. new SQLiteParameter("@eventtype", (int)VendorEvent.VendorEventType.PrintVoucher)).ToString());
  807. }
  808. private static DateTime GetVoucherFirstPrintDateFromLogs(DBHelper db, int VoucherId)
  809. {
  810. return (DateTime)db.ExecuteScalar(
  811. "SELECT MIN(EventDate) FROM Logs WHERE VoucherId=@voucherid AND EventType=@eventtype",
  812. new SQLiteParameter("@voucherid", VoucherId),
  813. new SQLiteParameter("@eventtype", (int)VendorEvent.VendorEventType.PrintVoucher));
  814. }
  815. private static decimal GetVoucherFaceValue(DBHelper db, int VoucherId)
  816. {
  817. return (decimal)db.ExecuteScalar(
  818. "SELECT b.FaceValue FROM Voucher v LEFT JOIN Batch b on v.BatchId = b.Id WHERE v.Id=@voucherid",
  819. new SQLiteParameter("@voucherid", VoucherId));
  820. }
  821. private static UserLimits CalculateUsageFromLogs(DBHelper db) {
  822. UserLimits CalculateUsage = new UserLimits();
  823. CalculateUsage.OfflinePrintValue = 0;
  824. CalculateUsage.OfflineReprintValue = 0;
  825. CalculateUsage.OnlinePrintValue = 0;
  826. CalculateUsage.OnlineReprintValue = 0;
  827. var CurrentDate = DateTime.Today.Date;
  828. CultureInfo IVC = CultureInfo.InvariantCulture;
  829. string Sql = "SELECT VoucherId FROM Logs WHERE UserId=@userid AND EventDate BETWEEN @eventdatea AND @eventdateb AND EventType=@eventtype";
  830. using (var Command = db.CreateCommand(Sql,
  831. new SQLiteParameter("@userid", Globals.SessionData.Credentials.Payload.User.Id),
  832. new SQLiteParameter("@eventdatea", CurrentDate.ToString("yyyy-MM-dd 00:00:00", IVC)),
  833. new SQLiteParameter("@eventdateb", CurrentDate.ToString("yyyy-MM-dd 23:59:59", IVC)),
  834. new SQLiteParameter("@eventtype", (int)VendorEvent.VendorEventType.PrintVoucher)))
  835. {
  836. int VoucherId;
  837. decimal VoucherFaceValue;
  838. using (SQLiteDataReader read = Command.ExecuteReader())
  839. {
  840. while (read.Read())
  841. {
  842. VoucherId = read.GetInt32(0);
  843. VoucherFaceValue = GetVoucherFaceValue(db, VoucherId);
  844. //Don't distinguish between online and offline usage for log recalculations
  845. CalculateUsage.OfflinePrintValue += VoucherFaceValue;
  846. CalculateUsage.OnlinePrintValue += VoucherFaceValue;
  847. if (GetVoucherPrintCountFromLogs(db, VoucherId) > 1)
  848. { //For reprinted vouchers just assume all prints were reprints - log recalculations should be stricter to prevent cheating
  849. CalculateUsage.OfflineReprintValue += VoucherFaceValue;
  850. CalculateUsage.OnlineReprintValue += VoucherFaceValue;
  851. }
  852. }
  853. }
  854. }
  855. return CalculateUsage;
  856. }
  857. public static void InitialiseUserLimits(DBHelper db)
  858. {
  859. Globals.UserCurrentUsage = new UserLimits();
  860. UserLimits CurrentUserLimits = new UserLimits();
  861. int Result = int.Parse(db.ExecuteScalar(
  862. "SELECT COUNT(*) FROM AccessControlTracking WHERE UserID = @userid",
  863. new SQLiteParameter("@userid", Globals.SessionData.Credentials.Payload.User.Id)).ToString());
  864. object ResultDate = db.ExecuteScalar(
  865. "SELECT Date FROM AccessControlTracking WHERE UserID = @userid",
  866. new SQLiteParameter("@userid", Globals.SessionData.Credentials.Payload.User.Id));
  867. bool DatePassed = false;
  868. if (ResultDate != null) {
  869. DateTime DateResult = DateTime.Parse(ResultDate.ToString());
  870. if (DateResult.Date < Globals.SessionData.Credentials.Payload.Date.Date) {
  871. DatePassed = true;
  872. }
  873. }
  874. if (DatePassed)
  875. { //Fresh day since last db write or data missing - fetch usage from logs incase of tampering
  876. Globals.UserCurrentUsage = CalculateUsageFromLogs(db);
  877. SaveCurrentUserUsage(db); //Immediately write calculated usage to DB
  878. }
  879. else
  880. {
  881. if (Result == 6)
  882. {
  883. var Sql = "SELECT UserID, Permission, CurrentUsage FROM AccessControlTracking WHERE UserID = @userid";
  884. using (var Command = db.CreateCommand(Sql,
  885. new SQLiteParameter("@userid", Globals.SessionData.Credentials.Payload.User.Id)))
  886. {
  887. using (SQLiteDataReader read = Command.ExecuteReader())
  888. {
  889. while (read.Read())
  890. {
  891. switch (read["Permission"])
  892. {
  893. case "OfflinePrint":
  894. Globals.UserCurrentUsage.OfflinePrintValue = (decimal)read["CurrentUsage"];
  895. break;
  896. case "OfflineReprint":
  897. Globals.UserCurrentUsage.OfflineReprintValue = (decimal)read["CurrentUsage"];
  898. break;
  899. case "OnlinePrint":
  900. Globals.UserCurrentUsage.OnlinePrintValue = (decimal)read["CurrentUsage"];
  901. break;
  902. case "OnlineReprint":
  903. Globals.UserCurrentUsage.OnlineReprintValue = (decimal)read["CurrentUsage"];
  904. break;
  905. case "BulkOrder":
  906. Globals.UserCurrentUsage.BulkOrderValue = (decimal)read["CurrentUsage"];
  907. break;
  908. case "BulkExport":
  909. Globals.UserCurrentUsage.BulkExportValue = (decimal)read["CurrentUsage"];
  910. break;
  911. }
  912. }
  913. }
  914. }
  915. }
  916. else
  917. {
  918. Globals.UserCurrentUsage = CalculateUsageFromLogs(db);
  919. SaveCurrentUserUsage(db); //Immediately write calculated usage to DB
  920. }
  921. }
  922. }
  923. public static int GetNumberOfUnprintedVouchersInRange(DBHelper db, int StartSeqNo, int EndSeqNo, int BatchId) {
  924. int UnprintedVouchers = 0;
  925. int VoucherId;
  926. string Sql = "SELECT Id FROM Voucher WHERE SequenceNumber=@sequencenumber AND BatchId=@batchid";
  927. using (var Command = db.CreateCommand(Sql))
  928. {
  929. for (int SeqNo = StartSeqNo; SeqNo <= EndSeqNo; SeqNo++)
  930. {
  931. Command.Parameters.Clear();
  932. Command.Parameters.AddWithValue("@sequencenumber", SeqNo);
  933. Command.Parameters.AddWithValue("@batchid", BatchId);
  934. VoucherId = int.Parse(Command.ExecuteScalar().ToString());
  935. int NumPrintedVouchers = GetVoucherPrintCountFromLogs(db, VoucherId);
  936. if (NumPrintedVouchers == 0)
  937. {
  938. UnprintedVouchers++;
  939. }
  940. }
  941. }
  942. return UnprintedVouchers;
  943. }
  944. }
  945. }