From ab81f4be4092c207353679dfa5318c97b7eb09af Mon Sep 17 00:00:00 2001 From: Ahmed Bouhuolia Date: Thu, 19 Mar 2020 17:02:37 +0200 Subject: [PATCH] WIP --- .env | 2 +- client/package-lock.json | 1402 +++++++++++++++++ client/src/components/DataTable.js | 6 +- client/src/config/sidebarMenu.js | 4 + .../FinancialStatements.connector.js | 21 +- .../src/connectors/ProfitLossSheet.connect.js | 23 + .../connectors/TrialBalanceSheet.connect.js | 21 + .../FinancialStatements/BalanceSheet.js | 48 - .../BalanceSheet/BalanceSheet.js | 86 + .../BalanceSheet/BalanceSheetHeader.js | 134 +- .../BalanceSheet/BalanceSheetTable.js | 111 +- .../GeneralLedger.js} | 0 .../GeneralLedger/GeneralLedgerHeader.js | 0 .../GeneralLedger/GeneralLedgerTable.js | 0 .../ProfitLossSheet/ProfitLossSheet.js | 78 + .../ProfitLossSheet/ProfitLossSheetHeader.js | 227 +++ .../ProfitLossSheet/ProfitLossSheetTable.js | 0 .../TrialBalanceSheet/TrialBalanceSheet.js | 75 + .../TrialBalanceSheetHeader.js | 113 ++ .../TrialBalanceSheetTable.js | 80 + client/src/routes/dashboard.js | 23 +- .../financialStatements.actions.js | 25 + .../financialStatements.reducer.js | 33 +- .../financialStatements.selectors.js | 91 ++ .../financialStatements.types.js | 1 + client/src/utils.js | 37 +- server/.env | 2 +- server/.env.test | 2 +- server/dist/bundle.js | 322 +++- server/package-lock.json | 117 +- server/package.json | 2 + server/src/app.js | 15 + server/src/database/manager.js | 17 + ...90822214244_create_user_has_roles_table.js | 4 +- ...0190822214247_create_oauth_tokens_table.js | 2 +- .../20190822214302_create_settings_table.js | 2 +- ...0822214905_create_resource_fields_table.js | 2 +- ...20190822214905_create_role_has_accounts.js | 4 +- ...90822214905_create_role_has_permissions.js | 6 +- ...20190822214905_create_views_roles_table.js | 2 +- ...2647_create_accounts_transactions_table.js | 4 +- .../20200105014405_create_expenses_table.js | 6 +- ...0105195823_create_manual_journals_table.js | 2 +- ...00120145342_create_budget_entries_table.js | 4 +- ...e_resource_custom_fields_metadata_table.js | 4 +- server/src/database/seeds/account_types.js | 9 + server/src/http/controllers/Accounting.js | 38 +- server/src/http/controllers/Customers.js | 61 + server/src/http/controllers/Expenses.js | 6 + .../http/controllers/FinancialStatements.js | 362 +++-- server/src/models/Account.js | 5 + .../src/services/Accounting/JournalPoster.js | 7 +- server/src/utils/index.js | 7 + server/tests/routes/accounting.test.js | 119 +- .../tests/routes/financial_statements.test.js | 464 ++++-- server/tests/routes/views.test.js | 2 +- server/tests/testInit.js | 9 +- 57 files changed, 3734 insertions(+), 515 deletions(-) create mode 100644 client/src/connectors/ProfitLossSheet.connect.js create mode 100644 client/src/connectors/TrialBalanceSheet.connect.js delete mode 100644 client/src/containers/Dashboard/FinancialStatements/BalanceSheet.js create mode 100644 client/src/containers/Dashboard/FinancialStatements/BalanceSheet/BalanceSheet.js rename client/src/containers/Dashboard/FinancialStatements/{TrialBalanceSheet.js => GeneralLedger/GeneralLedger.js} (100%) create mode 100644 client/src/containers/Dashboard/FinancialStatements/GeneralLedger/GeneralLedgerHeader.js create mode 100644 client/src/containers/Dashboard/FinancialStatements/GeneralLedger/GeneralLedgerTable.js create mode 100644 client/src/containers/Dashboard/FinancialStatements/ProfitLossSheet/ProfitLossSheet.js create mode 100644 client/src/containers/Dashboard/FinancialStatements/ProfitLossSheet/ProfitLossSheetHeader.js create mode 100644 client/src/containers/Dashboard/FinancialStatements/ProfitLossSheet/ProfitLossSheetTable.js create mode 100644 client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheet.js create mode 100644 client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheetHeader.js create mode 100644 client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheetTable.js create mode 100644 client/src/store/financialStatement/financialStatements.selectors.js create mode 100644 server/src/database/manager.js diff --git a/.env b/.env index e5b4a00f9..56cbd483a 100644 --- a/.env +++ b/.env @@ -10,5 +10,5 @@ MAIL_FROM_NAME= DB_CLIENT=mysql DB_HOST=127.0.0.1 DB_USER=root -DB_PASSWORD=123123123 +DB_PASSWORD=root DB_NAME=ratteb diff --git a/client/package-lock.json b/client/package-lock.json index d0046898c..1da8e3346 100644 --- a/client/package-lock.json +++ b/client/package-lock.json @@ -5504,6 +5504,44 @@ } } }, + "expand-range": { + "version": "1.8.2", + "resolved": "https://registry.npmjs.org/expand-range/-/expand-range-1.8.2.tgz", + "integrity": "sha1-opnv/TNf4nIeuujiV+x5ZE/IUzc=", + "requires": { + "fill-range": "^2.1.0" + }, + "dependencies": { + "fill-range": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-2.2.4.tgz", + "integrity": "sha512-cnrcCbj01+j2gTG921VZPnHbjmdAf8oQV/iGeV2kZxGSyfYjjTyY79ErsK1WJWMpw6DaApEX72binqJE+/d+5Q==", + "requires": { + "is-number": "^2.1.0", + "isobject": "^2.0.0", + "randomatic": "^3.0.0", + "repeat-element": "^1.1.2", + "repeat-string": "^1.5.2" + } + }, + "is-number": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-2.1.0.tgz", + "integrity": "sha1-Afy7s5NGOlSPL0ZszhbezknbkI8=", + "requires": { + "kind-of": "^3.0.2" + } + }, + "isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", + "requires": { + "isarray": "1.0.0" + } + } + } + }, "expect": { "version": "24.9.0", "resolved": "https://registry.npmjs.org/expect/-/expect-24.9.0.tgz", @@ -5808,6 +5846,11 @@ "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", "optional": true }, + "filename-regex": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/filename-regex/-/filename-regex-2.0.1.tgz", + "integrity": "sha1-wcS5vuPglyXdsQa3XB4wH+LxiyY=" + }, "filesize": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/filesize/-/filesize-6.0.1.tgz", @@ -6268,6 +6311,38 @@ "path-is-absolute": "^1.0.0" } }, + "glob-base": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/glob-base/-/glob-base-0.3.0.tgz", + "integrity": "sha1-27Fk9iIbHAscz4Kuoyi0l98Oo8Q=", + "requires": { + "glob-parent": "^2.0.0", + "is-glob": "^2.0.0" + }, + "dependencies": { + "glob-parent": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-2.0.0.tgz", + "integrity": "sha1-gTg9ctsFT8zPUzbaqQLxgvbtuyg=", + "requires": { + "is-glob": "^2.0.0" + } + }, + "is-extglob": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-1.0.0.tgz", + "integrity": "sha1-rEaBd8SUNAWgkvyPKXYMb/xiBsA=" + }, + "is-glob": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz", + "integrity": "sha1-0Jb5JqPe1WAPP9/ZEZjLCIjC2GM=", + "requires": { + "is-extglob": "^1.0.0" + } + } + } + }, "glob-parent": { "version": "5.1.0", "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.0.tgz", @@ -6943,6 +7018,11 @@ "ipaddr.js": "^1.9.0" } }, + "interpret": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.2.0.tgz", + "integrity": "sha512-mT34yGKMNceBQUoVn7iCDKDntA7SC6gycMAWzGx1z/CMCTV7b2AAtXlo3nRyHZ1FelRkQbQjprHSYGwzLtkVbw==" + }, "intl-format-cache": { "version": "4.2.21", "resolved": "https://registry.npmjs.org/intl-format-cache/-/intl-format-cache-4.2.21.tgz", @@ -7100,6 +7180,19 @@ "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.0.0.tgz", "integrity": "sha512-pJEdRugimx4fBMra5z2/5iRdZ63OhYV0vr0Dwm5+xtW4D1FvRkB8hamMIhnWfyJeDdyr/aa7BDyNbtG38VxgoQ==" }, + "is-dotfile": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/is-dotfile/-/is-dotfile-1.0.3.tgz", + "integrity": "sha1-pqLzL/0t+wT1yiXs0Pa4PPeYoeE=" + }, + "is-equal-shallow": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/is-equal-shallow/-/is-equal-shallow-0.1.3.tgz", + "integrity": "sha1-IjgJj8Ih3gvPpdnqxMRdY4qhxTQ=", + "requires": { + "is-primitive": "^2.0.0" + } + }, "is-extendable": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", @@ -7180,6 +7273,16 @@ "isobject": "^3.0.1" } }, + "is-posix-bracket": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/is-posix-bracket/-/is-posix-bracket-0.1.1.tgz", + "integrity": "sha1-MzTceXdDaOkvAW5vvAqI9c1ua8Q=" + }, + "is-primitive": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-primitive/-/is-primitive-2.0.0.tgz", + "integrity": "sha1-IHurkWOEmcB7Kt8kCkGochADRXU=" + }, "is-promise": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz", @@ -8677,6 +8780,11 @@ "resolved": "https://registry.npmjs.org/lodash._reinterpolate/-/lodash._reinterpolate-3.0.0.tgz", "integrity": "sha1-DM8tiRZq8Ds2Y8eWU4t1rG4RTZ0=" }, + "lodash.assign": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/lodash.assign/-/lodash.assign-4.2.0.tgz", + "integrity": "sha1-DZnzzNem0mHRm9rrkkUAXShYCOc=" + }, "lodash.memoize": { "version": "4.1.2", "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", @@ -8811,6 +8919,11 @@ "object-visit": "^1.0.0" } }, + "math-random": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/math-random/-/math-random-1.0.4.tgz", + "integrity": "sha512-rUxjysqif/BZQH2yhd5Aaq7vXMSx9NdEsQcyA07uEzIvxgI7zIr33gGsh+RU0/XjmQpCW7RsVof1vlkvQVCK5A==" + }, "md5.js": { "version": "1.3.5", "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz", @@ -9251,6 +9364,1215 @@ } } }, + "mocha-webpack": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/mocha-webpack/-/mocha-webpack-1.1.0.tgz", + "integrity": "sha512-brmE0tR6G5JbEzZXspJmF/uZm2J/YM/69M3VS6ND76i6wXbebFpE+bQDaehilK8CZanNSsTCcqTTLh1PZuRKJw==", + "requires": { + "babel-runtime": "^6.18.0", + "chalk": "^2.3.0", + "chokidar": "^1.6.1", + "glob-parent": "^3.1.0", + "globby": "^6.1.0", + "interpret": "^1.0.1", + "is-glob": "^4.0.0", + "loader-utils": "^1.1.0", + "lodash": "^4.3.0", + "memory-fs": "^0.4.1", + "nodent-runtime": "^3.0.3", + "normalize-path": "^2.0.1", + "progress": "^2.0.0", + "source-map-support": "^0.5.0", + "strip-ansi": "^4.0.0", + "toposort": "^1.0.0", + "yargs": "^4.8.0" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=" + }, + "anymatch": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-1.3.2.tgz", + "integrity": "sha512-0XNayC8lTHQ2OI8aljNCN3sSx6hsr/1+rlcDAotXJR7C1oZZHCNsfpbKwMjRA3Uqb5tF1Rae2oloTr4xpq+WjA==", + "requires": { + "micromatch": "^2.1.5", + "normalize-path": "^2.0.0" + } + }, + "arr-diff": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-2.0.0.tgz", + "integrity": "sha1-jzuCf5Vai9ZpaX5KQlasPOrjVs8=", + "requires": { + "arr-flatten": "^1.0.1" + } + }, + "array-unique": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.2.1.tgz", + "integrity": "sha1-odl8yvy8JiXMcPrc6zalDFiwGlM=" + }, + "binary-extensions": { + "version": "1.13.1", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", + "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==" + }, + "braces": { + "version": "1.8.5", + "resolved": "https://registry.npmjs.org/braces/-/braces-1.8.5.tgz", + "integrity": "sha1-uneWLhLf+WnWt2cR6RS3N4V79qc=", + "requires": { + "expand-range": "^1.8.1", + "preserve": "^0.2.0", + "repeat-element": "^1.1.2" + } + }, + "camelcase": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-3.0.0.tgz", + "integrity": "sha1-MvxLn82vhF/N9+c7uXysImHwqwo=" + }, + "chokidar": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-1.7.0.tgz", + "integrity": "sha1-eY5ol3gVHIB2tLNg5e3SjNortGg=", + "requires": { + "anymatch": "^1.3.0", + "async-each": "^1.0.0", + "fsevents": "^1.0.0", + "glob-parent": "^2.0.0", + "inherits": "^2.0.1", + "is-binary-path": "^1.0.0", + "is-glob": "^2.0.0", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.0.0" + }, + "dependencies": { + "glob-parent": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-2.0.0.tgz", + "integrity": "sha1-gTg9ctsFT8zPUzbaqQLxgvbtuyg=", + "requires": { + "is-glob": "^2.0.0" + } + }, + "is-glob": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz", + "integrity": "sha1-0Jb5JqPe1WAPP9/ZEZjLCIjC2GM=", + "requires": { + "is-extglob": "^1.0.0" + } + } + } + }, + "cliui": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-3.2.0.tgz", + "integrity": "sha1-EgYBU3qRbSmUD5NNo7SNWFo5IT0=", + "requires": { + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wrap-ansi": "^2.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "requires": { + "ansi-regex": "^2.0.0" + } + } + } + }, + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + } + }, + "expand-brackets": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-0.1.5.tgz", + "integrity": "sha1-3wcoTjQqgHzXM6xa9yQR5YHRF3s=", + "requires": { + "is-posix-bracket": "^0.1.0" + } + }, + "extglob": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/extglob/-/extglob-0.3.2.tgz", + "integrity": "sha1-Lhj/PS9JqydlzskCPwEdqo2DSaE=", + "requires": { + "is-extglob": "^1.0.0" + } + }, + "find-up": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-1.1.2.tgz", + "integrity": "sha1-ay6YIrGizgpgq2TWEOzK1TyyTQ8=", + "requires": { + "path-exists": "^2.0.0", + "pinkie-promise": "^2.0.0" + } + }, + "fsevents": { + "version": "1.2.11", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.11.tgz", + "integrity": "sha512-+ux3lx6peh0BpvY0JebGyZoiR4D+oYzdPZMKJwkZ+sFkNJzpL7tXc/wehS49gUAxg3tmMHPHZkA8JU2rhhgDHw==", + "optional": true, + "requires": { + "bindings": "^1.5.0", + "nan": "^2.12.1", + "node-pre-gyp": "*" + }, + "dependencies": { + "abbrev": { + "version": "1.1.1", + "bundled": true, + "optional": true + }, + "ansi-regex": { + "version": "2.1.1", + "bundled": true, + "optional": true + }, + "aproba": { + "version": "1.2.0", + "bundled": true, + "optional": true + }, + "are-we-there-yet": { + "version": "1.1.5", + "bundled": true, + "optional": true, + "requires": { + "delegates": "^1.0.0", + "readable-stream": "^2.0.6" + } + }, + "balanced-match": { + "version": "1.0.0", + "bundled": true, + "optional": true + }, + "brace-expansion": { + "version": "1.1.11", + "bundled": true, + "optional": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "chownr": { + "version": "1.1.3", + "bundled": true, + "optional": true + }, + "code-point-at": { + "version": "1.1.0", + "bundled": true, + "optional": true + }, + "concat-map": { + "version": "0.0.1", + "bundled": true, + "optional": true + }, + "console-control-strings": { + "version": "1.1.0", + "bundled": true, + "optional": true + }, + "core-util-is": { + "version": "1.0.2", + "bundled": true, + "optional": true + }, + "debug": { + "version": "3.2.6", + "bundled": true, + "optional": true, + "requires": { + "ms": "^2.1.1" + } + }, + "deep-extend": { + "version": "0.6.0", + "bundled": true, + "optional": true + }, + "delegates": { + "version": "1.0.0", + "bundled": true, + "optional": true + }, + "detect-libc": { + "version": "1.0.3", + "bundled": true, + "optional": true + }, + "fs-minipass": { + "version": "1.2.7", + "bundled": true, + "optional": true, + "requires": { + "minipass": "^2.6.0" + } + }, + "fs.realpath": { + "version": "1.0.0", + "bundled": true, + "optional": true + }, + "gauge": { + "version": "2.7.4", + "bundled": true, + "optional": true, + "requires": { + "aproba": "^1.0.3", + "console-control-strings": "^1.0.0", + "has-unicode": "^2.0.0", + "object-assign": "^4.1.0", + "signal-exit": "^3.0.0", + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wide-align": "^1.1.0" + } + }, + "glob": { + "version": "7.1.6", + "bundled": true, + "optional": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "has-unicode": { + "version": "2.0.1", + "bundled": true, + "optional": true + }, + "iconv-lite": { + "version": "0.4.24", + "bundled": true, + "optional": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + }, + "ignore-walk": { + "version": "3.0.3", + "bundled": true, + "optional": true, + "requires": { + "minimatch": "^3.0.4" + } + }, + "inflight": { + "version": "1.0.6", + "bundled": true, + "optional": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.4", + "bundled": true, + "optional": true + }, + "ini": { + "version": "1.3.5", + "bundled": true, + "optional": true + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "bundled": true, + "optional": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "isarray": { + "version": "1.0.0", + "bundled": true, + "optional": true + }, + "minimatch": { + "version": "3.0.4", + "bundled": true, + "optional": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "minimist": { + "version": "0.0.8", + "bundled": true, + "optional": true + }, + "minipass": { + "version": "2.9.0", + "bundled": true, + "optional": true, + "requires": { + "safe-buffer": "^5.1.2", + "yallist": "^3.0.0" + } + }, + "minizlib": { + "version": "1.3.3", + "bundled": true, + "optional": true, + "requires": { + "minipass": "^2.9.0" + } + }, + "mkdirp": { + "version": "0.5.1", + "bundled": true, + "optional": true, + "requires": { + "minimist": "0.0.8" + } + }, + "ms": { + "version": "2.1.2", + "bundled": true, + "optional": true + }, + "needle": { + "version": "2.4.0", + "bundled": true, + "optional": true, + "requires": { + "debug": "^3.2.6", + "iconv-lite": "^0.4.4", + "sax": "^1.2.4" + } + }, + "node-pre-gyp": { + "version": "0.14.0", + "bundled": true, + "optional": true, + "requires": { + "detect-libc": "^1.0.2", + "mkdirp": "^0.5.1", + "needle": "^2.2.1", + "nopt": "^4.0.1", + "npm-packlist": "^1.1.6", + "npmlog": "^4.0.2", + "rc": "^1.2.7", + "rimraf": "^2.6.1", + "semver": "^5.3.0", + "tar": "^4.4.2" + } + }, + "nopt": { + "version": "4.0.1", + "bundled": true, + "optional": true, + "requires": { + "abbrev": "1", + "osenv": "^0.1.4" + } + }, + "npm-bundled": { + "version": "1.1.1", + "bundled": true, + "optional": true, + "requires": { + "npm-normalize-package-bin": "^1.0.1" + } + }, + "npm-normalize-package-bin": { + "version": "1.0.1", + "bundled": true, + "optional": true + }, + "npm-packlist": { + "version": "1.4.7", + "bundled": true, + "optional": true, + "requires": { + "ignore-walk": "^3.0.1", + "npm-bundled": "^1.0.1" + } + }, + "npmlog": { + "version": "4.1.2", + "bundled": true, + "optional": true, + "requires": { + "are-we-there-yet": "~1.1.2", + "console-control-strings": "~1.1.0", + "gauge": "~2.7.3", + "set-blocking": "~2.0.0" + } + }, + "number-is-nan": { + "version": "1.0.1", + "bundled": true, + "optional": true + }, + "object-assign": { + "version": "4.1.1", + "bundled": true, + "optional": true + }, + "once": { + "version": "1.4.0", + "bundled": true, + "optional": true, + "requires": { + "wrappy": "1" + } + }, + "os-homedir": { + "version": "1.0.2", + "bundled": true, + "optional": true + }, + "os-tmpdir": { + "version": "1.0.2", + "bundled": true, + "optional": true + }, + "osenv": { + "version": "0.1.5", + "bundled": true, + "optional": true, + "requires": { + "os-homedir": "^1.0.0", + "os-tmpdir": "^1.0.0" + } + }, + "path-is-absolute": { + "version": "1.0.1", + "bundled": true, + "optional": true + }, + "process-nextick-args": { + "version": "2.0.1", + "bundled": true, + "optional": true + }, + "rc": { + "version": "1.2.8", + "bundled": true, + "optional": true, + "requires": { + "deep-extend": "^0.6.0", + "ini": "~1.3.0", + "minimist": "^1.2.0", + "strip-json-comments": "~2.0.1" + }, + "dependencies": { + "minimist": { + "version": "1.2.0", + "bundled": true, + "optional": true + } + } + }, + "readable-stream": { + "version": "2.3.6", + "bundled": true, + "optional": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "rimraf": { + "version": "2.7.1", + "bundled": true, + "optional": true, + "requires": { + "glob": "^7.1.3" + } + }, + "safe-buffer": { + "version": "5.1.2", + "bundled": true, + "optional": true + }, + "safer-buffer": { + "version": "2.1.2", + "bundled": true, + "optional": true + }, + "sax": { + "version": "1.2.4", + "bundled": true, + "optional": true + }, + "semver": { + "version": "5.7.1", + "bundled": true, + "optional": true + }, + "set-blocking": { + "version": "2.0.0", + "bundled": true, + "optional": true + }, + "signal-exit": { + "version": "3.0.2", + "bundled": true, + "optional": true + }, + "string-width": { + "version": "1.0.2", + "bundled": true, + "optional": true, + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + }, + "string_decoder": { + "version": "1.1.1", + "bundled": true, + "optional": true, + "requires": { + "safe-buffer": "~5.1.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "bundled": true, + "optional": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "strip-json-comments": { + "version": "2.0.1", + "bundled": true, + "optional": true + }, + "tar": { + "version": "4.4.13", + "bundled": true, + "optional": true, + "requires": { + "chownr": "^1.1.1", + "fs-minipass": "^1.2.5", + "minipass": "^2.8.6", + "minizlib": "^1.2.1", + "mkdirp": "^0.5.0", + "safe-buffer": "^5.1.2", + "yallist": "^3.0.3" + } + }, + "util-deprecate": { + "version": "1.0.2", + "bundled": true, + "optional": true + }, + "wide-align": { + "version": "1.1.3", + "bundled": true, + "optional": true, + "requires": { + "string-width": "^1.0.2 || 2" + } + }, + "wrappy": { + "version": "1.0.2", + "bundled": true, + "optional": true + }, + "yallist": { + "version": "3.1.1", + "bundled": true, + "optional": true + } + } + }, + "get-caller-file": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-1.0.3.tgz", + "integrity": "sha512-3t6rVToeoZfYSGd8YoLFR2DJkiQrIiUrGcjvFX2mDw3bn6k2OtwHN0TNCLbBO+w8qTvimhDkv+LSscbJY1vE6w==" + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=" + }, + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "globby": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-6.1.0.tgz", + "integrity": "sha1-9abXDoOV4hyFj7BInWTfAkJNUGw=", + "requires": { + "array-union": "^1.0.1", + "glob": "^7.0.3", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + } + }, + "invert-kv": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-1.0.0.tgz", + "integrity": "sha1-EEqOSqym09jNFXqO+L+rLXo//bY=" + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "requires": { + "kind-of": "^6.0.0" + }, + "dependencies": { + "kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==" + } + } + }, + "is-binary-path": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", + "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", + "requires": { + "binary-extensions": "^1.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "requires": { + "kind-of": "^6.0.0" + }, + "dependencies": { + "kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==" + } + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "dependencies": { + "kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==" + } + } + }, + "is-extglob": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-1.0.0.tgz", + "integrity": "sha1-rEaBd8SUNAWgkvyPKXYMb/xiBsA=" + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "lcid": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/lcid/-/lcid-1.0.0.tgz", + "integrity": "sha1-MIrMr6C8SDo4Z7S28rlQYlHRuDU=", + "requires": { + "invert-kv": "^1.0.0" + } + }, + "load-json-file": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-1.1.0.tgz", + "integrity": "sha1-lWkFcI1YtLq0wiYbBPWfMcmTdMA=", + "requires": { + "graceful-fs": "^4.1.2", + "parse-json": "^2.2.0", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0", + "strip-bom": "^2.0.0" + } + }, + "micromatch": { + "version": "2.3.11", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-2.3.11.tgz", + "integrity": "sha1-hmd8l9FyCzY0MdBNDRUpO9OMFWU=", + "requires": { + "arr-diff": "^2.0.0", + "array-unique": "^0.2.1", + "braces": "^1.8.2", + "expand-brackets": "^0.1.4", + "extglob": "^0.3.1", + "filename-regex": "^2.0.0", + "is-extglob": "^1.0.0", + "is-glob": "^2.0.1", + "kind-of": "^3.0.2", + "normalize-path": "^2.0.1", + "object.omit": "^2.0.0", + "parse-glob": "^3.0.4", + "regex-cache": "^0.4.2" + }, + "dependencies": { + "is-glob": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz", + "integrity": "sha1-0Jb5JqPe1WAPP9/ZEZjLCIjC2GM=", + "requires": { + "is-extglob": "^1.0.0" + } + } + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "os-locale": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-1.4.0.tgz", + "integrity": "sha1-IPnxeuKe00XoveWDsT0gCYA8FNk=", + "requires": { + "lcid": "^1.0.0" + } + }, + "parse-json": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", + "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=", + "requires": { + "error-ex": "^1.2.0" + } + }, + "path-exists": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-2.1.0.tgz", + "integrity": "sha1-D+tsZPD8UY2adU3V77YscCJ2H0s=", + "requires": { + "pinkie-promise": "^2.0.0" + } + }, + "path-type": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-1.1.0.tgz", + "integrity": "sha1-WcRPfuSR2nBNpBXaWkBwuk+P5EE=", + "requires": { + "graceful-fs": "^4.1.2", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + } + }, + "pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=" + }, + "read-pkg": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-1.1.0.tgz", + "integrity": "sha1-9f+qXs0pyzHAR0vKfXVra7KePyg=", + "requires": { + "load-json-file": "^1.0.0", + "normalize-package-data": "^2.3.2", + "path-type": "^1.0.0" + } + }, + "read-pkg-up": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-1.0.1.tgz", + "integrity": "sha1-nWPBMnbAZZGNV/ACpX9AobZD+wI=", + "requires": { + "find-up": "^1.0.0", + "read-pkg": "^1.0.0" + } + }, + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "readdirp": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", + "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", + "requires": { + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" + }, + "dependencies": { + "arr-diff": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", + "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=" + }, + "array-unique": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", + "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=" + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "expand-brackets": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", + "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", + "requires": { + "debug": "^2.3.3", + "define-property": "^0.2.5", + "extend-shallow": "^2.0.1", + "posix-character-classes": "^0.1.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "requires": { + "is-extendable": "^0.1.0" + } + }, + "is-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", + "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "requires": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + } + }, + "kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==" + } + } + }, + "extglob": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", + "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", + "requires": { + "array-unique": "^0.3.2", + "define-property": "^1.0.0", + "expand-brackets": "^2.1.4", + "extend-shallow": "^2.0.1", + "fragment-cache": "^0.2.1", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "is-accessor-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", + "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "is-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==" + }, + "micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", + "requires": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" + } + } + } + }, + "require-main-filename": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-1.0.1.tgz", + "integrity": "sha1-l/cXtp1IeE9fUmpsWqj/3aBVpNE=" + }, + "string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "requires": { + "ansi-regex": "^2.0.0" + } + } + } + }, + "string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "requires": { + "safe-buffer": "~5.1.0" + } + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "requires": { + "ansi-regex": "^3.0.0" + } + }, + "strip-bom": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-2.0.0.tgz", + "integrity": "sha1-YhmoVhZSBJHzV4i9vxRHqZx+aw4=", + "requires": { + "is-utf8": "^0.2.0" + } + }, + "toposort": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/toposort/-/toposort-1.0.7.tgz", + "integrity": "sha1-LmhELZ9k7HILjMieZEOsbKqVACk=" + }, + "which-module": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/which-module/-/which-module-1.0.0.tgz", + "integrity": "sha1-u6Y8qGGUiZT/MHc2CJ47lgJsKk8=" + }, + "wrap-ansi": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz", + "integrity": "sha1-2Pw9KE3QV5T+hJc8rs3Rz4JP3YU=", + "requires": { + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "requires": { + "ansi-regex": "^2.0.0" + } + } + } + }, + "y18n": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.1.tgz", + "integrity": "sha1-bRX7qITAhnnA136I53WegR4H+kE=" + }, + "yargs": { + "version": "4.8.1", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-4.8.1.tgz", + "integrity": "sha1-wMQpJMpKqmsObaFznfshZDn53cA=", + "requires": { + "cliui": "^3.2.0", + "decamelize": "^1.1.1", + "get-caller-file": "^1.0.1", + "lodash.assign": "^4.0.3", + "os-locale": "^1.4.0", + "read-pkg-up": "^1.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^1.0.1", + "set-blocking": "^2.0.0", + "string-width": "^1.0.1", + "which-module": "^1.0.0", + "window-size": "^0.2.0", + "y18n": "^3.2.1", + "yargs-parser": "^2.4.1" + } + }, + "yargs-parser": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-2.4.1.tgz", + "integrity": "sha1-hVaN488VD/SfpRgl8DqMiA3cxcQ=", + "requires": { + "camelcase": "^3.0.0", + "lodash.assign": "^4.0.6" + } + } + } + }, "moment": { "version": "2.24.0", "resolved": "https://registry.npmjs.org/moment/-/moment-2.24.0.tgz", @@ -9599,6 +10921,11 @@ } } }, + "nodent-runtime": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/nodent-runtime/-/nodent-runtime-3.2.1.tgz", + "integrity": "sha512-7Ws63oC+215smeKJQCxzrK21VFVlCFBkwl0MOObt0HOpVQXs3u483sAmtkF33nNqZ5rSOQjB76fgyPBmAUrtCA==" + }, "nopt": { "version": "3.0.6", "resolved": "https://registry.npmjs.org/nopt/-/nopt-3.0.6.tgz", @@ -9801,6 +11128,15 @@ "es-abstract": "^1.17.0-next.1" } }, + "object.omit": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/object.omit/-/object.omit-2.0.1.tgz", + "integrity": "sha1-Gpx0SCnznbuFjHbKNXmuKlTr0fo=", + "requires": { + "for-own": "^0.1.4", + "is-extendable": "^0.1.1" + } + }, "object.pick": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", @@ -10082,6 +11418,32 @@ "safe-buffer": "^5.1.1" } }, + "parse-glob": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/parse-glob/-/parse-glob-3.0.4.tgz", + "integrity": "sha1-ssN2z7EfNVE7rdFz7wu246OIORw=", + "requires": { + "glob-base": "^0.3.0", + "is-dotfile": "^1.0.0", + "is-extglob": "^1.0.0", + "is-glob": "^2.0.0" + }, + "dependencies": { + "is-extglob": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-1.0.0.tgz", + "integrity": "sha1-rEaBd8SUNAWgkvyPKXYMb/xiBsA=" + }, + "is-glob": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz", + "integrity": "sha1-0Jb5JqPe1WAPP9/ZEZjLCIjC2GM=", + "requires": { + "is-extglob": "^1.0.0" + } + } + } + }, "parse-json": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", @@ -11219,6 +12581,11 @@ "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz", "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=" }, + "preserve": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/preserve/-/preserve-0.2.0.tgz", + "integrity": "sha1-gV7R9uvGWSb4ZbMQwHE7yzMVzks=" + }, "pretty-bytes": { "version": "5.3.0", "resolved": "https://registry.npmjs.org/pretty-bytes/-/pretty-bytes-5.3.0.tgz", @@ -11422,6 +12789,28 @@ "performance-now": "^2.1.0" } }, + "randomatic": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/randomatic/-/randomatic-3.1.1.tgz", + "integrity": "sha512-TuDE5KxZ0J461RVjrJZCJc+J+zCkTb1MbH9AQUq68sMhOMcy9jLcb3BrZKgp9q9Ncltdg4QVqWrH02W2EFFVYw==", + "requires": { + "is-number": "^4.0.0", + "kind-of": "^6.0.0", + "math-random": "^1.0.1" + }, + "dependencies": { + "is-number": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-4.0.0.tgz", + "integrity": "sha512-rSklcAIlf1OmFdyAqbnWTLVelsQ58uvZ66S/ZyawjWqIviTWCjg2PzVGw8WUA+nNuPTqb4wgA+NszrJ+08LlgQ==" + }, + "kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==" + } + } + }, "randombytes": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", @@ -11952,6 +13341,14 @@ "private": "^0.1.6" } }, + "regex-cache": { + "version": "0.4.4", + "resolved": "https://registry.npmjs.org/regex-cache/-/regex-cache-0.4.4.tgz", + "integrity": "sha512-nVIZwtCjkC9YgvWkpM55B5rBhBYRZhAaJbgcFYXXsHnbZ9UZI9nnVWYZpBlCqv9ho2eZryPnWrZGsOdPwVWXWQ==", + "requires": { + "is-equal-shallow": "^0.1.3" + } + }, "regex-not": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.2.tgz", @@ -16104,6 +17501,11 @@ } } }, + "window-size": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/window-size/-/window-size-0.2.0.tgz", + "integrity": "sha1-tDFbtCFKPXBY6+7okuE/ok2YsHU=" + }, "word-wrap": { "version": "1.2.3", "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", diff --git a/client/src/components/DataTable.js b/client/src/components/DataTable.js index 16c984034..e8e25e85e 100644 --- a/client/src/components/DataTable.js +++ b/client/src/components/DataTable.js @@ -1,5 +1,5 @@ import React from 'react'; -import { useTable, usePagination } from 'react-table' +import { useTable, useExpanded, usePagination } from 'react-table' export default function DataTable({ columns, @@ -32,8 +32,10 @@ export default function DataTable({ // This means we'll also have to provide our own // pageCount. // pageCount: controlledPageCount, + getSubRows: row => row.children, }, - usePagination + useExpanded, + usePagination, ); return ( diff --git a/client/src/config/sidebarMenu.js b/client/src/config/sidebarMenu.js index bee0a5a8d..8e215cc65 100644 --- a/client/src/config/sidebarMenu.js +++ b/client/src/config/sidebarMenu.js @@ -116,6 +116,10 @@ export default [ text: 'Balance Sheet', href: '/dashboard/accounting/balance-sheet', }, + { + text: 'Trial Balance Sheet', + href: '/dashboard/accounting/trial-balance-sheet', + }, { divider: true, }, diff --git a/client/src/connectors/FinancialStatements.connector.js b/client/src/connectors/FinancialStatements.connector.js index 1a40ccc2f..1791e25de 100644 --- a/client/src/connectors/FinancialStatements.connector.js +++ b/client/src/connectors/FinancialStatements.connector.js @@ -3,10 +3,29 @@ import { fetchGeneralLedger, fetchBalanceSheet, } from 'store/financialStatement/financialStatements.actions'; -import t from 'store/types'; +import { + getBalanceSheetByQuery, + getBalanceSheetColumns, + getBalanceSheetIndexByQuery, + getBalanceSheetByIndex, + getBalanceSheetAssetsAccounts, + getBalanceSheetLiabilitiesAccounts, + getBalanceSheetQuery, +} from 'store/financialStatement/financialStatements.selectors'; export const mapStateToProps = (state, props) => ({ generalLedeger: state.financialStatements.generalLedger, + balanceSheets: state.financialStatements.balanceSheets, + + getBalanceSheetByQuery: (query) => getBalanceSheetByQuery(state.financialStatements.balanceSheets, query), + getBalanceSheetColumns: (sheetIndex) => getBalanceSheetColumns(state.financialStatements.balanceSheets, sheetIndex), + getBalanceSheetIndexByQuery: (query) => getBalanceSheetIndexByQuery(state.financialStatements.balanceSheets, query), + getBalanceSheetByIndex: (sheetIndex) => getBalanceSheetByIndex(state.financialStatements.balanceSheets, sheetIndex), + + getBalanceSheetAssetsAccounts: (sheetIndex) => getBalanceSheetAssetsAccounts(state.financialStatements.balanceSheets, sheetIndex), + getBalanceSheetLiabilitiesAccounts: (sheetIndex) => getBalanceSheetLiabilitiesAccounts(state.financialStatements.balanceSheets, sheetIndex), + + getBalanceSheetQuery: (sheetIndex) => getBalanceSheetQuery(state.financialStatements.balanceSheets, sheetIndex), }); export const mapDispatchToProps = (dispatch) => ({ diff --git a/client/src/connectors/ProfitLossSheet.connect.js b/client/src/connectors/ProfitLossSheet.connect.js new file mode 100644 index 000000000..2624086f3 --- /dev/null +++ b/client/src/connectors/ProfitLossSheet.connect.js @@ -0,0 +1,23 @@ +import {connect} from 'react-redux'; +import { + fetchProfitLossSheet, +} from 'store/financialStatement/financialStatements.actions'; +import { + getProfitLossSheetIndex, + getProfitLossSheet, + getProfitLossSheetColumns, + getProfitLossSheetAccounts, +} from 'store/financialStatement/financialStatements.selectors'; + +export const mapStateToProps = (state, props) => ({ + getProfitLossSheetIndex: (query) => getProfitLossSheetIndex(state.financialStatements.profitLossSheets, query), + getProfitLossSheet: (index) => getProfitLossSheet(state.financialStatements.profitLossSheets, index), + getProfitLossSheetColumns: (index) => getProfitLossSheetColumns(state.financialStatements.profitLossSheets, index), + getProfitLossSheetAccounts: (index) => getProfitLossSheetAccounts(state.financialStatements.profitLossSheets, index), +}); + +export const mapDispatchToProps = (dispatch) => ({ + fetchProfitLossSheet: (query = {}) => dispatch(fetchProfitLossSheet({ query })), +}); + +export default connect(mapStateToProps, mapDispatchToProps); \ No newline at end of file diff --git a/client/src/connectors/TrialBalanceSheet.connect.js b/client/src/connectors/TrialBalanceSheet.connect.js new file mode 100644 index 000000000..493bc4003 --- /dev/null +++ b/client/src/connectors/TrialBalanceSheet.connect.js @@ -0,0 +1,21 @@ +import {connect} from 'react-redux'; +import { + fetchTrialBalanceSheet +} from 'store/financialStatement/financialStatements.actions'; +import { + getTrialBalanceSheetIndex, + getTrialBalanceAccounts, + getTrialBalanceQuery, +} from 'store/financialStatement/financialStatements.selectors'; + +export const mapStateToProps = (state, props) => ({ + getTrialBalanceSheetIndex: (query) => getTrialBalanceSheetIndex(state.financialStatements.trialBalanceSheets, query), + getTrialBalanceAccounts: (sheetIndex) => getTrialBalanceAccounts(state.financialStatements.trialBalanceSheets, sheetIndex), + getTrialBalanceQuery: (sheetIndex) => getTrialBalanceQuery(state.financialStatements.trialBalanceSheets, sheetIndex), +}); + +export const mapDispatchToProps = (dispatch) => ({ + fetchTrialBalanceSheet: (query = {}) => dispatch(fetchTrialBalanceSheet({ query })), +}); + +export default connect(mapStateToProps, mapDispatchToProps); \ No newline at end of file diff --git a/client/src/containers/Dashboard/FinancialStatements/BalanceSheet.js b/client/src/containers/Dashboard/FinancialStatements/BalanceSheet.js deleted file mode 100644 index b4c782cc2..000000000 --- a/client/src/containers/Dashboard/FinancialStatements/BalanceSheet.js +++ /dev/null @@ -1,48 +0,0 @@ -import React, {useEffect} from 'react'; -import DashboardConnect from 'connectors/Dashboard.connector'; -import {compose} from 'utils'; -import useAsync from 'hooks/async'; -import FinancialStatementConnect from 'connectors/FinancialStatements.connector'; -import {useIntl} from 'react-intl'; -import BalanceSheetHeader from './BalanceSheet/BalanceSheetHeader'; -import LoadingIndicator from 'components/LoadingIndicator'; -import BalanceSheetTable from './BalanceSheet/BalanceSheetTable'; - -function BalanceSheet({ - fetchBalanceSheet, - changePageTitle, -}) { - const intl = useIntl(); - const handleDateChange = () => {}; - - const fetchHook = useAsync(async () => { - await Promise.all([ - fetchBalanceSheet({}), - ]); - }); - - useEffect(() => { - changePageTitle('Balance Sheet'); - }, []); - - const handleFilterSubmit = (filter) => { - - }; - - return ( -
- - -
- - - -
-
- ); -} - -export default compose( - DashboardConnect, - FinancialStatementConnect, -)(BalanceSheet); \ No newline at end of file diff --git a/client/src/containers/Dashboard/FinancialStatements/BalanceSheet/BalanceSheet.js b/client/src/containers/Dashboard/FinancialStatements/BalanceSheet/BalanceSheet.js new file mode 100644 index 000000000..970d8c235 --- /dev/null +++ b/client/src/containers/Dashboard/FinancialStatements/BalanceSheet/BalanceSheet.js @@ -0,0 +1,86 @@ +import React, {useEffect, useMemo, useState} from 'react'; +import DashboardConnect from 'connectors/Dashboard.connector'; +import {compose} from 'utils'; +import useAsync from 'hooks/async'; +import FinancialStatementConnect from 'connectors/FinancialStatements.connector'; +import {useIntl} from 'react-intl'; +import BalanceSheetHeader from './BalanceSheetHeader'; +import LoadingIndicator from 'components/LoadingIndicator'; +import BalanceSheetTable from './BalanceSheetTable'; +import moment from 'moment'; + +function BalanceSheet({ + fetchBalanceSheet, + changePageTitle, + getBalanceSheetByQuery, + getBalanceSheetIndexByQuery, + getBalanceSheetByIndex, + balanceSheets +}) { + const intl = useIntl(); + const [filter, setFilter] = useState({ + from_date: moment().startOf('year').format('YYYY-MM-DD'), + to_date: moment().endOf('year').format('YYYY-MM-DD'), + basis: 'cash', + display_columns_by: 'total', + none_zero: false, + }); + + const [reload, setReload] = useState(false); + + const fetchHook = useAsync(async () => { + await Promise.all([ + fetchBalanceSheet(filter), + ]); + setReload(false); + }); + + useEffect(() => { + if (!reload) { return; } + fetchHook.execute(); + }, [reload]); + + useEffect(() => { + changePageTitle('Balance Sheet'); + }, []); + + // Retrieve balance sheet index by the given filter query. + const balanceSheetIndex = useMemo(() => { + return getBalanceSheetIndexByQuery(filter); + }, [filter, balanceSheets]); + + // Retreive balance sheet by the given sheet index. + const balanceSheet = useMemo(() => { + return getBalanceSheetByIndex(balanceSheetIndex); + }, [balanceSheetIndex, balanceSheets]); + + // Handle re-fetch balance sheet after filter change. + const handleFilterSubmit = (filter) => { + setFilter({ + ...filter, + from_date: moment(filter.from_date).format('YYYY-MM-DD'), + to_date: moment(filter.to_date).format('YYYY-MM-DD'), + }); + setReload(true); + }; + return ( +
+ + +
+ + + +
+
+ ); +} + +export default compose( + DashboardConnect, + FinancialStatementConnect, +)(BalanceSheet); \ No newline at end of file diff --git a/client/src/containers/Dashboard/FinancialStatements/BalanceSheet/BalanceSheetHeader.js b/client/src/containers/Dashboard/FinancialStatements/BalanceSheet/BalanceSheetHeader.js index cfa083112..595655f31 100644 --- a/client/src/containers/Dashboard/FinancialStatements/BalanceSheet/BalanceSheetHeader.js +++ b/client/src/containers/Dashboard/FinancialStatements/BalanceSheet/BalanceSheetHeader.js @@ -1,4 +1,4 @@ -import React, {useState, useMemo} from 'react'; +import React, {useState, useMemo, useEffect} from 'react'; import FinancialStatementHeader from 'containers/Dashboard/FinancialStatements/FinancialStatementHeader'; import {Row, Col} from 'react-grid-system'; import { @@ -10,44 +10,53 @@ import { Radio, HTMLSelect, Intent, + Popover, } from "@blueprintjs/core"; import {Select} from '@blueprintjs/select'; import {DateInput} from '@blueprintjs/datetime'; import {useIntl} from 'react-intl'; -import {momentFormatter, handleStringChange} from 'utils'; +import { + momentFormatter, + handleStringChange, + parseDateRangeQuery, +} from 'utils'; import moment from 'moment'; export default function BalanceSheetHeader({ onSubmitFilter, + pageFilter, }) { const intl = useIntl(); - const [filter, setFilter] = useState({ - from_date: null, - to_date: null, - accounting_basis: 'cash', - display_columns_by: 'total', - }); - - const setFilterByName = (name, value) => { - setFilter({ - ...filter, - [name]: value, - }); - }; - - const handleFieldChange = (event) => { - setFilterByName(event.target.name, event.target.value); - }; - const displayColumnsByOptions = [ {key: 'total', name: 'Total'}, {key: 'year', name: 'Year'}, {key: 'month', name: 'Month'}, {key: 'week', name: 'Week'}, {key: 'day', name: 'Day'}, - {key: 'quarter', name: 'Quarter'} + {key: 'quarter', name: 'Quarter'}, ]; + const [filter, setFilter] = useState({ + ...pageFilter, + from_date: moment(pageFilter.from_date).toDate(), + to_date: moment(pageFilter.to_date).toDate() + }); + + const setFilterByKey = (name, value) => { + setFilter({ ...filter, [name]: value }); + }; + + const [reportDateRange, setReportDateRange] = useState('this_year'); + + useEffect(() => { + if (reportDateRange === 'custom') { return; } + const dateRange = parseDateRangeQuery(reportDateRange); + + if (dateRange) { + setFilter((filter) => ({ ...filter, ...dateRange, })); + } + }, [reportDateRange]) + const selectedDisplayColumnOpt = useMemo(() => { return displayColumnsByOptions.find(o => o.key === filter.display_columns_by); }, [filter.display_columns_by, displayColumnsByOptions]); @@ -56,31 +65,56 @@ export default function BalanceSheetHeader({ const accountTypeItem = (item, { handleClick, modifiers, query }) => { return (); }; - + + // Handle item select of `display columns by` field. const onItemSelectDisplayColumns = (item) => { - setFilterByName('display_columns_by', item.key); + setFilterByKey('display_columns_by', item.key); }; + // Handle any date change. const handleDateChange = (name) => (date) => { - setFilterByName(name, moment(date).format('YYYY-MM-DD')); + setReportDateRange('custom'); + setFilterByKey(name, date); }; + // handle submit filter submit button. const handleSubmitClick = () => { onSubmitFilter(filter); }; - const dateRangeOptions = [ {value: 'today', label: 'Today', }, {value: 'this_week', label: 'This Week'}, {value: 'this_month', label: 'This Month'}, {value: 'this_quarter', label: 'This Quarter'}, {value: 'this_year', label: 'This Year'}, + {value: 'custom', label: 'Custom Range'}, ]; + const [activeRowsColumns, setActiveRowsColumns] = useState(false); + + const onClickActiveRowsColumnsBtn = () => { + setActiveRowsColumns(!activeRowsColumns); + }; + + const activeRowsColumnsPopover = ( +
+
Columns
+ { + setFilterByKey('none_zero', value); + })} + > + + + +
+ ); return ( - + + options={dateRangeOptions} + value={reportDateRange} + onChange={(event) => setReportDateRange(event.target.value)} /> - + - + @@ -124,11 +160,11 @@ export default function BalanceSheetHeader({ - + + inline={false}> } + filterable={false} + itemRenderer={accountTypeItem} + popoverProps={{ minimal: true }} + onItemSelect={onItemSelectDisplayColumns}> + + + + + ) +} \ No newline at end of file diff --git a/client/src/containers/Dashboard/FinancialStatements/ProfitLossSheet/ProfitLossSheetTable.js b/client/src/containers/Dashboard/FinancialStatements/ProfitLossSheet/ProfitLossSheetTable.js new file mode 100644 index 000000000..e69de29bb diff --git a/client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheet.js b/client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheet.js new file mode 100644 index 000000000..bf54bd2bb --- /dev/null +++ b/client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheet.js @@ -0,0 +1,75 @@ +import React, { useEffect, useState, useMemo } from 'react'; +import TrialBalanceSheetHeader from "./TrialBalanceSheetHeader"; +import LoadingIndicator from 'components/LoadingIndicator'; +import TrialBalanceSheetTable from './TrialBalanceSheetTable'; +import { useAsync } from 'react-use'; +import moment from 'moment'; +import {compose} from 'utils'; +import TrialBalanceSheetConnect from 'connectors/TrialBalanceSheet.connect'; +import DashboardConnect from 'connectors/Dashboard.connector'; + +function TrialBalanceSheet({ + changePageTitle, + fetchTrialBalanceSheet, + getTrialBalanceSheetIndex, + getTrialBalanceAccounts, +}) { + const [filter, setFilter] = useState({ + from_date: moment().startOf('year').format('YYYY-MM-DD'), + to_date: moment().endOf('year').format('YYYY-MM-DD'), + basis: 'cash', + none_zero: false, + }); + const [reload, setReload] = useState(false); + + const fetchHook = useAsync(async () => { + await Promise.all([ + fetchTrialBalanceSheet(), + ]); + }); + + // Retrieve balance sheet index by the given filter query. + const trialBalanceSheetIndex = useMemo(() => { + return getTrialBalanceSheetIndex(filter); + }, [getTrialBalanceSheetIndex, filter]); + + // Retrieve balance sheet accounts bu the given sheet index. + const trialBalanceAccounts = useMemo(() => { + return getTrialBalanceAccounts(trialBalanceSheetIndex); + }, [trialBalanceSheetIndex]); + + // Change page title of the dashboard. + useEffect(() => { + changePageTitle('Trial Balance Sheet'); + }, []); + + const handleFilterSubmit = (filter) => { + setFilter({ + ...filter, + from_date: moment(filter.from_date).format('YYYY-MM-DD'), + to_date: moment(filter.to_date).format('YYYY-MM-DD'), + }); + setReload(true); + }; + + return ( +
+ + +
+ + + +
+
+ ) +} + +export default compose( + DashboardConnect, + TrialBalanceSheetConnect, +)(TrialBalanceSheet); \ No newline at end of file diff --git a/client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheetHeader.js b/client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheetHeader.js new file mode 100644 index 000000000..257a49a23 --- /dev/null +++ b/client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheetHeader.js @@ -0,0 +1,113 @@ +import React, {useState} from 'react'; +import FinancialStatementHeader from 'containers/Dashboard/FinancialStatements/FinancialStatementHeader'; +import {Row, Col} from 'react-grid-system'; +import { + Button, + FormGroup, + Position, + MenuItem, + RadioGroup, + Radio, + HTMLSelect, + Intent, + Popover, +} from "@blueprintjs/core"; +import {DateInput} from '@blueprintjs/datetime'; +import moment from 'moment'; +import {momentFormatter} from 'utils'; +import {useIntl} from 'react-intl'; + +export default function TrialBalanceSheetHeader({ + pageFilter, + onSubmitFilter, +}) { + const intl = useIntl(); + const [filter, setFilter] = useState({ + ...pageFilter, + from_date: moment(pageFilter.from_date).toDate(), + to_date: moment(pageFilter.to_date).toDate() + }) + + const setFilterByKey = (name, value) => { + setFilter({ ...filter, [name]: value }); + }; + + const [reportDateRange, setReportDateRange] = useState('this_year'); + + const dateRangeOptions = [ + {value: 'today', label: 'Today', }, + {value: 'this_week', label: 'This Week'}, + {value: 'this_month', label: 'This Month'}, + {value: 'this_quarter', label: 'This Quarter'}, + {value: 'this_year', label: 'This Year'}, + {value: 'custom', label: 'Custom Range'}, + ]; + + const handleDateChange = (name) => (date) => { + setReportDateRange('custom'); + setFilterByKey(name, date); + }; + + const handleSubmitClick = () => { onSubmitFilter(filter); }; + + return ( + + + + + + setReportDateRange(event.target.value)} /> + + + + + + + + + + + + + + + + + + + + + + + + + ); +} \ No newline at end of file diff --git a/client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheetTable.js b/client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheetTable.js new file mode 100644 index 000000000..46c5379b2 --- /dev/null +++ b/client/src/containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheetTable.js @@ -0,0 +1,80 @@ +import React, {useEffect, useState, useMemo} from 'react'; +import FinancialSheet from 'components/FinancialSheet'; +import DataTable from 'components/DataTable'; + + +export default function TrialBalanceSheetTable({ + trialBalanceSheetAccounts, + trialBalanceSheetIndex, +}) { + const [data, setData] = useState([]); + + const columns = useMemo(() => [ + { + // Build our expander column + id: 'expander', // Make sure it has an ID + Header: ({ + getToggleAllRowsExpandedProps, + isAllRowsExpanded + }) => ( + + {isAllRowsExpanded ? '👇' : '👉'} + + ), + Cell: ({ row }) => + // Use the row.canExpand and row.getToggleRowExpandedProps prop getter + // to build the toggle for expanding a row + row.canExpand ? ( + + {row.isExpanded ? '👇' : '👉'} + + ) : null, + }, + { + Header: 'Account Name', + accessor: 'name', + className: "actions", + }, + { + Header: 'Code', + accessor: 'code', + className: "note", + }, + { + Header: 'Credit', + accessor: 'credit', + className: 'credit', + }, + { + Header: 'debit', + accessor: 'debit', + className: 'debit', + }, + { + Header: 'Balance', + accessor: 'balance', + className: 'balance', + } + ], []); + + return ( + + + + + ); +} \ No newline at end of file diff --git a/client/src/routes/dashboard.js b/client/src/routes/dashboard.js index db9d76079..67c08d661 100644 --- a/client/src/routes/dashboard.js +++ b/client/src/routes/dashboard.js @@ -75,18 +75,25 @@ export default [ loader: () => import('containers/Dashboard/FinancialStatements/LedgerSheet') }), }, - { - path: `${BASE_URL}/accounting/trial-balance`, - name: 'dashboard.accounting.trial.balance', - component: LazyLoader({ - loader: () => import('containers/Dashboard/FinancialStatements/TrialBalanceSheet') - }), - }, { path: `${BASE_URL}/accounting/balance-sheet`, name: 'dashboard.accounting.balance.sheet', component: LazyLoader({ - loader: () => import('containers/Dashboard/FinancialStatements/BalanceSheet') + loader: () => import('containers/Dashboard/FinancialStatements/BalanceSheet/BalanceSheet') + }), + }, + { + path: `${BASE_URL}/accounting/trial-balance-sheet`, + name: 'dashboard.accounting.trial.balance', + component: LazyLoader({ + loader: () => import('containers/Dashboard/FinancialStatements/TrialBalanceSheet/TrialBalanceSheet') + }), + }, + { + path: `${BASE_URL}/accounting/profit-loss-sheet`, + name: 'dashboard.accounting.profit.loss.sheet', + component: LazyLoader({ + loader: () => import('containers/Dashboard/FinancialStatements/ProfitLossSheet/ProfitLossSheet') }), } ]; \ No newline at end of file diff --git a/client/src/store/financialStatement/financialStatements.actions.js b/client/src/store/financialStatement/financialStatements.actions.js index c2822b390..32673c09e 100644 --- a/client/src/store/financialStatement/financialStatements.actions.js +++ b/client/src/store/financialStatement/financialStatements.actions.js @@ -19,8 +19,33 @@ export const fetchBalanceSheet = ({ query }) => { dispatch({ type: t.BALANCE_SHEET_STATEMENT_SET, data: response.data, + query: query, }); resolve(response); }).catch((error) => { reject(error); }); }); +}; + +export const fetchTrialBalanceSheet = ({ query }) => { + return (dispatch) => new Promise((resolve, reject) => { + ApiService.get('/financial_statements/trial_balance_sheet').then((response) => { + dispatch({ + type: t.TRAIL_BALANCE_STATEMENT_SET, + data: response.data, + }); + resolve(response.data); + }).catch((error) => { reject(error); }) + }) +}; + +export const fetchProfitLossSheet = ({ query }) => { + return (dispatch) => new Promise((resolve, reject) => { + ApiService.get('/financial_statements/profit_loss_sheet').then((response) => { + dispatch({ + type: t.PROFIT_LOSS_STATEMENT_SET, + data: response.data, + }); + resolve(response.data); + }).catch((error) => { reject(error); }); + }) }; \ No newline at end of file diff --git a/client/src/store/financialStatement/financialStatements.reducer.js b/client/src/store/financialStatement/financialStatements.reducer.js index afce617a9..fbbb64744 100644 --- a/client/src/store/financialStatement/financialStatements.reducer.js +++ b/client/src/store/financialStatement/financialStatements.reducer.js @@ -1,14 +1,41 @@ import { createReducer } from '@reduxjs/toolkit'; import t from 'store/types'; +import {getBalanceSheetIndexByQuery, getTrialBalanceSheetIndex} from './financialStatements.selectors'; +import { actionComplete } from '@syncfusion/ej2-react-grids'; const initialState = { - balanceSheet: [], + balanceSheets: [], + trialBalanceSheets: [], generalLedger: [], - trialBalance: [], }; export default createReducer(initialState, { [t.BALANCE_SHEET_STATEMENT_SET]: (state, action) => { - state.balanceSheet.push(action.data); + const index = getBalanceSheetIndexByQuery(state.balanceSheets, action.query); + + const balanceSheet = { + balances: action.data.balance_sheet, + columns: Object.values(action.data.columns), + query: action.data.query, + }; + if (index !== -1) { + state.balanceSheets[index] = balanceSheet; + } else { + state.balanceSheets.push(balanceSheet); + } }, + + [t.TRAIL_BALANCE_STATEMENT_SET]: (state, action) => { + const index = getTrialBalanceSheetIndex(state.trialBalanceSheets, action.query); + + const trailBalanceSheet = { + accounts: action.data.accounts, + query: action.data.query, + }; + if (index !== -1) { + state.trialBalanceSheets[index] = trailBalanceSheet; + } else { + state.trailBalanceSheet.push(trailBalanceSheet); + } + } }); \ No newline at end of file diff --git a/client/src/store/financialStatement/financialStatements.selectors.js b/client/src/store/financialStatement/financialStatements.selectors.js new file mode 100644 index 000000000..d0ba0ec59 --- /dev/null +++ b/client/src/store/financialStatement/financialStatements.selectors.js @@ -0,0 +1,91 @@ +import {getObjectDiff} from 'utils'; + + +// Balance Sheet. +export const getBalanceSheetByQuery = (balanceSheets, query) => { + return balanceSheets.find(balanceSheet => { + return getObjectDiff(query, balanceSheet.query).length === 0; + }); +}; + +export const getBalanceSheetIndexByQuery = (balanceSheets, query) => { + return balanceSheets.findIndex((balanceSheet) => { + return getObjectDiff(query, balanceSheet.query).length === 0; + }); +}; + +export const getBalanceSheetByIndex = (balanceSheets, sheetIndex) => { + return balanceSheets[sheetIndex]; +}; + +export const getBalanceSheetQuery = (balanceSheets, sheetIndex) => { + if (typeof balanceSheets[sheetIndex] === 'object') { + return balanceSheets[sheetIndex].query || {}; + } + return {}; +}; + +export const getBalanceSheetAssetsAccounts = (balanceSheets, sheetIndex) => { + if (typeof balanceSheets[sheetIndex] === 'object') { + return balanceSheets[sheetIndex].balances.assets.accounts || []; + } + return []; +}; + +export const getBalanceSheetLiabilitiesAccounts = (balanceSheets, sheetIndex) => { + if (typeof balanceSheets[sheetIndex] === 'object') { + return balanceSheets[sheetIndex].balances.liabilities_equity.accounts || []; + } + return []; +}; + +export const getBalanceSheetColumns = (balanceSheets, sheetIndex) => { + if (typeof balanceSheets[sheetIndex] === 'object') { + return balanceSheets[sheetIndex].columns; + } + return []; +}; + + +// Trial Balance Sheet. +export const getTrialBalanceSheetIndex = (trialBalanceSheets, query) => { + return trialBalanceSheets.find((trialBalanceSheet) => { + return getObjectDiff(query, trialBalanceSheet.query).length === 0; + }); +}; + +export const getTrialBalanceAccounts = (trialBalanceSheets, sheetIndex) => { + if (typeof trialBalanceSheets[sheetIndex] === 'object') { + return trialBalanceSheets[sheetIndex].accounts; + } + return []; +}; + +export const getTrialBalanceQuery = (trialBalanceSheets, sheetIndex) => { + if (typeof trialBalanceSheets[sheetIndex] === 'object') { + return trialBalanceSheets[sheetIndex].query; + } + return []; +}; + +// Profit/Loss Sheet selectors. +export const getProfitLossSheetIndex = (profitLossSheets, query) => { + return profitLossSheets.find((profitLossSheet) => { + return getObjectDiff(query, profitLossSheet.query).length === 0; + }); +} + +export const getProfitLossSheet = (profitLossSheets, index) => { + return (typeof profitLossSheets[index] !== 'undefined') ? + profitLossSheets[index] : null; +}; + +export const getProfitLossSheetColumns = (profitLossSheets, index) => { + const sheet = getProfitLossSheet(profitLossSheets, index); + return (sheet) ? sheet.columns : []; +}; + +export const getProfitLossSheetAccounts = (profitLossSheets, index) => { + const sheet = getProfitLossSheet(profitLossSheets, index); + return (sheet) ? sheet.accounts : []; +}; \ No newline at end of file diff --git a/client/src/store/financialStatement/financialStatements.types.js b/client/src/store/financialStatement/financialStatements.types.js index d73a21887..8cd91f95c 100644 --- a/client/src/store/financialStatement/financialStatements.types.js +++ b/client/src/store/financialStatement/financialStatements.types.js @@ -3,4 +3,5 @@ export default { GENERAL_LEDGER_STATEMENT_SET: 'GENERAL_LEDGER_STATEMENT_SET', BALANCE_SHEET_STATEMENT_SET: 'BALANCE_SHEET_STATEMENT_SET', + TRAIL_BALANCE_STATEMENT_SET: 'TRAIL_BALANCE_STATEMENT_SET', } \ No newline at end of file diff --git a/client/src/utils.js b/client/src/utils.js index c993bab14..8f6f0936b 100644 --- a/client/src/utils.js +++ b/client/src/utils.js @@ -1,4 +1,5 @@ import moment from 'moment'; +import _ from 'lodash'; export function removeEmptyFromObject(obj) { obj = Object.assign({}, obj); @@ -76,4 +77,38 @@ export const objectKeysTransform = (obj, transform) => { }; export const compose = (...funcs) => - funcs.reduce((a, b) => (...args) => a(b(...args)), arg => arg); \ No newline at end of file + funcs.reduce((a, b) => (...args) => a(b(...args)), arg => arg); + +export const getObjectDiff = (a, b) => { + return _.reduce(a, (result, value, key) => { + return _.isEqual(value, b[key]) ? + result : result.concat(key); + }, []); +} + +export const parseDateRangeQuery = (keyword) => { + const queries = { + 'today': { + range: 'day', + }, + 'this_year': { + range: 'year', + }, + 'this_month': { + range: 'month' + }, + 'this_week': { + range: 'week' + } + }; + + if (typeof queries[keyword] === 'undefined') { + throw new Error(`The date range query ${keyword} is not defined.`); + } + const query = queries[keyword]; + + return { + from_date: moment().startOf(query.range).toDate(), + to_date: moment().endOf(query.range).toDate(), + }; +}; \ No newline at end of file diff --git a/server/.env b/server/.env index e5b4a00f9..56cbd483a 100644 --- a/server/.env +++ b/server/.env @@ -10,5 +10,5 @@ MAIL_FROM_NAME= DB_CLIENT=mysql DB_HOST=127.0.0.1 DB_USER=root -DB_PASSWORD=123123123 +DB_PASSWORD=root DB_NAME=ratteb diff --git a/server/.env.test b/server/.env.test index bfeba6ac6..4963c9f58 100644 --- a/server/.env.test +++ b/server/.env.test @@ -10,7 +10,7 @@ MAIL_FROM_NAME= DB_CLIENT=mysql DB_HOST=127.0.0.1 DB_USER=root -DB_PASSWORD=123123123 +DB_PASSWORD=root DB_NAME=ratteb JWT_SECRET_KEY=ahmedmohamked diff --git a/server/dist/bundle.js b/server/dist/bundle.js index 5ba6242b0..50daa6324 100644 --- a/server/dist/bundle.js +++ b/server/dist/bundle.js @@ -181,9 +181,12 @@ __webpack_require__.r(__webpack_exports__); /* harmony import */ var helmet__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(helmet__WEBPACK_IMPORTED_MODULE_1__); /* harmony import */ var express_boom__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! express-boom */ "express-boom"); /* harmony import */ var express_boom__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(express_boom__WEBPACK_IMPORTED_MODULE_2__); -/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../config */ "./config/index.js"); -/* harmony import */ var _http__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! @/http */ "./src/http/index.js"); -/* harmony import */ var _models__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! @/models */ "./src/models/index.js"); +/* harmony import */ var i18n__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! i18n */ "i18n"); +/* harmony import */ var i18n__WEBPACK_IMPORTED_MODULE_3___default = /*#__PURE__*/__webpack_require__.n(i18n__WEBPACK_IMPORTED_MODULE_3__); +/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../config */ "./config/index.js"); +/* harmony import */ var _http__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! @/http */ "./src/http/index.js"); +/* harmony import */ var _models__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! @/models */ "./src/models/index.js"); + @@ -193,6 +196,20 @@ __webpack_require__.r(__webpack_exports__); var app = express__WEBPACK_IMPORTED_MODULE_0___default()(); +// i18n.configure({ +// // setup some locales - other locales default to en silently +// locales: ['en'], + +// // sets a custom cookie name to parse locale settings from. +// cookie: 'yourcookiename', + +// // where to store json files - defaults to './locales' +// directory: `${__dirname}/resources/locale`, +// }); + +// i18n init parses req for language headers, cookies, etc. +// app.use(i18n.init); + // Express configuration app.set('port', process.env.PORT || 3000); @@ -200,7 +217,7 @@ app.use(helmet__WEBPACK_IMPORTED_MODULE_1___default()()); app.use(express_boom__WEBPACK_IMPORTED_MODULE_2___default()()); app.use(express__WEBPACK_IMPORTED_MODULE_0___default.a.json()); -Object(_http__WEBPACK_IMPORTED_MODULE_4__["default"])(app); +Object(_http__WEBPACK_IMPORTED_MODULE_5__["default"])(app); /* harmony default export */ __webpack_exports__["default"] = (app); @@ -726,12 +743,14 @@ __webpack_require__.r(__webpack_exports__); /* harmony import */ var express__WEBPACK_IMPORTED_MODULE_4___default = /*#__PURE__*/__webpack_require__.n(express__WEBPACK_IMPORTED_MODULE_4__); /* harmony import */ var lodash__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! lodash */ "lodash"); /* harmony import */ var lodash__WEBPACK_IMPORTED_MODULE_5___default = /*#__PURE__*/__webpack_require__.n(lodash__WEBPACK_IMPORTED_MODULE_5__); -/* harmony import */ var _models_Account__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! @/models/Account */ "./src/models/Account.js"); -/* harmony import */ var _http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! @/http/middleware/asyncMiddleware */ "./src/http/middleware/asyncMiddleware.js"); -/* harmony import */ var _http_middleware_jwtAuth__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! @/http/middleware/jwtAuth */ "./src/http/middleware/jwtAuth.js"); -/* harmony import */ var _services_Accounting_JournalPoster__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! @/services/Accounting/JournalPoster */ "./src/services/Accounting/JournalPoster.js"); -/* harmony import */ var _services_Accounting_JournalEntry__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! @/services/Accounting/JournalEntry */ "./src/services/Accounting/JournalEntry.js"); -/* harmony import */ var _models_JournalEntry__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! @/models/JournalEntry */ "./src/models/JournalEntry.js"); +/* harmony import */ var moment__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! moment */ "moment"); +/* harmony import */ var moment__WEBPACK_IMPORTED_MODULE_6___default = /*#__PURE__*/__webpack_require__.n(moment__WEBPACK_IMPORTED_MODULE_6__); +/* harmony import */ var _models_Account__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! @/models/Account */ "./src/models/Account.js"); +/* harmony import */ var _http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! @/http/middleware/asyncMiddleware */ "./src/http/middleware/asyncMiddleware.js"); +/* harmony import */ var _http_middleware_jwtAuth__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! @/http/middleware/jwtAuth */ "./src/http/middleware/jwtAuth.js"); +/* harmony import */ var _services_Accounting_JournalPoster__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! @/services/Accounting/JournalPoster */ "./src/services/Accounting/JournalPoster.js"); +/* harmony import */ var _services_Accounting_JournalEntry__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! @/services/Accounting/JournalEntry */ "./src/services/Accounting/JournalEntry.js"); +/* harmony import */ var _models_JournalEntry__WEBPACK_IMPORTED_MODULE_12__ = __webpack_require__(/*! @/models/JournalEntry */ "./src/models/JournalEntry.js"); function ownKeys(object, enumerableOnly) {var keys = Object.keys(object);if (Object.getOwnPropertySymbols) {var symbols = Object.getOwnPropertySymbols(object);if (enumerableOnly) symbols = symbols.filter(function (sym) {return Object.getOwnPropertyDescriptor(object, sym).enumerable;});keys.push.apply(keys, symbols);}return keys;}function _objectSpread(target) {for (var i = 1; i < arguments.length; i++) {var source = arguments[i] != null ? arguments[i] : {};if (i % 2) {ownKeys(source, true).forEach(function (key) {_babel_runtime_helpers_defineProperty__WEBPACK_IMPORTED_MODULE_1___default()(target, key, source[key]);});} else if (Object.getOwnPropertyDescriptors) {Object.defineProperties(target, Object.getOwnPropertyDescriptors(source));} else {ownKeys(source).forEach(function (key) {Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key));});}}return target;} @@ -742,25 +761,26 @@ function ownKeys(object, enumerableOnly) {var keys = Object.keys(object);if (Obj + /* harmony default export */ __webpack_exports__["default"] = ({ /** * Router constructor. */ router: function router() { var router = express__WEBPACK_IMPORTED_MODULE_4___default.a.Router(); - router.use(_http_middleware_jwtAuth__WEBPACK_IMPORTED_MODULE_8__["default"]); + router.use(_http_middleware_jwtAuth__WEBPACK_IMPORTED_MODULE_9__["default"]); router.post('/make-journal-entries', this.makeJournalEntries.validation, - Object(_http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_7__["default"])(this.makeJournalEntries.handler)); + Object(_http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_8__["default"])(this.makeJournalEntries.handler)); router.post('/recurring-journal-entries', this.recurringJournalEntries.validation, - Object(_http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_7__["default"])(this.recurringJournalEntries.handler)); + Object(_http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_8__["default"])(this.recurringJournalEntries.handler)); router.post('quick-journal-entries', this.quickJournalEntries.validation, - Object(_http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_7__["default"])(this.quickJournalEntries.handler)); + Object(_http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_8__["default"])(this.quickJournalEntries.handler)); return router; }, @@ -772,13 +792,14 @@ function ownKeys(object, enumerableOnly) {var keys = Object.keys(object);if (Obj validation: [ Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('date').isISO8601(), Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('reference').exists(), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('memo').optional().trim().escape(), Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('entries').isArray({ min: 1 }), - Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('entries.*.credit').isNumeric().toInt(), - Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('entries.*.debit').isNumeric().toInt(), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('entries.*.credit').optional({ nullable: true }).isNumeric().toInt(), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('entries.*.debit').optional({ nullable: true }).isNumeric().toInt(), Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('entries.*.account_id').isNumeric().toInt(), Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('entries.*.note').optional()], - handler: function () {var _handler = _babel_runtime_helpers_asyncToGenerator__WEBPACK_IMPORTED_MODULE_2___default()( /*#__PURE__*/_babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0___default.a.mark(function _callee(req, res) {var validationErrors, form, errorReasons, totalCredit, totalDebit, accountsIds, accounts, storedAccountsIds, journalReference, journalPoster;return _babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0___default.a.wrap(function _callee$(_context) {while (1) {switch (_context.prev = _context.next) {case 0: + handler: function () {var _handler = _babel_runtime_helpers_asyncToGenerator__WEBPACK_IMPORTED_MODULE_2___default()( /*#__PURE__*/_babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0___default.a.mark(function _callee(req, res) {var validationErrors, form, totalCredit, totalDebit, user, errorReasons, entries, formattedDate, accountsIds, accounts, storedAccountsIds, journalReference, manualJournal, journalPoster;return _babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0___default.a.wrap(function _callee$(_context) {while (1) {switch (_context.prev = _context.next) {case 0: validationErrors = Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["validationResult"])(req);if ( validationErrors.isEmpty()) {_context.next = 3;break;}return _context.abrupt("return", @@ -790,11 +811,16 @@ function ownKeys(object, enumerableOnly) {var keys = Object.keys(object);if (Obj date: new Date() }, req.body); - errorReasons = []; + totalCredit = 0; totalDebit = 0; - form.entries.forEach(function (entry) { + user = req.user; + errorReasons = []; + entries = form.entries.filter(function (entry) {return entry.credit || entry.debit;}); + formattedDate = moment__WEBPACK_IMPORTED_MODULE_6___default()(form.date).format('YYYY-MM-DD'); + + entries.forEach(function (entry) { if (entry.credit > 0) { totalCredit += entry.credit; } @@ -802,6 +828,7 @@ function ownKeys(object, enumerableOnly) {var keys = Object.keys(object);if (Obj totalDebit += entry.debit; } }); + if (totalCredit <= 0 || totalDebit <= 0) { errorReasons.push({ type: 'CREDIT.DEBIT.SUMATION.SHOULD.NOT.EQUAL.ZERO', @@ -811,36 +838,48 @@ function ownKeys(object, enumerableOnly) {var keys = Object.keys(object);if (Obj if (totalCredit !== totalDebit) { errorReasons.push({ type: 'CREDIT.DEBIT.NOT.EQUALS', code: 100 }); } - accountsIds = form.entries.map(function (entry) {return entry.account_id;});_context.next = 13;return ( - _models_Account__WEBPACK_IMPORTED_MODULE_6__["default"].query().whereIn('id', accountsIds). - withGraphFetched('type'));case 13:accounts = _context.sent; + accountsIds = entries.map(function (entry) {return entry.account_id;});_context.next = 16;return ( + _models_Account__WEBPACK_IMPORTED_MODULE_7__["default"].query().whereIn('id', accountsIds). + withGraphFetched('type'));case 16:accounts = _context.sent; storedAccountsIds = accounts.map(function (account) {return account.id;}); if (Object(lodash__WEBPACK_IMPORTED_MODULE_5__["difference"])(accountsIds, storedAccountsIds).length > 0) { errorReasons.push({ type: 'ACCOUNTS.IDS.NOT.FOUND', code: 200 }); - }_context.next = 18;return ( + }_context.next = 21;return ( - _models_JournalEntry__WEBPACK_IMPORTED_MODULE_11__["default"].query().where('reference', form.reference));case 18:journalReference = _context.sent; + _models_JournalEntry__WEBPACK_IMPORTED_MODULE_12__["default"].query().where('reference', form.reference));case 21:journalReference = _context.sent; if (journalReference.length > 0) { errorReasons.push({ type: 'REFERENCE.ALREADY.EXISTS', code: 300 }); }if (!( - errorReasons.length > 0)) {_context.next = 22;break;}return _context.abrupt("return", - res.status(400).send({ errors: errorReasons }));case 22: + errorReasons.length > 0)) {_context.next = 25;break;}return _context.abrupt("return", + res.status(400).send({ errors: errorReasons }));case 25:_context.next = 27;return ( - journalPoster = new _services_Accounting_JournalPoster__WEBPACK_IMPORTED_MODULE_9__["default"](); - form.entries.forEach(function (entry) { + + _models_JournalEntry__WEBPACK_IMPORTED_MODULE_12__["default"].query().insert({ + reference: form.reference, + transaction_type: 'Journal', + amount: totalCredit, + date: formattedDate, + note: form.memo, + user_id: user.id }));case 27:manualJournal = _context.sent; + + journalPoster = new _services_Accounting_JournalPoster__WEBPACK_IMPORTED_MODULE_10__["default"](); + + entries.forEach(function (entry) { var account = accounts.find(function (a) {return a.id === entry.account_id;}); - var jouranlEntry = new _services_Accounting_JournalEntry__WEBPACK_IMPORTED_MODULE_10__["default"]({ - date: entry.date, + var jouranlEntry = new _services_Accounting_JournalEntry__WEBPACK_IMPORTED_MODULE_11__["default"]({ debit: entry.debit, credit: entry.credit, account: account.id, + transactionType: 'Journal', accountNormal: account.type.normal, - note: entry.note }); + note: entry.note, + date: formattedDate, + userId: user.id }); if (entry.debit) { journalPoster.debit(jouranlEntry); @@ -850,17 +889,17 @@ function ownKeys(object, enumerableOnly) {var keys = Object.keys(object);if (Obj }); // Saves the journal entries and accounts balance changes. - _context.next = 26;return Promise.all([ + _context.next = 32;return Promise.all([ journalPoster.saveEntries(), - journalPoster.saveBalance()]);case 26:return _context.abrupt("return", + journalPoster.saveBalance()]);case 32:return _context.abrupt("return", - res.status(200).send());case 27:case "end":return _context.stop();}}}, _callee);}));function handler(_x, _x2) {return _handler.apply(this, arguments);}return handler;}() }, + res.status(200).send({ id: manualJournal.id }));case 33:case "end":return _context.stop();}}}, _callee);}));function handler(_x, _x2) {return _handler.apply(this, arguments);}return handler;}() }, /** - * Saves recurring journal entries template. - */ + * Saves recurring journal entries template. + */ recurringJournalEntries: { validation: [ Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('template_name').exists(), @@ -921,7 +960,7 @@ function ownKeys(object, enumerableOnly) {var keys = Object.keys(object);if (Obj errorReasons = []; form = _objectSpread({}, req.body);_context4.next = 7;return ( - _models_Account__WEBPACK_IMPORTED_MODULE_6__["default"].query(). + _models_Account__WEBPACK_IMPORTED_MODULE_7__["default"].query(). where('id', form.credit_account_id). orWhere('id', form.debit_account_id));case 7:foundAccounts = _context4.sent; @@ -2146,17 +2185,77 @@ __webpack_require__.r(__webpack_exports__); "use strict"; __webpack_require__.r(__webpack_exports__); -/* harmony import */ var express__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! express */ "express"); -/* harmony import */ var express__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(express__WEBPACK_IMPORTED_MODULE_0__); +/* harmony import */ var _babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @babel/runtime/regenerator */ "@babel/runtime/regenerator"); +/* harmony import */ var _babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0__); +/* harmony import */ var _babel_runtime_helpers_asyncToGenerator__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! @babel/runtime/helpers/asyncToGenerator */ "@babel/runtime/helpers/asyncToGenerator"); +/* harmony import */ var _babel_runtime_helpers_asyncToGenerator__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_babel_runtime_helpers_asyncToGenerator__WEBPACK_IMPORTED_MODULE_1__); +/* harmony import */ var express__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! express */ "express"); +/* harmony import */ var express__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(express__WEBPACK_IMPORTED_MODULE_2__); +/* harmony import */ var express_validator__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! express-validator */ "express-validator"); +/* harmony import */ var express_validator__WEBPACK_IMPORTED_MODULE_3___default = /*#__PURE__*/__webpack_require__.n(express_validator__WEBPACK_IMPORTED_MODULE_3__); +/* harmony import */ var _http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! @/http/middleware/asyncMiddleware */ "./src/http/middleware/asyncMiddleware.js"); + + /* harmony default export */ __webpack_exports__["default"] = ({ router: function router() { - var router = express__WEBPACK_IMPORTED_MODULE_0___default.a.Router(); + var router = express__WEBPACK_IMPORTED_MODULE_2___default.a.Router(); + + router.post('/', + this.newCustomer.validation, + Object(_http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_4__["default"])(this.newCustomer.handler)); + + router.post('/:id', + this.editCustomer.validation, + Object(_http_middleware_asyncMiddleware__WEBPACK_IMPORTED_MODULE_4__["default"])(this.editCustomer.handler)); return router; - } }); + }, + + newCustomer: { + validation: [ + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('custom_type').exists().trim().escape(), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('first_name').exists().trim().escape(), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('last_name'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('company_name'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('email'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('work_phone'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('personal_phone'), + + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('billing_address.country'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('billing_address.address'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('billing_address.city'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('billing_address.phone'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('billing_address.zip_code'), + + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('shiping_address.country'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('shiping_address.address'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('shiping_address.city'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('shiping_address.phone'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('shiping_address.zip_code'), + + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('contact.additional_phone'), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('contact.additional_email'), + + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('custom_fields').optional().isArray({ min: 1 }), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('custom_fields.*.key').exists().trim().escape(), + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('custom_fields.*.value').exists(), + + Object(express_validator__WEBPACK_IMPORTED_MODULE_3__["check"])('inactive').optional().isBoolean().toBoolean()], + + + handler: function () {var _handler = _babel_runtime_helpers_asyncToGenerator__WEBPACK_IMPORTED_MODULE_1___default()( /*#__PURE__*/_babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0___default.a.mark(function _callee(req, res) {return _babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0___default.a.wrap(function _callee$(_context) {while (1) {switch (_context.prev = _context.next) {case 0:case "end":return _context.stop();}}}, _callee);}));function handler(_x, _x2) {return _handler.apply(this, arguments);}return handler;}() }, + + + + + editCustomer: { + validation: [], + + + handler: function () {var _handler2 = _babel_runtime_helpers_asyncToGenerator__WEBPACK_IMPORTED_MODULE_1___default()( /*#__PURE__*/_babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0___default.a.mark(function _callee2(req, res) {return _babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_0___default.a.wrap(function _callee2$(_context2) {while (1) {switch (_context2.prev = _context2.next) {case 0:case "end":return _context2.stop();}}}, _callee2);}));function handler(_x3, _x4) {return _handler2.apply(this, arguments);}return handler;}() } }); /***/ }), @@ -3179,25 +3278,25 @@ var formatNumberClosure = function formatNumberClosure(filter) {return function res.status(200).send({ meta: _objectSpread({}, filter), - items: items }));case 20:case "end":return _context2.stop();}}}, _callee2);}));function handler(_x3, _x4) {return _handler2.apply(this, arguments);}return handler;}() }, + accounts: items }));case 20:case "end":return _context2.stop();}}}, _callee2);}));function handler(_x3, _x4) {return _handler2.apply(this, arguments);}return handler;}() }, /** - * Retrieve the balance sheet. - */ + * Retrieve the balance sheet. + */ balanceSheet: { validation: [ Object(express_validator__WEBPACK_IMPORTED_MODULE_5__["query"])('accounting_method').optional().isIn(['cash', 'accural']), Object(express_validator__WEBPACK_IMPORTED_MODULE_5__["query"])('from_date').optional(), Object(express_validator__WEBPACK_IMPORTED_MODULE_5__["query"])('to_date').optional(), - Object(express_validator__WEBPACK_IMPORTED_MODULE_5__["query"])('display_columns_by').optional().isIn(['year', 'month', 'week', 'day', 'quarter']), + Object(express_validator__WEBPACK_IMPORTED_MODULE_5__["query"])('display_columns_by').optional().isIn(['total', 'year', 'month', 'week', 'day', 'quarter']), Object(express_validator__WEBPACK_IMPORTED_MODULE_5__["query"])('number_format.no_cents').optional().isBoolean().toBoolean(), Object(express_validator__WEBPACK_IMPORTED_MODULE_5__["query"])('number_format.divide_1000').optional().isBoolean().toBoolean(), Object(express_validator__WEBPACK_IMPORTED_MODULE_5__["query"])('none_zero').optional().isBoolean().toBoolean()], - handler: function () {var _handler3 = _babel_runtime_helpers_asyncToGenerator__WEBPACK_IMPORTED_MODULE_3___default()( /*#__PURE__*/_babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_1___default.a.mark(function _callee3(req, res) {var validationErrors, filter, balanceSheetTypes, accounts, journalEntriesCollected, journalEntries, balanceFormatter, dateRangeSet, assets, liabilitiesEquity;return _babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_1___default.a.wrap(function _callee3$(_context3) {while (1) {switch (_context3.prev = _context3.next) {case 0: + handler: function () {var _handler3 = _babel_runtime_helpers_asyncToGenerator__WEBPACK_IMPORTED_MODULE_3___default()( /*#__PURE__*/_babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_1___default.a.mark(function _callee3(req, res) {var validationErrors, filter, balanceSheetTypes, accounts, journalEntriesCollected, journalEntries, balanceFormatter, filterDateType, dateRangeSet, assets, liabilitiesEquity;return _babel_runtime_regenerator__WEBPACK_IMPORTED_MODULE_1___default.a.wrap(function _callee3$(_context3) {while (1) {switch (_context3.prev = _context3.next) {case 0: validationErrors = Object(express_validator__WEBPACK_IMPORTED_MODULE_5__["validationResult"])(req);if ( validationErrors.isEmpty()) {_context3.next = 3;break;}return _context3.abrupt("return", @@ -3206,19 +3305,19 @@ var formatNumberClosure = function formatNumberClosure(filter) {return function filter = _objectSpread({ - display_columns_by: 'year', + display_columns_by: 'total', from_date: moment__WEBPACK_IMPORTED_MODULE_6___default()().startOf('year').format('YYYY-MM-DD'), to_date: moment__WEBPACK_IMPORTED_MODULE_6___default()().endOf('year').format('YYYY-MM-DD'), number_format: { no_cents: false, divide_1000: false }, - none_zero: false }, + none_zero: false, + basis: 'cash' }, req.query);_context3.next = 6;return ( - _models_AccountType__WEBPACK_IMPORTED_MODULE_11__["default"].query(). - where('balance_sheet', true));case 6:balanceSheetTypes = _context3.sent;_context3.next = 9;return ( + _models_AccountType__WEBPACK_IMPORTED_MODULE_11__["default"].query().where('balance_sheet', true));case 6:balanceSheetTypes = _context3.sent;_context3.next = 9;return ( _models_Account__WEBPACK_IMPORTED_MODULE_12__["default"].query(). @@ -3235,59 +3334,100 @@ var formatNumberClosure = function formatNumberClosure(filter) {return function // Account balance formmatter based on the given query. balanceFormatter = formatNumberClosure(filter.number_format); + filterDateType = filter.display_columns_by === 'total' ? + 'day' : filter.display_columns_by; // Gets the date range set from start to end date. dateRangeSet = Object(_utils__WEBPACK_IMPORTED_MODULE_14__["dateRangeCollection"])( filter.from_date, filter.to_date, - filter.display_columns_by); + filterDateType); + // Retrieve the asset balance sheet. - assets = _babel_runtime_helpers_toConsumableArray__WEBPACK_IMPORTED_MODULE_0___default()( - accounts. + assets = accounts. filter(function (account) {return ( account.type.normal === 'debit' && ( account.transactions.length > 0 || !filter.none_zero));}). - map(function (account) {return _objectSpread({}, - Object(lodash__WEBPACK_IMPORTED_MODULE_7__["pick"])(account, ['id', 'index', 'name', 'code']), { - transactions: dateRangeSet.map(function (date) { - var type = filter.display_columns_by; - var balance = journalEntries.getClosingBalance(account.id, date, type); - return { date: date, balance: balanceFormatter(balance) }; - }) });})); + map(function (account) { + // Calculates the closing balance to the given date. + var closingBalance = journalEntries.getClosingBalance(account.id, filter.to_date); + var type = filter.display_columns_by; + return _objectSpread({}, + Object(lodash__WEBPACK_IMPORTED_MODULE_7__["pick"])(account, ['id', 'index', 'name', 'code']), {}, + type !== 'total' ? { + periods_balance: dateRangeSet.map(function (date) { + var balance = journalEntries.getClosingBalance(account.id, date, filterDateType); + + return { + date: date, + formatted_amount: balanceFormatter(balance), + amount: balance }; + + }) } : + {}, { + balance: { + formatted_amount: balanceFormatter(closingBalance), + amount: closingBalance, + date: filter.to_date } }); + + + }); // Retrieve liabilities and equity balance sheet. - liabilitiesEquity = _babel_runtime_helpers_toConsumableArray__WEBPACK_IMPORTED_MODULE_0___default()( - accounts. + liabilitiesEquity = accounts. filter(function (account) {return ( account.type.normal === 'credit' && ( account.transactions.length > 0 || !filter.none_zero));}). - map(function (account) {return _objectSpread({}, - Object(lodash__WEBPACK_IMPORTED_MODULE_7__["pick"])(account, ['id', 'index', 'name', 'code']), { - transactions: dateRangeSet.map(function (date) { - var type = filter.display_columns_by; - var balance = journalEntries.getClosingBalance(account.id, date, type); - return { date: date, balance: balanceFormatter(balance) }; - }) });}));return _context3.abrupt("return", + map(function (account) { + // Calculates the closing balance to the given date. + var closingBalance = journalEntries.getClosingBalance(account.id, filter.to_date); + var type = filter.display_columns_by; + return _objectSpread({}, + Object(lodash__WEBPACK_IMPORTED_MODULE_7__["pick"])(account, ['id', 'index', 'name', 'code']), {}, + type !== 'total' ? { + periods_balance: dateRangeSet.map(function (date) { + var balance = journalEntries.getClosingBalance(account.id, date, filterDateType); + + return { + date: date, + formatted_amount: balanceFormatter(balance), + amount: balance }; + + }) } : + {}, { + balance: { + formattedAmount: balanceFormatter(closingBalance), + amount: closingBalance, + date: filter.to_date } }); + + + });return _context3.abrupt("return", res.status(200).send({ query: _objectSpread({}, filter), columns: _objectSpread({}, dateRangeSet), balance_sheet: { - assets: assets, - liabilities_equity: liabilitiesEquity } }));case 18:case "end":return _context3.stop();}}}, _callee3);}));function handler(_x5, _x6) {return _handler3.apply(this, arguments);}return handler;}() }, + assets: { + title: 'Assets', + accounts: _babel_runtime_helpers_toConsumableArray__WEBPACK_IMPORTED_MODULE_0___default()(assets) }, + + liabilities_equity: { + title: 'Liabilities & Equity', + accounts: _babel_runtime_helpers_toConsumableArray__WEBPACK_IMPORTED_MODULE_0___default()(liabilitiesEquity) } } }));case 19:case "end":return _context3.stop();}}}, _callee3);}));function handler(_x5, _x6) {return _handler3.apply(this, arguments);}return handler;}() }, + /** - * Retrieve the trial balance sheet. - */ + * Retrieve the trial balance sheet. + */ trialBalanceSheet: { validation: [ Object(express_validator__WEBPACK_IMPORTED_MODULE_5__["query"])('basis').optional(), @@ -3349,7 +3489,7 @@ var formatNumberClosure = function formatNumberClosure(filter) {return function });return _context4.abrupt("return", res.status(200).send({ - meta: _objectSpread({}, filter), + query: _objectSpread({}, filter), items: _babel_runtime_helpers_toConsumableArray__WEBPACK_IMPORTED_MODULE_0___default()(items) }));case 13:case "end":return _context4.stop();}}}, _callee4);}));function handler(_x7, _x8) {return _handler4.apply(this, arguments);}return handler;}() }, @@ -7969,10 +8109,12 @@ __webpack_require__.r(__webpack_exports__); /* harmony import */ var _services_Accounting_JournalEntry__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! @/services/Accounting/JournalEntry */ "./src/services/Accounting/JournalEntry.js"); /* harmony import */ var _models_AccountTransaction__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! @/models/AccountTransaction */ "./src/models/AccountTransaction.js"); /* harmony import */ var _models_AccountBalance__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! @/models/AccountBalance */ "./src/models/AccountBalance.js"); +/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! @/utils */ "./src/utils/index.js"); function ownKeys(object, enumerableOnly) {var keys = Object.keys(object);if (Object.getOwnPropertySymbols) {var symbols = Object.getOwnPropertySymbols(object);if (enumerableOnly) symbols = symbols.filter(function (sym) {return Object.getOwnPropertyDescriptor(object, sym).enumerable;});keys.push.apply(keys, symbols);}return keys;}function _objectSpread(target) {for (var i = 1; i < arguments.length; i++) {var source = arguments[i] != null ? arguments[i] : {};if (i % 2) {ownKeys(source, true).forEach(function (key) {_babel_runtime_helpers_defineProperty__WEBPACK_IMPORTED_MODULE_0___default()(target, key, source[key]);});} else if (Object.getOwnPropertyDescriptors) {Object.defineProperties(target, Object.getOwnPropertyDescriptors(source));} else {ownKeys(source).forEach(function (key) {Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key));});}}return target;} + var JournalPoster = /*#__PURE__*/function () { @@ -8096,17 +8238,17 @@ JournalPoster = /*#__PURE__*/function () { this.entries.forEach(function (entry) { var oper = _models_AccountTransaction__WEBPACK_IMPORTED_MODULE_8__["default"].query().insert(_objectSpread({ accountId: entry.account }, - Object(lodash__WEBPACK_IMPORTED_MODULE_5__["pick"])(entry, ['credit', 'debit', 'transactionType', + Object(lodash__WEBPACK_IMPORTED_MODULE_5__["pick"])(entry, ['credit', 'debit', 'transactionType', 'date', 'userId', 'referenceType', 'referenceId', 'note']))); - saveOperations.push(oper); + saveOperations.push(function () {return oper;}); });_context2.next = 4;return ( - Promise.all(saveOperations));case 4:case "end":return _context2.stop();}}}, _callee2, this);}));function saveEntries() {return _saveEntries.apply(this, arguments);}return saveEntries;}() + Object(_utils__WEBPACK_IMPORTED_MODULE_10__["promiseSerial"])(saveOperations));case 4:case "end":return _context2.stop();}}}, _callee2, this);}));function saveEntries() {return _saveEntries.apply(this, arguments);}return saveEntries;}() /** - * Reverses the stacked journal entries. - */ }, { key: "reverseEntries", value: function reverseEntries() + * Reverses the stacked journal entries. + */ }, { key: "reverseEntries", value: function reverseEntries() { var reverseEntries = []; @@ -8444,7 +8586,7 @@ var transporter = nodemailer__WEBPACK_IMPORTED_MODULE_0___default.a.createTransp /*!****************************!*\ !*** ./src/utils/index.js ***! \****************************/ -/*! exports provided: hashPassword, origin, dateRangeCollection, dateRangeFormat, mapValuesDeep, mapKeysDeep */ +/*! exports provided: hashPassword, origin, dateRangeCollection, dateRangeFormat, mapValuesDeep, mapKeysDeep, promiseSerial */ /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; @@ -8455,6 +8597,7 @@ __webpack_require__.r(__webpack_exports__); /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "dateRangeFormat", function() { return dateRangeFormat; }); /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "mapValuesDeep", function() { return mapValuesDeep; }); /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "mapKeysDeep", function() { return mapKeysDeep; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "promiseSerial", function() { return promiseSerial; }); /* harmony import */ var bcryptjs__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! bcryptjs */ "bcryptjs"); /* harmony import */ var bcryptjs__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(bcryptjs__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var moment__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! moment */ "moment"); @@ -8530,6 +8673,12 @@ var mapValuesDeep = function mapValuesDeep(v, callback) {return ( callback(v));}; +var promiseSerial = function promiseSerial(funcs) { + return funcs.reduce(function (promise, func) {return promise.then(function (result) {return func().then(Array.prototype.concat.bind(result));});}, + Promise.resolve([])); +}; + + /***/ }), @@ -8766,6 +8915,17 @@ module.exports = require("helmet"); /***/ }), +/***/ "i18n": +/*!***********************!*\ + !*** external "i18n" ***! + \***********************/ +/*! no static exports found */ +/***/ (function(module, exports) { + +module.exports = require("i18n"); + +/***/ }), + /***/ "jsonwebtoken": /*!*******************************!*\ !*** external "jsonwebtoken" ***! @@ -8866,4 +9026,4 @@ module.exports = require("path"); /***/ }) /******/ }); -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"bundle.js","sources":["webpack/bootstrap","/Users/ahmed/Documents/Ratteb/server/config/index.js","/Users/ahmed/Documents/Ratteb/server/knexfile.js","/Users/ahmed/Documents/Ratteb/server/src/app.js","/Users/ahmed/Documents/Ratteb/server/src/collection/BudgetEntriesSet.js","/Users/ahmed/Documents/Ratteb/server/src/collection/NestedSet/index.js","/Users/ahmed/Documents/Ratteb/server/src/collection/ResourceFieldMetadataCollection.js","/Users/ahmed/Documents/Ratteb/server/src/data/ResourceFieldsKeys.js","/Users/ahmed/Documents/Ratteb/server/src/database/knex.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/AccountOpeningBalance.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/AccountTypes.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Accounting.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Accounts.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Authentication.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Bills.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Budget.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/BudgetReports.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Currencies.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/CurrencyAdjustment.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Customers.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Expenses.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Fields.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/FinancialStatements.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/ItemCategories.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Items.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Options.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Resources.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Roles.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Suppliers.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Users.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Views.js","/Users/ahmed/Documents/Ratteb/server/src/http/index.js","/Users/ahmed/Documents/Ratteb/server/src/http/middleware/asyncMiddleware.js","/Users/ahmed/Documents/Ratteb/server/src/http/middleware/authorization.js","/Users/ahmed/Documents/Ratteb/server/src/http/middleware/jwtAuth.js","/Users/ahmed/Documents/Ratteb/server/src/lib/LogicEvaluation/Lexer.js","/Users/ahmed/Documents/Ratteb/server/src/lib/LogicEvaluation/Parser.js","/Users/ahmed/Documents/Ratteb/server/src/lib/LogicEvaluation/QueryParser.js","/Users/ahmed/Documents/Ratteb/server/src/lib/Metable/MetableCollection.js","/Users/ahmed/Documents/Ratteb/server/src/lib/ViewRolesBuilder/index.js","/Users/ahmed/Documents/Ratteb/server/src/models/Account.js","/Users/ahmed/Documents/Ratteb/server/src/models/AccountBalance.js","/Users/ahmed/Documents/Ratteb/server/src/models/AccountTransaction.js","/Users/ahmed/Documents/Ratteb/server/src/models/AccountType.js","/Users/ahmed/Documents/Ratteb/server/src/models/Budget.js","/Users/ahmed/Documents/Ratteb/server/src/models/BudgetEntry.js","/Users/ahmed/Documents/Ratteb/server/src/models/Expense.js","/Users/ahmed/Documents/Ratteb/server/src/models/Item.js","/Users/ahmed/Documents/Ratteb/server/src/models/ItemCategory.js","/Users/ahmed/Documents/Ratteb/server/src/models/JournalEntry.js","/Users/ahmed/Documents/Ratteb/server/src/models/ManualJournal.js","/Users/ahmed/Documents/Ratteb/server/src/models/Model.js","/Users/ahmed/Documents/Ratteb/server/src/models/Option.js","/Users/ahmed/Documents/Ratteb/server/src/models/PasswordReset.js","/Users/ahmed/Documents/Ratteb/server/src/models/Permission.js","/Users/ahmed/Documents/Ratteb/server/src/models/Resource.js","/Users/ahmed/Documents/Ratteb/server/src/models/ResourceField.js","/Users/ahmed/Documents/Ratteb/server/src/models/ResourceFieldMetadata.js","/Users/ahmed/Documents/Ratteb/server/src/models/Role.js","/Users/ahmed/Documents/Ratteb/server/src/models/User.js","/Users/ahmed/Documents/Ratteb/server/src/models/View.js","/Users/ahmed/Documents/Ratteb/server/src/models/ViewColumn.js","/Users/ahmed/Documents/Ratteb/server/src/models/ViewRole.js","/Users/ahmed/Documents/Ratteb/server/src/models/index.js","/Users/ahmed/Documents/Ratteb/server/src/server.js","/Users/ahmed/Documents/Ratteb/server/src/services/Accounting/JournalEntry.js","/Users/ahmed/Documents/Ratteb/server/src/services/Accounting/JournalPoster.js","/Users/ahmed/Documents/Ratteb/server/src/services/CustomFields/ResourceCustomFieldRepository.js","/Users/ahmed/Documents/Ratteb/server/src/services/Moment/index.js","/Users/ahmed/Documents/Ratteb/server/src/services/mail.js","/Users/ahmed/Documents/Ratteb/server/src/utils/index.js","external \"@babel/plugin-transform-runtime\"","external \"@babel/runtime/helpers/asyncToGenerator\"","external \"@babel/runtime/helpers/classCallCheck\"","external \"@babel/runtime/helpers/createClass\"","external \"@babel/runtime/helpers/defineProperty\"","external \"@babel/runtime/helpers/get\"","external \"@babel/runtime/helpers/getPrototypeOf\"","external \"@babel/runtime/helpers/inherits\"","external \"@babel/runtime/helpers/possibleConstructorReturn\"","external \"@babel/runtime/helpers/slicedToArray\"","external \"@babel/runtime/helpers/toConsumableArray\"","external \"@babel/runtime/regenerator\"","external \"bcryptjs\"","external \"dotenv\"","external \"errorhandler\"","external \"express\"","external \"express-boom\"","external \"express-validator\"","external \"fs\"","external \"helmet\"","external \"jsonwebtoken\"","external \"knex\"","external \"lodash\"","external \"moment\"","external \"moment-range\"","external \"mustache\"","external \"nodemailer\"","external \"objection\"","external \"path\""],"sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"dist/\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 0);\n","import path from 'path';\nimport dotenv from 'dotenv';\n\ndotenv.config({\n  path: path.resolve(process.cwd(), '.env.test'),\n});\n","require('dotenv').config();\n\nconst MIGRATIONS_DIR = './src/database/migrations';\nconst SEEDS_DIR = './src/database/seeds';\n\nmodule.exports = {\n  test: {\n    client: process.env.DB_CLIENT,\n    migrations: {\n      directory: MIGRATIONS_DIR,\n    },\n    connection: {\n      host: process.env.DB_HOST,\n      user: process.env.DB_USER,\n      password: process.env.DB_PASSWORD,\n      database: process.env.DB_NAME,\n      charset: 'utf8',\n    },\n  },\n  development: {\n    client: process.env.DB_CLIENT,\n    connection: {\n      host: process.env.DB_HOST,\n      user: process.env.DB_USER,\n      password: process.env.DB_PASSWORD,\n      database: process.env.DB_NAME,\n      charset: 'utf8',\n    },\n    migrations: {\n      directory: MIGRATIONS_DIR,\n    },\n    seeds: {\n      directory: SEEDS_DIR,\n    },\n  },\n  production: {\n    client: process.env.DB_CLIENT,\n    connection: {\n      host: process.env.DB_HOST,\n      user: process.env.DB_USER,\n      password: process.env.DB_PASSWORD,\n      database: process.env.DB_NAME,\n      charset: 'utf8',\n    },\n    migrations: {\n      directory: MIGRATIONS_DIR,\n    },\n    seeds: {\n      directory: SEEDS_DIR,\n    },\n  },\n};\n","import express from 'express';\nimport helmet from 'helmet';\nimport boom from 'express-boom';\nimport '../config';\nimport routes from '@/http';\nimport '@/models';\n\nconst app = express();\n\n// Express configuration\napp.set('port', process.env.PORT || 3000);\n\napp.use(helmet());\napp.use(boom());\napp.use(express.json());\n\nroutes(app);\n\nexport default app;\n","\n\nexport default class BudgetEntriesSet {\n\n  constructor() {\n    this.accounts = {}; \n    this.totalSummary = {}\n    this.orderSize = null;\n  }\n\n  setZeroPlaceholder() {\n    if (!this.orderSize) { return; }\n\n    Object.values(this.accounts).forEach((account) => {\n\n      for (let i = 0; i <= this.orderSize.length; i++) {\n        if (typeof account[i] === 'undefined') {\n          account[i] = { amount: 0 };\n        }\n      }\n    });\n  }\n\n  static from(accounts, configs) {\n    const collection = new this(configs);\n\n    accounts.forEach((entry) => {\n      if (typeof this.accounts[entry.accountId] === 'undefined') {\n        collection.accounts[entry.accountId] = {};\n      }\n      if (entry.order) {\n        collection.accounts[entry.accountId][entry.order] = entry;\n      }\n    });\n    return collection;\n  }\n\n  toArray() {\n    const output = [];\n\n    Object.key(this.accounts).forEach((accountId) => {\n      const entries = this.accounts[accountId];\n      output.push({\n        account_id: accountId,\n        entries: [\n          ...Object.key(entries).map((order) => {\n            const entry = entries[order];\n            return {\n              order,\n              amount: entry.amount,\n            };\n          }),\n        ],\n      });\n    });\n  }\n\n  calcTotalSummary() {\n    const totalSummary = {};\n\n    for (let i = 0; i < this.orderSize.length; i++) {\n      Object.value(this.accounts).forEach((account) => {\n        if (typeof totalSummary[i] !== 'undefined') {\n          totalSummary[i] = { amount: 0, order: i };\n        }\n        totalSummary[i].amount += account[i].amount;\n      });\n    }\n    this.totalSummary = totalSummary;\n  }\n\n  toArrayTotalSummary() {\n    return Object.values(this.totalSummary);\n  }\n\n}\n","\nexport default class NestedSet {\n  /**\n   * Constructor method.\n   * @param {Object} options -\n   */\n  constructor(items, options) {\n    this.options = {\n      parentId: 'parent_id',\n      id: 'id',\n      ...options,\n    };\n    this.items = items;\n    this.collection = {};\n  }\n\n  /**\n   * Link nodes children.\n   */\n  linkChildren() {\n    if (this.items.length <= 0) return false;\n\n    const map = {};\n    this.items.forEach((item) => {\n      map[item.id] = item;\n      map[item.id].children = [];\n    });\n\n    this.items.forEach((item) => {\n      const parentNodeId = item[this.options.parentId];\n      if (parentNodeId) {\n        map[parentNodeId].children.push(item);\n      }\n    });\n    return map;\n  }\n\n  toTree() {\n    const map = this.linkChildren();\n    const tree = {};\n\n    this.items.forEach((item) => {\n      const parentNodeId = item[this.options.parentId];\n      if (!parentNodeId) {\n        tree[item.id] = map[item.id];\n      }\n    });\n    this.collection = Object.values(tree);\n    return this.collection;\n  }\n\n  getTree() {\n    return this.collection;\n  }\n\n  flattenTree(nodeMapper) {\n    const flattenTree = [];\n\n    const traversal = (nodes, parentNode) => {\n      nodes.forEach((node) => {\n        let nodeMapped = node;\n\n        if (typeof nodeMapper === 'function') {\n          nodeMapped = nodeMapper(nodeMapped, parentNode);\n        }\n        flattenTree.push(nodeMapped);\n\n        if (node.children && node.children.length > 0) {\n          traversal(node.children, node);\n        }\n      });\n    };\n    traversal(this.collection);\n\n    return flattenTree;\n  }\n}\n","import MetableCollection from '@/lib/Metable/MetableCollection';\nimport ResourceFieldMetadata from '@/models/ResourceFieldMetadata';\n\nexport default class ResourceFieldMetadataCollection extends MetableCollection {\n  /**\n   * Constructor method.\n   */\n  constructor() {\n    super();\n\n    this.setModel(ResourceFieldMetadata);\n    this.extraColumns = ['resource_id', 'resource_item_id'];\n  }\n}\n","\nexport default {\n  \"expense_account\": 'expense_account_id',\n  \"payment_account\": 'payment_account_id',\n  \"account_type\": \"account_type_id\"\n}","import Knex from 'knex';\nimport { knexSnakeCaseMappers } from 'objection';\nimport knexfile from '@/../knexfile';\n\nconst config = knexfile[process.env.NODE_ENV];\nconst knex = Knex({\n  ...config,\n  ...knexSnakeCaseMappers({ upperCase: true }),\n});\n\nexport default knex;\n","import express from 'express';\nimport { check, validationResult, oneOf } from 'express-validator';\nimport { difference } from 'lodash';\nimport moment from 'moment';\nimport asyncMiddleware from '../middleware/asyncMiddleware';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport Account from '@/models/Account';\nimport JournalPoster from '@/services/Accounting/JournalPoster';\nimport JournalEntry from '@/services/Accounting/JournalEntry';\nimport ManualJournal from '@/models/ManualJournal';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(jwtAuth);\n\n    router.post('/',\n      this.openingBalnace.validation,\n      asyncMiddleware(this.openingBalnace.handler));\n\n    return router;\n  },\n\n  /**\n   * Opening balance to the given account.\n   * @param {Request} req -\n   * @param {Response} res -\n   */\n  openingBalnace: {\n    validation: [\n      check('date').optional(),\n      check('note').optional().trim().escape(),\n      check('balance_adjustment_account').exists().isNumeric().toInt(),\n      check('accounts').isArray({ min: 1 }),\n      check('accounts.*.id').exists().isInt(),\n      oneOf([\n        check('accounts.*.debit').exists().isNumeric().toFloat(),\n        check('accounts.*.credit').exists().isNumeric().toFloat(),\n      ]),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { accounts } = req.body;\n      const { user } = req;\n      const form = { ...req.body };\n      const date = moment(form.date).format('YYYY-MM-DD');\n\n      const accountsIds = accounts.map((account) => account.id);\n      const storedAccounts = await Account.query()\n        .select(['id']).whereIn('id', accountsIds)\n        .withGraphFetched('type');\n\n      const accountsCollection = new Map(storedAccounts.map(i => [i.id, i]));\n\n      // Get the stored accounts Ids and difference with submit accounts.\n      const accountsStoredIds = storedAccounts.map((account) => account.id);\n      const notFoundAccountsIds = difference(accountsIds, accountsStoredIds);\n      const errorReasons = [];\n\n      if (notFoundAccountsIds.length > 0) {\n        const ids = notFoundAccountsIds.map((a) => parseInt(a, 10));\n        errorReasons.push({ type: 'NOT_FOUND_ACCOUNT', code: 100, ids });\n      }\n      if (form.balance_adjustment_account) {\n        const account = await Account.query().findById(form.balance_adjustment_account);\n\n        if (!account) {\n          errorReasons.push({ type: 'BALANCE.ADJUSTMENT.ACCOUNT.NOT.EXIST', code: 300 });\n        }\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badData(null, { errors: errorReasons });\n      }\n\n      const journalEntries = new JournalPoster();\n\n      accounts.forEach((account) => {\n        const storedAccount = accountsCollection.get(account.id);\n\n        // Can't continue in case the stored account was not found.\n        if (!storedAccount) { return; }\n\n        const entryModel = new JournalEntry({\n          referenceType: 'OpeningBalance',\n          account: account.id,\n          accountNormal: storedAccount.type.normal,\n          userId: user.id,\n        });\n        if (account.credit) {\n          entryModel.entry.credit = account.credit;\n          journalEntries.credit(entryModel);\n        } else if (account.debit) {\n          entryModel.entry.debit = account.debit;\n          journalEntries.debit(entryModel);\n        }\n      });\n      // Calculates the credit and debit balance of stacked entries.\n      const trial = journalEntries.getTrialBalance();\n\n      if (trial.credit !== trial.debit) {\n        const entryModel = new JournalEntry({\n          referenceType: 'OpeningBalance',\n          account: form.balance_adjustment_account,\n          accountNormal: 'credit',\n          userId: user.id,\n        });\n\n        if (trial.credit > trial.debit) {\n          entryModel.entry.credit = Math.abs(trial.credit);\n          journalEntries.credit(entryModel);\n\n        } else if (trial.credit < trial.debit) {\n          entryModel.entry.debit = Math.abs(trial.debit);\n          journalEntries.debit(entryModel);\n        }\n      }\n      const manualJournal = await ManualJournal.query().insert({\n        amount: Math.max(trial.credit, trial.debit),\n        transaction_type: 'OpeningBalance',\n        date,\n        note: form.note,\n        user_id: user.id,\n      });\n\n      journalEntries.entries = journalEntries.entries.map((entry) => ({\n        ...entry,\n        referenceId: manualJournal.id,\n      }));\n      await Promise.all([\n        journalEntries.saveEntries(),\n        journalEntries.saveBalance(),\n      ]);\n      return res.status(200).send({ id: manualJournal.id });\n    },\n  },\n};\n","import express from 'express';\nimport JWTAuth from '@/http/middleware/jwtAuth';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport AccountType from '@/models/AccountType';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n    router.use(JWTAuth);\n\n    router.get('/',\n      this.getAccountTypesList.validation,\n      asyncMiddleware(this.getAccountTypesList.handler));\n\n    return router;\n  },\n\n  /**\n   * Retrieve accounts types list.\n   */\n  getAccountTypesList: {\n    validation: [],\n    async handler(req, res) {\n      const accountTypes = await AccountType.query();\n\n      return res.status(200).send({\n        account_types: accountTypes,\n      });\n    },\n  },\n};\n","import { check, query, validationResult } from 'express-validator';\nimport express from 'express';\nimport { difference } from 'lodash';\nimport Account from '@/models/Account';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport JWTAuth from '@/http/middleware/jwtAuth';\nimport JournalPoster from '@/services/Accounting/JournalPoster';\nimport JournalEntry from '@/services/Accounting/JournalEntry';\nimport ManualJournal from '@/models/JournalEntry';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n    router.use(JWTAuth);\n\n    router.post('/make-journal-entries',\n      this.makeJournalEntries.validation,\n      asyncMiddleware(this.makeJournalEntries.handler));\n\n    router.post('/recurring-journal-entries',\n      this.recurringJournalEntries.validation,\n      asyncMiddleware(this.recurringJournalEntries.handler));\n\n    router.post('quick-journal-entries',\n      this.quickJournalEntries.validation,\n      asyncMiddleware(this.quickJournalEntries.handler));\n\n    return router;\n  },\n\n  /**\n   * Make journal entrires.\n   */\n  makeJournalEntries: {\n    validation: [\n      check('date').isISO8601(),\n      check('reference').exists(),\n      check('entries').isArray({ min: 1 }),\n      check('entries.*.credit').isNumeric().toInt(),\n      check('entries.*.debit').isNumeric().toInt(),\n      check('entries.*.account_id').isNumeric().toInt(),\n      check('entries.*.note').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = {\n        date: new Date(),\n        ...req.body,\n      };\n      const errorReasons = [];\n      let totalCredit = 0;\n      let totalDebit = 0;\n\n      form.entries.forEach((entry) => {\n        if (entry.credit > 0) {\n          totalCredit += entry.credit;\n        }\n        if (entry.debit > 0) {\n          totalDebit += entry.debit;\n        }\n      });\n      if (totalCredit <= 0 || totalDebit <= 0) {\n        errorReasons.push({\n          type: 'CREDIT.DEBIT.SUMATION.SHOULD.NOT.EQUAL.ZERO',\n          code: 400,\n        });\n      }\n      if (totalCredit !== totalDebit) {\n        errorReasons.push({ type: 'CREDIT.DEBIT.NOT.EQUALS', code: 100 });\n      }\n      const accountsIds = form.entries.map((entry) => entry.account_id);\n      const accounts = await Account.query().whereIn('id', accountsIds)\n        .withGraphFetched('type');\n\n      const storedAccountsIds = accounts.map((account) => account.id);\n\n      if (difference(accountsIds, storedAccountsIds).length > 0) {\n        errorReasons.push({ type: 'ACCOUNTS.IDS.NOT.FOUND', code: 200 });\n      }\n\n      const journalReference = await ManualJournal.query().where('reference', form.reference);\n\n      if (journalReference.length > 0) {\n        errorReasons.push({ type: 'REFERENCE.ALREADY.EXISTS', code: 300 });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      const journalPoster = new JournalPoster();\n\n      form.entries.forEach((entry) => {\n        const account = accounts.find((a) => a.id === entry.account_id);\n\n        const jouranlEntry = new JournalEntry({\n          date: entry.date,\n          debit: entry.debit,\n          credit: entry.credit,\n          account: account.id,\n          accountNormal: account.type.normal,\n          note: entry.note,\n        });\n        if (entry.debit) {\n          journalPoster.debit(jouranlEntry);\n        } else {\n          journalPoster.credit(jouranlEntry);\n        }\n      });\n\n      // Saves the journal entries and accounts balance changes.\n      await Promise.all([\n        journalPoster.saveEntries(),\n        journalPoster.saveBalance(),\n      ]);\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Saves recurring journal entries template.\n   */\n  recurringJournalEntries: {\n    validation: [\n      check('template_name').exists(),\n      check('recurrence').exists(),\n      check('active').optional().isBoolean().toBoolean(),\n      check('entries').isArray({ min: 1 }),\n      check('entries.*.credit').isNumeric().toInt(),\n      check('entries.*.debit').isNumeric().toInt(),\n      check('entries.*.account_id').isNumeric().toInt(),\n      check('entries.*.note').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n    },\n  },\n\n  recurringJournalsList: {\n    validation: [\n      query('page').optional().isNumeric().toInt(),\n      query('page_size').optional().isNumeric().toInt(),\n      query('template_name').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n    },\n  },\n\n  quickJournalEntries: {\n    validation: [\n      check('date').exists().isISO8601(),\n      check('amount').exists().isNumeric().toFloat(),\n      check('credit_account_id').exists().isNumeric().toInt(),\n      check('debit_account_id').exists().isNumeric().toInt(),\n      check('transaction_type').exists(),\n      check('note').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const errorReasons = [];\n      const form = { ...req.body };\n\n      const foundAccounts = await Account.query()\n        .where('id', form.credit_account_id)\n        .orWhere('id', form.debit_account_id);\n\n      const creditAccount = foundAccounts.find((a) => a.id === form.credit_account_id);\n      const debitAccount = foundAccounts.find((a) => a.id === form.debit_account_id);\n\n      if (!creditAccount) {\n        errorReasons.push({ type: 'CREDIT_ACCOUNT.NOT.EXIST', code: 100 });\n      }\n      if (!debitAccount) {\n        errorReasons.push({ type: 'DEBIT_ACCOUNT.NOT.EXIST', code: 200 });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n\n      // const journalPoster = new JournalPoster();\n      // const journalCredit = new JournalEntry({\n      //   debit: \n      //   account: debitAccount.id,\n      //   referenceId: \n      // })\n\n      return res.status(200).send();\n    },\n  },\n};\n","import express from 'express';\nimport { check, validationResult, param, query } from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport Account from '@/models/Account';\nimport AccountType from '@/models/AccountType';\nimport AccountTransaction from '@/models/AccountTransaction';\nimport JournalPoster from '@/services/Accounting/JournalPoster';\nimport AccountBalance from '@/models/AccountBalance';\nimport Resource from '@/models/Resource';\nimport View from '@/models/View';\nimport JWTAuth from '@/http/middleware/jwtAuth';\nimport NestedSet from '../../collection/NestedSet';\nimport {\n  mapViewRolesToConditionals,\n  validateViewRoles,\n} from '@/lib/ViewRolesBuilder';\n\nexport default {\n  /**\n   * Router constructor method.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(JWTAuth);\n    router.post('/',\n      this.newAccount.validation,\n      asyncMiddleware(this.newAccount.handler));\n\n    router.post('/:id',\n      this.editAccount.validation,\n      asyncMiddleware(this.editAccount.handler));\n\n    router.get('/:id',\n      this.getAccount.validation,\n      asyncMiddleware(this.getAccount.handler));\n\n    router.get('/',\n      this.getAccountsList.validation,\n      asyncMiddleware(this.getAccountsList.handler));\n\n    router.delete('/:id',\n      this.deleteAccount.validation,\n      asyncMiddleware(this.deleteAccount.handler));\n\n    router.post('/:id/active',\n      this.activeAccount.validation,\n      asyncMiddleware(this.activeAccount.handler));\n\n    router.post('/:id/inactive',\n      this.inactiveAccount.validation,\n      asyncMiddleware(this.inactiveAccount.handler));\n\n    router.post('/:id/recalculate-balance',\n      this.recalcualteBalanace.validation,\n      asyncMiddleware(this.recalcualteBalanace.handler));\n\n    router.post('/:id/transfer_account/:toAccount',\n      this.transferToAnotherAccount.validation,\n      asyncMiddleware(this.transferToAnotherAccount.handler));\n\n    return router;\n  },\n\n  /**\n   * Creates a new account.\n   */\n  newAccount: {\n    validation: [\n      check('name').exists().isLength({ min: 3 }).trim().escape(),\n      check('code').exists().isLength({ max: 10 }).trim().escape(),\n      check('account_type_id').exists().isNumeric().toInt(),\n      check('description').optional().trim().escape(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = { ...req.body };\n\n      const foundAccountCodePromise = form.code\n        ? Account.query().where('code', form.code) : null;\n\n      const foundAccountTypePromise = AccountType.query()\n        .findById(form.account_type_id);\n\n      const [foundAccountCode, foundAccountType] = await Promise.all([\n        foundAccountCodePromise, foundAccountTypePromise,\n      ]);\n\n      if (foundAccountCodePromise && foundAccountCode.length > 0) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'NOT_UNIQUE_CODE', code: 100 }],\n        });\n      }\n      if (!foundAccountType) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'NOT_EXIST_ACCOUNT_TYPE', code: 200 }],\n        });\n      }\n      await Account.query().insert({ ...form });\n\n      return res.status(200).send({ item: { } });\n    },\n  },\n\n  /**\n   * Edit the given account details.\n   */\n  editAccount: {\n    validation: [\n      param('id').exists().toInt(),\n      check('name').exists().isLength({ min: 3 }).trim().escape(),\n      check('code').exists().isLength({ max: 10 }).trim().escape(),\n      check('account_type_id').exists().isNumeric().toInt(),\n      check('description').optional().trim().escape(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = { ...req.body };\n      const account = await Account.query().findById(id);\n\n      if (!account) {\n        return res.boom.notFound();\n      }\n      const foundAccountCodePromise = (form.code && form.code !== account.code)\n        ? Account.query().where('code', form.code).whereNot('id', account.id) : null;\n\n      const foundAccountTypePromise = (form.account_type_id !== account.account_type_id)\n        ? AccountType.query().where('id', form.account_type_id) : null;\n\n      const [foundAccountCode, foundAccountType] = await Promise.all([\n        foundAccountCodePromise, foundAccountTypePromise,\n      ]);\n      if (foundAccountCode.length > 0 && foundAccountCodePromise) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'NOT_UNIQUE_CODE', code: 100 }],\n        });\n      }\n      if (foundAccountType.length <= 0 && foundAccountTypePromise) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'NOT_EXIST_ACCOUNT_TYPE', code: 110 }],\n        });\n      }\n      await account.patch({ ...form });\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Get details of the given account.\n   */\n  getAccount: {\n    validation: [\n      param('id').toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const account = await Account.query().where('id', id).first();\n\n      if (!account) {\n        return res.boom.notFound();\n      }\n      return res.status(200).send({ account: { ...account } });\n    },\n  },\n\n  /**\n   * Delete the given account.\n   */\n  deleteAccount: {\n    validation: [\n      param('id').toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const account = await Account.query().findById(id);\n\n      if (!account) {\n        return res.boom.notFound();\n      }\n      const accountTransactions = await AccountTransaction.query()\n        .where('account_id', account.id);\n\n      if (accountTransactions.length > 0) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'ACCOUNT.HAS.ASSOCIATED.TRANSACTIONS', code: 100 }],\n        });\n      }\n      await Account.query().deleteById(account.id);\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrieve accounts list.\n   */\n  getAccountsList: {\n    validation: [\n      query('display_type').optional().isIn(['tree', 'flat']),\n      query('account_types').optional().isArray(),\n      query('account_types.*').optional().isNumeric().toInt(),\n      query('custom_view_id').optional().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const filter = {\n        account_types: [],\n        display_type: 'tree',\n        ...req.query,\n      };\n      const errorReasons = [];\n      const viewConditionals = [];\n      const accountsResource = await Resource.query().where('name', 'accounts').first();\n\n      if (!accountsResource) {\n        return res.status(400).send({\n          errors: [{ type: 'ACCOUNTS_RESOURCE_NOT_FOUND', code: 200 }],\n        });\n      }\n      const view = await View.query().onBuild((builder) => {\n        if (filter.custom_view_id) {\n          builder.where('id', filter.custom_view_id);\n        } else {\n          builder.where('favourite', true);\n        }\n        builder.where('resource_id', accountsResource.id);\n        builder.withGraphFetched('roles.field');\n        builder.withGraphFetched('columns');\n        builder.first();\n      });\n\n      if (view && view.roles.length > 0) {\n        viewConditionals.push(\n          ...mapViewRolesToConditionals(view.roles),\n        );\n        if (!validateViewRoles(viewConditionals, view.rolesLogicExpression)) {\n          errorReasons.push({ type: 'VIEW.LOGIC.EXPRESSION.INVALID', code: 400 });\n        }\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      const accounts = await Account.query().onBuild((builder) => {\n        builder.modify('filterAccountTypes', filter.account_types);\n        builder.withGraphFetched('type');\n\n        if (viewConditionals.length) {\n          builder.modify('viewRolesBuilder', viewConditionals, view.rolesLogicExpression);\n        }\n      });\n\n      const nestedAccounts = new NestedSet(accounts, { parentId: 'parentAccountId' });\n      const groupsAccounts = nestedAccounts.toTree();\n      const accountsList = [];\n\n      if (filter.display_type === 'tree') {\n        accountsList.push(...groupsAccounts);\n      } else if (filter.display_type === 'flat') {\n        const flattenAccounts = nestedAccounts.flattenTree((account, parentAccount) => {\n          if (parentAccount) {\n            account.name = `${parentAccount.name} ― ${account.name}`;\n          }\n          return account;\n        });\n        accountsList.push(...flattenAccounts);\n      }\n      return res.status(200).send({\n        accounts: accountsList,\n        ...(view) ? {\n          customViewId: view.id,\n        } : {},\n      });\n    },\n  },\n\n  /**\n   * Re-calculates balance of the given account.\n   */\n  recalcualteBalanace: {\n    validation: [\n      param('id').isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const account = await Account.findById(id);\n\n      if (!account) {\n        return res.status(400).send({\n          errors: [{ type: 'ACCOUNT.NOT.FOUND', code: 100 }],\n        });\n      }\n      const accountTransactions = AccountTransaction.query()\n        .where('account_id', account.id);\n\n      const journalEntries = new JournalPoster();\n      journalEntries.loadFromCollection(accountTransactions);\n\n      // Delete the balance of the given account id.\n      await AccountBalance.query().where('account_id', account.id).delete();\n\n      // Save calcualted account balance.\n      await journalEntries.saveBalance();\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Active the given account.\n   */\n  activeAccount: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const account = await Account.findById(id);\n\n      if (!account) {\n        return res.status(400).send({\n          errors: [{ type: 'ACCOUNT.NOT.FOUND', code: 100 }],\n        });\n      }\n      await account.patch({ active: true });\n\n      return res.status(200).send({ id: account.id });\n    },\n  },\n\n  /**\n   * Inactive the given account.\n   */\n  inactiveAccount: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const account = await Account.findById(id);\n\n      if (!account) {\n        return res.status(400).send({\n          errors: [{ type: 'ACCOUNT.NOT.FOUND', code: 100 }],\n        });\n      }\n      await account.patch({ active: false });\n\n      return res.status(200).send({ id: account.id });\n    },\n  },\n\n  /**\n   * Transfer all journal entries of the given account to another account.\n   */\n  transferToAnotherAccount: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n      param('toAccount').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      // const { id, toAccount: toAccountId } = req.params;\n\n      // const [fromAccount, toAccount] = await Promise.all([\n      //   Account.query().findById(id),\n      //   Account.query().findById(toAccountId),\n      // ]);\n\n      // const fromAccountTransactions = await AccountTransaction.query()\n      //   .where('account_id', fromAccount);\n\n      // return res.status(200).send();\n    },\n  },\n};\n","\nimport express from 'express';\nimport { check, validationResult } from 'express-validator';\nimport path from 'path';\nimport fs from 'fs';\nimport Mustache from 'mustache';\nimport jwt from 'jsonwebtoken';\nimport asyncMiddleware from '../middleware/asyncMiddleware';\nimport User from '@/models/User';\nimport PasswordReset from '@/models/PasswordReset';\nimport mail from '@/services/mail';\nimport { hashPassword } from '@/utils';\n\nexport default {\n  /**\n   * Constructor method.\n   */\n  router() {\n    const router = express.Router();\n\n    router.post('/login',\n      this.login.validation,\n      asyncMiddleware(this.login.handler));\n\n    router.post('/send_reset_password',\n      this.sendResetPassword.validation,\n      asyncMiddleware(this.sendResetPassword.handler));\n\n    router.post('/reset/:token',\n      this.resetPassword.validation,\n      asyncMiddleware(this.resetPassword.handler));\n\n    return router;\n  },\n\n  /**\n   * User login authentication request.\n   */\n  login: {\n    validation: [\n      check('crediential').exists().isEmail(),\n      check('password').exists().isLength({ min: 4 }),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { crediential, password } = req.body;\n      const { JWT_SECRET_KEY } = process.env;\n\n      const user = await User.query()\n        .where('email', crediential)\n        .orWhere('phone_number', crediential)\n        .first();\n\n      if (!user) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'INVALID_DETAILS', code: 100 }],\n        });\n      }\n      if (!user.verifyPassword(password)) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'INVALID_DETAILS', code: 100 }],\n        });\n      }\n      if (!user.active) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'USER_INACTIVE', code: 110 }],\n        });\n      }\n      // user.update({ last_login_at: new Date() });\n\n      const token = jwt.sign({\n        email: user.email,\n        _id: user.id,\n      }, JWT_SECRET_KEY, {\n        expiresIn: '1d',\n      });\n      return res.status(200).send({ token, user });\n    },\n  },\n\n  /**\n   * Send reset password link via email or SMS.\n   */\n  sendResetPassword: {\n    validation: [\n      check('email').exists().isEmail(),\n    ],\n    // eslint-disable-next-line consistent-return\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { email } = req.body;\n      const user = await User.where('email', email).fetch();\n\n      if (!user) {\n        return res.status(422).send();\n      }\n      // Delete all stored tokens of reset password that associate to the give email.\n      await PasswordReset.where({ email }).destroy({ require: false });\n\n      const passwordReset = PasswordReset.forge({\n        email,\n        token: '123123',\n      });\n      await passwordReset.save();\n\n      const filePath = path.join(__dirname, '../../views/mail/ResetPassword.html');\n      const template = fs.readFileSync(filePath, 'utf8');\n      const rendered = Mustache.render(template, {\n        url: `${req.protocol}://${req.hostname}/reset/${passwordReset.attributes.token}`,\n        first_name: user.attributes.first_name,\n        last_name: user.attributes.last_name,\n        contact_us_email: process.env.CONTACT_US_EMAIL,\n      });\n\n      const mailOptions = {\n        to: user.attributes.email,\n        from: `${process.env.MAIL_FROM_NAME} ${process.env.MAIL_FROM_ADDRESS}`,\n        subject: 'Ratteb Password Reset',\n        html: rendered,\n      };\n\n      // eslint-disable-next-line consistent-return\n      mail.sendMail(mailOptions, (error) => {\n        if (error) {\n          return res.status(400).send();\n        }\n        res.status(200).send({ data: { email: passwordReset.attributes.email } });\n      });\n    },\n  },\n\n  /**\n   * Reset password.\n   */\n  resetPassword: {\n    validation: [\n      check('password').exists().isLength({ min: 5 }).custom((value, { req }) => {\n        if (value !== req.body.confirm_password) {\n          throw new Error(\"Passwords don't match\");\n        } else {\n          return value;\n        }\n      }),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'VALIDATION_ERROR', ...validationErrors,\n        });\n      }\n      const { token } = req.params;\n      const { password } = req.body;\n\n      const tokenModel = await PasswordReset.query()\n        .where('token', token)\n        .where('created_at', '>=', Date.now() - 3600000)\n        .first();\n\n      if (!tokenModel) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'TOKEN_INVALID', code: 100 }],\n        });\n      }\n      const user = await User.where({\n        email: tokenModel.email,\n      });\n      if (!user) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'USER_NOT_FOUND', code: 120 }],\n        });\n      }\n      const hashedPassword = await hashPassword(password);\n\n      user.password = hashedPassword;\n      await user.save();\n\n      await PasswordReset.where('email', user.get('email')).destroy({ require: false });\n\n      return res.status(200).send({});\n    },\n  },\n};\n","import express from 'express';\n\nexport default {\n\n  router() {\n    const router = express.Router();\n\n    return router;\n  },\n};\n","import express from 'express';\nimport {\n  check,\n  query,\n  param,\n  validationResult,\n} from 'express-validator';\nimport { pick, difference, groupBy } from 'lodash';\nimport asyncMiddleware from \"@/http/middleware/asyncMiddleware\";\nimport JWTAuth from '@/http/middleware/jwtAuth';\nimport Budget from '@/models/Budget';\nimport BudgetEntry from '@/models/BudgetEntry';\nimport Account from '@/models/Account';\nimport moment from '@/services/Moment';\nimport BudgetEntriesSet from '@/collection/BudgetEntriesSet';\nimport AccountType from '@/models/AccountType';\nimport NestedSet from '@/collection/NestedSet';\nimport { dateRangeFormat } from '@/utils';\n\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(JWTAuth);\n\n    router.post('/',\n      this.newBudget.validation,\n      asyncMiddleware(this.newBudget.handler));\n\n    router.get('/:id',\n      this.getBudget.validation,\n      asyncMiddleware(this.getBudget.handler));\n\n    router.get('/:id',\n      this.deleteBudget.validation,\n      asyncMiddleware(this.deleteBudget.handler));\n\n    router.get('/',\n      this.listBudgets.validation,\n      asyncMiddleware(this.listBudgets.handler));\n\n    return router;\n  },\n\n  /**\n   * Retrieve budget details of the given id.\n   */\n  getBudget: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { id } = req.params;\n      const budget = await Budget.query().findById(id);\n\n      if (!budget) {\n        return res.status(404).send({\n          errors: [{ type: 'budget.not.found', code: 100 }],\n        });\n      }\n      const accountTypes = await AccountType.query().where('balance_sheet', true);\n\n      const [budgetEntries, accounts] = await Promise.all([\n        BudgetEntry.query().where('budget_id', budget.id),\n        Account.query().whereIn('account_type_id', accountTypes.map((a) => a.id)),\n      ]);\n\n      const accountsNestedSet = new NestedSet(accounts);\n\n      const columns = [];\n      const fromDate = moment(budget.year).startOf('year')\n        .add(budget.rangeOffset, budget.rangeBy).toDate();\n\n      const toDate = moment(budget.year).endOf('year').toDate();\n\n      const dateRange = moment.range(fromDate, toDate);\n      const dateRangeCollection = Array.from(dateRange.by(budget.rangeBy, {\n        step: budget.rangeIncrement, excludeEnd: false, excludeStart: false,\n      }));\n\n      dateRangeCollection.forEach((date) => {\n        columns.push(date.format(dateRangeFormat(budget.rangeBy)));\n      });\n      const budgetEntriesSet = BudgetEntriesSet.from(budgetEntries, {\n        orderSize: columns.length,\n      });\n      budgetEntriesSet.setZeroPlaceholder();\n      budgetEntriesSet.calcTotalSummary();\n\n      return res.status(200).send({\n        columns,\n        accounts: budgetEntriesSet.toArray(),\n        total: budgetEntriesSet.toArrayTotalSummary(),\n      });\n    },\n  },\n\n  /**\n   * Delete the given budget.\n   */\n  deleteBudget: {\n    validation: [\n      param('id').exists(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { id } = req.params;\n      const budget = await Budget.query().findById(id);\n\n      if (!budget) {\n        return res.status(404).send({\n          errors: [{ type: 'budget.not.found', code: 100 }],\n        });\n      }\n      await BudgetEntry.query().where('budget_id', budget.id).delete();\n      await budget.delete();\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Saves the new budget.\n   */\n  newBudget: {\n    validation: [\n      check('name').exists(),\n      check('fiscal_year').exists(),\n      check('period').exists().isIn(['year', 'month', 'quarter', 'half-year']),\n      check('accounts_type').exists().isIn(['balance_sheet', 'profit_loss']),\n      check('accounts').isArray(),\n      check('accounts.*.account_id').exists().isNumeric().toInt(),\n      check('accounts.*.entries').exists().isArray(),\n      check('accounts.*.entries.*.amount').exists().isNumeric().toFloat(),\n      check('accounts.*.entries.*.order').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const form = { ...req.body };\n      const submitAccountsIds = form.accounts.map((a) => a.account_id);\n      const storedAccounts = await Account.query().whereIn('id', submitAccountsIds);\n      const storedAccountsIds = storedAccounts.map((a) => a.id);\n\n      const errorReasons = [];\n      const notFoundAccountsIds = difference(submitAccountsIds, storedAccountsIds);\n\n      if (notFoundAccountsIds.length > 0) {\n        errorReasons.push({\n          type: 'ACCOUNT.NOT.FOUND', code: 200, accounts: notFoundAccountsIds,\n        });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      // validation entries order.\n      const budget = await Budget.query().insert({\n        ...pick(form, ['name', 'fiscal_year', 'period', 'accounts_type']),\n      });\n\n      const promiseOpers = [];\n\n      form.accounts.forEach((account) => {\n        account.entries.forEach((entry) => {\n          const budgetEntry = BudgetEntry.query().insert({\n            account_id: account.account_id,\n            amount: entry.amount,\n            order: entry.order,\n          });\n          promiseOpers.push(budgetEntry);\n        });\n      });\n      await Promise.all(promiseOpers);\n\n      return res.status(200).send({ id: budget.id });\n    },\n  },\n\n  /**\n   * List of paginated budgets items.\n   */\n  listBudgets: {\n    validation: [\n      query('year').optional(),\n      query('income_statement').optional().isBoolean().toBoolean(),\n      query('profit_loss').optional().isBoolean().toBoolean(),\n      query('page').optional().isNumeric().toInt(),\n      query('page_size').isNumeric().toInt(),\n      query('custom_view_id').optional().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const filter = {\n        page_size: 10,\n        page: 1,\n        ...req.query,\n      };\n      const budgets = await Budget.query().runBefore((result, q) => {\n        if (filter.profit_loss) {\n          q.modify('filterByYear', filter.year);\n        }\n        if (filter.income_statement) {\n          q.modify('filterByIncomeStatement', filter.income_statement);\n        }\n        if (filter.profit_loss) {\n          q.modify('filterByProfitLoss', filter.profit_loss);\n        }\n        q.page(filter.page, filter.page_size);\n        return result;\n      });\n      return res.status(200).send({\n        items: budgets.items,\n      });\n    },\n  },\n};\n","import express from 'express';\nimport { query, validationResult } from 'express-validator';\nimport moment from 'moment';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport Budget from '@/models/Budget';\nimport Account from '@/models/Account';\nimport AccountType from '@/models/AccountType';\nimport NestedSet from '@/collection/NestedSet';\nimport BudgetEntry from '@/models/BudgetEntry';\nimport { dateRangeFormat } from '@/utils';\n\nexport default {\n\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(jwtAuth);\n\n    router.get('/budget_verses_actual/:reportId',\n      this.budgetVersesActual.validation,\n      asyncMiddleware(this.budgetVersesActual.handler));\n\n    return router;\n  },\n\n  budgetVersesActual: {\n    validation: [\n      query('basis').optional().isIn(['cash', 'accural']),\n      query('period').optional(),\n      query('active_accounts').optional().toBoolean(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { reportId } = req.params;\n      const form = { ...req.body };\n      const errorReasons = [];\n\n      const budget = await Budget.query().findById(reportId);\n\n      if (!budget) {\n        errorReasons.push({ type: 'BUDGET_NOT_FOUND', code: 100 });\n      }\n      const budgetEntries = await BudgetEntry.query().where('budget_id', budget.id);\n\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      const accountTypes = await AccountType.query()\n        .where('balance_sheet', budget.accountTypes === 'balance_sheet')\n        .where('income_sheet', budget.accountTypes === 'profit_losss');\n\n      const accounts = await Account.query().runBefore((result, q) => {\n        const accountTypesIds = accountTypes.map((t) => t.id);\n\n        if (accountTypesIds.length > 0) {\n          q.whereIn('account_type_id', accountTypesIds);\n        }\n        q.where('active', form.active_accounts === true);\n        q.withGraphFetched('transactions');\n      });\n        \n      // const accountsNestedSet = NestedSet.from(accounts);\n\n      const fromDate = moment(budget.year).startOf('year')\n        .add(budget.rangeOffset, budget.rangeBy).toDate();\n\n      const toDate = moment(budget.year).endOf('year').toDate();\n\n      const dateRange = moment.range(fromDate, toDate);\n      const dateRangeCollection = Array.from(dateRange.by(budget.rangeBy, {\n        step: budget.rangeIncrement, excludeEnd: false, excludeStart: false,\n      }));\n\n    //   // const accounts = {\n    //   //   assets: [\n    //   //     {\n    //   //       name: '',\n    //   //       code: '',\n    //   //       totalEntries: [\n    //   //         {\n\n    //   //         }\n    //   //       ],\n    //   //       children: [\n    //   //         {\n    //   //           name: '',\n    //   //           code: '',\n    //   //           entries: [\n    //   //             {\n\n    //   //             }\n    //   //           ]\n    //   //         }\n    //   //       ]\n    //   //     }\n    //   //   ]\n    //   // }\n\n      return res.status(200).send({\n        columns: dateRangeCollection.map(d => d.format(dateRangeFormat(budget.rangeBy))),\n        // accounts: {\n        //   asset: [],\n        //   liabilities: [],\n        //   equaity: [],\n\n        //   income: [],\n        //   expenses: [],\n        // }\n      });\n    },\n  },\n}","import express from 'express';\nimport { check, validationResult } from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\n\nexport default {\n\n  router() {\n    const router = express.Router();\n\n    router.get('/all',\n      this.all.validation,\n      asyncMiddleware(this.all.handler));\n\n    router.get('/registered',\n      this.registered.validation,\n      asyncMiddleware(this.registered.handler));\n\n    return router;\n  },\n\n  all: {\n    validation: [],\n    async handler(req, res) {\n\n      return res.status(200).send({\n        currencies: [\n          { currency_code: 'USD', currency_sign: '$' },\n          { currency_code: 'LYD', currency_sign: '' },\n        ],\n      });\n    },\n  },\n\n  registered: {\n    validation: [],\n    async handler(req, res) {\n\n      return res.status(200).send({\n        currencies: [\n          { currency_code: 'USD', currency_sign: '$' },\n          { currency_code: 'LYD', currency_sign: '' },\n        ],\n      });\n    },\n  },\n};","\nexport default {\n\n\n  router() {\n\n  },\n  \n  addExchangePrice: {\n    validation: {\n      \n    },\n    async handler(req, res) {\n\n    },\n  },\n}","import express from 'express';\n\nexport default {\n\n  router() {\n    const router = express.Router();\n\n    return router;\n  },\n};\n","import express from 'express';\nimport {\n  check,\n  param,\n  query,\n  validationResult,\n} from 'express-validator';\nimport moment from 'moment';\nimport { difference, chain, omit } from 'lodash';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport Expense from '@/models/Expense';\nimport Account from '@/models/Account';\nimport JournalPoster from '@/services/Accounting/JournalPoster';\nimport JournalEntry from '@/services/Accounting/JournalEntry';\nimport JWTAuth from '@/http/middleware/jwtAuth';\nimport AccountTransaction from '@/models/AccountTransaction';\nimport View from '@/models/View';\nimport Resource from '../../models/Resource';\nimport ResourceCustomFieldRepository from '@/services/CustomFields/ResourceCustomFieldRepository';\nimport {\n  validateViewRoles,\n  mapViewRolesToConditionals,\n} from '@/lib/ViewRolesBuilder';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n    router.use(JWTAuth);\n\n    router.post('/',\n      this.newExpense.validation,\n      asyncMiddleware(this.newExpense.handler));\n\n    router.post('/:id/publish',\n      this.publishExpense.validation,\n      asyncMiddleware(this.publishExpense.handler));\n\n    router.delete('/:id',\n      this.deleteExpense.validation,\n      asyncMiddleware(this.deleteExpense.handler));\n\n    router.post('/bulk',\n      this.bulkAddExpenses.validation,\n      asyncMiddleware(this.bulkAddExpenses.handler));\n\n    router.post('/:id',\n      this.updateExpense.validation,\n      asyncMiddleware(this.updateExpense.handler));\n\n    router.get('/',\n      this.listExpenses.validation,\n      asyncMiddleware(this.listExpenses.handler));\n\n    // router.get('/:id',\n    //   this.getExpense.validation,\n    //   asyncMiddleware(this.getExpense.handler));\n\n    return router;\n  },\n\n  /**\n   * Saves a new expense.\n   */\n  newExpense: {\n    validation: [\n      check('date').optional(),\n      check('payment_account_id').exists().isNumeric().toInt(),\n      check('expense_account_id').exists().isNumeric().toInt(),\n      check('description').optional(),\n      check('amount').exists().isNumeric().toFloat(),\n      check('currency_code').optional(),\n      check('exchange_rate').optional().isNumeric().toFloat(),\n      check('publish').optional().isBoolean().toBoolean(),\n      check('custom_fields').optional().isArray({ min: 1 }),\n      check('custom_fields.*.key').exists().trim().escape(),\n      check('custom_fields.*.value').exists(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = {\n        date: new Date(),\n        published: false,\n        custom_fields: [],\n        ...req.body,\n      };\n      // Convert the date to the general format.\n      form.date = moment(form.date).format('YYYY-MM-DD');\n\n      const errorReasons = [];\n      const paymentAccount = await Account.query()\n        .findById(form.payment_account_id).first();\n\n      if (!paymentAccount) {\n        errorReasons.push({ type: 'PAYMENT.ACCOUNT.NOT.FOUND', code: 100 });\n      }\n      const expenseAccount = await Account.query().findById(form.expense_account_id).first();\n\n      if (!expenseAccount) {\n        errorReasons.push({ type: 'EXPENSE.ACCOUNT.NOT.FOUND', code: 200 });\n      }\n      // const customFields = new ResourceCustomFieldRepository(Expense);\n      // await customFields.load();\n\n      // if (customFields.validateExistCustomFields()) {\n      //   errorReasons.push({ type: 'CUSTOM.FIELDS.SLUGS.NOT.EXISTS', code: 400 });\n      // }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      const expenseTransaction = await Expense.query().insertAndFetch({\n        ...omit(form, ['custom_fields']),\n      });\n      // customFields.fillCustomFields(expenseTransaction.id, form.custom_fields);\n\n      const journalEntries = new JournalPoster();\n      const creditEntry = new JournalEntry({\n        credit: form.amount,\n        referenceId: expenseTransaction.id,\n        referenceType: Expense.referenceType,\n        date: form.date,\n        account: expenseAccount.id,\n        accountNormal: 'debit',\n        draft: !form.published,\n      });\n      const debitEntry = new JournalEntry({\n        debit: form.amount,\n        referenceId: expenseTransaction.id,\n        referenceType: Expense.referenceType,\n        date: form.date,\n        account: paymentAccount.id,\n        accountNormal: 'debit',\n        draft: !form.published,\n      });\n      journalEntries.credit(creditEntry);\n      journalEntries.debit(debitEntry);\n\n      await Promise.all([\n        // customFields.saveCustomFields(expenseTransaction.id),\n        journalEntries.saveEntries(),\n        journalEntries.saveBalance(),\n      ]);\n      return res.status(200).send({ id: expenseTransaction.id });\n    },\n  },\n\n  /**\n   * Bulk add expneses to the given accounts.\n   */\n  bulkAddExpenses: {\n    validation: [\n      check('expenses').exists().isArray({ min: 1 }),\n      check('expenses.*.date').optional().isISO8601(),\n      check('expenses.*.payment_account_id').exists().isNumeric().toInt(),\n      check('expenses.*.expense_account_id').exists().isNumeric().toInt(),\n      check('expenses.*.description').optional(),\n      check('expenses.*.amount').exists().isNumeric().toFloat(),\n      check('expenses.*.currency_code').optional(),\n      check('expenses.*.exchange_rate').optional().isNumeric().toFloat(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = { ...req.body };\n      const errorReasons = [];\n\n      const paymentAccountsIds = chain(form.expenses)\n        .map((e) => e.payment_account_id).uniq().value();\n      const expenseAccountsIds = chain(form.expenses)\n        .map((e) => e.expense_account_id).uniq().value();\n\n      const [expensesAccounts, paymentAccounts] = await Promise.all([\n        Account.query().whereIn('id', expenseAccountsIds),\n        Account.query().whereIn('id', paymentAccountsIds),\n      ]);\n      const storedExpensesAccountsIds = expensesAccounts.map((a) => a.id);\n      const storedPaymentAccountsIds = paymentAccounts.map((a) => a.id);\n\n      const notFoundPaymentAccountsIds = difference(expenseAccountsIds, storedExpensesAccountsIds);\n      const notFoundExpenseAccountsIds = difference(paymentAccountsIds, storedPaymentAccountsIds);\n\n      if (notFoundPaymentAccountsIds.length > 0) {\n        errorReasons.push({\n          type: 'PAYMENY.ACCOUNTS.NOT.FOUND',\n          code: 100,\n          accounts: notFoundPaymentAccountsIds,\n        });\n      }\n      if (notFoundExpenseAccountsIds.length > 0) {\n        errorReasons.push({\n          type: 'EXPENSE.ACCOUNTS.NOT.FOUND',\n          code: 200,\n          accounts: notFoundExpenseAccountsIds,\n        });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { reasons: errorReasons });\n      }\n      const expenseSaveOpers = [];\n      const journalPoster = new JournalPoster();\n\n      form.expenses.forEach(async (expense) => {\n        const expenseSaveOper = Expense.query().insert({ ...expense });\n        expenseSaveOpers.push(expenseSaveOper);\n      });\n      // Wait unit save all expense transactions.\n      const savedExpenseTransactions = await Promise.all(expenseSaveOpers);\n\n      savedExpenseTransactions.forEach((expense) => {\n        const date = moment(expense.date).format('YYYY-DD-MM');\n\n        const debit = new JournalEntry({\n          debit: expense.amount,\n          referenceId: expense.id,\n          referenceType: Expense.referenceType,\n          account: expense.payment_account_id,\n          accountNormal: 'debit',\n          date,\n        });\n        const credit = new JournalEntry({\n          credit: expense.amount,\n          referenceId: expense.id,\n          referenceType: Expense.referenceId,\n          account: expense.expense_account_id,\n          accountNormal: 'debit',\n          date,\n        });\n        journalPoster.credit(credit);\n        journalPoster.debit(debit);\n      });\n\n      // Save expense journal entries and balance change.\n      await Promise.all([\n        journalPoster.saveEntries(),\n        journalPoster.saveBalance(),\n      ]);\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Publish the given expense id.\n   */\n  publishExpense: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { id } = req.params;\n      const errorReasons = [];\n      const expense = await Expense.query().findById(id);\n\n      if (!expense) {\n        errorReasons.push({ type: 'EXPENSE.NOT.FOUND', code: 100 });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n\n      if (expense.published) {\n        errorReasons.push({ type: 'EXPENSE.ALREADY.PUBLISHED', code: 200 });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n\n      await AccountTransaction.query()\n        .where('reference_id', expense.id)\n        .where('reference_type', 'Expense')\n        .patch({\n          draft: false,\n        });\n\n      await Expense.query()\n        .where('id', expense.id)\n        .update({ published: true });\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrieve paginated expenses list.\n   */\n  listExpenses: {\n    validation: [\n      query('expense_account_id').optional().isNumeric().toInt(),\n      query('payment_account_id').optional().isNumeric().toInt(),\n      query('note').optional(),\n      query('range_from').optional().isNumeric().toFloat(),\n      query('range_to').optional().isNumeric().toFloat(),\n      query('date_from').optional().isISO8601(),\n      query('date_to').optional().isISO8601(),\n      query('column_sort_order').optional().isIn(['created_at', 'date', 'amount']),\n      query('sort_order').optional().isIn(['desc', 'asc']),\n      query('page').optional().isNumeric().toInt(),\n      query('page_size').optional().isNumeric().toInt(),\n      query('custom_view_id').optional().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        page_size: 10,\n        page: 1,\n        ...req.query,\n      };\n      const errorReasons = [];\n      const expenseResource = await Resource.query().where('name', 'expenses').first();\n\n      if (!expenseResource) {\n        errorReasons.push({ type: 'EXPENSE_RESOURCE_NOT_FOUND', code: 300 });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      const view = await View.query().onBuild((builder) => {\n        if (filter.custom_view_id) {\n          builder.where('id', filter.custom_view_id);\n        } else {\n          builder.where('favourite', true);\n        }\n        builder.where('resource_id', expenseResource.id);\n        builder.withGraphFetched('viewRoles.field');\n        builder.withGraphFetched('columns');\n\n        builder.first();\n      });\n      let viewConditionals = [];\n\n      if (view && view.viewRoles.length > 0) {\n        viewConditionals = mapViewRolesToConditionals(view.viewRoles);\n\n        if (!validateViewRoles(viewConditionals, view.rolesLogicExpression)) {\n          errorReasons.push({ type: 'VIEW.LOGIC.EXPRESSION.INVALID', code: 400 })\n        }\n      }\n      if (!view && filter.custom_view_id) {\n        errorReasons.push({ type: 'VIEW_NOT_FOUND', code: 100 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      const expenses = await Expense.query().onBuild((builder) => {\n        builder.withGraphFetched('paymentAccount');\n        builder.withGraphFetched('expenseAccount');\n        builder.withGraphFetched('user');\n\n        if (viewConditionals.length) {\n          builder.modify('viewRolesBuilder', viewConditionals, view.rolesLogicExpression);\n        }\n        builder.modify('filterByAmountRange', filter.range_from, filter.to_range);\n        builder.modify('filterByDateRange', filter.date_from, filter.date_to);\n        builder.modify('filterByExpenseAccount', filter.expense_account_id);\n        builder.modify('filterByPaymentAccount', filter.payment_account_id);\n        builder.modify('orderBy', filter.column_sort_order, filter.sort_order);\n      }).page(filter.page - 1, filter.page_size);\n\n      return res.status(200).send({\n        ...(view) ? {\n          customViewId: view.id, \n          viewColumns: view.columns,\n          viewConditionals,\n        } : {},\n        expenses,\n      });\n    },\n  },\n\n  /**\n   * Delete the given account.\n   */\n  deleteExpense: {\n    validation: [\n      param('id').isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { id } = req.params;\n      const expenseTransaction = await Expense.query().findById(id);\n\n      if (!expenseTransaction) {\n        return res.status(404).send({\n          errors: [{ type: 'EXPENSE.TRANSACTION.NOT.FOUND', code: 100 }],\n        });\n      }\n      const expenseEntries = await AccountTransaction.query()\n        .where('reference_type', 'Expense')\n        .where('reference_id', expenseTransaction.id);\n\n      const expenseEntriesCollect = new JournalPoster();\n      expenseEntriesCollect.loadEntries(expenseEntries);\n      expenseEntriesCollect.reverseEntries();\n\n      await Promise.all([\n        Expense.query().findById(expenseTransaction.id).delete(),\n        expenseEntriesCollect.deleteEntries(),\n        expenseEntriesCollect.saveBalance(),\n      ]);\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Update details of the given account.\n   */\n  updateExpense: {\n    validation: [\n      param('id').isNumeric().toInt(),\n      check('date').optional().isISO8601(),\n      check('payment_account_id').exists().isNumeric().toInt(),\n      check('expense_account_id').exists().isNumeric().toInt(),\n      check('description').optional(),\n      check('amount').exists().isNumeric().toFloat(),\n      check('currency_code').optional(),\n      check('exchange_rate').optional().isNumeric().toFloat(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { id } = req.params;\n      const expenseTransaction = await Expense.query().findById(id);\n\n      if (!expenseTransaction) {\n        return res.status(404).send({\n          errors: [{ type: 'EXPENSE.TRANSACTION.NOT.FOUND', code: 100 }],\n        });\n      }\n    },\n  },\n\n  /**\n   * Retrieve details of the given expense id.\n   */\n  getExpense: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { id } = req.params;\n      const expenseTransaction = await Expense.query().findById(id);\n\n      if (!expenseTransaction) {\n        return res.status(404).send({\n          errors: [{ type: 'EXPENSE.TRANSACTION.NOT.FOUND', code: 100 }],\n        });\n      }\n\n      const expenseCFMetadataRepo = new ResourceCustomFieldRepository(Expense);\n      await expenseCFMetadataRepo.load();\n      await expenseCFMetadataRepo.fetchCustomFieldsMetadata(expenseTransaction.id);\n\n      const expenseCusFieldsMetadata = expenseCFMetadataRepo.getMetadata(expenseTransaction.id);\n\n      return res.status(200).send({\n        ...expenseTransaction,\n        custom_fields: [\n          ...expenseCusFieldsMetadata.toArray(),\n        ],\n      });\n    },\n  },\n};\n","import express from 'express';\nimport { check, param, validationResult } from 'express-validator';\nimport ResourceField from '@/models/ResourceField';\nimport Resource from '@/models/Resource';\nimport asyncMiddleware from '../middleware/asyncMiddleware';\n\n/**\n * Types of the custom fields.\n */\nconst TYPES = ['text', 'email', 'number', 'url', 'percentage', 'checkbox', 'radio', 'textarea'];\n\nexport default {\n  /**\n   * Router constructor method.\n   */\n  router() {\n    const router = express.Router();\n\n    router.post('/resource/:resource_name',\n      this.addNewField.validation,\n      asyncMiddleware(this.addNewField.handler));\n\n    router.post('/:field_id',\n      this.editField.validation,\n      asyncMiddleware(this.editField.handler));\n\n    router.post('/status/:field_id',\n      this.changeStatus.validation,\n      asyncMiddleware(this.changeStatus.handler));\n\n    // router.get('/:field_id',\n    //   asyncMiddleware(this.getField.handler));\n\n    // router.delete('/:field_id',\n    //   asyncMiddleware(this.deleteField.handler));\n\n    return router;\n  },\n\n  /**\n   * Adds a new field control to the given resource.\n   * @param {Request} req -\n   * @param {Response} res -\n   */\n  addNewField: {\n    validation: [\n      param('resource_name').exists().trim().escape(),\n      check('label').exists().escape().trim(),\n      check('data_type').exists().isIn(TYPES),\n      check('help_text').optional(),\n      check('default').optional(),\n      check('options').optional().isArray(),\n      check('options.*.key').exists().isNumeric().toInt(),\n      check('options.*.value').exists(),\n    ],\n    async handler(req, res) {\n      const { resource_name: resourceName } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const resource = await Resource.query().where('name', resourceName).first();\n\n      if (!resource) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'RESOURCE_NOT_FOUND', code: 100 }],\n        });\n      }\n      const form = { options: [], ...req.body };\n      const choices = form.options.map((option) => ({ key: option.key, value: option.value }));\n\n      const storedResource = await ResourceField.query().insertAndFetch({\n        data_type: form.data_type,\n        label_name: form.label,\n        help_text: form.help_text,\n        default: form.default,\n        resource_id: resource.id,\n        options: choices,\n        index: -1,\n      });\n      return res.status(200).send({ id: storedResource.id });\n    },\n  },\n\n  /**\n   * Edit details of the given field.\n   */\n  editField: {\n    validation: [\n      param('field_id').exists().isNumeric().toInt(),\n      check('label').exists().escape().trim(),\n      check('data_type').exists().isIn(TYPES),\n      check('help_text').optional(),\n      check('default').optional(),\n      check('options').optional().isArray(),\n      check('options.*.key').exists().isNumeric().toInt(),\n      check('options.*.value').exists(),\n    ],\n    async handler(req, res) {\n      const { field_id: fieldId } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const field = await ResourceField.query().findById(fieldId);\n\n      if (!field) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'FIELD_NOT_FOUND', code: 100 }],\n        });\n      }\n      // Sets the default value of optional fields.\n      const form = { options: [], ...req.body };\n      const choices = form.options.map((option) => ({ key: option.key, value: option.value }));\n\n      await ResourceField.query().findById(field.id).update({\n        data_type: form.data_type,\n        label_name: form.label,\n        help_text: form.help_text,\n        default: form.default,\n        options: choices,\n      });\n      return res.status(200).send({ id: field.id });\n    },\n  },\n\n  /**\n   * Retrieve the fields list of the given resource.\n   * @param {Request} req -\n   * @param {Response} res -\n   */\n  fieldsList: {\n    validation: [\n      param('resource_name').toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { resource_name: resourceName } = req.params;\n      const resource = await Resource.query().where('name', resourceName).first();\n\n      if (!resource) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'RESOURCE_NOT_FOUND', code: 100 }],\n        });\n      }\n      const fields = await ResourceField.where('resource_id', resource.id).fetchAll();\n\n      return res.status(200).send({ fields: fields.toJSON() });\n    },\n  },\n\n  /**\n   * Change status of the given field.\n   */\n  changeStatus: {\n    validation: [\n      param('field_id').toInt(),\n      check('active').isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const { field_id: fieldId } = req.params;\n      const field = await ResourceField.query().findById(fieldId);\n\n      if (!field) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'NOT_FOUND_FIELD', code: 100 }],\n        });\n      }\n\n      const { active } = req.body;\n      await ResourceField.query().findById(field.id).patch({ active });\n\n      return res.status(200).send({ id: field.id });\n    },\n  },\n\n  /**\n   * Retrieve details of the given field.\n   */\n  getField: {\n    validation: [\n      param('field_id').toInt(),\n    ],\n    async handler(req, res) {\n      const { field_id: id } = req.params;\n      const field = await ResourceField.where('id', id).fetch();\n\n      if (!field) {\n        return res.boom.notFound();\n      }\n\n      return res.status(200).send({\n        field: field.toJSON(),\n      });\n    },\n  },\n\n  /**\n   * Delete the given field.\n   */\n  deleteField: {\n    validation: [\n      param('field_id').toInt(),\n    ],\n    async handler(req, res) {\n      const { field_id: id } = req.params;\n      const field = await ResourceField.where('id', id).fetch();\n\n      if (!field) {\n        return res.boom.notFound();\n      }\n      if (field.attributes.predefined) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'PREDEFINED_FIELD', code: 100 }],\n        });\n      }\n      await field.destroy();\n\n      return res.status(200).send({ id: field.get('id') });\n    },\n  },\n};\n","import express from 'express';\nimport { query, validationResult } from 'express-validator';\nimport moment from 'moment';\nimport { pick } from 'lodash';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport AccountTransaction from '@/models/AccountTransaction';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport AccountType from '@/models/AccountType';\nimport Account from '@/models/Account';\nimport JournalPoster from '@/services/Accounting/JournalPoster';\nimport { dateRangeCollection } from '@/utils';\n\nconst formatNumberClosure = (filter) => (balance) => {\n  let formattedBalance = parseFloat(balance);\n\n  if (filter.no_cents) {\n    formattedBalance = parseInt(formattedBalance, 10);\n  }\n  if (filter.divide_1000) {\n    formattedBalance /= 1000;\n  }\n  return formattedBalance;\n};\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n    router.use(jwtAuth);\n\n    router.get('/ledger',\n      this.ledger.validation,\n      asyncMiddleware(this.ledger.handler));\n\n    router.get('/general_ledger',\n      this.generalLedger.validation,\n      asyncMiddleware(this.generalLedger.handler));\n\n    router.get('/balance_sheet',\n      this.balanceSheet.validation,\n      asyncMiddleware(this.balanceSheet.handler));\n\n    router.get('/trial_balance_sheet',\n      this.trialBalanceSheet.validation,\n      asyncMiddleware(this.trialBalanceSheet.handler));\n\n    router.get('/profit_loss_sheet',\n      this.profitLossSheet.validation,\n      asyncMiddleware(this.profitLossSheet.handler));\n\n    router.get('/cash_flow_statement',\n      this.cashFlowStatement.validation,\n      asyncMiddleware(this.cashFlowStatement.handler));\n\n    return router;\n  },\n\n  /**\n   * Retrieve the ledger report of the given account.\n   */\n  ledger: {\n    validation: [\n      query('from_date').optional().isISO8601(),\n      query('to_date').optional().isISO8601(),\n      query('transaction_types').optional().isArray({ min: 1 }),\n      query('account_ids').optional().isArray({ min: 1 }),\n      query('account_ids.*').optional().isNumeric().toInt(),\n      query('from_range').optional().isNumeric().toInt(),\n      query('to_range').optional().isNumeric().toInt(),\n      query('number_format.no_cents').optional().isBoolean().toBoolean(),\n      query('number_format.divide_1000').optional().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        from_range: null,\n        to_range: null,\n        account_ids: [],\n        transaction_types: [],\n        number_format: {\n          no_cents: false,\n          divide_1000: false,\n        },\n        ...req.query,\n      };\n      const accountsJournalEntries = await AccountTransaction.query()\n        .modify('filterDateRange', filter.from_date, filter.to_date)\n        .modify('filterAccounts', filter.account_ids)\n        .modify('filterTransactionTypes', filter.transaction_types)\n        .modify('filterAmountRange', filter.from_range, filter.to_range)\n        .withGraphFetched('account');\n\n      const formatNumber = formatNumberClosure(filter.number_format);\n\n      return res.status(200).send({\n        meta: { ...filter },\n        items: accountsJournalEntries.map((entry) => ({\n          ...entry,\n          credit: formatNumber(entry.credit),\n          debit: formatNumber(entry.debit),\n        })),\n      });\n    },\n  },\n\n  /**\n   * Retrieve the general ledger financial statement.\n   */\n  generalLedger: {\n    validation: [\n      query('from_date').optional().isISO8601(),\n      query('to_date').optional().isISO8601(),\n      query('basis').optional(),\n      query('number_format.no_cents').optional().isBoolean().toBoolean(),\n      query('number_format.divide_1000').optional().isBoolean().toBoolean(),\n      query('none_zero').optional().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        from_date: moment().startOf('year').format('YYYY-MM-DD'),\n        to_date: moment().endOf('year').format('YYYY-MM-DD'),\n        number_format: {\n          no_cents: false,\n          divide_1000: false,\n        },\n        none_zero: false,\n        ...req.query,\n      };\n      const accounts = await Account.query()\n        .orderBy('index', 'DESC')\n        .withGraphFetched('transactions')\n        .modifyGraph('transactions', (builder) => {\n          builder.modify('filterDateRange', filter.from_date, filter.to_date);\n        });\n\n      const openingBalanceTransactions = await AccountTransaction.query()\n        .modify('filterDateRange', null, filter.from_date)\n        .modify('sumationCreditDebit')\n        .withGraphFetched('account.type');\n\n      const closingBalanceTransactions = await AccountTransaction.query()\n        .modify('filterDateRange', null, filter.to_date)\n        .modify('sumationCreditDebit')\n        .withGraphFetched('account.type');\n\n      const opeingBalanceCollection = new JournalPoster();\n      const closingBalanceCollection = new JournalPoster();\n\n      opeingBalanceCollection.loadEntries(openingBalanceTransactions);\n      closingBalanceCollection.loadEntries(closingBalanceTransactions);\n\n      // Transaction amount formatter based on the given query.\n      const formatNumber = formatNumberClosure(filter.number_format);\n\n      const items = [\n        ...accounts\n          .filter((account) => (\n            account.transactions.length > 0 || !filter.none_zero\n          ))\n          .map((account) => ({\n            ...pick(account, ['id', 'name', 'code', 'index']),\n            transactions: [\n              ...account.transactions.map((transaction) => ({\n                ...transaction,\n                credit: formatNumber(transaction.credit),\n                debit: formatNumber(transaction.debit),\n              })),\n            ],\n            opening: {\n              date: filter.from_date,\n              balance: opeingBalanceCollection.getClosingBalance(account.id),\n            },\n            closing: {\n              date: filter.to_date,\n              balance: closingBalanceCollection.getClosingBalance(account.id),\n            },\n          })),\n      ];\n      return res.status(200).send({\n        meta: { ...filter },\n        items,\n      });\n    },\n  },\n\n  /**\n   * Retrieve the balance sheet.\n   */\n  balanceSheet: {\n    validation: [\n      query('accounting_method').optional().isIn(['cash', 'accural']),\n      query('from_date').optional(),\n      query('to_date').optional(),\n      query('display_columns_by').optional().isIn(['year', 'month', 'week', 'day', 'quarter']),\n      query('number_format.no_cents').optional().isBoolean().toBoolean(),\n      query('number_format.divide_1000').optional().isBoolean().toBoolean(),\n      query('none_zero').optional().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        display_columns_by: 'year',\n        from_date: moment().startOf('year').format('YYYY-MM-DD'),\n        to_date: moment().endOf('year').format('YYYY-MM-DD'),\n        number_format: {\n          no_cents: false,\n          divide_1000: false,\n        },\n        none_zero: false,\n        ...req.query,\n      };\n\n      const balanceSheetTypes = await AccountType.query()\n        .where('balance_sheet', true);\n\n      // Fetch all balance sheet accounts.\n      const accounts = await Account.query()\n        .whereIn('account_type_id', balanceSheetTypes.map((a) => a.id))\n        .withGraphFetched('type')\n        .withGraphFetched('transactions')\n        .modifyGraph('transactions', (builder) => {\n          builder.modify('filterDateRange', null, filter.to_date);\n        });\n\n      const journalEntriesCollected = Account.collectJournalEntries(accounts);\n      const journalEntries = new JournalPoster();\n      journalEntries.loadEntries(journalEntriesCollected);\n\n      // Account balance formmatter based on the given query.\n      const balanceFormatter = formatNumberClosure(filter.number_format);\n\n      // Gets the date range set from start to end date.\n      const dateRangeSet = dateRangeCollection(\n        filter.from_date,\n        filter.to_date,\n        filter.display_columns_by,\n      );\n      // Retrieve the asset balance sheet.\n      const assets = [\n        ...accounts\n          .filter((account) => (\n            account.type.normal === 'debit'\n            && (account.transactions.length > 0 || !filter.none_zero)\n          ))\n          .map((account) => ({\n            ...pick(account, ['id', 'index', 'name', 'code']),\n            transactions: dateRangeSet.map((date) => {\n              const type = filter.display_columns_by;\n              const balance = journalEntries.getClosingBalance(account.id, date, type);\n              return { date, balance: balanceFormatter(balance) };\n            }),\n          })),\n      ];\n      // Retrieve liabilities and equity balance sheet.\n      const liabilitiesEquity = [\n        ...accounts\n          .filter((account) => (\n            account.type.normal === 'credit'\n            && (account.transactions.length > 0 || !filter.none_zero)\n          ))\n          .map((account) => ({\n            ...pick(account, ['id', 'index', 'name', 'code']),\n            transactions: dateRangeSet.map((date) => {\n              const type = filter.display_columns_by;\n              const balance = journalEntries.getClosingBalance(account.id, date, type);\n              return { date, balance: balanceFormatter(balance) };\n            }),\n          })),\n      ];\n      return res.status(200).send({\n        query: { ...filter },\n        columns: { ...dateRangeSet },\n        balance_sheet: {\n          assets,\n          liabilities_equity: liabilitiesEquity,\n        },\n      });\n    },\n  },\n\n  /**\n   * Retrieve the trial balance sheet.\n   */\n  trialBalanceSheet: {\n    validation: [\n      query('basis').optional(),\n      query('from_date').optional().isISO8601(),\n      query('to_date').optional().isISO8601(),\n      query('number_format.no_cents').optional().isBoolean(),\n      query('number_format.1000_divide').optional().isBoolean(),\n      query('basis').optional(),\n      query('none_zero').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        from_date: moment().startOf('year').format('YYYY-MM-DD'),\n        to_date: moment().endOf('year').format('YYYY-MM-DD'),\n        number_format: {\n          no_cents: false,\n          divide_1000: false,\n        },\n        basis: 'accural',\n        none_zero: false,\n        ...req.query,\n      };\n\n      const accounts = await Account.query()\n        .withGraphFetched('type')\n        .withGraphFetched('transactions')\n        .modifyGraph('transactions', (builder) => {\n          builder.modify('sumationCreditDebit');\n          builder.modify('filterDateRange', filter.from_date, filter.to_date);\n        });\n\n      const journalEntriesCollect = Account.collectJournalEntries(accounts);\n      const journalEntries = new JournalPoster();\n      journalEntries.loadEntries(journalEntriesCollect);\n\n      // Account balance formmatter based on the given query.\n      const balanceFormatter = formatNumberClosure(filter.number_format);\n\n      const items = accounts\n        .filter((account) => (\n          account.transactions.length > 0 || !filter.none_zero\n        ))\n        .map((account) => {\n          const trial = journalEntries.getTrialBalance(account.id);\n          return {\n            account_id: account.id,\n            code: account.code,\n            accountNormal: account.type.normal,\n            credit: balanceFormatter(trial.credit),\n            debit: balanceFormatter(trial.debit),\n            balance: balanceFormatter(trial.balance),\n          };\n        });\n      return res.status(200).send({\n        meta: { ...filter },\n        items: [...items],\n      });\n    },\n  },\n\n  /**\n   * Retrieve profit/loss financial statement.\n   */\n  profitLossSheet: {\n    validation: [\n      query('basis').optional(),\n      query('from_date').optional().isISO8601(),\n      query('to_date').optional().isISO8601(),\n      query('number_format.no_cents').optional().isBoolean(),\n      query('number_format.divide_1000').optional().isBoolean(),\n      query('basis').optional(),\n      query('none_zero').optional(),\n      query('display_columns_by').optional().isIn(['year', 'month', 'week', 'day', 'quarter']),\n      query('accounts').optional().isArray(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        from_date: moment().startOf('year').format('YYYY-MM-DD'),\n        to_date: moment().endOf('year').format('YYYY-MM-DD'),\n        number_format: {\n          no_cents: false,\n          divide_1000: false,\n        },\n        basis: 'accural',\n        none_zero: false,\n        display_columns_by: 'month',\n        ...req.query,\n      };\n      const incomeStatementTypes = await AccountType.query().where('income_sheet', true);\n\n      const accounts = await Account.query()\n        .whereIn('account_type_id', incomeStatementTypes.map((t) => t.id))\n        .withGraphFetched('type')\n        .withGraphFetched('transactions');\n\n      const filteredAccounts = accounts.filter((account) => {\n        return account.transactions.length > 0 || !filter.none_zero;\n      });\n      const journalEntriesCollected = Account.collectJournalEntries(accounts);\n      const journalEntries = new JournalPoster();\n      journalEntries.loadEntries(journalEntriesCollected);\n\n      // Account balance formmatter based on the given query.\n      const numberFormatter = formatNumberClosure(filter.number_format);\n\n      // Gets the date range set from start to end date.\n      const dateRangeSet = dateRangeCollection(\n        filter.from_date,\n        filter.to_date,\n        filter.display_columns_by,\n      );\n      const accountsIncome = filteredAccounts\n        .filter((account) => account.type.normal === 'credit')\n        .map((account) => ({\n          ...pick(account, ['id', 'index', 'name', 'code']),\n          dates: dateRangeSet.map((date) => {\n            const type = filter.display_columns_by;\n            const amount = journalEntries.getClosingBalance(account.id, date, type);\n\n            return { date, rawAmount: amount, amount: numberFormatter(amount) };\n          }),\n        }));\n\n      const accountsExpenses = filteredAccounts\n        .filter((account) => account.type.normal === 'debit')\n        .map((account) => ({\n          ...pick(account, ['id', 'index', 'name', 'code']),\n          dates: dateRangeSet.map((date) => {\n            const type = filter.display_columns_by;\n            const amount = journalEntries.getClosingBalance(account.id, date, type);\n\n            return { date, rawAmount: amount, amount: numberFormatter(amount) };\n          }),\n        }));\n\n      // Calculates the total income of income accounts.\n      const totalAccountsIncome = dateRangeSet.reduce((acc, date, index) => {\n        let amount = 0;\n        accountsIncome.forEach((account) => {\n          const currentDate = account.dates[index];\n          amount += currentDate.rawAmount || 0;\n        });\n        acc[date] = { date, rawAmount: amount, amount: numberFormatter(amount) };\n        return acc;\n      }, {});\n\n      // Calculates the total expenses of expenses accounts.\n      const totalAccountsExpenses = dateRangeSet.reduce((acc, date, index) => {\n        let amount = 0;\n        accountsExpenses.forEach((account) => {\n          const currentDate = account.dates[index];\n          amount += currentDate.rawAmount || 0;\n        });\n        acc[date] = { date, rawAmount: amount, amount: numberFormatter(amount) };\n        return acc;\n      }, {});\n\n      // Total income(date) - Total expenses(date) = Net income(date)\n      const netIncome = dateRangeSet.map((date) => {\n        const totalIncome = totalAccountsIncome[date];\n        const totalExpenses = totalAccountsExpenses[date];\n\n        let amount = totalIncome.rawAmount || 0;\n        amount -= totalExpenses.rawAmount || 0;\n        return { date, rawAmount: amount, amount: numberFormatter(amount) };\n      });\n\n      return res.status(200).send({\n        meta: { ...filter },\n        income: {\n          entry_normal: 'credit',\n          accounts: accountsIncome,\n        },\n        expenses: {\n          entry_normal: 'debit',\n          accounts: accountsExpenses,\n        },\n        total_income: Object.values(totalAccountsIncome),\n        total_expenses: Object.values(totalAccountsExpenses),\n        total_net_income: netIncome,\n      });\n    },\n  },\n\n  cashFlowStatement: {\n    validation: [\n      query('date_from').optional(),\n      query('date_to').optional(),\n    ],\n    async handler(req, res) {\n      \n      return res.status(200).send({\n        meta: {},\n        operating: [],\n        financing: [],\n        investing: [],\n      });\n    },\n  },\n}\n","import express from 'express';\nimport { check, param, validationResult } from 'express-validator';\nimport asyncMiddleware from '../middleware/asyncMiddleware';\nimport ItemCategory from '@/models/ItemCategory';\nimport Authorization from '@/http/middleware/authorization';\nimport JWTAuth from '@/http/middleware/jwtAuth';\n\nexport default {\n  /**\n   * Router constructor method.\n   */\n  router() {\n    const router = express.Router();\n    const permit = Authorization('items_categories');\n\n    router.use(JWTAuth);\n\n    router.post('/:id',\n      permit('create', 'edit'),\n      this.editCategory.validation,\n      asyncMiddleware(this.editCategory.handler));\n\n    router.post('/',\n      permit('create'),\n      this.newCategory.validation,\n      asyncMiddleware(this.newCategory.handler));\n\n    router.delete('/:id',\n      permit('create', 'edit', 'delete'),\n      this.deleteItem.validation,\n      asyncMiddleware(this.deleteItem.handler));\n\n    router.get('/:id',\n      permit('view'),\n      this.getCategory.validation,\n      asyncMiddleware(this.getCategory.handler));\n\n    router.get('/',\n      permit('view'),\n      this.getList.validation,\n      asyncMiddleware(this.getList.validation));\n\n    return router;\n  },\n\n  /**\n   * Creates a new item category.\n   */\n  newCategory: {\n    validation: [\n      check('name').exists({ checkFalsy: true }).trim().escape(),\n      check('parent_category_id').optional().isNumeric().toInt(),\n      check('description').optional().trim().escape(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { name, parent_category_id: parentCategoryId, description } = req.body;\n\n      if (parentCategoryId) {\n        const foundParentCategory = await ItemCategory.where('id', parentCategoryId).fetch();\n\n        if (!foundParentCategory) {\n          return res.boom.notFound('The parent category ID is not found.', {\n            errors: [{ type: 'PARENT_CATEGORY_NOT_FOUND', code: 100 }],\n          });\n        }\n      }\n      const category = await ItemCategory.forge({\n        label: name,\n        parent_category_id: parentCategoryId,\n        description,\n      });\n\n      await category.save();\n      return res.status(200).send({ id: category.get('id') });\n    },\n  },\n\n  /**\n   * Edit details of the given category item.\n   */\n  editCategory: {\n    validation: [\n      param('id').toInt(),\n      check('name').exists({ checkFalsy: true }).trim().escape(),\n      check('parent_category_id').optional().isNumeric().toInt(),\n      check('description').optional().trim().escape(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { name, parent_category_id: parentCategoryId, description } = req.body;\n      const itemCategory = await ItemCategory.where('id', id).fetch();\n\n      if (!itemCategory) {\n        return res.boom.notFound();\n      }\n      if (parentCategoryId && parentCategoryId !== itemCategory.attributes.parent_category_id) {\n        const foundParentCategory = await ItemCategory.where('id', parentCategoryId).fetch();\n\n        if (!foundParentCategory) {\n          return res.boom.notFound('The parent category ID is not found.', {\n            errors: [{ type: 'PARENT_CATEGORY_NOT_FOUND', code: 100 }],\n          });\n        }\n      }\n      await itemCategory.save({\n        label: name,\n        description,\n        parent_category_id: parentCategoryId,\n      });\n\n      return res.status(200).send({ id: itemCategory.id });\n    },\n  },\n\n  /**\n   * Delete the give item category.\n   */\n  deleteItem: {\n    validation: [\n      param('id').toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const itemCategory = await ItemCategory.where('id', id).fetch();\n\n      if (!itemCategory) {\n        return res.boom.notFound();\n      }\n      await itemCategory.destroy();\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrieve the list of items.\n   */\n  getList: {\n    validation: [],\n    async handler(req, res) {\n      const items = await ItemCategory.fetch();\n\n      if (!items) {\n        return res.boom.notFound();\n      }\n      return res.status(200).send({ items: items.toJSON() });\n    },\n  },\n\n  /**\n   * Retrieve details of the given category.\n   */\n  getCategory: {\n    validation: [\n      param('category_id').toInt(),\n    ],\n    async handler(req, res) {\n      const { category_id: categoryId } = req.params;\n      const item = await ItemCategory.where('id', categoryId).fetch();\n\n      if (!item) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'CATEGORY_NOT_FOUND', code: 100 }],\n        });\n      }\n\n      return res.status(200).send({ category: item.toJSON() });\n    },\n  },\n};\n","import express from 'express';\nimport { check, validationResult } from 'express-validator';\nimport moment from 'moment';\nimport { difference } from 'lodash';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport Item from '@/models/Item';\nimport Account from '@/models/Account';\nimport ItemCategory from '@/models/ItemCategory';\nimport Resource from '@/models/Resource';\nimport ResourceField from '@/models/ResourceField';\nimport Authorization from '@/http/middleware/authorization';\n\nexport default {\n\n  router() {\n    const router = express.Router();\n    const permit = Authorization('items');\n\n    router.use(jwtAuth);\n\n    router.post('/:id',\n      this.editItem.validation,\n      asyncMiddleware(this.editItem.handler));\n\n    router.post('/',\n      // permit('create'),\n      this.newItem.validation,\n      asyncMiddleware(this.newItem.handler));\n\n    router.delete('/:id',\n      this.deleteItem.validation,\n      asyncMiddleware(this.deleteItem.handler));\n\n    // router.get('/:id',\n    //   this.getCategory.validation,\n    //   asyncMiddleware(this.getCategory.handler));\n\n    // router.get('/',\n    //   this.categoriesList.validation,\n    //   asyncMiddleware(this.categoriesList.validation));\n\n    return router;\n  },\n\n  /**\n   * Creates a new item.\n   */\n  newItem: {\n    validation: [\n      check('name').exists(),\n      check('type').exists().trim().escape().isIn(['service', 'product']),\n      check('cost_price').exists().isNumeric(),\n      check('sell_price').exists().isNumeric(),\n      check('cost_account_id').exists().isInt().toInt(),\n      check('sell_account_id').exists().isInt().toInt(),\n      check('category_id').optional().isInt().toInt(),\n\n      check('custom_fields').optional().isArray({ min: 1 }),\n      check('custom_fields.*.key').exists().isNumeric().toInt(),\n      check('custom_fields.*.value').exists(),\n\n      check('note').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = {\n        custom_fields: [],\n        ...req.body,\n      };\n      const errorReasons = [];\n\n      const costAccountPromise = Account.query().findById(form.cost_account_id);\n      const sellAccountPromise = Account.query().findById(form.sell_account_id);\n      const itemCategoryPromise = (form.category_id)\n        ? ItemCategory.query().findById(form.category_id) : null;\n\n      // Validate the custom fields key and value type.\n      if (form.custom_fields.length > 0) {\n        const customFieldsKeys = form.custom_fields.map((field) => field.key);\n\n        // Get resource id than get all resource fields.\n        const resource = await Resource.where('name', 'items').fetch();\n        const fields = await ResourceField.query((query) => {\n          query.where('resource_id', resource.id);\n          query.whereIn('key', customFieldsKeys);\n        }).fetchAll();\n\n        const storedFieldsKey = fields.map((f) => f.attributes.key);\n\n        // Get all not defined resource fields.\n        const notFoundFields = difference(customFieldsKeys, storedFieldsKey);\n\n        if (notFoundFields.length > 0) {\n          errorReasons.push({ type: 'FIELD_KEY_NOT_FOUND', code: 150, fields: notFoundFields });\n        }\n      }\n      const [costAccount, sellAccount, itemCategory] = await Promise.all([\n        costAccountPromise, sellAccountPromise, itemCategoryPromise,\n      ]);\n      if (!costAccount) {\n        errorReasons.push({ type: 'COST_ACCOUNT_NOT_FOUND', code: 100 });\n      }\n      if (!sellAccount) {\n        errorReasons.push({ type: 'SELL_ACCOUNT_NOT_FOUND', code: 120 });\n      }\n      if (!itemCategory && form.category_id) {\n        errorReasons.push({ type: 'ITEM_CATEGORY_NOT_FOUND', code: 140 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n      const item = await Item.query().insertAndFetch({\n        name: form.name,\n        type: form.type,\n        cost_price: form.cost_price,\n        sell_price: form.sell_price,\n        sell_account_id: form.sell_account_id,\n        cost_account_id: form.cost_account_id,\n        currency_code: form.currency_code,\n        note: form.note,\n      });\n      return res.status(200).send({ id: item.id });\n    },\n  },\n\n  /**\n   * Edit the given item.\n   */\n  editItem: {\n    validation: [\n      check('name').exists(),\n      check('type').exists().trim().escape().isIn(['product', 'service']),\n      check('cost_price').exists().isNumeric(),\n      check('sell_price').exists().isNumeric(),\n      check('cost_account_id').exists().isInt(),\n      check('sell_account_id').exists().isInt(),\n      check('category_id').optional().isInt(),\n      check('note').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { id } = req.params;\n      const form = {\n        custom_fields: [],\n        ...req.body,\n      };\n      const item = await Item.query().findById(id);\n      \n      if (!item) {\n        return res.boom.notFound(null, { errors: [\n          { type: 'ITEM.NOT.FOUND', code: 100 },\n        ]});\n      }\n      const errorReasons = [];\n\n      const costAccountPromise = Account.query().findById(form.cost_account_id);\n      const sellAccountPromise = Account.query().findById(form.sell_account_id);\n      const itemCategoryPromise = (form.category_id)\n        ? ItemCategory.query().findById(form.category_id) : null;\n\n      const [costAccount, sellAccount, itemCategory] = await Promise.all([\n        costAccountPromise, sellAccountPromise, itemCategoryPromise,\n      ]);\n      if (!costAccount) {\n        errorReasons.push({ type: 'COST_ACCOUNT_NOT_FOUND', code: 100 });\n      }\n      if (!sellAccount) {\n        errorReasons.push({ type: 'SELL_ACCOUNT_NOT_FOUND', code: 120 });\n      }\n      if (!itemCategory && form.category_id) {\n        errorReasons.push({ type: 'ITEM_CATEGORY_NOT_FOUND', code: 140 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      const updatedItem = await Item.query().findById(id).patch({\n        name: form.name,\n        type: form.type,\n        cost_price: form.cost_price,\n        sell_price: form.sell_price,\n        currency_code: form.currency_code,\n        sell_account_id: form.sell_account_id,\n        cost_account_id: form.cost_account_id,\n        category_id: form.category_id,\n        note: form.note,\n      });\n      return res.status(200).send({ id: updatedItem.id });\n    },\n  },\n\n  /**\n   * Delete the given item from the storage.\n   */\n  deleteItem: {\n    validation: [],\n    async handler(req, res) {\n      const { id } = req.params;\n      const item = await Item.query().findById(id);\n\n      if (!item) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'ITEM_NOT_FOUND', code: 100 }],\n        });\n      }\n\n      // Delete the fucking the given item id.\n      await Item.query().findById(item.id).delete();\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrive the list items with pagination meta.\n   */\n  listItems: {\n    validation: [],\n    async handler(req, res) {\n      const filter = {\n        name: '',\n        description: '',\n        SKU: '',\n        account_id: null,\n        page_size: 10,\n        page: 1,\n        start_date: null,\n        end_date: null,\n        ...req.query,\n      };\n\n      const items = await Item.query((query) => {\n        if (filter.description) {\n          query.where('description', 'like', `%${filter.description}%`);\n        }\n        if (filter.description) {\n          query.where('SKU', filter.SKY);\n        }\n        if (filter.name) {\n          query.where('name', filter.name);\n        }\n        if (filter.start_date) {\n          const startDateFormatted = moment(filter.start_date).format('YYYY-MM-DD HH:mm:SS');\n          query.where('created_at', '>=', startDateFormatted);\n        }\n        if (filter.end_date) {\n          const endDateFormatted = moment(filter.end_date).format('YYYY-MM-DD HH:mm:SS');\n          query.where('created_at', '<=', endDateFormatted);\n        }\n      }).fetchPage({\n        page_size: filter.page_size,\n        page: filter.page,\n      });\n\n      return res.status(200).send({\n        items: items.toJSON(),\n        pagination: items.pagination,\n      });\n    },\n  },\n};\n","import express from 'express';\nimport { body, query, validationResult } from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport Option from '@/models/Option';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.post('/',\n      this.saveOptions.validation,\n      asyncMiddleware(this.saveOptions.handler));\n\n    router.get('/',\n      this.getOptions.validation,\n      asyncMiddleware(this.getSettings));\n\n    return router;\n  },\n\n  /**\n   * Saves the given options to the storage.\n   */\n  saveOptions: {\n    validation: [\n      body('options').isArray(),\n      body('options.*.key').exists(),\n      body('options.*.value').exists(),\n      body('options.*.group').exists(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'VALIDATION_ERROR', ...validationErrors,\n        });\n      }\n      const form = { ...req.body };\n      const optionsCollections = await Option.query();\n\n      form.options.forEach((option) => {\n        optionsCollections.setMeta(option.key, option.value, option.group);\n      });\n      await optionsCollections.saveMeta();\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrieve the application options from the storage.\n   */\n  getOptions: {\n    validation: [\n      query('key').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'VALIDATION_ERROR', ...validationErrors,\n        });\n      }\n      const options = await Option.query();\n\n      return res.status(200).sends({ options });\n    },\n  },\n};\n","import express from 'express';\nimport {\n  param,\n  query,\n  validationResult,\n} from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport Resource from '@/models/Resource';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(jwtAuth);\n\n    router.get('/:resource_slug/columns',\n      this.resourceColumns.validation,\n      asyncMiddleware(this.resourceColumns.handler));\n\n    router.get('/:resource_slug/fields',\n      this.resourceFields.validation,\n      asyncMiddleware(this.resourceFields.handler));\n\n    return router;\n  },\n\n  /**\n   * Retrieve resource columns of the given resource.\n   */\n  resourceColumns: {\n    validation: [\n      param('resource_slug').trim().escape().exists(),\n    ],\n    async handler(req, res) {\n      const { resource_slug: resourceSlug } = req.params;\n\n      const resource = await Resource.query()\n        .where('name', resourceSlug)\n        .withGraphFetched('fields')\n        .first();\n\n      if (!resource) {\n        return res.status(400).send({\n          errors: [{ type: 'RESOURCE.SLUG.NOT.FOUND', code: 200 }],\n        });\n      }\n      const resourceFields = resource.fields\n        .filter((field) => field.columnable)\n        .map((field) => ({\n          id: field.id,\n          label: field.labelName,\n          key: field.key,\n        }));\n\n      return res.status(200).send({\n        resource_columns: resourceFields,\n        resource_slug: resourceSlug,\n      });\n    },\n  },\n\n  /**\n   * Retrieve resource fields of the given resource.\n   */\n  resourceFields: {\n    validation: [\n      param('resource_slug').trim().escape().exists(),\n      query('predefined').optional().isBoolean().toBoolean(),\n      query('builtin').optional().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const { resource_slug: resourceSlug } = req.params;\n\n      const resource = await Resource.query()\n        .where('name', resourceSlug)\n        .withGraphFetched('fields')\n        .first();\n\n      if (!resource) {\n        return res.status(400).send({\n          errors: [{ type: 'RESOURCE.SLUG.NOT.FOUND', code: 200 }],\n        });\n      }\n      return res.status(200).send({\n        resource_fields: resource.fields,\n        resource_slug: resourceSlug,\n      });\n    },\n  },\n};\n","/* eslint-disable no-unused-vars */\nimport express from 'express';\nimport { check, validationResult } from 'express-validator';\nimport { difference } from 'lodash';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport Role from '@/models/Role';\nimport Permission from '@/models/Permission';\nimport Resource from '@/models/Resource';\nimport knex from '@/database/knex';\n\nconst AccessControllSchema = [\n  {\n    resource: 'items',\n    label: 'products_services',\n    permissions: ['create', 'edit', 'delete', 'view'],\n    fullAccess: true,\n    ownControl: true,\n  },\n];\n\nconst getResourceSchema = (resource) => AccessControllSchema\n  .find((schema) => schema.resource === resource);\n\nconst getResourcePermissions = (resource) => {\n  const foundResource = getResourceSchema(resource);\n  return foundResource ? foundResource.permissions : [];\n};\n\nconst findNotFoundResources = (resourcesSlugs) => {\n  const schemaResourcesSlugs = AccessControllSchema.map((s) => s.resource);\n  return difference(resourcesSlugs, schemaResourcesSlugs);\n};\n\nconst findNotFoundPermissions = (permissions, resourceSlug) => {\n  const schemaPermissions = getResourcePermissions(resourceSlug);\n  return difference(permissions, schemaPermissions);\n};\n\nexport default {\n  /**\n   * Router constructor method.\n   */\n  router() {\n    const router = express.Router();\n\n    router.post('/',\n      this.newRole.validation,\n      asyncMiddleware(this.newRole.handler));\n\n    router.post('/:id',\n      this.editRole.validation,\n      asyncMiddleware(this.editRole.handler.bind(this)));\n\n    router.delete('/:id',\n      this.deleteRole.validation,\n      asyncMiddleware(this.deleteRole.handler));\n\n    return router;\n  },\n\n  /**\n   * Creates a new role.\n   */\n  newRole: {\n    validation: [\n      check('name').exists().trim().escape(),\n      check('description').optional().trim().escape(),\n      check('permissions').isArray({ min: 0 }),\n      check('permissions.*.resource_slug').exists().whitelist('^[a-z0-9]+(?:-[a-z0-9]+)*$'),\n      check('permissions.*.permissions').isArray({ min: 1 }),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { name, description, permissions } = req.body;\n\n      const resourcesSlugs = permissions.map((perm) => perm.resource_slug);\n      const permissionsSlugs = [];\n      const resourcesNotFound = findNotFoundResources(resourcesSlugs);\n\n      const errorReasons = [];\n      const notFoundPermissions = [];\n\n      if (resourcesNotFound.length > 0) {\n        errorReasons.push({\n          type: 'RESOURCE_SLUG_NOT_FOUND', code: 100, resources: resourcesNotFound,\n        });\n      }\n      permissions.forEach((perm) => {\n        const abilities = perm.permissions.map((ability) => ability);\n\n        // Gets the not found permissions in the schema.\n        const notFoundAbilities = findNotFoundPermissions(abilities, perm.resource_slug);\n        \n        if (notFoundAbilities.length > 0) {\n          notFoundPermissions.push({\n            resource_slug: perm.resource_slug,\n            permissions: notFoundAbilities,\n          });\n        } else {\n          const perms = perm.permissions || [];\n          perms.forEach((permission) => {\n            if (perms.indexOf(permission) !== -1) {\n              permissionsSlugs.push(permission);\n            }\n          });\n        }\n      });\n      if (notFoundPermissions.length > 0) {\n        errorReasons.push({\n          type: 'PERMISSIONS_SLUG_NOT_FOUND',\n          code: 200,\n          permissions: notFoundPermissions,\n        });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n      // Permissions.\n      const [resourcesCollection, permsCollection] = await Promise.all([\n        Resource.query((query) => { query.whereIn('name', resourcesSlugs); }).fetchAll(),\n        Permission.query((query) => { query.whereIn('name', permissionsSlugs); }).fetchAll(),\n      ]);\n\n      const notStoredResources = difference(\n        resourcesSlugs, resourcesCollection.map((s) => s.name),\n      );\n      const notStoredPermissions = difference(\n        permissionsSlugs, permsCollection.map((perm) => perm.slug),\n      );\n\n      const insertThread = [];\n\n      if (notStoredResources.length > 0) {\n        insertThread.push(knex('resources').insert([\n          ...notStoredResources.map((resource) => ({ name: resource })),\n        ]));\n      }\n      if (notStoredPermissions.length > 0) {\n        insertThread.push(knex('permissions').insert([\n          ...notStoredPermissions.map((permission) => ({ name: permission })),\n        ]));\n      }\n\n      await Promise.all(insertThread);\n\n      const [storedPermissions, storedResources] = await Promise.all([\n        Permission.query((q) => { q.whereIn('name', permissionsSlugs); }).fetchAll(),\n        Resource.query((q) => { q.whereIn('name', resourcesSlugs); }).fetchAll(),\n      ]);\n\n      const storedResourcesSet = new Map(storedResources.map((resource) => [\n        resource.attributes.name, resource.attributes.id,\n      ]));\n      const storedPermissionsSet = new Map(storedPermissions.map((perm) => [\n        perm.attributes.name, perm.attributes.id,\n      ]));\n      const role = Role.forge({ name, description });\n\n      await role.save();\n\n      const roleHasPerms = permissions.map((resource) => resource.permissions.map((perm) => ({\n        role_id: role.id,\n        resource_id: storedResourcesSet.get(resource.resource_slug),\n        permission_id: storedPermissionsSet.get(perm),\n      })));\n\n      if (roleHasPerms.length > 0) {\n        await knex('role_has_permissions').insert(roleHasPerms[0]);\n      }\n      return res.status(200).send({ id: role.get('id') });\n    },\n  },\n\n  /**\n   * Edit the give role.\n   */\n  editRole: {\n    validation: [\n      check('name').exists().trim().escape(),\n      check('description').optional().trim().escape(),\n      check('permissions').isArray({ min: 0 }),\n      check('permissions.*.resource_slug').exists().whitelist('^[a-z0-9]+(?:-[a-z0-9]+)*$'),\n      check('permissions.*.permissions').isArray({ min: 1 }),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { id } = req.params;\n      const role = await Role.where('id', id).fetch();\n\n      if (!role) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'ROLE_NOT_FOUND', code: 100 }],\n        });\n      }\n\n      const { permissions } = req.body;\n      const errorReasons = [];\n      const permissionsSlugs = [];\n      const notFoundPermissions = [];\n\n      const resourcesSlugs = permissions.map((perm) => perm.resource_slug);\n      const resourcesNotFound = findNotFoundResources(resourcesSlugs);\n\n      if (resourcesNotFound.length > 0) {\n        errorReasons.push({\n          type: 'RESOURCE_SLUG_NOT_FOUND',\n          code: 100,\n          resources: resourcesNotFound,\n        });\n      }\n\n      permissions.forEach((perm) => {\n        const abilities = perm.permissions.map((ability) => ability);\n        // Gets the not found permissions in the schema.\n        const notFoundAbilities = findNotFoundPermissions(abilities, perm.resource_slug);\n\n        if (notFoundAbilities.length > 0) {\n          notFoundPermissions.push({\n            resource_slug: perm.resource_slug, permissions: notFoundAbilities,\n          });\n        } else {\n          const perms = perm.permissions || [];\n          perms.forEach((permission) => {\n            if (perms.indexOf(permission) !== -1) {\n              permissionsSlugs.push(permission);\n            }\n          });\n        }\n      });\n\n      if (notFoundPermissions.length > 0) {\n        errorReasons.push({\n          type: 'PERMISSIONS_SLUG_NOT_FOUND',\n          code: 200,\n          permissions: notFoundPermissions,\n        });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      // Permissions.\n      const [resourcesCollection, permsCollection] = await Promise.all([\n        Resource.query((query) => { query.whereIn('name', resourcesSlugs); }).fetchAll(),\n        Permission.query((query) => { query.whereIn('name', permissionsSlugs); }).fetchAll(),\n      ]);\n\n      const notStoredResources = difference(\n        resourcesSlugs, resourcesCollection.map((s) => s.name),\n      );\n      const notStoredPermissions = difference(\n        permissionsSlugs, permsCollection.map((perm) => perm.slug),\n      );\n      const insertThread = [];\n\n      if (notStoredResources.length > 0) {\n        insertThread.push(knex('resources').insert([\n          ...notStoredResources.map((resource) => ({ name: resource })),\n        ]));\n      }\n      if (notStoredPermissions.length > 0) {\n        insertThread.push(knex('permissions').insert([\n          ...notStoredPermissions.map((permission) => ({ name: permission })),\n        ]));\n      }\n\n      await Promise.all(insertThread);\n\n      const [storedPermissions, storedResources] = await Promise.all([\n        Permission.query((q) => { q.whereIn('name', permissionsSlugs); }).fetchAll(),\n        Resource.query((q) => { q.whereIn('name', resourcesSlugs); }).fetchAll(),\n      ]);\n\n      const storedResourcesSet = new Map(storedResources.map((resource) => [\n        resource.attributes.name, resource.attributes.id,\n      ]));\n      const storedPermissionsSet = new Map(storedPermissions.map((perm) => [\n        perm.attributes.name, perm.attributes.id,\n      ]));\n\n      await role.save();\n\n\n      const savedRoleHasPerms = await knex('role_has_permissions').where({\n        role_id: role.id,\n      });\n\n      console.log(savedRoleHasPerms);\n\n      // const roleHasPerms = permissions.map((resource) => resource.permissions.map((perm) => ({\n      //   role_id: role.id,\n      //   resource_id: storedResourcesSet.get(resource.resource_slug),\n      //   permission_id: storedPermissionsSet.get(perm),\n      // })));\n\n      // if (roleHasPerms.length > 0) {\n      //   await knex('role_has_permissions').insert(roleHasPerms[0]);\n      // }\n      return res.status(200).send({ id: role.get('id') });\n    },\n  },\n\n  deleteRole: {\n    validation: [],\n    async handler(req, res) {\n      const { id } = req.params;\n      const role = await Role.where('id', id).fetch();\n\n      if (!role) {\n        return res.boom.notFound();\n      }\n      if (role.attributes.predefined) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'ROLE_PREDEFINED', code: 100 }],\n        });\n      }\n\n      await knex('role_has_permissions')\n        .where('role_id', role.id).delete({ require: false });\n\n      await role.destroy();\n\n      return res.status(200).send();\n    },\n  },\n\n  getRole: {\n    validation: [],\n    handler(req, res) {\n      return res.status(200).send();\n    },\n  },\n};\n","import express from 'express';\n\nexport default {\n\n  router() {\n    const router = express.Router();\n\n    return router;\n  },\n};\n","import express from 'express';\nimport {\n  check,\n  query,\n  param,\n  validationResult,\n} from 'express-validator';\nimport User from '@/models/User';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport Authorization from '@/http/middleware/authorization';\n\nexport default {\n\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n    // const permit = Authorization('users');\n\n    router.use(jwtAuth);\n\n    router.post('/',\n      // permit('create'),\n      this.newUser.validation,\n      asyncMiddleware(this.newUser.handler));\n\n    router.post('/:id',\n      // permit('create', 'edit'),\n      this.editUser.validation,\n      asyncMiddleware(this.editUser.handler));\n\n    router.get('/',\n      // permit('view'),\n      this.listUsers.validation,\n      asyncMiddleware(this.listUsers.handler));\n\n    router.get('/:id',\n      // permit('view'),\n      this.getUser.validation,\n      asyncMiddleware(this.getUser.handler));\n\n    router.delete('/:id',\n      // permit('create', 'edit', 'delete'),\n      this.deleteUser.validation,\n      asyncMiddleware(this.deleteUser.handler));\n\n    return router;\n  },\n\n  /**\n   * Creates a new user.\n   */\n  newUser: {\n    validation: [\n      check('first_name').trim().escape().exists(),\n      check('last_name').trim().escape().exists(),\n      check('email').exists().isEmail(),\n      check('phone_number').optional().isMobilePhone(),\n      check('password').isLength({ min: 4 }).exists().custom((value, { req }) => {\n        if (value !== req.body.confirm_password) {\n          throw new Error(\"Passwords don't match\");\n        } else {\n          return value;\n        }\n      }),\n      check('status').exists().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { email, phone_number: phoneNumber } = req.body;\n\n      const foundUsers = await User.query()\n        .where('email', email)\n        .orWhere('phone_number', phoneNumber);\n\n      const foundUserEmail = foundUsers.find((u) => u.email === email);\n      const foundUserPhone = foundUsers.find((u) => u.phoneNumber === phoneNumber);\n\n      const errorReasons = [];\n\n      if (foundUserEmail) {\n        errorReasons.push({ type: 'EMAIL_ALREADY_EXIST', code: 100 });\n      }\n      if (foundUserPhone) {\n        errorReasons.push({ type: 'PHONE_NUMBER_ALREADY_EXIST', code: 120 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      const user = await User.query().insert({\n        first_name: req.body.first_name,\n        last_name: req.body.last_name,\n        email: req.body.email,\n        phone_number: req.body.phone_number,\n        active: req.body.status,\n      });\n\n      return res.status(200).send({ user });\n    },\n  },\n\n  /**\n   * Edit details of the given user.\n   */\n  editUser: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n      check('first_name').exists(),\n      check('last_name').exists(),\n      check('email').exists().isEmail(),\n      check('phone_number').optional().isMobilePhone(),\n      check('password').isLength({ min: 4 }).exists().custom((value, { req }) => {\n        if (value !== req.body.confirm_password) {\n          throw new Error(\"Passwords don't match\");\n        } else {\n          return value;\n        }\n      }),\n      check('status').exists().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const user = await User.query().where('id', id).first();\n\n      if (!user) {\n        return res.boom.notFound();\n      }\n      const { email, phone_number: phoneNumber } = req.body;\n\n      const foundUsers = await User.query()\n        .whereNot('id', id)\n        .andWhere((q) => {\n          q.where('email', email);\n          q.orWhere('phone_number', phoneNumber);\n        });\n\n      const foundUserEmail = foundUsers.find((u) => u.email === email);\n      const foundUserPhone = foundUsers.find((u) => u.phoneNumber === phoneNumber);\n\n      const errorReasons = [];\n\n      if (foundUserEmail) {\n        errorReasons.push({ type: 'EMAIL_ALREADY_EXIST', code: 100 });\n      }\n      if (foundUserPhone) {\n        errorReasons.push({ type: 'PHONE_NUMBER_ALREADY_EXIST', code: 120 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      await User.query().where('id', id).update({\n        first_name: req.body.first_name,\n        last_name: req.body.last_name,\n        email: req.body.email,\n        phone_number: req.body.phone_number,\n        active: req.body.status,\n      });\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Soft deleting the given user.\n   */\n  deleteUser: {\n    validation: [],\n    async handler(req, res) {\n      const { id } = req.params;\n      const user = await User.query().where('id', id).first();\n\n      if (!user) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'USER_NOT_FOUND', code: 100 }],\n        });\n      }\n      await User.query().where('id', id).delete();\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrieve user details of the given user id.\n   */\n  getUser: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const user = await User.query().where('id', id).first();\n\n      if (!user) {\n        return res.boom.notFound();\n      }\n      return res.status(200).send({ user });\n    },\n  },\n\n  /**\n   * Retrieve the list of users.\n   */\n  listUsers: {\n    validation: [\n      query('page_size').optional().isNumeric().toInt(),\n      query('page').optional().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const filter = {\n        first_name: '',\n        last_name: '',\n        email: '',\n        phone_number: '',\n\n        page_size: 10,\n        page: 1,\n        ...req.query,\n      };\n\n      const users = await User.query()\n        .page(filter.page - 1, filter.page_size);\n\n      return res.status(200).send({ users });\n    },\n  },\n};\n","import { difference, pick } from 'lodash';\nimport express from 'express';\nimport {\n  check,\n  query,\n  param,\n  oneOf,\n  validationResult,\n} from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport Resource from '@/models/Resource';\nimport View from '@/models/View';\nimport ViewRole from '@/models/ViewRole';\nimport ViewColumn from '@/models/ViewColumn';\nimport {\n  validateViewLogicExpression,\n} from '@/lib/ViewRolesBuilder';\n\nexport default {\n  resource: 'items',\n\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(jwtAuth);\n\n    router.get('/',\n      this.listViews.validation,\n      asyncMiddleware(this.listViews.handler));\n\n    router.post('/',\n      this.createView.validation,\n      asyncMiddleware(this.createView.handler));\n\n    router.post('/:view_id',\n      this.editView.validation,\n      asyncMiddleware(this.editView.handler));\n\n    router.delete('/:view_id',\n      this.deleteView.validation,\n      asyncMiddleware(this.deleteView.handler));\n\n    router.get('/:view_id',\n      asyncMiddleware(this.getView.handler));\n\n    return router;\n  },\n\n  /**\n   * List all views that associated with the given resource.\n   */\n  listViews: {\n    validation: [\n      oneOf([\n        query('resource_name').exists().trim().escape(),\n      ], [\n        query('resource_id').exists().isNumeric().toInt(),\n      ]),\n    ],\n    async handler(req, res) {\n      const filter = { ...req.query };\n\n      const resource = await Resource.query().onBuild((builder) => {\n        if (filter.resource_id) {\n          builder.where('id', filter.resource_id);\n        }\n        if (filter.resource_name) {\n          builder.where('name', filter.resource_name);\n        }\n        builder.first();\n      });\n\n      const views = await View.query().where('resource_id', resource.id);\n\n      return res.status(200).send({ views });\n    },\n  },\n\n  /**\n   * Retrieve view details of the given view id.\n   */\n  getView: {\n    validation: [\n      param('view_id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { view_id: viewId } = req.params;\n      const view = await View.query()\n        .where('id', viewId)\n        .withGraphFetched('resource')\n        .withGraphFetched('columns')\n        .withGraphFetched('roles.field')\n        .first();\n\n      if (!view) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'VIEW_NOT_FOUND', code: 100 }],\n        });\n      }\n      return res.status(200).send({ view: view.toJSON() });\n    },\n  },\n\n  /**\n   * Delete the given view of the resource.\n   */\n  deleteView: {\n    validation: [\n      param('view_id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { view_id: viewId } = req.params;\n      const view = await View.query().findById(viewId);\n\n      if (!view) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'VIEW_NOT_FOUND', code: 100 }],\n        });\n      }\n      if (view.predefined) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'PREDEFINED_VIEW', code: 200 }],\n        });\n      }\n      await Promise.all([\n        view.$relatedQuery('roles').delete(),\n        view.$relatedQuery('columns').delete(),\n      ]);\n      await View.query().where('id', view.id).delete();\n\n      return res.status(200).send({ id: view.id });\n    },\n  },\n\n  /**\n   * Creates a new view.\n   */\n  createView: {\n    validation: [\n      check('resource_name').exists().escape().trim(),\n      check('name').exists().escape().trim(),\n      check('logic_expression').exists().trim().escape(),\n      check('roles').isArray({ min: 1 }),\n      check('roles.*.field_key').exists().escape().trim(),\n      check('roles.*.comparator').exists(),\n      check('roles.*.value').exists(),\n      check('roles.*.index').exists().isNumeric().toInt(),\n      check('columns').exists().isArray({ min: 1 }),\n      check('columns.*.key').exists().escape().trim(),\n      check('columns.*.index').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = { ...req.body };\n      const resource = await Resource.query().where('name', form.resource_name).first();\n\n      if (!resource) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'RESOURCE_NOT_FOUND', code: 100 }],\n        });\n      }\n      const errorReasons = [];\n      const fieldsSlugs = form.roles.map((role) => role.field_key);\n\n      const resourceFields = await resource.$relatedQuery('fields');\n      const resourceFieldsKeys = resourceFields.map((f) => f.key);\n      const resourceFieldsKeysMap = new Map(resourceFields.map((field) => [field.key, field]));\n      const columnsKeys = form.columns.map((c) => c.key);\n\n      // The difference between the stored resource fields and submit fields keys.\n      const notFoundFields = difference(fieldsSlugs, resourceFieldsKeys);\n\n      if (notFoundFields.length > 0) {\n        errorReasons.push({ type: 'RESOURCE_FIELDS_NOT_EXIST', code: 100, fields: notFoundFields });\n      }\n      // The difference between the stored resource fields and the submit columns keys.\n      const notFoundColumns = difference(columnsKeys, resourceFieldsKeys);\n\n      if (notFoundColumns.length > 0) {\n        errorReasons.push({ type: 'COLUMNS_NOT_EXIST', code: 200, columns: notFoundColumns });\n      }\n      // Validates the view conditional logic expression.\n      if (!validateViewLogicExpression(form.logic_expression, form.roles.map((r) => r.index))) {\n        errorReasons.push({ type: 'VIEW.ROLES.LOGIC.EXPRESSION.INVALID', code: 400 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      // Save view details.\n      const view = await View.query().insert({\n        name: form.name,\n        predefined: false,\n        resource_id: resource.id,\n        roles_logic_expression: form.logic_expression,\n      });\n      // Save view roles async operations.\n      const saveViewRolesOpers = [];\n\n      form.roles.forEach((role) => {\n        const fieldModel = resourceFieldsKeysMap.get(role.field_key);\n        \n        const saveViewRoleOper = ViewRole.query().insert({\n          ...pick(role, ['comparator', 'value', 'index']),\n          field_id: fieldModel.id,\n          view_id: view.id,\n        });\n        saveViewRolesOpers.push(saveViewRoleOper);\n      });\n\n      form.columns.forEach((column) => {\n        const fieldModel = resourceFieldsKeysMap.get(column.key);\n\n        const saveViewColumnOper = ViewColumn.query().insert({\n          field_id: fieldModel.id,\n          view_id: view.id,\n          index: column.index,\n        });\n        saveViewRolesOpers.push(saveViewColumnOper);\n      });\n      await Promise.all(saveViewRolesOpers);\n\n      return res.status(200).send({ id: view.id });\n    },\n  },\n\n  editView: {\n    validation: [\n      param('view_id').exists().isNumeric().toInt(),\n      check('label').exists().escape().trim(),\n      check('columns').isArray({ min: 3 }),\n      check('roles').isArray(),\n      check('roles.*.field').exists().escape().trim(),\n      check('roles.*.comparator').exists(),\n      check('roles.*.value').exists(),\n      check('roles.*.index').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { view_id: viewId } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const view = await View.where('id', viewId).fetch();\n\n      if (!view) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'ROLE_NOT_FOUND', code: 100 }],\n        });\n      }\n      return res.status(200).send();\n    },\n  },\n};\n","// import OAuth2 from '@/http/controllers/OAuth2';\nimport Authentication from '@/http/controllers/Authentication';\nimport Users from '@/http/controllers/Users';\nimport Roles from '@/http/controllers/Roles';\nimport Items from '@/http/controllers/Items';\nimport ItemCategories from '@/http/controllers/ItemCategories';\nimport Accounts from '@/http/controllers/Accounts';\nimport AccountTypes from '@/http/controllers/AccountTypes';\nimport AccountOpeningBalance from '@/http/controllers/AccountOpeningBalance';\nimport Views from '@/http/controllers/Views';\nimport CustomFields from '@/http/controllers/Fields';\nimport Accounting from '@/http/controllers/Accounting';\nimport FinancialStatements from '@/http/controllers/FinancialStatements';\nimport Expenses from '@/http/controllers/Expenses';\nimport Options from '@/http/controllers/Options';\nimport Budget from '@/http/controllers/Budget';\nimport BudgetReports from '@/http/controllers/BudgetReports';\nimport Currencies from '@/http/controllers/Currencies';\nimport Customers from '@/http/controllers/Customers';\nimport Suppliers from '@/http/controllers/Suppliers';\nimport Bills from '@/http/controllers/Bills';\nimport CurrencyAdjustment from './controllers/CurrencyAdjustment';\nimport Resources from './controllers/Resources';\n// import SalesReports from '@/http/controllers/SalesReports';\n// import PurchasesReports from '@/http/controllers/PurchasesReports';\n\nexport default (app) => {\n  // app.use('/api/oauth2', OAuth2.router());\n  app.use('/api/auth', Authentication.router());\n  app.use('/api/currencies', Currencies.router());\n  app.use('/api/users', Users.router());\n  app.use('/api/roles', Roles.router());\n  app.use('/api/accounts', Accounts.router());\n  app.use('/api/account_types', AccountTypes.router());\n  app.use('/api/accounting', Accounting.router());\n  app.use('/api/accounts_opening_balances', AccountOpeningBalance.router());\n  app.use('/api/views', Views.router());\n  app.use('/api/fields', CustomFields.router());\n  app.use('/api/items', Items.router());\n  app.use('/api/item_categories', ItemCategories.router());\n  app.use('/api/expenses', Expenses.router());\n  app.use('/api/financial_statements', FinancialStatements.router());\n  app.use('/api/options', Options.router());\n  app.use('/api/budget_reports', BudgetReports.router());\n  // app.use('/api/customers', Customers.router());\n  // app.use('/api/suppliers', Suppliers.router());\n  // app.use('/api/bills', Bills.router());\n  app.use('/api/budget', Budget.router());\n  app.use('/api/resources', Resources.router());\n  // app.use('/api/currency_adjustment', CurrencyAdjustment.router());\n  // app.use('/api/reports/sales', SalesReports.router());\n  // app.use('/api/reports/purchases', PurchasesReports.router());\n};\n","const asyncMiddleware = (fn) => (req, res, next) => {\n  Promise.resolve(fn(req, res, next))\n    .catch((error) => {\n      console.log(error);\n      next(error);\n    });\n};\n\nexport default asyncMiddleware;\n","/* eslint-disable consistent-return */\nconst authorization = (resourceName) => (...permissions) => (req, res, next) => {\n  const { user } = req;\n  const onError = () => {\n    res.boom.unauthorized();\n  };\n  user.hasPermissions(resourceName, permissions)\n    .then((authorized) => {\n      if (!authorized) {\n        return onError();\n      }\n      next();\n    }).catch(onError);\n};\n\nexport default authorization;\n","/* eslint-disable consistent-return */\nimport jwt from 'jsonwebtoken';\nimport User from '@/models/User';\n// import Auth from '@/models/Auth';\n\nconst authMiddleware = (req, res, next) => {\n  const { JWT_SECRET_KEY } = process.env;\n  const token = req.headers['x-access-token'] || req.query.token;\n\n  const onError = () => {\n    // Auth.loggedOut();\n    res.status(401).send({\n      success: false,\n      message: 'unauthorized',\n    });\n  };\n\n  if (!token) {\n    return onError();\n  }\n\n  const verify = new Promise((resolve, reject) => {\n    jwt.verify(token, JWT_SECRET_KEY, async (error, decoded) => {\n      if (error) {\n        reject(error);\n      } else {\n        // eslint-disable-next-line no-underscore-dangle\n        req.user = await User.query().findById(decoded._id);\n        // Auth.setAuthenticatedUser(req.user);\n\n        if (!req.user) {\n          return onError();\n        }\n        resolve(decoded);\n      }\n    });\n  });\n\n  verify.then(() => { next(); }).catch(onError);\n};\nexport default authMiddleware;\n","\nconst OperationType = {\n  LOGIC: 'LOGIC',\n  STRING: 'STRING',\n  COMPARISON: 'COMPARISON',\n  MATH: 'MATH',\n};\n\nexport class Lexer {\n  // operation table\n  static get optable() {\n    return {\n      '=': OperationType.LOGIC,\n      '&': OperationType.LOGIC,\n      '|': OperationType.LOGIC,\n      '?': OperationType.LOGIC,\n      ':': OperationType.LOGIC,\n\n      '\\'': OperationType.STRING,\n      '\"': OperationType.STRING,\n\n      '!': OperationType.COMPARISON,\n      '>': OperationType.COMPARISON,\n      '<': OperationType.COMPARISON,\n\n      '(': OperationType.MATH,\n      ')': OperationType.MATH,\n      '+': OperationType.MATH,\n      '-': OperationType.MATH,\n      '*': OperationType.MATH,\n      '/': OperationType.MATH,\n      '%': OperationType.MATH,\n    };\n  }\n\n  /**\n   * Constructor\n   * @param {*} expression -\n   */\n  constructor(expression) {\n    this.currentIndex = 0;\n    this.input = expression;\n    this.tokenList = [];\n  }\n\n  getTokens() {\n    let tok;\n    do {\n      // read current token, so step should be -1\n      tok = this.pickNext(-1);\n      const pos = this.currentIndex;\n      switch (Lexer.optable[tok]) {\n        case OperationType.LOGIC:\n          // == && || ===\n          this.readLogicOpt(tok);\n          break;\n\n        case OperationType.STRING:\n          this.readString(tok);\n          break;\n\n        case OperationType.COMPARISON:\n          this.readCompare(tok);\n          break;\n\n        case OperationType.MATH:\n          this.receiveToken();\n          break;\n\n        default:\n          this.readValue(tok);\n      }\n\n      // if the pos not changed, this loop will go into a infinite loop, every step of while loop,\n      // we must move the pos forward\n      // so here we should throw error, for example `1 & 2`\n      if (pos === this.currentIndex && tok !== undefined) {\n        const err = new Error(`unkonw token ${tok} from input string ${this.input}`);\n        err.name = 'UnknowToken';\n        throw err;\n      }\n    } while (tok !== undefined)\n\n    return this.tokenList;\n  }\n\n  /**\n   * read next token, the index param can set next step, default go foward 1 step\n   *\n   * @param index next postion\n   */\n  pickNext(index = 0) {\n    return this.input[index + this.currentIndex + 1];\n  }\n\n  /**\n   * Store token into result tokenList, and move the pos index\n   *\n   * @param index\n   */\n  receiveToken(index = 1) {\n    const tok = this.input.slice(this.currentIndex, this.currentIndex + index).trim();\n    // skip empty string\n    if (tok) {\n      this.tokenList.push(tok);\n    }\n\n    this.currentIndex += index;\n  }\n\n  // ' or \"\n  readString(tok) {\n    let next;\n    let index = 0;\n    do {\n      next = this.pickNext(index);\n      index += 1;\n    } while (next !== tok && next !== undefined);\n    this.receiveToken(index + 1);\n  }\n\n  // > or < or >= or <= or !==\n  // tok in (>, <, !)\n  readCompare(tok) {\n    if (this.pickNext() !== '=') {\n      this.receiveToken(1);\n      return;\n    }\n    // !==\n    if (tok === '!' && this.pickNext(1) === '=') {\n      this.receiveToken(3);\n      return;\n    }\n    this.receiveToken(2);\n  }\n\n  // === or ==\n  // && ||\n  readLogicOpt(tok) {\n    if (this.pickNext() === tok) {\n      // ===\n      if (tok === '=' && this.pickNext(1) === tok) {\n        return this.receiveToken(3);\n      }\n      // == && ||\n      return this.receiveToken(2);\n    }\n    // handle as &&\n    // a ? b : c is equal to a && b || c\n    if (tok === '?' || tok === ':') {\n      return this.receiveToken(1);\n    }\n  }\n\n  readValue(tok) {\n    if (!tok) {\n      return;\n    }\n\n    let index = 0;\n    while (!Lexer.optable[tok] && tok !== undefined) {\n      tok = this.pickNext(index);\n      index += 1;\n    }\n    this.receiveToken(index);\n  }\n}\n\nexport default function token(expression) {\n  const lexer = new Lexer(expression);\n  return lexer.getTokens();\n}\n","export const OPERATION = {\n  '!': 5,\n  '*': 4,\n  '/': 4,\n  '%': 4,\n  '+': 3,\n  '-': 3,\n  '>': 2,\n  '<': 2,\n  '>=': 2,\n  '<=': 2,\n  '===': 2,\n  '!==': 2,\n  '==': 2,\n  '!=': 2,\n  '&&': 1,\n  '||': 1,\n  '?': 1,\n  ':': 1,\n};\n\n// export interface Node {\n//   left: Node | string | null;\n//   right: Node | string | null;\n//   operation: string;\n//   grouped?: boolean;\n// };\n\nexport default class Parser {\n\n  constructor(token) {\n    this.index = -1;\n    this.blockLevel = 0;\n    this.token = token;\n  }\n\n  /**\n   * \n   * @return {Node | string} =- \n   */\n  parse() {\n    let tok;\n    let root = {\n      left: null,\n      right: null,\n      operation: null,\n    };\n\n    do {\n      tok = this.parseStatement();\n\n      if (tok === null || tok === undefined) {\n        break;\n      }\n\n      if (root.left === null) {\n        root.left = tok;\n        root.operation = this.nextToken();\n\n        if (!root.operation) {\n          return tok;\n        }\n\n        root.right = this.parseStatement();\n      } else {\n        if (typeof tok !== 'string') {\n          throw new Error('operation must be string, but get ' + JSON.stringify(tok));\n        }\n        root = this.addNode(tok, this.parseStatement(), root);\n      }\n    } while (tok);\n\n    return root;\n  }\n\n  nextToken() {\n    this.index += 1;\n    return this.token[this.index];\n  }\n\n  prevToken() {\n    return this.token[this.index - 1];\n  }\n\n  /**\n   * \n   * @param {string} operation \n   * @param {Node|String|null} right \n   * @param {Node} root \n   */\n  addNode(operation, right, root) {\n    let pre = root;\n    \n    if (this.compare(pre.operation, operation) < 0 && !pre.grouped) {\n      \n      while (pre.right !== null &&\n        typeof pre.right !== 'string' &&\n        this.compare(pre.right.operation, operation) < 0 && !pre.right.grouped) {\n        pre = pre.right;\n      }\n\n      pre.right = {\n        operation,\n        left: pre.right,\n        right,\n      };\n      return root;\n    }\n    return {\n      left: pre,\n      right,\n      operation,\n    }\n  }\n\n  /**\n   * \n   * @param {String} a \n   * @param {String} b \n   */\n  compare(a, b) {\n    if (!OPERATION.hasOwnProperty(a) || !OPERATION.hasOwnProperty(b)) {\n      throw new Error(`unknow operation ${a} or ${b}`);\n    }\n    return OPERATION[a] - OPERATION[b];\n  }\n\n  /**\n   * @return string | Node | null\n   */\n  parseStatement() {\n    const token = this.nextToken();\n    if (token === '(') {\n      this.blockLevel += 1;\n      const node = this.parse();\n      this.blockLevel -= 1;\n\n      if (typeof node !== 'string') {\n        node.grouped = true;\n      }\n      return node;\n    }\n\n    if (token === ')') {\n      return null;\n    }\n\n    if (token === '!') {\n      return { left: null, operation: token, right: this.parseStatement() }\n    }\n\n    // 3 > -12 or -12 + 10\n    if (token === '-' && (OPERATION[this.prevToken()] > 0 || this.prevToken() === undefined)) {\n      return { left: '0', operation: token, right: this.parseStatement(), grouped: true };\n    }\n\n    return token;\n  }\n}\n","import { OPERATION } from './Parser';\n\nexport default class QueryParser {\n\n  constructor(tree, queries) {\n    this.tree = tree;\n    this.queries = queries;\n    this.query = null;\n  }\n\n  setQuery(query) {\n    this.query = query.clone();\n  }\n\n  parse() {\n    return this.parseNode(this.tree);\n  }\n\n  parseNode(node) {\n    if (typeof node === 'string') {\n      const nodeQuery = this.getQuery(node);\n      return (query) => { nodeQuery(query); };\n    }\n    if (OPERATION[node.operation] === undefined) {\n      throw new Error(`unknow expression ${node.operation}`);\n    }\n    const leftQuery = this.getQuery(node.left);\n    const rightQuery = this.getQuery(node.right);\n\n    switch (node.operation) {\n      case '&&':\n      case 'AND':\n      default:\n        return (nodeQuery) => nodeQuery.where((query) => {\n          query.where((q) => { leftQuery(q); });\n          query.andWhere((q) => { rightQuery(q); });\n        });\n      case '||':\n      case 'OR':\n        return (nodeQuery) => nodeQuery.where((query) => {\n          query.where((q) => { leftQuery(q); });\n          query.orWhere((q) => { rightQuery(q); });\n        });\n    }\n  }\n\n  getQuery(node) {\n    if (typeof node !== 'string' && node !== null) {\n      return this.parseNode(node);\n    }\n    const value = parseFloat(node);\n\n    if (!isNaN(value)) {\n      if (typeof this.queries[node] === 'undefined') {\n        throw new Error(`unknow query under index ${node}`);\n      }\n      return this.queries[node];\n    }\n    return null;\n  }\n}","\nexport default class MetableCollection {\n  constructor() {\n    this.metadata = [];\n    this.KEY_COLUMN = 'key';\n    this.VALUE_COLUMN = 'value';\n    this.TYPE_COLUMN = 'type';\n    this.model = null;\n    this.extraColumns = [];\n\n    this.extraQuery = (query, meta) => {\n      query.where('key', meta[this.KEY_COLUMN]);\n    };\n  }\n\n  /**\n   * Set model of this metadata collection.\n   * @param {Object} model -\n   */\n  setModel(model) {\n    this.model = model;\n  }\n\n  /**\n   * Find the given metadata key.\n   * @param {String} key -\n   * @return {object} - Metadata object.\n   */\n  findMeta(key) {\n    return this.allMetadata().find((meta) => meta.key === key);\n  }\n\n  /**\n   * Retrieve all metadata.\n   */\n  allMetadata() {\n    return this.metadata.filter((meta) => !meta.markAsDeleted);\n  }\n\n  /**\n   * Retrieve metadata of the given key.\n   * @param {String} key -\n   * @param {Mixied} defaultValue -\n   */\n  getMeta(key, defaultValue) {\n    const metadata = this.findMeta(key);\n    return metadata ? metadata.value : defaultValue || false;\n  }\n\n  /**\n   * Markes the metadata to should be deleted.\n   * @param {String} key -\n   */\n  removeMeta(key) {\n    const metadata = this.findMeta(key);\n\n    if (metadata) {\n      metadata.markAsDeleted = true;\n    }\n  }\n\n  /**\n   * Remove all meta data of the given group.\n   * @param {*} group\n   */\n  removeAllMeta(group = 'default') {\n    this.metadata = this.metadata.map((meta) => ({\n      ...meta,\n      markAsDeleted: true,\n    }));\n  }\n\n  setExtraQuery(callback) {\n    this.extraQuery = callback;\n  }\n\n  /**\n   * Set the meta data to the stack.\n   * @param {String} key -\n   * @param {String} value -\n   */\n  setMeta(key, value, payload) {\n    if (Array.isArray(key)) {\n      const metadata = key;\n\n      metadata.forEach((meta) => {\n        this.setMeta(meta.key, meta.value);\n      });\n      return;\n    }\n    const metadata = this.findMeta(key);\n\n    if (metadata) {\n      metadata.value = value;\n      metadata.markAsUpdated = true;\n    } else {\n      this.metadata.push({\n        value, key, ...payload, markAsInserted: true,\n      });\n    }\n  }\n\n  /**\n   * Saved the modified/deleted and inserted metadata.\n   */\n  async saveMeta() {\n    const inserted = this.metadata.filter((m) => (m.markAsInserted === true));\n    const updated = this.metadata.filter((m) => (m.markAsUpdated === true));\n    const deleted = this.metadata.filter((m) => (m.markAsDeleted === true));\n    const opers = [];\n\n    if (deleted.length > 0) {\n      deleted.forEach((meta) => {\n        const deleteOper = this.model.query().beforeRun((query, result) => {\n          this.extraQuery(query, meta);\n          return result;\n        }).delete();\n        opers.push(deleteOper);\n      });\n    }\n    inserted.forEach((meta) => {\n      const insertOper = this.model.query().insert({\n        [this.KEY_COLUMN]: meta.key,\n        [this.VALUE_COLUMN]: meta.value,\n        ...this.extraColumns.reduce((obj, column) => {\n          if (typeof meta[column] !== 'undefined') {\n            obj[column] = meta[column];\n          }\n          return obj;\n        }, {}),\n      });\n      opers.push(insertOper);\n    });\n    updated.forEach((meta) => {\n      const updateOper = this.model.query().onBuild((query) => {\n        this.extraQuery(query, meta);\n      }).patch({\n        [this.VALUE_COLUMN]: meta.value,\n      });\n      opers.push(updateOper);\n    });\n    await Promise.all(opers);\n  }\n\n  /**\n   * Loads the metadata from the storage.\n   * @param {String|Array} key -\n   * @param {Boolean} force -\n   */\n  async load() {\n    const metadata = await this.query();\n\n    const metadataArray = this.mapMetadataCollection(metadata);\n    metadataArray.forEach((meta) => {\n      this.metadata.push(meta);\n    });\n  }\n\n  /**\n   * Format the metadata before saving to the database.\n   * @param {String|Number|Boolean} value -\n   * @param {String} valueType -\n   * @return {String|Number|Boolean} -\n   */\n  static formatMetaValue(value, valueType) {\n    let parsedValue;\n\n    switch (valueType) {\n      case 'number':\n        parsedValue = `${value}`;\n        break;\n      case 'boolean':\n        parsedValue = value ? '1' : '0';\n        break;\n      case 'json':\n        parsedValue = JSON.stringify(parsedValue);\n        break;\n      default:\n        parsedValue = value;\n        break;\n    }\n    return parsedValue;\n  }\n\n  /**\n   * Mapping and parse metadata to collection entries.\n   * @param {Meta} attr -\n   * @param {String} parseType -\n   */\n  mapMetadata(attr, parseType = 'parse') {\n    return {\n      key: attr[this.KEY_COLUMN],\n      value: (parseType === 'parse')\n        ? MetableCollection.parseMetaValue(\n          attr[this.VALUE_COLUMN],\n          this.TYPE_COLUMN ? attr[this.TYPE_COLUMN] : false,\n        )\n        : MetableCollection.formatMetaValue(\n          attr[this.VALUE_COLUMN],\n          this.TYPE_COLUMN ? attr[this.TYPE_COLUMN] : false,\n        ),\n      ...this.extraColumns.map((extraCol) => ({\n        [extraCol]: attr[extraCol] || null,\n      })),\n    };\n  }\n\n  /**\n   * Parse the metadata to the collection.\n   * @param {Array} collection -\n   */\n  mapMetadataToCollection(metadata, parseType = 'parse') {\n    return metadata.map((model) => this.mapMetadataToCollection(model, parseType));\n  }\n\n  /**\n   * Load metadata to the metable collection.\n   * @param {Array} meta -\n   */\n  from(meta) {\n    if (Array.isArray(meta)) {\n      meta.forEach((m) => { this.from(m); });\n      return;\n    }\n    this.metadata.push(meta);\n  }\n\n\n  toArray() {\n    return this.metadata;\n  }\n\n  /**\n   * Static method to load metadata to the collection.\n   * @param {Array} meta \n   */\n  static from(meta) {\n    const collection = new MetableCollection();\n    collection.from(meta);\n\n    return collection;\n  }\n}\n","import { difference } from 'lodash';\nimport { Lexer } from '@/lib/LogicEvaluation/Lexer';\nimport Parser from '@/lib/LogicEvaluation/Parser';\nimport QueryParser from '@/lib/LogicEvaluation/QueryParser';\nimport resourceFieldsKeys from '@/data/ResourceFieldsKeys';\n\n//  const role = {\n//   compatotor: '',\n//   value: '',\n//   columnKey: '',\n//   columnSlug: '',\n//   index: 1,\n// }\n\nexport function buildRoleQuery(role) {\n  const columnName = resourceFieldsKeys[role.columnKey];\n\n  switch (role.comparator) {\n    case 'equals':\n    default:\n      return (builder) => {\n        builder.where(columnName, role.value);\n      };\n    case 'not_equal':\n    case 'not_equals':\n      return (builder) => {\n        builder.whereNot(columnName, role.value);\n      };\n  }\n}\n\n/**\n * Builds database query from stored view roles.\n *\n * @param {Array} roles -\n * @return {Function}\n */\nexport function viewRolesBuilder(roles, logicExpression = '') {\n  const rolesIndexSet = {};\n\n  roles.forEach((role) => {\n    rolesIndexSet[role.index] = buildRoleQuery(role);\n  });\n  // Lexer for logic expression.\n  const lexer = new Lexer(logicExpression);\n  const tokens = lexer.getTokens();\n\n  // Parse the logic expression.\n  const parser = new Parser(tokens);\n  const parsedTree = parser.parse();\n\n  const queryParser = new QueryParser(parsedTree, rolesIndexSet);\n  return queryParser.parse();\n}\n\n/**\n * Validates the view logic expression.\n * @param {String} logicExpression \n * @param {Array} indexes \n */\nexport function validateViewLogicExpression(logicExpression, indexes) {\n  const logicExpIndexes = logicExpression.match(/\\d+/g) || [];\n  return !difference(logicExpIndexes.map(Number), indexes).length;\n}\n\n/**\n * \n * @param {Array} roles -\n * @param {String} logicExpression -\n * @return {Boolean}\n */\nexport function validateViewRoles(roles, logicExpression) {\n  return validateViewLogicExpression(logicExpression, roles.map((r) => r.index));\n}\n\n/**\n * Mapes the view roles to view conditionals.\n * @param {Array} viewRoles -\n * @return {Array}\n */\nexport function mapViewRolesToConditionals(viewRoles) {\n  return viewRoles.map((viewRole) => ({\n    comparator: viewRole.comparator,\n    value: viewRole.value,\n    columnKey: viewRole.field.columnKey,\n    slug: viewRole.field.slug,\n    index: viewRole.index,\n  }));\n}","/* eslint-disable global-require */\nimport { Model } from 'objection';\nimport { flatten } from 'lodash';\nimport BaseModel from '@/models/Model';\nimport {viewRolesBuilder} from '@/lib/ViewRolesBuilder';\n\nexport default class Account extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'accounts';\n  }\n\n  /**\n   * Model modifiers.\n   */\n  static get modifiers() {\n    return {\n      filterAccountTypes(query, typesIds) {\n        if (typesIds.length > 0) {\n          query.whereIn('accoun_type_id', typesIds);\n        }\n      },\n      viewRolesBuilder(query, conditionals, expression) {\n        viewRolesBuilder(conditionals, expression)(query);\n      },\n    };\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const AccountType = require('@/models/AccountType');\n    const AccountBalance = require('@/models/AccountBalance');\n    const AccountTransaction = require('@/models/AccountTransaction');\n\n    return {\n      /**\n       * Account model may belongs to account type.\n       */\n      type: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: AccountType.default,\n        join: {\n          from: 'accounts.accountTypeId',\n          to: 'account_types.id',\n        },\n      },\n\n      /**\n       * Account model may has many balances accounts.\n       */\n      balance: {\n        relation: Model.HasOneRelation,\n        modelClass: AccountBalance.default,\n        join: {\n          from: 'accounts.id',\n          to: 'account_balances.accountId',\n        },\n      },\n\n      /**\n       * Account model may has many transactions.\n       */\n      transactions: {\n        relation: Model.HasManyRelation,\n        modelClass: AccountTransaction.default,\n        join: {\n          from: 'accounts.id',\n          to: 'accounts_transactions.accountId',\n        },\n      },\n    };\n  }\n\n  static collectJournalEntries(accounts) {\n    return flatten(accounts.map((account) => account.transactions.map((transaction) => ({\n      accountId: account.id,\n      ...transaction,\n      accountNormal: account.type.normal,\n    }))));\n  }\n}\n","import { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class AccountBalance extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'account_balances';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Account = require('@/models/Account');\n\n    return {\n      account: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: Account.default,\n        join: {\n          from: 'account_balance.account_id',\n          to: 'accounts.id',\n        },\n      },\n    };\n  }\n}\n","import { Model } from 'objection';\nimport moment from 'moment';\nimport BaseModel from '@/models/Model';\n\nexport default class AccountTransaction extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'accounts_transactions';\n  }\n\n  /**\n   * Model modifiers.\n   */\n  static get modifiers() {\n    return {\n      filterAccounts(query, accountsIds) {\n        if (accountsIds.length > 0) {\n          query.whereIn('account_id', accountsIds);\n        }\n      },\n      filterTransactionTypes(query, types) {\n        if (Array.isArray(types) && types.length > 0) {\n          query.whereIn('reference_type', types);\n        } else if (typeof types === 'string') {\n          query.where('reference_type', types);\n        }\n      },\n      filterDateRange(query, startDate, endDate, type = 'day') {\n        const dateFormat = 'YYYY-MM-DD HH:mm:ss';\n        const fromDate = moment(startDate).startOf(type).format(dateFormat);\n        const toDate = moment(endDate).endOf(type).format(dateFormat);\n\n        if (startDate) {\n          query.where('date', '>=', fromDate);\n        }\n        if (endDate) {\n          query.where('date', '<=', toDate);\n        }\n      },\n      filterAmountRange(query, fromAmount, toAmount) {\n        if (fromAmount) {\n          query.andWhere((q) => {\n            q.where('credit', '>=', fromAmount);\n            q.orWhere('debit', '>=', fromAmount);\n          });\n        }\n        if (toAmount) {\n          query.andWhere((q) => {\n            q.where('credit', '<=', toAmount);\n            q.orWhere('debit', '<=', toAmount);\n          });\n        }\n      },\n      sumationCreditDebit(query) {\n        query.sum('credit as credit');\n        query.sum('debit as debit');\n        query.groupBy('account_id');\n      },\n    };\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Account = require('@/models/Account');\n\n    return {\n      account: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: Account.default,\n        join: {\n          from: 'accounts_transactions.accountId',\n          to: 'accounts.id',\n        },\n      },\n    };\n  }\n}\n","// import path from 'path';\nimport { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class AccountType extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'account_types';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Account = require('@/models/Account');\n\n    return {\n      /**\n       * Account type may has many associated accounts.\n       */\n      accounts: {\n        relation: Model.HasManyRelation,\n        modelClass: Account.default,\n        join: {\n          from: 'account_types.id',\n          to: 'accounts.accountTypeId',\n        },\n      },\n    };\n  }\n}\n","import BaseModel from '@/models/Model';\n\nexport default class Budget extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'budgets';\n  }\n\n  static get virtualAttributes() {\n    return ['rangeBy', 'rangeIncrement'];\n  }\n\n  /**\n   * Model modifiers.\n   */\n  static get modifiers() {\n    return {\n      filterByYear(query, year) {\n        query.where('year', year);\n      },\n      filterByIncomeStatement(query) {\n        query.where('account_types', 'income_statement');\n      },\n      filterByProfitLoss(query) {\n        query.where('accounts_types', 'profit_loss');\n      },\n    };\n  }\n\n  get rangeBy() {\n    switch (this.period) {\n      case 'half-year':\n      case 'quarter':\n        return 'month';\n      default:\n        return this.period;\n    }\n  }\n\n  get rangeIncrement() {\n    switch (this.period) {\n      case 'half-year':\n        return 6;\n      case 'quarter':\n        return 3;\n      default:\n        return 1;\n    }\n  }\n\n  get rangeOffset() {\n    switch (this.period) {\n      case 'half-year': return 5;\n      case 'quarter': return 2;\n      default: return 0;\n    }\n  }\n}\n","import BaseModel from '@/models/Model';\n\nexport default class Budget extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'budget_entries';\n  }\n}\n","import { Model } from 'objection';\nimport BaseModel from '@/models/Model';\nimport {viewRolesBuilder} from '@/lib/ViewRolesBuilder';\nexport default class Expense extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'expenses';\n  }\n\n  static get referenceType() {\n    return 'Expense';\n  }\n\n  /**\n   * Model modifiers.\n   */\n  static get modifiers() {\n    return {\n      filterByDateRange(query, startDate, endDate) {\n        if (startDate) {\n          query.where('date', '>=', startDate);\n        }\n        if (endDate) {\n          query.where('date', '<=', endDate);\n        }\n      },\n      filterByAmountRange(query, from, to) {\n        if (from) {\n          query.where('amount', '>=', from);\n        }\n        if (to) {\n          query.where('amount', '<=', to);\n        }\n      },\n      filterByExpenseAccount(query, accountId) {\n        if (accountId) {\n          query.where('expense_account_id', accountId);\n        }\n      },\n      filterByPaymentAccount(query, accountId) {\n        if (accountId) {\n          query.where('payment_account_id', accountId);\n        }\n      },\n\n      viewRolesBuilder(query, conditionals, expression) {\n        viewRolesBuilder(conditionals, expression)(query);\n      },\n\n      orderBy(query) {\n        \n      }\n    };\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Account = require('@/models/Account');\n    const User = require('@/models/User');\n\n    return {\n      paymentAccount: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: Account.default,\n        join: {\n          from: 'expenses.paymentAccountId',\n          to: 'accounts.id',\n        },\n      },\n\n      expenseAccount: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: Account.default,\n        join: {\n          from: 'expenses.expenseAccountId',\n          to: 'accounts.id',\n        },\n      },\n\n      user: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: User.default,\n        join: {\n          from: 'expenses.userId',\n          to: 'users.id',\n        },\n      },\n    };\n  }\n}\n","import { Model } from 'objection';\nimport path from 'path';\nimport BaseModel from '@/models/Model';\n\nexport default class Item extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'items';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    return {\n      /**\n       * Item may has many meta data.\n       */\n      metadata: {\n        relation: Model.HasManyRelation,\n        modelBase: path.join(__dirname, 'ItemMetadata'),\n        join: {\n          from: 'items.id',\n          to: 'items_metadata.item_id',\n        },\n      },\n\n      /**\n       * Item may belongs to cateogory model.\n       */\n      category: {\n        relation: Model.BelongsToOneRelation,\n        modelBase: path.join(__dirname, 'ItemCategory'),\n        join: {\n          from: 'items.categoryId',\n          to: 'items_categories.id',\n        },\n      },\n    };\n  }\n}\n","import path from 'path';\nimport { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class ItemCategory extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'items_categories';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    return {\n      /**\n       * Item category may has many items.\n       */\n      items: {\n        relation: Model.HasManyRelation,\n        modelBase: path.join(__dirname, 'Item'),\n        join: {\n          from: 'items_categories.item_id',\n          to: 'items.id',\n        },\n      },\n    };\n  }\n}\n","import BaseModel from '@/models/Model';\n\nexport default class JournalEntry extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'manual_journals';\n  }\n}\n","import BaseModel from '@/models/Model';\n\nexport default class ManualJournal extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'manual_journals';\n  }\n}\n","import { Model } from 'objection';\nimport {transform, snakeCase} from 'lodash';\nimport {mapKeysDeep} from '@/utils';\n\nexport default class ModelBase extends Model {\n  static get collection() {\n    return Array;\n  }\n\n  static query(...args) {\n    return super.query(...args).runAfter((result) => {\n      if (Array.isArray(result)) {\n        return this.collection.from(result);\n      }\n      return result;\n    });\n  }\n\n  $formatJson(json, opt) {\n    const transformed = mapKeysDeep(json, (value, key) => {\n      return snakeCase(key);\n    });\n    const parsedJson = super.$formatJson(transformed, opt);\n\n    return parsedJson;\n  }\n}\n","import { mixin } from 'objection';\nimport BaseModel from '@/models/Model';\nimport MetableCollection from '@/lib/Metable/MetableCollection';\n\nexport default class Option extends mixin(BaseModel, [mixin]) {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'options';\n  }\n\n  /**\n   * Override the model query.\n   * @param  {...any} args -\n   */\n  static query(...args) {\n    return super.query(...args).runAfter((result) => {\n      if (result instanceof MetableCollection) {\n        result.setModel(Option);\n      }\n      return result;\n    });\n  }\n\n  static get collection() {\n    return MetableCollection;\n  }\n}\n","import Model from '@/models/Model';\n\nexport default class PasswordResets extends Model {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'password_resets';\n  }\n}\n","import { Model } from 'objection';\nimport path from 'path';\nimport BaseModel from '@/models/Model';\n\nexport default class Permission extends BaseModel {\n  /**\n   * Table name of Role model.\n   * @type {String}\n   */\n  static get tableName() {\n    return 'permissions';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    return {\n      /**\n       * Permission model may belongs to role model.\n       */\n      role: {\n        relation: Model.BelongsToOneRelation,\n        modelBase: path.join(__dirname, 'Role'),\n        join: {\n          from: 'permissions.role_id',\n          to: 'roles.id',\n        },\n      },\n\n      // resource: {\n      //   relation: Model.BelongsToOneRelation,\n      //   modelBase: path.join(__dirname, 'Resource'),\n      //   join: {\n      //     from: 'permissions.',\n      //     to: '',\n      //   }\n      // }\n    };\n  }\n}\n","import path from 'path';\nimport { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class Resource extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'resources';\n  }\n\n  /**\n   * Timestamp columns.\n   */\n  static get hasTimestamps() {\n    return false;\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const View = require('@/models/View');\n    const ResourceField = require('@/models/ResourceField');\n    const Permission = require('@/models/Permission');\n\n    return {\n      /**\n       * Resource model may has many views.\n       */\n      views: {\n        relation: Model.HasManyRelation,\n        modelClass: View.default,\n        join: {\n          from: 'resources.id',\n          to: 'views.resourceId',\n        },\n      },\n\n      /**\n       * Resource model may has many fields.\n       */\n      fields: {\n        relation: Model.HasManyRelation,\n        modelClass: ResourceField.default,\n        join: {\n          from: 'resources.id',\n          to: 'resource_fields.resourceId',\n        },\n      },\n\n      /**\n       * Resource model may has many associated permissions.\n       */\n      permissions: {\n        relation: Model.ManyToManyRelation,\n        modelClass: Permission.default,\n        join: {\n          from: 'resources.id',\n          through: {\n            from: 'role_has_permissions.resourceId',\n            to: 'role_has_permissions.permissionId',\n          },\n          to: 'permissions.id',\n        },\n      },\n    };\n  }\n}\n","import { snakeCase } from 'lodash';\nimport { Model } from 'objection';\nimport path from 'path';\nimport BaseModel from '@/models/Model';\n\nexport default class ResourceField extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'resource_fields';\n  }\n\n  static get jsonAttributes() {\n    return ['options'];\n  }\n\n  /**\n   * Model modifiers.\n   */\n  static get modifiers() {\n    return {\n      whereNotPredefined(query) {\n        query.whereNot('predefined', true);\n      },\n    };\n  }\n\n  /**\n   * Timestamp columns.\n   */\n  static get hasTimestamps() {\n    return false;\n  }\n\n  /**\n   * Virtual attributes.\n   */\n  static get virtualAttributes() {\n    return ['key'];\n  }\n\n  /**\n   * Resource field key.\n   */\n  key() {\n    return snakeCase(this.labelName);\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    return {\n      /**\n       * Resource field may belongs to resource model.\n       */\n      resource: {\n        relation: Model.BelongsToOneRelation,\n        modelBase: path.join(__dirname, 'Resource'),\n        join: {\n          from: 'resource_fields.resource_id',\n          to: 'resources.id',\n        },\n      },\n    };\n  }\n}\n","import { Model } from 'objection';\nimport path from 'path';\nimport BaseModel from '@/models/Model';\nimport ResourceFieldMetadataCollection from '@/collection/ResourceFieldMetadataCollection';\n\nexport default class ResourceFieldMetadata extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'resource_custom_fields_metadata';\n  }\n\n  /**\n   * Override the resource field metadata collection.\n   */\n  static get collection() {\n    return ResourceFieldMetadataCollection;\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    return {\n      /**\n       * Resource field may belongs to resource model.\n       */\n      resource: {\n        relation: Model.BelongsToOneRelation,\n        modelBase: path.join(__dirname, 'Resource'),\n        join: {\n          from: 'resource_fields.resource_id',\n          to: 'resources.id',\n        },\n      },\n    };\n  }\n}\n","import { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class Role extends BaseModel {\n  /**\n   * Table name of Role model.\n   * @type {String}\n   */\n  static get tableName() {\n    return 'roles';\n  }\n\n  /**\n   * Timestamp columns.\n   */\n  static get hasTimestamps() {\n    return false;\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Permission = require('@/models/Permission');\n    const Resource = require('@/models/Resource');\n    const User = require('@/models/User');\n    const ResourceField = require('@/models/ResourceField');\n\n    return {\n      /**\n       * Role may has many permissions.\n       */\n      permissions: {\n        relation: Model.ManyToManyRelation,\n        modelClass: Permission.default,\n        join: {\n          from: 'roles.id',\n          through: {\n            from: 'role_has_permissions.roleId',\n            to: 'role_has_permissions.permissionId',\n          },\n          to: 'permissions.id',\n        },\n      },\n\n      /**\n       * Role may has many resources.\n       */\n      resources: {\n        relation: Model.ManyToManyRelation,\n        modelClass: Resource.default,\n        join: {\n          from: 'roles.id',\n          through: {\n            from: 'role_has_permissions.roleId',\n            to: 'role_has_permissions.resourceId',\n          },\n          to: 'resources.id',\n        },\n      },\n\n      /**\n       * Role may has resource field.\n       */\n      field: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: ResourceField.default,\n        join: {\n          from: 'roles.fieldId',\n          to: 'resource_fields.id',\n        }\n      },\n\n      /**\n       * Role may has many associated users.\n       */\n      users: {\n        relation: Model.ManyToManyRelation,\n        modelClass: User.default,\n        join: {\n          from: 'roles.id',\n          through: {\n            from: 'user_has_roles.roleId',\n            to: 'user_has_roles.userId',\n          },\n          to: 'users.id',\n        },\n      },\n    };\n  }\n}\n","import bcrypt from 'bcryptjs';\nimport { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n// import PermissionsService from '@/services/PermissionsService';\n\nexport default class User extends BaseModel {\n  // ...PermissionsService\n\n  static get virtualAttributes() {\n    return ['fullName'];\n  }\n\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'users';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Role = require('@/models/Role');\n\n    return {\n      roles: {\n        relation: Model.ManyToManyRelation,\n        modelClass: Role.default,\n        join: {\n          from: 'users.id',\n          through: {\n            from: 'user_has_roles.userId',\n            to: 'user_has_roles.roleId',\n          },\n          to: 'roles.id',\n        },\n      },\n    };\n  }\n\n  /**\n   * Verify the password of the user.\n   * @param  {String} password - The given password.\n   * @return {Boolean}\n   */\n  verifyPassword(password) {\n    return bcrypt.compareSync(password, this.password);\n  }\n\n  fullName() {\n    return `${this.firstName} ${this.lastName}`;\n  }\n}\n","import path from 'path';\nimport { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class View extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'views';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Resource = require('@/models/Resource');\n    const ViewColumn = require('@/models/ViewColumn');\n    const ViewRole = require('@/models/ViewRole');\n\n    return {\n      /**\n       * View model belongs to resource model.\n       */\n      resource: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: Resource.default,\n        join: {\n          from: 'views.resourceId',\n          to: 'resources.id',\n        },\n      },\n\n      /**\n       * View model may has many columns.\n       */\n      columns: {\n        relation: Model.HasManyRelation,\n        modelClass: ViewColumn.default,\n        join: {\n          from: 'views.id',\n          to: 'view_has_columns.viewId',\n        },\n      },\n\n      /**\n       * View model may has many view roles.\n       */\n      roles: {\n        relation: Model.HasManyRelation,\n        modelClass: ViewRole.default,\n        join: {\n          from: 'views.id',\n          to: 'view_roles.viewId',\n        },\n      },\n    };\n  }\n}\n","import BaseModel from '@/models/Model';\n\nexport default class ViewColumn extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'view_has_columns';\n  }\n\n  /**\n   * Timestamp columns.\n   */\n  static get hasTimestamps() {\n    return false;\n  }\n}\n","import { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class ViewRole extends BaseModel {\n\n  /**\n   * Virtual attributes.\n   */\n  static get virtualAttributes() {\n    return ['comparators'];\n  }\n\n  static get comparators() {\n    return [\n      'equals', 'not_equal', 'contains', 'not_contain',\n    ];\n  }\n\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'view_roles';\n  }\n\n  /**\n   * Timestamp columns.\n   */\n  static get hasTimestamps() {\n    return false;\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const ResourceField = require('@/models/ResourceField');\n    const View = require('@/models/View');\n\n    return {\n      /**\n       * View role model may belongs to view model.\n       */\n      view: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: View.default,\n        join: {\n          from: 'view_roles.viewId',\n          to: 'views.id',\n        },\n      },\n\n      /**\n       * View role model may belongs to resource field model.\n       */\n      field: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: ResourceField.default,\n        join: {\n          from: 'view_roles.fieldId',\n          to: 'resource_fields.id',\n        },\n      },\n    };\n  }\n}\n","import { Model } from 'objection';\nimport knex from '@/database/knex';\n\n// Bind all Models to a knex instance. If you only have one database in\n// your server this is all you have to do. For multi database systems, see\n// the Model.bindKnex() method.\nModel.knex(knex);\n","import errorHandler from 'errorhandler';\nimport app from '@/app';\n\napp.use(errorHandler);\n\nconst server = app.listen(app.get('port'), () => {\n  console.log(\n    '  App is running at http://localhost:%d in %s mode',\n    app.get('port'),\n    app.get('env'),\n  );\n  console.log('  Press CTRL-C to stop');\n});\n\nexport default server;\n","\nexport default class JournalEntry {\n  constructor(entry) {\n    const defaults = {\n      credit: 0,\n      debit: 0,\n    };\n    this.entry = { ...defaults, ...entry };\n  }\n}\n","import { pick } from 'lodash';\nimport moment from 'moment';\nimport JournalEntry from '@/services/Accounting/JournalEntry';\nimport AccountTransaction from '@/models/AccountTransaction';\nimport AccountBalance from '@/models/AccountBalance';\n\nexport default class JournalPoster {\n  /**\n   * Journal poster constructor.\n   */\n  constructor() {\n    this.entries = [];\n    this.balancesChange = {};\n  }\n\n  /**\n   * Writes the credit entry for the given account.\n   * @param {JournalEntry} entry -\n   */\n  credit(entryModel) {\n    if (entryModel instanceof JournalEntry === false) {\n      throw new Error('The entry is not instance of JournalEntry.');\n    }\n    this.entries.push(entryModel.entry);\n    this.setAccountBalanceChange(entryModel.entry, 'credit');\n  }\n\n  /**\n   * Writes the debit entry for the given account.\n   * @param {JournalEntry} entry -\n   */\n  debit(entryModel) {\n    if (entryModel instanceof JournalEntry === false) {\n      throw new Error('The entry is not instance of JournalEntry.');\n    }\n    this.entries.push(entryModel.entry);\n    this.setAccountBalanceChange(entryModel.entry, 'debit');\n  }\n\n  /**\n   * Sets account balance change.\n   * @param {JournalEntry} entry\n   * @param {String} type\n   */\n  setAccountBalanceChange(entry, type) {\n    if (!this.balancesChange[entry.account]) {\n      this.balancesChange[entry.account] = 0;\n    }\n    let change = 0;\n\n    if (entry.accountNormal === 'credit') {\n      change = (type === 'credit') ? entry.credit : -1 * entry.debit;\n    } else if (entry.accountNormal === 'debit') {\n      change = (type === 'debit') ? entry.debit : -1 * entry.credit;\n    }\n    this.balancesChange[entry.account] += change;\n  }\n\n  /**\n   * Mapping the balance change to list.\n   */\n  mapBalanceChangesToList() {\n    const mappedList = [];\n\n    Object.keys(this.balancesChange).forEach((accountId) => {\n      const balance = this.balancesChange[accountId];\n\n      mappedList.push({\n        account_id: accountId,\n        amount: balance,\n      });\n    });\n    return mappedList;\n  }\n\n  /**\n   * Saves the balance change of journal entries.\n   */\n  async saveBalance() {\n    const balancesList = this.mapBalanceChangesToList();\n    const balanceUpdateOpers = [];\n    const balanceInsertOpers = [];\n    const balanceFindOneOpers = [];\n    let balanceAccounts = [];\n\n    balancesList.forEach((balance) => {\n      const oper = AccountBalance.query().findOne('account_id', balance.account_id);\n      balanceFindOneOpers.push(oper);\n    });\n    balanceAccounts = await Promise.all(balanceFindOneOpers);\n\n    balancesList.forEach((balance) => {\n      const method = balance.amount < 0 ? 'decrement' : 'increment';\n\n      // Detarmine if the account balance is already exists or not.\n      const foundAccBalance = balanceAccounts.some((account) => (\n        account && account.account_id === balance.account_id\n      ));\n      if (foundAccBalance) {\n        const query = AccountBalance\n          .query()[method]('amount', Math.abs(balance.amount))\n          .where('account_id', balance.account_id);\n\n        balanceUpdateOpers.push(query);\n      } else {\n        const query = AccountBalance.query().insert({\n          account_id: balance.account_id,\n          amount: balance.amount,\n          currency_code: 'USD',\n        });\n        balanceInsertOpers.push(query);\n      }\n    });\n    await Promise.all([\n      ...balanceUpdateOpers, ...balanceInsertOpers,\n    ]);\n  }\n\n  /**\n   * Saves the stacked journal entries to the storage.\n   */\n  async saveEntries() {\n    const saveOperations = [];\n\n    this.entries.forEach((entry) => {\n      const oper = AccountTransaction.query().insert({\n        accountId: entry.account,\n        ...pick(entry, ['credit', 'debit', 'transactionType',\n          'referenceType', 'referenceId', 'note']),\n      });\n      saveOperations.push(oper);\n    });\n    await Promise.all(saveOperations);\n  }\n\n  /**\n   * Reverses the stacked journal entries.\n   */\n  reverseEntries() {\n    const reverseEntries = [];\n\n    this.entries.forEach((entry) => {\n      const reverseEntry = { ...entry };\n\n      if (entry.credit) {\n        reverseEntry.debit = entry.credit;\n      }\n      if (entry.debit) {\n        reverseEntry.credit = entry.debit;\n      }\n      reverseEntries.push(reverseEntry);\n    });\n    this.entries = reverseEntries;\n  }\n\n  /**\n   * Delete the given or all stacked entries.\n   * @param {Array} ids -\n   */\n  async deleteEntries(ids) {\n    const entriesIds = ids || this.entries.map((e) => e.id);\n\n    if (entriesIds.length > 0) {\n      await AccountTransaction.query().whereIn('id', entriesIds).delete();\n    }\n  }\n\n  /**\n   * Retrieve the closing balance for the given account and closing date.\n   * @param {Number} accountId -\n   * @param {Date} closingDate -\n   */\n  getClosingBalance(accountId, closingDate, dateType = 'day') {\n    let closingBalance = 0;\n    const momentClosingDate = moment(closingDate);\n\n    this.entries.forEach((entry) => {\n      // Can not continue if not before or event same closing date.\n      if ((!momentClosingDate.isAfter(entry.date, dateType)\n        && !momentClosingDate.isSame(entry.date, dateType))\n        || (entry.account !== accountId && accountId)) {\n        return;\n      }\n      if (entry.accountNormal === 'credit') {\n        closingBalance += (entry.credit) ? entry.credit : -1 * entry.debit;\n      } else if (entry.accountNormal === 'debit') {\n        closingBalance += (entry.debit) ? entry.debit : -1 * entry.credit;\n      }\n    });\n    return closingBalance;\n  }\n\n  /**\n   * Retrieve the credit/debit sumation for the given account and date.\n   * @param {Number} account -\n   * @param {Date|String} closingDate -\n   */\n  getTrialBalance(accountId, closingDate, dateType) {\n    const momentClosingDate = moment(closingDate);\n    const result = {\n      credit: 0,\n      debit: 0,\n      balance: 0,\n    };\n    this.entries.forEach((entry) => {\n      if ((!momentClosingDate.isAfter(entry.date, dateType)\n        && !momentClosingDate.isSame(entry.date, dateType))\n        || (entry.account !== accountId && accountId)) {\n        return;\n      }\n      result.credit += entry.credit;\n      result.debit += entry.debit;\n\n      if (entry.accountNormal === 'credit') {\n        result.balance += (entry.credit) ? entry.credit : -1 * entry.debit;\n      } else if (entry.accountNormal === 'debit') {\n        result.balance += (entry.debit) ? entry.debit : -1 * entry.credit;\n      }\n    });\n    return result;\n  }\n\n  /**\n   * Load fetched accounts journal entries.\n   * @param {Array} entries -\n   */\n  loadEntries(entries) {\n    entries.forEach((entry) => {\n      this.entries.push({\n        ...entry,\n        account: entry.account ? entry.account.id : entry.accountId,\n        accountNormal: (entry.account && entry.account.type)\n          ? entry.account.type.normal : entry.accountNormal,\n      });\n    });\n  }\n\n  static loadAccounts() {\n\n  }\n}\n","import Resource from '@/models/Resource';\nimport ResourceField from '@/models/ResourceField';\nimport ResourceFieldMetadata from '@/models/ResourceFieldMetadata';\nimport ResourceFieldMetadataCollection from '@/collection/ResourceFieldMetadataCollection';\n\nexport default class ResourceCustomFieldRepository {\n  /**\n   * Class constructor.\n   */\n  constructor(model) {\n    if (typeof model === 'function') {\n      this.resourceName = model.name;\n    } else if (typeof model === 'string') {\n      this.resourceName = model;\n    }\n    // Custom fields of the given resource.\n    this.customFields = [];\n    this.filledCustomFields = {};\n\n    // metadata of custom fields of the given resource.\n    this.fieldsMetadata = {};\n    this.resource = {};\n  }\n\n  /**\n   * Fetches metadata of custom fields of the given resource.\n   * @param {Integer} id - Resource item id.\n   */\n  async fetchCustomFieldsMetadata(id) {\n    if (typeof id === 'undefined') {\n      throw new Error('Please define the resource item id.');\n    }\n    if (!this.resource) {\n      throw new Error('Target resource model is not found.');\n    }\n    const metadata = await ResourceFieldMetadata.query()\n      .where('resource_id', this.resource.id)\n      .where('resource_item_id', id);\n\n    this.fieldsMetadata[id] = metadata;\n  }\n\n  /**\n   * Load resource.\n   */\n  async loadResource() {\n    const resource = await Resource.query().where('name', this.resourceName).first();\n\n    if (!resource) {\n      throw new Error('There is no stored resource in the storage with the given model name.');\n    }\n    this.setResource(resource);\n  }\n\n  /**\n   * Load metadata of the resource.\n   */\n  async loadResourceCustomFields() {\n    if (typeof this.resource.id === 'undefined') {\n      throw new Error('Please fetch resource details before fetch custom fields of the resource.');\n    }\n    const customFields = await ResourceField.query()\n      .where('resource_id', this.resource.id)\n      .modify('whereNotPredefined');\n\n    this.setResourceCustomFields(customFields);\n  }\n\n  /**\n   * Sets resource model.\n   * @param {Resource} resource -\n   */\n  setResource(resource) {\n    this.resource = resource;\n  }\n\n  /**\n   * Sets resource custom fields collection.\n   * @param {Array} customFields -\n   */\n  setResourceCustomFields(customFields) {\n    this.customFields = customFields;\n  }\n\n  /**\n   * Retrieve metadata of the resource custom fields.\n   * @param {Integer} itemId -\n   */\n  getMetadata(itemId) {\n    return this.fieldsMetadata[itemId] || this.fieldsMetadata;\n  }\n\n  /**\n   * Fill metadata of the custom fields that associated to the resource.\n   * @param {Inter} id - Resource item id.\n   * @param {Array} attributes -\n   */\n  fillCustomFields(id, attributes) {\n    if (typeof this.filledCustomFields[id] === 'undefined') {\n      this.filledCustomFields[id] = [];\n    }\n    attributes.forEach((attr) => {\n      this.filledCustomFields[id].push(attr);\n\n      if (!this.fieldsMetadata[id]) {\n        this.fieldsMetadata[id] = new ResourceFieldMetadataCollection();\n      }\n      this.fieldsMetadata[id].setMeta(attr.key, attr.value, {\n        resource_id: this.resource.id,\n        resource_item_id: id,\n      });\n    });\n  }\n\n  /**\n   * Saves the instered, updated and deleted  custom fields metadata.\n   * @param {Integer} id - Optional resource item id.\n   */\n  async saveCustomFields(id) {\n    if (id) {\n      if (typeof this.fieldsMetadata[id] === 'undefined') {\n        throw new Error('There is no resource item with the given id.');\n      }\n      await this.fieldsMetadata[id].saveMeta();\n    } else {\n      const opers = [];\n      this.fieldsMetadata.forEach((metadata) => {\n        const oper = metadata.saveMeta();\n        opers.push(oper);\n      });\n      await Promise.all(opers);\n    }\n  }\n\n  /**\n   * Validates the exist custom fields.\n   */\n  validateExistCustomFields() {\n\n  }\n\n  toArray() {\n    return this.fieldsMetadata.toArray();\n  }\n\n  async load() {\n    await this.loadResource();\n    await this.loadResourceCustomFields();\n  }\n\n  static forgeMetadataCollection() {\n\n  }\n}\n","import Moment from 'moment';\nimport { extendMoment } from 'moment-range';\n\nconst moment = extendMoment(Moment);\n\nexport default moment;\n","import nodemailer from 'nodemailer';\n\n// create reusable transporter object using the default SMTP transport\nconst transporter = nodemailer.createTransport({\n  host: process.env.MAIL_HOST,\n  port: Number(process.env.MAIL_PORT),\n  secure: process.env.MAIL_SECURE === 'true', // true for 465, false for other ports\n  auth: {\n    user: process.env.MAIL_USERNAME,\n    pass: process.env.MAIL_PASSWORD,\n  },\n});\n\nexport default transporter;\n","import bcrypt from 'bcryptjs';\nimport moment from 'moment';\nimport _ from 'lodash';\nconst { map, isArray, isPlainObject, mapKeys, mapValues } = require('lodash')\n\n\nconst hashPassword = (password) => new Promise((resolve) => {\n  bcrypt.genSalt(10, (error, salt) => {\n    bcrypt.hash(password, salt, (err, hash) => { resolve(hash); });\n  });\n});\n\nconst origin = (request) => `${request.protocol}://${request.hostname}`;\n\nconst dateRangeCollection = (fromDate, toDate, addType = 'day', increment = 1) => {\n  const collection = [];\n  const momentFromDate = moment(fromDate);\n  let dateFormat = '';\n\n  switch (addType) {\n    case 'day':\n    default:\n      dateFormat = 'YYYY-MM-DD'; break;\n    case 'month':\n    case 'quarter':\n      dateFormat = 'YYYY-MM'; break;\n    case 'year':\n      dateFormat = 'YYYY'; break;\n  }\n  for (let i = momentFromDate;\n    (i.isBefore(toDate, addType) || i.isSame(toDate, addType));\n    i.add(increment, `${addType}s`)) {\n    collection.push(i.endOf(addType).format(dateFormat));\n  }\n  return collection;\n};\n\nconst dateRangeFormat = (rangeType) => {\n  switch (rangeType) {\n    case 'year':\n      return 'YYYY';\n    case 'month':\n    case 'quarter':\n    default:\n      return 'YYYY-MM';\n  }\n};\n\n\nconst mapKeysDeep = (obj, cb) => {\n  if (_.isArray(obj)) {\n      return obj.map(innerObj => mapKeysDeep(innerObj, cb));\n  }\n  else if (_.isObject(obj)) {\n      return _.mapValues(\n          _.mapKeys(obj, cb),\n          val => mapKeysDeep(val, cb),\n      )\n  } else {\n      return obj;\n  }\n}\n\nconst mapValuesDeep = (v, callback) => (\n  _.isObject(v)\n    ? _.mapValues(v, v => mapValuesDeep(v, callback))\n    : callback(v));\n\nexport {\n  hashPassword,\n  origin,\n  dateRangeCollection,\n  dateRangeFormat,\n  mapValuesDeep,\n  mapKeysDeep,\n};\n","module.exports = require(\"@babel/plugin-transform-runtime\");","module.exports = require(\"@babel/runtime/helpers/asyncToGenerator\");","module.exports = require(\"@babel/runtime/helpers/classCallCheck\");","module.exports = require(\"@babel/runtime/helpers/createClass\");","module.exports = require(\"@babel/runtime/helpers/defineProperty\");","module.exports = require(\"@babel/runtime/helpers/get\");","module.exports = require(\"@babel/runtime/helpers/getPrototypeOf\");","module.exports = require(\"@babel/runtime/helpers/inherits\");","module.exports = require(\"@babel/runtime/helpers/possibleConstructorReturn\");","module.exports = require(\"@babel/runtime/helpers/slicedToArray\");","module.exports = require(\"@babel/runtime/helpers/toConsumableArray\");","module.exports = require(\"@babel/runtime/regenerator\");","module.exports = require(\"bcryptjs\");","module.exports = require(\"dotenv\");","module.exports = require(\"errorhandler\");","module.exports = require(\"express\");","module.exports = require(\"express-boom\");","module.exports = require(\"express-validator\");","module.exports = require(\"fs\");","module.exports = require(\"helmet\");","module.exports = require(\"jsonwebtoken\");","module.exports = require(\"knex\");","module.exports = require(\"lodash\");","module.exports = require(\"moment\");","module.exports = require(\"moment-range\");","module.exports = require(\"mustache\");","module.exports = require(\"nodemailer\");","module.exports = require(\"objection\");","module.exports = require(\"path\");"],"mappings":";AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AClFA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;;;;;;;;;;ACJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AChDA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;ACxEA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC3EA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACXA;AACA;AACA;AACA;;;;;;;;;;;;ACJA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACVA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC/IA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC7BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAjFA;AAmFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACrNA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAIA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAnBA;AACA;AACA;AAqBA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AChYA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AChMA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AClPA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AA7EA;AA+EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA;;;;;;;;;;;;AC5GA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACZA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAIA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAhDA;AAkDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AA7DA;AACA;AA+DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAvFA;AAyFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACtfA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACtOA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACjgBA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACpLA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAnCA;AAqCA;AACA;AACA;AAvCA;AAyCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC9QA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACtEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzFA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAhIA;AACA;AACA;AACA;AAkIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACvVA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC/OA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AAOA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACvQA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACpDA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACfA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AADA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AA3BA;AA+BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AA9EA;AAmFA;AACA;AACA;AACA;AACA;;;;AAvFA;AA4FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAtGA;AAwGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAlHA;AAoHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAjIA;AAmIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC3KA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC7JA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;AA0BA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AApEA;AAwEA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACjPA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACxFA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAzEA;;AAOA;;AAgBA;;AASA;;AAYA;;AAYA;;;;;;;;;;;;AClEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC3BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC/EA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC/BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AApDA;;AAWA;;;;;;;;;;;;ACjBA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC5FA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACzCA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC7BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzBA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAfA;AAkBA;AACA;;;;;;;;;;;;AC3BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AApBA;AAsBA;;;;;;;;;;;;;ACvCA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACpEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;AA3CA;;AAGA;;AAWA;;AAWA;;AAOA;;AAmBA;AAKA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AClEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACrCA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzFA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AA5CA;;AAOA;;AAOA;;;;;;;;;;;;ACtBA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzDA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;;;;;;;;;;ACfA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AChEA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACNA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC/OA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACxJA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACLA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACbA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;ACnEA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;A","sourceRoot":""} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"bundle.js","sources":["webpack/bootstrap","/Users/ahmed/Documents/Ratteb/server/config/index.js","/Users/ahmed/Documents/Ratteb/server/knexfile.js","/Users/ahmed/Documents/Ratteb/server/src/app.js","/Users/ahmed/Documents/Ratteb/server/src/collection/BudgetEntriesSet.js","/Users/ahmed/Documents/Ratteb/server/src/collection/NestedSet/index.js","/Users/ahmed/Documents/Ratteb/server/src/collection/ResourceFieldMetadataCollection.js","/Users/ahmed/Documents/Ratteb/server/src/data/ResourceFieldsKeys.js","/Users/ahmed/Documents/Ratteb/server/src/database/knex.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/AccountOpeningBalance.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/AccountTypes.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Accounting.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Accounts.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Authentication.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Bills.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Budget.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/BudgetReports.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Currencies.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/CurrencyAdjustment.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Customers.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Expenses.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Fields.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/FinancialStatements.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/ItemCategories.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Items.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Options.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Resources.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Roles.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Suppliers.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Users.js","/Users/ahmed/Documents/Ratteb/server/src/http/controllers/Views.js","/Users/ahmed/Documents/Ratteb/server/src/http/index.js","/Users/ahmed/Documents/Ratteb/server/src/http/middleware/asyncMiddleware.js","/Users/ahmed/Documents/Ratteb/server/src/http/middleware/authorization.js","/Users/ahmed/Documents/Ratteb/server/src/http/middleware/jwtAuth.js","/Users/ahmed/Documents/Ratteb/server/src/lib/LogicEvaluation/Lexer.js","/Users/ahmed/Documents/Ratteb/server/src/lib/LogicEvaluation/Parser.js","/Users/ahmed/Documents/Ratteb/server/src/lib/LogicEvaluation/QueryParser.js","/Users/ahmed/Documents/Ratteb/server/src/lib/Metable/MetableCollection.js","/Users/ahmed/Documents/Ratteb/server/src/lib/ViewRolesBuilder/index.js","/Users/ahmed/Documents/Ratteb/server/src/models/Account.js","/Users/ahmed/Documents/Ratteb/server/src/models/AccountBalance.js","/Users/ahmed/Documents/Ratteb/server/src/models/AccountTransaction.js","/Users/ahmed/Documents/Ratteb/server/src/models/AccountType.js","/Users/ahmed/Documents/Ratteb/server/src/models/Budget.js","/Users/ahmed/Documents/Ratteb/server/src/models/BudgetEntry.js","/Users/ahmed/Documents/Ratteb/server/src/models/Expense.js","/Users/ahmed/Documents/Ratteb/server/src/models/Item.js","/Users/ahmed/Documents/Ratteb/server/src/models/ItemCategory.js","/Users/ahmed/Documents/Ratteb/server/src/models/JournalEntry.js","/Users/ahmed/Documents/Ratteb/server/src/models/ManualJournal.js","/Users/ahmed/Documents/Ratteb/server/src/models/Model.js","/Users/ahmed/Documents/Ratteb/server/src/models/Option.js","/Users/ahmed/Documents/Ratteb/server/src/models/PasswordReset.js","/Users/ahmed/Documents/Ratteb/server/src/models/Permission.js","/Users/ahmed/Documents/Ratteb/server/src/models/Resource.js","/Users/ahmed/Documents/Ratteb/server/src/models/ResourceField.js","/Users/ahmed/Documents/Ratteb/server/src/models/ResourceFieldMetadata.js","/Users/ahmed/Documents/Ratteb/server/src/models/Role.js","/Users/ahmed/Documents/Ratteb/server/src/models/User.js","/Users/ahmed/Documents/Ratteb/server/src/models/View.js","/Users/ahmed/Documents/Ratteb/server/src/models/ViewColumn.js","/Users/ahmed/Documents/Ratteb/server/src/models/ViewRole.js","/Users/ahmed/Documents/Ratteb/server/src/models/index.js","/Users/ahmed/Documents/Ratteb/server/src/server.js","/Users/ahmed/Documents/Ratteb/server/src/services/Accounting/JournalEntry.js","/Users/ahmed/Documents/Ratteb/server/src/services/Accounting/JournalPoster.js","/Users/ahmed/Documents/Ratteb/server/src/services/CustomFields/ResourceCustomFieldRepository.js","/Users/ahmed/Documents/Ratteb/server/src/services/Moment/index.js","/Users/ahmed/Documents/Ratteb/server/src/services/mail.js","/Users/ahmed/Documents/Ratteb/server/src/utils/index.js","external \"@babel/plugin-transform-runtime\"","external \"@babel/runtime/helpers/asyncToGenerator\"","external \"@babel/runtime/helpers/classCallCheck\"","external \"@babel/runtime/helpers/createClass\"","external \"@babel/runtime/helpers/defineProperty\"","external \"@babel/runtime/helpers/get\"","external \"@babel/runtime/helpers/getPrototypeOf\"","external \"@babel/runtime/helpers/inherits\"","external \"@babel/runtime/helpers/possibleConstructorReturn\"","external \"@babel/runtime/helpers/slicedToArray\"","external \"@babel/runtime/helpers/toConsumableArray\"","external \"@babel/runtime/regenerator\"","external \"bcryptjs\"","external \"dotenv\"","external \"errorhandler\"","external \"express\"","external \"express-boom\"","external \"express-validator\"","external \"fs\"","external \"helmet\"","external \"i18n\"","external \"jsonwebtoken\"","external \"knex\"","external \"lodash\"","external \"moment\"","external \"moment-range\"","external \"mustache\"","external \"nodemailer\"","external \"objection\"","external \"path\""],"sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"dist/\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 0);\n","import path from 'path';\nimport dotenv from 'dotenv';\n\ndotenv.config({\n  path: path.resolve(process.cwd(), '.env.test'),\n});\n","require('dotenv').config();\n\nconst MIGRATIONS_DIR = './src/database/migrations';\nconst SEEDS_DIR = './src/database/seeds';\n\nmodule.exports = {\n  test: {\n    client: process.env.DB_CLIENT,\n    migrations: {\n      directory: MIGRATIONS_DIR,\n    },\n    connection: {\n      host: process.env.DB_HOST,\n      user: process.env.DB_USER,\n      password: process.env.DB_PASSWORD,\n      database: process.env.DB_NAME,\n      charset: 'utf8',\n    },\n  },\n  development: {\n    client: process.env.DB_CLIENT,\n    connection: {\n      host: process.env.DB_HOST,\n      user: process.env.DB_USER,\n      password: process.env.DB_PASSWORD,\n      database: process.env.DB_NAME,\n      charset: 'utf8',\n    },\n    migrations: {\n      directory: MIGRATIONS_DIR,\n    },\n    seeds: {\n      directory: SEEDS_DIR,\n    },\n  },\n  production: {\n    client: process.env.DB_CLIENT,\n    connection: {\n      host: process.env.DB_HOST,\n      user: process.env.DB_USER,\n      password: process.env.DB_PASSWORD,\n      database: process.env.DB_NAME,\n      charset: 'utf8',\n    },\n    migrations: {\n      directory: MIGRATIONS_DIR,\n    },\n    seeds: {\n      directory: SEEDS_DIR,\n    },\n  },\n};\n","import express from 'express';\nimport helmet from 'helmet';\nimport boom from 'express-boom';\nimport i18n from 'i18n';\nimport '../config';\nimport routes from '@/http';\nimport '@/models';\n\nconst app = express();\n\n// i18n.configure({\n//   // setup some locales - other locales default to en silently\n//   locales: ['en'],\n\n//   // sets a custom cookie name to parse locale settings from.\n//   cookie: 'yourcookiename',\n\n//   // where to store json files - defaults to './locales'\n//   directory: `${__dirname}/resources/locale`,\n// });\n\n// i18n init parses req for language headers, cookies, etc.\n// app.use(i18n.init);\n\n// Express configuration\napp.set('port', process.env.PORT || 3000);\n\napp.use(helmet());\napp.use(boom());\napp.use(express.json());\n\nroutes(app);\n\nexport default app;\n","\n\nexport default class BudgetEntriesSet {\n\n  constructor() {\n    this.accounts = {}; \n    this.totalSummary = {}\n    this.orderSize = null;\n  }\n\n  setZeroPlaceholder() {\n    if (!this.orderSize) { return; }\n\n    Object.values(this.accounts).forEach((account) => {\n\n      for (let i = 0; i <= this.orderSize.length; i++) {\n        if (typeof account[i] === 'undefined') {\n          account[i] = { amount: 0 };\n        }\n      }\n    });\n  }\n\n  static from(accounts, configs) {\n    const collection = new this(configs);\n\n    accounts.forEach((entry) => {\n      if (typeof this.accounts[entry.accountId] === 'undefined') {\n        collection.accounts[entry.accountId] = {};\n      }\n      if (entry.order) {\n        collection.accounts[entry.accountId][entry.order] = entry;\n      }\n    });\n    return collection;\n  }\n\n  toArray() {\n    const output = [];\n\n    Object.key(this.accounts).forEach((accountId) => {\n      const entries = this.accounts[accountId];\n      output.push({\n        account_id: accountId,\n        entries: [\n          ...Object.key(entries).map((order) => {\n            const entry = entries[order];\n            return {\n              order,\n              amount: entry.amount,\n            };\n          }),\n        ],\n      });\n    });\n  }\n\n  calcTotalSummary() {\n    const totalSummary = {};\n\n    for (let i = 0; i < this.orderSize.length; i++) {\n      Object.value(this.accounts).forEach((account) => {\n        if (typeof totalSummary[i] !== 'undefined') {\n          totalSummary[i] = { amount: 0, order: i };\n        }\n        totalSummary[i].amount += account[i].amount;\n      });\n    }\n    this.totalSummary = totalSummary;\n  }\n\n  toArrayTotalSummary() {\n    return Object.values(this.totalSummary);\n  }\n\n}\n","\nexport default class NestedSet {\n  /**\n   * Constructor method.\n   * @param {Object} options -\n   */\n  constructor(items, options) {\n    this.options = {\n      parentId: 'parent_id',\n      id: 'id',\n      ...options,\n    };\n    this.items = items;\n    this.collection = {};\n  }\n\n  /**\n   * Link nodes children.\n   */\n  linkChildren() {\n    if (this.items.length <= 0) return false;\n\n    const map = {};\n    this.items.forEach((item) => {\n      map[item.id] = item;\n      map[item.id].children = [];\n    });\n\n    this.items.forEach((item) => {\n      const parentNodeId = item[this.options.parentId];\n      if (parentNodeId) {\n        map[parentNodeId].children.push(item);\n      }\n    });\n    return map;\n  }\n\n  toTree() {\n    const map = this.linkChildren();\n    const tree = {};\n\n    this.items.forEach((item) => {\n      const parentNodeId = item[this.options.parentId];\n      if (!parentNodeId) {\n        tree[item.id] = map[item.id];\n      }\n    });\n    this.collection = Object.values(tree);\n    return this.collection;\n  }\n\n  getTree() {\n    return this.collection;\n  }\n\n  flattenTree(nodeMapper) {\n    const flattenTree = [];\n\n    const traversal = (nodes, parentNode) => {\n      nodes.forEach((node) => {\n        let nodeMapped = node;\n\n        if (typeof nodeMapper === 'function') {\n          nodeMapped = nodeMapper(nodeMapped, parentNode);\n        }\n        flattenTree.push(nodeMapped);\n\n        if (node.children && node.children.length > 0) {\n          traversal(node.children, node);\n        }\n      });\n    };\n    traversal(this.collection);\n\n    return flattenTree;\n  }\n}\n","import MetableCollection from '@/lib/Metable/MetableCollection';\nimport ResourceFieldMetadata from '@/models/ResourceFieldMetadata';\n\nexport default class ResourceFieldMetadataCollection extends MetableCollection {\n  /**\n   * Constructor method.\n   */\n  constructor() {\n    super();\n\n    this.setModel(ResourceFieldMetadata);\n    this.extraColumns = ['resource_id', 'resource_item_id'];\n  }\n}\n","\nexport default {\n  \"expense_account\": 'expense_account_id',\n  \"payment_account\": 'payment_account_id',\n  \"account_type\": \"account_type_id\"\n}","import Knex from 'knex';\nimport { knexSnakeCaseMappers } from 'objection';\nimport knexfile from '@/../knexfile';\n\nconst config = knexfile[process.env.NODE_ENV];\nconst knex = Knex({\n  ...config,\n  ...knexSnakeCaseMappers({ upperCase: true }),\n});\n\nexport default knex;\n","import express from 'express';\nimport { check, validationResult, oneOf } from 'express-validator';\nimport { difference } from 'lodash';\nimport moment from 'moment';\nimport asyncMiddleware from '../middleware/asyncMiddleware';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport Account from '@/models/Account';\nimport JournalPoster from '@/services/Accounting/JournalPoster';\nimport JournalEntry from '@/services/Accounting/JournalEntry';\nimport ManualJournal from '@/models/ManualJournal';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(jwtAuth);\n\n    router.post('/',\n      this.openingBalnace.validation,\n      asyncMiddleware(this.openingBalnace.handler));\n\n    return router;\n  },\n\n  /**\n   * Opening balance to the given account.\n   * @param {Request} req -\n   * @param {Response} res -\n   */\n  openingBalnace: {\n    validation: [\n      check('date').optional(),\n      check('note').optional().trim().escape(),\n      check('balance_adjustment_account').exists().isNumeric().toInt(),\n      check('accounts').isArray({ min: 1 }),\n      check('accounts.*.id').exists().isInt(),\n      oneOf([\n        check('accounts.*.debit').exists().isNumeric().toFloat(),\n        check('accounts.*.credit').exists().isNumeric().toFloat(),\n      ]),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { accounts } = req.body;\n      const { user } = req;\n      const form = { ...req.body };\n      const date = moment(form.date).format('YYYY-MM-DD');\n\n      const accountsIds = accounts.map((account) => account.id);\n      const storedAccounts = await Account.query()\n        .select(['id']).whereIn('id', accountsIds)\n        .withGraphFetched('type');\n\n      const accountsCollection = new Map(storedAccounts.map(i => [i.id, i]));\n\n      // Get the stored accounts Ids and difference with submit accounts.\n      const accountsStoredIds = storedAccounts.map((account) => account.id);\n      const notFoundAccountsIds = difference(accountsIds, accountsStoredIds);\n      const errorReasons = [];\n\n      if (notFoundAccountsIds.length > 0) {\n        const ids = notFoundAccountsIds.map((a) => parseInt(a, 10));\n        errorReasons.push({ type: 'NOT_FOUND_ACCOUNT', code: 100, ids });\n      }\n      if (form.balance_adjustment_account) {\n        const account = await Account.query().findById(form.balance_adjustment_account);\n\n        if (!account) {\n          errorReasons.push({ type: 'BALANCE.ADJUSTMENT.ACCOUNT.NOT.EXIST', code: 300 });\n        }\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badData(null, { errors: errorReasons });\n      }\n\n      const journalEntries = new JournalPoster();\n\n      accounts.forEach((account) => {\n        const storedAccount = accountsCollection.get(account.id);\n\n        // Can't continue in case the stored account was not found.\n        if (!storedAccount) { return; }\n\n        const entryModel = new JournalEntry({\n          referenceType: 'OpeningBalance',\n          account: account.id,\n          accountNormal: storedAccount.type.normal,\n          userId: user.id,\n        });\n        if (account.credit) {\n          entryModel.entry.credit = account.credit;\n          journalEntries.credit(entryModel);\n        } else if (account.debit) {\n          entryModel.entry.debit = account.debit;\n          journalEntries.debit(entryModel);\n        }\n      });\n      // Calculates the credit and debit balance of stacked entries.\n      const trial = journalEntries.getTrialBalance();\n\n      if (trial.credit !== trial.debit) {\n        const entryModel = new JournalEntry({\n          referenceType: 'OpeningBalance',\n          account: form.balance_adjustment_account,\n          accountNormal: 'credit',\n          userId: user.id,\n        });\n\n        if (trial.credit > trial.debit) {\n          entryModel.entry.credit = Math.abs(trial.credit);\n          journalEntries.credit(entryModel);\n\n        } else if (trial.credit < trial.debit) {\n          entryModel.entry.debit = Math.abs(trial.debit);\n          journalEntries.debit(entryModel);\n        }\n      }\n      const manualJournal = await ManualJournal.query().insert({\n        amount: Math.max(trial.credit, trial.debit),\n        transaction_type: 'OpeningBalance',\n        date,\n        note: form.note,\n        user_id: user.id,\n      });\n\n      journalEntries.entries = journalEntries.entries.map((entry) => ({\n        ...entry,\n        referenceId: manualJournal.id,\n      }));\n      await Promise.all([\n        journalEntries.saveEntries(),\n        journalEntries.saveBalance(),\n      ]);\n      return res.status(200).send({ id: manualJournal.id });\n    },\n  },\n};\n","import express from 'express';\nimport JWTAuth from '@/http/middleware/jwtAuth';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport AccountType from '@/models/AccountType';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n    router.use(JWTAuth);\n\n    router.get('/',\n      this.getAccountTypesList.validation,\n      asyncMiddleware(this.getAccountTypesList.handler));\n\n    return router;\n  },\n\n  /**\n   * Retrieve accounts types list.\n   */\n  getAccountTypesList: {\n    validation: [],\n    async handler(req, res) {\n      const accountTypes = await AccountType.query();\n\n      return res.status(200).send({\n        account_types: accountTypes,\n      });\n    },\n  },\n};\n","import { check, query, oneOf, validationResult } from 'express-validator';\nimport express from 'express';\nimport { difference } from 'lodash';\nimport moment from 'moment';\nimport Account from '@/models/Account';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport JWTAuth from '@/http/middleware/jwtAuth';\nimport JournalPoster from '@/services/Accounting/JournalPoster';\nimport JournalEntry from '@/services/Accounting/JournalEntry';\nimport ManualJournal from '@/models/JournalEntry';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n    router.use(JWTAuth);\n\n    router.post('/make-journal-entries',\n      this.makeJournalEntries.validation,\n      asyncMiddleware(this.makeJournalEntries.handler));\n\n    router.post('/recurring-journal-entries',\n      this.recurringJournalEntries.validation,\n      asyncMiddleware(this.recurringJournalEntries.handler));\n\n    router.post('quick-journal-entries',\n      this.quickJournalEntries.validation,\n      asyncMiddleware(this.quickJournalEntries.handler));\n\n    return router;\n  },\n\n  /**\n   * Make journal entrires.\n   */\n  makeJournalEntries: {\n    validation: [\n      check('date').isISO8601(),\n      check('reference').exists(),\n      check('memo').optional().trim().escape(),\n      check('entries').isArray({ min: 1 }),\n      check('entries.*.credit').optional({ nullable: true }).isNumeric().toInt(),\n      check('entries.*.debit').optional({ nullable: true }).isNumeric().toInt(),\n      check('entries.*.account_id').isNumeric().toInt(),\n      check('entries.*.note').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = {\n        date: new Date(),\n        ...req.body,\n      };\n\n      let totalCredit = 0;\n      let totalDebit = 0;\n\n      const { user } = req;\n      const errorReasons = [];\n      const entries = form.entries.filter((entry) => (entry.credit || entry.debit));\n      const formattedDate = moment(form.date).format('YYYY-MM-DD');\n\n      entries.forEach((entry) => {\n        if (entry.credit > 0) {\n          totalCredit += entry.credit;\n        }\n        if (entry.debit > 0) {\n          totalDebit += entry.debit;\n        }\n      });\n\n      if (totalCredit <= 0 || totalDebit <= 0) {\n        errorReasons.push({\n          type: 'CREDIT.DEBIT.SUMATION.SHOULD.NOT.EQUAL.ZERO',\n          code: 400,\n        });\n      }\n      if (totalCredit !== totalDebit) {\n        errorReasons.push({ type: 'CREDIT.DEBIT.NOT.EQUALS', code: 100 });\n      }\n      const accountsIds = entries.map((entry) => entry.account_id);\n      const accounts = await Account.query().whereIn('id', accountsIds)\n        .withGraphFetched('type');\n\n      const storedAccountsIds = accounts.map((account) => account.id);\n\n      if (difference(accountsIds, storedAccountsIds).length > 0) {\n        errorReasons.push({ type: 'ACCOUNTS.IDS.NOT.FOUND', code: 200 });\n      }\n\n      const journalReference = await ManualJournal.query().where('reference', form.reference);\n\n      if (journalReference.length > 0) {\n        errorReasons.push({ type: 'REFERENCE.ALREADY.EXISTS', code: 300 });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n\n      // Save manual journal transaction.\n      const manualJournal = await ManualJournal.query().insert({\n        reference: form.reference,\n        transaction_type: 'Journal',\n        amount: totalCredit,\n        date: formattedDate,\n        note: form.memo,\n        user_id: user.id,\n      });\n      const journalPoster = new JournalPoster();\n\n      entries.forEach((entry) => {\n        const account = accounts.find((a) => a.id === entry.account_id);\n\n        const jouranlEntry = new JournalEntry({\n          debit: entry.debit,\n          credit: entry.credit,\n          account: account.id,\n          transactionType: 'Journal',\n          accountNormal: account.type.normal,\n          note: entry.note,\n          date: formattedDate,\n          userId: user.id,\n        });\n        if (entry.debit) {\n          journalPoster.debit(jouranlEntry);\n        } else {\n          journalPoster.credit(jouranlEntry);\n        }\n      });\n\n      // Saves the journal entries and accounts balance changes.\n      await Promise.all([\n        journalPoster.saveEntries(),\n        journalPoster.saveBalance(),\n      ]);\n      return res.status(200).send({ id: manualJournal.id });\n    },\n  },\n\n  /**\n   * Saves recurring journal entries template.\n   */\n  recurringJournalEntries: {\n    validation: [\n      check('template_name').exists(),\n      check('recurrence').exists(),\n      check('active').optional().isBoolean().toBoolean(),\n      check('entries').isArray({ min: 1 }),\n      check('entries.*.credit').isNumeric().toInt(),\n      check('entries.*.debit').isNumeric().toInt(),\n      check('entries.*.account_id').isNumeric().toInt(),\n      check('entries.*.note').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n    },\n  },\n\n  recurringJournalsList: {\n    validation: [\n      query('page').optional().isNumeric().toInt(),\n      query('page_size').optional().isNumeric().toInt(),\n      query('template_name').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n    },\n  },\n\n  quickJournalEntries: {\n    validation: [\n      check('date').exists().isISO8601(),\n      check('amount').exists().isNumeric().toFloat(),\n      check('credit_account_id').exists().isNumeric().toInt(),\n      check('debit_account_id').exists().isNumeric().toInt(),\n      check('transaction_type').exists(),\n      check('note').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const errorReasons = [];\n      const form = { ...req.body };\n\n      const foundAccounts = await Account.query()\n        .where('id', form.credit_account_id)\n        .orWhere('id', form.debit_account_id);\n\n      const creditAccount = foundAccounts.find((a) => a.id === form.credit_account_id);\n      const debitAccount = foundAccounts.find((a) => a.id === form.debit_account_id);\n\n      if (!creditAccount) {\n        errorReasons.push({ type: 'CREDIT_ACCOUNT.NOT.EXIST', code: 100 });\n      }\n      if (!debitAccount) {\n        errorReasons.push({ type: 'DEBIT_ACCOUNT.NOT.EXIST', code: 200 });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n\n      // const journalPoster = new JournalPoster();\n      // const journalCredit = new JournalEntry({\n      //   debit: \n      //   account: debitAccount.id,\n      //   referenceId: \n      // })\n\n      return res.status(200).send();\n    },\n  },\n};\n","import express from 'express';\nimport { check, validationResult, param, query } from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport Account from '@/models/Account';\nimport AccountType from '@/models/AccountType';\nimport AccountTransaction from '@/models/AccountTransaction';\nimport JournalPoster from '@/services/Accounting/JournalPoster';\nimport AccountBalance from '@/models/AccountBalance';\nimport Resource from '@/models/Resource';\nimport View from '@/models/View';\nimport JWTAuth from '@/http/middleware/jwtAuth';\nimport NestedSet from '../../collection/NestedSet';\nimport {\n  mapViewRolesToConditionals,\n  validateViewRoles,\n} from '@/lib/ViewRolesBuilder';\n\nexport default {\n  /**\n   * Router constructor method.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(JWTAuth);\n    router.post('/',\n      this.newAccount.validation,\n      asyncMiddleware(this.newAccount.handler));\n\n    router.post('/:id',\n      this.editAccount.validation,\n      asyncMiddleware(this.editAccount.handler));\n\n    router.get('/:id',\n      this.getAccount.validation,\n      asyncMiddleware(this.getAccount.handler));\n\n    router.get('/',\n      this.getAccountsList.validation,\n      asyncMiddleware(this.getAccountsList.handler));\n\n    router.delete('/:id',\n      this.deleteAccount.validation,\n      asyncMiddleware(this.deleteAccount.handler));\n\n    router.post('/:id/active',\n      this.activeAccount.validation,\n      asyncMiddleware(this.activeAccount.handler));\n\n    router.post('/:id/inactive',\n      this.inactiveAccount.validation,\n      asyncMiddleware(this.inactiveAccount.handler));\n\n    router.post('/:id/recalculate-balance',\n      this.recalcualteBalanace.validation,\n      asyncMiddleware(this.recalcualteBalanace.handler));\n\n    router.post('/:id/transfer_account/:toAccount',\n      this.transferToAnotherAccount.validation,\n      asyncMiddleware(this.transferToAnotherAccount.handler));\n\n    return router;\n  },\n\n  /**\n   * Creates a new account.\n   */\n  newAccount: {\n    validation: [\n      check('name').exists().isLength({ min: 3 }).trim().escape(),\n      check('code').exists().isLength({ max: 10 }).trim().escape(),\n      check('account_type_id').exists().isNumeric().toInt(),\n      check('description').optional().trim().escape(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = { ...req.body };\n\n      const foundAccountCodePromise = form.code\n        ? Account.query().where('code', form.code) : null;\n\n      const foundAccountTypePromise = AccountType.query()\n        .findById(form.account_type_id);\n\n      const [foundAccountCode, foundAccountType] = await Promise.all([\n        foundAccountCodePromise, foundAccountTypePromise,\n      ]);\n\n      if (foundAccountCodePromise && foundAccountCode.length > 0) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'NOT_UNIQUE_CODE', code: 100 }],\n        });\n      }\n      if (!foundAccountType) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'NOT_EXIST_ACCOUNT_TYPE', code: 200 }],\n        });\n      }\n      await Account.query().insert({ ...form });\n\n      return res.status(200).send({ item: { } });\n    },\n  },\n\n  /**\n   * Edit the given account details.\n   */\n  editAccount: {\n    validation: [\n      param('id').exists().toInt(),\n      check('name').exists().isLength({ min: 3 }).trim().escape(),\n      check('code').exists().isLength({ max: 10 }).trim().escape(),\n      check('account_type_id').exists().isNumeric().toInt(),\n      check('description').optional().trim().escape(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = { ...req.body };\n      const account = await Account.query().findById(id);\n\n      if (!account) {\n        return res.boom.notFound();\n      }\n      const foundAccountCodePromise = (form.code && form.code !== account.code)\n        ? Account.query().where('code', form.code).whereNot('id', account.id) : null;\n\n      const foundAccountTypePromise = (form.account_type_id !== account.account_type_id)\n        ? AccountType.query().where('id', form.account_type_id) : null;\n\n      const [foundAccountCode, foundAccountType] = await Promise.all([\n        foundAccountCodePromise, foundAccountTypePromise,\n      ]);\n      if (foundAccountCode.length > 0 && foundAccountCodePromise) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'NOT_UNIQUE_CODE', code: 100 }],\n        });\n      }\n      if (foundAccountType.length <= 0 && foundAccountTypePromise) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'NOT_EXIST_ACCOUNT_TYPE', code: 110 }],\n        });\n      }\n      await account.patch({ ...form });\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Get details of the given account.\n   */\n  getAccount: {\n    validation: [\n      param('id').toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const account = await Account.query().where('id', id).first();\n\n      if (!account) {\n        return res.boom.notFound();\n      }\n      return res.status(200).send({ account: { ...account } });\n    },\n  },\n\n  /**\n   * Delete the given account.\n   */\n  deleteAccount: {\n    validation: [\n      param('id').toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const account = await Account.query().findById(id);\n\n      if (!account) {\n        return res.boom.notFound();\n      }\n      const accountTransactions = await AccountTransaction.query()\n        .where('account_id', account.id);\n\n      if (accountTransactions.length > 0) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'ACCOUNT.HAS.ASSOCIATED.TRANSACTIONS', code: 100 }],\n        });\n      }\n      await Account.query().deleteById(account.id);\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrieve accounts list.\n   */\n  getAccountsList: {\n    validation: [\n      query('display_type').optional().isIn(['tree', 'flat']),\n      query('account_types').optional().isArray(),\n      query('account_types.*').optional().isNumeric().toInt(),\n      query('custom_view_id').optional().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const filter = {\n        account_types: [],\n        display_type: 'tree',\n        ...req.query,\n      };\n      const errorReasons = [];\n      const viewConditionals = [];\n      const accountsResource = await Resource.query().where('name', 'accounts').first();\n\n      if (!accountsResource) {\n        return res.status(400).send({\n          errors: [{ type: 'ACCOUNTS_RESOURCE_NOT_FOUND', code: 200 }],\n        });\n      }\n      const view = await View.query().onBuild((builder) => {\n        if (filter.custom_view_id) {\n          builder.where('id', filter.custom_view_id);\n        } else {\n          builder.where('favourite', true);\n        }\n        builder.where('resource_id', accountsResource.id);\n        builder.withGraphFetched('roles.field');\n        builder.withGraphFetched('columns');\n        builder.first();\n      });\n\n      if (view && view.roles.length > 0) {\n        viewConditionals.push(\n          ...mapViewRolesToConditionals(view.roles),\n        );\n        if (!validateViewRoles(viewConditionals, view.rolesLogicExpression)) {\n          errorReasons.push({ type: 'VIEW.LOGIC.EXPRESSION.INVALID', code: 400 });\n        }\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      const accounts = await Account.query().onBuild((builder) => {\n        builder.modify('filterAccountTypes', filter.account_types);\n        builder.withGraphFetched('type');\n\n        if (viewConditionals.length) {\n          builder.modify('viewRolesBuilder', viewConditionals, view.rolesLogicExpression);\n        }\n      });\n\n      const nestedAccounts = new NestedSet(accounts, { parentId: 'parentAccountId' });\n      const groupsAccounts = nestedAccounts.toTree();\n      const accountsList = [];\n\n      if (filter.display_type === 'tree') {\n        accountsList.push(...groupsAccounts);\n      } else if (filter.display_type === 'flat') {\n        const flattenAccounts = nestedAccounts.flattenTree((account, parentAccount) => {\n          if (parentAccount) {\n            account.name = `${parentAccount.name} ― ${account.name}`;\n          }\n          return account;\n        });\n        accountsList.push(...flattenAccounts);\n      }\n      return res.status(200).send({\n        accounts: accountsList,\n        ...(view) ? {\n          customViewId: view.id,\n        } : {},\n      });\n    },\n  },\n\n  /**\n   * Re-calculates balance of the given account.\n   */\n  recalcualteBalanace: {\n    validation: [\n      param('id').isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const account = await Account.findById(id);\n\n      if (!account) {\n        return res.status(400).send({\n          errors: [{ type: 'ACCOUNT.NOT.FOUND', code: 100 }],\n        });\n      }\n      const accountTransactions = AccountTransaction.query()\n        .where('account_id', account.id);\n\n      const journalEntries = new JournalPoster();\n      journalEntries.loadFromCollection(accountTransactions);\n\n      // Delete the balance of the given account id.\n      await AccountBalance.query().where('account_id', account.id).delete();\n\n      // Save calcualted account balance.\n      await journalEntries.saveBalance();\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Active the given account.\n   */\n  activeAccount: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const account = await Account.findById(id);\n\n      if (!account) {\n        return res.status(400).send({\n          errors: [{ type: 'ACCOUNT.NOT.FOUND', code: 100 }],\n        });\n      }\n      await account.patch({ active: true });\n\n      return res.status(200).send({ id: account.id });\n    },\n  },\n\n  /**\n   * Inactive the given account.\n   */\n  inactiveAccount: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const account = await Account.findById(id);\n\n      if (!account) {\n        return res.status(400).send({\n          errors: [{ type: 'ACCOUNT.NOT.FOUND', code: 100 }],\n        });\n      }\n      await account.patch({ active: false });\n\n      return res.status(200).send({ id: account.id });\n    },\n  },\n\n  /**\n   * Transfer all journal entries of the given account to another account.\n   */\n  transferToAnotherAccount: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n      param('toAccount').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      // const { id, toAccount: toAccountId } = req.params;\n\n      // const [fromAccount, toAccount] = await Promise.all([\n      //   Account.query().findById(id),\n      //   Account.query().findById(toAccountId),\n      // ]);\n\n      // const fromAccountTransactions = await AccountTransaction.query()\n      //   .where('account_id', fromAccount);\n\n      // return res.status(200).send();\n    },\n  },\n};\n","\nimport express from 'express';\nimport { check, validationResult } from 'express-validator';\nimport path from 'path';\nimport fs from 'fs';\nimport Mustache from 'mustache';\nimport jwt from 'jsonwebtoken';\nimport asyncMiddleware from '../middleware/asyncMiddleware';\nimport User from '@/models/User';\nimport PasswordReset from '@/models/PasswordReset';\nimport mail from '@/services/mail';\nimport { hashPassword } from '@/utils';\n\nexport default {\n  /**\n   * Constructor method.\n   */\n  router() {\n    const router = express.Router();\n\n    router.post('/login',\n      this.login.validation,\n      asyncMiddleware(this.login.handler));\n\n    router.post('/send_reset_password',\n      this.sendResetPassword.validation,\n      asyncMiddleware(this.sendResetPassword.handler));\n\n    router.post('/reset/:token',\n      this.resetPassword.validation,\n      asyncMiddleware(this.resetPassword.handler));\n\n    return router;\n  },\n\n  /**\n   * User login authentication request.\n   */\n  login: {\n    validation: [\n      check('crediential').exists().isEmail(),\n      check('password').exists().isLength({ min: 4 }),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { crediential, password } = req.body;\n      const { JWT_SECRET_KEY } = process.env;\n\n      const user = await User.query()\n        .where('email', crediential)\n        .orWhere('phone_number', crediential)\n        .first();\n\n      if (!user) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'INVALID_DETAILS', code: 100 }],\n        });\n      }\n      if (!user.verifyPassword(password)) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'INVALID_DETAILS', code: 100 }],\n        });\n      }\n      if (!user.active) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'USER_INACTIVE', code: 110 }],\n        });\n      }\n      // user.update({ last_login_at: new Date() });\n\n      const token = jwt.sign({\n        email: user.email,\n        _id: user.id,\n      }, JWT_SECRET_KEY, {\n        expiresIn: '1d',\n      });\n      return res.status(200).send({ token, user });\n    },\n  },\n\n  /**\n   * Send reset password link via email or SMS.\n   */\n  sendResetPassword: {\n    validation: [\n      check('email').exists().isEmail(),\n    ],\n    // eslint-disable-next-line consistent-return\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { email } = req.body;\n      const user = await User.where('email', email).fetch();\n\n      if (!user) {\n        return res.status(422).send();\n      }\n      // Delete all stored tokens of reset password that associate to the give email.\n      await PasswordReset.where({ email }).destroy({ require: false });\n\n      const passwordReset = PasswordReset.forge({\n        email,\n        token: '123123',\n      });\n      await passwordReset.save();\n\n      const filePath = path.join(__dirname, '../../views/mail/ResetPassword.html');\n      const template = fs.readFileSync(filePath, 'utf8');\n      const rendered = Mustache.render(template, {\n        url: `${req.protocol}://${req.hostname}/reset/${passwordReset.attributes.token}`,\n        first_name: user.attributes.first_name,\n        last_name: user.attributes.last_name,\n        contact_us_email: process.env.CONTACT_US_EMAIL,\n      });\n\n      const mailOptions = {\n        to: user.attributes.email,\n        from: `${process.env.MAIL_FROM_NAME} ${process.env.MAIL_FROM_ADDRESS}`,\n        subject: 'Ratteb Password Reset',\n        html: rendered,\n      };\n\n      // eslint-disable-next-line consistent-return\n      mail.sendMail(mailOptions, (error) => {\n        if (error) {\n          return res.status(400).send();\n        }\n        res.status(200).send({ data: { email: passwordReset.attributes.email } });\n      });\n    },\n  },\n\n  /**\n   * Reset password.\n   */\n  resetPassword: {\n    validation: [\n      check('password').exists().isLength({ min: 5 }).custom((value, { req }) => {\n        if (value !== req.body.confirm_password) {\n          throw new Error(\"Passwords don't match\");\n        } else {\n          return value;\n        }\n      }),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'VALIDATION_ERROR', ...validationErrors,\n        });\n      }\n      const { token } = req.params;\n      const { password } = req.body;\n\n      const tokenModel = await PasswordReset.query()\n        .where('token', token)\n        .where('created_at', '>=', Date.now() - 3600000)\n        .first();\n\n      if (!tokenModel) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'TOKEN_INVALID', code: 100 }],\n        });\n      }\n      const user = await User.where({\n        email: tokenModel.email,\n      });\n      if (!user) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'USER_NOT_FOUND', code: 120 }],\n        });\n      }\n      const hashedPassword = await hashPassword(password);\n\n      user.password = hashedPassword;\n      await user.save();\n\n      await PasswordReset.where('email', user.get('email')).destroy({ require: false });\n\n      return res.status(200).send({});\n    },\n  },\n};\n","import express from 'express';\n\nexport default {\n\n  router() {\n    const router = express.Router();\n\n    return router;\n  },\n};\n","import express from 'express';\nimport {\n  check,\n  query,\n  param,\n  validationResult,\n} from 'express-validator';\nimport { pick, difference, groupBy } from 'lodash';\nimport asyncMiddleware from \"@/http/middleware/asyncMiddleware\";\nimport JWTAuth from '@/http/middleware/jwtAuth';\nimport Budget from '@/models/Budget';\nimport BudgetEntry from '@/models/BudgetEntry';\nimport Account from '@/models/Account';\nimport moment from '@/services/Moment';\nimport BudgetEntriesSet from '@/collection/BudgetEntriesSet';\nimport AccountType from '@/models/AccountType';\nimport NestedSet from '@/collection/NestedSet';\nimport { dateRangeFormat } from '@/utils';\n\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(JWTAuth);\n\n    router.post('/',\n      this.newBudget.validation,\n      asyncMiddleware(this.newBudget.handler));\n\n    router.get('/:id',\n      this.getBudget.validation,\n      asyncMiddleware(this.getBudget.handler));\n\n    router.get('/:id',\n      this.deleteBudget.validation,\n      asyncMiddleware(this.deleteBudget.handler));\n\n    router.get('/',\n      this.listBudgets.validation,\n      asyncMiddleware(this.listBudgets.handler));\n\n    return router;\n  },\n\n  /**\n   * Retrieve budget details of the given id.\n   */\n  getBudget: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { id } = req.params;\n      const budget = await Budget.query().findById(id);\n\n      if (!budget) {\n        return res.status(404).send({\n          errors: [{ type: 'budget.not.found', code: 100 }],\n        });\n      }\n      const accountTypes = await AccountType.query().where('balance_sheet', true);\n\n      const [budgetEntries, accounts] = await Promise.all([\n        BudgetEntry.query().where('budget_id', budget.id),\n        Account.query().whereIn('account_type_id', accountTypes.map((a) => a.id)),\n      ]);\n\n      const accountsNestedSet = new NestedSet(accounts);\n\n      const columns = [];\n      const fromDate = moment(budget.year).startOf('year')\n        .add(budget.rangeOffset, budget.rangeBy).toDate();\n\n      const toDate = moment(budget.year).endOf('year').toDate();\n\n      const dateRange = moment.range(fromDate, toDate);\n      const dateRangeCollection = Array.from(dateRange.by(budget.rangeBy, {\n        step: budget.rangeIncrement, excludeEnd: false, excludeStart: false,\n      }));\n\n      dateRangeCollection.forEach((date) => {\n        columns.push(date.format(dateRangeFormat(budget.rangeBy)));\n      });\n      const budgetEntriesSet = BudgetEntriesSet.from(budgetEntries, {\n        orderSize: columns.length,\n      });\n      budgetEntriesSet.setZeroPlaceholder();\n      budgetEntriesSet.calcTotalSummary();\n\n      return res.status(200).send({\n        columns,\n        accounts: budgetEntriesSet.toArray(),\n        total: budgetEntriesSet.toArrayTotalSummary(),\n      });\n    },\n  },\n\n  /**\n   * Delete the given budget.\n   */\n  deleteBudget: {\n    validation: [\n      param('id').exists(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { id } = req.params;\n      const budget = await Budget.query().findById(id);\n\n      if (!budget) {\n        return res.status(404).send({\n          errors: [{ type: 'budget.not.found', code: 100 }],\n        });\n      }\n      await BudgetEntry.query().where('budget_id', budget.id).delete();\n      await budget.delete();\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Saves the new budget.\n   */\n  newBudget: {\n    validation: [\n      check('name').exists(),\n      check('fiscal_year').exists(),\n      check('period').exists().isIn(['year', 'month', 'quarter', 'half-year']),\n      check('accounts_type').exists().isIn(['balance_sheet', 'profit_loss']),\n      check('accounts').isArray(),\n      check('accounts.*.account_id').exists().isNumeric().toInt(),\n      check('accounts.*.entries').exists().isArray(),\n      check('accounts.*.entries.*.amount').exists().isNumeric().toFloat(),\n      check('accounts.*.entries.*.order').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const form = { ...req.body };\n      const submitAccountsIds = form.accounts.map((a) => a.account_id);\n      const storedAccounts = await Account.query().whereIn('id', submitAccountsIds);\n      const storedAccountsIds = storedAccounts.map((a) => a.id);\n\n      const errorReasons = [];\n      const notFoundAccountsIds = difference(submitAccountsIds, storedAccountsIds);\n\n      if (notFoundAccountsIds.length > 0) {\n        errorReasons.push({\n          type: 'ACCOUNT.NOT.FOUND', code: 200, accounts: notFoundAccountsIds,\n        });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      // validation entries order.\n      const budget = await Budget.query().insert({\n        ...pick(form, ['name', 'fiscal_year', 'period', 'accounts_type']),\n      });\n\n      const promiseOpers = [];\n\n      form.accounts.forEach((account) => {\n        account.entries.forEach((entry) => {\n          const budgetEntry = BudgetEntry.query().insert({\n            account_id: account.account_id,\n            amount: entry.amount,\n            order: entry.order,\n          });\n          promiseOpers.push(budgetEntry);\n        });\n      });\n      await Promise.all(promiseOpers);\n\n      return res.status(200).send({ id: budget.id });\n    },\n  },\n\n  /**\n   * List of paginated budgets items.\n   */\n  listBudgets: {\n    validation: [\n      query('year').optional(),\n      query('income_statement').optional().isBoolean().toBoolean(),\n      query('profit_loss').optional().isBoolean().toBoolean(),\n      query('page').optional().isNumeric().toInt(),\n      query('page_size').isNumeric().toInt(),\n      query('custom_view_id').optional().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const filter = {\n        page_size: 10,\n        page: 1,\n        ...req.query,\n      };\n      const budgets = await Budget.query().runBefore((result, q) => {\n        if (filter.profit_loss) {\n          q.modify('filterByYear', filter.year);\n        }\n        if (filter.income_statement) {\n          q.modify('filterByIncomeStatement', filter.income_statement);\n        }\n        if (filter.profit_loss) {\n          q.modify('filterByProfitLoss', filter.profit_loss);\n        }\n        q.page(filter.page, filter.page_size);\n        return result;\n      });\n      return res.status(200).send({\n        items: budgets.items,\n      });\n    },\n  },\n};\n","import express from 'express';\nimport { query, validationResult } from 'express-validator';\nimport moment from 'moment';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport Budget from '@/models/Budget';\nimport Account from '@/models/Account';\nimport AccountType from '@/models/AccountType';\nimport NestedSet from '@/collection/NestedSet';\nimport BudgetEntry from '@/models/BudgetEntry';\nimport { dateRangeFormat } from '@/utils';\n\nexport default {\n\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(jwtAuth);\n\n    router.get('/budget_verses_actual/:reportId',\n      this.budgetVersesActual.validation,\n      asyncMiddleware(this.budgetVersesActual.handler));\n\n    return router;\n  },\n\n  budgetVersesActual: {\n    validation: [\n      query('basis').optional().isIn(['cash', 'accural']),\n      query('period').optional(),\n      query('active_accounts').optional().toBoolean(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { reportId } = req.params;\n      const form = { ...req.body };\n      const errorReasons = [];\n\n      const budget = await Budget.query().findById(reportId);\n\n      if (!budget) {\n        errorReasons.push({ type: 'BUDGET_NOT_FOUND', code: 100 });\n      }\n      const budgetEntries = await BudgetEntry.query().where('budget_id', budget.id);\n\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      const accountTypes = await AccountType.query()\n        .where('balance_sheet', budget.accountTypes === 'balance_sheet')\n        .where('income_sheet', budget.accountTypes === 'profit_losss');\n\n      const accounts = await Account.query().runBefore((result, q) => {\n        const accountTypesIds = accountTypes.map((t) => t.id);\n\n        if (accountTypesIds.length > 0) {\n          q.whereIn('account_type_id', accountTypesIds);\n        }\n        q.where('active', form.active_accounts === true);\n        q.withGraphFetched('transactions');\n      });\n        \n      // const accountsNestedSet = NestedSet.from(accounts);\n\n      const fromDate = moment(budget.year).startOf('year')\n        .add(budget.rangeOffset, budget.rangeBy).toDate();\n\n      const toDate = moment(budget.year).endOf('year').toDate();\n\n      const dateRange = moment.range(fromDate, toDate);\n      const dateRangeCollection = Array.from(dateRange.by(budget.rangeBy, {\n        step: budget.rangeIncrement, excludeEnd: false, excludeStart: false,\n      }));\n\n    //   // const accounts = {\n    //   //   assets: [\n    //   //     {\n    //   //       name: '',\n    //   //       code: '',\n    //   //       totalEntries: [\n    //   //         {\n\n    //   //         }\n    //   //       ],\n    //   //       children: [\n    //   //         {\n    //   //           name: '',\n    //   //           code: '',\n    //   //           entries: [\n    //   //             {\n\n    //   //             }\n    //   //           ]\n    //   //         }\n    //   //       ]\n    //   //     }\n    //   //   ]\n    //   // }\n\n      return res.status(200).send({\n        columns: dateRangeCollection.map(d => d.format(dateRangeFormat(budget.rangeBy))),\n        // accounts: {\n        //   asset: [],\n        //   liabilities: [],\n        //   equaity: [],\n\n        //   income: [],\n        //   expenses: [],\n        // }\n      });\n    },\n  },\n}","import express from 'express';\nimport { check, validationResult } from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\n\nexport default {\n\n  router() {\n    const router = express.Router();\n\n    router.get('/all',\n      this.all.validation,\n      asyncMiddleware(this.all.handler));\n\n    router.get('/registered',\n      this.registered.validation,\n      asyncMiddleware(this.registered.handler));\n\n    return router;\n  },\n\n  all: {\n    validation: [],\n    async handler(req, res) {\n\n      return res.status(200).send({\n        currencies: [\n          { currency_code: 'USD', currency_sign: '$' },\n          { currency_code: 'LYD', currency_sign: '' },\n        ],\n      });\n    },\n  },\n\n  registered: {\n    validation: [],\n    async handler(req, res) {\n\n      return res.status(200).send({\n        currencies: [\n          { currency_code: 'USD', currency_sign: '$' },\n          { currency_code: 'LYD', currency_sign: '' },\n        ],\n      });\n    },\n  },\n};","\nexport default {\n\n\n  router() {\n\n  },\n  \n  addExchangePrice: {\n    validation: {\n      \n    },\n    async handler(req, res) {\n\n    },\n  },\n}","import express from 'express';\nimport {\n  check,\n  param,\n  query,\n  validationResult,\n} from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\n\nexport default {\n\n  router() {\n    const router = express.Router();\n\n    router.post('/',\n      this.newCustomer.validation,\n      asyncMiddleware(this.newCustomer.handler));\n\n    router.post('/:id',\n      this.editCustomer.validation,\n      asyncMiddleware(this.editCustomer.handler));\n\n    return router;\n  },\n\n  newCustomer: {\n    validation: [\n      check('custom_type').exists().trim().escape(),\n      check('first_name').exists().trim().escape(),\n      check('last_name'),\n      check('company_name'),\n      check('email'),\n      check('work_phone'),\n      check('personal_phone'),\n\n      check('billing_address.country'),\n      check('billing_address.address'),\n      check('billing_address.city'),\n      check('billing_address.phone'),\n      check('billing_address.zip_code'),\n\n      check('shiping_address.country'),\n      check('shiping_address.address'),\n      check('shiping_address.city'),\n      check('shiping_address.phone'),\n      check('shiping_address.zip_code'),\n\n      check('contact.additional_phone'),\n      check('contact.additional_email'),\n\n      check('custom_fields').optional().isArray({ min: 1 }),\n      check('custom_fields.*.key').exists().trim().escape(),\n      check('custom_fields.*.value').exists(),\n\n      check('inactive').optional().isBoolean().toBoolean(),\n    ],\n\n    async handler(req, res) {\n\n    },\n  },\n\n  editCustomer: {\n    validation: [\n\n    ],\n    async handler(req, res) {\n\n    },\n  },\n};\n","import express from 'express';\nimport {\n  check,\n  param,\n  query,\n  validationResult,\n} from 'express-validator';\nimport moment from 'moment';\nimport { difference, chain, omit } from 'lodash';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport Expense from '@/models/Expense';\nimport Account from '@/models/Account';\nimport JournalPoster from '@/services/Accounting/JournalPoster';\nimport JournalEntry from '@/services/Accounting/JournalEntry';\nimport JWTAuth from '@/http/middleware/jwtAuth';\nimport AccountTransaction from '@/models/AccountTransaction';\nimport View from '@/models/View';\nimport Resource from '../../models/Resource';\nimport ResourceCustomFieldRepository from '@/services/CustomFields/ResourceCustomFieldRepository';\nimport {\n  validateViewRoles,\n  mapViewRolesToConditionals,\n} from '@/lib/ViewRolesBuilder';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n    router.use(JWTAuth);\n\n    router.post('/',\n      this.newExpense.validation,\n      asyncMiddleware(this.newExpense.handler));\n\n    router.post('/:id/publish',\n      this.publishExpense.validation,\n      asyncMiddleware(this.publishExpense.handler));\n\n    router.delete('/:id',\n      this.deleteExpense.validation,\n      asyncMiddleware(this.deleteExpense.handler));\n\n    router.post('/bulk',\n      this.bulkAddExpenses.validation,\n      asyncMiddleware(this.bulkAddExpenses.handler));\n\n    router.post('/:id',\n      this.updateExpense.validation,\n      asyncMiddleware(this.updateExpense.handler));\n\n    router.get('/',\n      this.listExpenses.validation,\n      asyncMiddleware(this.listExpenses.handler));\n\n    // router.get('/:id',\n    //   this.getExpense.validation,\n    //   asyncMiddleware(this.getExpense.handler));\n\n    return router;\n  },\n\n  /**\n   * Saves a new expense.\n   */\n  newExpense: {\n    validation: [\n      check('date').optional(),\n      check('payment_account_id').exists().isNumeric().toInt(),\n      check('expense_account_id').exists().isNumeric().toInt(),\n      check('description').optional(),\n      check('amount').exists().isNumeric().toFloat(),\n      check('currency_code').optional(),\n      check('exchange_rate').optional().isNumeric().toFloat(),\n      check('publish').optional().isBoolean().toBoolean(),\n      check('custom_fields').optional().isArray({ min: 1 }),\n      check('custom_fields.*.key').exists().trim().escape(),\n      check('custom_fields.*.value').exists(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = {\n        date: new Date(),\n        published: false,\n        custom_fields: [],\n        ...req.body,\n      };\n      // Convert the date to the general format.\n      form.date = moment(form.date).format('YYYY-MM-DD');\n\n      const errorReasons = [];\n      const paymentAccount = await Account.query()\n        .findById(form.payment_account_id).first();\n\n      if (!paymentAccount) {\n        errorReasons.push({ type: 'PAYMENT.ACCOUNT.NOT.FOUND', code: 100 });\n      }\n      const expenseAccount = await Account.query().findById(form.expense_account_id).first();\n\n      if (!expenseAccount) {\n        errorReasons.push({ type: 'EXPENSE.ACCOUNT.NOT.FOUND', code: 200 });\n      }\n      // const customFields = new ResourceCustomFieldRepository(Expense);\n      // await customFields.load();\n\n      // if (customFields.validateExistCustomFields()) {\n      //   errorReasons.push({ type: 'CUSTOM.FIELDS.SLUGS.NOT.EXISTS', code: 400 });\n      // }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      const expenseTransaction = await Expense.query().insertAndFetch({\n        ...omit(form, ['custom_fields']),\n      });\n      // customFields.fillCustomFields(expenseTransaction.id, form.custom_fields);\n\n      const journalEntries = new JournalPoster();\n      const creditEntry = new JournalEntry({\n        credit: form.amount,\n        referenceId: expenseTransaction.id,\n        referenceType: Expense.referenceType,\n        date: form.date,\n        account: expenseAccount.id,\n        accountNormal: 'debit',\n        draft: !form.published,\n      });\n      const debitEntry = new JournalEntry({\n        debit: form.amount,\n        referenceId: expenseTransaction.id,\n        referenceType: Expense.referenceType,\n        date: form.date,\n        account: paymentAccount.id,\n        accountNormal: 'debit',\n        draft: !form.published,\n      });\n      journalEntries.credit(creditEntry);\n      journalEntries.debit(debitEntry);\n\n      await Promise.all([\n        // customFields.saveCustomFields(expenseTransaction.id),\n        journalEntries.saveEntries(),\n        journalEntries.saveBalance(),\n      ]);\n      return res.status(200).send({ id: expenseTransaction.id });\n    },\n  },\n\n  /**\n   * Bulk add expneses to the given accounts.\n   */\n  bulkAddExpenses: {\n    validation: [\n      check('expenses').exists().isArray({ min: 1 }),\n      check('expenses.*.date').optional().isISO8601(),\n      check('expenses.*.payment_account_id').exists().isNumeric().toInt(),\n      check('expenses.*.expense_account_id').exists().isNumeric().toInt(),\n      check('expenses.*.description').optional(),\n      check('expenses.*.amount').exists().isNumeric().toFloat(),\n      check('expenses.*.currency_code').optional(),\n      check('expenses.*.exchange_rate').optional().isNumeric().toFloat(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = { ...req.body };\n      const errorReasons = [];\n\n      const paymentAccountsIds = chain(form.expenses)\n        .map((e) => e.payment_account_id).uniq().value();\n      const expenseAccountsIds = chain(form.expenses)\n        .map((e) => e.expense_account_id).uniq().value();\n\n      const [expensesAccounts, paymentAccounts] = await Promise.all([\n        Account.query().whereIn('id', expenseAccountsIds),\n        Account.query().whereIn('id', paymentAccountsIds),\n      ]);\n      const storedExpensesAccountsIds = expensesAccounts.map((a) => a.id);\n      const storedPaymentAccountsIds = paymentAccounts.map((a) => a.id);\n\n      const notFoundPaymentAccountsIds = difference(expenseAccountsIds, storedExpensesAccountsIds);\n      const notFoundExpenseAccountsIds = difference(paymentAccountsIds, storedPaymentAccountsIds);\n\n      if (notFoundPaymentAccountsIds.length > 0) {\n        errorReasons.push({\n          type: 'PAYMENY.ACCOUNTS.NOT.FOUND',\n          code: 100,\n          accounts: notFoundPaymentAccountsIds,\n        });\n      }\n      if (notFoundExpenseAccountsIds.length > 0) {\n        errorReasons.push({\n          type: 'EXPENSE.ACCOUNTS.NOT.FOUND',\n          code: 200,\n          accounts: notFoundExpenseAccountsIds,\n        });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { reasons: errorReasons });\n      }\n      const expenseSaveOpers = [];\n      const journalPoster = new JournalPoster();\n\n      form.expenses.forEach(async (expense) => {\n        const expenseSaveOper = Expense.query().insert({ ...expense });\n        expenseSaveOpers.push(expenseSaveOper);\n      });\n      // Wait unit save all expense transactions.\n      const savedExpenseTransactions = await Promise.all(expenseSaveOpers);\n\n      savedExpenseTransactions.forEach((expense) => {\n        const date = moment(expense.date).format('YYYY-DD-MM');\n\n        const debit = new JournalEntry({\n          debit: expense.amount,\n          referenceId: expense.id,\n          referenceType: Expense.referenceType,\n          account: expense.payment_account_id,\n          accountNormal: 'debit',\n          date,\n        });\n        const credit = new JournalEntry({\n          credit: expense.amount,\n          referenceId: expense.id,\n          referenceType: Expense.referenceId,\n          account: expense.expense_account_id,\n          accountNormal: 'debit',\n          date,\n        });\n        journalPoster.credit(credit);\n        journalPoster.debit(debit);\n      });\n\n      // Save expense journal entries and balance change.\n      await Promise.all([\n        journalPoster.saveEntries(),\n        journalPoster.saveBalance(),\n      ]);\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Publish the given expense id.\n   */\n  publishExpense: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { id } = req.params;\n      const errorReasons = [];\n      const expense = await Expense.query().findById(id);\n\n      if (!expense) {\n        errorReasons.push({ type: 'EXPENSE.NOT.FOUND', code: 100 });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n\n      if (expense.published) {\n        errorReasons.push({ type: 'EXPENSE.ALREADY.PUBLISHED', code: 200 });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n\n      await AccountTransaction.query()\n        .where('reference_id', expense.id)\n        .where('reference_type', 'Expense')\n        .patch({\n          draft: false,\n        });\n\n      await Expense.query()\n        .where('id', expense.id)\n        .update({ published: true });\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrieve paginated expenses list.\n   */\n  listExpenses: {\n    validation: [\n      query('expense_account_id').optional().isNumeric().toInt(),\n      query('payment_account_id').optional().isNumeric().toInt(),\n      query('note').optional(),\n      query('range_from').optional().isNumeric().toFloat(),\n      query('range_to').optional().isNumeric().toFloat(),\n      query('date_from').optional().isISO8601(),\n      query('date_to').optional().isISO8601(),\n      query('column_sort_order').optional().isIn(['created_at', 'date', 'amount']),\n      query('sort_order').optional().isIn(['desc', 'asc']),\n      query('page').optional().isNumeric().toInt(),\n      query('page_size').optional().isNumeric().toInt(),\n      query('custom_view_id').optional().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        page_size: 10,\n        page: 1,\n        ...req.query,\n      };\n      const errorReasons = [];\n      const expenseResource = await Resource.query().where('name', 'expenses').first();\n\n      if (!expenseResource) {\n        errorReasons.push({ type: 'EXPENSE_RESOURCE_NOT_FOUND', code: 300 });\n      }\n      if (errorReasons.length > 0) {\n        return res.status(400).send({ errors: errorReasons });\n      }\n      const view = await View.query().onBuild((builder) => {\n        if (filter.custom_view_id) {\n          builder.where('id', filter.custom_view_id);\n        } else {\n          builder.where('favourite', true);\n        }\n        builder.where('resource_id', expenseResource.id);\n        builder.withGraphFetched('viewRoles.field');\n        builder.withGraphFetched('columns');\n\n        builder.first();\n      });\n      let viewConditionals = [];\n\n      if (view && view.viewRoles.length > 0) {\n        viewConditionals = mapViewRolesToConditionals(view.viewRoles);\n\n        if (!validateViewRoles(viewConditionals, view.rolesLogicExpression)) {\n          errorReasons.push({ type: 'VIEW.LOGIC.EXPRESSION.INVALID', code: 400 })\n        }\n      }\n      if (!view && filter.custom_view_id) {\n        errorReasons.push({ type: 'VIEW_NOT_FOUND', code: 100 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      const expenses = await Expense.query().onBuild((builder) => {\n        builder.withGraphFetched('paymentAccount');\n        builder.withGraphFetched('expenseAccount');\n        builder.withGraphFetched('user');\n\n        if (viewConditionals.length) {\n          builder.modify('viewRolesBuilder', viewConditionals, view.rolesLogicExpression);\n        }\n        builder.modify('filterByAmountRange', filter.range_from, filter.to_range);\n        builder.modify('filterByDateRange', filter.date_from, filter.date_to);\n        builder.modify('filterByExpenseAccount', filter.expense_account_id);\n        builder.modify('filterByPaymentAccount', filter.payment_account_id);\n        builder.modify('orderBy', filter.column_sort_order, filter.sort_order);\n      }).page(filter.page - 1, filter.page_size);\n\n      return res.status(200).send({\n        ...(view) ? {\n          customViewId: view.id, \n          viewColumns: view.columns,\n          viewConditionals,\n        } : {},\n        expenses,\n      });\n    },\n  },\n\n  /**\n   * Delete the given account.\n   */\n  deleteExpense: {\n    validation: [\n      param('id').isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { id } = req.params;\n      const expenseTransaction = await Expense.query().findById(id);\n\n      if (!expenseTransaction) {\n        return res.status(404).send({\n          errors: [{ type: 'EXPENSE.TRANSACTION.NOT.FOUND', code: 100 }],\n        });\n      }\n      const expenseEntries = await AccountTransaction.query()\n        .where('reference_type', 'Expense')\n        .where('reference_id', expenseTransaction.id);\n\n      const expenseEntriesCollect = new JournalPoster();\n      expenseEntriesCollect.loadEntries(expenseEntries);\n      expenseEntriesCollect.reverseEntries();\n\n      await Promise.all([\n        Expense.query().findById(expenseTransaction.id).delete(),\n        expenseEntriesCollect.deleteEntries(),\n        expenseEntriesCollect.saveBalance(),\n      ]);\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Update details of the given account.\n   */\n  updateExpense: {\n    validation: [\n      param('id').isNumeric().toInt(),\n      check('date').optional().isISO8601(),\n      check('payment_account_id').exists().isNumeric().toInt(),\n      check('expense_account_id').exists().isNumeric().toInt(),\n      check('description').optional(),\n      check('amount').exists().isNumeric().toFloat(),\n      check('currency_code').optional(),\n      check('exchange_rate').optional().isNumeric().toFloat(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { id } = req.params;\n      const expenseTransaction = await Expense.query().findById(id);\n\n      if (!expenseTransaction) {\n        return res.status(404).send({\n          errors: [{ type: 'EXPENSE.TRANSACTION.NOT.FOUND', code: 100 }],\n        });\n      }\n    },\n  },\n\n  /**\n   * Retrieve details of the given expense id.\n   */\n  getExpense: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { id } = req.params;\n      const expenseTransaction = await Expense.query().findById(id);\n\n      if (!expenseTransaction) {\n        return res.status(404).send({\n          errors: [{ type: 'EXPENSE.TRANSACTION.NOT.FOUND', code: 100 }],\n        });\n      }\n\n      const expenseCFMetadataRepo = new ResourceCustomFieldRepository(Expense);\n      await expenseCFMetadataRepo.load();\n      await expenseCFMetadataRepo.fetchCustomFieldsMetadata(expenseTransaction.id);\n\n      const expenseCusFieldsMetadata = expenseCFMetadataRepo.getMetadata(expenseTransaction.id);\n\n      return res.status(200).send({\n        ...expenseTransaction,\n        custom_fields: [\n          ...expenseCusFieldsMetadata.toArray(),\n        ],\n      });\n    },\n  },\n};\n","import express from 'express';\nimport { check, param, validationResult } from 'express-validator';\nimport ResourceField from '@/models/ResourceField';\nimport Resource from '@/models/Resource';\nimport asyncMiddleware from '../middleware/asyncMiddleware';\n\n/**\n * Types of the custom fields.\n */\nconst TYPES = ['text', 'email', 'number', 'url', 'percentage', 'checkbox', 'radio', 'textarea'];\n\nexport default {\n  /**\n   * Router constructor method.\n   */\n  router() {\n    const router = express.Router();\n\n    router.post('/resource/:resource_name',\n      this.addNewField.validation,\n      asyncMiddleware(this.addNewField.handler));\n\n    router.post('/:field_id',\n      this.editField.validation,\n      asyncMiddleware(this.editField.handler));\n\n    router.post('/status/:field_id',\n      this.changeStatus.validation,\n      asyncMiddleware(this.changeStatus.handler));\n\n    // router.get('/:field_id',\n    //   asyncMiddleware(this.getField.handler));\n\n    // router.delete('/:field_id',\n    //   asyncMiddleware(this.deleteField.handler));\n\n    return router;\n  },\n\n  /**\n   * Adds a new field control to the given resource.\n   * @param {Request} req -\n   * @param {Response} res -\n   */\n  addNewField: {\n    validation: [\n      param('resource_name').exists().trim().escape(),\n      check('label').exists().escape().trim(),\n      check('data_type').exists().isIn(TYPES),\n      check('help_text').optional(),\n      check('default').optional(),\n      check('options').optional().isArray(),\n      check('options.*.key').exists().isNumeric().toInt(),\n      check('options.*.value').exists(),\n    ],\n    async handler(req, res) {\n      const { resource_name: resourceName } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const resource = await Resource.query().where('name', resourceName).first();\n\n      if (!resource) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'RESOURCE_NOT_FOUND', code: 100 }],\n        });\n      }\n      const form = { options: [], ...req.body };\n      const choices = form.options.map((option) => ({ key: option.key, value: option.value }));\n\n      const storedResource = await ResourceField.query().insertAndFetch({\n        data_type: form.data_type,\n        label_name: form.label,\n        help_text: form.help_text,\n        default: form.default,\n        resource_id: resource.id,\n        options: choices,\n        index: -1,\n      });\n      return res.status(200).send({ id: storedResource.id });\n    },\n  },\n\n  /**\n   * Edit details of the given field.\n   */\n  editField: {\n    validation: [\n      param('field_id').exists().isNumeric().toInt(),\n      check('label').exists().escape().trim(),\n      check('data_type').exists().isIn(TYPES),\n      check('help_text').optional(),\n      check('default').optional(),\n      check('options').optional().isArray(),\n      check('options.*.key').exists().isNumeric().toInt(),\n      check('options.*.value').exists(),\n    ],\n    async handler(req, res) {\n      const { field_id: fieldId } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const field = await ResourceField.query().findById(fieldId);\n\n      if (!field) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'FIELD_NOT_FOUND', code: 100 }],\n        });\n      }\n      // Sets the default value of optional fields.\n      const form = { options: [], ...req.body };\n      const choices = form.options.map((option) => ({ key: option.key, value: option.value }));\n\n      await ResourceField.query().findById(field.id).update({\n        data_type: form.data_type,\n        label_name: form.label,\n        help_text: form.help_text,\n        default: form.default,\n        options: choices,\n      });\n      return res.status(200).send({ id: field.id });\n    },\n  },\n\n  /**\n   * Retrieve the fields list of the given resource.\n   * @param {Request} req -\n   * @param {Response} res -\n   */\n  fieldsList: {\n    validation: [\n      param('resource_name').toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { resource_name: resourceName } = req.params;\n      const resource = await Resource.query().where('name', resourceName).first();\n\n      if (!resource) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'RESOURCE_NOT_FOUND', code: 100 }],\n        });\n      }\n      const fields = await ResourceField.where('resource_id', resource.id).fetchAll();\n\n      return res.status(200).send({ fields: fields.toJSON() });\n    },\n  },\n\n  /**\n   * Change status of the given field.\n   */\n  changeStatus: {\n    validation: [\n      param('field_id').toInt(),\n      check('active').isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const { field_id: fieldId } = req.params;\n      const field = await ResourceField.query().findById(fieldId);\n\n      if (!field) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'NOT_FOUND_FIELD', code: 100 }],\n        });\n      }\n\n      const { active } = req.body;\n      await ResourceField.query().findById(field.id).patch({ active });\n\n      return res.status(200).send({ id: field.id });\n    },\n  },\n\n  /**\n   * Retrieve details of the given field.\n   */\n  getField: {\n    validation: [\n      param('field_id').toInt(),\n    ],\n    async handler(req, res) {\n      const { field_id: id } = req.params;\n      const field = await ResourceField.where('id', id).fetch();\n\n      if (!field) {\n        return res.boom.notFound();\n      }\n\n      return res.status(200).send({\n        field: field.toJSON(),\n      });\n    },\n  },\n\n  /**\n   * Delete the given field.\n   */\n  deleteField: {\n    validation: [\n      param('field_id').toInt(),\n    ],\n    async handler(req, res) {\n      const { field_id: id } = req.params;\n      const field = await ResourceField.where('id', id).fetch();\n\n      if (!field) {\n        return res.boom.notFound();\n      }\n      if (field.attributes.predefined) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'PREDEFINED_FIELD', code: 100 }],\n        });\n      }\n      await field.destroy();\n\n      return res.status(200).send({ id: field.get('id') });\n    },\n  },\n};\n","import express from 'express';\nimport { query, validationResult } from 'express-validator';\nimport moment from 'moment';\nimport { pick } from 'lodash';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport AccountTransaction from '@/models/AccountTransaction';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport AccountType from '@/models/AccountType';\nimport Account from '@/models/Account';\nimport JournalPoster from '@/services/Accounting/JournalPoster';\nimport { dateRangeCollection } from '@/utils';\n\nconst formatNumberClosure = (filter) => (balance) => {\n  let formattedBalance = parseFloat(balance);\n\n  if (filter.no_cents) {\n    formattedBalance = parseInt(formattedBalance, 10);\n  }\n  if (filter.divide_1000) {\n    formattedBalance /= 1000;\n  }\n  return formattedBalance;\n};\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n    router.use(jwtAuth);\n\n    router.get('/ledger',\n      this.ledger.validation,\n      asyncMiddleware(this.ledger.handler));\n\n    router.get('/general_ledger',\n      this.generalLedger.validation,\n      asyncMiddleware(this.generalLedger.handler));\n\n    router.get('/balance_sheet',\n      this.balanceSheet.validation,\n      asyncMiddleware(this.balanceSheet.handler));\n\n    router.get('/trial_balance_sheet',\n      this.trialBalanceSheet.validation,\n      asyncMiddleware(this.trialBalanceSheet.handler));\n\n    router.get('/profit_loss_sheet',\n      this.profitLossSheet.validation,\n      asyncMiddleware(this.profitLossSheet.handler));\n\n    router.get('/cash_flow_statement',\n      this.cashFlowStatement.validation,\n      asyncMiddleware(this.cashFlowStatement.handler));\n\n    return router;\n  },\n\n  /**\n   * Retrieve the ledger report of the given account.\n   */\n  ledger: {\n    validation: [\n      query('from_date').optional().isISO8601(),\n      query('to_date').optional().isISO8601(),\n      query('transaction_types').optional().isArray({ min: 1 }),\n      query('account_ids').optional().isArray({ min: 1 }),\n      query('account_ids.*').optional().isNumeric().toInt(),\n      query('from_range').optional().isNumeric().toInt(),\n      query('to_range').optional().isNumeric().toInt(),\n      query('number_format.no_cents').optional().isBoolean().toBoolean(),\n      query('number_format.divide_1000').optional().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        from_range: null,\n        to_range: null,\n        account_ids: [],\n        transaction_types: [],\n        number_format: {\n          no_cents: false,\n          divide_1000: false,\n        },\n        ...req.query,\n      };\n      const accountsJournalEntries = await AccountTransaction.query()\n        .modify('filterDateRange', filter.from_date, filter.to_date)\n        .modify('filterAccounts', filter.account_ids)\n        .modify('filterTransactionTypes', filter.transaction_types)\n        .modify('filterAmountRange', filter.from_range, filter.to_range)\n        .withGraphFetched('account');\n\n      const formatNumber = formatNumberClosure(filter.number_format);\n\n      return res.status(200).send({\n        meta: { ...filter },\n        items: accountsJournalEntries.map((entry) => ({\n          ...entry,\n          credit: formatNumber(entry.credit),\n          debit: formatNumber(entry.debit),\n        })),\n      });\n    },\n  },\n\n  /**\n   * Retrieve the general ledger financial statement.\n   */\n  generalLedger: {\n    validation: [\n      query('from_date').optional().isISO8601(),\n      query('to_date').optional().isISO8601(),\n      query('basis').optional(),\n      query('number_format.no_cents').optional().isBoolean().toBoolean(),\n      query('number_format.divide_1000').optional().isBoolean().toBoolean(),\n      query('none_zero').optional().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        from_date: moment().startOf('year').format('YYYY-MM-DD'),\n        to_date: moment().endOf('year').format('YYYY-MM-DD'),\n        number_format: {\n          no_cents: false,\n          divide_1000: false,\n        },\n        none_zero: false,\n        ...req.query,\n      };\n      const accounts = await Account.query()\n        .orderBy('index', 'DESC')\n        .withGraphFetched('transactions')\n        .modifyGraph('transactions', (builder) => {\n          builder.modify('filterDateRange', filter.from_date, filter.to_date);\n        });\n\n      const openingBalanceTransactions = await AccountTransaction.query()\n        .modify('filterDateRange', null, filter.from_date)\n        .modify('sumationCreditDebit')\n        .withGraphFetched('account.type');\n\n      const closingBalanceTransactions = await AccountTransaction.query()\n        .modify('filterDateRange', null, filter.to_date)\n        .modify('sumationCreditDebit')\n        .withGraphFetched('account.type');\n\n      const opeingBalanceCollection = new JournalPoster();\n      const closingBalanceCollection = new JournalPoster();\n\n      opeingBalanceCollection.loadEntries(openingBalanceTransactions);\n      closingBalanceCollection.loadEntries(closingBalanceTransactions);\n\n      // Transaction amount formatter based on the given query.\n      const formatNumber = formatNumberClosure(filter.number_format);\n\n      const items = [\n        ...accounts\n          .filter((account) => (\n            account.transactions.length > 0 || !filter.none_zero\n          ))\n          .map((account) => ({\n            ...pick(account, ['id', 'name', 'code', 'index']),\n            transactions: [\n              ...account.transactions.map((transaction) => ({\n                ...transaction,\n                credit: formatNumber(transaction.credit),\n                debit: formatNumber(transaction.debit),\n              })),\n            ],\n            opening: {\n              date: filter.from_date,\n              balance: opeingBalanceCollection.getClosingBalance(account.id),\n            },\n            closing: {\n              date: filter.to_date,\n              balance: closingBalanceCollection.getClosingBalance(account.id),\n            },\n          })),\n      ];\n      return res.status(200).send({\n        meta: { ...filter },\n        accounts: items,\n      });\n    },\n  },\n\n  /**\n   * Retrieve the balance sheet.\n   */\n  balanceSheet: {\n    validation: [\n      query('accounting_method').optional().isIn(['cash', 'accural']),\n      query('from_date').optional(),\n      query('to_date').optional(),\n      query('display_columns_by').optional().isIn(['total', 'year', 'month', 'week', 'day', 'quarter']),\n      query('number_format.no_cents').optional().isBoolean().toBoolean(),\n      query('number_format.divide_1000').optional().isBoolean().toBoolean(),\n      query('none_zero').optional().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        display_columns_by: 'total',\n        from_date: moment().startOf('year').format('YYYY-MM-DD'),\n        to_date: moment().endOf('year').format('YYYY-MM-DD'),\n        number_format: {\n          no_cents: false,\n          divide_1000: false,\n        },\n        none_zero: false,\n        basis: 'cash',\n        ...req.query,\n      };\n\n      const balanceSheetTypes = await AccountType.query().where('balance_sheet', true);\n\n      // Fetch all balance sheet accounts.\n      const accounts = await Account.query()\n        .whereIn('account_type_id', balanceSheetTypes.map((a) => a.id))\n        .withGraphFetched('type')\n        .withGraphFetched('transactions')\n        .modifyGraph('transactions', (builder) => {\n          builder.modify('filterDateRange', null, filter.to_date);\n        });\n\n      const journalEntriesCollected = Account.collectJournalEntries(accounts);\n      const journalEntries = new JournalPoster();\n      journalEntries.loadEntries(journalEntriesCollected);\n\n      // Account balance formmatter based on the given query.\n      const balanceFormatter = formatNumberClosure(filter.number_format);\n      const filterDateType = filter.display_columns_by === 'total'\n        ? 'day' : filter.display_columns_by;\n\n      // Gets the date range set from start to end date.\n      const dateRangeSet = dateRangeCollection(\n        filter.from_date,\n        filter.to_date,\n        filterDateType,\n      );\n\n      // Retrieve the asset balance sheet.\n      const assets = accounts\n        .filter((account) => (\n          account.type.normal === 'debit'\n            && (account.transactions.length > 0 || !filter.none_zero)\n        ))\n        .map((account) => {\n          // Calculates the closing balance to the given date.\n          const closingBalance = journalEntries.getClosingBalance(account.id, filter.to_date);\n          const type = filter.display_columns_by;\n\n          return {\n            ...pick(account, ['id', 'index', 'name', 'code']),\n            ...(type !== 'total') ? {\n              periods_balance: dateRangeSet.map((date) => {\n                const balance = journalEntries.getClosingBalance(account.id, date, filterDateType);\n\n                return {\n                  date,\n                  formatted_amount: balanceFormatter(balance),\n                  amount: balance,\n                };\n              }),\n            } : {},\n            balance: {\n              formatted_amount: balanceFormatter(closingBalance),\n              amount: closingBalance,\n              date: filter.to_date,\n            },\n          };\n        });\n\n      // Retrieve liabilities and equity balance sheet.\n      const liabilitiesEquity = accounts\n        .filter((account) => (\n          account.type.normal === 'credit'\n            && (account.transactions.length > 0 || !filter.none_zero)\n        ))\n        .map((account) => {\n          // Calculates the closing balance to the given date.\n          const closingBalance = journalEntries.getClosingBalance(account.id, filter.to_date);\n          const type = filter.display_columns_by;\n\n          return {\n            ...pick(account, ['id', 'index', 'name', 'code']),\n            ...(type !== 'total') ? {\n              periods_balance: dateRangeSet.map((date) => {\n                const balance = journalEntries.getClosingBalance(account.id, date, filterDateType);\n\n                return {\n                  date,\n                  formatted_amount: balanceFormatter(balance),\n                  amount: balance,\n                };\n              }),\n            } : {},\n            balance: {\n              formattedAmount: balanceFormatter(closingBalance),\n              amount: closingBalance,\n              date: filter.to_date,\n            },\n          };\n        });\n\n      return res.status(200).send({\n        query: { ...filter },\n        columns: { ...dateRangeSet },\n        balance_sheet: {\n          assets: {\n            title: 'Assets',\n            accounts: [...assets],\n          },\n          liabilities_equity: {\n            title: 'Liabilities & Equity',\n            accounts: [...liabilitiesEquity],\n          },\n        },\n      });\n    },\n  },\n\n  /**\n   * Retrieve the trial balance sheet.\n   */\n  trialBalanceSheet: {\n    validation: [\n      query('basis').optional(),\n      query('from_date').optional().isISO8601(),\n      query('to_date').optional().isISO8601(),\n      query('number_format.no_cents').optional().isBoolean(),\n      query('number_format.1000_divide').optional().isBoolean(),\n      query('basis').optional(),\n      query('none_zero').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        from_date: moment().startOf('year').format('YYYY-MM-DD'),\n        to_date: moment().endOf('year').format('YYYY-MM-DD'),\n        number_format: {\n          no_cents: false,\n          divide_1000: false,\n        },\n        basis: 'accural',\n        none_zero: false,\n        ...req.query,\n      };\n\n      const accounts = await Account.query()\n        .withGraphFetched('type')\n        .withGraphFetched('transactions')\n        .modifyGraph('transactions', (builder) => {\n          builder.modify('sumationCreditDebit');\n          builder.modify('filterDateRange', filter.from_date, filter.to_date);\n        });\n\n      const journalEntriesCollect = Account.collectJournalEntries(accounts);\n      const journalEntries = new JournalPoster();\n      journalEntries.loadEntries(journalEntriesCollect);\n\n      // Account balance formmatter based on the given query.\n      const balanceFormatter = formatNumberClosure(filter.number_format);\n\n      const items = accounts\n        .filter((account) => (\n          account.transactions.length > 0 || !filter.none_zero\n        ))\n        .map((account) => {\n          const trial = journalEntries.getTrialBalance(account.id);\n          return {\n            account_id: account.id,\n            code: account.code,\n            accountNormal: account.type.normal,\n            credit: balanceFormatter(trial.credit),\n            debit: balanceFormatter(trial.debit),\n            balance: balanceFormatter(trial.balance),\n          };\n        });\n      return res.status(200).send({\n        query: { ...filter },\n        items: [...items],\n      });\n    },\n  },\n\n  /**\n   * Retrieve profit/loss financial statement.\n   */\n  profitLossSheet: {\n    validation: [\n      query('basis').optional(),\n      query('from_date').optional().isISO8601(),\n      query('to_date').optional().isISO8601(),\n      query('number_format.no_cents').optional().isBoolean(),\n      query('number_format.divide_1000').optional().isBoolean(),\n      query('basis').optional(),\n      query('none_zero').optional(),\n      query('display_columns_by').optional().isIn(['year', 'month', 'week', 'day', 'quarter']),\n      query('accounts').optional().isArray(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const filter = {\n        from_date: moment().startOf('year').format('YYYY-MM-DD'),\n        to_date: moment().endOf('year').format('YYYY-MM-DD'),\n        number_format: {\n          no_cents: false,\n          divide_1000: false,\n        },\n        basis: 'accural',\n        none_zero: false,\n        display_columns_by: 'month',\n        ...req.query,\n      };\n      const incomeStatementTypes = await AccountType.query().where('income_sheet', true);\n\n      const accounts = await Account.query()\n        .whereIn('account_type_id', incomeStatementTypes.map((t) => t.id))\n        .withGraphFetched('type')\n        .withGraphFetched('transactions');\n\n      const filteredAccounts = accounts.filter((account) => {\n        return account.transactions.length > 0 || !filter.none_zero;\n      });\n      const journalEntriesCollected = Account.collectJournalEntries(accounts);\n      const journalEntries = new JournalPoster();\n      journalEntries.loadEntries(journalEntriesCollected);\n\n      // Account balance formmatter based on the given query.\n      const numberFormatter = formatNumberClosure(filter.number_format);\n\n      // Gets the date range set from start to end date.\n      const dateRangeSet = dateRangeCollection(\n        filter.from_date,\n        filter.to_date,\n        filter.display_columns_by,\n      );\n      const accountsIncome = filteredAccounts\n        .filter((account) => account.type.normal === 'credit')\n        .map((account) => ({\n          ...pick(account, ['id', 'index', 'name', 'code']),\n          dates: dateRangeSet.map((date) => {\n            const type = filter.display_columns_by;\n            const amount = journalEntries.getClosingBalance(account.id, date, type);\n\n            return { date, rawAmount: amount, amount: numberFormatter(amount) };\n          }),\n        }));\n\n      const accountsExpenses = filteredAccounts\n        .filter((account) => account.type.normal === 'debit')\n        .map((account) => ({\n          ...pick(account, ['id', 'index', 'name', 'code']),\n          dates: dateRangeSet.map((date) => {\n            const type = filter.display_columns_by;\n            const amount = journalEntries.getClosingBalance(account.id, date, type);\n\n            return { date, rawAmount: amount, amount: numberFormatter(amount) };\n          }),\n        }));\n\n      // Calculates the total income of income accounts.\n      const totalAccountsIncome = dateRangeSet.reduce((acc, date, index) => {\n        let amount = 0;\n        accountsIncome.forEach((account) => {\n          const currentDate = account.dates[index];\n          amount += currentDate.rawAmount || 0;\n        });\n        acc[date] = { date, rawAmount: amount, amount: numberFormatter(amount) };\n        return acc;\n      }, {});\n\n      // Calculates the total expenses of expenses accounts.\n      const totalAccountsExpenses = dateRangeSet.reduce((acc, date, index) => {\n        let amount = 0;\n        accountsExpenses.forEach((account) => {\n          const currentDate = account.dates[index];\n          amount += currentDate.rawAmount || 0;\n        });\n        acc[date] = { date, rawAmount: amount, amount: numberFormatter(amount) };\n        return acc;\n      }, {});\n\n      // Total income(date) - Total expenses(date) = Net income(date)\n      const netIncome = dateRangeSet.map((date) => {\n        const totalIncome = totalAccountsIncome[date];\n        const totalExpenses = totalAccountsExpenses[date];\n\n        let amount = totalIncome.rawAmount || 0;\n        amount -= totalExpenses.rawAmount || 0;\n        return { date, rawAmount: amount, amount: numberFormatter(amount) };\n      });\n\n      return res.status(200).send({\n        meta: { ...filter },\n        income: {\n          entry_normal: 'credit',\n          accounts: accountsIncome,\n        },\n        expenses: {\n          entry_normal: 'debit',\n          accounts: accountsExpenses,\n        },\n        total_income: Object.values(totalAccountsIncome),\n        total_expenses: Object.values(totalAccountsExpenses),\n        total_net_income: netIncome,\n      });\n    },\n  },\n\n  cashFlowStatement: {\n    validation: [\n      query('date_from').optional(),\n      query('date_to').optional(),\n    ],\n    async handler(req, res) {\n      \n      return res.status(200).send({\n        meta: {},\n        operating: [],\n        financing: [],\n        investing: [],\n      });\n    },\n  },\n}\n","import express from 'express';\nimport { check, param, validationResult } from 'express-validator';\nimport asyncMiddleware from '../middleware/asyncMiddleware';\nimport ItemCategory from '@/models/ItemCategory';\nimport Authorization from '@/http/middleware/authorization';\nimport JWTAuth from '@/http/middleware/jwtAuth';\n\nexport default {\n  /**\n   * Router constructor method.\n   */\n  router() {\n    const router = express.Router();\n    const permit = Authorization('items_categories');\n\n    router.use(JWTAuth);\n\n    router.post('/:id',\n      permit('create', 'edit'),\n      this.editCategory.validation,\n      asyncMiddleware(this.editCategory.handler));\n\n    router.post('/',\n      permit('create'),\n      this.newCategory.validation,\n      asyncMiddleware(this.newCategory.handler));\n\n    router.delete('/:id',\n      permit('create', 'edit', 'delete'),\n      this.deleteItem.validation,\n      asyncMiddleware(this.deleteItem.handler));\n\n    router.get('/:id',\n      permit('view'),\n      this.getCategory.validation,\n      asyncMiddleware(this.getCategory.handler));\n\n    router.get('/',\n      permit('view'),\n      this.getList.validation,\n      asyncMiddleware(this.getList.validation));\n\n    return router;\n  },\n\n  /**\n   * Creates a new item category.\n   */\n  newCategory: {\n    validation: [\n      check('name').exists({ checkFalsy: true }).trim().escape(),\n      check('parent_category_id').optional().isNumeric().toInt(),\n      check('description').optional().trim().escape(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { name, parent_category_id: parentCategoryId, description } = req.body;\n\n      if (parentCategoryId) {\n        const foundParentCategory = await ItemCategory.where('id', parentCategoryId).fetch();\n\n        if (!foundParentCategory) {\n          return res.boom.notFound('The parent category ID is not found.', {\n            errors: [{ type: 'PARENT_CATEGORY_NOT_FOUND', code: 100 }],\n          });\n        }\n      }\n      const category = await ItemCategory.forge({\n        label: name,\n        parent_category_id: parentCategoryId,\n        description,\n      });\n\n      await category.save();\n      return res.status(200).send({ id: category.get('id') });\n    },\n  },\n\n  /**\n   * Edit details of the given category item.\n   */\n  editCategory: {\n    validation: [\n      param('id').toInt(),\n      check('name').exists({ checkFalsy: true }).trim().escape(),\n      check('parent_category_id').optional().isNumeric().toInt(),\n      check('description').optional().trim().escape(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { name, parent_category_id: parentCategoryId, description } = req.body;\n      const itemCategory = await ItemCategory.where('id', id).fetch();\n\n      if (!itemCategory) {\n        return res.boom.notFound();\n      }\n      if (parentCategoryId && parentCategoryId !== itemCategory.attributes.parent_category_id) {\n        const foundParentCategory = await ItemCategory.where('id', parentCategoryId).fetch();\n\n        if (!foundParentCategory) {\n          return res.boom.notFound('The parent category ID is not found.', {\n            errors: [{ type: 'PARENT_CATEGORY_NOT_FOUND', code: 100 }],\n          });\n        }\n      }\n      await itemCategory.save({\n        label: name,\n        description,\n        parent_category_id: parentCategoryId,\n      });\n\n      return res.status(200).send({ id: itemCategory.id });\n    },\n  },\n\n  /**\n   * Delete the give item category.\n   */\n  deleteItem: {\n    validation: [\n      param('id').toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const itemCategory = await ItemCategory.where('id', id).fetch();\n\n      if (!itemCategory) {\n        return res.boom.notFound();\n      }\n      await itemCategory.destroy();\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrieve the list of items.\n   */\n  getList: {\n    validation: [],\n    async handler(req, res) {\n      const items = await ItemCategory.fetch();\n\n      if (!items) {\n        return res.boom.notFound();\n      }\n      return res.status(200).send({ items: items.toJSON() });\n    },\n  },\n\n  /**\n   * Retrieve details of the given category.\n   */\n  getCategory: {\n    validation: [\n      param('category_id').toInt(),\n    ],\n    async handler(req, res) {\n      const { category_id: categoryId } = req.params;\n      const item = await ItemCategory.where('id', categoryId).fetch();\n\n      if (!item) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'CATEGORY_NOT_FOUND', code: 100 }],\n        });\n      }\n\n      return res.status(200).send({ category: item.toJSON() });\n    },\n  },\n};\n","import express from 'express';\nimport { check, validationResult } from 'express-validator';\nimport moment from 'moment';\nimport { difference } from 'lodash';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport Item from '@/models/Item';\nimport Account from '@/models/Account';\nimport ItemCategory from '@/models/ItemCategory';\nimport Resource from '@/models/Resource';\nimport ResourceField from '@/models/ResourceField';\nimport Authorization from '@/http/middleware/authorization';\n\nexport default {\n\n  router() {\n    const router = express.Router();\n    const permit = Authorization('items');\n\n    router.use(jwtAuth);\n\n    router.post('/:id',\n      this.editItem.validation,\n      asyncMiddleware(this.editItem.handler));\n\n    router.post('/',\n      // permit('create'),\n      this.newItem.validation,\n      asyncMiddleware(this.newItem.handler));\n\n    router.delete('/:id',\n      this.deleteItem.validation,\n      asyncMiddleware(this.deleteItem.handler));\n\n    // router.get('/:id',\n    //   this.getCategory.validation,\n    //   asyncMiddleware(this.getCategory.handler));\n\n    // router.get('/',\n    //   this.categoriesList.validation,\n    //   asyncMiddleware(this.categoriesList.validation));\n\n    return router;\n  },\n\n  /**\n   * Creates a new item.\n   */\n  newItem: {\n    validation: [\n      check('name').exists(),\n      check('type').exists().trim().escape().isIn(['service', 'product']),\n      check('cost_price').exists().isNumeric(),\n      check('sell_price').exists().isNumeric(),\n      check('cost_account_id').exists().isInt().toInt(),\n      check('sell_account_id').exists().isInt().toInt(),\n      check('category_id').optional().isInt().toInt(),\n\n      check('custom_fields').optional().isArray({ min: 1 }),\n      check('custom_fields.*.key').exists().isNumeric().toInt(),\n      check('custom_fields.*.value').exists(),\n\n      check('note').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = {\n        custom_fields: [],\n        ...req.body,\n      };\n      const errorReasons = [];\n\n      const costAccountPromise = Account.query().findById(form.cost_account_id);\n      const sellAccountPromise = Account.query().findById(form.sell_account_id);\n      const itemCategoryPromise = (form.category_id)\n        ? ItemCategory.query().findById(form.category_id) : null;\n\n      // Validate the custom fields key and value type.\n      if (form.custom_fields.length > 0) {\n        const customFieldsKeys = form.custom_fields.map((field) => field.key);\n\n        // Get resource id than get all resource fields.\n        const resource = await Resource.where('name', 'items').fetch();\n        const fields = await ResourceField.query((query) => {\n          query.where('resource_id', resource.id);\n          query.whereIn('key', customFieldsKeys);\n        }).fetchAll();\n\n        const storedFieldsKey = fields.map((f) => f.attributes.key);\n\n        // Get all not defined resource fields.\n        const notFoundFields = difference(customFieldsKeys, storedFieldsKey);\n\n        if (notFoundFields.length > 0) {\n          errorReasons.push({ type: 'FIELD_KEY_NOT_FOUND', code: 150, fields: notFoundFields });\n        }\n      }\n      const [costAccount, sellAccount, itemCategory] = await Promise.all([\n        costAccountPromise, sellAccountPromise, itemCategoryPromise,\n      ]);\n      if (!costAccount) {\n        errorReasons.push({ type: 'COST_ACCOUNT_NOT_FOUND', code: 100 });\n      }\n      if (!sellAccount) {\n        errorReasons.push({ type: 'SELL_ACCOUNT_NOT_FOUND', code: 120 });\n      }\n      if (!itemCategory && form.category_id) {\n        errorReasons.push({ type: 'ITEM_CATEGORY_NOT_FOUND', code: 140 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n      const item = await Item.query().insertAndFetch({\n        name: form.name,\n        type: form.type,\n        cost_price: form.cost_price,\n        sell_price: form.sell_price,\n        sell_account_id: form.sell_account_id,\n        cost_account_id: form.cost_account_id,\n        currency_code: form.currency_code,\n        note: form.note,\n      });\n      return res.status(200).send({ id: item.id });\n    },\n  },\n\n  /**\n   * Edit the given item.\n   */\n  editItem: {\n    validation: [\n      check('name').exists(),\n      check('type').exists().trim().escape().isIn(['product', 'service']),\n      check('cost_price').exists().isNumeric(),\n      check('sell_price').exists().isNumeric(),\n      check('cost_account_id').exists().isInt(),\n      check('sell_account_id').exists().isInt(),\n      check('category_id').optional().isInt(),\n      check('note').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { id } = req.params;\n      const form = {\n        custom_fields: [],\n        ...req.body,\n      };\n      const item = await Item.query().findById(id);\n      \n      if (!item) {\n        return res.boom.notFound(null, { errors: [\n          { type: 'ITEM.NOT.FOUND', code: 100 },\n        ]});\n      }\n      const errorReasons = [];\n\n      const costAccountPromise = Account.query().findById(form.cost_account_id);\n      const sellAccountPromise = Account.query().findById(form.sell_account_id);\n      const itemCategoryPromise = (form.category_id)\n        ? ItemCategory.query().findById(form.category_id) : null;\n\n      const [costAccount, sellAccount, itemCategory] = await Promise.all([\n        costAccountPromise, sellAccountPromise, itemCategoryPromise,\n      ]);\n      if (!costAccount) {\n        errorReasons.push({ type: 'COST_ACCOUNT_NOT_FOUND', code: 100 });\n      }\n      if (!sellAccount) {\n        errorReasons.push({ type: 'SELL_ACCOUNT_NOT_FOUND', code: 120 });\n      }\n      if (!itemCategory && form.category_id) {\n        errorReasons.push({ type: 'ITEM_CATEGORY_NOT_FOUND', code: 140 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      const updatedItem = await Item.query().findById(id).patch({\n        name: form.name,\n        type: form.type,\n        cost_price: form.cost_price,\n        sell_price: form.sell_price,\n        currency_code: form.currency_code,\n        sell_account_id: form.sell_account_id,\n        cost_account_id: form.cost_account_id,\n        category_id: form.category_id,\n        note: form.note,\n      });\n      return res.status(200).send({ id: updatedItem.id });\n    },\n  },\n\n  /**\n   * Delete the given item from the storage.\n   */\n  deleteItem: {\n    validation: [],\n    async handler(req, res) {\n      const { id } = req.params;\n      const item = await Item.query().findById(id);\n\n      if (!item) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'ITEM_NOT_FOUND', code: 100 }],\n        });\n      }\n\n      // Delete the fucking the given item id.\n      await Item.query().findById(item.id).delete();\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrive the list items with pagination meta.\n   */\n  listItems: {\n    validation: [],\n    async handler(req, res) {\n      const filter = {\n        name: '',\n        description: '',\n        SKU: '',\n        account_id: null,\n        page_size: 10,\n        page: 1,\n        start_date: null,\n        end_date: null,\n        ...req.query,\n      };\n\n      const items = await Item.query((query) => {\n        if (filter.description) {\n          query.where('description', 'like', `%${filter.description}%`);\n        }\n        if (filter.description) {\n          query.where('SKU', filter.SKY);\n        }\n        if (filter.name) {\n          query.where('name', filter.name);\n        }\n        if (filter.start_date) {\n          const startDateFormatted = moment(filter.start_date).format('YYYY-MM-DD HH:mm:SS');\n          query.where('created_at', '>=', startDateFormatted);\n        }\n        if (filter.end_date) {\n          const endDateFormatted = moment(filter.end_date).format('YYYY-MM-DD HH:mm:SS');\n          query.where('created_at', '<=', endDateFormatted);\n        }\n      }).fetchPage({\n        page_size: filter.page_size,\n        page: filter.page,\n      });\n\n      return res.status(200).send({\n        items: items.toJSON(),\n        pagination: items.pagination,\n      });\n    },\n  },\n};\n","import express from 'express';\nimport { body, query, validationResult } from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport Option from '@/models/Option';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.post('/',\n      this.saveOptions.validation,\n      asyncMiddleware(this.saveOptions.handler));\n\n    router.get('/',\n      this.getOptions.validation,\n      asyncMiddleware(this.getSettings));\n\n    return router;\n  },\n\n  /**\n   * Saves the given options to the storage.\n   */\n  saveOptions: {\n    validation: [\n      body('options').isArray(),\n      body('options.*.key').exists(),\n      body('options.*.value').exists(),\n      body('options.*.group').exists(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'VALIDATION_ERROR', ...validationErrors,\n        });\n      }\n      const form = { ...req.body };\n      const optionsCollections = await Option.query();\n\n      form.options.forEach((option) => {\n        optionsCollections.setMeta(option.key, option.value, option.group);\n      });\n      await optionsCollections.saveMeta();\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrieve the application options from the storage.\n   */\n  getOptions: {\n    validation: [\n      query('key').optional(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'VALIDATION_ERROR', ...validationErrors,\n        });\n      }\n      const options = await Option.query();\n\n      return res.status(200).sends({ options });\n    },\n  },\n};\n","import express from 'express';\nimport {\n  param,\n  query,\n  validationResult,\n} from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport Resource from '@/models/Resource';\n\nexport default {\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(jwtAuth);\n\n    router.get('/:resource_slug/columns',\n      this.resourceColumns.validation,\n      asyncMiddleware(this.resourceColumns.handler));\n\n    router.get('/:resource_slug/fields',\n      this.resourceFields.validation,\n      asyncMiddleware(this.resourceFields.handler));\n\n    return router;\n  },\n\n  /**\n   * Retrieve resource columns of the given resource.\n   */\n  resourceColumns: {\n    validation: [\n      param('resource_slug').trim().escape().exists(),\n    ],\n    async handler(req, res) {\n      const { resource_slug: resourceSlug } = req.params;\n\n      const resource = await Resource.query()\n        .where('name', resourceSlug)\n        .withGraphFetched('fields')\n        .first();\n\n      if (!resource) {\n        return res.status(400).send({\n          errors: [{ type: 'RESOURCE.SLUG.NOT.FOUND', code: 200 }],\n        });\n      }\n      const resourceFields = resource.fields\n        .filter((field) => field.columnable)\n        .map((field) => ({\n          id: field.id,\n          label: field.labelName,\n          key: field.key,\n        }));\n\n      return res.status(200).send({\n        resource_columns: resourceFields,\n        resource_slug: resourceSlug,\n      });\n    },\n  },\n\n  /**\n   * Retrieve resource fields of the given resource.\n   */\n  resourceFields: {\n    validation: [\n      param('resource_slug').trim().escape().exists(),\n      query('predefined').optional().isBoolean().toBoolean(),\n      query('builtin').optional().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const { resource_slug: resourceSlug } = req.params;\n\n      const resource = await Resource.query()\n        .where('name', resourceSlug)\n        .withGraphFetched('fields')\n        .first();\n\n      if (!resource) {\n        return res.status(400).send({\n          errors: [{ type: 'RESOURCE.SLUG.NOT.FOUND', code: 200 }],\n        });\n      }\n      return res.status(200).send({\n        resource_fields: resource.fields,\n        resource_slug: resourceSlug,\n      });\n    },\n  },\n};\n","/* eslint-disable no-unused-vars */\nimport express from 'express';\nimport { check, validationResult } from 'express-validator';\nimport { difference } from 'lodash';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport Role from '@/models/Role';\nimport Permission from '@/models/Permission';\nimport Resource from '@/models/Resource';\nimport knex from '@/database/knex';\n\nconst AccessControllSchema = [\n  {\n    resource: 'items',\n    label: 'products_services',\n    permissions: ['create', 'edit', 'delete', 'view'],\n    fullAccess: true,\n    ownControl: true,\n  },\n];\n\nconst getResourceSchema = (resource) => AccessControllSchema\n  .find((schema) => schema.resource === resource);\n\nconst getResourcePermissions = (resource) => {\n  const foundResource = getResourceSchema(resource);\n  return foundResource ? foundResource.permissions : [];\n};\n\nconst findNotFoundResources = (resourcesSlugs) => {\n  const schemaResourcesSlugs = AccessControllSchema.map((s) => s.resource);\n  return difference(resourcesSlugs, schemaResourcesSlugs);\n};\n\nconst findNotFoundPermissions = (permissions, resourceSlug) => {\n  const schemaPermissions = getResourcePermissions(resourceSlug);\n  return difference(permissions, schemaPermissions);\n};\n\nexport default {\n  /**\n   * Router constructor method.\n   */\n  router() {\n    const router = express.Router();\n\n    router.post('/',\n      this.newRole.validation,\n      asyncMiddleware(this.newRole.handler));\n\n    router.post('/:id',\n      this.editRole.validation,\n      asyncMiddleware(this.editRole.handler.bind(this)));\n\n    router.delete('/:id',\n      this.deleteRole.validation,\n      asyncMiddleware(this.deleteRole.handler));\n\n    return router;\n  },\n\n  /**\n   * Creates a new role.\n   */\n  newRole: {\n    validation: [\n      check('name').exists().trim().escape(),\n      check('description').optional().trim().escape(),\n      check('permissions').isArray({ min: 0 }),\n      check('permissions.*.resource_slug').exists().whitelist('^[a-z0-9]+(?:-[a-z0-9]+)*$'),\n      check('permissions.*.permissions').isArray({ min: 1 }),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { name, description, permissions } = req.body;\n\n      const resourcesSlugs = permissions.map((perm) => perm.resource_slug);\n      const permissionsSlugs = [];\n      const resourcesNotFound = findNotFoundResources(resourcesSlugs);\n\n      const errorReasons = [];\n      const notFoundPermissions = [];\n\n      if (resourcesNotFound.length > 0) {\n        errorReasons.push({\n          type: 'RESOURCE_SLUG_NOT_FOUND', code: 100, resources: resourcesNotFound,\n        });\n      }\n      permissions.forEach((perm) => {\n        const abilities = perm.permissions.map((ability) => ability);\n\n        // Gets the not found permissions in the schema.\n        const notFoundAbilities = findNotFoundPermissions(abilities, perm.resource_slug);\n        \n        if (notFoundAbilities.length > 0) {\n          notFoundPermissions.push({\n            resource_slug: perm.resource_slug,\n            permissions: notFoundAbilities,\n          });\n        } else {\n          const perms = perm.permissions || [];\n          perms.forEach((permission) => {\n            if (perms.indexOf(permission) !== -1) {\n              permissionsSlugs.push(permission);\n            }\n          });\n        }\n      });\n      if (notFoundPermissions.length > 0) {\n        errorReasons.push({\n          type: 'PERMISSIONS_SLUG_NOT_FOUND',\n          code: 200,\n          permissions: notFoundPermissions,\n        });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n      // Permissions.\n      const [resourcesCollection, permsCollection] = await Promise.all([\n        Resource.query((query) => { query.whereIn('name', resourcesSlugs); }).fetchAll(),\n        Permission.query((query) => { query.whereIn('name', permissionsSlugs); }).fetchAll(),\n      ]);\n\n      const notStoredResources = difference(\n        resourcesSlugs, resourcesCollection.map((s) => s.name),\n      );\n      const notStoredPermissions = difference(\n        permissionsSlugs, permsCollection.map((perm) => perm.slug),\n      );\n\n      const insertThread = [];\n\n      if (notStoredResources.length > 0) {\n        insertThread.push(knex('resources').insert([\n          ...notStoredResources.map((resource) => ({ name: resource })),\n        ]));\n      }\n      if (notStoredPermissions.length > 0) {\n        insertThread.push(knex('permissions').insert([\n          ...notStoredPermissions.map((permission) => ({ name: permission })),\n        ]));\n      }\n\n      await Promise.all(insertThread);\n\n      const [storedPermissions, storedResources] = await Promise.all([\n        Permission.query((q) => { q.whereIn('name', permissionsSlugs); }).fetchAll(),\n        Resource.query((q) => { q.whereIn('name', resourcesSlugs); }).fetchAll(),\n      ]);\n\n      const storedResourcesSet = new Map(storedResources.map((resource) => [\n        resource.attributes.name, resource.attributes.id,\n      ]));\n      const storedPermissionsSet = new Map(storedPermissions.map((perm) => [\n        perm.attributes.name, perm.attributes.id,\n      ]));\n      const role = Role.forge({ name, description });\n\n      await role.save();\n\n      const roleHasPerms = permissions.map((resource) => resource.permissions.map((perm) => ({\n        role_id: role.id,\n        resource_id: storedResourcesSet.get(resource.resource_slug),\n        permission_id: storedPermissionsSet.get(perm),\n      })));\n\n      if (roleHasPerms.length > 0) {\n        await knex('role_has_permissions').insert(roleHasPerms[0]);\n      }\n      return res.status(200).send({ id: role.get('id') });\n    },\n  },\n\n  /**\n   * Edit the give role.\n   */\n  editRole: {\n    validation: [\n      check('name').exists().trim().escape(),\n      check('description').optional().trim().escape(),\n      check('permissions').isArray({ min: 0 }),\n      check('permissions.*.resource_slug').exists().whitelist('^[a-z0-9]+(?:-[a-z0-9]+)*$'),\n      check('permissions.*.permissions').isArray({ min: 1 }),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n\n      const { id } = req.params;\n      const role = await Role.where('id', id).fetch();\n\n      if (!role) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'ROLE_NOT_FOUND', code: 100 }],\n        });\n      }\n\n      const { permissions } = req.body;\n      const errorReasons = [];\n      const permissionsSlugs = [];\n      const notFoundPermissions = [];\n\n      const resourcesSlugs = permissions.map((perm) => perm.resource_slug);\n      const resourcesNotFound = findNotFoundResources(resourcesSlugs);\n\n      if (resourcesNotFound.length > 0) {\n        errorReasons.push({\n          type: 'RESOURCE_SLUG_NOT_FOUND',\n          code: 100,\n          resources: resourcesNotFound,\n        });\n      }\n\n      permissions.forEach((perm) => {\n        const abilities = perm.permissions.map((ability) => ability);\n        // Gets the not found permissions in the schema.\n        const notFoundAbilities = findNotFoundPermissions(abilities, perm.resource_slug);\n\n        if (notFoundAbilities.length > 0) {\n          notFoundPermissions.push({\n            resource_slug: perm.resource_slug, permissions: notFoundAbilities,\n          });\n        } else {\n          const perms = perm.permissions || [];\n          perms.forEach((permission) => {\n            if (perms.indexOf(permission) !== -1) {\n              permissionsSlugs.push(permission);\n            }\n          });\n        }\n      });\n\n      if (notFoundPermissions.length > 0) {\n        errorReasons.push({\n          type: 'PERMISSIONS_SLUG_NOT_FOUND',\n          code: 200,\n          permissions: notFoundPermissions,\n        });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      // Permissions.\n      const [resourcesCollection, permsCollection] = await Promise.all([\n        Resource.query((query) => { query.whereIn('name', resourcesSlugs); }).fetchAll(),\n        Permission.query((query) => { query.whereIn('name', permissionsSlugs); }).fetchAll(),\n      ]);\n\n      const notStoredResources = difference(\n        resourcesSlugs, resourcesCollection.map((s) => s.name),\n      );\n      const notStoredPermissions = difference(\n        permissionsSlugs, permsCollection.map((perm) => perm.slug),\n      );\n      const insertThread = [];\n\n      if (notStoredResources.length > 0) {\n        insertThread.push(knex('resources').insert([\n          ...notStoredResources.map((resource) => ({ name: resource })),\n        ]));\n      }\n      if (notStoredPermissions.length > 0) {\n        insertThread.push(knex('permissions').insert([\n          ...notStoredPermissions.map((permission) => ({ name: permission })),\n        ]));\n      }\n\n      await Promise.all(insertThread);\n\n      const [storedPermissions, storedResources] = await Promise.all([\n        Permission.query((q) => { q.whereIn('name', permissionsSlugs); }).fetchAll(),\n        Resource.query((q) => { q.whereIn('name', resourcesSlugs); }).fetchAll(),\n      ]);\n\n      const storedResourcesSet = new Map(storedResources.map((resource) => [\n        resource.attributes.name, resource.attributes.id,\n      ]));\n      const storedPermissionsSet = new Map(storedPermissions.map((perm) => [\n        perm.attributes.name, perm.attributes.id,\n      ]));\n\n      await role.save();\n\n\n      const savedRoleHasPerms = await knex('role_has_permissions').where({\n        role_id: role.id,\n      });\n\n      console.log(savedRoleHasPerms);\n\n      // const roleHasPerms = permissions.map((resource) => resource.permissions.map((perm) => ({\n      //   role_id: role.id,\n      //   resource_id: storedResourcesSet.get(resource.resource_slug),\n      //   permission_id: storedPermissionsSet.get(perm),\n      // })));\n\n      // if (roleHasPerms.length > 0) {\n      //   await knex('role_has_permissions').insert(roleHasPerms[0]);\n      // }\n      return res.status(200).send({ id: role.get('id') });\n    },\n  },\n\n  deleteRole: {\n    validation: [],\n    async handler(req, res) {\n      const { id } = req.params;\n      const role = await Role.where('id', id).fetch();\n\n      if (!role) {\n        return res.boom.notFound();\n      }\n      if (role.attributes.predefined) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'ROLE_PREDEFINED', code: 100 }],\n        });\n      }\n\n      await knex('role_has_permissions')\n        .where('role_id', role.id).delete({ require: false });\n\n      await role.destroy();\n\n      return res.status(200).send();\n    },\n  },\n\n  getRole: {\n    validation: [],\n    handler(req, res) {\n      return res.status(200).send();\n    },\n  },\n};\n","import express from 'express';\n\nexport default {\n\n  router() {\n    const router = express.Router();\n\n    return router;\n  },\n};\n","import express from 'express';\nimport {\n  check,\n  query,\n  param,\n  validationResult,\n} from 'express-validator';\nimport User from '@/models/User';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport Authorization from '@/http/middleware/authorization';\n\nexport default {\n\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n    // const permit = Authorization('users');\n\n    router.use(jwtAuth);\n\n    router.post('/',\n      // permit('create'),\n      this.newUser.validation,\n      asyncMiddleware(this.newUser.handler));\n\n    router.post('/:id',\n      // permit('create', 'edit'),\n      this.editUser.validation,\n      asyncMiddleware(this.editUser.handler));\n\n    router.get('/',\n      // permit('view'),\n      this.listUsers.validation,\n      asyncMiddleware(this.listUsers.handler));\n\n    router.get('/:id',\n      // permit('view'),\n      this.getUser.validation,\n      asyncMiddleware(this.getUser.handler));\n\n    router.delete('/:id',\n      // permit('create', 'edit', 'delete'),\n      this.deleteUser.validation,\n      asyncMiddleware(this.deleteUser.handler));\n\n    return router;\n  },\n\n  /**\n   * Creates a new user.\n   */\n  newUser: {\n    validation: [\n      check('first_name').trim().escape().exists(),\n      check('last_name').trim().escape().exists(),\n      check('email').exists().isEmail(),\n      check('phone_number').optional().isMobilePhone(),\n      check('password').isLength({ min: 4 }).exists().custom((value, { req }) => {\n        if (value !== req.body.confirm_password) {\n          throw new Error(\"Passwords don't match\");\n        } else {\n          return value;\n        }\n      }),\n      check('status').exists().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const { email, phone_number: phoneNumber } = req.body;\n\n      const foundUsers = await User.query()\n        .where('email', email)\n        .orWhere('phone_number', phoneNumber);\n\n      const foundUserEmail = foundUsers.find((u) => u.email === email);\n      const foundUserPhone = foundUsers.find((u) => u.phoneNumber === phoneNumber);\n\n      const errorReasons = [];\n\n      if (foundUserEmail) {\n        errorReasons.push({ type: 'EMAIL_ALREADY_EXIST', code: 100 });\n      }\n      if (foundUserPhone) {\n        errorReasons.push({ type: 'PHONE_NUMBER_ALREADY_EXIST', code: 120 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      const user = await User.query().insert({\n        first_name: req.body.first_name,\n        last_name: req.body.last_name,\n        email: req.body.email,\n        phone_number: req.body.phone_number,\n        active: req.body.status,\n      });\n\n      return res.status(200).send({ user });\n    },\n  },\n\n  /**\n   * Edit details of the given user.\n   */\n  editUser: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n      check('first_name').exists(),\n      check('last_name').exists(),\n      check('email').exists().isEmail(),\n      check('phone_number').optional().isMobilePhone(),\n      check('password').isLength({ min: 4 }).exists().custom((value, { req }) => {\n        if (value !== req.body.confirm_password) {\n          throw new Error(\"Passwords don't match\");\n        } else {\n          return value;\n        }\n      }),\n      check('status').exists().isBoolean().toBoolean(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const user = await User.query().where('id', id).first();\n\n      if (!user) {\n        return res.boom.notFound();\n      }\n      const { email, phone_number: phoneNumber } = req.body;\n\n      const foundUsers = await User.query()\n        .whereNot('id', id)\n        .andWhere((q) => {\n          q.where('email', email);\n          q.orWhere('phone_number', phoneNumber);\n        });\n\n      const foundUserEmail = foundUsers.find((u) => u.email === email);\n      const foundUserPhone = foundUsers.find((u) => u.phoneNumber === phoneNumber);\n\n      const errorReasons = [];\n\n      if (foundUserEmail) {\n        errorReasons.push({ type: 'EMAIL_ALREADY_EXIST', code: 100 });\n      }\n      if (foundUserPhone) {\n        errorReasons.push({ type: 'PHONE_NUMBER_ALREADY_EXIST', code: 120 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      await User.query().where('id', id).update({\n        first_name: req.body.first_name,\n        last_name: req.body.last_name,\n        email: req.body.email,\n        phone_number: req.body.phone_number,\n        active: req.body.status,\n      });\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Soft deleting the given user.\n   */\n  deleteUser: {\n    validation: [],\n    async handler(req, res) {\n      const { id } = req.params;\n      const user = await User.query().where('id', id).first();\n\n      if (!user) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'USER_NOT_FOUND', code: 100 }],\n        });\n      }\n      await User.query().where('id', id).delete();\n\n      return res.status(200).send();\n    },\n  },\n\n  /**\n   * Retrieve user details of the given user id.\n   */\n  getUser: {\n    validation: [\n      param('id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { id } = req.params;\n      const user = await User.query().where('id', id).first();\n\n      if (!user) {\n        return res.boom.notFound();\n      }\n      return res.status(200).send({ user });\n    },\n  },\n\n  /**\n   * Retrieve the list of users.\n   */\n  listUsers: {\n    validation: [\n      query('page_size').optional().isNumeric().toInt(),\n      query('page').optional().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const filter = {\n        first_name: '',\n        last_name: '',\n        email: '',\n        phone_number: '',\n\n        page_size: 10,\n        page: 1,\n        ...req.query,\n      };\n\n      const users = await User.query()\n        .page(filter.page - 1, filter.page_size);\n\n      return res.status(200).send({ users });\n    },\n  },\n};\n","import { difference, pick } from 'lodash';\nimport express from 'express';\nimport {\n  check,\n  query,\n  param,\n  oneOf,\n  validationResult,\n} from 'express-validator';\nimport asyncMiddleware from '@/http/middleware/asyncMiddleware';\nimport jwtAuth from '@/http/middleware/jwtAuth';\nimport Resource from '@/models/Resource';\nimport View from '@/models/View';\nimport ViewRole from '@/models/ViewRole';\nimport ViewColumn from '@/models/ViewColumn';\nimport {\n  validateViewLogicExpression,\n} from '@/lib/ViewRolesBuilder';\n\nexport default {\n  resource: 'items',\n\n  /**\n   * Router constructor.\n   */\n  router() {\n    const router = express.Router();\n\n    router.use(jwtAuth);\n\n    router.get('/',\n      this.listViews.validation,\n      asyncMiddleware(this.listViews.handler));\n\n    router.post('/',\n      this.createView.validation,\n      asyncMiddleware(this.createView.handler));\n\n    router.post('/:view_id',\n      this.editView.validation,\n      asyncMiddleware(this.editView.handler));\n\n    router.delete('/:view_id',\n      this.deleteView.validation,\n      asyncMiddleware(this.deleteView.handler));\n\n    router.get('/:view_id',\n      asyncMiddleware(this.getView.handler));\n\n    return router;\n  },\n\n  /**\n   * List all views that associated with the given resource.\n   */\n  listViews: {\n    validation: [\n      oneOf([\n        query('resource_name').exists().trim().escape(),\n      ], [\n        query('resource_id').exists().isNumeric().toInt(),\n      ]),\n    ],\n    async handler(req, res) {\n      const filter = { ...req.query };\n\n      const resource = await Resource.query().onBuild((builder) => {\n        if (filter.resource_id) {\n          builder.where('id', filter.resource_id);\n        }\n        if (filter.resource_name) {\n          builder.where('name', filter.resource_name);\n        }\n        builder.first();\n      });\n\n      const views = await View.query().where('resource_id', resource.id);\n\n      return res.status(200).send({ views });\n    },\n  },\n\n  /**\n   * Retrieve view details of the given view id.\n   */\n  getView: {\n    validation: [\n      param('view_id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { view_id: viewId } = req.params;\n      const view = await View.query()\n        .where('id', viewId)\n        .withGraphFetched('resource')\n        .withGraphFetched('columns')\n        .withGraphFetched('roles.field')\n        .first();\n\n      if (!view) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'VIEW_NOT_FOUND', code: 100 }],\n        });\n      }\n      return res.status(200).send({ view: view.toJSON() });\n    },\n  },\n\n  /**\n   * Delete the given view of the resource.\n   */\n  deleteView: {\n    validation: [\n      param('view_id').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { view_id: viewId } = req.params;\n      const view = await View.query().findById(viewId);\n\n      if (!view) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'VIEW_NOT_FOUND', code: 100 }],\n        });\n      }\n      if (view.predefined) {\n        return res.boom.badRequest(null, {\n          errors: [{ type: 'PREDEFINED_VIEW', code: 200 }],\n        });\n      }\n      await Promise.all([\n        view.$relatedQuery('roles').delete(),\n        view.$relatedQuery('columns').delete(),\n      ]);\n      await View.query().where('id', view.id).delete();\n\n      return res.status(200).send({ id: view.id });\n    },\n  },\n\n  /**\n   * Creates a new view.\n   */\n  createView: {\n    validation: [\n      check('resource_name').exists().escape().trim(),\n      check('name').exists().escape().trim(),\n      check('logic_expression').exists().trim().escape(),\n      check('roles').isArray({ min: 1 }),\n      check('roles.*.field_key').exists().escape().trim(),\n      check('roles.*.comparator').exists(),\n      check('roles.*.value').exists(),\n      check('roles.*.index').exists().isNumeric().toInt(),\n      check('columns').exists().isArray({ min: 1 }),\n      check('columns.*.key').exists().escape().trim(),\n      check('columns.*.index').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const form = { ...req.body };\n      const resource = await Resource.query().where('name', form.resource_name).first();\n\n      if (!resource) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'RESOURCE_NOT_FOUND', code: 100 }],\n        });\n      }\n      const errorReasons = [];\n      const fieldsSlugs = form.roles.map((role) => role.field_key);\n\n      const resourceFields = await resource.$relatedQuery('fields');\n      const resourceFieldsKeys = resourceFields.map((f) => f.key);\n      const resourceFieldsKeysMap = new Map(resourceFields.map((field) => [field.key, field]));\n      const columnsKeys = form.columns.map((c) => c.key);\n\n      // The difference between the stored resource fields and submit fields keys.\n      const notFoundFields = difference(fieldsSlugs, resourceFieldsKeys);\n\n      if (notFoundFields.length > 0) {\n        errorReasons.push({ type: 'RESOURCE_FIELDS_NOT_EXIST', code: 100, fields: notFoundFields });\n      }\n      // The difference between the stored resource fields and the submit columns keys.\n      const notFoundColumns = difference(columnsKeys, resourceFieldsKeys);\n\n      if (notFoundColumns.length > 0) {\n        errorReasons.push({ type: 'COLUMNS_NOT_EXIST', code: 200, columns: notFoundColumns });\n      }\n      // Validates the view conditional logic expression.\n      if (!validateViewLogicExpression(form.logic_expression, form.roles.map((r) => r.index))) {\n        errorReasons.push({ type: 'VIEW.ROLES.LOGIC.EXPRESSION.INVALID', code: 400 });\n      }\n      if (errorReasons.length > 0) {\n        return res.boom.badRequest(null, { errors: errorReasons });\n      }\n\n      // Save view details.\n      const view = await View.query().insert({\n        name: form.name,\n        predefined: false,\n        resource_id: resource.id,\n        roles_logic_expression: form.logic_expression,\n      });\n      // Save view roles async operations.\n      const saveViewRolesOpers = [];\n\n      form.roles.forEach((role) => {\n        const fieldModel = resourceFieldsKeysMap.get(role.field_key);\n        \n        const saveViewRoleOper = ViewRole.query().insert({\n          ...pick(role, ['comparator', 'value', 'index']),\n          field_id: fieldModel.id,\n          view_id: view.id,\n        });\n        saveViewRolesOpers.push(saveViewRoleOper);\n      });\n\n      form.columns.forEach((column) => {\n        const fieldModel = resourceFieldsKeysMap.get(column.key);\n\n        const saveViewColumnOper = ViewColumn.query().insert({\n          field_id: fieldModel.id,\n          view_id: view.id,\n          index: column.index,\n        });\n        saveViewRolesOpers.push(saveViewColumnOper);\n      });\n      await Promise.all(saveViewRolesOpers);\n\n      return res.status(200).send({ id: view.id });\n    },\n  },\n\n  editView: {\n    validation: [\n      param('view_id').exists().isNumeric().toInt(),\n      check('label').exists().escape().trim(),\n      check('columns').isArray({ min: 3 }),\n      check('roles').isArray(),\n      check('roles.*.field').exists().escape().trim(),\n      check('roles.*.comparator').exists(),\n      check('roles.*.value').exists(),\n      check('roles.*.index').exists().isNumeric().toInt(),\n    ],\n    async handler(req, res) {\n      const { view_id: viewId } = req.params;\n      const validationErrors = validationResult(req);\n\n      if (!validationErrors.isEmpty()) {\n        return res.boom.badData(null, {\n          code: 'validation_error', ...validationErrors,\n        });\n      }\n      const view = await View.where('id', viewId).fetch();\n\n      if (!view) {\n        return res.boom.notFound(null, {\n          errors: [{ type: 'ROLE_NOT_FOUND', code: 100 }],\n        });\n      }\n      return res.status(200).send();\n    },\n  },\n};\n","// import OAuth2 from '@/http/controllers/OAuth2';\nimport Authentication from '@/http/controllers/Authentication';\nimport Users from '@/http/controllers/Users';\nimport Roles from '@/http/controllers/Roles';\nimport Items from '@/http/controllers/Items';\nimport ItemCategories from '@/http/controllers/ItemCategories';\nimport Accounts from '@/http/controllers/Accounts';\nimport AccountTypes from '@/http/controllers/AccountTypes';\nimport AccountOpeningBalance from '@/http/controllers/AccountOpeningBalance';\nimport Views from '@/http/controllers/Views';\nimport CustomFields from '@/http/controllers/Fields';\nimport Accounting from '@/http/controllers/Accounting';\nimport FinancialStatements from '@/http/controllers/FinancialStatements';\nimport Expenses from '@/http/controllers/Expenses';\nimport Options from '@/http/controllers/Options';\nimport Budget from '@/http/controllers/Budget';\nimport BudgetReports from '@/http/controllers/BudgetReports';\nimport Currencies from '@/http/controllers/Currencies';\nimport Customers from '@/http/controllers/Customers';\nimport Suppliers from '@/http/controllers/Suppliers';\nimport Bills from '@/http/controllers/Bills';\nimport CurrencyAdjustment from './controllers/CurrencyAdjustment';\nimport Resources from './controllers/Resources';\n// import SalesReports from '@/http/controllers/SalesReports';\n// import PurchasesReports from '@/http/controllers/PurchasesReports';\n\nexport default (app) => {\n  // app.use('/api/oauth2', OAuth2.router());\n  app.use('/api/auth', Authentication.router());\n  app.use('/api/currencies', Currencies.router());\n  app.use('/api/users', Users.router());\n  app.use('/api/roles', Roles.router());\n  app.use('/api/accounts', Accounts.router());\n  app.use('/api/account_types', AccountTypes.router());\n  app.use('/api/accounting', Accounting.router());\n  app.use('/api/accounts_opening_balances', AccountOpeningBalance.router());\n  app.use('/api/views', Views.router());\n  app.use('/api/fields', CustomFields.router());\n  app.use('/api/items', Items.router());\n  app.use('/api/item_categories', ItemCategories.router());\n  app.use('/api/expenses', Expenses.router());\n  app.use('/api/financial_statements', FinancialStatements.router());\n  app.use('/api/options', Options.router());\n  app.use('/api/budget_reports', BudgetReports.router());\n  // app.use('/api/customers', Customers.router());\n  // app.use('/api/suppliers', Suppliers.router());\n  // app.use('/api/bills', Bills.router());\n  app.use('/api/budget', Budget.router());\n  app.use('/api/resources', Resources.router());\n  // app.use('/api/currency_adjustment', CurrencyAdjustment.router());\n  // app.use('/api/reports/sales', SalesReports.router());\n  // app.use('/api/reports/purchases', PurchasesReports.router());\n};\n","const asyncMiddleware = (fn) => (req, res, next) => {\n  Promise.resolve(fn(req, res, next))\n    .catch((error) => {\n      console.log(error);\n      next(error);\n    });\n};\n\nexport default asyncMiddleware;\n","/* eslint-disable consistent-return */\nconst authorization = (resourceName) => (...permissions) => (req, res, next) => {\n  const { user } = req;\n  const onError = () => {\n    res.boom.unauthorized();\n  };\n  user.hasPermissions(resourceName, permissions)\n    .then((authorized) => {\n      if (!authorized) {\n        return onError();\n      }\n      next();\n    }).catch(onError);\n};\n\nexport default authorization;\n","/* eslint-disable consistent-return */\nimport jwt from 'jsonwebtoken';\nimport User from '@/models/User';\n// import Auth from '@/models/Auth';\n\nconst authMiddleware = (req, res, next) => {\n  const { JWT_SECRET_KEY } = process.env;\n  const token = req.headers['x-access-token'] || req.query.token;\n\n  const onError = () => {\n    // Auth.loggedOut();\n    res.status(401).send({\n      success: false,\n      message: 'unauthorized',\n    });\n  };\n\n  if (!token) {\n    return onError();\n  }\n\n  const verify = new Promise((resolve, reject) => {\n    jwt.verify(token, JWT_SECRET_KEY, async (error, decoded) => {\n      if (error) {\n        reject(error);\n      } else {\n        // eslint-disable-next-line no-underscore-dangle\n        req.user = await User.query().findById(decoded._id);\n        // Auth.setAuthenticatedUser(req.user);\n\n        if (!req.user) {\n          return onError();\n        }\n        resolve(decoded);\n      }\n    });\n  });\n\n  verify.then(() => { next(); }).catch(onError);\n};\nexport default authMiddleware;\n","\nconst OperationType = {\n  LOGIC: 'LOGIC',\n  STRING: 'STRING',\n  COMPARISON: 'COMPARISON',\n  MATH: 'MATH',\n};\n\nexport class Lexer {\n  // operation table\n  static get optable() {\n    return {\n      '=': OperationType.LOGIC,\n      '&': OperationType.LOGIC,\n      '|': OperationType.LOGIC,\n      '?': OperationType.LOGIC,\n      ':': OperationType.LOGIC,\n\n      '\\'': OperationType.STRING,\n      '\"': OperationType.STRING,\n\n      '!': OperationType.COMPARISON,\n      '>': OperationType.COMPARISON,\n      '<': OperationType.COMPARISON,\n\n      '(': OperationType.MATH,\n      ')': OperationType.MATH,\n      '+': OperationType.MATH,\n      '-': OperationType.MATH,\n      '*': OperationType.MATH,\n      '/': OperationType.MATH,\n      '%': OperationType.MATH,\n    };\n  }\n\n  /**\n   * Constructor\n   * @param {*} expression -\n   */\n  constructor(expression) {\n    this.currentIndex = 0;\n    this.input = expression;\n    this.tokenList = [];\n  }\n\n  getTokens() {\n    let tok;\n    do {\n      // read current token, so step should be -1\n      tok = this.pickNext(-1);\n      const pos = this.currentIndex;\n      switch (Lexer.optable[tok]) {\n        case OperationType.LOGIC:\n          // == && || ===\n          this.readLogicOpt(tok);\n          break;\n\n        case OperationType.STRING:\n          this.readString(tok);\n          break;\n\n        case OperationType.COMPARISON:\n          this.readCompare(tok);\n          break;\n\n        case OperationType.MATH:\n          this.receiveToken();\n          break;\n\n        default:\n          this.readValue(tok);\n      }\n\n      // if the pos not changed, this loop will go into a infinite loop, every step of while loop,\n      // we must move the pos forward\n      // so here we should throw error, for example `1 & 2`\n      if (pos === this.currentIndex && tok !== undefined) {\n        const err = new Error(`unkonw token ${tok} from input string ${this.input}`);\n        err.name = 'UnknowToken';\n        throw err;\n      }\n    } while (tok !== undefined)\n\n    return this.tokenList;\n  }\n\n  /**\n   * read next token, the index param can set next step, default go foward 1 step\n   *\n   * @param index next postion\n   */\n  pickNext(index = 0) {\n    return this.input[index + this.currentIndex + 1];\n  }\n\n  /**\n   * Store token into result tokenList, and move the pos index\n   *\n   * @param index\n   */\n  receiveToken(index = 1) {\n    const tok = this.input.slice(this.currentIndex, this.currentIndex + index).trim();\n    // skip empty string\n    if (tok) {\n      this.tokenList.push(tok);\n    }\n\n    this.currentIndex += index;\n  }\n\n  // ' or \"\n  readString(tok) {\n    let next;\n    let index = 0;\n    do {\n      next = this.pickNext(index);\n      index += 1;\n    } while (next !== tok && next !== undefined);\n    this.receiveToken(index + 1);\n  }\n\n  // > or < or >= or <= or !==\n  // tok in (>, <, !)\n  readCompare(tok) {\n    if (this.pickNext() !== '=') {\n      this.receiveToken(1);\n      return;\n    }\n    // !==\n    if (tok === '!' && this.pickNext(1) === '=') {\n      this.receiveToken(3);\n      return;\n    }\n    this.receiveToken(2);\n  }\n\n  // === or ==\n  // && ||\n  readLogicOpt(tok) {\n    if (this.pickNext() === tok) {\n      // ===\n      if (tok === '=' && this.pickNext(1) === tok) {\n        return this.receiveToken(3);\n      }\n      // == && ||\n      return this.receiveToken(2);\n    }\n    // handle as &&\n    // a ? b : c is equal to a && b || c\n    if (tok === '?' || tok === ':') {\n      return this.receiveToken(1);\n    }\n  }\n\n  readValue(tok) {\n    if (!tok) {\n      return;\n    }\n\n    let index = 0;\n    while (!Lexer.optable[tok] && tok !== undefined) {\n      tok = this.pickNext(index);\n      index += 1;\n    }\n    this.receiveToken(index);\n  }\n}\n\nexport default function token(expression) {\n  const lexer = new Lexer(expression);\n  return lexer.getTokens();\n}\n","export const OPERATION = {\n  '!': 5,\n  '*': 4,\n  '/': 4,\n  '%': 4,\n  '+': 3,\n  '-': 3,\n  '>': 2,\n  '<': 2,\n  '>=': 2,\n  '<=': 2,\n  '===': 2,\n  '!==': 2,\n  '==': 2,\n  '!=': 2,\n  '&&': 1,\n  '||': 1,\n  '?': 1,\n  ':': 1,\n};\n\n// export interface Node {\n//   left: Node | string | null;\n//   right: Node | string | null;\n//   operation: string;\n//   grouped?: boolean;\n// };\n\nexport default class Parser {\n\n  constructor(token) {\n    this.index = -1;\n    this.blockLevel = 0;\n    this.token = token;\n  }\n\n  /**\n   * \n   * @return {Node | string} =- \n   */\n  parse() {\n    let tok;\n    let root = {\n      left: null,\n      right: null,\n      operation: null,\n    };\n\n    do {\n      tok = this.parseStatement();\n\n      if (tok === null || tok === undefined) {\n        break;\n      }\n\n      if (root.left === null) {\n        root.left = tok;\n        root.operation = this.nextToken();\n\n        if (!root.operation) {\n          return tok;\n        }\n\n        root.right = this.parseStatement();\n      } else {\n        if (typeof tok !== 'string') {\n          throw new Error('operation must be string, but get ' + JSON.stringify(tok));\n        }\n        root = this.addNode(tok, this.parseStatement(), root);\n      }\n    } while (tok);\n\n    return root;\n  }\n\n  nextToken() {\n    this.index += 1;\n    return this.token[this.index];\n  }\n\n  prevToken() {\n    return this.token[this.index - 1];\n  }\n\n  /**\n   * \n   * @param {string} operation \n   * @param {Node|String|null} right \n   * @param {Node} root \n   */\n  addNode(operation, right, root) {\n    let pre = root;\n    \n    if (this.compare(pre.operation, operation) < 0 && !pre.grouped) {\n      \n      while (pre.right !== null &&\n        typeof pre.right !== 'string' &&\n        this.compare(pre.right.operation, operation) < 0 && !pre.right.grouped) {\n        pre = pre.right;\n      }\n\n      pre.right = {\n        operation,\n        left: pre.right,\n        right,\n      };\n      return root;\n    }\n    return {\n      left: pre,\n      right,\n      operation,\n    }\n  }\n\n  /**\n   * \n   * @param {String} a \n   * @param {String} b \n   */\n  compare(a, b) {\n    if (!OPERATION.hasOwnProperty(a) || !OPERATION.hasOwnProperty(b)) {\n      throw new Error(`unknow operation ${a} or ${b}`);\n    }\n    return OPERATION[a] - OPERATION[b];\n  }\n\n  /**\n   * @return string | Node | null\n   */\n  parseStatement() {\n    const token = this.nextToken();\n    if (token === '(') {\n      this.blockLevel += 1;\n      const node = this.parse();\n      this.blockLevel -= 1;\n\n      if (typeof node !== 'string') {\n        node.grouped = true;\n      }\n      return node;\n    }\n\n    if (token === ')') {\n      return null;\n    }\n\n    if (token === '!') {\n      return { left: null, operation: token, right: this.parseStatement() }\n    }\n\n    // 3 > -12 or -12 + 10\n    if (token === '-' && (OPERATION[this.prevToken()] > 0 || this.prevToken() === undefined)) {\n      return { left: '0', operation: token, right: this.parseStatement(), grouped: true };\n    }\n\n    return token;\n  }\n}\n","import { OPERATION } from './Parser';\n\nexport default class QueryParser {\n\n  constructor(tree, queries) {\n    this.tree = tree;\n    this.queries = queries;\n    this.query = null;\n  }\n\n  setQuery(query) {\n    this.query = query.clone();\n  }\n\n  parse() {\n    return this.parseNode(this.tree);\n  }\n\n  parseNode(node) {\n    if (typeof node === 'string') {\n      const nodeQuery = this.getQuery(node);\n      return (query) => { nodeQuery(query); };\n    }\n    if (OPERATION[node.operation] === undefined) {\n      throw new Error(`unknow expression ${node.operation}`);\n    }\n    const leftQuery = this.getQuery(node.left);\n    const rightQuery = this.getQuery(node.right);\n\n    switch (node.operation) {\n      case '&&':\n      case 'AND':\n      default:\n        return (nodeQuery) => nodeQuery.where((query) => {\n          query.where((q) => { leftQuery(q); });\n          query.andWhere((q) => { rightQuery(q); });\n        });\n      case '||':\n      case 'OR':\n        return (nodeQuery) => nodeQuery.where((query) => {\n          query.where((q) => { leftQuery(q); });\n          query.orWhere((q) => { rightQuery(q); });\n        });\n    }\n  }\n\n  getQuery(node) {\n    if (typeof node !== 'string' && node !== null) {\n      return this.parseNode(node);\n    }\n    const value = parseFloat(node);\n\n    if (!isNaN(value)) {\n      if (typeof this.queries[node] === 'undefined') {\n        throw new Error(`unknow query under index ${node}`);\n      }\n      return this.queries[node];\n    }\n    return null;\n  }\n}","\nexport default class MetableCollection {\n  constructor() {\n    this.metadata = [];\n    this.KEY_COLUMN = 'key';\n    this.VALUE_COLUMN = 'value';\n    this.TYPE_COLUMN = 'type';\n    this.model = null;\n    this.extraColumns = [];\n\n    this.extraQuery = (query, meta) => {\n      query.where('key', meta[this.KEY_COLUMN]);\n    };\n  }\n\n  /**\n   * Set model of this metadata collection.\n   * @param {Object} model -\n   */\n  setModel(model) {\n    this.model = model;\n  }\n\n  /**\n   * Find the given metadata key.\n   * @param {String} key -\n   * @return {object} - Metadata object.\n   */\n  findMeta(key) {\n    return this.allMetadata().find((meta) => meta.key === key);\n  }\n\n  /**\n   * Retrieve all metadata.\n   */\n  allMetadata() {\n    return this.metadata.filter((meta) => !meta.markAsDeleted);\n  }\n\n  /**\n   * Retrieve metadata of the given key.\n   * @param {String} key -\n   * @param {Mixied} defaultValue -\n   */\n  getMeta(key, defaultValue) {\n    const metadata = this.findMeta(key);\n    return metadata ? metadata.value : defaultValue || false;\n  }\n\n  /**\n   * Markes the metadata to should be deleted.\n   * @param {String} key -\n   */\n  removeMeta(key) {\n    const metadata = this.findMeta(key);\n\n    if (metadata) {\n      metadata.markAsDeleted = true;\n    }\n  }\n\n  /**\n   * Remove all meta data of the given group.\n   * @param {*} group\n   */\n  removeAllMeta(group = 'default') {\n    this.metadata = this.metadata.map((meta) => ({\n      ...meta,\n      markAsDeleted: true,\n    }));\n  }\n\n  setExtraQuery(callback) {\n    this.extraQuery = callback;\n  }\n\n  /**\n   * Set the meta data to the stack.\n   * @param {String} key -\n   * @param {String} value -\n   */\n  setMeta(key, value, payload) {\n    if (Array.isArray(key)) {\n      const metadata = key;\n\n      metadata.forEach((meta) => {\n        this.setMeta(meta.key, meta.value);\n      });\n      return;\n    }\n    const metadata = this.findMeta(key);\n\n    if (metadata) {\n      metadata.value = value;\n      metadata.markAsUpdated = true;\n    } else {\n      this.metadata.push({\n        value, key, ...payload, markAsInserted: true,\n      });\n    }\n  }\n\n  /**\n   * Saved the modified/deleted and inserted metadata.\n   */\n  async saveMeta() {\n    const inserted = this.metadata.filter((m) => (m.markAsInserted === true));\n    const updated = this.metadata.filter((m) => (m.markAsUpdated === true));\n    const deleted = this.metadata.filter((m) => (m.markAsDeleted === true));\n    const opers = [];\n\n    if (deleted.length > 0) {\n      deleted.forEach((meta) => {\n        const deleteOper = this.model.query().beforeRun((query, result) => {\n          this.extraQuery(query, meta);\n          return result;\n        }).delete();\n        opers.push(deleteOper);\n      });\n    }\n    inserted.forEach((meta) => {\n      const insertOper = this.model.query().insert({\n        [this.KEY_COLUMN]: meta.key,\n        [this.VALUE_COLUMN]: meta.value,\n        ...this.extraColumns.reduce((obj, column) => {\n          if (typeof meta[column] !== 'undefined') {\n            obj[column] = meta[column];\n          }\n          return obj;\n        }, {}),\n      });\n      opers.push(insertOper);\n    });\n    updated.forEach((meta) => {\n      const updateOper = this.model.query().onBuild((query) => {\n        this.extraQuery(query, meta);\n      }).patch({\n        [this.VALUE_COLUMN]: meta.value,\n      });\n      opers.push(updateOper);\n    });\n    await Promise.all(opers);\n  }\n\n  /**\n   * Loads the metadata from the storage.\n   * @param {String|Array} key -\n   * @param {Boolean} force -\n   */\n  async load() {\n    const metadata = await this.query();\n\n    const metadataArray = this.mapMetadataCollection(metadata);\n    metadataArray.forEach((meta) => {\n      this.metadata.push(meta);\n    });\n  }\n\n  /**\n   * Format the metadata before saving to the database.\n   * @param {String|Number|Boolean} value -\n   * @param {String} valueType -\n   * @return {String|Number|Boolean} -\n   */\n  static formatMetaValue(value, valueType) {\n    let parsedValue;\n\n    switch (valueType) {\n      case 'number':\n        parsedValue = `${value}`;\n        break;\n      case 'boolean':\n        parsedValue = value ? '1' : '0';\n        break;\n      case 'json':\n        parsedValue = JSON.stringify(parsedValue);\n        break;\n      default:\n        parsedValue = value;\n        break;\n    }\n    return parsedValue;\n  }\n\n  /**\n   * Mapping and parse metadata to collection entries.\n   * @param {Meta} attr -\n   * @param {String} parseType -\n   */\n  mapMetadata(attr, parseType = 'parse') {\n    return {\n      key: attr[this.KEY_COLUMN],\n      value: (parseType === 'parse')\n        ? MetableCollection.parseMetaValue(\n          attr[this.VALUE_COLUMN],\n          this.TYPE_COLUMN ? attr[this.TYPE_COLUMN] : false,\n        )\n        : MetableCollection.formatMetaValue(\n          attr[this.VALUE_COLUMN],\n          this.TYPE_COLUMN ? attr[this.TYPE_COLUMN] : false,\n        ),\n      ...this.extraColumns.map((extraCol) => ({\n        [extraCol]: attr[extraCol] || null,\n      })),\n    };\n  }\n\n  /**\n   * Parse the metadata to the collection.\n   * @param {Array} collection -\n   */\n  mapMetadataToCollection(metadata, parseType = 'parse') {\n    return metadata.map((model) => this.mapMetadataToCollection(model, parseType));\n  }\n\n  /**\n   * Load metadata to the metable collection.\n   * @param {Array} meta -\n   */\n  from(meta) {\n    if (Array.isArray(meta)) {\n      meta.forEach((m) => { this.from(m); });\n      return;\n    }\n    this.metadata.push(meta);\n  }\n\n\n  toArray() {\n    return this.metadata;\n  }\n\n  /**\n   * Static method to load metadata to the collection.\n   * @param {Array} meta \n   */\n  static from(meta) {\n    const collection = new MetableCollection();\n    collection.from(meta);\n\n    return collection;\n  }\n}\n","import { difference } from 'lodash';\nimport { Lexer } from '@/lib/LogicEvaluation/Lexer';\nimport Parser from '@/lib/LogicEvaluation/Parser';\nimport QueryParser from '@/lib/LogicEvaluation/QueryParser';\nimport resourceFieldsKeys from '@/data/ResourceFieldsKeys';\n\n//  const role = {\n//   compatotor: '',\n//   value: '',\n//   columnKey: '',\n//   columnSlug: '',\n//   index: 1,\n// }\n\nexport function buildRoleQuery(role) {\n  const columnName = resourceFieldsKeys[role.columnKey];\n\n  switch (role.comparator) {\n    case 'equals':\n    default:\n      return (builder) => {\n        builder.where(columnName, role.value);\n      };\n    case 'not_equal':\n    case 'not_equals':\n      return (builder) => {\n        builder.whereNot(columnName, role.value);\n      };\n  }\n}\n\n/**\n * Builds database query from stored view roles.\n *\n * @param {Array} roles -\n * @return {Function}\n */\nexport function viewRolesBuilder(roles, logicExpression = '') {\n  const rolesIndexSet = {};\n\n  roles.forEach((role) => {\n    rolesIndexSet[role.index] = buildRoleQuery(role);\n  });\n  // Lexer for logic expression.\n  const lexer = new Lexer(logicExpression);\n  const tokens = lexer.getTokens();\n\n  // Parse the logic expression.\n  const parser = new Parser(tokens);\n  const parsedTree = parser.parse();\n\n  const queryParser = new QueryParser(parsedTree, rolesIndexSet);\n  return queryParser.parse();\n}\n\n/**\n * Validates the view logic expression.\n * @param {String} logicExpression \n * @param {Array} indexes \n */\nexport function validateViewLogicExpression(logicExpression, indexes) {\n  const logicExpIndexes = logicExpression.match(/\\d+/g) || [];\n  return !difference(logicExpIndexes.map(Number), indexes).length;\n}\n\n/**\n * \n * @param {Array} roles -\n * @param {String} logicExpression -\n * @return {Boolean}\n */\nexport function validateViewRoles(roles, logicExpression) {\n  return validateViewLogicExpression(logicExpression, roles.map((r) => r.index));\n}\n\n/**\n * Mapes the view roles to view conditionals.\n * @param {Array} viewRoles -\n * @return {Array}\n */\nexport function mapViewRolesToConditionals(viewRoles) {\n  return viewRoles.map((viewRole) => ({\n    comparator: viewRole.comparator,\n    value: viewRole.value,\n    columnKey: viewRole.field.columnKey,\n    slug: viewRole.field.slug,\n    index: viewRole.index,\n  }));\n}","/* eslint-disable global-require */\nimport { Model } from 'objection';\nimport { flatten } from 'lodash';\nimport BaseModel from '@/models/Model';\nimport {viewRolesBuilder} from '@/lib/ViewRolesBuilder';\n\nexport default class Account extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'accounts';\n  }\n\n  /**\n   * Model modifiers.\n   */\n  static get modifiers() {\n    return {\n      filterAccountTypes(query, typesIds) {\n        if (typesIds.length > 0) {\n          query.whereIn('accoun_type_id', typesIds);\n        }\n      },\n      viewRolesBuilder(query, conditionals, expression) {\n        viewRolesBuilder(conditionals, expression)(query);\n      },\n    };\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const AccountType = require('@/models/AccountType');\n    const AccountBalance = require('@/models/AccountBalance');\n    const AccountTransaction = require('@/models/AccountTransaction');\n\n    return {\n      /**\n       * Account model may belongs to account type.\n       */\n      type: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: AccountType.default,\n        join: {\n          from: 'accounts.accountTypeId',\n          to: 'account_types.id',\n        },\n      },\n\n      /**\n       * Account model may has many balances accounts.\n       */\n      balance: {\n        relation: Model.HasOneRelation,\n        modelClass: AccountBalance.default,\n        join: {\n          from: 'accounts.id',\n          to: 'account_balances.accountId',\n        },\n      },\n\n      /**\n       * Account model may has many transactions.\n       */\n      transactions: {\n        relation: Model.HasManyRelation,\n        modelClass: AccountTransaction.default,\n        join: {\n          from: 'accounts.id',\n          to: 'accounts_transactions.accountId',\n        },\n      },\n    };\n  }\n\n  static collectJournalEntries(accounts) {\n    return flatten(accounts.map((account) => account.transactions.map((transaction) => ({\n      accountId: account.id,\n      ...transaction,\n      accountNormal: account.type.normal,\n    }))));\n  }\n}\n","import { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class AccountBalance extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'account_balances';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Account = require('@/models/Account');\n\n    return {\n      account: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: Account.default,\n        join: {\n          from: 'account_balance.account_id',\n          to: 'accounts.id',\n        },\n      },\n    };\n  }\n}\n","import { Model } from 'objection';\nimport moment from 'moment';\nimport BaseModel from '@/models/Model';\n\nexport default class AccountTransaction extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'accounts_transactions';\n  }\n\n  /**\n   * Model modifiers.\n   */\n  static get modifiers() {\n    return {\n      filterAccounts(query, accountsIds) {\n        if (accountsIds.length > 0) {\n          query.whereIn('account_id', accountsIds);\n        }\n      },\n      filterTransactionTypes(query, types) {\n        if (Array.isArray(types) && types.length > 0) {\n          query.whereIn('reference_type', types);\n        } else if (typeof types === 'string') {\n          query.where('reference_type', types);\n        }\n      },\n      filterDateRange(query, startDate, endDate, type = 'day') {\n        const dateFormat = 'YYYY-MM-DD HH:mm:ss';\n        const fromDate = moment(startDate).startOf(type).format(dateFormat);\n        const toDate = moment(endDate).endOf(type).format(dateFormat);\n\n        if (startDate) {\n          query.where('date', '>=', fromDate);\n        }\n        if (endDate) {\n          query.where('date', '<=', toDate);\n        }\n      },\n      filterAmountRange(query, fromAmount, toAmount) {\n        if (fromAmount) {\n          query.andWhere((q) => {\n            q.where('credit', '>=', fromAmount);\n            q.orWhere('debit', '>=', fromAmount);\n          });\n        }\n        if (toAmount) {\n          query.andWhere((q) => {\n            q.where('credit', '<=', toAmount);\n            q.orWhere('debit', '<=', toAmount);\n          });\n        }\n      },\n      sumationCreditDebit(query) {\n        query.sum('credit as credit');\n        query.sum('debit as debit');\n        query.groupBy('account_id');\n      },\n    };\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Account = require('@/models/Account');\n\n    return {\n      account: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: Account.default,\n        join: {\n          from: 'accounts_transactions.accountId',\n          to: 'accounts.id',\n        },\n      },\n    };\n  }\n}\n","// import path from 'path';\nimport { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class AccountType extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'account_types';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Account = require('@/models/Account');\n\n    return {\n      /**\n       * Account type may has many associated accounts.\n       */\n      accounts: {\n        relation: Model.HasManyRelation,\n        modelClass: Account.default,\n        join: {\n          from: 'account_types.id',\n          to: 'accounts.accountTypeId',\n        },\n      },\n    };\n  }\n}\n","import BaseModel from '@/models/Model';\n\nexport default class Budget extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'budgets';\n  }\n\n  static get virtualAttributes() {\n    return ['rangeBy', 'rangeIncrement'];\n  }\n\n  /**\n   * Model modifiers.\n   */\n  static get modifiers() {\n    return {\n      filterByYear(query, year) {\n        query.where('year', year);\n      },\n      filterByIncomeStatement(query) {\n        query.where('account_types', 'income_statement');\n      },\n      filterByProfitLoss(query) {\n        query.where('accounts_types', 'profit_loss');\n      },\n    };\n  }\n\n  get rangeBy() {\n    switch (this.period) {\n      case 'half-year':\n      case 'quarter':\n        return 'month';\n      default:\n        return this.period;\n    }\n  }\n\n  get rangeIncrement() {\n    switch (this.period) {\n      case 'half-year':\n        return 6;\n      case 'quarter':\n        return 3;\n      default:\n        return 1;\n    }\n  }\n\n  get rangeOffset() {\n    switch (this.period) {\n      case 'half-year': return 5;\n      case 'quarter': return 2;\n      default: return 0;\n    }\n  }\n}\n","import BaseModel from '@/models/Model';\n\nexport default class Budget extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'budget_entries';\n  }\n}\n","import { Model } from 'objection';\nimport BaseModel from '@/models/Model';\nimport {viewRolesBuilder} from '@/lib/ViewRolesBuilder';\nexport default class Expense extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'expenses';\n  }\n\n  static get referenceType() {\n    return 'Expense';\n  }\n\n  /**\n   * Model modifiers.\n   */\n  static get modifiers() {\n    return {\n      filterByDateRange(query, startDate, endDate) {\n        if (startDate) {\n          query.where('date', '>=', startDate);\n        }\n        if (endDate) {\n          query.where('date', '<=', endDate);\n        }\n      },\n      filterByAmountRange(query, from, to) {\n        if (from) {\n          query.where('amount', '>=', from);\n        }\n        if (to) {\n          query.where('amount', '<=', to);\n        }\n      },\n      filterByExpenseAccount(query, accountId) {\n        if (accountId) {\n          query.where('expense_account_id', accountId);\n        }\n      },\n      filterByPaymentAccount(query, accountId) {\n        if (accountId) {\n          query.where('payment_account_id', accountId);\n        }\n      },\n\n      viewRolesBuilder(query, conditionals, expression) {\n        viewRolesBuilder(conditionals, expression)(query);\n      },\n\n      orderBy(query) {\n        \n      }\n    };\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Account = require('@/models/Account');\n    const User = require('@/models/User');\n\n    return {\n      paymentAccount: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: Account.default,\n        join: {\n          from: 'expenses.paymentAccountId',\n          to: 'accounts.id',\n        },\n      },\n\n      expenseAccount: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: Account.default,\n        join: {\n          from: 'expenses.expenseAccountId',\n          to: 'accounts.id',\n        },\n      },\n\n      user: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: User.default,\n        join: {\n          from: 'expenses.userId',\n          to: 'users.id',\n        },\n      },\n    };\n  }\n}\n","import { Model } from 'objection';\nimport path from 'path';\nimport BaseModel from '@/models/Model';\n\nexport default class Item extends BaseModel {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'items';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    return {\n      /**\n       * Item may has many meta data.\n       */\n      metadata: {\n        relation: Model.HasManyRelation,\n        modelBase: path.join(__dirname, 'ItemMetadata'),\n        join: {\n          from: 'items.id',\n          to: 'items_metadata.item_id',\n        },\n      },\n\n      /**\n       * Item may belongs to cateogory model.\n       */\n      category: {\n        relation: Model.BelongsToOneRelation,\n        modelBase: path.join(__dirname, 'ItemCategory'),\n        join: {\n          from: 'items.categoryId',\n          to: 'items_categories.id',\n        },\n      },\n    };\n  }\n}\n","import path from 'path';\nimport { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class ItemCategory extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'items_categories';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    return {\n      /**\n       * Item category may has many items.\n       */\n      items: {\n        relation: Model.HasManyRelation,\n        modelBase: path.join(__dirname, 'Item'),\n        join: {\n          from: 'items_categories.item_id',\n          to: 'items.id',\n        },\n      },\n    };\n  }\n}\n","import BaseModel from '@/models/Model';\n\nexport default class JournalEntry extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'manual_journals';\n  }\n}\n","import BaseModel from '@/models/Model';\n\nexport default class ManualJournal extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'manual_journals';\n  }\n}\n","import { Model } from 'objection';\nimport {transform, snakeCase} from 'lodash';\nimport {mapKeysDeep} from '@/utils';\n\nexport default class ModelBase extends Model {\n  static get collection() {\n    return Array;\n  }\n\n  static query(...args) {\n    return super.query(...args).runAfter((result) => {\n      if (Array.isArray(result)) {\n        return this.collection.from(result);\n      }\n      return result;\n    });\n  }\n\n  $formatJson(json, opt) {\n    const transformed = mapKeysDeep(json, (value, key) => {\n      return snakeCase(key);\n    });\n    const parsedJson = super.$formatJson(transformed, opt);\n\n    return parsedJson;\n  }\n}\n","import { mixin } from 'objection';\nimport BaseModel from '@/models/Model';\nimport MetableCollection from '@/lib/Metable/MetableCollection';\n\nexport default class Option extends mixin(BaseModel, [mixin]) {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'options';\n  }\n\n  /**\n   * Override the model query.\n   * @param  {...any} args -\n   */\n  static query(...args) {\n    return super.query(...args).runAfter((result) => {\n      if (result instanceof MetableCollection) {\n        result.setModel(Option);\n      }\n      return result;\n    });\n  }\n\n  static get collection() {\n    return MetableCollection;\n  }\n}\n","import Model from '@/models/Model';\n\nexport default class PasswordResets extends Model {\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'password_resets';\n  }\n}\n","import { Model } from 'objection';\nimport path from 'path';\nimport BaseModel from '@/models/Model';\n\nexport default class Permission extends BaseModel {\n  /**\n   * Table name of Role model.\n   * @type {String}\n   */\n  static get tableName() {\n    return 'permissions';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    return {\n      /**\n       * Permission model may belongs to role model.\n       */\n      role: {\n        relation: Model.BelongsToOneRelation,\n        modelBase: path.join(__dirname, 'Role'),\n        join: {\n          from: 'permissions.role_id',\n          to: 'roles.id',\n        },\n      },\n\n      // resource: {\n      //   relation: Model.BelongsToOneRelation,\n      //   modelBase: path.join(__dirname, 'Resource'),\n      //   join: {\n      //     from: 'permissions.',\n      //     to: '',\n      //   }\n      // }\n    };\n  }\n}\n","import path from 'path';\nimport { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class Resource extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'resources';\n  }\n\n  /**\n   * Timestamp columns.\n   */\n  static get hasTimestamps() {\n    return false;\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const View = require('@/models/View');\n    const ResourceField = require('@/models/ResourceField');\n    const Permission = require('@/models/Permission');\n\n    return {\n      /**\n       * Resource model may has many views.\n       */\n      views: {\n        relation: Model.HasManyRelation,\n        modelClass: View.default,\n        join: {\n          from: 'resources.id',\n          to: 'views.resourceId',\n        },\n      },\n\n      /**\n       * Resource model may has many fields.\n       */\n      fields: {\n        relation: Model.HasManyRelation,\n        modelClass: ResourceField.default,\n        join: {\n          from: 'resources.id',\n          to: 'resource_fields.resourceId',\n        },\n      },\n\n      /**\n       * Resource model may has many associated permissions.\n       */\n      permissions: {\n        relation: Model.ManyToManyRelation,\n        modelClass: Permission.default,\n        join: {\n          from: 'resources.id',\n          through: {\n            from: 'role_has_permissions.resourceId',\n            to: 'role_has_permissions.permissionId',\n          },\n          to: 'permissions.id',\n        },\n      },\n    };\n  }\n}\n","import { snakeCase } from 'lodash';\nimport { Model } from 'objection';\nimport path from 'path';\nimport BaseModel from '@/models/Model';\n\nexport default class ResourceField extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'resource_fields';\n  }\n\n  static get jsonAttributes() {\n    return ['options'];\n  }\n\n  /**\n   * Model modifiers.\n   */\n  static get modifiers() {\n    return {\n      whereNotPredefined(query) {\n        query.whereNot('predefined', true);\n      },\n    };\n  }\n\n  /**\n   * Timestamp columns.\n   */\n  static get hasTimestamps() {\n    return false;\n  }\n\n  /**\n   * Virtual attributes.\n   */\n  static get virtualAttributes() {\n    return ['key'];\n  }\n\n  /**\n   * Resource field key.\n   */\n  key() {\n    return snakeCase(this.labelName);\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    return {\n      /**\n       * Resource field may belongs to resource model.\n       */\n      resource: {\n        relation: Model.BelongsToOneRelation,\n        modelBase: path.join(__dirname, 'Resource'),\n        join: {\n          from: 'resource_fields.resource_id',\n          to: 'resources.id',\n        },\n      },\n    };\n  }\n}\n","import { Model } from 'objection';\nimport path from 'path';\nimport BaseModel from '@/models/Model';\nimport ResourceFieldMetadataCollection from '@/collection/ResourceFieldMetadataCollection';\n\nexport default class ResourceFieldMetadata extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'resource_custom_fields_metadata';\n  }\n\n  /**\n   * Override the resource field metadata collection.\n   */\n  static get collection() {\n    return ResourceFieldMetadataCollection;\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    return {\n      /**\n       * Resource field may belongs to resource model.\n       */\n      resource: {\n        relation: Model.BelongsToOneRelation,\n        modelBase: path.join(__dirname, 'Resource'),\n        join: {\n          from: 'resource_fields.resource_id',\n          to: 'resources.id',\n        },\n      },\n    };\n  }\n}\n","import { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class Role extends BaseModel {\n  /**\n   * Table name of Role model.\n   * @type {String}\n   */\n  static get tableName() {\n    return 'roles';\n  }\n\n  /**\n   * Timestamp columns.\n   */\n  static get hasTimestamps() {\n    return false;\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Permission = require('@/models/Permission');\n    const Resource = require('@/models/Resource');\n    const User = require('@/models/User');\n    const ResourceField = require('@/models/ResourceField');\n\n    return {\n      /**\n       * Role may has many permissions.\n       */\n      permissions: {\n        relation: Model.ManyToManyRelation,\n        modelClass: Permission.default,\n        join: {\n          from: 'roles.id',\n          through: {\n            from: 'role_has_permissions.roleId',\n            to: 'role_has_permissions.permissionId',\n          },\n          to: 'permissions.id',\n        },\n      },\n\n      /**\n       * Role may has many resources.\n       */\n      resources: {\n        relation: Model.ManyToManyRelation,\n        modelClass: Resource.default,\n        join: {\n          from: 'roles.id',\n          through: {\n            from: 'role_has_permissions.roleId',\n            to: 'role_has_permissions.resourceId',\n          },\n          to: 'resources.id',\n        },\n      },\n\n      /**\n       * Role may has resource field.\n       */\n      field: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: ResourceField.default,\n        join: {\n          from: 'roles.fieldId',\n          to: 'resource_fields.id',\n        }\n      },\n\n      /**\n       * Role may has many associated users.\n       */\n      users: {\n        relation: Model.ManyToManyRelation,\n        modelClass: User.default,\n        join: {\n          from: 'roles.id',\n          through: {\n            from: 'user_has_roles.roleId',\n            to: 'user_has_roles.userId',\n          },\n          to: 'users.id',\n        },\n      },\n    };\n  }\n}\n","import bcrypt from 'bcryptjs';\nimport { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n// import PermissionsService from '@/services/PermissionsService';\n\nexport default class User extends BaseModel {\n  // ...PermissionsService\n\n  static get virtualAttributes() {\n    return ['fullName'];\n  }\n\n  /**\n   * Table name\n   */\n  static get tableName() {\n    return 'users';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Role = require('@/models/Role');\n\n    return {\n      roles: {\n        relation: Model.ManyToManyRelation,\n        modelClass: Role.default,\n        join: {\n          from: 'users.id',\n          through: {\n            from: 'user_has_roles.userId',\n            to: 'user_has_roles.roleId',\n          },\n          to: 'roles.id',\n        },\n      },\n    };\n  }\n\n  /**\n   * Verify the password of the user.\n   * @param  {String} password - The given password.\n   * @return {Boolean}\n   */\n  verifyPassword(password) {\n    return bcrypt.compareSync(password, this.password);\n  }\n\n  fullName() {\n    return `${this.firstName} ${this.lastName}`;\n  }\n}\n","import path from 'path';\nimport { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class View extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'views';\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const Resource = require('@/models/Resource');\n    const ViewColumn = require('@/models/ViewColumn');\n    const ViewRole = require('@/models/ViewRole');\n\n    return {\n      /**\n       * View model belongs to resource model.\n       */\n      resource: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: Resource.default,\n        join: {\n          from: 'views.resourceId',\n          to: 'resources.id',\n        },\n      },\n\n      /**\n       * View model may has many columns.\n       */\n      columns: {\n        relation: Model.HasManyRelation,\n        modelClass: ViewColumn.default,\n        join: {\n          from: 'views.id',\n          to: 'view_has_columns.viewId',\n        },\n      },\n\n      /**\n       * View model may has many view roles.\n       */\n      roles: {\n        relation: Model.HasManyRelation,\n        modelClass: ViewRole.default,\n        join: {\n          from: 'views.id',\n          to: 'view_roles.viewId',\n        },\n      },\n    };\n  }\n}\n","import BaseModel from '@/models/Model';\n\nexport default class ViewColumn extends BaseModel {\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'view_has_columns';\n  }\n\n  /**\n   * Timestamp columns.\n   */\n  static get hasTimestamps() {\n    return false;\n  }\n}\n","import { Model } from 'objection';\nimport BaseModel from '@/models/Model';\n\nexport default class ViewRole extends BaseModel {\n\n  /**\n   * Virtual attributes.\n   */\n  static get virtualAttributes() {\n    return ['comparators'];\n  }\n\n  static get comparators() {\n    return [\n      'equals', 'not_equal', 'contains', 'not_contain',\n    ];\n  }\n\n  /**\n   * Table name.\n   */\n  static get tableName() {\n    return 'view_roles';\n  }\n\n  /**\n   * Timestamp columns.\n   */\n  static get hasTimestamps() {\n    return false;\n  }\n\n  /**\n   * Relationship mapping.\n   */\n  static get relationMappings() {\n    const ResourceField = require('@/models/ResourceField');\n    const View = require('@/models/View');\n\n    return {\n      /**\n       * View role model may belongs to view model.\n       */\n      view: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: View.default,\n        join: {\n          from: 'view_roles.viewId',\n          to: 'views.id',\n        },\n      },\n\n      /**\n       * View role model may belongs to resource field model.\n       */\n      field: {\n        relation: Model.BelongsToOneRelation,\n        modelClass: ResourceField.default,\n        join: {\n          from: 'view_roles.fieldId',\n          to: 'resource_fields.id',\n        },\n      },\n    };\n  }\n}\n","import { Model } from 'objection';\nimport knex from '@/database/knex';\n\n// Bind all Models to a knex instance. If you only have one database in\n// your server this is all you have to do. For multi database systems, see\n// the Model.bindKnex() method.\nModel.knex(knex);\n","import errorHandler from 'errorhandler';\nimport app from '@/app';\n\napp.use(errorHandler);\n\nconst server = app.listen(app.get('port'), () => {\n  console.log(\n    '  App is running at http://localhost:%d in %s mode',\n    app.get('port'),\n    app.get('env'),\n  );\n  console.log('  Press CTRL-C to stop');\n});\n\nexport default server;\n","\nexport default class JournalEntry {\n  constructor(entry) {\n    const defaults = {\n      credit: 0,\n      debit: 0,\n    };\n    this.entry = { ...defaults, ...entry };\n  }\n}\n","import { pick } from 'lodash';\nimport moment from 'moment';\nimport JournalEntry from '@/services/Accounting/JournalEntry';\nimport AccountTransaction from '@/models/AccountTransaction';\nimport AccountBalance from '@/models/AccountBalance';\nimport {promiseSerial} from '@/utils';\n\nexport default class JournalPoster {\n  /**\n   * Journal poster constructor.\n   */\n  constructor() {\n    this.entries = [];\n    this.balancesChange = {};\n  }\n\n  /**\n   * Writes the credit entry for the given account.\n   * @param {JournalEntry} entry -\n   */\n  credit(entryModel) {\n    if (entryModel instanceof JournalEntry === false) {\n      throw new Error('The entry is not instance of JournalEntry.');\n    }\n    this.entries.push(entryModel.entry);\n    this.setAccountBalanceChange(entryModel.entry, 'credit');\n  }\n\n  /**\n   * Writes the debit entry for the given account.\n   * @param {JournalEntry} entry -\n   */\n  debit(entryModel) {\n    if (entryModel instanceof JournalEntry === false) {\n      throw new Error('The entry is not instance of JournalEntry.');\n    }\n    this.entries.push(entryModel.entry);\n    this.setAccountBalanceChange(entryModel.entry, 'debit');\n  }\n\n  /**\n   * Sets account balance change.\n   * @param {JournalEntry} entry\n   * @param {String} type\n   */\n  setAccountBalanceChange(entry, type) {\n    if (!this.balancesChange[entry.account]) {\n      this.balancesChange[entry.account] = 0;\n    }\n    let change = 0;\n\n    if (entry.accountNormal === 'credit') {\n      change = (type === 'credit') ? entry.credit : -1 * entry.debit;\n    } else if (entry.accountNormal === 'debit') {\n      change = (type === 'debit') ? entry.debit : -1 * entry.credit;\n    }\n    this.balancesChange[entry.account] += change;\n  }\n\n  /**\n   * Mapping the balance change to list.\n   */\n  mapBalanceChangesToList() {\n    const mappedList = [];\n\n    Object.keys(this.balancesChange).forEach((accountId) => {\n      const balance = this.balancesChange[accountId];\n\n      mappedList.push({\n        account_id: accountId,\n        amount: balance,\n      });\n    });\n    return mappedList;\n  }\n\n  /**\n   * Saves the balance change of journal entries.\n   */\n  async saveBalance() {\n    const balancesList = this.mapBalanceChangesToList();\n    const balanceUpdateOpers = [];\n    const balanceInsertOpers = [];\n    const balanceFindOneOpers = [];\n    let balanceAccounts = [];\n\n    balancesList.forEach((balance) => {\n      const oper = AccountBalance.query().findOne('account_id', balance.account_id);\n      balanceFindOneOpers.push(oper);\n    });\n    balanceAccounts = await Promise.all(balanceFindOneOpers);\n\n    balancesList.forEach((balance) => {\n      const method = balance.amount < 0 ? 'decrement' : 'increment';\n\n      // Detarmine if the account balance is already exists or not.\n      const foundAccBalance = balanceAccounts.some((account) => (\n        account && account.account_id === balance.account_id\n      ));\n      if (foundAccBalance) {\n        const query = AccountBalance\n          .query()[method]('amount', Math.abs(balance.amount))\n          .where('account_id', balance.account_id);\n\n        balanceUpdateOpers.push(query);\n      } else {\n        const query = AccountBalance.query().insert({\n          account_id: balance.account_id,\n          amount: balance.amount,\n          currency_code: 'USD',\n        });\n        balanceInsertOpers.push(query);\n      }\n    });\n    await Promise.all([\n      ...balanceUpdateOpers, ...balanceInsertOpers,\n    ]);\n  }\n\n  /**\n   * Saves the stacked journal entries to the storage.\n   */\n  async saveEntries() {\n    const saveOperations = [];\n\n    this.entries.forEach((entry) => {\n      const oper = AccountTransaction.query().insert({\n        accountId: entry.account,\n        ...pick(entry, ['credit', 'debit', 'transactionType', 'date', 'userId',\n          'referenceType', 'referenceId', 'note']),\n      });\n      saveOperations.push(() => oper);\n    });\n    await promiseSerial(saveOperations);\n  }\n\n  /**\n   * Reverses the stacked journal entries.\n   */\n  reverseEntries() {\n    const reverseEntries = [];\n\n    this.entries.forEach((entry) => {\n      const reverseEntry = { ...entry };\n\n      if (entry.credit) {\n        reverseEntry.debit = entry.credit;\n      }\n      if (entry.debit) {\n        reverseEntry.credit = entry.debit;\n      }\n      reverseEntries.push(reverseEntry);\n    });\n    this.entries = reverseEntries;\n  }\n\n  /**\n   * Delete the given or all stacked entries.\n   * @param {Array} ids -\n   */\n  async deleteEntries(ids) {\n    const entriesIds = ids || this.entries.map((e) => e.id);\n\n    if (entriesIds.length > 0) {\n      await AccountTransaction.query().whereIn('id', entriesIds).delete();\n    }\n  }\n\n  /**\n   * Retrieve the closing balance for the given account and closing date.\n   * @param {Number} accountId -\n   * @param {Date} closingDate -\n   */\n  getClosingBalance(accountId, closingDate, dateType = 'day') {\n    let closingBalance = 0;\n    const momentClosingDate = moment(closingDate);\n\n    this.entries.forEach((entry) => {\n      // Can not continue if not before or event same closing date.\n      if ((!momentClosingDate.isAfter(entry.date, dateType)\n        && !momentClosingDate.isSame(entry.date, dateType))\n        || (entry.account !== accountId && accountId)) {\n        return;\n      }\n      if (entry.accountNormal === 'credit') {\n        closingBalance += (entry.credit) ? entry.credit : -1 * entry.debit;\n      } else if (entry.accountNormal === 'debit') {\n        closingBalance += (entry.debit) ? entry.debit : -1 * entry.credit;\n      }\n    });\n    return closingBalance;\n  }\n\n  /**\n   * Retrieve the credit/debit sumation for the given account and date.\n   * @param {Number} account -\n   * @param {Date|String} closingDate -\n   */\n  getTrialBalance(accountId, closingDate, dateType) {\n    const momentClosingDate = moment(closingDate);\n    const result = {\n      credit: 0,\n      debit: 0,\n      balance: 0,\n    };\n    this.entries.forEach((entry) => {\n      if ((!momentClosingDate.isAfter(entry.date, dateType)\n        && !momentClosingDate.isSame(entry.date, dateType))\n        || (entry.account !== accountId && accountId)) {\n        return;\n      }\n      result.credit += entry.credit;\n      result.debit += entry.debit;\n\n      if (entry.accountNormal === 'credit') {\n        result.balance += (entry.credit) ? entry.credit : -1 * entry.debit;\n      } else if (entry.accountNormal === 'debit') {\n        result.balance += (entry.debit) ? entry.debit : -1 * entry.credit;\n      }\n    });\n    return result;\n  }\n\n  /**\n   * Load fetched accounts journal entries.\n   * @param {Array} entries -\n   */\n  loadEntries(entries) {\n    entries.forEach((entry) => {\n      this.entries.push({\n        ...entry,\n        account: entry.account ? entry.account.id : entry.accountId,\n        accountNormal: (entry.account && entry.account.type)\n          ? entry.account.type.normal : entry.accountNormal,\n      });\n    });\n  }\n\n  static loadAccounts() {\n\n  }\n}\n","import Resource from '@/models/Resource';\nimport ResourceField from '@/models/ResourceField';\nimport ResourceFieldMetadata from '@/models/ResourceFieldMetadata';\nimport ResourceFieldMetadataCollection from '@/collection/ResourceFieldMetadataCollection';\n\nexport default class ResourceCustomFieldRepository {\n  /**\n   * Class constructor.\n   */\n  constructor(model) {\n    if (typeof model === 'function') {\n      this.resourceName = model.name;\n    } else if (typeof model === 'string') {\n      this.resourceName = model;\n    }\n    // Custom fields of the given resource.\n    this.customFields = [];\n    this.filledCustomFields = {};\n\n    // metadata of custom fields of the given resource.\n    this.fieldsMetadata = {};\n    this.resource = {};\n  }\n\n  /**\n   * Fetches metadata of custom fields of the given resource.\n   * @param {Integer} id - Resource item id.\n   */\n  async fetchCustomFieldsMetadata(id) {\n    if (typeof id === 'undefined') {\n      throw new Error('Please define the resource item id.');\n    }\n    if (!this.resource) {\n      throw new Error('Target resource model is not found.');\n    }\n    const metadata = await ResourceFieldMetadata.query()\n      .where('resource_id', this.resource.id)\n      .where('resource_item_id', id);\n\n    this.fieldsMetadata[id] = metadata;\n  }\n\n  /**\n   * Load resource.\n   */\n  async loadResource() {\n    const resource = await Resource.query().where('name', this.resourceName).first();\n\n    if (!resource) {\n      throw new Error('There is no stored resource in the storage with the given model name.');\n    }\n    this.setResource(resource);\n  }\n\n  /**\n   * Load metadata of the resource.\n   */\n  async loadResourceCustomFields() {\n    if (typeof this.resource.id === 'undefined') {\n      throw new Error('Please fetch resource details before fetch custom fields of the resource.');\n    }\n    const customFields = await ResourceField.query()\n      .where('resource_id', this.resource.id)\n      .modify('whereNotPredefined');\n\n    this.setResourceCustomFields(customFields);\n  }\n\n  /**\n   * Sets resource model.\n   * @param {Resource} resource -\n   */\n  setResource(resource) {\n    this.resource = resource;\n  }\n\n  /**\n   * Sets resource custom fields collection.\n   * @param {Array} customFields -\n   */\n  setResourceCustomFields(customFields) {\n    this.customFields = customFields;\n  }\n\n  /**\n   * Retrieve metadata of the resource custom fields.\n   * @param {Integer} itemId -\n   */\n  getMetadata(itemId) {\n    return this.fieldsMetadata[itemId] || this.fieldsMetadata;\n  }\n\n  /**\n   * Fill metadata of the custom fields that associated to the resource.\n   * @param {Inter} id - Resource item id.\n   * @param {Array} attributes -\n   */\n  fillCustomFields(id, attributes) {\n    if (typeof this.filledCustomFields[id] === 'undefined') {\n      this.filledCustomFields[id] = [];\n    }\n    attributes.forEach((attr) => {\n      this.filledCustomFields[id].push(attr);\n\n      if (!this.fieldsMetadata[id]) {\n        this.fieldsMetadata[id] = new ResourceFieldMetadataCollection();\n      }\n      this.fieldsMetadata[id].setMeta(attr.key, attr.value, {\n        resource_id: this.resource.id,\n        resource_item_id: id,\n      });\n    });\n  }\n\n  /**\n   * Saves the instered, updated and deleted  custom fields metadata.\n   * @param {Integer} id - Optional resource item id.\n   */\n  async saveCustomFields(id) {\n    if (id) {\n      if (typeof this.fieldsMetadata[id] === 'undefined') {\n        throw new Error('There is no resource item with the given id.');\n      }\n      await this.fieldsMetadata[id].saveMeta();\n    } else {\n      const opers = [];\n      this.fieldsMetadata.forEach((metadata) => {\n        const oper = metadata.saveMeta();\n        opers.push(oper);\n      });\n      await Promise.all(opers);\n    }\n  }\n\n  /**\n   * Validates the exist custom fields.\n   */\n  validateExistCustomFields() {\n\n  }\n\n  toArray() {\n    return this.fieldsMetadata.toArray();\n  }\n\n  async load() {\n    await this.loadResource();\n    await this.loadResourceCustomFields();\n  }\n\n  static forgeMetadataCollection() {\n\n  }\n}\n","import Moment from 'moment';\nimport { extendMoment } from 'moment-range';\n\nconst moment = extendMoment(Moment);\n\nexport default moment;\n","import nodemailer from 'nodemailer';\n\n// create reusable transporter object using the default SMTP transport\nconst transporter = nodemailer.createTransport({\n  host: process.env.MAIL_HOST,\n  port: Number(process.env.MAIL_PORT),\n  secure: process.env.MAIL_SECURE === 'true', // true for 465, false for other ports\n  auth: {\n    user: process.env.MAIL_USERNAME,\n    pass: process.env.MAIL_PASSWORD,\n  },\n});\n\nexport default transporter;\n","import bcrypt from 'bcryptjs';\nimport moment from 'moment';\nimport _ from 'lodash';\nconst { map, isArray, isPlainObject, mapKeys, mapValues } = require('lodash')\n\n\nconst hashPassword = (password) => new Promise((resolve) => {\n  bcrypt.genSalt(10, (error, salt) => {\n    bcrypt.hash(password, salt, (err, hash) => { resolve(hash); });\n  });\n});\n\nconst origin = (request) => `${request.protocol}://${request.hostname}`;\n\nconst dateRangeCollection = (fromDate, toDate, addType = 'day', increment = 1) => {\n  const collection = [];\n  const momentFromDate = moment(fromDate);\n  let dateFormat = '';\n\n  switch (addType) {\n    case 'day':\n    default:\n      dateFormat = 'YYYY-MM-DD'; break;\n    case 'month':\n    case 'quarter':\n      dateFormat = 'YYYY-MM'; break;\n    case 'year':\n      dateFormat = 'YYYY'; break;\n  }\n  for (let i = momentFromDate;\n    (i.isBefore(toDate, addType) || i.isSame(toDate, addType));\n    i.add(increment, `${addType}s`)) {\n    collection.push(i.endOf(addType).format(dateFormat));\n  }\n  return collection;\n};\n\nconst dateRangeFormat = (rangeType) => {\n  switch (rangeType) {\n    case 'year':\n      return 'YYYY';\n    case 'month':\n    case 'quarter':\n    default:\n      return 'YYYY-MM';\n  }\n};\n\n\nconst mapKeysDeep = (obj, cb) => {\n  if (_.isArray(obj)) {\n      return obj.map(innerObj => mapKeysDeep(innerObj, cb));\n  }\n  else if (_.isObject(obj)) {\n      return _.mapValues(\n          _.mapKeys(obj, cb),\n          val => mapKeysDeep(val, cb),\n      )\n  } else {\n      return obj;\n  }\n}\n\nconst mapValuesDeep = (v, callback) => (\n  _.isObject(v)\n    ? _.mapValues(v, v => mapValuesDeep(v, callback))\n    : callback(v));\n\n\nconst promiseSerial = (funcs) => {\n  return funcs.reduce((promise, func) => promise.then((result) => func().then(Array.prototype.concat.bind(result))),\n    Promise.resolve([]));\n}\n\nexport {\n  hashPassword,\n  origin,\n  dateRangeCollection,\n  dateRangeFormat,\n  mapValuesDeep,\n  mapKeysDeep,\n  promiseSerial,\n};\n","module.exports = require(\"@babel/plugin-transform-runtime\");","module.exports = require(\"@babel/runtime/helpers/asyncToGenerator\");","module.exports = require(\"@babel/runtime/helpers/classCallCheck\");","module.exports = require(\"@babel/runtime/helpers/createClass\");","module.exports = require(\"@babel/runtime/helpers/defineProperty\");","module.exports = require(\"@babel/runtime/helpers/get\");","module.exports = require(\"@babel/runtime/helpers/getPrototypeOf\");","module.exports = require(\"@babel/runtime/helpers/inherits\");","module.exports = require(\"@babel/runtime/helpers/possibleConstructorReturn\");","module.exports = require(\"@babel/runtime/helpers/slicedToArray\");","module.exports = require(\"@babel/runtime/helpers/toConsumableArray\");","module.exports = require(\"@babel/runtime/regenerator\");","module.exports = require(\"bcryptjs\");","module.exports = require(\"dotenv\");","module.exports = require(\"errorhandler\");","module.exports = require(\"express\");","module.exports = require(\"express-boom\");","module.exports = require(\"express-validator\");","module.exports = require(\"fs\");","module.exports = require(\"helmet\");","module.exports = require(\"i18n\");","module.exports = require(\"jsonwebtoken\");","module.exports = require(\"knex\");","module.exports = require(\"lodash\");","module.exports = require(\"moment\");","module.exports = require(\"moment-range\");","module.exports = require(\"mustache\");","module.exports = require(\"nodemailer\");","module.exports = require(\"objection\");","module.exports = require(\"path\");"],"mappings":";AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AClFA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;;;;;;;;;;ACJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AChDA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;ACxEA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC3EA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACXA;AACA;AACA;AACA;;;;;;;;;;;;ACJA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACVA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC/IA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC7BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AApGA;AAsGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzOA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAIA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAnBA;AACA;AACA;AAqBA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AChYA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AChMA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AClPA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AA7EA;AA+EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA;;;;;;;;;;;;AC5GA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACZA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AClEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAIA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAhDA;AAkDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AA7DA;AACA;AA+DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAvFA;AAyFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACtfA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACtOA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC1iBA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACpLA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAnCA;AAqCA;AACA;AACA;AAvCA;AAyCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC9QA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACtEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzFA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAhIA;AACA;AACA;AACA;AAkIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACvVA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC/OA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AAOA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACvQA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACpDA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACfA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AADA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AA3BA;AA+BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AA9EA;AAmFA;AACA;AACA;AACA;AACA;;;;AAvFA;AA4FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAtGA;AAwGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAlHA;AAoHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAjIA;AAmIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC3KA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC7JA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;AA0BA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AApEA;AAwEA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACjPA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACxFA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAzEA;;AAOA;;AAgBA;;AASA;;AAYA;;AAYA;;;;;;;;;;;;AClEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC3BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC/EA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC/BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AApDA;;AAWA;;;;;;;;;;;;ACjBA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC5FA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACzCA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC7BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzBA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAfA;AAkBA;AACA;;;;;;;;;;;;AC3BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AApBA;AAsBA;;;;;;;;;;;;;ACvCA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACpEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;AA3CA;;AAGA;;AAWA;;AAWA;;AAOA;;AAmBA;AAKA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AClEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACrCA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzFA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AA5CA;;AAOA;;AAOA;;;;;;;;;;;;ACtBA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzDA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;;;;;;;;;;;;ACfA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AChEA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACNA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACRA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AChPA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACxJA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACLA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACbA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;ACzEA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;;;;;;;;ACAA;;;;A","sourceRoot":""} \ No newline at end of file diff --git a/server/package-lock.json b/server/package-lock.json index 795c59d81..398d78773 100644 --- a/server/package-lock.json +++ b/server/package-lock.json @@ -2228,14 +2228,6 @@ "object-visit": "^1.0.0" } }, - "collections": { - "version": "5.1.9", - "resolved": "https://registry.npmjs.org/collections/-/collections-5.1.9.tgz", - "integrity": "sha512-omsKk7VkxuYKsxKG9jdyNtqTVJXZuwLnK53lc7M7AW4cXKcEyK3F+4cfQFVOy4ivT+UWhJlvClrJI7qF2Ts6iA==", - "requires": { - "weak-map": "~1.0.x" - } - }, "color-convert": { "version": "1.9.3", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", @@ -3915,8 +3907,7 @@ "fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", - "dev": true + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" }, "fsevents": { "version": "1.2.9", @@ -4788,6 +4779,26 @@ "integrity": "sha1-7AbBDgo0wPL68Zn3/X/Hj//QPHM=", "dev": true }, + "i18n": { + "version": "0.8.5", + "resolved": "https://registry.npmjs.org/i18n/-/i18n-0.8.5.tgz", + "integrity": "sha512-6UgLbhJGgn4XFeuZc/dDdrrri0ij24EK4hxv4Pbi5hloYAZ1B2+0eQchEryBFezLKYOHhVGV/5+H4i0oxng94w==", + "requires": { + "debug": "*", + "make-plural": "^6.0.1", + "math-interval-parser": "^2.0.1", + "messageformat": "^2.3.0", + "mustache": "*", + "sprintf-js": "^1.1.2" + }, + "dependencies": { + "sprintf-js": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.2.tgz", + "integrity": "sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==" + } + } + }, "iconv-lite": { "version": "0.4.24", "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", @@ -4869,7 +4880,6 @@ "version": "1.0.6", "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", - "dev": true, "requires": { "once": "^1.3.0", "wrappy": "1" @@ -5570,6 +5580,36 @@ } } }, + "knex-db-manager": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/knex-db-manager/-/knex-db-manager-0.6.1.tgz", + "integrity": "sha512-H1IBb/zUwFXYyok8n/WmtRNV/DcZ77lV9qXqjpoF4otFPgf3W85/l79Dq/8gOE+0WnawLsYKXD8Hdz5VsaHIzQ==", + "requires": { + "bluebird": "^3.7.2", + "glob": "^7.1.6", + "lodash": "^4.17.15" + }, + "dependencies": { + "bluebird": { + "version": "3.7.2", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", + "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==" + }, + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + } + } + }, "knex-factory": { "version": "0.0.6", "resolved": "https://registry.npmjs.org/knex-factory/-/knex-factory-0.0.6.tgz", @@ -5847,6 +5887,11 @@ "kind-of": "^6.0.2" } }, + "make-plural": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/make-plural/-/make-plural-6.1.0.tgz", + "integrity": "sha512-0ekbPHqxcdRcmjZ43TkRuejK5rXgMF1OjG4FVnVHgCvOcjrexaSX7a0dfAvqhOm1qWPgjYnXtmz3cHpHW5ZewA==" + }, "mamacro": { "version": "0.0.3", "resolved": "https://registry.npmjs.org/mamacro/-/mamacro-0.0.3.tgz", @@ -5875,6 +5920,11 @@ "object-visit": "^1.0.0" } }, + "math-interval-parser": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/math-interval-parser/-/math-interval-parser-2.0.1.tgz", + "integrity": "sha512-VmlAmb0UJwlvMyx8iPhXUDnVW1F9IrGEd9CIOmv+XL8AErCUUuozoDMrgImvnYt2A+53qVX/tPW6YJurMKYsvA==" + }, "md5.js": { "version": "1.3.5", "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz", @@ -5939,6 +5989,42 @@ } } }, + "messageformat": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/messageformat/-/messageformat-2.3.0.tgz", + "integrity": "sha512-uTzvsv0lTeQxYI2y1NPa1lItL5VRI8Gb93Y2K2ue5gBPyrbJxfDi/EYWxh2PKv5yO42AJeeqblS9MJSh/IEk4w==", + "requires": { + "make-plural": "^4.3.0", + "messageformat-formatters": "^2.0.1", + "messageformat-parser": "^4.1.2" + }, + "dependencies": { + "make-plural": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/make-plural/-/make-plural-4.3.0.tgz", + "integrity": "sha512-xTYd4JVHpSCW+aqDof6w/MebaMVNTVYBZhbB/vi513xXdiPT92JMVCo0Jq8W2UZnzYRFeVbQiQ+I25l13JuKvA==", + "requires": { + "minimist": "^1.2.0" + } + }, + "minimist": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", + "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", + "optional": true + } + } + }, + "messageformat-formatters": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/messageformat-formatters/-/messageformat-formatters-2.0.1.tgz", + "integrity": "sha512-E/lQRXhtHwGuiQjI7qxkLp8AHbMD5r2217XNe/SREbBlSawe0lOqsFb7rflZJmlQFSULNLIqlcjjsCPlB3m3Mg==" + }, + "messageformat-parser": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/messageformat-parser/-/messageformat-parser-4.1.2.tgz", + "integrity": "sha512-7dWuifeyldz7vhEuL96Kwq1fhZXBW+TUfbnHN4UCrCxoXQTYjHnR78eI66Gk9LaLLsAvzPNVJBaa66DRfFNaiA==" + }, "methods": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", @@ -6944,7 +7030,6 @@ "version": "1.4.0", "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", - "dev": true, "requires": { "wrappy": "1" } @@ -9100,11 +9185,6 @@ "neo-async": "^2.5.0" } }, - "weak-map": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/weak-map/-/weak-map-1.0.5.tgz", - "integrity": "sha1-eWkVhNmGB/UHC9O3CkDmuyLkAes=" - }, "webpack": { "version": "4.39.2", "resolved": "https://registry.npmjs.org/webpack/-/webpack-4.39.2.tgz", @@ -9556,8 +9636,7 @@ "wrappy": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", - "dev": true + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" }, "write": { "version": "1.0.3", diff --git a/server/package.json b/server/package.json index 0b5e6447a..6f919349c 100644 --- a/server/package.json +++ b/server/package.json @@ -29,8 +29,10 @@ "express-oauth-server": "^2.0.0", "express-validator": "^6.2.0", "helmet": "^3.21.0", + "i18n": "^0.8.5", "jsonwebtoken": "^8.5.1", "knex": "^0.20.3", + "knex-db-manager": "^0.6.1", "lodash": "^4.17.15", "memory-cache": "^0.2.0", "moment": "^2.24.0", diff --git a/server/src/app.js b/server/src/app.js index d7b76f077..55403acfc 100644 --- a/server/src/app.js +++ b/server/src/app.js @@ -1,12 +1,27 @@ import express from 'express'; import helmet from 'helmet'; import boom from 'express-boom'; +import i18n from 'i18n'; import '../config'; import routes from '@/http'; import '@/models'; const app = express(); +// i18n.configure({ +// // setup some locales - other locales default to en silently +// locales: ['en'], + +// // sets a custom cookie name to parse locale settings from. +// cookie: 'yourcookiename', + +// // where to store json files - defaults to './locales' +// directory: `${__dirname}/resources/locale`, +// }); + +// i18n init parses req for language headers, cookies, etc. +// app.use(i18n.init); + // Express configuration app.set('port', process.env.PORT || 3000); diff --git a/server/src/database/manager.js b/server/src/database/manager.js new file mode 100644 index 000000000..b42a3a3f0 --- /dev/null +++ b/server/src/database/manager.js @@ -0,0 +1,17 @@ +import knexManager from 'knex-db-manager'; +import knexfile from '@/../knexfile'; + +const config = knexfile[process.env.NODE_ENV]; + +const dbManager = knexManager.databaseManagerFactory({ + knex: config, + dbManager: { + // db manager related configuration + collate: [], + superUser: 'root', + superPassword: 'root', + // populatePathPattern: 'data/**/*.js', // glob format for searching seeds + }, +}); + +export default dbManager; \ No newline at end of file diff --git a/server/src/database/migrations/20190822214244_create_user_has_roles_table.js b/server/src/database/migrations/20190822214244_create_user_has_roles_table.js index 44f49755a..c35b826c4 100644 --- a/server/src/database/migrations/20190822214244_create_user_has_roles_table.js +++ b/server/src/database/migrations/20190822214244_create_user_has_roles_table.js @@ -2,8 +2,8 @@ exports.up = function (knex) { return knex.schema.createTable('user_has_roles', (table) => { table.increments(); - table.integer('user_id').unsigned().references('id').inTable('users'); - table.integer('role_id').unsigned().references('id').inTable('roles'); + table.integer('user_id').unsigned(); + table.integer('role_id').unsigned(); }); }; diff --git a/server/src/database/migrations/20190822214247_create_oauth_tokens_table.js b/server/src/database/migrations/20190822214247_create_oauth_tokens_table.js index f91d4a01a..0b7ded2ef 100644 --- a/server/src/database/migrations/20190822214247_create_oauth_tokens_table.js +++ b/server/src/database/migrations/20190822214247_create_oauth_tokens_table.js @@ -7,7 +7,7 @@ exports.up = function(knex) { table.integer('client_id').unsigned(); table.string('refresh_token'); table.date('refresh_token_expires_on'); - table.integer('user_id').unsigned().references('id').inTable('users'); + table.integer('user_id').unsigned(); }); }; diff --git a/server/src/database/migrations/20190822214302_create_settings_table.js b/server/src/database/migrations/20190822214302_create_settings_table.js index 57aeb37b3..bd7c0b43a 100644 --- a/server/src/database/migrations/20190822214302_create_settings_table.js +++ b/server/src/database/migrations/20190822214302_create_settings_table.js @@ -2,7 +2,7 @@ exports.up = function (knex) { return knex.schema.createTable('settings', (table) => { table.increments(); - table.integer('user_id').unsigned().references('id').inTable('users'); + table.integer('user_id').unsigned(); table.string('group'); table.string('type'); table.string('key'); diff --git a/server/src/database/migrations/20190822214905_create_resource_fields_table.js b/server/src/database/migrations/20190822214905_create_resource_fields_table.js index 9e01be943..d8fbe2d32 100644 --- a/server/src/database/migrations/20190822214905_create_resource_fields_table.js +++ b/server/src/database/migrations/20190822214905_create_resource_fields_table.js @@ -13,7 +13,7 @@ exports.up = function (knex) { table.boolean('columnable'); table.integer('index'); table.json('options'); - table.integer('resource_id').unsigned().references('id').inTable('resources'); + table.integer('resource_id').unsigned(); }); }; diff --git a/server/src/database/migrations/20190822214905_create_role_has_accounts.js b/server/src/database/migrations/20190822214905_create_role_has_accounts.js index e66e7df5e..6bc9c0d15 100644 --- a/server/src/database/migrations/20190822214905_create_role_has_accounts.js +++ b/server/src/database/migrations/20190822214905_create_role_has_accounts.js @@ -2,8 +2,8 @@ exports.up = function (knex) { return knex.schema.createTable('role_has_accounts', (table) => { table.increments(); - table.integer('role_id').unsigned().references('id').inTable('roles'); - table.integer('account_id').unsigned().references('id').inTable('accounts'); + table.integer('role_id').unsigned(); + table.integer('account_id').unsigned(); }); }; diff --git a/server/src/database/migrations/20190822214905_create_role_has_permissions.js b/server/src/database/migrations/20190822214905_create_role_has_permissions.js index 26e19e8be..d25020eab 100644 --- a/server/src/database/migrations/20190822214905_create_role_has_permissions.js +++ b/server/src/database/migrations/20190822214905_create_role_has_permissions.js @@ -2,9 +2,9 @@ exports.up = function (knex) { return knex.schema.createTable('role_has_permissions', (table) => { table.increments(); - table.integer('role_id').unsigned().references('id').inTable('roles'); - table.integer('permission_id').unsigned().references('id').inTable('permissions'); - table.integer('resource_id').unsigned().references('id').inTable('resources'); + table.integer('role_id').unsigned(); + table.integer('permission_id').unsigned(); + table.integer('resource_id').unsigned(); }); }; diff --git a/server/src/database/migrations/20190822214905_create_views_roles_table.js b/server/src/database/migrations/20190822214905_create_views_roles_table.js index c7a30b359..79a2bb0cd 100644 --- a/server/src/database/migrations/20190822214905_create_views_roles_table.js +++ b/server/src/database/migrations/20190822214905_create_views_roles_table.js @@ -3,7 +3,7 @@ exports.up = function (knex) { return knex.schema.createTable('view_roles', (table) => { table.increments(); table.integer('index'); - table.integer('field_id').unsigned().references('id').inTable('resource_fields'); + table.integer('field_id').unsigned(); table.string('comparator'); table.string('value'); table.integer('view_id').unsigned(); diff --git a/server/src/database/migrations/20200104232647_create_accounts_transactions_table.js b/server/src/database/migrations/20200104232647_create_accounts_transactions_table.js index 389472121..2ca5339e3 100644 --- a/server/src/database/migrations/20200104232647_create_accounts_transactions_table.js +++ b/server/src/database/migrations/20200104232647_create_accounts_transactions_table.js @@ -7,10 +7,10 @@ exports.up = function(knex) { table.string('transaction_type'); table.string('reference_type'); table.integer('reference_id'); - table.integer('account_id').unsigned().references('id').inTable('accounts'); + table.integer('account_id').unsigned(); table.string('note'); table.boolean('draft').defaultTo(false); - table.integer('user_id').unsigned().references('id').inTable('users'); + table.integer('user_id').unsigned(); table.date('date'); table.timestamps(); }); diff --git a/server/src/database/migrations/20200105014405_create_expenses_table.js b/server/src/database/migrations/20200105014405_create_expenses_table.js index 4cb326302..f6f8e383d 100644 --- a/server/src/database/migrations/20200105014405_create_expenses_table.js +++ b/server/src/database/migrations/20200105014405_create_expenses_table.js @@ -6,11 +6,11 @@ exports.up = function(knex) { table.string('currency_code'); table.decimal('exchange_rate'); table.text('description'); - table.integer('expense_account_id').unsigned().references('id').inTable('accounts'); - table.integer('payment_account_id').unsigned().references('id').inTable('accounts'); + table.integer('expense_account_id').unsigned(); + table.integer('payment_account_id').unsigned(); table.string('reference'); table.boolean('published').defaultTo(false); - table.integer('user_id').unsigned().references('id').inTable('users'); + table.integer('user_id').unsigned(); table.date('date'); // table.timestamps(); }) diff --git a/server/src/database/migrations/20200105195823_create_manual_journals_table.js b/server/src/database/migrations/20200105195823_create_manual_journals_table.js index f254aec7c..5dee2b6e4 100644 --- a/server/src/database/migrations/20200105195823_create_manual_journals_table.js +++ b/server/src/database/migrations/20200105195823_create_manual_journals_table.js @@ -7,7 +7,7 @@ exports.up = function(knex) { table.decimal('amount'); table.date('date'); table.string('note'); - table.integer('user_id').unsigned().references('id').inTable('users'); + table.integer('user_id').unsigned(); table.timestamps(); }); }; diff --git a/server/src/database/migrations/20200120145342_create_budget_entries_table.js b/server/src/database/migrations/20200120145342_create_budget_entries_table.js index 60a7d49a2..bd7795ef5 100644 --- a/server/src/database/migrations/20200120145342_create_budget_entries_table.js +++ b/server/src/database/migrations/20200120145342_create_budget_entries_table.js @@ -2,8 +2,8 @@ exports.up = function(knex) { return knex.schema.createTable('budget_entries', (table) => { table.increments(); - table.integer('budget_id').unsigned().references('id').inTable('budgets'); - table.integer('account_id').unsigned().references('id').inTable('accounts'); + table.integer('budget_id').unsigned(); + table.integer('account_id').unsigned(); table.decimal('amount', 15, 5); table.integer('order'); }) diff --git a/server/src/database/migrations/20200125173323_create_resource_custom_fields_metadata_table.js b/server/src/database/migrations/20200125173323_create_resource_custom_fields_metadata_table.js index 74c1c89c3..b1731a3ea 100644 --- a/server/src/database/migrations/20200125173323_create_resource_custom_fields_metadata_table.js +++ b/server/src/database/migrations/20200125173323_create_resource_custom_fields_metadata_table.js @@ -2,11 +2,11 @@ exports.up = function(knex) { return knex.schema.createTable('resource_custom_fields_metadata', (table) => { table.increments(); - table.integer('resource_id').unsigned().references('id').inTable('resources'); + table.integer('resource_id').unsigned(); table.integer('resource_item_id').unsigned(); table.string('key'); table.string('value'); - }) + }); }; exports.down = function(knex) { diff --git a/server/src/database/seeds/account_types.js b/server/src/database/seeds/account_types.js index 003a2f05b..91a07d65e 100644 --- a/server/src/database/seeds/account_types.js +++ b/server/src/database/seeds/account_types.js @@ -8,54 +8,63 @@ exports.seed = (knex) => { { id: 1, name: 'Fixed Asset', + normal: 'debit', balance_sheet: true, income_sheet: false, }, { id: 2, name: 'Current Asset', + normal: 'debit', balance_sheet: true, income_sheet: false, }, { id: 3, name: 'Long Term Liability', + normal: 'credit', balance_sheet: false, income_sheet: true, }, { id: 4, name: 'Current Liability', + normal: 'credit', balance_sheet: false, income_sheet: true, }, { id: 5, name: 'Equity', + normal: 'credit', balance_sheet: false, income_sheet: true, }, { id: 6, name: 'Expense', + normal: 'debit', balance_sheet: false, income_sheet: true, }, { id: 7, name: 'Income', + normal: 'credit', balance_sheet: false, income_sheet: true, }, { id: 8, name: 'Accounts Receivable', + normal: 'debit', balance_sheet: true, income_sheet: false, }, { id: 9, name: 'Accounts Payable', + normal: 'credit', balance_sheet: true, income_sheet: false, }, diff --git a/server/src/http/controllers/Accounting.js b/server/src/http/controllers/Accounting.js index a3696ec1a..bf611bdc7 100644 --- a/server/src/http/controllers/Accounting.js +++ b/server/src/http/controllers/Accounting.js @@ -1,6 +1,7 @@ -import { check, query, validationResult } from 'express-validator'; +import { check, query, oneOf, validationResult } from 'express-validator'; import express from 'express'; import { difference } from 'lodash'; +import moment from 'moment'; import Account from '@/models/Account'; import asyncMiddleware from '@/http/middleware/asyncMiddleware'; import JWTAuth from '@/http/middleware/jwtAuth'; @@ -38,9 +39,10 @@ export default { validation: [ check('date').isISO8601(), check('reference').exists(), + check('memo').optional().trim().escape(), check('entries').isArray({ min: 1 }), - check('entries.*.credit').isNumeric().toInt(), - check('entries.*.debit').isNumeric().toInt(), + check('entries.*.credit').optional({ nullable: true }).isNumeric().toInt(), + check('entries.*.debit').optional({ nullable: true }).isNumeric().toInt(), check('entries.*.account_id').isNumeric().toInt(), check('entries.*.note').optional(), ], @@ -56,11 +58,16 @@ export default { date: new Date(), ...req.body, }; - const errorReasons = []; + let totalCredit = 0; let totalDebit = 0; - form.entries.forEach((entry) => { + const { user } = req; + const errorReasons = []; + const entries = form.entries.filter((entry) => (entry.credit || entry.debit)); + const formattedDate = moment(form.date).format('YYYY-MM-DD'); + + entries.forEach((entry) => { if (entry.credit > 0) { totalCredit += entry.credit; } @@ -68,6 +75,7 @@ export default { totalDebit += entry.debit; } }); + if (totalCredit <= 0 || totalDebit <= 0) { errorReasons.push({ type: 'CREDIT.DEBIT.SUMATION.SHOULD.NOT.EQUAL.ZERO', @@ -77,7 +85,7 @@ export default { if (totalCredit !== totalDebit) { errorReasons.push({ type: 'CREDIT.DEBIT.NOT.EQUALS', code: 100 }); } - const accountsIds = form.entries.map((entry) => entry.account_id); + const accountsIds = entries.map((entry) => entry.account_id); const accounts = await Account.query().whereIn('id', accountsIds) .withGraphFetched('type'); @@ -95,18 +103,30 @@ export default { if (errorReasons.length > 0) { return res.status(400).send({ errors: errorReasons }); } + + // Save manual journal transaction. + const manualJournal = await ManualJournal.query().insert({ + reference: form.reference, + transaction_type: 'Journal', + amount: totalCredit, + date: formattedDate, + note: form.memo, + user_id: user.id, + }); const journalPoster = new JournalPoster(); - form.entries.forEach((entry) => { + entries.forEach((entry) => { const account = accounts.find((a) => a.id === entry.account_id); const jouranlEntry = new JournalEntry({ - date: entry.date, debit: entry.debit, credit: entry.credit, account: account.id, + transactionType: 'Journal', accountNormal: account.type.normal, note: entry.note, + date: formattedDate, + userId: user.id, }); if (entry.debit) { journalPoster.debit(jouranlEntry); @@ -120,7 +140,7 @@ export default { journalPoster.saveEntries(), journalPoster.saveBalance(), ]); - return res.status(200).send(); + return res.status(200).send({ id: manualJournal.id }); }, }, diff --git a/server/src/http/controllers/Customers.js b/server/src/http/controllers/Customers.js index f7a91021d..4a56bd130 100644 --- a/server/src/http/controllers/Customers.js +++ b/server/src/http/controllers/Customers.js @@ -1,10 +1,71 @@ import express from 'express'; +import { + check, + param, + query, + validationResult, +} from 'express-validator'; +import asyncMiddleware from '@/http/middleware/asyncMiddleware'; export default { router() { const router = express.Router(); + router.post('/', + this.newCustomer.validation, + asyncMiddleware(this.newCustomer.handler)); + + router.post('/:id', + this.editCustomer.validation, + asyncMiddleware(this.editCustomer.handler)); + return router; }, + + newCustomer: { + validation: [ + check('custom_type').exists().trim().escape(), + check('first_name').exists().trim().escape(), + check('last_name'), + check('company_name'), + check('email'), + check('work_phone'), + check('personal_phone'), + + check('billing_address.country'), + check('billing_address.address'), + check('billing_address.city'), + check('billing_address.phone'), + check('billing_address.zip_code'), + + check('shiping_address.country'), + check('shiping_address.address'), + check('shiping_address.city'), + check('shiping_address.phone'), + check('shiping_address.zip_code'), + + check('contact.additional_phone'), + check('contact.additional_email'), + + check('custom_fields').optional().isArray({ min: 1 }), + check('custom_fields.*.key').exists().trim().escape(), + check('custom_fields.*.value').exists(), + + check('inactive').optional().isBoolean().toBoolean(), + ], + + async handler(req, res) { + + }, + }, + + editCustomer: { + validation: [ + + ], + async handler(req, res) { + + }, + }, }; diff --git a/server/src/http/controllers/Expenses.js b/server/src/http/controllers/Expenses.js index 8d2e66d45..9a55775b8 100644 --- a/server/src/http/controllers/Expenses.js +++ b/server/src/http/controllers/Expenses.js @@ -317,6 +317,12 @@ export default { query('page').optional().isNumeric().toInt(), query('page_size').optional().isNumeric().toInt(), query('custom_view_id').optional().isNumeric().toInt(), + + query('filter_roles').optional().isArray(), + query('filter_roles.*.field_key').exists().escape().trim(), + query('filter_roles.*.value').exists().escape().trim(), + query('filter_roles.*.comparator').exists().escape().trim(), + query('filter_roles.*.index').exists().isNumeric().toInt(), ], async handler(req, res) { const validationErrors = validationResult(req); diff --git a/server/src/http/controllers/FinancialStatements.js b/server/src/http/controllers/FinancialStatements.js index 27684dc2c..d8abec9d9 100644 --- a/server/src/http/controllers/FinancialStatements.js +++ b/server/src/http/controllers/FinancialStatements.js @@ -122,6 +122,7 @@ export default { query('number_format.no_cents').optional().isBoolean().toBoolean(), query('number_format.divide_1000').optional().isBoolean().toBoolean(), query('none_zero').optional().isBoolean().toBoolean(), + query('accounts_ids').optional().trim().escape(), ], async handler(req, res) { const validationErrors = validationResult(req); @@ -134,16 +135,21 @@ export default { const filter = { from_date: moment().startOf('year').format('YYYY-MM-DD'), to_date: moment().endOf('year').format('YYYY-MM-DD'), + basis: 'cash', number_format: { no_cents: false, divide_1000: false, }, none_zero: false, + accounts_ids: [], ...req.query, }; + const accounts = await Account.query() .orderBy('index', 'DESC') + .modify('filterAccounts', filter.accounts_ids) .withGraphFetched('transactions') + .withGraphFetched('type') .modifyGraph('transactions', (builder) => { builder.modify('filterDateRange', filter.from_date, filter.to_date); }); @@ -167,33 +173,40 @@ export default { // Transaction amount formatter based on the given query. const formatNumber = formatNumberClosure(filter.number_format); - const items = [ - ...accounts - .filter((account) => ( - account.transactions.length > 0 || !filter.none_zero - )) - .map((account) => ({ - ...pick(account, ['id', 'name', 'code', 'index']), - transactions: [ - ...account.transactions.map((transaction) => ({ + const items = accounts + .filter((account) => ( + account.transactions.length > 0 || !filter.none_zero + )) + .map((account) => ({ + ...pick(account, ['id', 'name', 'code', 'index']), + transactions: [ + ...account.transactions.map((transaction) => { + let amount = 0; + + if (account.type.normal === 'credit') { + amount += transaction.credit - transaction.credit; + } else if (account.type.normal === 'debit') { + amount += transaction.debit - transaction.credit; + } + return { ...transaction, - credit: formatNumber(transaction.credit), - debit: formatNumber(transaction.debit), - })), - ], - opening: { - date: filter.from_date, - balance: opeingBalanceCollection.getClosingBalance(account.id), - }, - closing: { - date: filter.to_date, - balance: closingBalanceCollection.getClosingBalance(account.id), - }, - })), - ]; + amount: formatNumber(amount), + }; + }), + ], + opening: { + date: filter.from_date, + balance: opeingBalanceCollection.getClosingBalance(account.id), + }, + closing: { + date: filter.to_date, + balance: closingBalanceCollection.getClosingBalance(account.id), + }, + })); + return res.status(200).send({ - meta: { ...filter }, - items, + query: { ...filter }, + accounts: items, }); }, }, @@ -206,7 +219,7 @@ export default { query('accounting_method').optional().isIn(['cash', 'accural']), query('from_date').optional(), query('to_date').optional(), - query('display_columns_by').optional().isIn(['year', 'month', 'week', 'day', 'quarter']), + query('display_columns_by').optional().isIn(['total', 'year', 'month', 'week', 'day', 'quarter']), query('number_format.no_cents').optional().isBoolean().toBoolean(), query('number_format.divide_1000').optional().isBoolean().toBoolean(), query('none_zero').optional().isBoolean().toBoolean(), @@ -220,7 +233,7 @@ export default { }); } const filter = { - display_columns_by: 'year', + display_columns_by: 'total', from_date: moment().startOf('year').format('YYYY-MM-DD'), to_date: moment().endOf('year').format('YYYY-MM-DD'), number_format: { @@ -228,11 +241,11 @@ export default { divide_1000: false, }, none_zero: false, + basis: 'cash', ...req.query, }; - const balanceSheetTypes = await AccountType.query() - .where('balance_sheet', true); + const balanceSheetTypes = await AccountType.query().where('balance_sheet', true); // Fetch all balance sheet accounts. const accounts = await Account.query() @@ -249,51 +262,92 @@ export default { // Account balance formmatter based on the given query. const balanceFormatter = formatNumberClosure(filter.number_format); + const filterDateType = filter.display_columns_by === 'total' + ? 'day' : filter.display_columns_by; // Gets the date range set from start to end date. const dateRangeSet = dateRangeCollection( filter.from_date, filter.to_date, - filter.display_columns_by, + filterDateType, ); + // Retrieve the asset balance sheet. - const assets = [ - ...accounts - .filter((account) => ( - account.type.normal === 'debit' + const assets = accounts + .filter((account) => ( + account.type.normal === 'debit' && (account.transactions.length > 0 || !filter.none_zero) - )) - .map((account) => ({ + )) + .map((account) => { + // Calculates the closing balance to the given date. + const closingBalance = journalEntries.getClosingBalance(account.id, filter.to_date); + const type = filter.display_columns_by; + + return { ...pick(account, ['id', 'index', 'name', 'code']), - transactions: dateRangeSet.map((date) => { - const type = filter.display_columns_by; - const balance = journalEntries.getClosingBalance(account.id, date, type); - return { date, balance: balanceFormatter(balance) }; - }), - })), - ]; + ...(type !== 'total') ? { + periods_balance: dateRangeSet.map((date) => { + const balance = journalEntries.getClosingBalance(account.id, date, filterDateType); + + return { + date, + formatted_amount: balanceFormatter(balance), + amount: balance, + }; + }), + } : {}, + balance: { + formatted_amount: balanceFormatter(closingBalance), + amount: closingBalance, + date: filter.to_date, + }, + }; + }); + // Retrieve liabilities and equity balance sheet. - const liabilitiesEquity = [ - ...accounts - .filter((account) => ( - account.type.normal === 'credit' + const liabilitiesEquity = accounts + .filter((account) => ( + account.type.normal === 'credit' && (account.transactions.length > 0 || !filter.none_zero) - )) - .map((account) => ({ + )) + .map((account) => { + // Calculates the closing balance to the given date. + const closingBalance = journalEntries.getClosingBalance(account.id, filter.to_date); + const type = filter.display_columns_by; + + return { ...pick(account, ['id', 'index', 'name', 'code']), - transactions: dateRangeSet.map((date) => { - const type = filter.display_columns_by; - const balance = journalEntries.getClosingBalance(account.id, date, type); - return { date, balance: balanceFormatter(balance) }; - }), - })), - ]; + ...(type !== 'total') ? { + periods_balance: dateRangeSet.map((date) => { + const balance = journalEntries.getClosingBalance(account.id, date, filterDateType); + + return { + date, + formatted_amount: balanceFormatter(balance), + amount: balance, + }; + }), + } : {}, + balance: { + formattedAmount: balanceFormatter(closingBalance), + amount: closingBalance, + date: filter.to_date, + }, + }; + }); + return res.status(200).send({ query: { ...filter }, columns: { ...dateRangeSet }, balance_sheet: { - assets, - liabilities_equity: liabilitiesEquity, + assets: { + title: 'Assets', + accounts: [...assets], + }, + liabilities_equity: { + title: 'Liabilities & Equity', + accounts: [...liabilitiesEquity], + }, }, }); }, @@ -363,7 +417,7 @@ export default { }; }); return res.status(200).send({ - meta: { ...filter }, + query: { ...filter }, items: [...items], }); }, @@ -381,8 +435,12 @@ export default { query('number_format.divide_1000').optional().isBoolean(), query('basis').optional(), query('none_zero').optional(), - query('display_columns_by').optional().isIn(['year', 'month', 'week', 'day', 'quarter']), - query('accounts').optional().isArray(), + query('display_columns_type').optional().isIn([ + 'total', 'date_periods', + ]), + query('display_columns_by').optional().isIn([ + 'year', 'month', 'week', 'day', 'quarter', + ]), ], async handler(req, res) { const validationErrors = validationResult(req); @@ -401,19 +459,22 @@ export default { }, basis: 'accural', none_zero: false, - display_columns_by: 'month', + display_columns_type: 'total', + display_columns_by: 'total', ...req.query, }; const incomeStatementTypes = await AccountType.query().where('income_sheet', true); + // Fetch all income accounts from storage. const accounts = await Account.query() .whereIn('account_type_id', incomeStatementTypes.map((t) => t.id)) .withGraphFetched('type') .withGraphFetched('transactions'); - const filteredAccounts = accounts.filter((account) => { - return account.transactions.length > 0 || !filter.none_zero; - }); + // Filter all none zero accounts if it was enabled. + const filteredAccounts = accounts.filter((account) => ( + account.transactions.length > 0 || !filter.none_zero + )); const journalEntriesCollected = Account.collectJournalEntries(accounts); const journalEntries = new JournalPoster(); journalEntries.loadEntries(journalEntriesCollected); @@ -427,75 +488,130 @@ export default { filter.to_date, filter.display_columns_by, ); - const accountsIncome = filteredAccounts - .filter((account) => account.type.normal === 'credit') - .map((account) => ({ + + const accountsMapper = (incomeExpenseAccounts) => ( + incomeExpenseAccounts.map((account) => ({ ...pick(account, ['id', 'index', 'name', 'code']), - dates: dateRangeSet.map((date) => { - const type = filter.display_columns_by; - const amount = journalEntries.getClosingBalance(account.id, date, type); - return { date, rawAmount: amount, amount: numberFormatter(amount) }; - }), - })); + // Total closing balance of the account. + ...(filter.display_columns_type === 'total') && { + total: (() => { + const amount = journalEntries.getClosingBalance(account.id, filter.to_date); + return { amount, date: filter.to_date, formatted_amount: numberFormatter(amount) }; + })(), + }, + // Date periods when display columns type `periods`. + ...(filter.display_columns_type === 'date_periods') && { + periods: dateRangeSet.map((date) => { + const type = filter.display_columns_by; + const amount = journalEntries.getClosingBalance(account.id, date, type); - const accountsExpenses = filteredAccounts - .filter((account) => account.type.normal === 'debit') - .map((account) => ({ - ...pick(account, ['id', 'index', 'name', 'code']), - dates: dateRangeSet.map((date) => { - const type = filter.display_columns_by; - const amount = journalEntries.getClosingBalance(account.id, date, type); + return { date, amount, formatted_amount: numberFormatter(amount) }; + }), + }, + }))); - return { date, rawAmount: amount, amount: numberFormatter(amount) }; - }), - })); + const totalAccountsReducer = (incomeExpenseAccounts) => ( + incomeExpenseAccounts.reduce((acc, account) => { + const amount = (account) ? account.total.amount : 0; + return amount + acc; + }, 0)); - // Calculates the total income of income accounts. - const totalAccountsIncome = dateRangeSet.reduce((acc, date, index) => { - let amount = 0; - accountsIncome.forEach((account) => { - const currentDate = account.dates[index]; - amount += currentDate.rawAmount || 0; - }); - acc[date] = { date, rawAmount: amount, amount: numberFormatter(amount) }; - return acc; - }, {}); + const accountsIncome = accountsMapper(filteredAccounts + .filter((account) => account.type.normal === 'credit')); - // Calculates the total expenses of expenses accounts. - const totalAccountsExpenses = dateRangeSet.reduce((acc, date, index) => { - let amount = 0; - accountsExpenses.forEach((account) => { - const currentDate = account.dates[index]; - amount += currentDate.rawAmount || 0; - }); - acc[date] = { date, rawAmount: amount, amount: numberFormatter(amount) }; - return acc; - }, {}); + const accountsExpenses = accountsMapper(filteredAccounts + .filter((account) => account.type.normal === 'debit')); + + // @return {Array} + const totalPeriodsMapper = (incomeExpenseAccounts) => ( + Object.values(dateRangeSet.reduce((acc, date, index) => { + let amount = 0; + + incomeExpenseAccounts.forEach((account) => { + const currentDate = account.periods[index]; + amount += currentDate.amount || 0; + }); + acc[date] = { date, amount, formatted_amount: numberFormatter(amount) }; + return acc; + }, {}))); // Total income(date) - Total expenses(date) = Net income(date) - const netIncome = dateRangeSet.map((date) => { - const totalIncome = totalAccountsIncome[date]; - const totalExpenses = totalAccountsExpenses[date]; + // @return {Array} + const netIncomePeriodsMapper = (totalIncomeAcocunts, totalExpenseAccounts) => ( + dateRangeSet.map((date, index) => { + const totalIncome = totalIncomeAcocunts[index]; + const totalExpenses = totalExpenseAccounts[index]; - let amount = totalIncome.rawAmount || 0; - amount -= totalExpenses.rawAmount || 0; - return { date, rawAmount: amount, amount: numberFormatter(amount) }; - }); + let amount = totalIncome.amount || 0; + amount -= totalExpenses.amount || 0; + return { date, amount, formatted_amount: numberFormatter(amount) }; + })); + // @return {Object} + const netIncomeTotal = (totalIncome, totalExpenses) => { + const netIncomeAmount = totalIncome.amount - totalExpenses.amount; + return { amount: netIncomeAmount, formatted_amount: netIncomeAmount }; + }; + + const totalIncomeAccounts = totalAccountsReducer(accountsIncome); + const totalExpensesAccounts = totalAccountsReducer(accountsExpenses); + + const incomeResponse = { + entry_normal: 'credit', + accounts: accountsIncome, + + ...(filter.display_columns_type === 'total') && { + total: { + amount: totalIncomeAccounts, + date: filter.to_date, + formatted_amount: numberFormatter(totalIncomeAccounts), + }, + }, + ...(filter.display_columns_type === 'date_periods') && { + total_periods: [ + ...totalPeriodsMapper(accountsIncome), + ], + }, + }; + const expenseResponse = { + entry_normal: 'debit', + accounts: accountsExpenses, + + ...(filter.display_columns_type === 'total') && { + total: { + amount: totalExpensesAccounts, + date: filter.to_date, + formatted_amount: numberFormatter(totalExpensesAccounts), + }, + }, + ...(filter.display_columns_type === 'date_periods') && { + total_periods: [ + ...totalPeriodsMapper(accountsExpenses), + ], + }, + }; + const netIncomeResponse = { + ...(filter.display_columns_type === 'total') && { + total: { + ...netIncomeTotal(incomeResponse.total, expenseResponse.total), + }, + }, + ...(filter.display_columns_type === 'date_periods') && { + total_periods: [ + ...netIncomePeriodsMapper( + incomeResponse.total_periods, + expenseResponse.total_periods, + ), + ], + }, + }; return res.status(200).send({ - meta: { ...filter }, - income: { - entry_normal: 'credit', - accounts: accountsIncome, - }, - expenses: { - entry_normal: 'debit', - accounts: accountsExpenses, - }, - total_income: Object.values(totalAccountsIncome), - total_expenses: Object.values(totalAccountsExpenses), - total_net_income: netIncome, + query: { ...filter }, + columns: [...dateRangeSet], + income: incomeResponse, + expenses: expenseResponse, + net_income: netIncomeResponse, }); }, }, diff --git a/server/src/models/Account.js b/server/src/models/Account.js index dbad7f73f..53e900af2 100644 --- a/server/src/models/Account.js +++ b/server/src/models/Account.js @@ -17,6 +17,11 @@ export default class Account extends BaseModel { */ static get modifiers() { return { + filterAccounts(query, accountIds) { + if (accountIds.length > 0) { + query.whereIn('id', accountIds); + } + }, filterAccountTypes(query, typesIds) { if (typesIds.length > 0) { query.whereIn('accoun_type_id', typesIds); diff --git a/server/src/services/Accounting/JournalPoster.js b/server/src/services/Accounting/JournalPoster.js index d6ab875c6..b18dec636 100644 --- a/server/src/services/Accounting/JournalPoster.js +++ b/server/src/services/Accounting/JournalPoster.js @@ -3,6 +3,7 @@ import moment from 'moment'; import JournalEntry from '@/services/Accounting/JournalEntry'; import AccountTransaction from '@/models/AccountTransaction'; import AccountBalance from '@/models/AccountBalance'; +import {promiseSerial} from '@/utils'; export default class JournalPoster { /** @@ -125,12 +126,12 @@ export default class JournalPoster { this.entries.forEach((entry) => { const oper = AccountTransaction.query().insert({ accountId: entry.account, - ...pick(entry, ['credit', 'debit', 'transactionType', + ...pick(entry, ['credit', 'debit', 'transactionType', 'date', 'userId', 'referenceType', 'referenceId', 'note']), }); - saveOperations.push(oper); + saveOperations.push(() => oper); }); - await Promise.all(saveOperations); + await promiseSerial(saveOperations); } /** diff --git a/server/src/utils/index.js b/server/src/utils/index.js index 06c1fad3a..6dd921b8f 100644 --- a/server/src/utils/index.js +++ b/server/src/utils/index.js @@ -66,6 +66,12 @@ const mapValuesDeep = (v, callback) => ( ? _.mapValues(v, v => mapValuesDeep(v, callback)) : callback(v)); + +const promiseSerial = (funcs) => { + return funcs.reduce((promise, func) => promise.then((result) => func().then(Array.prototype.concat.bind(result))), + Promise.resolve([])); +} + export { hashPassword, origin, @@ -73,4 +79,5 @@ export { dateRangeFormat, mapValuesDeep, mapKeysDeep, + promiseSerial, }; diff --git a/server/tests/routes/accounting.test.js b/server/tests/routes/accounting.test.js index 49ce26757..1ba7c7587 100644 --- a/server/tests/routes/accounting.test.js +++ b/server/tests/routes/accounting.test.js @@ -1,8 +1,16 @@ -import { request, expect, create, login } from '~/testInit'; +import { + request, + expect, + create, + login, +} from '~/testInit'; +import moment from 'moment'; +import ManualJournal from '@/models/ManualJournal'; +import AccountTransaction from '@/models/AccountTransaction'; let loginRes; -describe('routes: /accounting', () => { +describe('routes: `/accounting`', () => { beforeEach(async () => { loginRes = await login(); }); @@ -127,8 +135,113 @@ describe('routes: /accounting', () => { }); }); - it('Should store all journal entries to the storage.', async () => { + it('Should discard journal entries that has null credit and debit amount.', async () => { + const account1 = await create('account'); + const account2 = await create('account'); + const res = await request() + .post('/api/accounting/make-journal-entries') + .set('x-access-token', loginRes.body.token) + .send({ + date: new Date().toISOString(), + reference: '1000', + entries: [ + { + credit: null, + debit: 0, + account_id: account1.id, + }, + { + credit: null, + debit: 0, + account_id: account2.id, + }, + ], + }); + + expect(res.status).equals(400); + expect(res.body.errors).include.something.that.deep.equal({ + type: 'CREDIT.DEBIT.SUMATION.SHOULD.NOT.EQUAL.ZERO', + code: 400, + }); + }); + + it('Should store manual journal transaction to the storage.', async () => { + const account1 = await create('account'); + const account2 = await create('account'); + + const res = await request() + .post('/api/accounting/make-journal-entries') + .set('x-access-token', loginRes.body.token) + .send({ + date: new Date('2020-2-2').toISOString(), + reference: '1000', + memo: 'Description here.', + entries: [ + { + credit: 1000, + account_id: account1.id, + }, + { + debit: 1000, + account_id: account2.id, + }, + ], + }); + + const foundManualJournal = await ManualJournal.query(); + + expect(foundManualJournal.length).equals(1); + expect(foundManualJournal[0].reference).equals('1000'); + expect(foundManualJournal[0].transactionType).equals('Journal'); + expect(foundManualJournal[0].amount).equals(1000); + expect(moment(foundManualJournal[0].date).format('YYYY-MM-DD')).equals('2020-02-02'); + expect(foundManualJournal[0].note).equals('Description here.'); + expect(foundManualJournal[0].userId).equals(1); + }); + + it('Should store journal transactions to the storage.', async () => { + const account1 = await create('account'); + const account2 = await create('account'); + + const res = await request() + .post('/api/accounting/make-journal-entries') + .set('x-access-token', loginRes.body.token) + .send({ + date: new Date('2020-1-1').toISOString(), + reference: '1000', + memo: 'Description here.', + entries: [ + { + credit: 1000, + account_id: account1.id, + note: 'First note', + }, + { + debit: 1000, + account_id: account2.id, + note: 'Second note', + }, + ], + }); + + const foundAccountsTransactions = await AccountTransaction.query(); + + expect(foundAccountsTransactions.length).equals(2); + + expect(foundAccountsTransactions[0].credit).equals(1000); + expect(foundAccountsTransactions[0].debit).equals(null); + expect(foundAccountsTransactions[0].accountId).equals(account1.id); + expect(foundAccountsTransactions[0].note).equals('First note'); + expect(foundAccountsTransactions[0].transactionType).equals('Journal'); + expect(foundAccountsTransactions[0].userId).equals(1); + + expect(foundAccountsTransactions[1].credit).equals(null); + expect(foundAccountsTransactions[1].debit).equals(1000); + expect(foundAccountsTransactions[1].accountId).equals(account2.id); + expect(foundAccountsTransactions[1].note).equals('Second note'); + expect(foundAccountsTransactions[1].transactionType).equals('Journal'); + expect(foundAccountsTransactions[1].userId).equals(1); }); }); diff --git a/server/tests/routes/financial_statements.test.js b/server/tests/routes/financial_statements.test.js index 0ff4ab5a1..5fb167d68 100644 --- a/server/tests/routes/financial_statements.test.js +++ b/server/tests/routes/financial_statements.test.js @@ -4,6 +4,7 @@ import { login, create, } from '~/testInit'; +import moment from 'moment'; let loginRes; let creditAccount; @@ -145,56 +146,95 @@ describe('routes: `/financial_statements`', () => { }); }); - describe('routes: `/financial_statements/general_ledger`', () => { + describe.only('routes: `/financial_statements/general_ledger`', () => { it('Should response unauthorized in case the user was not authorized.', async () => { const res = await request() .get('/api/financial_statements/general_ledger') .send(); - expect(res.status).equals(400); + expect(res.status).equals(401); }); - it('Should retrieve the genereal ledger transactions.', async () => { + it('Should retrieve request query meta on response schema.', async () => { const res = await request() .get('/api/financial_statements/general_ledger') .set('x-access-token', loginRes.body.token) .send(); - expect(res.body.items).to.be.a('array'); - expect(res.body.items.length).equals(4); + expect(res.body.query.from_date).equals(moment().startOf('year').format('YYYY-MM-DD')); + expect(res.body.query.to_date).equals(moment().endOf('year').format('YYYY-MM-DD')); + expect(res.body.query.basis).equals('cash'); + expect(res.body.query.number_format.no_cents).equals(false); + expect(res.body.query.number_format.divide_1000).equals(false); + expect(res.body.query.none_zero).equals(false); + expect(res.body.query.accounts_ids).to.be.an('array'); }); - it('Should retrieve opeing and closing balance in each account.', async () => { + it('Should retrieve the general ledger accounts with associated transactions and opening/closing balance.', async () => { const res = await request() .get('/api/financial_statements/general_ledger') .set('x-access-token', loginRes.body.token) .send(); - const foundCreditAccount = res.body.items.find((a) => a.id === creditAccount.id); - - expect(foundCreditAccount.closing.balance).equals(2000); - expect(foundCreditAccount.opening.balance).equals(0); + expect(res.body.accounts).is.an('array'); + expect(res.body.accounts[0].id).to.be.an('number'); + expect(res.body.accounts[0].name).to.be.a('string'); + expect(res.body.accounts[0].code).to.be.a('string'); + expect(res.body.accounts[0].transactions).to.be.a('array'); + expect(res.body.accounts[0].opening).to.be.a('object'); + expect(res.body.accounts[0].opening.balance).to.be.a('number'); + expect(res.body.accounts[0].opening.date).to.be.a('string'); + expect(res.body.accounts[0].closing).to.be.a('object'); + expect(res.body.accounts[0].closing.balance).to.be.a('number'); + expect(res.body.accounts[0].closing.date).to.be.a('string'); }); - it('Should retrieve the general ledger transactions between date range.', async () => { + it('Should retrieve opening and closing balance.', async () => { const res = await request() .get('/api/financial_statements/general_ledger') .set('x-access-token', loginRes.body.token) - .query({ - from_date: '2020-04-04', - to_date: '2020-05-05', - }) .send(); - - const foundCreditAccount = res.body.items.find((a) => a.id === creditAccount.id); - expect(foundCreditAccount.transactions.length).equals(0); + + const targetAccount = res.body.accounts.find((a) => a.id === creditAccount.id); + + expect(targetAccount).to.be.an('object'); + expect(targetAccount.opening).to.deep.equal({ + balance: 0, date: '2020-01-01', + }); + expect(targetAccount.closing).to.deep.equal({ + balance: 2000, date: '2020-12-31', + }); }); - it('Should retrieve the general ledger transactions with no cents numbers.', () => { + it('Should retrieve opening and closing balance between the given date range.', () => { }); - it('Should retrieve the transacvtions divided on 1000.', () => { + it('Should retrieve transactions of accounts that has transactions between date range.', () => { + + }); + + it('Should retrieve accounts transactions only that between date range.', () => { + + }); + + it('Should not retrieve all accounts that have no transactions in the given date range when `none_zero` is `false`.', () => { + + }); + + it('Should retrieve all accounts even it have no transactions in the given date range when `none_zero` is `true`', () => { + + }); + + it('Should amount transactions divided on 1000 when `number_format.none_zero` is `true`.', () => { + + }); + + it('Should amount transactions rounded with no decimals when `number_format.no_cents` is `true`.', () => { + + }); + + it('Should retrieve only accounts that given in the query.', () => { }); }); @@ -208,6 +248,27 @@ describe('routes: `/financial_statements`', () => { expect(res.status).equals(401); }); + it('Should retrieve query of the balance sheet with default values.', async () => { + const res = await request() + .get('/api/financial_statements/balance_sheet') + .set('x-access-token', loginRes.body.token) + .query({ + display_columns_by: 'year', + from_date: '2020-01-01', + to_date: '2020-02-01', + }) + .send(); + + expect(res.body.query.display_columns_by).equals('year'); + expect(res.body.query.from_date).equals('2020-01-01'); + expect(res.body.query.to_date).equals('2020-02-01'); + + expect(res.body.query.number_format.no_cents).equals(false); + expect(res.body.query.number_format.divide_1000).equals(false); + + expect(res.body.query.none_zero).equals(false); + }); + it('Should retrieve the asset accounts balance.', async () => { const res = await request() .get('/api/financial_statements/balance_sheet') @@ -217,11 +278,30 @@ describe('routes: `/financial_statements`', () => { }) .send(); - expect(res.body.balance_sheet.assets).to.be.a('array'); - expect(res.body.balance_sheet.liabilities_equity).to.be.a('array'); + expect(res.body.balance_sheet.assets.accounts).to.be.a('array'); + expect(res.body.balance_sheet.liabilities_equity.accounts).to.be.a('array'); }); - it('Should retrieve asset/liabilities balance sheet between the given date range.', async () => { + it('Should retrieve assets/liabilities total balance between the given date range.', async () => { + const res = await request() + .get('/api/financial_statements/balance_sheet') + .set('x-access-token', loginRes.body.token) + .query({ + display_columns_by: 'total', + from_date: '2012-01-01', + to_date: '2032-02-02', + }) + .send(); + + expect(res.body.balance_sheet.assets.accounts[0].balance).deep.equals({ + amount: 4000, formattedAmount: 4000, date: '2032-02-02', + }); + expect(res.body.balance_sheet.liabilities_equity.accounts[0].balance).deep.equals({ + amount: 2000, formattedAmount: 2000, date: '2032-02-02', + }); + }); + + it('Should retrieve asset/liabilities balance sheet with display columns by `year`.', async () => { const res = await request() .get('/api/financial_statements/balance_sheet') .set('x-access-token', loginRes.body.token) @@ -232,121 +312,142 @@ describe('routes: `/financial_statements`', () => { }) .send(); - const { balance_sheet: balanceSheet } = res.body; - const foundCreditAccount = balanceSheet.assets.find((account) => { - return account.id === debitAccount.id; - }); + expect(res.body.balance_sheet.assets.accounts[0].periods_balance.length).equals(7); + expect(res.body.balance_sheet.liabilities_equity.accounts[0].periods_balance.length).equals(7); - expect(foundCreditAccount.transactions.length).equals(6); - foundCreditAccount.transactions.forEach((transaction) => { - expect(transaction.balance).equals(0); - }); - - const foundDebitAccount = balanceSheet.liabilities_equity.find((account) => { - return account.id === creditAccount.id; - }); - - expect(foundDebitAccount.transactions.length).equals(6); - foundDebitAccount.transactions.forEach((transaction) => { - expect(transaction.balance).equals(0); - }); + expect(res.body.balance_sheet.assets.accounts[0].periods_balance).deep.equals([ + { + amount: 0, + formatted_amount: 0, + date: '2012', + }, + { + amount: 0, + formatted_amount: 0, + date: '2013', + }, + { + amount: 0, + formatted_amount: 0, + date: '2014', + }, + { + amount: 0, + formatted_amount: 0, + date: '2015', + }, + { + amount: 0, + formatted_amount: 0, + date: '2016', + }, + { + amount: 0, + formatted_amount: 0, + date: '2017', + }, + { + amount: 0, + formatted_amount: 0, + date: '2018', + }, + ]); }); - it('Should retrieve balance sheet with display columns day.', async () => { + it('Should retrieve balance sheet with display columns by `day`.', async () => { const res = await request() .get('/api/financial_statements/balance_sheet') .set('x-access-token', loginRes.body.token) .query({ display_columns_by: 'day', - from_date: '2020-03-01', - to_date: '2020-04-01', + from_date: '2020-01-08', + to_date: '2020-01-12', }) .send(); - const { balance_sheet: balanceSheet } = res.body; - - const foundDebitAccount = balanceSheet.assets.find((account) => { - return account.id === debitAccount.id; - }); - const foundCreditAccount = balanceSheet.liabilities_equity.find((account) => { - return account.id === creditAccount.id; - }); - - expect(foundDebitAccount.transactions.length).equals(31); - expect(foundCreditAccount.transactions.length).equals(31); - - foundDebitAccount.transactions.forEach((transaction) => { - expect(transaction.balance).equals(4000); - }); - foundCreditAccount.transactions.forEach((transaction) => { - expect(transaction.balance).equals(2000); - }); + expect(res.body.balance_sheet.assets.accounts[0].periods_balance).deep.equals([ + { date: '2020-01-08', formatted_amount: 0, amount: 0 }, + { date: '2020-01-09', formatted_amount: 0, amount: 0 }, + { date: '2020-01-10', formatted_amount: 4000, amount: 4000 }, + { date: '2020-01-11', formatted_amount: 4000, amount: 4000 }, + { date: '2020-01-12', formatted_amount: 4000, amount: 4000 }, + ]); }); - it('Should retrieve the balance sheet with display columns month.', async () => { + it('Should retrieve the balance sheet with display columns by `month`.', async () => { const res = await request() .get('/api/financial_statements/balance_sheet') .set('x-access-token', loginRes.body.token) .query({ display_columns_by: 'month', - from_date: '2020', - to_date: '2021', + from_date: '2019-07-01', + to_date: '2020-06-30', }) .send(); - const { balance_sheet: balanceSheet } = res.body; + expect(res.body.balance_sheet.assets.accounts[0].periods_balance.length).equals(12); + expect(res.body.balance_sheet.liabilities_equity.accounts[0].periods_balance.length).equals(12); - const foundDebitAccount = balanceSheet.assets.find((account) => { - return account.id === debitAccount.id; - }); - const foundCreditAccount = balanceSheet.liabilities_equity.find((account) => { - return account.id === creditAccount.id; - }); - - expect(foundDebitAccount.transactions.length).equals(12); - expect(foundCreditAccount.transactions.length).equals(12); - - foundDebitAccount.transactions.forEach((transaction) => { - expect(transaction.balance).equals(4000); - }); - foundCreditAccount.transactions.forEach((transaction) => { - expect(transaction.balance).equals(2000); - }); + expect(res.body.balance_sheet.assets.accounts[0].periods_balance).deep.equals([ + { date: '2019-07', formatted_amount: 0, amount: 0 }, + { date: '2019-08', formatted_amount: 0, amount: 0 }, + { date: '2019-09', formatted_amount: 0, amount: 0 }, + { date: '2019-10', formatted_amount: 0, amount: 0 }, + { date: '2019-11', formatted_amount: 0, amount: 0 }, + { date: '2019-12', formatted_amount: 0, amount: 0 }, + { date: '2020-01', formatted_amount: 4000, amount: 4000 }, + { date: '2020-02', formatted_amount: 4000, amount: 4000 }, + { date: '2020-03', formatted_amount: 4000, amount: 4000 }, + { date: '2020-04', formatted_amount: 4000, amount: 4000 }, + { date: '2020-05', formatted_amount: 4000, amount: 4000 }, + { date: '2020-06', formatted_amount: 4000, amount: 4000 }, + ]); }); - it('Should retrieve the balance sheet with display columns quarter.', async () => { + it('Should retrieve the balance sheet with display columns `quarter`.', async () => { const res = await request() .get('/api/financial_statements/balance_sheet') .set('x-access-token', loginRes.body.token) .query({ display_columns_by: 'quarter', - from_date: '2020', - to_date: '2021', + from_date: '2020-01-01', + to_date: '2020-12-31', }) .send(); - const { balance_sheet: balanceSheet } = res.body; - - const foundDebitAccount = balanceSheet.assets.find((account) => { - return account.id === debitAccount.id; - }); - const foundCreditAccount = balanceSheet.liabilities_equity.find((account) => { - return account.id === creditAccount.id; - }); - - expect(foundDebitAccount.transactions.length).equals(4); - expect(foundCreditAccount.transactions.length).equals(4); - - foundDebitAccount.transactions.forEach((transaction) => { - expect(transaction.balance).equals(4000); - }); - foundCreditAccount.transactions.forEach((transaction) => { - expect(transaction.balance).equals(2000); - }); + expect(res.body.balance_sheet.assets.accounts[0].periods_balance.length).equals(4); + expect(res.body.balance_sheet.assets.accounts[0].periods_balance).deep.equals([ + { date: '2020-03', formatted_amount: 4000, amount: 4000 }, + { date: '2020-06', formatted_amount: 4000, amount: 4000 }, + { date: '2020-09', formatted_amount: 4000, amount: 4000 }, + { date: '2020-12', formatted_amount: 4000, amount: 4000 }, + ]); }); - it('Should retrieve the balance sheet amounts without cents.', () => { - + it('Should retrieve the balance sheet amounts without cents.', async () => { + await create('account_transaction', { + debit: 0.25, credit: 0, account_id: debitAccount.id, date: '2020-1-10', + }); + const res = await request() + .get('/api/financial_statements/balance_sheet') + .set('x-access-token', loginRes.body.token) + .query({ + display_columns_by: 'quarter', + from_date: '2020-01-01', + to_date: '2020-12-31', + number_format: { + no_cents: true, + }, + }) + .send(); + + expect(res.body.balance_sheet.assets.accounts[0].periods_balance.length).equals(4); + expect(res.body.balance_sheet.assets.accounts[0].periods_balance).deep.equals([ + { date: '2020-03', formatted_amount: 4000, amount: 4000.25 }, + { date: '2020-06', formatted_amount: 4000, amount: 4000.25 }, + { date: '2020-09', formatted_amount: 4000, amount: 4000.25 }, + { date: '2020-12', formatted_amount: 4000, amount: 4000.25 }, + ]); }); it('Should retrieve the balance sheet amounts divided on 1000.', async () => { @@ -363,23 +464,13 @@ describe('routes: `/financial_statements`', () => { }) .send(); - const { balance_sheet: balanceSheet } = res.body; - const foundDebitAccount = balanceSheet.assets.find((account) => { - return account.id === debitAccount.id; - }); - const foundCreditAccount = balanceSheet.liabilities_equity.find((account) => { - return account.id === creditAccount.id; - }); - - expect(foundDebitAccount.transactions.length).equals(4); - expect(foundCreditAccount.transactions.length).equals(4); - - foundDebitAccount.transactions.forEach((transaction) => { - expect(transaction.balance).equals(4); - }); - foundCreditAccount.transactions.forEach((transaction) => { - expect(transaction.balance).equals(2); - }); + expect(res.body.balance_sheet.assets.accounts[0].periods_balance).deep.equals([ + { date: '2020-03', formatted_amount: 4, amount: 4000 }, + { date: '2020-06', formatted_amount: 4, amount: 4000 }, + { date: '2020-09', formatted_amount: 4, amount: 4000 }, + { date: '2020-12', formatted_amount: 4, amount: 4000 }, + { date: '2021-03', formatted_amount: 4, amount: 4000 }, + ]); }); it('Should not retrieve accounts has no transactions between the given date range in case query none_zero is true.', async () => { @@ -397,16 +488,8 @@ describe('routes: `/financial_statements`', () => { }) .send(); - const { balance_sheet: balanceSheet } = res.body; - const foundDebitAccount = balanceSheet.assets.find((account) => { - return account.id === debitAccount.id; - }); - - const foundCreditAccount = balanceSheet.liabilities_equity.find((account) => { - return account.id === creditAccount.id; - }); - expect(foundDebitAccount).equals(undefined); - expect(foundCreditAccount).equals(undefined); + expect(res.body.balance_sheet.assets.accounts.length).equals(0); + expect(res.body.balance_sheet.liabilities_equity.accounts.length).equals(0); }); }); @@ -441,14 +524,11 @@ describe('routes: `/financial_statements`', () => { // There is no transactions between these dates. from_date: '2002-01-01', to_date: '2003-01-01', + none_zero: true, }) .send(); - res.body.items.forEach((item) => { - expect(item.credit).equals(0); - expect(item.debit).equals(0); - expect(item.balance).equals(0); - }); + expect(res.body.items.length).equals(0); }); it('Should retrieve trial balance of accounts between the given date range.', async () => { @@ -506,6 +586,10 @@ describe('routes: `/financial_statements`', () => { }) .send(); }); + + it('Should retrieve associated account details in accounts list.', async () => { + + }); }); describe('routes: `/api/financial_statements/profit_loss_sheet`', () => { @@ -515,7 +599,111 @@ describe('routes: `/financial_statements`', () => { .send(); expect(res.status).equals(401); - expect(res.body.message).equals('unauthorzied'); + expect(res.body.message).equals('unauthorized'); + }); + + it('Should retrieve columns when display type `date_periods` and columns by `month` between date range.', async () => { + const res = await request() + .get('/api/financial_statements/profit_loss_sheet') + .set('x-access-token', loginRes.body.token) + .query({ + from_date: '2020-01-01', + to_date: '2020-12-12', + display_columns_type: 'date_periods', + display_columns_by: 'month', + }) + .send(); + + expect(res.body.columns.length).equals(12); + expect(res.body.columns).deep.equals([ + '2020-01', '2020-02', + '2020-03', '2020-04', + '2020-05', '2020-06', + '2020-07', '2020-08', + '2020-09', '2020-10', + '2020-11', '2020-12', + ]); + }); + + it('Should retrieve columns when display type `date_periods` and columns by `quarter`.', async () => { + const res = await request() + .get('/api/financial_statements/profit_loss_sheet') + .set('x-access-token', loginRes.body.token) + .query({ + from_date: moment().startOf('year').format('YYYY-MM-DD'), + to_date: moment().endOf('year').format('YYYY-MM-DD'), + display_columns_type: 'date_periods', + display_columns_by: 'month', + }) + .send(); + + expect(res.body.columns.length).equals(12); + expect(res.body.columns).deep.equals([ + '2020-03', '2020-06', '2020-09', '2020-12', + ]); + }); + + it('Should retrieve columns when display type `date_periods` and columns by `day` between date range.', async () => { + const res = await request() + .get('/api/financial_statements/profit_loss_sheet') + .set('x-access-token', loginRes.body.token) + .query({ + from_date: moment('2020-01-01').startOf('month').format('YYYY-MM-DD'), + to_date: moment('2020-01-01').endOf('month').format('YYYY-MM-DD'), + display_columns_type: 'date_periods', + display_columns_by: 'day', + }) + .send(); + + expect(res.body.columns.length).equals(31); + expect(res.body.columns).deep.equals([ + '2020-01-01', '2020-01-02', '2020-01-03', + '2020-01-04', '2020-01-05', '2020-01-06', + '2020-01-07', '2020-01-08', '2020-01-09', + '2020-01-10', '2020-01-11', '2020-01-12', + '2020-01-13', '2020-01-14', '2020-01-15', + '2020-01-16', '2020-01-17', '2020-01-18', + '2020-01-19', '2020-01-20', '2020-01-21', + '2020-01-22', '2020-01-23', '2020-01-24', + '2020-01-25', '2020-01-26', '2020-01-27', + '2020-01-28', '2020-01-29', '2020-01-30', + '2020-01-31', + ]); + }); + + it('Should retrieve all income accounts even it has no transactions.', async () => { + const res = await request() + .get('/api/financial_statements/profit_loss_sheet') + .set('x-access-token', loginRes.body.token) + .query({ + from_date: moment('2020-01-01').startOf('month').format('YYYY-MM-DD'), + to_date: moment('2020-01-01').endOf('month').format('YYYY-MM-DD'), + display_columns_type: 'total', + display_columns_by: 'day', + }) + .send(); + + console.log(res.body); + + // expect(res.body.income.accounts.length).equals(2); + // expect(res.body.income.accounts[0].name).to.be.an('string'); + // expect(res.body.income.accounts[0].code).to.be.an('string'); + // expect(res.body.income.accounts[0].periods).to.be.an('array'); + // expect(res.body.income.accounts[0].periods.length).equals(31); + }); + + it('Should retrieve total of each income account when display columns by `total`.', async () => { + const res = await request() + .get('/api/financial_statements/profit_loss_sheet') + .set('x-access-token', loginRes.body.token) + .query({ + from_date: moment('2020-01-01').startOf('month').format('YYYY-MM-DD'), + to_date: moment('2020-01-01').endOf('month').format('YYYY-MM-DD'), + display_columns_by: 'day', + }) + .send(); + + expect(res.body.income).deep.equals(); }); it('Should retrieve credit sumation of income accounts.', async () => { @@ -531,6 +719,8 @@ describe('routes: `/financial_statements`', () => { }) .send(); + console.log(res.body); + res.body.income.accounts[0].dates.forEach((item) => { expect(item.rawAmount).equals(2000); }); diff --git a/server/tests/routes/views.test.js b/server/tests/routes/views.test.js index d4ecdbf0b..e82ede2b7 100644 --- a/server/tests/routes/views.test.js +++ b/server/tests/routes/views.test.js @@ -11,7 +11,7 @@ import ViewColumn from '../../src/models/ViewColumn'; let loginRes; -describe.only('routes: `/views`', () => { +describe('routes: `/views`', () => { beforeEach(async () => { loginRes = await login(); }); diff --git a/server/tests/testInit.js b/server/tests/testInit.js index e75284053..86667ee09 100644 --- a/server/tests/testInit.js +++ b/server/tests/testInit.js @@ -1,22 +1,29 @@ import chai from 'chai'; import chaiHttp from 'chai-http'; import chaiThings from 'chai-things'; + import knex from '@/database/knex'; import '@/models'; import app from '@/app'; import factory from '@/database/factories'; +import knexConfig from '@/../knexfile'; +import dbManager from '@/database/manager'; // import { hashPassword } from '@/utils'; const request = () => chai.request(app); const { expect } = chai; +before(async () => { + await dbManager.dropDb(); + await dbManager.createDb('ratteb'); +}); + beforeEach(async () => { await knex.migrate.rollback(); await knex.migrate.latest(); }); afterEach(async () => { - await knex.migrate.rollback(); }); chai.use(chaiHttp);