taxtank-core 0.32.0 → 0.32.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (133) hide show
  1. package/esm2022/lib/collections/subscription/service-subscription.collection.mjs +3 -3
  2. package/esm2022/lib/collections/vehicle/best-vehicle-logbook.collection.mjs +18 -14
  3. package/esm2022/lib/collections/vehicle/vehicle-logbook.collection.mjs +10 -5
  4. package/esm2022/lib/db/Models/firm/client-invite.mjs +3 -3
  5. package/esm2022/lib/db/Models/subscription/service-product.mjs +1 -1
  6. package/esm2022/lib/db/Models/subscription/service-subscription.mjs +1 -1
  7. package/esm2022/lib/services/account-setup/account-setup.service.mjs +2 -2
  8. package/esm2022/lib/services/http/bank/bank-connection/bank-connection.service.mjs +4 -6
  9. package/esm2022/lib/services/http/chat/chat.service.mjs +4 -5
  10. package/esm2022/lib/services/http/chat/message.service.mjs +4 -5
  11. package/esm2022/lib/services/http/firm/client-invite/client-invite-messages.enum.mjs +5 -2
  12. package/esm2022/lib/services/http/firm/client-invite/client-invite.service.mjs +56 -63
  13. package/esm2022/lib/services/http/firm/client-movement/client-movement.service.mjs +32 -32
  14. package/esm2022/lib/services/http/rest/rest.service.mjs +23 -2
  15. package/esm2022/lib/services/http/subscription/service-payment-method/service-payment-method.service.mjs +4 -5
  16. package/esm2022/lib/services/http/subscription/service-subscription/subscription.service.mjs +4 -7
  17. package/fesm2022/taxtank-core.mjs +507 -488
  18. package/fesm2022/taxtank-core.mjs.map +1 -1
  19. package/lib/collections/vehicle/best-vehicle-logbook.collection.d.ts +2 -1
  20. package/lib/collections/vehicle/vehicle-logbook.collection.d.ts +3 -2
  21. package/lib/db/Models/firm/client-invite.d.ts +2 -2
  22. package/lib/db/Models/subscription/service-product.d.ts +2 -1
  23. package/lib/db/Models/subscription/service-subscription.d.ts +2 -0
  24. package/lib/services/http/bank/bank-connection/bank-connection.service.d.ts +1 -3
  25. package/lib/services/http/chat/chat.service.d.ts +2 -3
  26. package/lib/services/http/chat/message.service.d.ts +2 -3
  27. package/lib/services/http/firm/client-invite/client-invite-messages.enum.d.ts +5 -2
  28. package/lib/services/http/firm/client-invite/client-invite.service.d.ts +15 -14
  29. package/lib/services/http/firm/client-movement/client-movement.service.d.ts +19 -13
  30. package/lib/services/http/rest/rest.service.d.ts +13 -1
  31. package/lib/services/http/subscription/service-payment-method/service-payment-method.service.d.ts +2 -3
  32. package/lib/services/http/subscription/service-subscription/subscription.service.d.ts +2 -5
  33. package/package.json +1 -1
  34. package/lib/db/Models/address.d.ts.map +0 -1
  35. package/lib/db/Models/bank/bank-account-balance.d.ts.map +0 -1
  36. package/lib/db/Models/bank/bank-account-property.d.ts.map +0 -1
  37. package/lib/db/Models/bank/bank-account.d.ts.map +0 -1
  38. package/lib/db/Models/bank/bank-connection.d.ts.map +0 -1
  39. package/lib/db/Models/bank/bank-transaction-import.d.ts.map +0 -1
  40. package/lib/db/Models/bank/bank-transaction.d.ts.map +0 -1
  41. package/lib/db/Models/bank/bank.d.ts.map +0 -1
  42. package/lib/db/Models/bank/basiq-job.d.ts.map +0 -1
  43. package/lib/db/Models/chart-accounts/chart-accounts-depreciation.d.ts.map +0 -1
  44. package/lib/db/Models/chart-accounts/chart-accounts-heading.d.ts.map +0 -1
  45. package/lib/db/Models/chart-accounts/chart-accounts-meta-field.d.ts.map +0 -1
  46. package/lib/db/Models/chart-accounts/chart-accounts-value.d.ts.map +0 -1
  47. package/lib/db/Models/chart-accounts/chart-accounts.d.ts.map +0 -1
  48. package/lib/db/Models/client/client-details.d.ts.map +0 -1
  49. package/lib/db/Models/client/client-income-types.d.ts.map +0 -1
  50. package/lib/db/Models/client/occupation.d.ts.map +0 -1
  51. package/lib/db/Models/country.d.ts.map +0 -1
  52. package/lib/db/Models/depreciation/depreciation-capital-project.d.ts.map +0 -1
  53. package/lib/db/Models/depreciation/depreciation-forecast.d.ts.map +0 -1
  54. package/lib/db/Models/depreciation/depreciation.d.ts.map +0 -1
  55. package/lib/db/Models/document/document-base.d.ts.map +0 -1
  56. package/lib/db/Models/document/document-folder.d.ts.map +0 -1
  57. package/lib/db/Models/document/document.d.ts.map +0 -1
  58. package/lib/db/Models/document/property-document.d.ts.map +0 -1
  59. package/lib/db/Models/file.d.ts.map +0 -1
  60. package/lib/db/Models/firm/chat.d.ts.map +0 -1
  61. package/lib/db/Models/firm/client-invite.d.ts.map +0 -1
  62. package/lib/db/Models/firm/client-movement.d.ts.map +0 -1
  63. package/lib/db/Models/firm/employee-invite.d.ts.map +0 -1
  64. package/lib/db/Models/firm/firm-invite.d.ts.map +0 -1
  65. package/lib/db/Models/firm/firm.d.ts.map +0 -1
  66. package/lib/db/Models/firm/message-document.d.ts.map +0 -1
  67. package/lib/db/Models/firm/message.d.ts.map +0 -1
  68. package/lib/db/Models/holding/holding-sale.d.ts.map +0 -1
  69. package/lib/db/Models/incomeSource/income-source-forecast.d.ts.map +0 -1
  70. package/lib/db/Models/incomeSource/income-source-type.d.ts.map +0 -1
  71. package/lib/db/Models/incomeSource/income-source.d.ts.map +0 -1
  72. package/lib/db/Models/incomeSource/salary-forecast.d.ts.map +0 -1
  73. package/lib/db/Models/loan/borrowing-expense-loan.d.ts.map +0 -1
  74. package/lib/db/Models/loan/borrowing-expense.d.ts.map +0 -1
  75. package/lib/db/Models/loan/loan-payout.d.ts.map +0 -1
  76. package/lib/db/Models/loan/loan.d.ts.map +0 -1
  77. package/lib/db/Models/phone.d.ts.map +0 -1
  78. package/lib/db/Models/property/borrowing-report.d.ts.map +0 -1
  79. package/lib/db/Models/property/property-category-movement.d.ts.map +0 -1
  80. package/lib/db/Models/property/property-category.d.ts.map +0 -1
  81. package/lib/db/Models/property/property-forecast.d.ts.map +0 -1
  82. package/lib/db/Models/property/property-sale/property-sale-tax-exemption-meta-field.d.ts.map +0 -1
  83. package/lib/db/Models/property/property-sale/property-sale.d.ts.map +0 -1
  84. package/lib/db/Models/property/property-sale/tax-exemption-meta-field.d.ts.map +0 -1
  85. package/lib/db/Models/property/property-sale/tax-exemption.d.ts.map +0 -1
  86. package/lib/db/Models/property/property-share.d.ts.map +0 -1
  87. package/lib/db/Models/property/property-subscription.d.ts.map +0 -1
  88. package/lib/db/Models/property/property-valuation.d.ts.map +0 -1
  89. package/lib/db/Models/property/property.d.ts.map +0 -1
  90. package/lib/db/Models/service-notification.d.ts.map +0 -1
  91. package/lib/db/Models/sole/bas-report.d.ts.map +0 -1
  92. package/lib/db/Models/sole/sole-business-activity.d.ts.map +0 -1
  93. package/lib/db/Models/sole/sole-business-allocation.d.ts.map +0 -1
  94. package/lib/db/Models/sole/sole-business-loss-offset-rule.d.ts.map +0 -1
  95. package/lib/db/Models/sole/sole-business-loss.d.ts.map +0 -1
  96. package/lib/db/Models/sole/sole-business.d.ts.map +0 -1
  97. package/lib/db/Models/sole/sole-contact.d.ts.map +0 -1
  98. package/lib/db/Models/sole/sole-depreciation-method.d.ts.map +0 -1
  99. package/lib/db/Models/sole/sole-details.d.ts.map +0 -1
  100. package/lib/db/Models/sole/sole-forecast.d.ts.map +0 -1
  101. package/lib/db/Models/sole/sole-invoice-item.d.ts.map +0 -1
  102. package/lib/db/Models/sole/sole-invoice-template.d.ts.map +0 -1
  103. package/lib/db/Models/sole/sole-invoice.d.ts.map +0 -1
  104. package/lib/db/Models/spare/spare-document-category.d.ts.map +0 -1
  105. package/lib/db/Models/spare/spare-document.d.ts.map +0 -1
  106. package/lib/db/Models/subscription/service-payment-method.d.ts.map +0 -1
  107. package/lib/db/Models/subscription/service-payment.d.ts.map +0 -1
  108. package/lib/db/Models/subscription/service-price.d.ts.map +0 -1
  109. package/lib/db/Models/subscription/service-product.d.ts.map +0 -1
  110. package/lib/db/Models/subscription/service-subscription-item.d.ts.map +0 -1
  111. package/lib/db/Models/subscription/service-subscription.d.ts.map +0 -1
  112. package/lib/db/Models/tax-calculation.d.ts.map +0 -1
  113. package/lib/db/Models/tax-return/tax-return-category.d.ts.map +0 -1
  114. package/lib/db/Models/tax-return/tax-return-item.d.ts.map +0 -1
  115. package/lib/db/Models/tax-return/tax-return.d.ts.map +0 -1
  116. package/lib/db/Models/transaction/allocation-rule-condition.d.ts.map +0 -1
  117. package/lib/db/Models/transaction/allocation-rule-transaction-meta-field.d.ts.map +0 -1
  118. package/lib/db/Models/transaction/allocation-rule-transaction.d.ts.map +0 -1
  119. package/lib/db/Models/transaction/allocation-rule.d.ts.map +0 -1
  120. package/lib/db/Models/transaction/transaction-allocation.d.ts.map +0 -1
  121. package/lib/db/Models/transaction/transaction-meta-field.d.ts.map +0 -1
  122. package/lib/db/Models/transaction/transaction.d.ts.map +0 -1
  123. package/lib/db/Models/user/employee-details.d.ts.map +0 -1
  124. package/lib/db/Models/user/registration-invite.d.ts.map +0 -1
  125. package/lib/db/Models/user/user-event-setting.d.ts.map +0 -1
  126. package/lib/db/Models/user/user-event-type-category.d.ts.map +0 -1
  127. package/lib/db/Models/user/user-event-type.d.ts.map +0 -1
  128. package/lib/db/Models/user/user-event.d.ts.map +0 -1
  129. package/lib/db/Models/vehicle/vehicle-claim-details.d.ts.map +0 -1
  130. package/lib/db/Models/vehicle/vehicle-claim.d.ts.map +0 -1
  131. package/lib/db/Models/vehicle/vehicle-logbook.d.ts.map +0 -1
  132. package/lib/db/Models/vehicle/vehicle.d.ts.map +0 -1
  133. package/taxtank-core.d.ts.map +0 -1
