ReceiptMYOBPoster.cs 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. using Comal.Classes;
  2. using InABox.Core;
  3. using InABox.Poster.MYOB;
  4. using MYOB.AccountRight.SDK.Contracts.Version2.Sale;
  5. using MYOB.AccountRight.SDK.Services.Sale;
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Text;
  10. using System.Threading.Tasks;
  11. using System.Windows.Markup.Localizer;
  12. using Invoice = Comal.Classes.Invoice;
  13. using MYOBReceipt = MYOB.AccountRight.SDK.Contracts.Version2.Sale.CustomerPayment;
  14. namespace PRS.Shared.Posters.MYOB;
  15. public class ReceiptMYOBPosterSettings : MYOBPosterSettings
  16. {
  17. }
  18. public class ReceiptMYOBPoster : IMYOBPoster<Receipt, ReceiptMYOBPosterSettings>
  19. {
  20. public MYOBConnectionData ConnectionData { get; set; }
  21. public ReceiptMYOBPosterSettings Settings { get; set; }
  22. public MYOBGlobalPosterSettings GlobalSettings { get; set; }
  23. public bool BeforePost(IDataModel<Receipt> model)
  24. {
  25. foreach(var (_, table) in model.ModelTables)
  26. {
  27. table.ShouldLoad = false;
  28. }
  29. model.SetShouldLoad<Receipt>(true);
  30. model.SetColumns<Receipt>(RequiredReceiptColumns());
  31. model.SetIsDefault<InvoiceReceipt>(true, alias: "Receipt_InvoiceReceipt");
  32. model.SetColumns<InvoiceReceipt>(RequiredInvoiceReceiptColumns(), alias: "Receipt_InvoiceReceipt");
  33. model.SetIsDefault<Customer>(true, alias: "Receipt_Customer");
  34. model.SetColumns<Customer>(RequiredCustomerColumns(), alias: "Receipt_Customer");
  35. return true;
  36. }
  37. private static Columns<Receipt> RequiredReceiptColumns()
  38. {
  39. return Columns.None<Receipt>()
  40. .Add(x => x.ID)
  41. .Add(x => x.PostedReference)
  42. .Add(x => x.PostedStatus)
  43. .Add(x => x.CustomerLink.ID)
  44. .Add(x => x.Date)
  45. .Add(x => x.Total)
  46. .Add(x => x.Notes);
  47. }
  48. private static Columns<InvoiceReceipt> RequiredInvoiceReceiptColumns()
  49. {
  50. return Columns.None<InvoiceReceipt>()
  51. .Add(x => x.ID)
  52. .Add(x => x.Amount)
  53. .Add(x => x.ReceiptLink.ID)
  54. .Add(x => x.InvoiceLink.Number)
  55. .Add(x => x.InvoiceLink.PostedReference);
  56. }
  57. private static Columns<Customer> RequiredCustomerColumns()
  58. {
  59. return CustomerMYOBPoster.RequiredColumns();
  60. }
  61. public IPostResult<Receipt> Process(IDataModel<Receipt> model)
  62. {
  63. // https://developer.myob.com/api/myob-business-api/v2/sale/customerpayment/
  64. var results = new PostResult<Receipt>();
  65. var service = new CustomerPaymentService(ConnectionData.Configuration, null, ConnectionData.AuthKey);
  66. var receipts = model.GetTable<Receipt>().ToArray<Receipt>();
  67. var customers = model.GetTable<Customer>("Receipt_Customer")
  68. .ToObjects<Customer>().ToDictionary(x => x.ID);
  69. var invoices = model.GetTable<InvoiceReceipt>("Receipt_InvoiceReceipt")
  70. .ToObjects<InvoiceReceipt>().GroupBy(x => x.ReceiptLink.ID).ToDictionary(x => x.Key, x => x.ToArray());
  71. foreach(var receipt in receipts)
  72. {
  73. // For receipts, always create a new one, so we cannot posted already posted stuff.
  74. if(receipt.PostedStatus == PostedStatus.Posted)
  75. {
  76. continue;
  77. }
  78. var myobReceipt = new MYOBReceipt();
  79. myobReceipt.DepositTo = DepositTo.UndepositedFunds;
  80. // Setting this to null right now.
  81. myobReceipt.Account ??= new();
  82. myobReceipt.Account.UID = Guid.Empty;
  83. // Account =
  84. if(customers.TryGetValue(receipt.CustomerLink.ID, out var customer))
  85. {
  86. if(!CustomerMYOBPoster.MapCustomer(ConnectionData, customer).Get(out var customerID, out var error))
  87. {
  88. CoreUtils.LogException("", error, $"Error while posting receipt {receipt.ID}");
  89. results.AddFailed(receipt, error.Message);
  90. continue;
  91. }
  92. myobReceipt.Customer ??= new();
  93. myobReceipt.Customer.UID = customerID;
  94. }
  95. // myobReceipt.ReceiptNumber =
  96. myobReceipt.Date = receipt.Date;
  97. myobReceipt.AmountReceived = (decimal)receipt.Total;
  98. // myobReceipt.AmountReceivedForeign = 0;
  99. // myobReceipt.PaymentMethod =
  100. myobReceipt.Memo = receipt.Notes.Truncate(255);
  101. if(invoices.TryGetValue(receipt.ID, out var receiptInvoices))
  102. {
  103. var myobInvoices = new CustomerPaymentLine[receiptInvoices.Length];
  104. string? failed = null;
  105. for(int i = 0; i < receiptInvoices.Length; ++i)
  106. {
  107. var invoice = receiptInvoices[i];
  108. var line = new CustomerPaymentLine();
  109. line.RowID = i + 1;
  110. line.Number = invoice.InvoiceLink.Number.ToString().Truncate(8);
  111. line.AmountApplied = (decimal)invoice.Amount;
  112. line.Type = CustomerPaymentLineType.Invoice;
  113. if(Guid.TryParse(invoice.InvoiceLink.PostedReference, out var myobInvoiceID))
  114. {
  115. line.UID = myobInvoiceID;
  116. }
  117. else
  118. {
  119. failed = $"Invoice {invoice.InvoiceLink.Number} hasn't been posted yet; it must be posted before this receipt can be.";
  120. break;
  121. }
  122. myobInvoices[i] = line;
  123. }
  124. if(failed is not null)
  125. {
  126. results.AddFailed(receipt, failed);
  127. continue;
  128. }
  129. myobReceipt.Invoices = myobInvoices;
  130. }
  131. else
  132. {
  133. myobReceipt.Invoices = [];
  134. }
  135. if(service.Save(ConnectionData, myobReceipt).Get(out var result, out var e))
  136. {
  137. receipt.PostedReference = result.UID.ToString();
  138. results.AddSuccess(receipt);
  139. }
  140. else
  141. {
  142. CoreUtils.LogException("", e, $"Error while posting receipt {receipt.ID}");
  143. results.AddFailed(receipt, e.Message);
  144. }
  145. }
  146. return results;
  147. }
  148. }
  149. public class ReceiptMYOBPosterEngine<T> : MYOBPosterEngine<Receipt, ReceiptMYOBPosterSettings> { }