ReceiptMYOBPoster.cs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  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.UID = Guid.Empty;
  82. // Account =
  83. if(customers.TryGetValue(receipt.CustomerLink.ID, out var customer))
  84. {
  85. if(!CustomerMYOBPoster.MapCustomer(ConnectionData, customer).Get(out var customerID, out var error))
  86. {
  87. CoreUtils.LogException("", error, $"Error while posting receipt {receipt.ID}");
  88. results.AddFailed(receipt, error.Message);
  89. continue;
  90. }
  91. myobReceipt.Customer.UID = customerID;
  92. }
  93. // myobReceipt.ReceiptNumber =
  94. myobReceipt.Date = receipt.Date;
  95. myobReceipt.AmountReceived = (decimal)receipt.Total;
  96. // myobReceipt.AmountReceivedForeign = 0;
  97. // myobReceipt.PaymentMethod =
  98. myobReceipt.Memo = receipt.Notes.Truncate(255);
  99. if(invoices.TryGetValue(receipt.ID, out var receiptInvoices))
  100. {
  101. var myobInvoices = new CustomerPaymentLine[receiptInvoices.Length];
  102. string? failed = null;
  103. for(int i = 0; i < receiptInvoices.Length; ++i)
  104. {
  105. var invoice = receiptInvoices[i];
  106. var line = new CustomerPaymentLine();
  107. line.RowID = i + 1;
  108. line.Number = invoice.InvoiceLink.Number.ToString().Truncate(8);
  109. line.AmountApplied = (decimal)invoice.Amount;
  110. line.Type = CustomerPaymentLineType.Invoice;
  111. if(Guid.TryParse(invoice.InvoiceLink.PostedReference, out var myobInvoiceID))
  112. {
  113. line.UID = myobInvoiceID;
  114. }
  115. else
  116. {
  117. failed = $"Invoice {invoice.InvoiceLink.Number} hasn't been posted yet; it must be posted before this receipt can be.";
  118. break;
  119. }
  120. myobInvoices[i] = line;
  121. }
  122. if(failed is not null)
  123. {
  124. results.AddFailed(receipt, failed);
  125. continue;
  126. }
  127. myobReceipt.Invoices = myobInvoices;
  128. }
  129. else
  130. {
  131. myobReceipt.Invoices = [];
  132. }
  133. try
  134. {
  135. var result = service.UpdateEx(ConnectionData.CompanyFile, myobReceipt, ConnectionData.CompanyFileCredentials);
  136. receipt.PostedReference = result.UID.ToString();
  137. results.AddSuccess(receipt);
  138. }
  139. catch(Exception e)
  140. {
  141. CoreUtils.LogException("", e, $"Error while posting receipt {receipt.ID}");
  142. results.AddFailed(receipt, e.Message);
  143. }
  144. }
  145. return results;
  146. }
  147. }