@@ -1,12 +1,15 @@
1
1
  export var ClientInviteMessages;
2
2
  (function (ClientInviteMessages) {
3
3
  ClientInviteMessages["TT_ADVISOR_INVITED"] = "You have Invited TaxTank Support successfully";
4
+ ClientInviteMessages["TT_INVITE_ACCEPT"] = "You successfully accepted the invitation";
4
5
  ClientInviteMessages["REVOKE_TT_ADVISOR_CONFIRM"] = "Are you sure you want to revoke access?";
5
6
  ClientInviteMessages["TT_ADVISOR_REVOKED"] = "Access revoked";
6
- ClientInviteMessages["TT_INVITE_ACCEPT"] = "You successfully accepted the invitation";
7
7
  ClientInviteMessages["SEND_INVITE_MESSAGES_TITLE"] = "TaxTank support is here to help";
8
8
  ClientInviteMessages["SEND_INVITE_MESSAGES_DESCRIPTION"] = "Before inviting TaxTank support be sure to check out our help articles and message our support team.";
9
9
  ClientInviteMessages["TT_ACCOUNTANT_INVITED"] = "Accountant invitation has been sent successfully";
10
10
  ClientInviteMessages["TT_ACCOUNTANT_INVETED_TOOLTIP"] = "Need an accountant? TaxTank Accountants delivers virtual accounting service exclusive to TaxTank subscribers. Experience a fast, seamless, and cost-effective lodgement service that rewards your TaxTank efforts throughout the year. Our team of experienced accountants brings their expertise in tax and TaxTank to the virtual realm. It all starts with a 15-minute consultation that sets the stage for personalised service. With us, you\u2019ll receive the best of both worlds at tax time.";
11
+ ClientInviteMessages["CONFIRM_REJECT"] = "Are you sure you want to reject this invitation?";
12
+ ClientInviteMessages["REJECTED"] = "Invitation rejected";
13
+ ClientInviteMessages["ACCEPTED"] = "Client invitation accepted";
11
14
  })(ClientInviteMessages || (ClientInviteMessages = {}));
12
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xpZW50LWludml0ZS1tZXNzYWdlcy5lbnVtLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdHQtY29yZS9zcmMvbGliL3NlcnZpY2VzL2h0dHAvZmlybS9jbGllbnQtaW52aXRlL2NsaWVudC1pbnZpdGUtbWVzc2FnZXMuZW51bS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQU4sSUFBWSxvQkFZWDtBQVpELFdBQVksb0JBQW9CO0lBQzlCLDRGQUFvRSxDQUFBO0lBQ3BFLDZGQUFxRSxDQUFBO0lBQ3JFLDZEQUFxQyxDQUFBO0lBQ3JDLHFGQUE2RCxDQUFBO0lBQzdELHNGQUE4RCxDQUFBO0lBQzlELGlLQUF5SSxDQUFBO0lBQ3pJLGtHQUEwRSxDQUFBO0lBQzFFLGdpQkFHNkgsQ0FBQTtBQUMvSCxDQUFDLEVBWlcsb0JBQW9CLEtBQXBCLG9CQUFvQixRQVkvQiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBlbnVtIENsaWVudEludml0ZU1lc3NhZ2VzIHtcbiAgVFRfQURWSVNPUl9JTlZJVEVEID0gJ1lvdSBoYXZlIEludml0ZWQgVGF4VGFuayBTdXBwb3J0IHN1Y2Nlc3NmdWxseScsXG4gIFJFVk9LRV9UVF9BRFZJU09SX0NPTkZJUk0gPSAnQXJlIHlvdSBzdXJlIHlvdSB3YW50IHRvIHJldm9rZSBhY2Nlc3M/JyxcbiAgVFRfQURWSVNPUl9SRVZPS0VEID0gJ0FjY2VzcyByZXZva2VkJyxcbiAgVFRfSU5WSVRFX0FDQ0VQVCA9ICdZb3Ugc3VjY2Vzc2Z1bGx5IGFjY2VwdGVkIHRoZSBpbnZpdGF0aW9uJyxcbiAgU0VORF9JTlZJVEVfTUVTU0FHRVNfVElUTEUgPSAnVGF4VGFuayBzdXBwb3J0IGlzIGhlcmUgdG8gaGVscCcsXG4gIFNFTkRfSU5WSVRFX01FU1NBR0VTX0RFU0NSSVBUSU9OID0gJ0JlZm9yZSBpbnZpdGluZyBUYXhUYW5rIHN1cHBvcnQgYmUgc3VyZSB0byBjaGVjayBvdXQgb3VyIGhlbHAgYXJ0aWNsZXMgYW5kIG1lc3NhZ2Ugb3VyIHN1cHBvcnQgdGVhbS4nLFxuICBUVF9BQ0NPVU5UQU5UX0lOVklURUQgPSAnQWNjb3VudGFudCBpbnZpdGF0aW9uIGhhcyBiZWVuIHNlbnQgc3VjY2Vzc2Z1bGx5JyxcbiAgVFRfQUNDT1VOVEFOVF9JTlZFVEVEX1RPT0xUSVAgPSAnTmVlZCBhbiBhY2NvdW50YW50PyBUYXhUYW5rIEFjY291bnRhbnRzIGRlbGl2ZXJzIHZpcnR1YWwgYWNjb3VudGluZyBzZXJ2aWNlIGV4Y2x1c2l2ZSB0byBUYXhUYW5rIHN1YnNjcmliZXJzLiAnICtcbiAgICAgICdFeHBlcmllbmNlIGEgZmFzdCwgc2VhbWxlc3MsIGFuZCBjb3N0LWVmZmVjdGl2ZSBsb2RnZW1lbnQgc2VydmljZSB0aGF0IHJld2FyZHMgeW91ciBUYXhUYW5rIGVmZm9ydHMgdGhyb3VnaG91dCB0aGUgeWVhci4nICtcbiAgICAgICcgT3VyIHRlYW0gb2YgZXhwZXJpZW5jZWQgYWNjb3VudGFudHMgYnJpbmdzIHRoZWlyIGV4cGVydGlzZSBpbiB0YXggYW5kIFRheFRhbmsgdG8gdGhlIHZpcnR1YWwgcmVhbG0uIEl0IGFsbCBzdGFydHMgd2l0aCBhIDE1LW1pbnV0ZSAnICtcbiAgICAgICdjb25zdWx0YXRpb24gdGhhdCBzZXRzIHRoZSBzdGFnZSBmb3IgcGVyc29uYWxpc2VkIHNlcnZpY2UuIFdpdGggdXMsIHlvdeKAmWxsIHJlY2VpdmUgdGhlIGJlc3Qgb2YgYm90aCB3b3JsZHMgYXQgdGF4IHRpbWUuJ1xufVxuIl19
15
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xpZW50LWludml0ZS1tZXNzYWdlcy5lbnVtLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdHQtY29yZS9zcmMvbGliL3NlcnZpY2VzL2h0dHAvZmlybS9jbGllbnQtaW52aXRlL2NsaWVudC1pbnZpdGUtbWVzc2FnZXMuZW51bS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQU4sSUFBWSxvQkFlWDtBQWZELFdBQVksb0JBQW9CO0lBQzlCLDRGQUFvRSxDQUFBO0lBQ3BFLHFGQUE2RCxDQUFBO0lBQzdELDZGQUFxRSxDQUFBO0lBQ3JFLDZEQUFxQyxDQUFBO0lBQ3JDLHNGQUE4RCxDQUFBO0lBQzlELGlLQUF5SSxDQUFBO0lBQ3pJLGtHQUEwRSxDQUFBO0lBQzFFLGdpQkFHeUgsQ0FBQTtJQUN6SCwyRkFBbUUsQ0FBQTtJQUNuRSx3REFBZ0MsQ0FBQTtJQUNoQywrREFBdUMsQ0FBQTtBQUN6QyxDQUFDLEVBZlcsb0JBQW9CLEtBQXBCLG9CQUFvQixRQWUvQiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBlbnVtIENsaWVudEludml0ZU1lc3NhZ2VzIHtcbiAgVFRfQURWSVNPUl9JTlZJVEVEID0gJ1lvdSBoYXZlIEludml0ZWQgVGF4VGFuayBTdXBwb3J0IHN1Y2Nlc3NmdWxseScsXG4gIFRUX0lOVklURV9BQ0NFUFQgPSAnWW91IHN1Y2Nlc3NmdWxseSBhY2NlcHRlZCB0aGUgaW52aXRhdGlvbicsXG4gIFJFVk9LRV9UVF9BRFZJU09SX0NPTkZJUk0gPSAnQXJlIHlvdSBzdXJlIHlvdSB3YW50IHRvIHJldm9rZSBhY2Nlc3M/JyxcbiAgVFRfQURWSVNPUl9SRVZPS0VEID0gJ0FjY2VzcyByZXZva2VkJyxcbiAgU0VORF9JTlZJVEVfTUVTU0FHRVNfVElUTEUgPSAnVGF4VGFuayBzdXBwb3J0IGlzIGhlcmUgdG8gaGVscCcsXG4gIFNFTkRfSU5WSVRFX01FU1NBR0VTX0RFU0NSSVBUSU9OID0gJ0JlZm9yZSBpbnZpdGluZyBUYXhUYW5rIHN1cHBvcnQgYmUgc3VyZSB0byBjaGVjayBvdXQgb3VyIGhlbHAgYXJ0aWNsZXMgYW5kIG1lc3NhZ2Ugb3VyIHN1cHBvcnQgdGVhbS4nLFxuICBUVF9BQ0NPVU5UQU5UX0lOVklURUQgPSAnQWNjb3VudGFudCBpbnZpdGF0aW9uIGhhcyBiZWVuIHNlbnQgc3VjY2Vzc2Z1bGx5JyxcbiAgVFRfQUNDT1VOVEFOVF9JTlZFVEVEX1RPT0xUSVAgPSAnTmVlZCBhbiBhY2NvdW50YW50PyBUYXhUYW5rIEFjY291bnRhbnRzIGRlbGl2ZXJzIHZpcnR1YWwgYWNjb3VudGluZyBzZXJ2aWNlIGV4Y2x1c2l2ZSB0byBUYXhUYW5rIHN1YnNjcmliZXJzLiAnICtcbiAgJ0V4cGVyaWVuY2UgYSBmYXN0LCBzZWFtbGVzcywgYW5kIGNvc3QtZWZmZWN0aXZlIGxvZGdlbWVudCBzZXJ2aWNlIHRoYXQgcmV3YXJkcyB5b3VyIFRheFRhbmsgZWZmb3J0cyB0aHJvdWdob3V0IHRoZSB5ZWFyLicgK1xuICAnIE91ciB0ZWFtIG9mIGV4cGVyaWVuY2VkIGFjY291bnRhbnRzIGJyaW5ncyB0aGVpciBleHBlcnRpc2UgaW4gdGF4IGFuZCBUYXhUYW5rIHRvIHRoZSB2aXJ0dWFsIHJlYWxtLiBJdCBhbGwgc3RhcnRzIHdpdGggYSAxNS1taW51dGUgJyArXG4gICdjb25zdWx0YXRpb24gdGhhdCBzZXRzIHRoZSBzdGFnZSBmb3IgcGVyc29uYWxpc2VkIHNlcnZpY2UuIFdpdGggdXMsIHlvdeKAmWxsIHJlY2VpdmUgdGhlIGJlc3Qgb2YgYm90aCB3b3JsZHMgYXQgdGF4IHRpbWUuJyxcbiAgQ09ORklSTV9SRUpFQ1QgPSAnQXJlIHlvdSBzdXJlIHlvdSB3YW50IHRvIHJlamVjdCB0aGlzIGludml0YXRpb24/JyxcbiAgUkVKRUNURUQgPSAnSW52aXRhdGlvbiByZWplY3RlZCcsXG4gIEFDQ0VQVEVEID0gJ0NsaWVudCBpbnZpdGF0aW9uIGFjY2VwdGVkJ1xufVxuIl19
@@ -1,110 +1,103 @@
1
1
  import { Injectable } from '@angular/core';
2
- import { map } from 'rxjs/operators';
3
- import { replace } from '../../../../functions/array';
4
- import cloneDeep from 'lodash/cloneDeep';
2
+ import { throwError } from 'rxjs';
3
+ import { catchError, map } from 'rxjs/operators';
5
4
  import { plainToClass } from 'class-transformer';
6
- import { RestService } from '../../rest/rest-old.service';
5
+ import { RestService } from '../../rest/rest.service';
7
6
  import { AppEvent, AppEventTypeEnum, ClientInvite } from '../../../../models';
8
- import { UserEventTypeTypeEnum } from '../../../../db/Enums/user-event-type-type.enum';
9
7
  import { ClientInviteCollection } from '../../../../collections';
8
+ import last from 'lodash/last';
9
+ import merge from 'lodash/merge';
10
10
  import * as i0 from "@angular/core";
11
11
  export class ClientInviteService extends RestService {
12
12
  constructor() {
13
13
  super(...arguments);
14
- this.url = 'clients/invites';
14
+ this.endpointUri = 'clients/invites';
15
+ this.collectionClass = ClientInviteCollection;
15
16
  this.modelClass = ClientInvite;
16
17
  }
17
- listenEvents() {
18
- this.listenNotifications();
19
- }
20
18
  getFirmInvites() {
21
- return super.get(`${this.environment.apiV2}/${this.url}/all`);
19
+ return super.get(`${this.apiUrl}/all`);
22
20
  }
21
+ /**
22
+ * Get invite which employee is TaxTank Advisor
23
+ * @TODO move to collection? but we need to inject enviroment
24
+ */
23
25
  getTTAdvisor() {
24
- return this.get()
25
- .pipe(map((invites) => new ClientInviteCollection(invites).findBy('employee.email', this.environment.ttAdvisor) || null));
26
+ return this.get().pipe(map((invites) => invites.findBy('employee.email', this.environment.ttAdvisor)));
26
27
  }
27
28
  /**
28
- * Import employees for firm from CSV file
29
- * @param file
29
+ * Import clients from CSV file
30
30
  */
31
31
  import(file) {
32
32
  const formData = new FormData();
33
33
  formData.append('file', file);
34
- return this.http.post(`${this.environment.apiV2}/${this.url}/import`, formData)
35
- .pipe(map((clientInvite) => clientInvite));
34
+ // cant use parent method because of different parameters
35
+ return this.http.post(`${this.apiUrl}/import`, formData).pipe(map((clientInvite) => clientInvite));
36
36
  }
37
37
  /**
38
38
  * Accept client invitation from firm to user
39
- * @param invite
40
39
  */
41
40
  accept(invite) {
42
- return this.http.post(`${this.environment.apiV2}/${this.url}/${invite.id}/accept`, null)
43
- .pipe(map(() => {
44
- this.eventDispatcherService.dispatch(new AppEvent(AppEventTypeEnum.CLIENT_INVITE_ACCEPTED, invite));
45
- // set cache without accepted invite
46
- this.cache = this.cache.filter((clientInvite) => invite.id !== clientInvite.id);
47
- this.updateCache();
41
+ // cant use parent method because of custom specific handler
42
+ return this.http.put(`${this.apiUrl}/${invite.id}/accept`, invite).pipe(map(() => {
43
+ // @TODO remove when TaxReviewService refactored with the new rest and use there listenCSE instead
44
+ // TaxReviewService is listening client invites acception. With the new rest we are using listenCSE method, but old rest does not have this method.
45
+ // We need to refactor TaxReviewService with new rest first, then we can listenCSE and remove manual events
46
+ this.eventDispatcherService.dispatch(new AppEvent(AppEventTypeEnum.CLIENT_INVITE_ACCEPTED, null));
47
+ this.handleResponse([invite], 'delete');
48
48
  }));
49
49
  }
50
50
  /**
51
51
  * Reject client invitation from firm to user
52
- * @param invite
53
52
  */
54
53
  reject(invite) {
55
- return this.http.post(`${this.environment.apiV2}/${this.url}/${invite.id}/reject`, null)
56
- .pipe(map((rejectedInvite) => {
57
- const updatedInvite = plainToClass(ClientInvite, rejectedInvite);
58
- // avoid cache changes
59
- // @TODO make cache readonly
60
- const tempCache = cloneDeep(this.cache);
61
- replace(tempCache, plainToClass(ClientInvite, updatedInvite));
62
- this.cache = tempCache;
63
- this.updateCache();
64
- return updatedInvite;
65
- }));
54
+ return super.put(invite, null, `${this.apiUrl}/${invite.id}/reject`);
66
55
  }
67
56
  /**
68
57
  * Resend invitation from firm to client
69
- * @param invite
70
58
  */
71
59
  resend(invite) {
72
- return this.http.post(`${this.environment.apiV2}/${this.url}/${invite.id}/resend`, null)
73
- .pipe(map((rejectedInvite) => {
74
- const updatedInvite = plainToClass(ClientInvite, rejectedInvite);
75
- // avoid cache changes
76
- // @TODO make cache readonly
77
- const tempCache = cloneDeep(this.cache);
78
- replace(tempCache, plainToClass(ClientInvite, updatedInvite));
79
- this.cache = tempCache;
80
- this.updateCache();
81
- return updatedInvite;
60
+ // cant use parent method because of custom specific handler
61
+ return this.http.post(`${this.apiUrl}/${invite.id}/resend`, null)
62
+ .pipe(map((inviteBase) => plainToClass(ClientInvite, inviteBase)), map((resentInvite) => {
63
+ this.handleResponse([resentInvite], 'put');
64
+ return resentInvite;
82
65
  }));
83
66
  }
84
67
  /**
85
68
  * Send invitation from client to firm
86
- * @param invite
87
69
  */
88
70
  inviteFirmByUser(invite) {
89
- return this.http.post(`${this.environment.apiV2}/firms/invite`, invite)
90
- .pipe(map((inviteBase) => {
91
- const newInvite = plainToClass(ClientInvite, inviteBase);
92
- // avoid cache changes
93
- // @TODO make cache readonly
94
- const tempCache = cloneDeep(this.cache);
95
- tempCache.push(newInvite);
96
- this.cache = tempCache;
97
- this.updateCache();
98
- return newInvite;
71
+ return super.post(invite, null, `${this.environment.apiV2}/firms/invite`)
72
+ .pipe(catchError((error) => {
73
+ if (error.error.email) {
74
+ this.toastService.error(error.error.email[0]);
75
+ }
76
+ else {
77
+ this.toastService.error(error.error);
78
+ }
79
+ return throwError(error);
99
80
  }));
100
81
  }
101
82
  /**
102
- * Reset cache on client/firm invite events
83
+ * Listem clientInvites db changes via mercure to update cache
84
+ * @TODO make it universal and move to parent
103
85
  */
104
- listenNotifications() {
105
- this.eventDispatcherService.on(AppEventTypeEnum.NOTIFICATION_ADDED).subscribe((notification) => {
106
- if (!notification.isRead && (notification.eventType === UserEventTypeTypeEnum.FIRM_INVITE_ACCEPTED || notification.eventType === UserEventTypeTypeEnum.CLIENT_INVITE)) {
107
- this.refreshCache();
86
+ listenSSE() {
87
+ this.sseService.on('clientInvites')
88
+ .pipe(map((clientInvite) => plainToClass(ClientInvite, clientInvite)))
89
+ .subscribe((clientInvite) => {
90
+ // we got empty body when item deleted so we should get id from Api Platform data (@id)
91
+ const id = clientInvite.id ?? +last(clientInvite['@id'].split('/'));
92
+ switch (true) {
93
+ case !clientInvite.id:
94
+ this.updateCache([merge(clientInvite, { id })], 'delete');
95
+ break;
96
+ case this.hasInCache(id):
97
+ this.updateCache([clientInvite], 'put');
98
+ break;
99
+ case !this.hasInCache(id):
100
+ this.updateCache([clientInvite], 'post');
108
101
  }
109
102
  });
110
103
  }
@@ -117,4 +110,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
117
110
  providedIn: 'root'
118
111
  }]
119
112
  }] });
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client-invite.service.js","sourceRoot":"","sources":["../../../../../../../../projects/tt-core/src/lib/services/http/firm/client-invite/client-invite.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Observable } from 'rxjs';\nimport { ClientInvite as ClientInviteBase } from '../../../../db/Models/firm/client-invite';\nimport { map } from 'rxjs/operators';\nimport { replace } from '../../../../functions/array';\nimport cloneDeep from 'lodash/cloneDeep';\nimport { plainToClass } from 'class-transformer';\nimport { RestService } from '../../rest/rest-old.service';\nimport { AppEvent, AppEventTypeEnum, ClientInvite, Notification } from '../../../../models';\nimport { IEventListener } from '../../../../interfaces';\nimport { UserEventTypeTypeEnum } from '../../../../db/Enums/user-event-type-type.enum';\nimport { ClientInviteCollection } from '../../../../collections';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class ClientInviteService extends RestService<ClientInviteBase, ClientInvite> implements IEventListener {\n  url = 'clients/invites';\n  modelClass: typeof ClientInvite = ClientInvite;\n\n  listenEvents(): void {\n    this.listenNotifications();\n  }\n\n  getFirmInvites(): Observable<ClientInvite[]> {\n    return super.get(`${this.environment.apiV2}/${this.url}/all`);\n  }\n\n  getTTAdvisor(): Observable<ClientInvite> {\n    return this.get()\n      .pipe(\n        map((invites: ClientInvite[]) => new ClientInviteCollection(invites).findBy('employee.email', this.environment.ttAdvisor) || null)\n      );\n  }\n\n  /**\n   * Import employees for firm from CSV file\n   * @param file\n   */\n  import(file: File): Observable<ClientInvite[]> {\n    const formData: FormData = new FormData();\n    formData.append('file', file);\n\n    return this.http.post(`${this.environment.apiV2}/${this.url}/import`, formData)\n      .pipe(\n        map((clientInvite: ClientInvite[]) => clientInvite)\n      );\n  }\n\n  /**\n   * Accept client invitation from firm to user\n   * @param invite\n   */\n  accept(invite: ClientInvite): Observable<any> {\n    return this.http.post(`${this.environment.apiV2}/${this.url}/${invite.id}/accept`, null)\n      .pipe(\n        map((): void => {\n          this.eventDispatcherService.dispatch(new AppEvent(AppEventTypeEnum.CLIENT_INVITE_ACCEPTED, invite));\n          // set cache without accepted invite\n          this.cache = this.cache.filter((clientInvite: ClientInvite) => invite.id !== clientInvite.id);\n          this.updateCache();\n        })\n      );\n  }\n\n  /**\n   * Reject client invitation from firm to user\n   * @param invite\n   */\n  reject(invite: ClientInvite): Observable<ClientInvite> {\n    return this.http.post(`${this.environment.apiV2}/${this.url}/${invite.id}/reject`, null)\n      .pipe(\n        map((rejectedInvite: ClientInviteBase): ClientInvite => {\n          const updatedInvite: ClientInvite = plainToClass(ClientInvite, rejectedInvite);\n          // avoid cache changes\n          // @TODO make cache readonly\n          const tempCache: ClientInvite[] = cloneDeep(this.cache);\n          replace(tempCache, plainToClass(ClientInvite, updatedInvite));\n          this.cache = tempCache;\n          this.updateCache();\n          return updatedInvite;\n        })\n      );\n  }\n\n  /**\n   * Resend invitation from firm to client\n   * @param invite\n   */\n  resend(invite: ClientInvite): Observable<ClientInvite> {\n    return this.http.post(`${this.environment.apiV2}/${this.url}/${invite.id}/resend`, null)\n      .pipe(\n        map((rejectedInvite: ClientInviteBase): ClientInvite => {\n          const updatedInvite: ClientInvite = plainToClass(ClientInvite, rejectedInvite);\n          // avoid cache changes\n          // @TODO make cache readonly\n          const tempCache: ClientInvite[] = cloneDeep(this.cache);\n          replace(tempCache, plainToClass(ClientInvite, updatedInvite));\n          this.cache = tempCache;\n          this.updateCache();\n          return updatedInvite;\n        })\n      );\n  }\n\n  /**\n   * Send invitation from client to firm\n   * @param invite\n   */\n  inviteFirmByUser(invite: ClientInvite): Observable<ClientInvite> {\n    return this.http.post(`${this.environment.apiV2}/firms/invite`, invite)\n      .pipe(\n        map((inviteBase: ClientInviteBase): ClientInvite => {\n          const newInvite: ClientInvite = plainToClass(ClientInvite, inviteBase);\n          // avoid cache changes\n          // @TODO make cache readonly\n          const tempCache: ClientInvite[] = cloneDeep(this.cache);\n          tempCache.push(newInvite);\n          this.cache = tempCache;\n          this.updateCache();\n          return newInvite;\n        })\n      );\n  }\n\n  /**\n   * Reset cache on client/firm invite events\n   */\n  listenNotifications(): void {\n    this.eventDispatcherService.on(AppEventTypeEnum.NOTIFICATION_ADDED).subscribe((notification: Notification) => {\n      if (!notification.isRead && (notification.eventType === UserEventTypeTypeEnum.FIRM_INVITE_ACCEPTED || notification.eventType === UserEventTypeTypeEnum.CLIENT_INVITE)) {\n        this.refreshCache();\n      }\n    });\n  }\n}\n"]}
113
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client-invite.service.js","sourceRoot":"","sources":["../../../../../../../../projects/tt-core/src/lib/services/http/firm/client-invite/client-invite.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Observable, throwError } from 'rxjs';\nimport { ClientInvite as ClientInviteBase } from '../../../../db/Models/firm/client-invite';\nimport { catchError, map } from 'rxjs/operators';\nimport { plainToClass } from 'class-transformer';\nimport { RestService } from '../../rest/rest.service';\nimport { AppEvent, AppEventTypeEnum, ClientInvite } from '../../../../models';\nimport { IEventListener } from '../../../../interfaces';\nimport { ClientInviteCollection } from '../../../../collections';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport last from 'lodash/last';\nimport merge from 'lodash/merge';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class ClientInviteService extends RestService<ClientInviteBase, ClientInvite, ClientInviteCollection> implements IEventListener {\n  protected endpointUri = 'clients/invites';\n  collectionClass: typeof ClientInviteCollection = ClientInviteCollection;\n  modelClass: typeof ClientInvite = ClientInvite;\n  \n  getFirmInvites(): Observable<ClientInviteCollection> {\n    return super.get(`${this.apiUrl}/all`);\n  }\n\n  /**\n   * Get invite which employee is TaxTank Advisor\n   * @TODO move to collection? but we need to inject enviroment \n   */\n  getTTAdvisor(): Observable<ClientInvite> {\n    return this.get().pipe(\n      map((invites: ClientInviteCollection) => invites.findBy('employee.email', this.environment.ttAdvisor))\n    );\n  }\n\n  /**\n   * Import clients from CSV file\n   */\n  import(file: File): Observable<ClientInvite[]> {\n    const formData: FormData = new FormData();\n    formData.append('file', file);\n\n    // cant use parent method because of different parameters\n    return this.http.post(`${this.apiUrl}/import`, formData).pipe(\n      map((clientInvite: ClientInvite[]) => clientInvite)\n    );\n  }\n\n  /**\n   * Accept client invitation from firm to user\n   */\n  accept(invite: ClientInvite): Observable<void> {\n    // cant use parent method because of custom specific handler\n    return this.http.put(`${this.apiUrl}/${invite.id}/accept`, invite).pipe(\n      map(() => {\n        // @TODO remove when TaxReviewService refactored with the new rest and use there listenCSE instead\n        // TaxReviewService is listening client invites acception. With the new rest we are using listenCSE method, but old rest does not have this method.\n        // We need to refactor TaxReviewService with new rest first, then we can listenCSE and remove manual events\n        this.eventDispatcherService.dispatch(new AppEvent(AppEventTypeEnum.CLIENT_INVITE_ACCEPTED, null));\n\n        this.handleResponse([invite], 'delete');\n      })\n    );\n  }\n\n  /**\n   * Reject client invitation from firm to user\n   */\n  reject(invite: ClientInvite): Observable<ClientInvite> {\n    return super.put(invite, null, `${this.apiUrl}/${invite.id}/reject`);\n  }\n\n  /**\n   * Resend invitation from firm to client\n   */\n  resend(invite: ClientInvite): Observable<ClientInvite> {\n    // cant use parent method because of custom specific handler\n    return this.http.post(`${this.apiUrl}/${invite.id}/resend`, null)\n      .pipe(\n        map((inviteBase: ClientInviteBase) => plainToClass(ClientInvite, inviteBase)),\n        map((resentInvite: ClientInvite) => {\n          this.handleResponse([resentInvite], 'put');\n\n          return resentInvite;\n        })\n      );\n  }\n\n  /**\n   * Send invitation from client to firm\n   */\n  inviteFirmByUser(invite: ClientInvite): Observable<ClientInvite> {\n    return super.post(invite, null, `${this.environment.apiV2}/firms/invite`)\n      .pipe(\n        catchError((error: HttpErrorResponse) => {\n          if (error.error.email) {\n            this.toastService.error(error.error.email[0]);\n          } else {\n            this.toastService.error(error.error);\n          }\n\n          return throwError(error);\n        })\n      );\n  }\n\n  /**\n   * Listem clientInvites db changes via mercure to update cache\n   * @TODO make it universal and move to parent\n   */\n  listenSSE(): void {\n    this.sseService.on('clientInvites')\n      .pipe(map((clientInvite: ClientInviteBase) => plainToClass(ClientInvite, clientInvite)))\n      .subscribe((clientInvite: ClientInvite) => {\n\n        // we got empty body when item deleted so we should get id from Api Platform data (@id)\n        const id = clientInvite.id ?? +last(clientInvite['@id'].split('/'));\n\n        switch (true) {\n          case !clientInvite.id:\n            this.updateCache([merge(clientInvite, {id})], 'delete');\n            break;\n\n          case this.hasInCache(id):\n            this.updateCache([clientInvite], 'put');\n            break;\n\n          case !this.hasInCache(id):\n            this.updateCache([clientInvite], 'post');\n        }\n      })\n  }\n}\n"]}
@@ -3,31 +3,27 @@ import { forkJoin } from 'rxjs';
3
3
  import { map } from 'rxjs/operators';
4
4
  import { ClientMovementCollection } from '../../../../collections/client-movement.collection';
5
5
  import { plainToClass } from 'class-transformer';
6
- import { RestService } from '../../rest/rest-old.service';
6
+ import { RestService } from '../../rest/rest.service';
7
7
  import { ClientMovement } from '../../../../models/client/client-movement';
8
8
  import { AppEvent } from '../../../../models/event/app-event';
9
9
  import { AppEventTypeEnum } from '../../../../models/event/app-event-type.enum';
10
- import { UserEventTypeTypeEnum } from '../../../../db/Enums/user-event-type-type.enum';
10
+ import { ClientInvite } from '../../../../models';
11
11
  import * as i0 from "@angular/core";
12
12
  export class ClientMovementService extends RestService {
13
13
  constructor() {
14
14
  super(...arguments);
15
+ this.endpointUri = 'client-movements';
16
+ this.collectionClass = ClientMovementCollection;
15
17
  this.modelClass = ClientMovement;
16
- this.url = 'client-movements';
17
- this.isHydra = true;
18
+ this.isApiPlatform = true;
19
+ this.disabledMethods = ['postBatch', 'delete', 'deleteBatch'];
18
20
  }
19
21
  listenEvents() {
20
- this.listenNotifications();
21
- this.listenClientInvites();
22
- }
23
- /**
24
- * @TODO BaseRestService should return collection
25
- */
26
- getAll() {
27
- return this.get().pipe(map((clientMovements) => new ClientMovementCollection(clientMovements)));
22
+ // @TODO handle with mercure
23
+ this.listenCSE(ClientInvite, ['post', 'put', 'delete'], this.refreshCache);
28
24
  }
29
25
  getActive() {
30
- return this.getAll().pipe(map((clientMovements) => clientMovements.active));
26
+ return this.get().pipe(map((clientMovements) => clientMovements.active));
31
27
  }
32
28
  getTTAdvisor() {
33
29
  return this.getActive()
@@ -36,18 +32,25 @@ export class ClientMovementService extends RestService {
36
32
  getActiveByFirmType(type) {
37
33
  return this.getActive().pipe(map((clientMovements) => clientMovements.getByFirmType(type)));
38
34
  }
39
- update(movement) {
35
+ /**
36
+ * Method is using for transfer client between employees.
37
+ * On backend we close passed movement and create a new one instead
38
+ */
39
+ put(movement) {
40
40
  // extend base 'add' method and pass into it movement without id
41
- return super.add(plainToClass(ClientMovement, Object.assign({}, movement, { id: null }))).pipe(map((updatedMovement) => {
41
+ return super.post(plainToClass(ClientMovement, Object.assign({}, movement, { id: null }))).pipe(map((updatedMovement) => {
42
42
  // remove provided movement from cache
43
43
  this.cache = this.cache.filter((item) => item.id !== movement.id);
44
- this.updateCache();
44
+ this.updateCache([updatedMovement], 'post');
45
45
  this.eventDispatcherService.dispatch(new AppEvent(AppEventTypeEnum.CLIENT_TRANSFER_TO_OTHER_EMPLOYEE, [updatedMovement.client]));
46
46
  return updatedMovement;
47
47
  }));
48
48
  }
49
- updateBatch(clientMovements) {
50
- const batch = clientMovements.map((clientMovement) => this.update(clientMovement));
49
+ /**
50
+ * There is no actually direct updating of client movement, but we are using this method for transfer clients between employees
51
+ */
52
+ putBatch(clientMovements) {
53
+ const batch = clientMovements.map((clientMovement) => this.put(clientMovement));
51
54
  return forkJoin(batch);
52
55
  }
53
56
  /**
@@ -55,27 +58,24 @@ export class ClientMovementService extends RestService {
55
58
  */
56
59
  close(clientMovement) {
57
60
  clientMovement.dateTo = new Date();
61
+ // cant use parent method because of custom specific handler
58
62
  return this.http.put(`${this.environment.apiV2}/client-movements/${clientMovement.id}/close`, clientMovement)
59
63
  .pipe(map(() => {
64
+ // @TODO remove when TaxReviewService refactored with the new rest and use there listenCSE instead
60
65
  this.eventDispatcherService.dispatch(new AppEvent(AppEventTypeEnum.CLIENT_MOVEMENT_CLOSED, null));
61
66
  this.cache = this.cache.filter((movement) => movement.id !== clientMovement.id);
62
- this.updateCache();
67
+ this.updateCache([clientMovement], 'delete');
63
68
  }));
64
69
  }
65
- listenNotifications() {
66
- this.eventDispatcherService.on(AppEventTypeEnum.NOTIFICATION_ADDED).subscribe((notification) => {
67
- if (!notification.isRead && notification.eventType === UserEventTypeTypeEnum.FIRM_INVITE_ACCEPTED) {
68
- this.refreshCache();
69
- }
70
- });
71
- }
72
70
  /**
73
- * Listen to EventDispatcherService event related to clients invitations
71
+ * Listem clientMovements db changes via mercure to update cache
72
+ * @TODO make it universal and move to parent
74
73
  */
75
- listenClientInvites() {
76
- this.eventDispatcherService.on(AppEventTypeEnum.CLIENT_INVITE_ACCEPTED)
77
- .subscribe(() => {
78
- this.refreshCache();
74
+ listenSSE() {
75
+ this.sseService.on('clientMovements')
76
+ .pipe(map((clientMovement) => plainToClass(ClientMovement, clientMovement)))
77
+ .subscribe((clientMovement) => {
78
+ this.hasInCache(clientMovement.id) ? this.updateCache([clientMovement], 'put') : this.updateCache([clientMovement], 'post');
79
79
  });
80
80
  }
81
81
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ClientMovementService, deps: null, target: i0.ɵɵFactoryTarget.Injectable }); }
@@ -87,4 +87,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
87
87
  providedIn: 'root'
88
88
  }]
89
89
  }] });
90
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client-movement.service.js","sourceRoot":"","sources":["../../../../../../../../projects/tt-core/src/lib/services/http/firm/client-movement/client-movement.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { ClientMovement as ClientMovementBase } from '../../../../db/Models/firm/client-movement';\nimport { Observable, forkJoin } from 'rxjs';\nimport { map } from 'rxjs/operators';\nimport { ClientMovementCollection } from '../../../../collections/client-movement.collection';\nimport { FirmTypeEnum } from '../../../../db/Enums/firm-type.enum';\nimport { plainToClass } from 'class-transformer';\nimport { RestService } from '../../rest/rest-old.service';\nimport { ClientMovement } from '../../../../models/client/client-movement';\nimport { AppEvent } from '../../../../models/event/app-event';\nimport { AppEventTypeEnum } from '../../../../models/event/app-event-type.enum';\nimport { IEventListener } from '../../../../interfaces/event-listener.interface';\nimport { Notification } from '../../../../models/notification/notification';\nimport { UserEventTypeTypeEnum } from '../../../../db/Enums/user-event-type-type.enum';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class ClientMovementService extends RestService<ClientMovementBase, ClientMovement> implements IEventListener {\n  modelClass: typeof ClientMovement = ClientMovement;\n  url = 'client-movements';\n  isHydra = true;\n\n  listenEvents(): void {\n    this.listenNotifications();\n    this.listenClientInvites();\n  }\n\n  /**\n   * @TODO BaseRestService should return collection\n   */\n  getAll(): Observable<ClientMovementCollection> {\n    return this.get().pipe(\n      map((clientMovements: ClientMovement[]) => new ClientMovementCollection(clientMovements))\n    );\n  }\n\n  getActive(): Observable<ClientMovementCollection> {\n    return this.getAll().pipe(\n      map((clientMovements: ClientMovementCollection): ClientMovementCollection => clientMovements.active)\n    );\n  }\n\n  getTTAdvisor(): Observable<ClientMovement> {\n    return this.getActive()\n      .pipe(\n        map((movements: ClientMovementCollection) => movements.findBy('employee.email', this.environment.ttAdvisor) || null)\n      );\n  }\n\n  getActiveByFirmType(type: FirmTypeEnum): Observable<ClientMovementCollection> {\n    return this.getActive().pipe(\n      map((clientMovements: ClientMovementCollection): ClientMovementCollection => clientMovements.getByFirmType(type))\n    );\n  }\n\n  update(movement: ClientMovement): Observable<ClientMovement> {\n    // extend base 'add' method and pass into it movement without id\n    return super.add(plainToClass(ClientMovement, Object.assign({}, movement, {id: null}))).pipe(\n      map((updatedMovement: ClientMovement) => {\n        // remove provided movement from cache\n        this.cache = this.cache.filter((item: ClientMovement) => item.id !== movement.id);\n        this.updateCache();\n\n        this.eventDispatcherService.dispatch(new AppEvent(AppEventTypeEnum.CLIENT_TRANSFER_TO_OTHER_EMPLOYEE, [updatedMovement.client]));\n\n        return updatedMovement;\n      })\n    );\n  }\n\n  updateBatch(clientMovements: ClientMovement[]): Observable<ClientMovement[]> {\n    const batch: Observable<ClientMovement>[] = clientMovements.map((clientMovement: ClientMovement) => this.update(clientMovement));\n    return forkJoin(batch);\n  }\n\n  /**\n   * Reject current firm\n   */\n  close(clientMovement: ClientMovement): Observable<void> {\n    clientMovement.dateTo = new Date();\n    return this.http.put(`${this.environment.apiV2}/client-movements/${clientMovement.id}/close`, clientMovement)\n      .pipe(\n        map((): void => {\n          this.eventDispatcherService.dispatch(new AppEvent(AppEventTypeEnum.CLIENT_MOVEMENT_CLOSED, null));\n          this.cache = this.cache.filter((movement: ClientMovement): boolean => movement.id !== clientMovement.id);\n          this.updateCache();\n        })\n      );\n  }\n\n  private listenNotifications(): void {\n    this.eventDispatcherService.on(AppEventTypeEnum.NOTIFICATION_ADDED).subscribe((notification: Notification) => {\n      if (!notification.isRead && notification.eventType === UserEventTypeTypeEnum.FIRM_INVITE_ACCEPTED) {\n        this.refreshCache();\n      }\n    });\n  }\n\n  /**\n   * Listen to EventDispatcherService event related to clients invitations\n   */\n  private listenClientInvites(): void {\n    this.eventDispatcherService.on(AppEventTypeEnum.CLIENT_INVITE_ACCEPTED)\n      .subscribe((): void => {\n        this.refreshCache();\n      });\n  }\n}\n"]}
90
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client-movement.service.js","sourceRoot":"","sources":["../../../../../../../../projects/tt-core/src/lib/services/http/firm/client-movement/client-movement.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { ClientMovement as ClientMovementBase } from '../../../../db/Models/firm/client-movement';\nimport { Observable, forkJoin } from 'rxjs';\nimport { map } from 'rxjs/operators';\nimport { ClientMovementCollection } from '../../../../collections/client-movement.collection';\nimport { FirmTypeEnum } from '../../../../db/Enums/firm-type.enum';\nimport { plainToClass } from 'class-transformer';\nimport { RestService } from '../../rest/rest.service';\nimport { ClientMovement } from '../../../../models/client/client-movement';\nimport { AppEvent } from '../../../../models/event/app-event';\nimport { AppEventTypeEnum } from '../../../../models/event/app-event-type.enum';\nimport { IEventListener } from '../../../../interfaces/event-listener.interface';\nimport { ClientInvite } from '../../../../models';\nimport { RestMethod } from '../../rest';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class ClientMovementService extends RestService<ClientMovementBase, ClientMovement, ClientMovementCollection> implements IEventListener {\n  protected endpointUri = 'client-movements';\n  collectionClass: typeof ClientMovementCollection = ClientMovementCollection;\n  modelClass: typeof ClientMovement = ClientMovement;\n  isApiPlatform = true;\n  disabledMethods: RestMethod[] = ['postBatch', 'delete', 'deleteBatch'];\n\n  listenEvents(): void {\n    // @TODO handle with mercure\n    this.listenCSE(ClientInvite, ['post', 'put', 'delete'], this.refreshCache);\n  }\n\n  getActive(): Observable<ClientMovementCollection> {\n    return this.get().pipe(\n      map((clientMovements: ClientMovementCollection): ClientMovementCollection => clientMovements.active)\n    );\n  }\n\n  getTTAdvisor(): Observable<ClientMovement> {\n    return this.getActive()\n      .pipe(\n        map((movements: ClientMovementCollection) => movements.findBy('employee.email', this.environment.ttAdvisor) || null)\n      );\n  }\n\n  getActiveByFirmType(type: FirmTypeEnum): Observable<ClientMovementCollection> {\n    return this.getActive().pipe(\n      map((clientMovements: ClientMovementCollection): ClientMovementCollection => clientMovements.getByFirmType(type))\n    );\n  }\n\n  /**\n   * Method is using for transfer client between employees.\n   * On backend we close passed movement and create a new one instead\n   */\n  put(movement: ClientMovement): Observable<ClientMovement> {\n    // extend base 'add' method and pass into it movement without id\n    return super.post(plainToClass(ClientMovement, Object.assign({}, movement, {id: null}))).pipe(\n      map((updatedMovement: ClientMovement) => {\n        // remove provided movement from cache\n        this.cache = this.cache.filter((item: ClientMovement) => item.id !== movement.id);\n        this.updateCache([updatedMovement], 'post');\n\n        this.eventDispatcherService.dispatch(new AppEvent(AppEventTypeEnum.CLIENT_TRANSFER_TO_OTHER_EMPLOYEE, [updatedMovement.client]));\n\n        return updatedMovement;\n      })\n    );\n  }\n\n  /**\n   * There is no actually direct updating of client movement, but we are using this method for transfer clients between employees\n   */\n  putBatch(clientMovements: ClientMovement[]): Observable<ClientMovement[]> {\n    const batch: Observable<ClientMovement>[] = clientMovements.map((clientMovement: ClientMovement) => this.put(clientMovement));\n  \n    return forkJoin(batch);\n  }\n\n  /**\n   * Reject current firm\n   */\n  close(clientMovement: ClientMovement): Observable<void> {\n    clientMovement.dateTo = new Date();\n    // cant use parent method because of custom specific handler\n    return this.http.put(`${this.environment.apiV2}/client-movements/${clientMovement.id}/close`, clientMovement)\n      .pipe(\n        map((): void => {\n          // @TODO remove when TaxReviewService refactored with the new rest and use there listenCSE instead\n          this.eventDispatcherService.dispatch(new AppEvent(AppEventTypeEnum.CLIENT_MOVEMENT_CLOSED, null));\n\n          this.cache = this.cache.filter((movement: ClientMovement): boolean => movement.id !== clientMovement.id);\n          this.updateCache([clientMovement], 'delete');\n        })\n      );\n  }\n  \n  /**\n   * Listem clientMovements db changes via mercure to update cache\n   * @TODO make it universal and move to parent\n   */\n  listenSSE(): void {\n    this.sseService.on('clientMovements')\n      .pipe(map((clientMovement: ClientMovementBase) => plainToClass(ClientMovement, clientMovement)))\n      .subscribe((clientMovement: ClientMovement) => {\n        this.hasInCache(clientMovement.id) ? this.updateCache([clientMovement], 'put') : this.updateCache([clientMovement], 'post');\n      })\n  }\n}\n"]}
@@ -1,6 +1,7 @@
1
1
  import { Inject, Injectable, inject } from '@angular/core';
2
2
  import { ReplaySubject } from 'rxjs';
3
3
  import { catchError, first, map } from 'rxjs/operators';
4
+ import { SseService } from '../../event';
4
5
  import { classToPlain } from 'class-transformer';
5
6
  import { ObservableModel } from '../../../db/Models';
6
7
  import { AppEvent2 } from '../../../models';
@@ -8,6 +9,7 @@ import { toArray } from '../../../functions/array';
8
9
  import { DataService } from '../../data.service';
9
10
  import { MixpanelService } from '../../mixpanel/mixpanel.service';
10
11
  import { throwError as observableThrowError } from 'rxjs';
12
+ import { ToastService } from '../../toast';
11
13
  import intersection from 'lodash/intersection';
12
14
  import * as i0 from "@angular/core";
13
15
  import * as i1 from "@angular/common/http";
@@ -36,9 +38,17 @@ export class RestService extends DataService {
36
38
  */
37
39
  this.disabledMethods = [];
38
40
  this.mpService = inject(MixpanelService);
41
+ this.sseService = inject(SseService);
42
+ this.toastService = inject(ToastService);
39
43
  this.roles = [];
44
+ /**
45
+ * disable cache update by requests, update only by SSE events
46
+ */
47
+ this.disableCache = false;
40
48
  // @TODO Alex remove, bad idea to call empty method overrided in child (because it will be called with parent context)
41
49
  this.listenEvents();
50
+ // @TODO Alex remove, bad idea to call empty method overrided in child (because it will be called with parent context)
51
+ this.listenSSE();
42
52
  }
43
53
  get apiUrl() {
44
54
  return `${this.environment.apiV2}/${this.endpointUri}`;
@@ -83,6 +93,9 @@ export class RestService extends DataService {
83
93
  this.handleAccessError('get');
84
94
  return this.get().pipe(map((collection) => collection.first));
85
95
  }
96
+ hasInCache(id) {
97
+ return !!this.getCache().findBy('id', id);
98
+ }
86
99
  getArray() {
87
100
  return this.get().pipe(map((collection) => collection.toArray()));
88
101
  }
@@ -173,6 +186,10 @@ export class RestService extends DataService {
173
186
  }));
174
187
  }
175
188
  track(response, method, mpData, postfix = '') {
189
+ // check if result exist, sometimes response may be empty (for example ClientInvite accept/reject/etc)
190
+ if (!response[0]) {
191
+ return;
192
+ }
176
193
  // no other way to check interface
177
194
  if ('getMpData' in response[0]) {
178
195
  response.forEach((model) => {
@@ -189,7 +206,7 @@ export class RestService extends DataService {
189
206
  */
190
207
  handleResponse(response, method, mpData) {
191
208
  this.track(response, method, mpData);
192
- if (this.getCache()) {
209
+ if (this.getCache() && !this.disableCache) {
193
210
  this.updateCache(response, method);
194
211
  }
195
212
  // dispatch event for interested services
@@ -266,6 +283,10 @@ export class RestService extends DataService {
266
283
  * Method that call all listeners. Empty by default. Should be redefined by child services if required
267
284
  */
268
285
  listenEvents() { }
286
+ /**
287
+ * Listen SSE events
288
+ */
289
+ listenSSE() { }
269
290
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: RestService, deps: [{ token: i1.HttpClient }, { token: i2.EventDispatcherService }, { token: 'environment' }], target: i0.ɵɵFactoryTarget.Injectable }); }
270
291
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: RestService, providedIn: 'root' }); }
271
292
  }
@@ -278,4 +299,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
278
299
  type: Inject,
279
300
  args: ['environment']
280
301
  }] }]; } });
281
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rest.service.js","sourceRoot":"","sources":["../../../../../../../projects/tt-core/src/lib/services/http/rest/rest.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, inject } from '@angular/core';\nimport { Observable, ReplaySubject } from 'rxjs';\nimport { catchError, first, map } from 'rxjs/operators';\nimport { HttpClient, HttpErrorResponse } from '@angular/common/http';\nimport { EventDispatcherService } from '../../event';\nimport { classToPlain } from 'class-transformer';\nimport { IEventListener } from '../../../interfaces';\nimport { Collection } from '../../../collections';\nimport { AbstractModel, ObservableModel } from '../../../db/Models';\nimport { AppEvent2 } from '../../../models';\nimport { toArray } from '../../../functions/array';\nimport { RestMethod } from './rest-method.type';\nimport { HttpMethod } from './http-method.type';\nimport { DataService } from '../../data.service';\nimport { UserRolesEnum } from '../../../db/Enums/user-roles.enum';\nimport { MixpanelService } from '../../mixpanel/mixpanel.service';\nimport { throwError as observableThrowError } from 'rxjs';\nimport intersection from 'lodash/intersection';\n\n/**\n * Abstract base service that implements common services functionality\n * and describe abstract methods/properties that have to be implemented in child services\n * Model - entity service is working with\n * BaseModel - base entity model that extends by Model\n * CollectionModel - entity collection class\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport abstract class RestService<BaseModel, Model extends AbstractModel, CollectionModel extends Collection<Model>>\n  extends DataService<BaseModel, Model, CollectionModel> implements IEventListener {\n\n  /**\n   * Url part for service entity API\n   */\n  protected abstract endpointUri: string;\n  /**\n   * Subject for service cache\n   */\n  protected cacheSubject: ReplaySubject<CollectionModel> = new ReplaySubject(1);\n  /**\n   * Flag for work with API platform\n   */\n  protected isApiPlatform: boolean;\n  /**\n   * List of methods unavailable for current API\n   * @TODO Alex: add and handle enabled methods too\n   * @TODO Alex: disable batch by default\n   */\n  disabledMethods: RestMethod[] = [];\n  protected mpService: MixpanelService = inject(MixpanelService);\n  roles: UserRolesEnum[] = [];\n\n  constructor(\n    protected http: HttpClient,\n    protected eventDispatcherService: EventDispatcherService,\n    @Inject('environment') protected environment: any\n  ) {\n    super();\n    // @TODO Alex remove, bad idea to call empty method overrided in child (because it will be called with parent context)\n    this.listenEvents();\n  }\n\n  protected get apiUrl(): string {\n    return `${this.environment.apiV2}/${this.endpointUri}`;\n  }\n\n  setCache(data: any[], next = false): void {\n    super.setCache(data);\n\n    if (next) {\n      this.cacheSubject.next(this.cache);\n    }\n  }\n\n  /**\n   * Refresh cache with actual backend data\n   */\n  refreshCache(): void {\n    this.cache = undefined;\n    this.get();\n  }\n\n  /**\n   * Get data from backend and fill the cache\n   */\n  protected fetch(path = this.apiUrl): Observable<CollectionModel> {\n    this.handleAccessError('get');\n\n    if (!this.hasRoles()) {\n      this.setCache([], true);\n      return this.cacheSubject.asObservable();\n    }\n\n    // Set cache as empty collection to avoid multiple requests before cache filled\n    this.setCache([]);\n    return this.http.get(path)\n      .pipe(\n        map((response: object) => this.isApiPlatform ? response['hydra:member'] : toArray(response)),\n        map((response: BaseModel[]) => {\n          this.setCache(response, true);\n\n          return this.cache;\n        })\n      );\n  }\n\n  get(path: string = this.apiUrl): Observable<CollectionModel> {\n    if (!this.cache) {\n      this.fetch(path).pipe(first()).subscribe();\n    }\n\n    return this.cacheSubject.asObservable();\n  }\n\n  getFirst(): Observable<Model> {\n    this.handleAccessError('get');\n\n    return this.get().pipe(\n      map((collection: CollectionModel) => collection.first)\n    );\n  }\n\n  getArray(): Observable<Model[]> {\n    return this.get().pipe(map((collection: CollectionModel) => collection.toArray()));\n  }\n\n  /**\n   * Create a new Model instance in database\n   */\n  post(model: Model, mpData?: AbstractModel, path: string = this.apiUrl): Observable<Model> {\n    this.handleAccessError('post');\n\n    return this.http.post(path, classToPlain(model))\n      .pipe(\n        map((response: BaseModel) => {\n          const result: Model = this.createModelInstance(response);\n          this.handleResponse([result], 'post', mpData);\n\n          return result;\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError([model], 'post', mpData);\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  /**\n   * Create multiple new Model instances in database\n   */\n  postBatch(models: Model[]): Observable<Model[]> {\n    this.handleAccessError('postBatch');\n\n    return this.http.post(this.apiUrl, classToPlain(models))\n      .pipe(\n        map((response: BaseModel[]) => {\n          const result: Model[] = response.map((item: BaseModel): Model => this.createModelInstance(item));\n          this.handleResponse(result, 'post');\n\n          return result;\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError(models, 'post');\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  /**\n   * Change an existing Model instance in database\n   */\n  put(model: Model, mpData?: AbstractModel, path = `${this.apiUrl}/${model.id}`): Observable<Model> {\n    this.handleAccessError('put');\n\n    return this.http.put(path, classToPlain(model))\n      .pipe(\n        map((response: BaseModel) => {\n          const result: Model = this.createModelInstance(response);\n          this.handleResponse([result], 'put', mpData);\n\n          return result;\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError([model], 'put', mpData);\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  /**\n   * Change multiple existing Model instances in database\n   */\n  putBatch(models: Model[]): Observable<Model[]> {\n    this.handleAccessError('putBatch');\n\n    return this.http.put(this.apiUrl, classToPlain(models))\n      .pipe(\n        map((response: BaseModel[]) => {\n          const result: Model[] = response.map((item: BaseModel): Model => this.createModelInstance(item));\n          this.handleResponse(result, 'put');\n\n          return result;\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError(models, 'put');\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  /**\n   * Remove a Model instance from database\n   */\n  delete(model: Model, mpData?: AbstractModel): Observable<void> {\n    this.handleAccessError('delete');\n\n    return this.http.delete(`${this.apiUrl}/${model.id}`)\n      .pipe(\n        map(() => {\n          this.handleResponse([model], 'delete', mpData);\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError([model], 'delete', mpData);\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  /**\n   * Remove multiple Model instances from database\n   */\n  deleteBatch(models: Model[]): Observable<void> {\n    this.handleAccessError('deleteBatch');\n\n    return this.http.post(`${this.apiUrl}/delete`, classToPlain(models))\n      .pipe(\n        map(() => {\n          this.handleResponse(models, 'delete');\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError(models, 'delete');\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  protected track(response: Model[], method: HttpMethod, mpData?: AbstractModel, postfix = '') {\n    // no other way to check interface\n    if ('getMpData' in response[0]) {\n      response.forEach((model) => {\n        this.mpService.track(`${method}${this.modelClass.name}${postfix}`, model['getMpData'](mpData))\n      });\n    }\n  }\n\n  protected handleResponseError(response: Model[], method: HttpMethod, mpData?: AbstractModel) {\n    // @TODO error notification?\n    this.track(response, method, mpData, 'Error');\n  }\n\n  /**\n   * Handle response data - update cache and dispatch event if it is needed\n   */\n  protected handleResponse(response: Model[], method: HttpMethod, mpData?: AbstractModel) {\n    this.track(response, method, mpData);\n\n    if (this.getCache()) {\n      this.updateCache(response, method);\n    }\n\n    // dispatch event for interested services\n    if (response[0] instanceof ObservableModel) {\n      this.dispatchEvent(method, response);\n    }\n  }\n\n  /**\n   * Update cache with passed items. Add/Update detects automatically, Delete via optional flag\n   */\n  protected updateCache(items: Model[], method: HttpMethod): void {\n    let cache: CollectionModel;\n\n    switch (method) {\n      case 'post':\n        cache = this.cache.push(...items);\n        break;\n\n      case 'put':\n        // @TODO Alex old instance\n        items.forEach((item: Model) => {\n          cache = this.cache.replaceBy('id', item.id, item);\n        });\n        break;\n\n      case 'delete':\n        cache = this.cache.removeBy('id', items.map((item: Model) => item.id));\n        break;\n\n      case 'get':\n        cache = this.createCollectionInstance(this.collectionClass, items);\n    }\n\n    this.setCache(cache.toArray(), true);\n  }\n\n  /**\n   * Generate and dispatch rest event\n   */\n  protected dispatchEvent(method: HttpMethod, items: Model[]): void {\n    const eventName: string = (this.modelClass as typeof ObservableModel).getEventName(method);\n    this.eventDispatcherService.dispatch2(new AppEvent2(eventName, items));\n  }\n\n  /**\n   * Check if method is not disabled. Throw exception otherwise.\n   * Some entities does not have endpoints for all methods.\n   */\n  private handleAccessError(method: RestMethod): void {\n    if (!this.disabledMethods.includes(method)) {\n      return;\n    }\n\n    throw new Error(`Method ${method}() is not allowed for ${this.constructor.name}`);\n  }\n\n  /**\n   * Subscribe to http events and run callback.\n   * CSE - Cleint Sent Events\n   * @param modelClass\n   * @param methods The list of http methods should be listened for\n   * @param callback The function to be called when event triggered\n   */\n  protected listenCSE(modelClass: typeof ObservableModel, methods: HttpMethod[], callback: Function): void {\n    callback = callback.bind(this);\n\n    methods.forEach((method: HttpMethod) => {\n      this.eventDispatcherService.on2(modelClass.getEventName(method)).subscribe((data: ObservableModel[]) => {\n        callback(data);\n      });\n    })\n  }\n\n  /**\n   * check if user has subscription role to access api\n   */\n  hasRoles(): boolean {\n    if (!this.roles.length) {\n      return true;\n    }\n\n    return !!intersection(JSON.parse(localStorage.getItem('roles')), this.roles).length;\n  }\n\n  /**\n   * // @TODO Alex remove\n   * Method that call all listeners. Empty by default. Should be redefined by child services if required\n   */\n  listenEvents(): void {}\n}\n"]}
302
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rest.service.js","sourceRoot":"","sources":["../../../../../../../projects/tt-core/src/lib/services/http/rest/rest.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, inject } from '@angular/core';\nimport { Observable, ReplaySubject } from 'rxjs';\nimport { catchError, first, map } from 'rxjs/operators';\nimport { HttpClient, HttpErrorResponse } from '@angular/common/http';\nimport { EventDispatcherService, SseService } from '../../event';\nimport { classToPlain } from 'class-transformer';\nimport { IEventListener } from '../../../interfaces';\nimport { Collection } from '../../../collections';\nimport { AbstractModel, ObservableModel } from '../../../db/Models';\nimport { AppEvent2 } from '../../../models';\nimport { toArray } from '../../../functions/array';\nimport { RestMethod } from './rest-method.type';\nimport { HttpMethod } from './http-method.type';\nimport { DataService } from '../../data.service';\nimport { UserRolesEnum } from '../../../db/Enums/user-roles.enum';\nimport { MixpanelService } from '../../mixpanel/mixpanel.service';\nimport { throwError as observableThrowError } from 'rxjs';\nimport { ToastService } from '../../toast';\nimport intersection from 'lodash/intersection';\n\n/**\n * Abstract base service that implements common services functionality\n * and describe abstract methods/properties that have to be implemented in child services\n * Model - entity service is working with\n * BaseModel - base entity model that extends by Model\n * CollectionModel - entity collection class\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport abstract class RestService<BaseModel, Model extends AbstractModel, CollectionModel extends Collection<Model>>\n  extends DataService<BaseModel, Model, CollectionModel> implements IEventListener {\n\n  /**\n   * Url part for service entity API\n   */\n  protected abstract endpointUri: string;\n  /**\n   * Subject for service cache\n   */\n  protected cacheSubject: ReplaySubject<CollectionModel> = new ReplaySubject(1);\n  /**\n   * Flag for work with API platform\n   */\n  protected isApiPlatform: boolean;\n  /**\n   * List of methods unavailable for current API\n   * @TODO Alex: add and handle enabled methods too\n   * @TODO Alex: disable batch by default\n   */\n  disabledMethods: RestMethod[] = [];\n  protected mpService: MixpanelService = inject(MixpanelService);\n  protected sseService: SseService = inject(SseService);\n  protected toastService: ToastService = inject(ToastService);\n  roles: UserRolesEnum[] = [];\n\n  /**\n   * disable cache update by requests, update only by SSE events\n   */\n  disableCache: boolean = false;\n\n  constructor(\n    protected http: HttpClient,\n    protected eventDispatcherService: EventDispatcherService,\n    @Inject('environment') protected environment: any\n  ) {\n    super();\n    // @TODO Alex remove, bad idea to call empty method overrided in child (because it will be called with parent context)\n    this.listenEvents();\n    // @TODO Alex remove, bad idea to call empty method overrided in child (because it will be called with parent context)\n    this.listenSSE();\n  }\n\n  protected get apiUrl(): string {\n    return `${this.environment.apiV2}/${this.endpointUri}`;\n  }\n\n  setCache(data: any[], next = false): void {\n    super.setCache(data);\n\n    if (next) {\n      this.cacheSubject.next(this.cache);\n    }\n  }\n\n  /**\n   * Refresh cache with actual backend data\n   */\n  refreshCache(): void {\n    this.cache = undefined;\n    this.get();\n  }\n\n  /**\n   * Get data from backend and fill the cache\n   */\n  protected fetch(path = this.apiUrl): Observable<CollectionModel> {\n    this.handleAccessError('get');\n\n    if (!this.hasRoles()) {\n      this.setCache([], true);\n      return this.cacheSubject.asObservable();\n    }\n\n    // Set cache as empty collection to avoid multiple requests before cache filled\n    this.setCache([]);\n    return this.http.get(path)\n      .pipe(\n        map((response: object) => this.isApiPlatform ? response['hydra:member'] : toArray(response)),\n        map((response: BaseModel[]) => {\n          this.setCache(response, true);\n\n          return this.cache;\n        })\n      );\n  }\n\n  get(path: string = this.apiUrl): Observable<CollectionModel> {\n    if (!this.cache) {\n      this.fetch(path).pipe(first()).subscribe();\n    }\n\n    return this.cacheSubject.asObservable();\n  }\n\n  getFirst(): Observable<Model> {\n    this.handleAccessError('get');\n\n    return this.get().pipe(\n      map((collection: CollectionModel) => collection.first)\n    );\n  }\n\n  hasInCache(id: number): boolean {\n    return !!this.getCache().findBy('id', id);\n  }\n\n  getArray(): Observable<Model[]> {\n    return this.get().pipe(map((collection: CollectionModel) => collection.toArray()));\n  }\n\n  /**\n   * Create a new Model instance in database\n   */\n  post(model: Model, mpData?: AbstractModel, path: string = this.apiUrl): Observable<Model> {\n    this.handleAccessError('post');\n\n    return this.http.post(path, classToPlain(model))\n      .pipe(\n        map((response: BaseModel) => {\n          const result: Model = this.createModelInstance(response);\n          this.handleResponse([result], 'post', mpData);\n\n          return result;\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError([model], 'post', mpData);\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  /**\n   * Create multiple new Model instances in database\n   */\n  postBatch(models: Model[]): Observable<Model[]> {\n    this.handleAccessError('postBatch');\n\n    return this.http.post(this.apiUrl, classToPlain(models))\n      .pipe(\n        map((response: BaseModel[]) => {\n          const result: Model[] = response.map((item: BaseModel): Model => this.createModelInstance(item));\n          this.handleResponse(result, 'post');\n\n          return result;\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError(models, 'post');\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  /**\n   * Change an existing Model instance in database\n   */\n  put(model: Model, mpData?: AbstractModel, path = `${this.apiUrl}/${model.id}`): Observable<Model> {\n    this.handleAccessError('put');\n\n    return this.http.put(path, classToPlain(model))\n      .pipe(\n        map((response: BaseModel) => {\n          const result: Model = this.createModelInstance(response);\n          this.handleResponse([result], 'put', mpData);\n\n          return result;\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError([model], 'put', mpData);\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  /**\n   * Change multiple existing Model instances in database\n   */\n  putBatch(models: Model[]): Observable<Model[]> {\n    this.handleAccessError('putBatch');\n\n    return this.http.put(this.apiUrl, classToPlain(models))\n      .pipe(\n        map((response: BaseModel[]) => {\n          const result: Model[] = response.map((item: BaseModel): Model => this.createModelInstance(item));\n          this.handleResponse(result, 'put');\n\n          return result;\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError(models, 'put');\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  /**\n   * Remove a Model instance from database\n   */\n  delete(model: Model, mpData?: AbstractModel): Observable<void> {\n    this.handleAccessError('delete');\n\n    return this.http.delete(`${this.apiUrl}/${model.id}`)\n      .pipe(\n        map(() => {\n          this.handleResponse([model], 'delete', mpData);\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError([model], 'delete', mpData);\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  /**\n   * Remove multiple Model instances from database\n   */\n  deleteBatch(models: Model[]): Observable<void> {\n    this.handleAccessError('deleteBatch');\n\n    return this.http.post(`${this.apiUrl}/delete`, classToPlain(models))\n      .pipe(\n        map(() => {\n          this.handleResponse(models, 'delete');\n        }),\n        catchError((error: HttpErrorResponse) => {\n          this.handleResponseError(models, 'delete');\n\n          return observableThrowError(error);\n        })\n      );\n  }\n\n  protected track(response: Model[], method: HttpMethod, mpData?: AbstractModel, postfix = '') {\n    // check if result exist, sometimes response may be empty (for example ClientInvite accept/reject/etc)\n    if (!response[0]) {\n      return;\n    }\n\n    // no other way to check interface\n    if ('getMpData' in response[0]) {\n      response.forEach((model) => {\n        this.mpService.track(`${method}${this.modelClass.name}${postfix}`, model['getMpData'](mpData))\n      });\n    }\n  }\n\n  protected handleResponseError(response: Model[], method: HttpMethod, mpData?: AbstractModel) {\n    // @TODO error notification?\n    this.track(response, method, mpData, 'Error');\n  }\n\n  /**\n   * Handle response data - update cache and dispatch event if it is needed\n   */\n  protected handleResponse(response: Model[], method: HttpMethod, mpData?: AbstractModel) {\n    this.track(response, method, mpData);\n\n    if (this.getCache() && !this.disableCache) {\n      this.updateCache(response, method);\n    }\n\n    // dispatch event for interested services\n    if (response[0] instanceof ObservableModel) {\n      this.dispatchEvent(method, response);\n    }\n  }\n\n  /**\n   * Update cache with passed items. Add/Update detects automatically, Delete via optional flag\n   */\n  protected updateCache(items: Model[], method: HttpMethod): void {\n    let cache: CollectionModel;\n\n    switch (method) {\n      case 'post':\n        cache = this.cache.push(...items);\n        break;\n\n      case 'put':\n        // @TODO Alex old instance\n        items.forEach((item: Model) => {\n          cache = this.cache.replaceBy('id', item.id, item);\n        });\n        break;\n\n      case 'delete':\n        cache = this.cache.removeBy('id', items.map((item: Model) => item.id));\n        break;\n\n      case 'get':\n        cache = this.createCollectionInstance(this.collectionClass, items);\n    }\n\n    this.setCache(cache.toArray(), true);\n  }\n\n  /**\n   * Generate and dispatch rest event\n   */\n  protected dispatchEvent(method: HttpMethod, items: Model[]): void {\n    const eventName: string = (this.modelClass as typeof ObservableModel).getEventName(method);\n    this.eventDispatcherService.dispatch2(new AppEvent2(eventName, items));\n  }\n\n  /**\n   * Check if method is not disabled. Throw exception otherwise.\n   * Some entities does not have endpoints for all methods.\n   */\n  private handleAccessError(method: RestMethod): void {\n    if (!this.disabledMethods.includes(method)) {\n      return;\n    }\n\n    throw new Error(`Method ${method}() is not allowed for ${this.constructor.name}`);\n  }\n\n  /**\n   * Subscribe to http events and run callback.\n   * CSE - Cleint Sent Events\n   * @param modelClass\n   * @param methods The list of http methods should be listened for\n   * @param callback The function to be called when event triggered\n   */\n  protected listenCSE(modelClass: typeof ObservableModel, methods: HttpMethod[], callback: Function): void {\n    callback = callback.bind(this);\n\n    methods.forEach((method: HttpMethod) => {\n      this.eventDispatcherService.on2(modelClass.getEventName(method)).subscribe((data: ObservableModel[]) => {\n        callback(data);\n      });\n    })\n  }\n\n  /**\n   * check if user has subscription role to access api\n   */\n  hasRoles(): boolean {\n    if (!this.roles.length) {\n      return true;\n    }\n\n    return !!intersection(JSON.parse(localStorage.getItem('roles')), this.roles).length;\n  }\n\n  /**\n   * // @TODO Alex remove\n   * Method that call all listeners. Empty by default. Should be redefined by child services if required\n   */\n  listenEvents(): void {}\n\n  /**\n   * Listen SSE events\n   */\n  listenSSE(): void {}\n}\n"]}