diff --git a/gmail/.clasp.json b/gmail/.clasp.json deleted file mode 100644 index fa7fcc7bd..000000000 --- a/gmail/.clasp.json +++ /dev/null @@ -1,4 +0,0 @@ -{ - "scriptId": "1wAzxJBkBYbIs_P2K76RpoBGovgjNNfSoRASf7660wgkxwYa89WZmh2gS", - "projectId": "odoo-gmail-304313" -} diff --git a/gmail/.claspignore b/gmail/.claspignore deleted file mode 100644 index 36f2a398f..000000000 --- a/gmail/.claspignore +++ /dev/null @@ -1,13 +0,0 @@ -.git -.git/* -node_modules -node_modules/** -node_modules/**/.*/** -node_modules/**/.* - -# ignore all files… -**/** - -# include appscript and build result -!appsscript.json -!build/*.js diff --git a/gmail/.prettierignore b/gmail/.prettierignore new file mode 100644 index 000000000..1eae0cf67 --- /dev/null +++ b/gmail/.prettierignore @@ -0,0 +1,2 @@ +dist/ +node_modules/ diff --git a/gmail/.prettierrc b/gmail/.prettierrc index 7c8e25071..e4a483fb7 100644 --- a/gmail/.prettierrc +++ b/gmail/.prettierrc @@ -2,6 +2,9 @@ "semi": true, "trailingComma": "all", "singleQuote": false, - "printWidth": 120, - "tabWidth": 4 + "printWidth": 100, + "tabWidth": 4, + "plugins": [ + "prettier-plugin-organize-imports" + ] } diff --git a/gmail/README.md b/gmail/README.md index 36b4ad7f8..76d57f560 100644 --- a/gmail/README.md +++ b/gmail/README.md @@ -2,89 +2,30 @@ This addons allows you to find information about the sender of the emails you received and also to link your Gmail contacts to your Odoo partners, to create leads from Gmail,... -![Odoo Gmail Extension](./assets/img/readme.png) - # Development -## Requirements -First you need npm, -> apt-get install -y npm - -Install the dependencies -> npm install - -## Prettier -You should auto-format the code using the prettier configuration, -> `npx prettier --config .prettierrc 'src/**/*.ts' --write` - -## Compiling - -We use [rollup.js](https://github.com/rollup/rollup) to package all of the source files into a single one. -This is necessary as App Scripts do not support ES6 import/export statements yet. - -Once you have applied the necessary changes, run the following command: -> npx rollup -c - -This will simultaneously compile and package the typescript sourcecode inside `build/main.js` - -Now all you need to do is upload the script to your account and deploy it! - -## Uploading method 1: Manually copying the file -If you do not plan on updating this script regularly, perhaps you will prefer using Google's GUI. - -- Head to [the App Scripts manager](https://script.google.com/) and create a project -- Go to the project settings and enable appscript.json editing: `Show "appsscript.json" manifest file in editor` -- Copy the contents of your local `appscript.json` to the remote one in the project editor -- Create a file `main.gs` and remove the existing `Code.gs` if any. -- Copy the contents of your local `build/main.js` to the `main.gs` file in the project editor +Create the database and fill the credentials in `consts.ts` +> psql -U root -d postgres -f init_db.sql -## Uploading method 2: Using Clasp -You may want to use the Google's CLI tool [clasp](https://github.com/google/clasp) to manage, compile and update your app script. +To serve the addin, you need a public HTTPS connection to your application. +You cannot use nrgok, because Gmail store all images we use in the addin, +and it won't fetch them if they come from a free ngrok domain name. +So the simplest is to use a VPS (and to do a reverse SSH proxy to develop locally). -First install -> npm install -g @google/clasp - -Login to your account to be able to push on your Gmail project, -> clasp login - -Note: the `--no-localhost` option we previously recommended was [deprecated by google](https://developers.google.com/identity/protocols/oauth2/resources/oob-migration) - -### If you already have a project -Update `.clasp.json` to use your own script id and project. -If you do not have a specific project, use `Default`. - -### If you do not have a project yet -Remove `.clasp.json` - -Create a project -> clasp create - -For the project type, select "api". - -### Push your project -Push the project -> clasp push - - -# Deployment -Finally, you can enable the add-on for your account. - -Head to [the App Scripts manager](https://script.google.com/). -- Select your project and click "Deploy". -- For testing on your account just select "Test deployments". "Google Workspace Add-on" should be automatically selected as the type. -- Click "Install" and the add-on should appear in the addons tab of Gmail. +Then run +> npm install +> npm run dev -You're done! +Go to this page: +https://console.cloud.google.com/apis/api/appsmarket-component.googleapis.com/googleapps_sdk_gsao -For final deployments you will need to create a Google Cloud Project with the GMail API and link it to this script. -Refer to Google's documentation for more information. +Then create an HTTP deployment using `deployment.json`, and update the URL in `onTriggerFunction` to contain your ngrok URL. -# Documentation -`GmailApp` object, -https://developers.google.com/apps-script/reference/gmail/gmail-app +Then click on "install", and the addin will be available in your Gmail account. -`URL fetch API` -https://developers.google.com/apps-script/reference/url-fetch +Before committing, run prettier +> npm run prettier -`Storage` -https://developers.google.com/apps-script/reference/cache -https://developers.google.com/apps-script/reference/properties +# Production +Update the `CLIENT_ID` and the public URL in `consts.ts`, then run +> npm run build +> node dist diff --git a/gmail/appsscript.json b/gmail/appsscript.json deleted file mode 100644 index 0f1164ef0..000000000 --- a/gmail/appsscript.json +++ /dev/null @@ -1,54 +0,0 @@ -{ - "oauthScopes": [ - "https://www.googleapis.com/auth/userinfo.email", - "https://www.googleapis.com/auth/gmail.addons.execute", - "https://www.googleapis.com/auth/gmail.addons.current.message.readonly", - "https://www.googleapis.com/auth/script.external_request" - ], - "gmail": { - "name": "Odoo", - "logoUrl": "https://raw.githubusercontent.com/odoo/mail-client-extensions/master/outlook/assets/odoo.png", - "contextualTriggers": [ - { - "unconditional": {}, - "onTriggerFunction": "onGmailMessageOpen" - } - ], - "primaryColor": "#875A7B", - "secondaryColor": "#00A09D", - "openLinkUrlPrefixes": ["*"] - }, - "urlFetchWhitelist": [ - "https://*.odoo.com/mail_plugin/get_translations", - "https://*.odoo.com/mail_plugin/partner/get", - "https://*.odoo.com/mail_plugin/log_mail_content", - "https://*.odoo.com/mail_plugin/partner/search", - "https://*.odoo.com/mail_plugin/partner/create", - "https://*.odoo.com/mail_plugin/partner/enrich_and_create_company", - "https://*.odoo.com/mail_plugin/partner/enrich_and_update_company", - "https://*.odoo.com/mail_plugin/lead/create", - "https://*.odoo.com/mail_plugin/ticket/create", - "https://*.odoo.com/mail_plugin/project/search", - "https://*.odoo.com/mail_plugin/project/create", - "https://*.odoo.com/mail_plugin/task/create", - "https://*.odoo.com/web/login", - "https://*.odoo.com/mail_plugin/auth", - "https://*.odoo.com/mail_plugin/auth/access_token", - "https://odoo.com/mail_plugin/get_translations", - "https://odoo.com/mail_plugin/partner/get", - "https://odoo.com/mail_plugin/log_mail_content", - "https://odoo.com/mail_plugin/partner/search", - "https://odoo.com/mail_plugin/partner/create", - "https://odoo.com/mail_plugin/partner/enrich_and_create_company", - "https://odoo.com/mail_plugin/partner/enrich_and_update_company", - "https://odoo.com/mail_plugin/lead/create", - "https://odoo.com/mail_plugin/ticket/create", - "https://odoo.com/mail_plugin/project/search", - "https://odoo.com/mail_plugin/project/create", - "https://odoo.com/mail_plugin/task/create", - "https://odoo.com/web/login", - "https://odoo.com/mail_plugin/auth", - "https://odoo.com/mail_plugin/auth/access_token", - "https://iap-services.odoo.com/iap/mail_extension/enrich" - ] -} diff --git a/gmail/assets/img/odoo.png b/gmail/assets/img/odoo.png deleted file mode 100644 index 3d83a91f9..000000000 Binary files a/gmail/assets/img/odoo.png and /dev/null differ diff --git a/gmail/assets/img/odoo_full.png b/gmail/assets/img/odoo_full.png deleted file mode 100644 index 8acd02add..000000000 Binary files a/gmail/assets/img/odoo_full.png and /dev/null differ diff --git a/gmail/assets/img/readme.png b/gmail/assets/img/readme.png deleted file mode 100644 index 37952b8b8..000000000 Binary files a/gmail/assets/img/readme.png and /dev/null differ diff --git a/gmail/deployment.json b/gmail/deployment.json new file mode 100644 index 000000000..8b9c1b0ef --- /dev/null +++ b/gmail/deployment.json @@ -0,0 +1,26 @@ +{ + "oauthScopes": [ + "https://www.googleapis.com/auth/userinfo.email", + "https://www.googleapis.com/auth/gmail.addons.execute", + "https://www.googleapis.com/auth/gmail.addons.current.message.readonly" + ], + "addOns": { + "common": { + "name": "Odoo", + "logoUrl": "https://raw.githubusercontent.com/odoo/mail-client-extensions/master/outlook/assets/odoo.png" + }, + "gmail": { + "primaryColor": "#875A7B", + "secondaryColor": "#00A09D", + "contextualTriggers": [ + { + "unconditional": {}, + "onTriggerFunction": "https://mister7f.xyz/on_open_email" + } + ] + }, + "httpOptions": { + "granularOauthPermissionSupport": "OPT_IN" + } + } +} diff --git a/gmail/iap_instruction.md b/gmail/iap_instruction.md deleted file mode 100644 index 30722153b..000000000 --- a/gmail/iap_instruction.md +++ /dev/null @@ -1,13 +0,0 @@ -# Shared secret between IAP and the add-on -Go to your Google project, -> clasp open - -Then File -> Project properties -> Script Properties - -And add a row, -> `ODOO_SHARED_SECRET` `` - -On the IAP side, add a system parameter -> `iap_mail_extension.shared_secret` `` - -This secret will allow the add-on to authenticate to IAP. diff --git a/gmail/init_db.sql b/gmail/init_db.sql new file mode 100644 index 000000000..10d2bb281 --- /dev/null +++ b/gmail/init_db.sql @@ -0,0 +1,25 @@ +CREATE DATABASE gmail_addin_db; + +\c gmail_addin_db; + +CREATE TABLE users_settings ( + id SERIAL PRIMARY KEY, + email TEXT UNIQUE NOT NULL, + odoo_url TEXT, + odoo_token TEXT, + login_token TEXT, + login_token_expire_at TIMESTAMP WITH TIME ZONE, + translations JSON, + translations_expire_at TIMESTAMP WITH TIME ZONE +); + +-- Remember that the user logged the email on the giver record +CREATE TABLE email_logs ( + id SERIAL PRIMARY KEY, + user_id INTEGER NOT NULL, + message_id TEXT NOT NULL, + res_id INTEGER NOT NULL, + res_model TEXT NOT NULL, + create_date TIMESTAMP WITH TIME ZONE DEFAULT NOW(), + FOREIGN KEY (user_id) REFERENCES users_settings(id) ON DELETE CASCADE +); diff --git a/gmail/package-lock.json b/gmail/package-lock.json index 93b2c28c3..5436d5f4b 100644 --- a/gmail/package-lock.json +++ b/gmail/package-lock.json @@ -1,17 +1,2832 @@ { + "name": "gmail_http", + "version": "1.0.0", + "lockfileVersion": 3, "requires": true, - "lockfileVersion": 1, - "dependencies": { - "@types/google-apps-script": { - "version": "1.0.31", - "resolved": "https://registry.npmjs.org/@types/google-apps-script/-/google-apps-script-1.0.31.tgz", - "integrity": "sha512-tgsJKk20fwFoh0Ml4Li3pqKQ5uu3Nr3XeRsee2+pkPGrJxDlA3qsHAA2q3/HRv5yi9U6QVvdGwJ16USnmA7wAA==" - }, - "prettier": { + "packages": { + "": { + "name": "gmail_http", + "version": "1.0.0", + "dependencies": { + "@resvg/resvg-js": "^2.6.2", + "dotenv": "^17.2.3", + "express": "^5.2.1", + "express-async-handler": "^1.2.0", + "google-auth-library": "^10.5.0", + "googleapis": "^167.0.0", + "jsonwebtoken": "^9.0.3", + "mailparser": "^3.9.0", + "node-cron": "^4.2.1", + "pg": "^8.16.3" + }, + "devDependencies": { + "@types/express": "^5.0.6", + "@types/node": "^24.10.1", + "@types/pg": "^8.15.6", + "concurrently": "^9.2.1", + "prettier": "^3.7.4", + "prettier-plugin-organize-imports": "^4.3.0", + "tsx": "^4.21.0", + "typescript": "^5.9.3" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.27.1", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.27.1.tgz", + "integrity": "sha512-z3H/HYI9MM0HTv3hQZ81f+AKb+yEoCRlUby1F80vbQ5XdzEMyY/9iNlAmhqiBKw4MJXwfgsh7ERGEOhrM1niMA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@isaacs/cliui": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz", + "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==", + "license": "ISC", + "dependencies": { + "string-width": "^5.1.2", + "string-width-cjs": "npm:string-width@^4.2.0", + "strip-ansi": "^7.0.1", + "strip-ansi-cjs": "npm:strip-ansi@^6.0.1", + "wrap-ansi": "^8.1.0", + "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@pkgjs/parseargs": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz", + "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==", + "license": "MIT", + "optional": true, + "engines": { + "node": ">=14" + } + }, + "node_modules/@resvg/resvg-js": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/@resvg/resvg-js/-/resvg-js-2.6.2.tgz", + "integrity": "sha512-xBaJish5OeGmniDj9cW5PRa/PtmuVU3ziqrbr5xJj901ZDN4TosrVaNZpEiLZAxdfnhAe7uQ7QFWfjPe9d9K2Q==", + "license": "MPL-2.0", + "engines": { + "node": ">= 10" + }, + "optionalDependencies": { + "@resvg/resvg-js-android-arm-eabi": "2.6.2", + "@resvg/resvg-js-android-arm64": "2.6.2", + "@resvg/resvg-js-darwin-arm64": "2.6.2", + "@resvg/resvg-js-darwin-x64": "2.6.2", + "@resvg/resvg-js-linux-arm-gnueabihf": "2.6.2", + "@resvg/resvg-js-linux-arm64-gnu": "2.6.2", + "@resvg/resvg-js-linux-arm64-musl": "2.6.2", + "@resvg/resvg-js-linux-x64-gnu": "2.6.2", + "@resvg/resvg-js-linux-x64-musl": "2.6.2", + "@resvg/resvg-js-win32-arm64-msvc": "2.6.2", + "@resvg/resvg-js-win32-ia32-msvc": "2.6.2", + "@resvg/resvg-js-win32-x64-msvc": "2.6.2" + } + }, + "node_modules/@resvg/resvg-js-linux-x64-gnu": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/@resvg/resvg-js-linux-x64-gnu/-/resvg-js-linux-x64-gnu-2.6.2.tgz", + "integrity": "sha512-IVUe+ckIerA7xMZ50duAZzwf1U7khQe2E0QpUxu5MBJNao5RqC0zwV/Zm965vw6D3gGFUl7j4m+oJjubBVoftw==", + "cpu": [ + "x64" + ], + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 10" + } + }, + "node_modules/@selderee/plugin-htmlparser2": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/@selderee/plugin-htmlparser2/-/plugin-htmlparser2-0.11.0.tgz", + "integrity": "sha512-P33hHGdldxGabLFjPPpaTxVolMrzrcegejx+0GxjrIb9Zv48D8yAIA/QTDR2dFl7Uz7urX8aX6+5bCZslr+gWQ==", + "license": "MIT", + "dependencies": { + "domhandler": "^5.0.3", + "selderee": "^0.11.0" + }, + "funding": { + "url": "https://ko-fi.com/killymxi" + } + }, + "node_modules/@types/body-parser": { + "version": "1.19.6", + "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.6.tgz", + "integrity": "sha512-HLFeCYgz89uk22N5Qg3dvGvsv46B8GLvKKo1zKG4NybA8U2DiEO3w9lqGg29t/tfLRJpJ6iQxnVw4OnB7MoM9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/connect": "*", + "@types/node": "*" + } + }, + "node_modules/@types/connect": { + "version": "3.4.38", + "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.38.tgz", + "integrity": "sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/express": { + "version": "5.0.6", + "resolved": "https://registry.npmjs.org/@types/express/-/express-5.0.6.tgz", + "integrity": "sha512-sKYVuV7Sv9fbPIt/442koC7+IIwK5olP1KWeD88e/idgoJqDm3JV/YUiPwkoKK92ylff2MGxSz1CSjsXelx0YA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/body-parser": "*", + "@types/express-serve-static-core": "^5.0.0", + "@types/serve-static": "^2" + } + }, + "node_modules/@types/express-serve-static-core": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-5.1.0.tgz", + "integrity": "sha512-jnHMsrd0Mwa9Cf4IdOzbz543y4XJepXrbia2T4b6+spXC2We3t1y6K44D3mR8XMFSXMCf3/l7rCgddfx7UNVBA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*", + "@types/qs": "*", + "@types/range-parser": "*", + "@types/send": "*" + } + }, + "node_modules/@types/http-errors": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@types/http-errors/-/http-errors-2.0.5.tgz", + "integrity": "sha512-r8Tayk8HJnX0FztbZN7oVqGccWgw98T/0neJphO91KkmOzug1KkofZURD4UaD5uH8AqcFLfdPErnBod0u71/qg==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/node": { + "version": "24.10.1", + "resolved": "https://registry.npmjs.org/@types/node/-/node-24.10.1.tgz", + "integrity": "sha512-GNWcUTRBgIRJD5zj+Tq0fKOJ5XZajIiBroOF0yvj2bSU1WvNdYS/dn9UxwsujGW4JX06dnHyjV2y9rRaybH0iQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "undici-types": "~7.16.0" + } + }, + "node_modules/@types/pg": { + "version": "8.15.6", + "resolved": "https://registry.npmjs.org/@types/pg/-/pg-8.15.6.tgz", + "integrity": "sha512-NoaMtzhxOrubeL/7UZuNTrejB4MPAJ0RpxZqXQf2qXuVlTPuG6Y8p4u9dKRaue4yjmC7ZhzVO2/Yyyn25znrPQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*", + "pg-protocol": "*", + "pg-types": "^2.2.0" + } + }, + "node_modules/@types/qs": { + "version": "6.14.0", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.14.0.tgz", + "integrity": "sha512-eOunJqu0K1923aExK6y8p6fsihYEn/BYuQ4g0CxAAgFc4b/ZLN4CrsRZ55srTdqoiLzU2B2evC+apEIxprEzkQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/range-parser": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.7.tgz", + "integrity": "sha512-hKormJbkJqzQGhziax5PItDUTMAM9uE2XXQmM37dyd4hVM+5aVl7oVxMVUiVQn2oCQFN/LKCZdvSM0pFRqbSmQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/send": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@types/send/-/send-1.2.1.tgz", + "integrity": "sha512-arsCikDvlU99zl1g69TcAB3mzZPpxgw0UQnaHeC1Nwb015xp8bknZv5rIfri9xTOcMuaVgvabfIRA7PSZVuZIQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/serve-static": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-2.2.0.tgz", + "integrity": "sha512-8mam4H1NHLtu7nmtalF7eyBH14QyOASmcxHhSfEoRyr0nP/YdoesEtU+uSRvMe96TW/HPTtkoKqQLl53N7UXMQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/http-errors": "*", + "@types/node": "*" + } + }, + "node_modules/@zone-eu/mailsplit": { + "version": "5.4.8", + "resolved": "https://registry.npmjs.org/@zone-eu/mailsplit/-/mailsplit-5.4.8.tgz", + "integrity": "sha512-eEyACj4JZ7sjzRvy26QhLgKEMWwQbsw1+QZnlLX+/gihcNH07lVPOcnwf5U6UAL7gkc//J3jVd76o/WS+taUiA==", + "license": "(MIT OR EUPL-1.1+)", + "dependencies": { + "libbase64": "1.3.0", + "libmime": "5.3.7", + "libqp": "2.1.1" + } + }, + "node_modules/accepts": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-2.0.0.tgz", + "integrity": "sha512-5cvg6CtKwfgdmVqY1WIiXKc3Q1bkRqGLi+2W/6ao+6Y7gu/RCwRuAhGEzh5B4KlszSuTLgZYuqFqo5bImjNKng==", + "license": "MIT", + "dependencies": { + "mime-types": "^3.0.0", + "negotiator": "^1.0.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/agent-base": { + "version": "7.1.4", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.4.tgz", + "integrity": "sha512-MnA+YT8fwfJPgBx3m60MNqakm30XOkyIoH1y6huTQvC0PwZG7ki8NacLBcrPbNoo8vEZy7Jpuk7+jMO+CUovTQ==", + "license": "MIT", + "engines": { + "node": ">= 14" + } + }, + "node_modules/ansi-regex": { + "version": "6.2.2", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.2.2.tgz", + "integrity": "sha512-Bq3SmSpyFHaWjPk8If9yc6svM8c56dB5BAtW4Qbw5jHTwwXXcTLoRMkpDJp6VL0XzlWaCHTXrkFURMYmD0sLqg==", + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-regex?sponsor=1" + } + }, + "node_modules/ansi-styles": { + "version": "6.2.3", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.3.tgz", + "integrity": "sha512-4Dj6M28JB+oAH8kFkTLUo+a2jwOFkuqb3yucU0CANcRRUbxS0cP0nZYCGjcc3BNXwRIsUVmDGgzawme7zvJHvg==", + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "license": "MIT" + }, + "node_modules/base64-js": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/bignumber.js": { + "version": "9.3.1", + "resolved": "https://registry.npmjs.org/bignumber.js/-/bignumber.js-9.3.1.tgz", + "integrity": "sha512-Ko0uX15oIUS7wJ3Rb30Fs6SkVbLmPBAKdlm7q9+ak9bbIeFf0MwuBsQV6z7+X768/cHsfg+WlysDWJcmthjsjQ==", + "license": "MIT", + "engines": { + "node": "*" + } + }, + "node_modules/body-parser": { "version": "2.2.1", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.2.1.tgz", - "integrity": "sha512-PqyhM2yCjg/oKkFPtTGUojv7gnZAoG80ttl45O6x2Ug/rMJw4wcc9k6aaf2hibP7BGVCCM33gZoGjyvt9mm16Q==", - "dev": true + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-2.2.1.tgz", + "integrity": "sha512-nfDwkulwiZYQIGwxdy0RUmowMhKcFVcYXUU7m4QlKYim1rUtg83xm2yjZ40QjDuc291AJjjeSc9b++AWHSgSHw==", + "license": "MIT", + "dependencies": { + "bytes": "^3.1.2", + "content-type": "^1.0.5", + "debug": "^4.4.3", + "http-errors": "^2.0.0", + "iconv-lite": "^0.7.0", + "on-finished": "^2.4.1", + "qs": "^6.14.0", + "raw-body": "^3.0.1", + "type-is": "^2.0.1" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/brace-expansion": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.2.tgz", + "integrity": "sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==", + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==", + "license": "BSD-3-Clause" + }, + "node_modules/bytes": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", + "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/call-bind-apply-helpers": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", + "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/call-bound": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", + "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "get-intrinsic": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/chalk/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/cliui": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dev": true, + "license": "ISC", + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/cliui/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/cliui/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/cliui/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true, + "license": "MIT" + }, + "node_modules/cliui/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cliui/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cliui/node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "license": "MIT", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "license": "MIT" + }, + "node_modules/concurrently": { + "version": "9.2.1", + "resolved": "https://registry.npmjs.org/concurrently/-/concurrently-9.2.1.tgz", + "integrity": "sha512-fsfrO0MxV64Znoy8/l1vVIjjHa29SZyyqPgQBwhiDcaW8wJc2W3XWVOGx4M3oJBnv/zdUZIIp1gDeS98GzP8Ng==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "4.1.2", + "rxjs": "7.8.2", + "shell-quote": "1.8.3", + "supports-color": "8.1.1", + "tree-kill": "1.2.2", + "yargs": "17.7.2" + }, + "bin": { + "conc": "dist/bin/concurrently.js", + "concurrently": "dist/bin/concurrently.js" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/open-cli-tools/concurrently?sponsor=1" + } + }, + "node_modules/concurrently/node_modules/supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/supports-color?sponsor=1" + } + }, + "node_modules/content-disposition": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-1.0.1.tgz", + "integrity": "sha512-oIXISMynqSqm241k6kcQ5UwttDILMK4BiurCfGEREw6+X9jkkpEe5T9FZaApyLGGOnFuyMWZpdolTXMtvEJ08Q==", + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/content-type": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", + "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie": { + "version": "0.7.2", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.2.tgz", + "integrity": "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.2.2.tgz", + "integrity": "sha512-D76uU73ulSXrD1UXF4KE2TMxVVwhsnCgfAyTg9k8P6KGZjlXKrOLe4dJQKI3Bxi5wjesZoFXJWElNWBjPZMbhg==", + "license": "MIT", + "engines": { + "node": ">=6.6.0" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/data-uri-to-buffer": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-4.0.1.tgz", + "integrity": "sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==", + "license": "MIT", + "engines": { + "node": ">= 12" + } + }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/deepmerge": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", + "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/depd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", + "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/dom-serializer": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-2.0.0.tgz", + "integrity": "sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==", + "license": "MIT", + "dependencies": { + "domelementtype": "^2.3.0", + "domhandler": "^5.0.2", + "entities": "^4.2.0" + }, + "funding": { + "url": "https://github.com/cheeriojs/dom-serializer?sponsor=1" + } + }, + "node_modules/domelementtype": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.3.0.tgz", + "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/fb55" + } + ], + "license": "BSD-2-Clause" + }, + "node_modules/domhandler": { + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-5.0.3.tgz", + "integrity": "sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==", + "license": "BSD-2-Clause", + "dependencies": { + "domelementtype": "^2.3.0" + }, + "engines": { + "node": ">= 4" + }, + "funding": { + "url": "https://github.com/fb55/domhandler?sponsor=1" + } + }, + "node_modules/domutils": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.2.2.tgz", + "integrity": "sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw==", + "license": "BSD-2-Clause", + "dependencies": { + "dom-serializer": "^2.0.0", + "domelementtype": "^2.3.0", + "domhandler": "^5.0.3" + }, + "funding": { + "url": "https://github.com/fb55/domutils?sponsor=1" + } + }, + "node_modules/dotenv": { + "version": "17.2.3", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-17.2.3.tgz", + "integrity": "sha512-JVUnt+DUIzu87TABbhPmNfVdBDt18BLOWjMUFJMSi/Qqg7NTYtabbvSNJGOJ7afbRuv9D/lngizHtP7QyLQ+9w==", + "license": "BSD-2-Clause", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://dotenvx.com" + } + }, + "node_modules/dunder-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", + "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "es-errors": "^1.3.0", + "gopd": "^1.2.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/eastasianwidth": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", + "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", + "license": "MIT" + }, + "node_modules/ecdsa-sig-formatter": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", + "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", + "license": "Apache-2.0", + "dependencies": { + "safe-buffer": "^5.0.1" + } + }, + "node_modules/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", + "license": "MIT" + }, + "node_modules/emoji-regex": { + "version": "9.2.2", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", + "license": "MIT" + }, + "node_modules/encodeurl": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", + "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/encoding-japanese": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/encoding-japanese/-/encoding-japanese-2.2.0.tgz", + "integrity": "sha512-EuJWwlHPZ1LbADuKTClvHtwbaFn4rOD+dRAbWysqEOXRc2Uui0hJInNJrsdH0c+OhJA4nrCBdSkW4DD5YxAo6A==", + "license": "MIT", + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/entities": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", + "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==", + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.12" + }, + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/es-define-property": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", + "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-object-atoms": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", + "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/esbuild": { + "version": "0.27.1", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.27.1.tgz", + "integrity": "sha512-yY35KZckJJuVVPXpvjgxiCuVEJT67F6zDeVTv4rizyPrfGBUpZQsvmxnN+C371c2esD/hNMjj4tpBhuueLN7aA==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=18" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.27.1", + "@esbuild/android-arm": "0.27.1", + "@esbuild/android-arm64": "0.27.1", + "@esbuild/android-x64": "0.27.1", + "@esbuild/darwin-arm64": "0.27.1", + "@esbuild/darwin-x64": "0.27.1", + "@esbuild/freebsd-arm64": "0.27.1", + "@esbuild/freebsd-x64": "0.27.1", + "@esbuild/linux-arm": "0.27.1", + "@esbuild/linux-arm64": "0.27.1", + "@esbuild/linux-ia32": "0.27.1", + "@esbuild/linux-loong64": "0.27.1", + "@esbuild/linux-mips64el": "0.27.1", + "@esbuild/linux-ppc64": "0.27.1", + "@esbuild/linux-riscv64": "0.27.1", + "@esbuild/linux-s390x": "0.27.1", + "@esbuild/linux-x64": "0.27.1", + "@esbuild/netbsd-arm64": "0.27.1", + "@esbuild/netbsd-x64": "0.27.1", + "@esbuild/openbsd-arm64": "0.27.1", + "@esbuild/openbsd-x64": "0.27.1", + "@esbuild/openharmony-arm64": "0.27.1", + "@esbuild/sunos-x64": "0.27.1", + "@esbuild/win32-arm64": "0.27.1", + "@esbuild/win32-ia32": "0.27.1", + "@esbuild/win32-x64": "0.27.1" + } + }, + "node_modules/escalade": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", + "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==", + "license": "MIT" + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/express": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/express/-/express-5.2.1.tgz", + "integrity": "sha512-hIS4idWWai69NezIdRt2xFVofaF4j+6INOpJlVOLDO8zXGpUVEVzIYk12UUi2JzjEzWL3IOAxcTubgz9Po0yXw==", + "license": "MIT", + "dependencies": { + "accepts": "^2.0.0", + "body-parser": "^2.2.1", + "content-disposition": "^1.0.0", + "content-type": "^1.0.5", + "cookie": "^0.7.1", + "cookie-signature": "^1.2.1", + "debug": "^4.4.0", + "depd": "^2.0.0", + "encodeurl": "^2.0.0", + "escape-html": "^1.0.3", + "etag": "^1.8.1", + "finalhandler": "^2.1.0", + "fresh": "^2.0.0", + "http-errors": "^2.0.0", + "merge-descriptors": "^2.0.0", + "mime-types": "^3.0.0", + "on-finished": "^2.4.1", + "once": "^1.4.0", + "parseurl": "^1.3.3", + "proxy-addr": "^2.0.7", + "qs": "^6.14.0", + "range-parser": "^1.2.1", + "router": "^2.2.0", + "send": "^1.1.0", + "serve-static": "^2.2.0", + "statuses": "^2.0.1", + "type-is": "^2.0.1", + "vary": "^1.1.2" + }, + "engines": { + "node": ">= 18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/express-async-handler": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/express-async-handler/-/express-async-handler-1.2.0.tgz", + "integrity": "sha512-rCSVtPXRmQSW8rmik/AIb2P0op6l7r1fMW538yyvTMltCO4xQEWMmobfrIxN2V1/mVrgxB8Az3reYF6yUZw37w==", + "license": "MIT" + }, + "node_modules/extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", + "license": "MIT" + }, + "node_modules/fetch-blob": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/fetch-blob/-/fetch-blob-3.2.0.tgz", + "integrity": "sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/jimmywarting" + }, + { + "type": "paypal", + "url": "https://paypal.me/jimmywarting" + } + ], + "license": "MIT", + "dependencies": { + "node-domexception": "^1.0.0", + "web-streams-polyfill": "^3.0.3" + }, + "engines": { + "node": "^12.20 || >= 14.13" + } + }, + "node_modules/finalhandler": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-2.1.1.tgz", + "integrity": "sha512-S8KoZgRZN+a5rNwqTxlZZePjT/4cnm0ROV70LedRHZ0p8u9fRID0hJUZQpkKLzro8LfmC8sx23bY6tVNxv8pQA==", + "license": "MIT", + "dependencies": { + "debug": "^4.4.0", + "encodeurl": "^2.0.0", + "escape-html": "^1.0.3", + "on-finished": "^2.4.1", + "parseurl": "^1.3.3", + "statuses": "^2.0.1" + }, + "engines": { + "node": ">= 18.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/foreground-child": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.1.tgz", + "integrity": "sha512-gIXjKqtFuWEgzFRJA9WCQeSJLZDjgJUOMCMzxtvFq/37KojM1BFGufqsCy0r4qSQmYLsZYMeyRqzIWOMup03sw==", + "license": "ISC", + "dependencies": { + "cross-spawn": "^7.0.6", + "signal-exit": "^4.0.1" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/formdata-polyfill": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/formdata-polyfill/-/formdata-polyfill-4.0.10.tgz", + "integrity": "sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==", + "license": "MIT", + "dependencies": { + "fetch-blob": "^3.1.2" + }, + "engines": { + "node": ">=12.20.0" + } + }, + "node_modules/forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fresh": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-2.0.0.tgz", + "integrity": "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gaxios": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/gaxios/-/gaxios-7.1.3.tgz", + "integrity": "sha512-YGGyuEdVIjqxkxVH1pUTMY/XtmmsApXrCVv5EU25iX6inEPbV+VakJfLealkBtJN69AQmh1eGOdCl9Sm1UP6XQ==", + "license": "Apache-2.0", + "dependencies": { + "extend": "^3.0.2", + "https-proxy-agent": "^7.0.1", + "node-fetch": "^3.3.2", + "rimraf": "^5.0.1" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/gcp-metadata": { + "version": "8.1.2", + "resolved": "https://registry.npmjs.org/gcp-metadata/-/gcp-metadata-8.1.2.tgz", + "integrity": "sha512-zV/5HKTfCeKWnxG0Dmrw51hEWFGfcF2xiXqcA3+J90WDuP0SvoiSO5ORvcBsifmx/FoIjgQN3oNOGaQ5PhLFkg==", + "license": "Apache-2.0", + "dependencies": { + "gaxios": "^7.0.0", + "google-logging-utils": "^1.0.0", + "json-bigint": "^1.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dev": true, + "license": "ISC", + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-intrinsic": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", + "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", + "license": "MIT", + "dependencies": { + "call-bind-apply-helpers": "^1.0.2", + "es-define-property": "^1.0.1", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.1.1", + "function-bind": "^1.1.2", + "get-proto": "^1.0.1", + "gopd": "^1.2.0", + "has-symbols": "^1.1.0", + "hasown": "^2.0.2", + "math-intrinsics": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", + "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", + "license": "MIT", + "dependencies": { + "dunder-proto": "^1.0.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/get-tsconfig": { + "version": "4.13.0", + "resolved": "https://registry.npmjs.org/get-tsconfig/-/get-tsconfig-4.13.0.tgz", + "integrity": "sha512-1VKTZJCwBrvbd+Wn3AOgQP/2Av+TfTCOlE4AcRJE72W1ksZXbAx8PPBR9RzgTeSPzlPMHrbANMH3LbltH73wxQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "resolve-pkg-maps": "^1.0.0" + }, + "funding": { + "url": "https://github.com/privatenumber/get-tsconfig?sponsor=1" + } + }, + "node_modules/glob": { + "version": "10.5.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-10.5.0.tgz", + "integrity": "sha512-DfXN8DfhJ7NH3Oe7cFmu3NCu1wKbkReJ8TorzSAFbSKrlNaQSKfIzqYqVY8zlbs2NLBbWpRiU52GX2PbaBVNkg==", + "license": "ISC", + "dependencies": { + "foreground-child": "^3.1.0", + "jackspeak": "^3.1.2", + "minimatch": "^9.0.4", + "minipass": "^7.1.2", + "package-json-from-dist": "^1.0.0", + "path-scurry": "^1.11.1" + }, + "bin": { + "glob": "dist/esm/bin.mjs" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/google-auth-library": { + "version": "10.5.0", + "resolved": "https://registry.npmjs.org/google-auth-library/-/google-auth-library-10.5.0.tgz", + "integrity": "sha512-7ABviyMOlX5hIVD60YOfHw4/CxOfBhyduaYB+wbFWCWoni4N7SLcV46hrVRktuBbZjFC9ONyqamZITN7q3n32w==", + "license": "Apache-2.0", + "dependencies": { + "base64-js": "^1.3.0", + "ecdsa-sig-formatter": "^1.0.11", + "gaxios": "^7.0.0", + "gcp-metadata": "^8.0.0", + "google-logging-utils": "^1.0.0", + "gtoken": "^8.0.0", + "jws": "^4.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/google-logging-utils": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/google-logging-utils/-/google-logging-utils-1.1.3.tgz", + "integrity": "sha512-eAmLkjDjAFCVXg7A1unxHsLf961m6y17QFqXqAXGj/gVkKFrEICfStRfwUlGNfeCEjNRa32JEWOUTlYXPyyKvA==", + "license": "Apache-2.0", + "engines": { + "node": ">=14" + } + }, + "node_modules/googleapis": { + "version": "167.0.0", + "resolved": "https://registry.npmjs.org/googleapis/-/googleapis-167.0.0.tgz", + "integrity": "sha512-8Xqeki6K9u9jh6rGRA/OywRMXg8yXuv4ZLwSSuMBB3Ze1pErbR/iv00UmVtcrP2LcSW2Fqi+LUJ7WgFMDoxd7Q==", + "license": "Apache-2.0", + "dependencies": { + "google-auth-library": "^10.2.0", + "googleapis-common": "^8.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/googleapis-common": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/googleapis-common/-/googleapis-common-8.0.1.tgz", + "integrity": "sha512-eCzNACUXPb1PW5l0ULTzMHaL/ltPRADoPgjBlT8jWsTbxkCp6siv+qKJ/1ldaybCthGwsYFYallF7u9AkU4L+A==", + "license": "Apache-2.0", + "dependencies": { + "extend": "^3.0.2", + "gaxios": "^7.0.0-rc.4", + "google-auth-library": "^10.1.0", + "qs": "^6.7.0", + "url-template": "^2.0.8" + }, + "engines": { + "node": ">=18.0.0" + } + }, + "node_modules/gopd": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", + "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gtoken": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/gtoken/-/gtoken-8.0.0.tgz", + "integrity": "sha512-+CqsMbHPiSTdtSO14O51eMNlrp9N79gmeqmXeouJOhfucAedHw9noVe/n5uJk3tbKE6a+6ZCQg3RPhVhHByAIw==", + "license": "MIT", + "dependencies": { + "gaxios": "^7.0.0", + "jws": "^4.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/has-symbols": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", + "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "license": "MIT", + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/he": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", + "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", + "license": "MIT", + "bin": { + "he": "bin/he" + } + }, + "node_modules/html-to-text": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/html-to-text/-/html-to-text-9.0.5.tgz", + "integrity": "sha512-qY60FjREgVZL03vJU6IfMV4GDjGBIoOyvuFdpBDIX9yTlDw0TjxVBQp+P8NvpdIXNJvfWBTNul7fsAQJq2FNpg==", + "license": "MIT", + "dependencies": { + "@selderee/plugin-htmlparser2": "^0.11.0", + "deepmerge": "^4.3.1", + "dom-serializer": "^2.0.0", + "htmlparser2": "^8.0.2", + "selderee": "^0.11.0" + }, + "engines": { + "node": ">=14" + } + }, + "node_modules/htmlparser2": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.2.tgz", + "integrity": "sha512-GYdjWKDkbRLkZ5geuHs5NY1puJ+PXwP7+fHPRz06Eirsb9ugf6d8kkXav6ADhcODhFFPMIXyxkxSuMf3D6NCFA==", + "funding": [ + "https://github.com/fb55/htmlparser2?sponsor=1", + { + "type": "github", + "url": "https://github.com/sponsors/fb55" + } + ], + "license": "MIT", + "dependencies": { + "domelementtype": "^2.3.0", + "domhandler": "^5.0.3", + "domutils": "^3.0.1", + "entities": "^4.4.0" + } + }, + "node_modules/http-errors": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.1.tgz", + "integrity": "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ==", + "license": "MIT", + "dependencies": { + "depd": "~2.0.0", + "inherits": "~2.0.4", + "setprototypeof": "~1.2.0", + "statuses": "~2.0.2", + "toidentifier": "~1.0.1" + }, + "engines": { + "node": ">= 0.8" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/https-proxy-agent": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-7.0.6.tgz", + "integrity": "sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw==", + "license": "MIT", + "dependencies": { + "agent-base": "^7.1.2", + "debug": "4" + }, + "engines": { + "node": ">= 14" + } + }, + "node_modules/iconv-lite": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.7.0.tgz", + "integrity": "sha512-cf6L2Ds3h57VVmkZe+Pn+5APsT7FpqJtEhhieDCvrE2MK5Qk9MyffgQyuxQTm6BChfeZNtcOLHp9IcWRVcIcBQ==", + "license": "MIT", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "license": "ISC" + }, + "node_modules/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "license": "MIT", + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-promise": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz", + "integrity": "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ==", + "license": "MIT" + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "license": "ISC" + }, + "node_modules/jackspeak": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-3.4.3.tgz", + "integrity": "sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==", + "license": "BlueOak-1.0.0", + "dependencies": { + "@isaacs/cliui": "^8.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + }, + "optionalDependencies": { + "@pkgjs/parseargs": "^0.11.0" + } + }, + "node_modules/json-bigint": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/json-bigint/-/json-bigint-1.0.0.tgz", + "integrity": "sha512-SiPv/8VpZuWbvLSMtTDU8hEfrZWg/mH/nV/b4o0CYbSxu1UIQPLdwKOCIyLQX+VIPO5vrLX3i8qtqFyhdPSUSQ==", + "license": "MIT", + "dependencies": { + "bignumber.js": "^9.0.0" + } + }, + "node_modules/jsonwebtoken": { + "version": "9.0.3", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-9.0.3.tgz", + "integrity": "sha512-MT/xP0CrubFRNLNKvxJ2BYfy53Zkm++5bX9dtuPbqAeQpTVe0MQTFhao8+Cp//EmJp244xt6Drw/GVEGCUj40g==", + "license": "MIT", + "dependencies": { + "jws": "^4.0.1", + "lodash.includes": "^4.3.0", + "lodash.isboolean": "^3.0.3", + "lodash.isinteger": "^4.0.4", + "lodash.isnumber": "^3.0.3", + "lodash.isplainobject": "^4.0.6", + "lodash.isstring": "^4.0.1", + "lodash.once": "^4.0.0", + "ms": "^2.1.1", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=12", + "npm": ">=6" + } + }, + "node_modules/jwa": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-2.0.1.tgz", + "integrity": "sha512-hRF04fqJIP8Abbkq5NKGN0Bbr3JxlQ+qhZufXVr0DvujKy93ZCbXZMHDL4EOtodSbCWxOqR8MS1tXA5hwqCXDg==", + "license": "MIT", + "dependencies": { + "buffer-equal-constant-time": "^1.0.1", + "ecdsa-sig-formatter": "1.0.11", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/jws": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/jws/-/jws-4.0.1.tgz", + "integrity": "sha512-EKI/M/yqPncGUUh44xz0PxSidXFr/+r0pA70+gIYhjv+et7yxM+s29Y+VGDkovRofQem0fs7Uvf4+YmAdyRduA==", + "license": "MIT", + "dependencies": { + "jwa": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/leac": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/leac/-/leac-0.6.0.tgz", + "integrity": "sha512-y+SqErxb8h7nE/fiEX07jsbuhrpO9lL8eca7/Y1nuWV2moNlXhyd59iDGcRf6moVyDMbmTNzL40SUyrFU/yDpg==", + "license": "MIT", + "funding": { + "url": "https://ko-fi.com/killymxi" + } + }, + "node_modules/libbase64": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/libbase64/-/libbase64-1.3.0.tgz", + "integrity": "sha512-GgOXd0Eo6phYgh0DJtjQ2tO8dc0IVINtZJeARPeiIJqge+HdsWSuaDTe8ztQ7j/cONByDZ3zeB325AHiv5O0dg==", + "license": "MIT" + }, + "node_modules/libmime": { + "version": "5.3.7", + "resolved": "https://registry.npmjs.org/libmime/-/libmime-5.3.7.tgz", + "integrity": "sha512-FlDb3Wtha8P01kTL3P9M+ZDNDWPKPmKHWaU/cG/lg5pfuAwdflVpZE+wm9m7pKmC5ww6s+zTxBKS1p6yl3KpSw==", + "license": "MIT", + "dependencies": { + "encoding-japanese": "2.2.0", + "iconv-lite": "0.6.3", + "libbase64": "1.3.0", + "libqp": "2.1.1" + } + }, + "node_modules/libmime/node_modules/iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "license": "MIT", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/libqp": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/libqp/-/libqp-2.1.1.tgz", + "integrity": "sha512-0Wd+GPz1O134cP62YU2GTOPNA7Qgl09XwCqM5zpBv87ERCXdfDtyKXvV7c9U22yWJh44QZqBocFnXN11K96qow==", + "license": "MIT" + }, + "node_modules/linkify-it": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/linkify-it/-/linkify-it-5.0.0.tgz", + "integrity": "sha512-5aHCbzQRADcdP+ATqnDuhhJ/MRIqDkZX5pyjFHRRysS8vZ5AbqGEoFIb6pYHPZ+L/OC2Lc+xT8uHVVR5CAK/wQ==", + "license": "MIT", + "dependencies": { + "uc.micro": "^2.0.0" + } + }, + "node_modules/lodash.includes": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", + "integrity": "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w==", + "license": "MIT" + }, + "node_modules/lodash.isboolean": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", + "integrity": "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg==", + "license": "MIT" + }, + "node_modules/lodash.isinteger": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", + "integrity": "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA==", + "license": "MIT" + }, + "node_modules/lodash.isnumber": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", + "integrity": "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw==", + "license": "MIT" + }, + "node_modules/lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==", + "license": "MIT" + }, + "node_modules/lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw==", + "license": "MIT" + }, + "node_modules/lodash.once": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", + "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg==", + "license": "MIT" + }, + "node_modules/lru-cache": { + "version": "10.4.3", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.4.3.tgz", + "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==", + "license": "ISC" + }, + "node_modules/mailparser": { + "version": "3.9.1", + "resolved": "https://registry.npmjs.org/mailparser/-/mailparser-3.9.1.tgz", + "integrity": "sha512-6vHZcco3fWsDMkf4Vz9iAfxvwrKNGbHx0dV1RKVphQ/zaNY34Buc7D37LSa09jeSeybWzYcTPjhiZFxzVRJedA==", + "license": "MIT", + "dependencies": { + "@zone-eu/mailsplit": "5.4.8", + "encoding-japanese": "2.2.0", + "he": "1.2.0", + "html-to-text": "9.0.5", + "iconv-lite": "0.7.0", + "libmime": "5.3.7", + "linkify-it": "5.0.0", + "nodemailer": "7.0.11", + "punycode.js": "2.3.1", + "tlds": "1.261.0" + } + }, + "node_modules/math-intrinsics": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", + "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/media-typer": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-1.1.0.tgz", + "integrity": "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/merge-descriptors": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-2.0.0.tgz", + "integrity": "sha512-Snk314V5ayFLhp3fkUREub6WtjBfPdCPY1Ln8/8munuLuiYhsABgBVWsozAG+MWMbVEvcdcpbi9R7ww22l9Q3g==", + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/mime-db": { + "version": "1.54.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.54.0.tgz", + "integrity": "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-3.0.2.tgz", + "integrity": "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A==", + "license": "MIT", + "dependencies": { + "mime-db": "^1.54.0" + }, + "engines": { + "node": ">=18" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/minimatch": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", + "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/minipass": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", + "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==", + "license": "ISC", + "engines": { + "node": ">=16 || 14 >=14.17" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "license": "MIT" + }, + "node_modules/negotiator": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-1.0.0.tgz", + "integrity": "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/node-cron": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/node-cron/-/node-cron-4.2.1.tgz", + "integrity": "sha512-lgimEHPE/QDgFlywTd8yTR61ptugX3Qer29efeyWw2rv259HtGBNn1vZVmp8lB9uo9wC0t/AT4iGqXxia+CJFg==", + "license": "ISC", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/node-domexception": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz", + "integrity": "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==", + "deprecated": "Use your platform's native DOMException instead", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/jimmywarting" + }, + { + "type": "github", + "url": "https://paypal.me/jimmywarting" + } + ], + "license": "MIT", + "engines": { + "node": ">=10.5.0" + } + }, + "node_modules/node-fetch": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-3.3.2.tgz", + "integrity": "sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==", + "license": "MIT", + "dependencies": { + "data-uri-to-buffer": "^4.0.0", + "fetch-blob": "^3.1.4", + "formdata-polyfill": "^4.0.10" + }, + "engines": { + "node": "^12.20.0 || ^14.13.1 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/node-fetch" + } + }, + "node_modules/nodemailer": { + "version": "7.0.11", + "resolved": "https://registry.npmjs.org/nodemailer/-/nodemailer-7.0.11.tgz", + "integrity": "sha512-gnXhNRE0FNhD7wPSCGhdNh46Hs6nm+uTyg+Kq0cZukNQiYdnCsoQjodNP9BQVG9XrcK/v6/MgpAPBUFyzh9pvw==", + "license": "MIT-0", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/object-inspect": { + "version": "1.13.4", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", + "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/on-finished": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", + "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", + "license": "MIT", + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "license": "ISC", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/package-json-from-dist": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz", + "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==", + "license": "BlueOak-1.0.0" + }, + "node_modules/parseley": { + "version": "0.12.1", + "resolved": "https://registry.npmjs.org/parseley/-/parseley-0.12.1.tgz", + "integrity": "sha512-e6qHKe3a9HWr0oMRVDTRhKce+bRO8VGQR3NyVwcjwrbhMmFCX9KszEV35+rn4AdilFAq9VPxP/Fe1wC9Qjd2lw==", + "license": "MIT", + "dependencies": { + "leac": "^0.6.0", + "peberminta": "^0.9.0" + }, + "funding": { + "url": "https://ko-fi.com/killymxi" + } + }, + "node_modules/parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-scurry": { + "version": "1.11.1", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.11.1.tgz", + "integrity": "sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==", + "license": "BlueOak-1.0.0", + "dependencies": { + "lru-cache": "^10.2.0", + "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0" + }, + "engines": { + "node": ">=16 || 14 >=14.18" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/path-to-regexp": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-8.3.0.tgz", + "integrity": "sha512-7jdwVIRtsP8MYpdXSwOS0YdD0Du+qOoF/AEPIt88PcCFrZCzx41oxku1jD88hZBwbNUIEfpqvuhjFaMAqMTWnA==", + "license": "MIT", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, + "node_modules/peberminta": { + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/peberminta/-/peberminta-0.9.0.tgz", + "integrity": "sha512-XIxfHpEuSJbITd1H3EeQwpcZbTLHc+VVr8ANI9t5sit565tsI4/xK3KWTUFE2e6QiangUkh3B0jihzmGnNrRsQ==", + "license": "MIT", + "funding": { + "url": "https://ko-fi.com/killymxi" + } + }, + "node_modules/pg": { + "version": "8.16.3", + "resolved": "https://registry.npmjs.org/pg/-/pg-8.16.3.tgz", + "integrity": "sha512-enxc1h0jA/aq5oSDMvqyW3q89ra6XIIDZgCX9vkMrnz5DFTw/Ny3Li2lFQ+pt3L6MCgm/5o2o8HW9hiJji+xvw==", + "license": "MIT", + "dependencies": { + "pg-connection-string": "^2.9.1", + "pg-pool": "^3.10.1", + "pg-protocol": "^1.10.3", + "pg-types": "2.2.0", + "pgpass": "1.0.5" + }, + "engines": { + "node": ">= 16.0.0" + }, + "optionalDependencies": { + "pg-cloudflare": "^1.2.7" + }, + "peerDependencies": { + "pg-native": ">=3.0.1" + }, + "peerDependenciesMeta": { + "pg-native": { + "optional": true + } + } + }, + "node_modules/pg-cloudflare": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/pg-cloudflare/-/pg-cloudflare-1.2.7.tgz", + "integrity": "sha512-YgCtzMH0ptvZJslLM1ffsY4EuGaU0cx4XSdXLRFae8bPP4dS5xL1tNB3k2o/N64cHJpwU7dxKli/nZ2lUa5fLg==", + "license": "MIT", + "optional": true + }, + "node_modules/pg-connection-string": { + "version": "2.9.1", + "resolved": "https://registry.npmjs.org/pg-connection-string/-/pg-connection-string-2.9.1.tgz", + "integrity": "sha512-nkc6NpDcvPVpZXxrreI/FOtX3XemeLl8E0qFr6F2Lrm/I8WOnaWNhIPK2Z7OHpw7gh5XJThi6j6ppgNoaT1w4w==", + "license": "MIT" + }, + "node_modules/pg-int8": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/pg-int8/-/pg-int8-1.0.1.tgz", + "integrity": "sha512-WCtabS6t3c8SkpDBUlb1kjOs7l66xsGdKpIPZsg4wR+B3+u9UAum2odSsF9tnvxg80h4ZxLWMy4pRjOsFIqQpw==", + "license": "ISC", + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/pg-pool": { + "version": "3.10.1", + "resolved": "https://registry.npmjs.org/pg-pool/-/pg-pool-3.10.1.tgz", + "integrity": "sha512-Tu8jMlcX+9d8+QVzKIvM/uJtp07PKr82IUOYEphaWcoBhIYkoHpLXN3qO59nAI11ripznDsEzEv8nUxBVWajGg==", + "license": "MIT", + "peerDependencies": { + "pg": ">=8.0" + } + }, + "node_modules/pg-protocol": { + "version": "1.10.3", + "resolved": "https://registry.npmjs.org/pg-protocol/-/pg-protocol-1.10.3.tgz", + "integrity": "sha512-6DIBgBQaTKDJyxnXaLiLR8wBpQQcGWuAESkRBX/t6OwA8YsqP+iVSiond2EDy6Y/dsGk8rh/jtax3js5NeV7JQ==", + "license": "MIT" + }, + "node_modules/pg-types": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/pg-types/-/pg-types-2.2.0.tgz", + "integrity": "sha512-qTAAlrEsl8s4OiEQY69wDvcMIdQN6wdz5ojQiOy6YRMuynxenON0O5oCpJI6lshc6scgAY8qvJ2On/p+CXY0GA==", + "license": "MIT", + "dependencies": { + "pg-int8": "1.0.1", + "postgres-array": "~2.0.0", + "postgres-bytea": "~1.0.0", + "postgres-date": "~1.0.4", + "postgres-interval": "^1.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/pgpass": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/pgpass/-/pgpass-1.0.5.tgz", + "integrity": "sha512-FdW9r/jQZhSeohs1Z3sI1yxFQNFvMcnmfuj4WBMUTxOrAyLMaTcE1aAMBiTlbMNaXvBCQuVi0R7hd8udDSP7ug==", + "license": "MIT", + "dependencies": { + "split2": "^4.1.0" + } + }, + "node_modules/postgres-array": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/postgres-array/-/postgres-array-2.0.0.tgz", + "integrity": "sha512-VpZrUqU5A69eQyW2c5CA1jtLecCsN2U/bD6VilrFDWq5+5UIEVO7nazS3TEcHf1zuPYO/sqGvUvW62g86RXZuA==", + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/postgres-bytea": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/postgres-bytea/-/postgres-bytea-1.0.0.tgz", + "integrity": "sha512-xy3pmLuQqRBZBXDULy7KbaitYqLcmxigw14Q5sj8QBVLqEwXfeybIKVWiqAXTlcvdvb0+xkOtDbfQMOf4lST1w==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/postgres-date": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/postgres-date/-/postgres-date-1.0.7.tgz", + "integrity": "sha512-suDmjLVQg78nMK2UZ454hAG+OAW+HQPZ6n++TNDUX+L0+uUlLywnoxJKDou51Zm+zTCjrCl0Nq6J9C5hP9vK/Q==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/postgres-interval": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/postgres-interval/-/postgres-interval-1.2.0.tgz", + "integrity": "sha512-9ZhXKM/rw350N1ovuWHbGxnGh/SNJ4cnxHiM0rxE4VN41wsg8P8zWn9hv/buK00RP4WvlOyr/RBDiptyxVbkZQ==", + "license": "MIT", + "dependencies": { + "xtend": "^4.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/prettier": { + "version": "3.7.4", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.7.4.tgz", + "integrity": "sha512-v6UNi1+3hSlVvv8fSaoUbggEM5VErKmmpGA7Pl3HF8V6uKY7rvClBOJlH6yNwQtfTueNkGVpOv/mtWL9L4bgRA==", + "dev": true, + "license": "MIT", + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" + } + }, + "node_modules/prettier-plugin-organize-imports": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/prettier-plugin-organize-imports/-/prettier-plugin-organize-imports-4.3.0.tgz", + "integrity": "sha512-FxFz0qFhyBsGdIsb697f/EkvHzi5SZOhWAjxcx2dLt+Q532bAlhswcXGYB1yzjZ69kW8UoadFBw7TyNwlq96Iw==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "prettier": ">=2.0", + "typescript": ">=2.9", + "vue-tsc": "^2.1.0 || 3" + }, + "peerDependenciesMeta": { + "vue-tsc": { + "optional": true + } + } + }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "license": "MIT", + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/punycode.js": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode.js/-/punycode.js-2.3.1.tgz", + "integrity": "sha512-uxFIHU0YlHYhDQtV4R9J6a52SLx28BCjT+4ieh7IGbgwVJWO+km431c4yRlREUAsAmt/uMjQUyQHNEPf0M39CA==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/qs": { + "version": "6.14.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.14.0.tgz", + "integrity": "sha512-YWWTjgABSKcvs/nWBi9PycY/JiPJqOD4JA6o9Sej2AtvSGarXxKC3OQSk4pAarbdQlKAh5D4FCQkJNkW+GAn3w==", + "license": "BSD-3-Clause", + "dependencies": { + "side-channel": "^1.1.0" + }, + "engines": { + "node": ">=0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/raw-body": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-3.0.2.tgz", + "integrity": "sha512-K5zQjDllxWkf7Z5xJdV0/B0WTNqx6vxG70zJE4N0kBs4LovmEYWJzQGxC9bS9RAKu3bgM40lrd5zoLJ12MQ5BA==", + "license": "MIT", + "dependencies": { + "bytes": "~3.1.2", + "http-errors": "~2.0.1", + "iconv-lite": "~0.7.0", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve-pkg-maps": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz", + "integrity": "sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://github.com/privatenumber/resolve-pkg-maps?sponsor=1" + } + }, + "node_modules/rimraf": { + "version": "5.0.10", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-5.0.10.tgz", + "integrity": "sha512-l0OE8wL34P4nJH/H2ffoaniAokM2qSmrtXHmlpvYr5AVVX8msAyW0l8NVJFDxlSK4u3Uh/f41cQheDVdnYijwQ==", + "license": "ISC", + "dependencies": { + "glob": "^10.3.7" + }, + "bin": { + "rimraf": "dist/esm/bin.mjs" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/router": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/router/-/router-2.2.0.tgz", + "integrity": "sha512-nLTrUKm2UyiL7rlhapu/Zl45FwNgkZGaCpZbIHajDYgwlJCOzLSk+cIPAnsEqV955GjILJnKbdQC1nVPz+gAYQ==", + "license": "MIT", + "dependencies": { + "debug": "^4.4.0", + "depd": "^2.0.0", + "is-promise": "^4.0.0", + "parseurl": "^1.3.3", + "path-to-regexp": "^8.0.0" + }, + "engines": { + "node": ">= 18" + } + }, + "node_modules/rxjs": { + "version": "7.8.2", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.8.2.tgz", + "integrity": "sha512-dhKf903U/PQZY6boNNtAGdWbG85WAbjT/1xYoZIC7FAY0yWapOBQVsVrDl58W86//e1VpMNBtRV4MaXfdMySFA==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "tslib": "^2.1.0" + } + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "license": "MIT" + }, + "node_modules/selderee": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/selderee/-/selderee-0.11.0.tgz", + "integrity": "sha512-5TF+l7p4+OsnP8BCCvSyZiSPc4x4//p5uPwK8TCnVPJYRmU2aYKMpOXvw8zM5a5JvuuCGN1jmsMwuU2W02ukfA==", + "license": "MIT", + "dependencies": { + "parseley": "^0.12.0" + }, + "funding": { + "url": "https://ko-fi.com/killymxi" + } + }, + "node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/send": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/send/-/send-1.2.0.tgz", + "integrity": "sha512-uaW0WwXKpL9blXE2o0bRhoL2EGXIrZxQ2ZQ4mgcfoBxdFmQold+qWsD2jLrfZ0trjKL6vOw0j//eAwcALFjKSw==", + "license": "MIT", + "dependencies": { + "debug": "^4.3.5", + "encodeurl": "^2.0.0", + "escape-html": "^1.0.3", + "etag": "^1.8.1", + "fresh": "^2.0.0", + "http-errors": "^2.0.0", + "mime-types": "^3.0.1", + "ms": "^2.1.3", + "on-finished": "^2.4.1", + "range-parser": "^1.2.1", + "statuses": "^2.0.1" + }, + "engines": { + "node": ">= 18" + } + }, + "node_modules/serve-static": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-2.2.0.tgz", + "integrity": "sha512-61g9pCh0Vnh7IutZjtLGGpTA355+OPn2TyDv/6ivP2h/AdAVX9azsoxmg2/M6nZeQZNYBEwIcsne1mJd9oQItQ==", + "license": "MIT", + "dependencies": { + "encodeurl": "^2.0.0", + "escape-html": "^1.0.3", + "parseurl": "^1.3.3", + "send": "^1.2.0" + }, + "engines": { + "node": ">= 18" + } + }, + "node_modules/setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", + "license": "ISC" + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "license": "MIT", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/shell-quote": { + "version": "1.8.3", + "resolved": "https://registry.npmjs.org/shell-quote/-/shell-quote-1.8.3.tgz", + "integrity": "sha512-ObmnIF4hXNg1BqhnHmgbDETF8dLPCggZWBjkQfhZpbszZnYur5DUljTcCHii5LC3J5E0yeO/1LIMyH+UvHQgyw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", + "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3", + "side-channel-list": "^1.0.0", + "side-channel-map": "^1.0.1", + "side-channel-weakmap": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-list": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", + "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", + "license": "MIT", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-map": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", + "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-weakmap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", + "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", + "license": "MIT", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3", + "side-channel-map": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/signal-exit": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", + "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", + "license": "ISC", + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/split2": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/split2/-/split2-4.2.0.tgz", + "integrity": "sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==", + "license": "ISC", + "engines": { + "node": ">= 10.x" + } + }, + "node_modules/statuses": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.2.tgz", + "integrity": "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/string-width": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", + "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", + "license": "MIT", + "dependencies": { + "eastasianwidth": "^0.2.0", + "emoji-regex": "^9.2.2", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/string-width-cjs": { + "name": "string-width", + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width-cjs/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "license": "MIT" + }, + "node_modules/string-width-cjs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.2.tgz", + "integrity": "sha512-gmBGslpoQJtgnMAvOVqGZpEz9dyoKTCzy2nfz/n8aIFhN/jCE/rCmcxabB6jOOHV+0WNnylOxaxBQPSvcWklhA==", + "license": "MIT", + "dependencies": { + "ansi-regex": "^6.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/strip-ansi?sponsor=1" + } + }, + "node_modules/strip-ansi-cjs": { + "name": "strip-ansi", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-ansi-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/tlds": { + "version": "1.261.0", + "resolved": "https://registry.npmjs.org/tlds/-/tlds-1.261.0.tgz", + "integrity": "sha512-QXqwfEl9ddlGBaRFXIvNKK6OhipSiLXuRuLJX5DErz0o0Q0rYxulWLdFryTkV5PkdZct5iMInwYEGe/eR++1AA==", + "license": "MIT", + "bin": { + "tlds": "bin.js" + } + }, + "node_modules/toidentifier": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", + "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", + "license": "MIT", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/tree-kill": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", + "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", + "dev": true, + "license": "MIT", + "bin": { + "tree-kill": "cli.js" + } + }, + "node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "dev": true, + "license": "0BSD" + }, + "node_modules/tsx": { + "version": "4.21.0", + "resolved": "https://registry.npmjs.org/tsx/-/tsx-4.21.0.tgz", + "integrity": "sha512-5C1sg4USs1lfG0GFb2RLXsdpXqBSEhAaA/0kPL01wxzpMqLILNxIxIOKiILz+cdg/pLnOUxFYOR5yhHU666wbw==", + "dev": true, + "license": "MIT", + "dependencies": { + "esbuild": "~0.27.0", + "get-tsconfig": "^4.7.5" + }, + "bin": { + "tsx": "dist/cli.mjs" + }, + "engines": { + "node": ">=18.0.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.3" + } + }, + "node_modules/type-is": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-2.0.1.tgz", + "integrity": "sha512-OZs6gsjF4vMp32qrCbiVSkrFmXtG/AZhY3t0iAMrMBiAZyV9oALtXO8hsrHbMXF9x6L3grlFuwW2oAz7cav+Gw==", + "license": "MIT", + "dependencies": { + "content-type": "^1.0.5", + "media-typer": "^1.1.0", + "mime-types": "^3.0.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/typescript": { + "version": "5.9.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", + "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/uc.micro": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/uc.micro/-/uc.micro-2.1.0.tgz", + "integrity": "sha512-ARDJmphmdvUk6Glw7y9DQ2bFkKBHwQHLi2lsaH6PPmz/Ka9sFOBsBluozhDltWmnv9u/cF6Rt87znRTPV+yp/A==", + "license": "MIT" + }, + "node_modules/undici-types": { + "version": "7.16.0", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz", + "integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==", + "dev": true, + "license": "MIT" + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/url-template": { + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/url-template/-/url-template-2.0.8.tgz", + "integrity": "sha512-XdVKMF4SJ0nP/O7XIPB0JwAEuT9lDIYnNsK8yGVe43y0AWoKeJNdv3ZNWh7ksJ6KqQFjOO6ox/VEitLnaVNufw==", + "license": "BSD" + }, + "node_modules/vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/web-streams-polyfill": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.3.3.tgz", + "integrity": "sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==", + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/wrap-ansi": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz", + "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==", + "license": "MIT", + "dependencies": { + "ansi-styles": "^6.1.0", + "string-width": "^5.0.1", + "strip-ansi": "^7.0.1" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs": { + "name": "wrap-ansi", + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "license": "MIT" + }, + "node_modules/wrap-ansi-cjs/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi-cjs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "license": "ISC" + }, + "node_modules/xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "license": "MIT", + "engines": { + "node": ">=0.4" + } + }, + "node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=10" + } + }, + "node_modules/yargs": { + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "cliui": "^8.0.1", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.3", + "y18n": "^5.0.5", + "yargs-parser": "^21.1.1" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs-parser": { + "version": "21.1.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", + "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/yargs/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true, + "license": "MIT" + }, + "node_modules/yargs/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/yargs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } } } } diff --git a/gmail/package.json b/gmail/package.json index abe3215ab..849828cb1 100644 --- a/gmail/package.json +++ b/gmail/package.json @@ -1,11 +1,38 @@ { - "devDependencies": { - "@rollup/plugin-node-resolve": "^15.0.2", - "@rollup/plugin-typescript": "^11.1.1", - "@types/google-apps-script": "^1.0.64", - "prettier": "^2.2.1", - "rollup": "^3.22.0", - "tslib": "^2.5.3" + "name": "gmail_http", + "version": "1.0.0", + "main": "index.js", + "scripts": { + "dev": "npm run gen-secret && concurrently \"tsx watch src/index.ts\" \"npm run typecheck\"", + "typecheck": "tsc --noEmit --watch", + "build": "rm -rf dist && tsc && npm run gen-secret && mv .env dist/.env", + "start": "cd dist && node index.js", + "prettier": "prettier --write 'src/**/*.ts'", + "gen-secret": "echo APP_SECRET=$(node -e \"console.log(require('crypto').randomBytes(32).toString('hex'))\") > .env" + }, + "author": "", + "license": "", + "description": "", + "dependencies": { + "@resvg/resvg-js": "^2.6.2", + "dotenv": "^17.2.3", + "express": "^5.2.1", + "express-async-handler": "^1.2.0", + "google-auth-library": "^10.5.0", + "googleapis": "^167.0.0", + "jsonwebtoken": "^9.0.3", + "mailparser": "^3.9.0", + "node-cron": "^4.2.1", + "pg": "^8.16.3" }, - "type": "module" + "devDependencies": { + "@types/express": "^5.0.6", + "@types/node": "^24.10.1", + "@types/pg": "^8.15.6", + "concurrently": "^9.2.1", + "prettier": "^3.7.4", + "prettier-plugin-organize-imports": "^4.3.0", + "tsx": "^4.21.0", + "typescript": "^5.9.3" + } } diff --git a/gmail/rollup.config.js b/gmail/rollup.config.js deleted file mode 100644 index c57394761..000000000 --- a/gmail/rollup.config.js +++ /dev/null @@ -1,37 +0,0 @@ -import typescript from "@rollup/plugin-typescript"; -import { nodeResolve } from "@rollup/plugin-node-resolve"; - -const extensions = [".ts"]; - -/** - * Prevent tree-shaking the entry-point - * by not shaking any module that isn't imported by anyone. - * @returns side-effects or nothing - */ -function preventEntrypointShakingPlugin() { - return { - name: "no-treeshaking", - resolveId(id, importer) { - if (!importer) { - return { id, moduleSideEffects: "no-treeshake" }; - } - return null; - }, - }; -} - -export default { - input: "./src/main.ts", - output: { - dir: "./build", - format: "esm", - sourcemap: true, - }, - plugins: [ - preventEntrypointShakingPlugin(), - nodeResolve({ - extensions, - }), - typescript(), - ], -}; diff --git a/gmail/src/assets/Caveat.ttf b/gmail/src/assets/Caveat.ttf new file mode 100644 index 000000000..8c97eaf4f Binary files /dev/null and b/gmail/src/assets/Caveat.ttf differ diff --git a/gmail/src/assets/GoogleSans.ttf b/gmail/src/assets/GoogleSans.ttf new file mode 100644 index 000000000..ab605f9e2 Binary files /dev/null and b/gmail/src/assets/GoogleSans.ttf differ diff --git a/gmail/src/assets/button.svg b/gmail/src/assets/button.svg new file mode 100644 index 000000000..be200a407 --- /dev/null +++ b/gmail/src/assets/button.svg @@ -0,0 +1,6 @@ + + + + +__TEXT__ + diff --git a/gmail/src/assets/close.png b/gmail/src/assets/close.png new file mode 100644 index 000000000..e259397e0 Binary files /dev/null and b/gmail/src/assets/close.png differ diff --git a/gmail/src/assets/email_in_odoo.png b/gmail/src/assets/email_in_odoo.png new file mode 100644 index 000000000..2dbc2edbf Binary files /dev/null and b/gmail/src/assets/email_in_odoo.png differ diff --git a/gmail/src/assets/email_logged.png b/gmail/src/assets/email_logged.png new file mode 100644 index 000000000..00306e46f Binary files /dev/null and b/gmail/src/assets/email_logged.png differ diff --git a/gmail/src/assets/empty_folder.svg b/gmail/src/assets/empty_folder.svg new file mode 100644 index 000000000..70e6eeac1 --- /dev/null +++ b/gmail/src/assets/empty_folder.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + diff --git a/gmail/src/assets/link.svg b/gmail/src/assets/link.svg new file mode 100644 index 000000000..f812e4839 --- /dev/null +++ b/gmail/src/assets/link.svg @@ -0,0 +1 @@ + diff --git a/gmail/src/assets/login_header.svg b/gmail/src/assets/login_header.svg new file mode 100644 index 000000000..cc4a4101f --- /dev/null +++ b/gmail/src/assets/login_header.svg @@ -0,0 +1,66 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + The power of Odooin your inbox + + + + + + + + + + + + + for Gmail + + diff --git a/gmail/src/assets/odoo.png b/gmail/src/assets/odoo.png new file mode 100644 index 000000000..1fa68c204 Binary files /dev/null and b/gmail/src/assets/odoo.png differ diff --git a/gmail/src/assets/person.png b/gmail/src/assets/person.png new file mode 100644 index 000000000..6513bb2d4 Binary files /dev/null and b/gmail/src/assets/person.png differ diff --git a/gmail/src/assets/reload.png b/gmail/src/assets/reload.png new file mode 100644 index 000000000..8398346a2 Binary files /dev/null and b/gmail/src/assets/reload.png differ diff --git a/gmail/src/assets/search.png b/gmail/src/assets/search.png new file mode 100644 index 000000000..c737a5a4e Binary files /dev/null and b/gmail/src/assets/search.png differ diff --git a/gmail/src/assets/search_no_result.svg b/gmail/src/assets/search_no_result.svg new file mode 100644 index 000000000..4f1f08855 --- /dev/null +++ b/gmail/src/assets/search_no_result.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + + No record found. + Try using different keywords. + diff --git a/gmail/src/const.ts b/gmail/src/consts.ts similarity index 56% rename from gmail/src/const.ts rename to gmail/src/consts.ts index a59c7976e..dc756ef98 100644 --- a/gmail/src/const.ts +++ b/gmail/src/consts.ts @@ -1,28 +1,36 @@ +export const HOST = "https://mister7f.xyz"; +export const CLIENT_ID = "36859136832-1fiif7tqkl57sck2e80349oetgbhrb3a.apps.googleusercontent.com"; + +// PSQL config +export const PSQL_USER = "root"; +export const PSQL_PASS = "root"; +export const PSQL_DB = "gmail_addin_db"; +export const PSQL_HOST = "localhost"; +export const PSQL_PORT = 5432; + export const URLS: Record = { GET_TRANSLATIONS: "/mail_plugin/get_translations", LOG_EMAIL: "/mail_plugin/log_mail_content", + SEARCH_RECORDS: "/mail_plugin/search_records", // Partner GET_PARTNER: "/mail_plugin/partner/get", - SEARCH_PARTNER: "/mail_plugin/partner/search", + SEARCH_PARTNER: "/mail_plugin/search_records/res.partner", PARTNER_CREATE: "/mail_plugin/partner/create", - CREATE_COMPANY: "/mail_plugin/partner/enrich_and_create_company", - ENRICH_COMPANY: "/mail_plugin/partner/enrich_and_update_company", // CRM Lead CREATE_LEAD: "/mail_plugin/lead/create", // HELPDESK Ticket CREATE_TICKET: "/mail_plugin/ticket/create", // Project - SEARCH_PROJECT: "/mail_plugin/project/search", + SEARCH_PROJECT: "/mail_plugin/search_records/project.project", CREATE_PROJECT: "/mail_plugin/project/create", CREATE_TASK: "/mail_plugin/task/create", - // IAP - IAP_COMPANY_ENRICHMENT: "https://iap-services.odoo.com/iap/mail_extension/enrich", }; export const ODOO_AUTH_URLS: Record = { LOGIN: "/web/login", AUTH_CODE: "/mail_plugin/auth", CODE_VALIDATION: "/mail_plugin/auth/access_token", + CHECK_VERSION: "/mail_plugin/auth/check_version", SCOPE: "outlook", FRIENDLY_NAME: "Gmail", }; diff --git a/gmail/src/global.d.ts b/gmail/src/global.d.ts deleted file mode 100644 index 7be5c3cae..000000000 --- a/gmail/src/global.d.ts +++ /dev/null @@ -1,6 +0,0 @@ -declare let global: any; -type Card = any; -type ActionEvent = any; -type CardSection = any; -type Button = any; -type GmailAttachment = any; diff --git a/gmail/src/index.ts b/gmail/src/index.ts new file mode 100644 index 000000000..8869516ed --- /dev/null +++ b/gmail/src/index.ts @@ -0,0 +1,253 @@ +import express from "express"; +import asyncHandler from "express-async-handler"; +import fs from "fs/promises"; +import { google } from "googleapis"; +import jwt from "jsonwebtoken"; +import cron from "node-cron"; +import path from "path"; +import { Email } from "./models/email"; +import { Partner } from "./models/partner"; +import { State } from "./models/state"; +import { User } from "./models/user"; +import { odooAuthCallback } from "./services/odoo_auth"; +import { Translate } from "./services/translation"; +import { getEventHandler } from "./utils/actions"; +import pool from "./utils/db"; +import { htmlEscape } from "./utils/format"; +import { svgToPngResponse } from "./utils/svg"; +import { getLoginMainView } from "./views/login"; +import { getPartnerView } from "./views/partner"; +import { getSearchPartnerView } from "./views/search_partner"; + +const gmail = google.gmail({ version: "v1" }); + +const app = express(); +app.use(express.json()); + +// Load the application secret from `.env` +require("dotenv").config({ quiet: true }); +if (!process.env.APP_SECRET?.length) { + throw new Error("Application secret not configured"); +} + +/** + * Once a day, clean the old email log table. + */ +cron.schedule("0 0 * * *", async () => { + console.log("Clean the email logging table..."); + await pool.query( + ` + DELETE FROM email_logs + WHERE create_date < NOW() - INTERVAL '1 month' + `, + ); +}); + +/** + * Endpoint called the first time the user open an email, or when reloading. + */ +app.post( + "/on_open_email", + asyncHandler(async (req, res) => { + const [user, headers] = await Promise.all([ + User.getUserFromGoogleToken(req.body), + Email.getEmailHeadersFromGoogleToken(req.body), + ]); + + if (!user.odooUrl?.length || !user.odooToken?.length) { + res.json((await getLoginMainView(user)).build()); + return; + } + + const email = await Email.getEmailFromHeaders(req.body, headers, user); + + if (email.contacts.length > 1) { + // More than one contact, we will need to choose the right one + const [_t, [searchedPartners, error]] = await Promise.all([ + Translate.getTranslations(user), + Partner.searchPartner( + user, + email.contacts.map((c) => c.email), + ), + ]); + + if (error.code) { + res.json((await getLoginMainView(user)).build()); + return; + } + const existingPartnersEmails = searchedPartners.map((p) => p.email); + for (const contact of email.contacts) { + if (existingPartnersEmails.includes(contact.email)) { + continue; + } + searchedPartners.push( + Partner.fromJson({ name: contact.name, email: contact.email }), + ); + } + + const state = new State(null, false, email, searchedPartners, null, false); + const searchPartnerView = await getSearchPartnerView( + state, + _t, + user, + "", + false, + _t("In this conversation"), + true, + true, + ); + res.json(searchPartnerView.build()); + return; + } + + // Only one partner, we can open the view immediately + const [_t, [partner, canCreatePartner, canCreateProject, error]] = await Promise.all([ + Translate.getTranslations(user), + Partner.getPartner(user, email.contacts[0].name, email.contacts[0].email), + ]); + + if (error.code) { + res.json((await getLoginMainView(user)).build()); + return; + } + + const state = new State(partner, canCreatePartner, email, null, null, canCreateProject); + + res.json(getPartnerView(state, _t, user).build()); + }), +); + +/** + * Callback called by the addin when it executes an action. + */ +app.post( + "/execute_action", + asyncHandler(async (req, res) => { + const user = await User.getUserFromGoogleToken(req.body); + + const _t = await Translate.getTranslations(user); + + const rawFormInputs = req.body.commonEventObject.formInputs || {}; + const formInputs = Object.fromEntries( + Object.entries(rawFormInputs).map(([key, value]) => [ + key, + value["stringInputs"]["value"][0], + ]), + ); + const parameters = req.body.commonEventObject.parameters; + const decoded = jwt.verify(parameters.token, process.env.APP_SECRET, { + algorithms: ["HS256"], + }); + + const functionName = decoded.functionName; + const state = decoded.state && State.fromJson(decoded.state); + const args = decoded.arguments || {}; + + if (state?.email) { + // Update the Gmail tokens + state.email.userOAuthToken = req.body.authorizationEventObject.userOAuthToken; + state.email.accessToken = req.body.gmail.accessToken; + } + + const result = await getEventHandler(functionName)(state, _t, user, args, formInputs); + res.json(result.build()); + }), +); + +app.get( + "/auth_callback", + asyncHandler(async (req, res) => { + res.send(await odooAuthCallback(req)); + }), +); + +/** + * Serve the SVG files as PNG, because Google won't fetch SVG. + */ +app.use("/assets", async (req, res, next) => { + if (!req.path.endsWith(".svg.png")) { + return next(); + } + + const filename = req.path.slice(1, -4); + + // Prevent directory traversal + if (!/^[a-z0-9_-]+\.svg$/.test(filename)) { + res.sendStatus(404); + return; + } + const base = path.join(__dirname, "assets"); + const fullPath = path.resolve(path.join(base, filename)); + if (!fullPath.startsWith(`${base}/`)) { + res.sendStatus(404); + return; + } + + try { + svgToPngResponse(await fs.readFile(fullPath), res); + } catch (err) { + res.sendStatus(404); + } +}); + +/** + * For some views, we want button that takes the full width of the card, + * this is not possible with standard button widget, and so we use SVG + * file with a link. + */ +app.use("/render_button/:backgroundColor/:textColor/:label", async (req, res, next) => { + const { backgroundColor, textColor, label } = req.params; + if (!/^[0-9a-z]$/.test(backgroundColor) || !/^[0-9a-z]$/.test(textColor)) { + res.sendStatus(404); + return; + } + + const svg = await fs.readFile(path.join(__dirname, "assets/button.svg")); + const svgText = svg + .toString() + .replace("__TEXT__", htmlEscape(label)) + .replace("__STROKE__", `#${backgroundColor}`) + .replace("__FILL__", `#${backgroundColor}`) + .replace("__COLOR__", `#${textColor}`); + try { + svgToPngResponse(Buffer.from(svgText), res); + } catch (err) { + res.sendStatus(404); + } +}); + +/** + * For some views, we want button that takes the full width of the card, + * this is not possible with standard button widget, and so we use SVG + * file with a link. + */ +app.use("/render_search_no_result/:title/:subtitle", async (req, res, next) => { + const { title, subtitle } = req.params; + + const svg = await fs.readFile(path.join(__dirname, "assets/search_no_result.svg")); + const svgText = svg + .toString() + .replace("No record found.", htmlEscape(title)) + .replace("Try using different keywords.", htmlEscape(subtitle)); + try { + svgToPngResponse(Buffer.from(svgText), res); + } catch (err) { + res.sendStatus(404); + } +}); + +app.use( + "/assets", + express.static(path.join(__dirname, "assets"), { + fallthrough: false, + immutable: true, + maxAge: "1y", + }), +); + +const server = app.listen(5000, () => { + const address = server.address(); + if (typeof address === "object" && address?.port) { + console.log("Running on port", address.port); + } +}); diff --git a/gmail/src/main.ts b/gmail/src/main.ts deleted file mode 100644 index 028699406..000000000 --- a/gmail/src/main.ts +++ /dev/null @@ -1,42 +0,0 @@ -import { buildView } from "./views/index"; -import { Email } from "./models/email"; -import { State } from "./models/state"; -import { Partner } from "./models/partner"; -import { _t } from "./services/translation"; - -/** - * Entry point of the application, executed when an email is open. - * - * If the user is not connected to a Odoo database, we will contact IAP and enrich the - * domain of the op penned email. - * - * If the user is connected to a Odoo database, we will fetch the corresponding partner - * and other information like his leads, tickets, company... - */ -function onGmailMessageOpen(event) { - GmailApp.setCurrentMessageAccessToken(event.messageMetadata.accessToken); - const currentEmail = new Email(event.gmail.messageId, event.gmail.accessToken); - - const [partner, odooUserCompanies, canCreatePartner, canCreateProject, error] = Partner.enrichPartner( - currentEmail.contactEmail, - currentEmail.contactName - ); - - if (!partner) { - // Should at least use the FROM headers to generate the partner - throw new Error(_t("Error during enrichment")); - } - - const state = new State( - partner, - canCreatePartner, - currentEmail, - odooUserCompanies, - null, - null, - canCreateProject, - error - ); - - return [buildView(state)]; -} diff --git a/gmail/src/models/company.ts b/gmail/src/models/company.ts deleted file mode 100644 index ab0483ccd..000000000 --- a/gmail/src/models/company.ts +++ /dev/null @@ -1,152 +0,0 @@ -import { formatUrl, isTrue, first } from "../utils/format"; - -export class Company { - id: number; - name: string; - email: string; - phone: string; - isEnriched: boolean; - - // Additional Information - address: string; - annualRevenue: string; - companyType: string; - description: string; - emails: string; - employees: number; - foundedYear: number; - image: string; - industry: string; - mobile: string; - phones: string; - tags: string; - timezone: string; - timezoneUrl: string; - twitterFollowers: number; - twitterBio: string; - website: string; - - // Social Medias - crunchbase: string; - facebook: string; - linkedin: string; - twitter: string; - - /** - * Parse the dictionary returned by IAP. - */ - static fromIapResponse(values: any): Company { - const company = new Company(); - - company.name = isTrue(values.name); - company.email = first(values.email); - company.phone = first(values.phone_numbers); - company.isEnriched = !!Object.keys(values).length; - - company.emails = isTrue(values.email) ? values.email.join("\n") : null; - company.phones = isTrue(values.phone_numbers) ? values.phone_numbers.join("\n") : null; - - company.image = isTrue(values.logo); - company.website = formatUrl(values.domain); - company.description = isTrue(values.description); - company.address = isTrue(values.location); - - // Social Medias - company.facebook = isTrue(values.facebook); - company.twitter = isTrue(values.twitter); - company.linkedin = isTrue(values.linkedin); - company.crunchbase = isTrue(values.crunchbase); - - // Additional Information - company.employees = values.employees || null; - company.annualRevenue = isTrue(values.estimated_annual_revenue); - company.industry = isTrue(values.industry); - company.twitterBio = isTrue(values.twitter_bio); - company.twitterFollowers = values.twitter_followers || null; - company.foundedYear = values.founded_year; - company.timezone = isTrue(values.timezone); - company.timezoneUrl = isTrue(values.timezone_url); - company.tags = isTrue(values.tag) ? values.tag.join(", ") : null; - company.companyType = isTrue(values.company_type); - - return company; - } - - /** - * Unserialize the company object (reverse JSON.stringify). - */ - static fromJson(values: any): Company { - const company = new Company(); - - company.id = values.id; - company.name = values.name; - company.email = values.email; - company.phone = values.phone; - - company.address = values.address; - company.annualRevenue = values.annualRevenue; - company.companyType = values.companyType; - company.description = values.description; - company.emails = values.emails; - company.employees = values.employees; - company.foundedYear = values.foundedYear; - company.image = values.image; - company.industry = values.industry; - company.mobile = values.mobile; - company.phones = values.phones; - company.tags = values.tags; - company.timezone = values.timezone; - company.timezoneUrl = values.timezoneUrl; - company.twitterBio = values.twitterBio; - company.twitterFollowers = values.twitterFollowers; - company.website = values.website; - - company.crunchbase = values.crunchbase; - company.facebook = values.facebook; - company.twitter = values.twitter; - company.linkedin = values.linkedin; - - return company; - } - - /** - * Parse the dictionary returned by an Odoo database. - */ - static fromOdooResponse(values: any): Company { - if (!values.id || values.id < 0) { - return null; - } - - const iapInfo = values.additionalInfo || {}; - - const company = this.fromIapResponse(iapInfo); - - // Overwrite IAP information with the Odoo client database information - company.id = values.id; - company.name = values.name; - company.email = values.email; - company.phone = values.phone; - - company.mobile = values.mobile; - company.website = values.website; - company.image = values.image ? "data:image/png;base64," + values.image : null; - - if (values.address) { - company.address = ""; - - if (isTrue(values.address.street)) { - company.address += values.address.street + ", "; - } - if (isTrue(values.address.zip)) { - company.address += values.address.zip + " "; - } - if (isTrue(values.address.city)) { - company.address += values.address.city + " "; - } - if (isTrue(values.address.country)) { - company.address += values.address.country; - } - } - return company; - } -} diff --git a/gmail/src/models/email.ts b/gmail/src/models/email.ts index f623e8e70..43f543421 100644 --- a/gmail/src/models/email.ts +++ b/gmail/src/models/email.ts @@ -1,88 +1,189 @@ +import { OAuth2Client } from "google-auth-library"; +import { google } from "googleapis"; +import { simpleParser } from "mailparser"; import { ErrorMessage } from "../models/error_message"; +import pool from "../utils/db"; +import { User } from "./user"; + +const gmail = google.gmail({ version: "v1" }); /** * Represent the current email open in the Gmail application. */ export class Email { + userOAuthToken: string; accessToken: string; messageId: string; subject: string; - contactEmail: string; - contactFullEmail: string; - contactName: string; + emailFrom: string; + contacts: EmailContact[]; - constructor(messageId: string = null, accessToken: string = null) { - if (messageId) { - const userEmail = Session.getEffectiveUser().getEmail().toLowerCase(); + // Store on which record the current email has been logged + // >>> {"res.partner": [1, 2, 3]} + loggingState: Record; - this.accessToken = accessToken; + constructor( + userOAuthToken: string, + accessToken: string, + messageId: string, + subject: string, + emailFrom: string, + contacts: EmailContact[], + loggingState: Record, + ) { + this.userOAuthToken = userOAuthToken; + this.accessToken = accessToken; + this.messageId = messageId; + this.subject = subject; + this.emailFrom = emailFrom; + this.contacts = contacts; + this.loggingState = loggingState; + } - this.messageId = messageId; - const message = GmailApp.getMessageById(this.messageId); - this.subject = message.getSubject(); + /** + * Use the token we receive from Google to get the information about the opened email. + * + * Only get the headers of the email to not slow down the application + * (if we don't log the email, we only need the contacts that are in + * the email, we can delay the fetching of the email body and + * the attachments). + */ + static async getEmailHeadersFromGoogleToken(event: any): Promise> { + const messageId = event.gmail.messageId; + const auth = new OAuth2Client(); + auth.setCredentials({ access_token: event.authorizationEventObject.userOAuthToken }); + // @ts-ignore + const gmailResponse = await gmail.users.messages.get({ + id: messageId, + userId: "me", + format: "metadata", + auth, + headers: { "X-Goog-Gmail-Access-Token": event.gmail.accessToken }, + }); + // @ts-ignore + const rawHeaders = gmailResponse.data.payload.headers; + return Object.fromEntries(rawHeaders.map((h) => [h.name.toLowerCase(), h.value])); + } - const fromHeaders = message.getFrom(); - const sent = fromHeaders.toLowerCase().indexOf(userEmail) >= 0; - this.contactFullEmail = sent ? message.getTo() : message.getFrom(); - [this.contactName, this.contactEmail] = this._emailSplitTuple(this.contactFullEmail); - } + /** + * Once we got the headers and the user from the Gmail API, + * we can build the `Email` object. + */ + static async getEmailFromHeaders( + event: any, + headers: Record, + user: User, + ): Promise { + const userEmail = user.email.toLowerCase(); + const contacts = [ + ...this._emailSplitTuple(headers["to"] || "", userEmail), + ...this._emailSplitTuple(headers["from"] || "", userEmail), + ...this._emailSplitTuple(headers["cc"] || "", userEmail), + ...this._emailSplitTuple(headers["bcc"] || "", userEmail), + ]; + + return new Email( + event.authorizationEventObject.userOAuthToken, + event.gmail.accessToken, + event.gmail.messageId, + headers["subject"] || "", + headers["from"] || "", + contacts, + await this._getLoggingState(user, event.gmail.messageId), + ); } /** - * Ask the email body only if the user asked for it (e.g. asked to log the email). + * Fetch the information in the email that require the full EML. */ - public get body() { - GmailApp.setCurrentMessageAccessToken(this.accessToken); - const message = GmailApp.getMessageById(this.messageId); - return message.getBody(); + async getBodyAndAttachments(): Promise<[string, number, [string[][], ErrorMessage]]> { + const auth = new OAuth2Client(); + auth.setCredentials({ access_token: this.userOAuthToken }); + + // @ts-ignore + const gmailResponse = await gmail.users.messages.get({ + id: this.messageId, + userId: "me", + format: "raw", + auth, + headers: { "X-Goog-Gmail-Access-Token": this.accessToken }, + }); + + // @ts-ignore + const messageEmlB64 = gmailResponse.data.raw; + const messageEml = atob(messageEmlB64.replaceAll("-", "+").replaceAll("_", "/")); + + const mail = await simpleParser(messageEml); + return [ + mail.html || mail.text || "", + mail.date.getTime(), + this._getAttachments(mail.attachments), + ]; } /** - * Parse a full FROM header and return the name part and the email part. + * Parse a full FROM header and return the name and email parts. * * E.G. - * "BOB" => ["BOB", "bob@example.com"] - * bob@example.com => ["bob@example.com", "bob@example.com"] + * "BOB" + * => [["BOB", "bob@example.com"]] * + * bob@example.com + * => [["bob@example.com", "bob@example.com"]] + * + * alice@example.com, bob@example.com + * => [ + * ["alice@example.com", "alice@example.com"], + * ["bob@example.com", "bob@example.com"] + * ] + * + * "Alice" , "BOB" + * => [ + * ["alice@example.com", "alice@example.com"], + * ["bob@example.com", "bob@example.com"] + * ] + * + * , + * => [ + * ["alice@example.com", "alice@example.com"], + * ["bob@example.com", "bob@example.com"] + * ] */ - _emailSplitTuple(fullEmail: string): [string, string] { - const match = fullEmail.match(/(.*)<(.*)>/); - fullEmail = fullEmail.replace("<", "").replace(">", ""); - - if (!match) { - return [fullEmail, fullEmail]; - } - - const [_, name, email] = match; - - if (!name || !email) { - return [fullEmail, fullEmail]; - } + private static _emailSplitTuple(fullEmail: string, userEmail: string): EmailContact[] { + const contacts = []; + const re = /(.*?)<(.*?)>/; + for (const part of fullEmail.split(",")) { + if (part.toLowerCase().indexOf(userEmail) >= 0 || !part.trim()?.length) { + // Skip the user's email + continue; + } - const cleanedName = name.replace(/\"/g, "").trim(); - if (!cleanedName || !cleanedName.length) { - return [fullEmail, fullEmail]; + const result = part.match(re); + if (!result) { + contacts.push(new EmailContact(part.trim(), part.trim(), part.trim())); + continue; + } + const email = result[2].trim(); + let name = result[1].replace(/\"/g, "").trim() || email; + contacts.push(new EmailContact(name, email, part.trim())); } - - return [cleanedName, email]; + return contacts; } /** * Unserialize the email object (reverse JSON.stringify). */ static fromJson(values: any): Email { - const email = new Email(); - - email.accessToken = values.accessToken; - email.messageId = values.messageId; - email.subject = values.subject; - - email.contactEmail = values.contactEmail; - email.contactFullEmail = values.contactFullEmail; - email.contactName = values.contactName; - - return email; + return new Email( + values.userOAuthToken, + values.accessToken, + values.messageId, + values.subject, + values.emailFrom, + values.contacts.map((c) => EmailContact.fromJson(c)), + values.loggingState, + ); } /** @@ -96,10 +197,7 @@ export class Email { * - If no attachment, return an empty array and an empty error message. * - Otherwise, the list of attachments base 64 encoded and an empty error message */ - getAttachments(): [string[][], ErrorMessage] { - GmailApp.setCurrentMessageAccessToken(this.accessToken); - const message = GmailApp.getMessageById(this.messageId); - const gmailAttachments = message.getAttachments(); + private _getAttachments(gmailAttachments): [string[][], ErrorMessage] { const attachments: string[][] = []; // The size limit of the POST request are 50 MB @@ -109,18 +207,94 @@ export class Email { let totalAttachmentsSize = 0; for (const gmailAttachment of gmailAttachments) { - const bytesSize = gmailAttachment.getSize(); + if (gmailAttachment.contentDisposition === "inline") { + // Outlook inline images + continue; + } + const bytesSize = gmailAttachment.content.length; totalAttachmentsSize += bytesSize; if (totalAttachmentsSize > MAXIMUM_ATTACHMENTS_SIZE) { return [null, new ErrorMessage("attachments_size_exceeded")]; } - - const name = gmailAttachment.getName(); - const content = Utilities.base64Encode(gmailAttachment.getBytes()); + const name = gmailAttachment.filename; + const content = gmailAttachment.content.toString("base64"); attachments.push([name, content]); } return [attachments, new ErrorMessage(null)]; } + + /** + * Save the fact that we logged the email on the record, in the cache. + * + * Returns: + * True if the record was not yet marked as "logged" + * False if we already logged the email on the record + */ + async setLoggingState(user: User, resModel: string, resId: number) { + this.loggingState[resModel].push(resId); + await pool.query( + ` + INSERT INTO email_logs (user_id, message_id, res_id, res_model) + VALUES ($1, $2, $3, $4) + `, + [user.id, this.messageId, resId, resModel], + ); + } + + /** + * Check if the email has not been logged on the record. + * + * Returns: + * True if the record was not yet marked as "logged" + * False if we already logged the email on the record + */ + checkLoggingState(resModel: string, resId: number): boolean { + return this.loggingState[resModel].includes(resId); + } + + /** + * Get the logging state for the current email + * (that way, we do only one query for all the records we will see), + */ + private static async _getLoggingState( + user: User, + messageId: string, + ): Promise> { + const result = await pool.query( + ` + SELECT res_model, res_id + FROM email_logs + WHERE user_id = $1 AND message_id = $2 + `, + [user.id, messageId], + ); + const ret: Record = { + "res.partner": [], + "crm.lead": [], + "helpdesk.ticket": [], + "project.task": [], + }; + for (const row of result.rows) { + ret[row.res_model].push(row.res_id); + } + return ret; + } +} + +export class EmailContact { + name: string; + email: string; + fullEmail: string; + + constructor(name: string, email: string, fullEmail: string) { + this.name = name; + this.email = email; + this.fullEmail = fullEmail; + } + + static fromJson(values: any): EmailContact { + return new EmailContact(values.name, values.email, values.fullEmail); + } } diff --git a/gmail/src/models/error_message.ts b/gmail/src/models/error_message.ts index 05153ec7b..c5f5a747a 100644 --- a/gmail/src/models/error_message.ts +++ b/gmail/src/models/error_message.ts @@ -1,5 +1,3 @@ -import { _t } from "../services/translation"; - /** * Represent an error and translate its code to a message. */ @@ -7,14 +5,6 @@ import { _t } from "../services/translation"; const _ERROR_CODE_MESSAGES: Record = { odoo: null, // Message is contained in the additional information http_error_odoo: "Could not connect to database. Try to log out and in.", - insufficient_credit: "Not enough credits to enrich.", - company_created: null, - company_updated: null, - // IAP - http_error_iap: "Our IAP server is down, please come back later.", - exhausted_requests: - "Oops, looks like you have exhausted your free enrichment requests. Please log in to try again.", - missing_data: "No insights found for this address", unknown: "Something bad happened. Please, try again later.", // Attachment attachments_size_exceeded: @@ -27,37 +17,15 @@ const _ERROR_CODE_MESSAGES: Record = { */ export class ErrorMessage { code: string; - message: string; - information: string; - - // False if the error means that we can not contact the Odoo database - // (e.g. HTTP error) - canContactOdooDatabase: boolean = true; - - canCreateCompany: boolean = true; + private message: string; constructor(code: string = null, information: any = null) { - if (code) { - this.setError(code, information); - } - } - - /** - * Set the code error and find the appropriate message to display. - */ - setError(code: string, information: any = null) { - if (code === "no_data") { - code = "missing_data"; - information = null; - } - this.code = code; - this.information = information; - this.message = _t(_ERROR_CODE_MESSAGES[this.code]); + this.message = information || _ERROR_CODE_MESSAGES[code] || _ERROR_CODE_MESSAGES["unknown"]; + } - if (code === "http_error_odoo") { - this.canContactOdooDatabase = false; - } + toString(_t: Function): string { + return _t(this.message); } /** @@ -65,11 +33,7 @@ export class ErrorMessage { */ static fromJson(values: any) { const error = new ErrorMessage(); - error.code = values.code; error.message = values.message; - error.canContactOdooDatabase = values.canContactOdooDatabase; - error.canCreateCompany = values.canCreateCompany; - error.information = values.information; return error; } } diff --git a/gmail/src/models/lead.ts b/gmail/src/models/lead.ts index dce33c26c..ac8b87af1 100644 --- a/gmail/src/models/lead.ts +++ b/gmail/src/models/lead.ts @@ -1,7 +1,8 @@ +import { URLS } from "../consts"; import { postJsonRpc } from "../utils/http"; -import { isTrue } from "../utils/format"; -import { URLS } from "../const"; -import { getAccessToken } from "src/services/odoo_auth"; +import { Email } from "./email"; +import { Partner } from "./partner"; +import { User } from "./user"; /** * Represent a "crm.lead" record. @@ -9,26 +10,41 @@ import { getAccessToken } from "src/services/odoo_auth"; export class Lead { id: number; name: string; - expectedRevenue: string; - probability: number; - recurringRevenue: string; - recurringPlan: string; + revenuesDescription: string; /** * Make a RPC call to the Odoo database to create a lead * and return the ID of the newly created record. */ - static createLead(partnerId: number, emailBody: string, emailSubject: string): number { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.CREATE_LEAD; - const accessToken = getAccessToken(); + static async createLead( + user: User, + partner: Partner, + email: Email, + ): Promise<[Lead, Partner] | null> { + const [body, _, attachmentsParsed] = await email.getBodyAndAttachments(); - const response = postJsonRpc( - url, - { email_body: emailBody, email_subject: emailSubject, partner_id: partnerId }, - { Authorization: "Bearer " + accessToken }, + const response = await postJsonRpc( + user.odooUrl + URLS.CREATE_LEAD, + { + email_body: body, + email_subject: email.subject, + partner_id: partner.id, + partner_email: partner.email, + partner_name: partner.name, + attachments: attachmentsParsed[0], + }, + { Authorization: "Bearer " + user.odooToken }, ); - return response ? response.lead_id || null : null; + if (!response?.id) { + return null; + } + if (!partner.id) { + partner.id = response.partner_id; + partner.image = response.partner_image; + partner.isWritable = true; + } + return [Lead.fromOdooResponse(response), partner]; } /** @@ -38,10 +54,7 @@ export class Lead { const lead = new Lead(); lead.id = values.id; lead.name = values.name; - lead.expectedRevenue = values.expectedRevenue; - lead.probability = values.probability; - lead.recurringRevenue = values.recurringRevenue; - lead.recurringPlan = values.recurringPlan; + lead.revenuesDescription = values.revenuesDescription; return lead; } @@ -50,16 +63,9 @@ export class Lead { */ static fromOdooResponse(values: any): Lead { const lead = new Lead(); - lead.id = values.lead_id; + lead.id = values.id; lead.name = values.name; - lead.expectedRevenue = values.expected_revenue; - lead.probability = values.probability; - - if (isTrue(values.recurring_revenue) && isTrue(values.recurring_plan)) { - lead.recurringRevenue = values.recurring_revenue; - lead.recurringPlan = values.recurring_plan; - } - + lead.revenuesDescription = values.revenues_description; return lead; } } diff --git a/gmail/src/models/partner.ts b/gmail/src/models/partner.ts index 0ac450ea7..97a3494ea 100644 --- a/gmail/src/models/partner.ts +++ b/gmail/src/models/partner.ts @@ -1,12 +1,10 @@ -import { Company } from "./company"; +import { URLS } from "../consts"; +import { ErrorMessage } from "../models/error_message"; +import { postJsonRpc } from "../utils/http"; import { Lead } from "./lead"; import { Task } from "./task"; import { Ticket } from "./ticket"; -import { postJsonRpc, postJsonRpcCached } from "../utils/http"; -import { URLS } from "../const"; -import { ErrorMessage } from "../models/error_message"; -import { getAccessToken } from "src/services/odoo_auth"; -import { getOdooServerUrl } from "src/services/app_properties"; +import { User } from "./user"; /** * Represent the current partner and all the information about him. @@ -18,15 +16,28 @@ export class Partner { image: string; isCompany: boolean; + companyName: string; phone: string; mobile: string; - company: Company; leads: Lead[]; + leadCount: number; tickets: Ticket[]; + ticketCount: number; tasks: Task[]; + taskCount: number; + + isWritable: boolean; - isWriteable: boolean; + /** + * Return the image to show in the interface for the current partner. + */ + getImage() { + if (!this.id || this.id < 0 || !this.image) { + return "/assets/person.png"; + } + return this.image; + } /** * Unserialize the partner object (reverse JSON.stringify). @@ -40,19 +51,27 @@ export class Partner { partner.image = values.image; partner.isCompany = values.isCompany; + partner.companyName = values.companyName; partner.phone = values.phone; partner.mobile = values.mobile; - partner.company = values.company ? Company.fromJson(values.company) : null; - partner.isWriteable = values.isWriteable; + partner.leadCount = values.leadCount; + partner.ticketCount = values.ticketCount; + partner.taskCount = values.taskCount; - partner.leads = values.leads ? values.leads.map((leadValues: any) => Lead.fromJson(leadValues)) : null; + partner.isWritable = values.isWritable; + + partner.leads = values.leads + ? values.leads.map((leadValues: any) => Lead.fromJson(leadValues)) + : null; partner.tickets = values.tickets ? values.tickets.map((ticketValues: any) => Ticket.fromJson(ticketValues)) : null; - partner.tasks = values.tasks ? values.tasks.map((taskValues: any) => Task.fromJson(taskValues)) : null; + partner.tasks = values.tasks + ? values.tasks.map((taskValues: any) => Task.fromJson(taskValues)) + : null; return partner; } @@ -66,86 +85,38 @@ export class Partner { partner.name = values.name; partner.email = values.email; - partner.image = values.image ? "data:image/png;base64," + values.image : null; + partner.image = values.image; + partner.isCompany = values.is_company; partner.isCompany = values.is_company; + partner.companyName = values.company_name; + partner.phone = values.phone; partner.mobile = values.mobile; - - // Undefined should be considered as True for retro-compatibility - partner.isWriteable = values.can_write_on_partner !== false; - - if (values.company && values.company.id && values.company.id > 0) { - partner.company = Company.fromOdooResponse(values.company); - } + partner.isWritable = values.can_write_on_partner; return partner; } - /** - * Try to find information about the given email /name. - * - * If we are not logged to an Odoo database, enrich the email domain with IAP. - * Otherwise fetch the partner on the user database. - * - * See `getPartner` - */ - static enrichPartner(email: string, name: string): [Partner, number[], boolean, boolean, ErrorMessage] { - const odooServerUrl = getOdooServerUrl(); - const odooAccessToken = getAccessToken(); - - if (odooServerUrl && odooAccessToken) { - return this.getPartner(email, name); - } else { - const [partner, error] = this._enrichFromIap(email, name); - return [partner, null, false, false, error]; - } - } - - /** - * Extract the email domain and send a request to IAP - * to find information about the company. - */ - static _enrichFromIap(email: string, name: string): [Partner, ErrorMessage] { - const odooSharedSecret = PropertiesService.getScriptProperties().getProperty("ODOO_SHARED_SECRET"); - const userEmail = Session.getEffectiveUser().getEmail(); - - const senderDomain = email.split("@").pop(); - - const response = postJsonRpcCached(URLS.IAP_COMPANY_ENRICHMENT, { - email: userEmail, - domain: senderDomain, - secret: odooSharedSecret, - }); - - const error = new ErrorMessage(); - if (!response) { - error.setError("http_error_iap"); - } else if (response.error && response.error.length) { - error.setError(response.error); - } - - const partner = new Partner(); - partner.name = name; - partner.email = email; - - if (response && response.name) { - partner.company = Company.fromIapResponse(response); - } - - return [partner, error]; - } /** * Create a "res.partner" with the given values in the Odoo database. */ - static savePartner(partnerValues: any): number { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.PARTNER_CREATE; - const odooAccessToken = getAccessToken(); - - const response = postJsonRpc(url, partnerValues, { - Authorization: "Bearer " + odooAccessToken, + static async savePartner(user: User, partner: Partner): Promise { + const partnerValues = { + name: partner.name, + email: partner.email, + }; + + const response = await postJsonRpc(user.odooUrl + URLS.PARTNER_CREATE, partnerValues, { + Authorization: "Bearer " + user.odooToken, }); - return response && response.id; + if (!response?.id) { + return null; + } + partner.id = response.id; + partner.image = response.image; + partner.isWritable = true; + return partner; } /** @@ -153,119 +124,94 @@ export class Partner { * * Return * - The Partner related to the given email address - * - The list of Odoo companies in which the current user belongs * - True if the current user can create partner in his Odoo database * - True if the current user can create projects in his Odoo database * - The error message if something bad happened */ - static getPartner( - email: string, + static async getPartner( + user: User, name: string, + email: string, partnerId: number = null, - ): [Partner, number[], boolean, boolean, ErrorMessage] { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.GET_PARTNER; - const odooAccessToken = getAccessToken(); - - const response = postJsonRpc( - url, - { email: email, name: name, partner_id: partnerId }, - { Authorization: "Bearer " + odooAccessToken }, + ): Promise<[Partner, boolean, boolean, ErrorMessage]> { + if (!user.odooUrl || !user.odooToken) { + const error = new ErrorMessage("http_error_odoo"); + const partner = Partner.fromJson({ name, email }); + return [partner, false, false, error]; + } + + const response = await postJsonRpc( + user.odooUrl + URLS.GET_PARTNER, + { email: email, partner_id: partnerId }, + { Authorization: "Bearer " + user.odooToken }, ); + if (response && response.error) { + const error = new ErrorMessage("odoo", response.error); + const partner = Partner.fromJson({ name, email }); + return [partner, false, false, error]; + } + if (!response || !response.partner) { const error = new ErrorMessage("http_error_odoo"); - const partner = Partner.fromJson({ name: name, email: email }); - return [partner, null, false, false, error]; + const partner = Partner.fromJson({ name, email }); + return [partner, false, false, error]; } const error = new ErrorMessage(); - - if (response.enrichment_info && response.enrichment_info.type) { - error.setError(response.enrichment_info.type, response.enrichment_info.info); - } else if (response.partner.enrichment_info && response.partner.enrichment_info.type) { - error.setError(response.partner.enrichment_info.type, response.partner.enrichment_info.info); - } - - const partner = Partner.fromOdooResponse(response.partner); + const partner = Partner.fromOdooResponse({ name, email, ...response.partner }); // Parse leads if (response.leads) { - partner.leads = response.leads.map((leadValues: any) => Lead.fromOdooResponse(leadValues)); + partner.leadCount = response.lead_count; + partner.leads = response.leads.map((leadValues: any) => + Lead.fromOdooResponse(leadValues), + ); } // Parse tickets if (response.tickets) { - partner.tickets = response.tickets.map((ticketValues: any) => Ticket.fromOdooResponse(ticketValues)); + partner.ticketCount = response.ticket_count; + partner.tickets = response.tickets.map((ticketValues: any) => + Ticket.fromOdooResponse(ticketValues), + ); } // Parse tasks if (response.tasks) { - partner.tasks = response.tasks.map((taskValues: any) => Task.fromOdooResponse(taskValues)); + partner.taskCount = response.task_count; + partner.tasks = response.tasks.map((taskValues: any) => + Task.fromOdooResponse(taskValues), + ); } const canCreateProject = response.can_create_project !== false; - const odooUserCompanies = response.user_companies || null; // undefined must be considered as true const canCreatePartner = response.can_create_partner !== false; - return [partner, odooUserCompanies, canCreatePartner, canCreateProject, error]; + return [partner, canCreatePartner, canCreateProject, error]; } /** * Perform a search on the Odoo database and return the list of matched partners. */ - static searchPartner(query: string): [Partner[], ErrorMessage] { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.SEARCH_PARTNER; - const odooAccessToken = getAccessToken(); - - const response = postJsonRpc(url, { search_term: query }, { Authorization: "Bearer " + odooAccessToken }); + static async searchPartner( + user: User, + query: string | string[], + ): Promise<[Partner[], ErrorMessage]> { + const response = await postJsonRpc( + user.odooUrl + URLS.SEARCH_PARTNER, + { query }, + { Authorization: "Bearer " + user.odooToken }, + ); - if (!response || !response.partners) { + if (!response?.length) { return [[], new ErrorMessage("http_error_odoo")]; } - return [response.partners.map((values: any) => Partner.fromOdooResponse(values)), new ErrorMessage()]; - } - - /** - * Create and enrich the company of the given partner. - */ - static createCompany(partnerId: number): [Company, ErrorMessage] { - return this._enrichOrCreateCompany(partnerId, URLS.CREATE_COMPANY); - } - - /** - * Enrich the existing company. - */ - static enrichCompany(companyId: number): [Company, ErrorMessage] { - return this._enrichOrCreateCompany(companyId, URLS.ENRICH_COMPANY); - } - - static _enrichOrCreateCompany(partnerId: number, endpoint: string): [Company, ErrorMessage] { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + endpoint; - const odooAccessToken = getAccessToken(); - - const response = postJsonRpc(url, { partner_id: partnerId }, { Authorization: "Bearer " + odooAccessToken }); - - if (!response) { - return [null, new ErrorMessage("http_error_odoo")]; - } - - if (response.error) { - return [null, new ErrorMessage("odoo", response.error)]; - } - - let error = new ErrorMessage(); - - if (response.enrichment_info && response.enrichment_info.type) { - error.setError(response.enrichment_info.type, response.enrichment_info.info); - } - - if (error.code) { - error.canCreateCompany = false; - } - - const company = response.company ? Company.fromOdooResponse(response.company) : null; - return [company, error]; + return [ + response[0].map((values: any) => Partner.fromOdooResponse(values)), + new ErrorMessage(), + ]; } } diff --git a/gmail/src/models/project.ts b/gmail/src/models/project.ts index 02202e00d..835ca37d5 100644 --- a/gmail/src/models/project.ts +++ b/gmail/src/models/project.ts @@ -1,7 +1,7 @@ -import { postJsonRpc } from "../utils/http"; -import { URLS } from "../const"; +import { URLS } from "../consts"; import { ErrorMessage } from "../models/error_message"; -import { getAccessToken } from "src/services/odoo_auth"; +import { postJsonRpc } from "../utils/http"; +import { User } from "./user"; /** * Represent a "project.project" record. @@ -10,6 +10,8 @@ export class Project { id: number; name: string; partnerName: string; + stageName: string; + companyName: string; /** * Unserialize the project object (reverse JSON.stringify). @@ -19,6 +21,8 @@ export class Project { project.id = values.id; project.name = values.name; project.partnerName = values.partnerName; + project.stageName = values.stageName; + project.companyName = values.companyName; return project; } @@ -27,39 +31,46 @@ export class Project { */ static fromOdooResponse(values: any): Project { const project = new Project(); - project.id = values.project_id; + project.id = values.id; project.name = values.name; project.partnerName = values.partner_name; + project.stageName = values.stage_name; + project.companyName = values.company_name; return project; } /** * Make a RPC call to the Odoo database to search a project. */ - static searchProject(query: string): [Project[], ErrorMessage] { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.SEARCH_PROJECT; - const odooAccessToken = getAccessToken(); + static async searchProject(user: User, query: string): Promise<[Project[], ErrorMessage]> { + const response = await postJsonRpc( + user.odooUrl + URLS.SEARCH_PROJECT, + { query }, + { Authorization: "Bearer " + user.odooToken }, + ); - const response = postJsonRpc(url, { search_term: query }, { Authorization: "Bearer " + odooAccessToken }); - - if (!response) { + if (!response?.length) { return [[], new ErrorMessage("http_error_odoo")]; } - return [response.map((values: any) => Project.fromOdooResponse(values)), new ErrorMessage()]; + return [ + response[0].map((values: any) => Project.fromOdooResponse(values)), + new ErrorMessage(), + ]; } /** * Make a RPC call to the Odoo database to create a project * and return the newly created record. */ - static createProject(name: string): Project { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.CREATE_PROJECT; - const odooAccessToken = getAccessToken(); - - const response = postJsonRpc(url, { name: name }, { Authorization: "Bearer " + odooAccessToken }); + static async createProject(user: User, name: string): Promise { + const response = await postJsonRpc( + user.odooUrl + URLS.CREATE_PROJECT, + { name: name }, + { Authorization: "Bearer " + user.odooToken }, + ); - const projectId = response ? response.project_id || null : null; + const projectId = response ? response.id || null : null; if (!projectId) { return null; } diff --git a/gmail/src/models/state.ts b/gmail/src/models/state.ts index 1760dbaf5..0b2b42e82 100644 --- a/gmail/src/models/state.ts +++ b/gmail/src/models/state.ts @@ -1,59 +1,39 @@ -import { isTrue } from "../utils/format"; import { Email } from "./email"; import { Partner } from "./partner"; import { Project } from "./project"; -import { Lead } from "./lead"; -import { ErrorMessage } from "./error_message"; -import { getAccessToken, getOdooAuthUrl } from "../services/odoo_auth"; -import { getOdooServerUrl } from "src/services/app_properties"; /** * Object which contains all data for the application. * - * In App-Script, all event handler are function and not method. We can only pass string - * as arguments. So this object is serialized, then given to the event handler and then + * This object is serialized, then given to the event handler and then * unserialize to retrieve the original object. - * - * That's how we manage the state of the application without performing a big amount of - * read / write in the cache. */ - export class State { // Contact of the current card partner: Partner; canCreatePartner: boolean; // Opened email with headers email: Email; - // ID list of the Odoo user companies - odooUserCompanies: number[]; // Searched partners in the search view searchedPartners: Partner[]; // Searched projects in the search view searchedProjects: Project[]; canCreateProject: boolean; - // Current error message displayed on the card - error: ErrorMessage; - // Used in the company card - isCompanyDescriptionUnfolded: boolean; constructor( partner: Partner, canCreatePartner: boolean, email: Email, - odooUserCompanies: number[], partners: Partner[], searchedProjects: Project[], canCreateProject: boolean, - error: ErrorMessage, ) { this.partner = partner; this.canCreatePartner = canCreatePartner; this.email = email; - this.odooUserCompanies = odooUserCompanies; this.searchedPartners = partners; this.searchedProjects = searchedProjects; this.canCreateProject = canCreateProject; - this.error = error; } toJson(): string { @@ -63,21 +43,16 @@ export class State { /** * Unserialize the state object (reverse JSON.stringify). */ - static fromJson(json: string): State { - const values = JSON.parse(json); - + static fromJson(values: any): State { const partnerValues = values.partner || {}; const canCreatePartner = values.canCreatePartner; const emailValues = values.email || {}; - const errorValues = values.error || {}; const partnersValues = values.searchedPartners; const projectsValues = values.searchedProjects; const canCreateProject = values.canCreateProject; const partner = Partner.fromJson(partnerValues); const email = Email.fromJson(emailValues); - const error = ErrorMessage.fromJson(errorValues); - const odooUserCompanies = values.odooUserCompanies; const searchedPartners = partnersValues ? partnersValues.map((partnerValues: any) => Partner.fromJson(partnerValues)) : null; @@ -85,133 +60,13 @@ export class State { ? projectsValues.map((projectValues: any) => Project.fromJson(projectValues)) : null; - // "isCompanyDescriptionUnfolded" is not copied - // to re-fold the description if we go back / refresh - return new State( partner, canCreatePartner, email, - odooUserCompanies, searchedPartners, searchedProjects, canCreateProject, - error, ); } - - /** - * Return the companies of the Odoo user as a GET parameter to add in a URL or an - * empty string if the information is missing. - * - * e.g. - * &cids=1,3,7 - */ - get odooCompaniesParameter(): string { - if (this.odooUserCompanies && this.odooUserCompanies.length) { - const cids = this.odooUserCompanies.sort().join(","); - return `&cids=${cids}`; - } - return ""; - } - - /** - * Cache / user properties management. - * - * Introduced with static getter / setter because they are shared between all the - * application cards. - */ - static get accessToken() { - const accessToken = getAccessToken(); - return isTrue(accessToken); - } - - static get isLogged(): boolean { - return !!this.accessToken; - } - - /** - * Return the URL require to login to the Odoo database. - */ - static get odooLoginUrl(): string { - const loginUrl = getOdooAuthUrl(); - return isTrue(loginUrl); - } - /** - * Return the shared secret between the add-on and IAP - * (which is used to authenticate the add-on to IAP). - */ - static get odooSharedSecret(): string { - const scriptProperties = PropertiesService.getScriptProperties(); - const sharedSecret = scriptProperties.getProperty("ODOO_SHARED_SECRET"); - return isTrue(sharedSecret); - } - - /** - * Dictionary which inform us on which record we already logged the email. - * So the user can not log 2 times the same email on the same record. - * This is stored into the cache, so we don't need to modify the Odoo models. - * - * Note: the cache expire after 6 hours. - * - * Returns: - * { - * "partners": [3, 6], // email already logged on the partner 3 and 6 - * "leads": [7, 14], - * } - */ - static getLoggingState(messageId: string) { - const cache = CacheService.getUserCache(); - const loggingStateStr = cache.get("ODOO_LOGGING_STATE_" + getOdooServerUrl() + "_" + messageId); - - const defaultValues: Record = { - partners: [], - leads: [], - tickets: [], - tasks: [], - }; - - if (!loggingStateStr || !loggingStateStr.length) { - return defaultValues; - } - return { ...defaultValues, ...JSON.parse(loggingStateStr) }; - } - - /** - * Save the fact that we logged the email on the record, in the cache. - * - * Returns: - * True if the record was not yet marked as "logged" - * False if we already logged the email on the record - */ - static setLoggingState(messageId: string, res_model: string, res_id: number): boolean { - const loggingState = this.getLoggingState(messageId); - if (loggingState[res_model].indexOf(res_id) < 0) { - loggingState[res_model].push(res_id); - const cache = CacheService.getUserCache(); - - // The cache key depend on the current email open and on the Odoo database - const cacheKey = "ODOO_LOGGING_STATE_" + getOdooServerUrl() + "_" + messageId; - - cache.put( - cacheKey, - JSON.stringify(loggingState), - 21600, // 6 hours, maximum cache life time - ); - return true; - } - return false; - } - - /** - * Check if the email has not been logged on the record. - * - * Returns: - * True if the record was not yet marked as "logged" - * False if we already logged the email on the record - */ - static checkLoggingState(messageId: string, res_model: string, res_id: number): boolean { - const loggingState = this.getLoggingState(messageId); - return loggingState[res_model].indexOf(res_id) < 0; - } } diff --git a/gmail/src/models/task.ts b/gmail/src/models/task.ts index ba8b54530..a83dd3bfc 100644 --- a/gmail/src/models/task.ts +++ b/gmail/src/models/task.ts @@ -1,6 +1,8 @@ +import { URLS } from "../consts"; import { postJsonRpc } from "../utils/http"; -import { URLS } from "../const"; -import { getAccessToken } from "src/services/odoo_auth"; +import { Email } from "./email"; +import { Partner } from "./partner"; +import { User } from "./user"; /** * Represent a "project.task" record. @@ -26,7 +28,7 @@ export class Task { */ static fromOdooResponse(values: any): Task { const task = new Task(); - task.id = values.task_id; + task.id = values.id; task.name = values.name; task.projectName = values.project_name; return task; @@ -36,25 +38,34 @@ export class Task { * Make a RPC call to the Odoo database to create a task * and return the ID of the newly created record. */ - static createTask(partnerId: number, projectId: number, emailBody: string, emailSubject: string): Task { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.CREATE_TASK; - const odooAccessToken = getAccessToken(); - - const response = postJsonRpc( - url, - { email_subject: emailSubject, email_body: emailBody, project_id: projectId, partner_id: partnerId }, - { Authorization: "Bearer " + odooAccessToken }, + static async createTask( + user: User, + partner: Partner, + projectId: number, + email: Email, + ): Promise<[Task, Partner] | null> { + const [body, _, attachmentsParsed] = await email.getBodyAndAttachments(); + const response = await postJsonRpc( + user.odooUrl + URLS.CREATE_TASK, + { + email_body: body, + email_subject: email.subject, + partner_email: partner.email, + partner_id: partner.id, + partner_name: partner.name, + project_id: projectId, + attachments: attachmentsParsed[0], + }, + { Authorization: "Bearer " + user.odooToken }, ); - - const taskId = response ? response.task_id || null : null; - - if (!taskId) { + if (!response?.id) { return null; } - - return Task.fromJson({ - id: taskId, - name: response.name, - }); + if (!partner.id) { + partner.id = response.partner_id; + partner.image = response.partner_image; + partner.isWritable = true; + } + return [Task.fromOdooResponse(response), partner]; } } diff --git a/gmail/src/models/ticket.ts b/gmail/src/models/ticket.ts index fec7f420f..e43948327 100644 --- a/gmail/src/models/ticket.ts +++ b/gmail/src/models/ticket.ts @@ -1,6 +1,8 @@ +import { URLS } from "../consts"; import { postJsonRpc } from "../utils/http"; -import { URLS } from "../const"; -import { getAccessToken } from "src/services/odoo_auth"; +import { Email } from "./email"; +import { Partner } from "./partner"; +import { User } from "./user"; /** * Represent a "helpdesk.ticket" record. @@ -8,22 +10,40 @@ import { getAccessToken } from "src/services/odoo_auth"; export class Ticket { id: number; name: string; + stageName: string; /** * Make a RPC call to the Odoo database to create a ticket * and return the ID of the newly created record. */ - static createTicket(partnerId: number, emailBody: string, emailSubject: string): number { - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.CREATE_TICKET; - const odooAccessToken = getAccessToken(); - - const response = postJsonRpc( - url, - { email_body: emailBody, email_subject: emailSubject, partner_id: partnerId }, - { Authorization: "Bearer " + odooAccessToken }, + static async createTicket( + user: User, + partner: Partner, + email: Email, + ): Promise<[Ticket, Partner] | null> { + const [body, _, attachmentsParsed] = await email.getBodyAndAttachments(); + const response = await postJsonRpc( + user.odooUrl + URLS.CREATE_TICKET, + { + email_body: body, + email_subject: email.subject, + partner_email: partner.email, + partner_id: partner.id, + partner_name: partner.name, + attachments: attachmentsParsed[0], + }, + { Authorization: "Bearer " + user.odooToken }, ); - return response ? response.ticket_id || null : null; + if (!response?.id) { + return null; + } + if (!partner.id) { + partner.id = response.partner_id; + partner.image = response.partner_image; + partner.isWritable = true; + } + return [Ticket.fromOdooResponse(response), partner]; } /** @@ -33,6 +53,7 @@ export class Ticket { const ticket = new Ticket(); ticket.id = values.id; ticket.name = values.name; + ticket.stageName = values.stageName; return ticket; } @@ -41,8 +62,9 @@ export class Ticket { */ static fromOdooResponse(values: any): Ticket { const ticket = new Ticket(); - ticket.id = values.ticket_id; + ticket.id = values.id; ticket.name = values.name; + ticket.stageName = values.stage_name; return ticket; } } diff --git a/gmail/src/models/user.ts b/gmail/src/models/user.ts new file mode 100644 index 000000000..6ed855563 --- /dev/null +++ b/gmail/src/models/user.ts @@ -0,0 +1,174 @@ +import * as crypto from "crypto"; +import { OAuth2Client } from "google-auth-library"; +import { CLIENT_ID } from "../consts"; +import pool from "../utils/db"; + +export class User { + id?: number; + email: string; + odooUrl?: string; + odooToken?: string; + + // That token is used to authenticate the user when he's redirected + // to the callback URL, and can be used only once + loginToken?: string; + loginTokenExpireAt?: Date; + + // Store the translation for the current user, based on the language + // of his `res.users` on the Odoo side + translations?: any; + translationsExpireAt?: Date; + + constructor( + id: number, + email: string, + odooUrl?: string, + odooToken?: string, + loginToken?: string, + loginTokenExpireAt?: Date, + translations?: any, + translationsExpireAt?: Date, + ) { + this.id = id; + this.email = email; + this.odooUrl = odooUrl; + this.odooToken = odooToken; + this.loginToken = loginToken; + this.loginTokenExpireAt = loginTokenExpireAt; + this.translations = translations; + this.translationsExpireAt = translationsExpireAt; + } + + async save() { + await pool.query( + ` + INSERT INTO users_settings ( + email, + odoo_url, + odoo_token, + login_token, + login_token_expire_at, + translations, + translations_expire_at + ) + VALUES ($1, $2, $3, $4, $5, $6, $7) + ON CONFLICT (email) DO UPDATE + SET odoo_url = EXCLUDED.odoo_url, + odoo_token = EXCLUDED.odoo_token, + login_token = EXCLUDED.login_token, + login_token_expire_at = EXCLUDED.login_token_expire_at, + translations = EXCLUDED.translations, + translations_expire_at = EXCLUDED.translations_expire_at + `, + [ + this.email, + this.odooUrl, + this.odooToken, + this.loginToken, + this.loginTokenExpireAt, + this.translations, + this.translationsExpireAt, + ], + ); + } + + /** + * Generate the login token and set the expiration date in 1 hour. + */ + async generateLoginToken(): Promise { + const EXPIRATION_DURATION_MS = 60 * 60 * 1000; + this.loginTokenExpireAt = new Date(Date.now() + EXPIRATION_DURATION_MS); + this.loginToken = crypto.randomBytes(64).toString("hex"); + await this.save(); + return this.loginToken; + } + + /** + * Check the token we receive from Google, and get the user base on the email. + */ + static async getUserFromGoogleToken(event: any): Promise { + const oAuth2Client = new OAuth2Client(); + const decodedToken = await oAuth2Client.verifyIdToken({ + idToken: event.authorizationEventObject.userIdToken, + audience: CLIENT_ID, + }); + const payload = decodedToken.getPayload(); + if (!payload.email || !payload.email_verified) { + throw new Error("Failed to authenticate the user"); + } + return await User._getUserFromEmail(payload.email); + } + + /** + * Check the token we receive from the user's browser, and get the user. + * + * The login token can only be used once, and is reset after getting the user. + */ + static async getUserFromLoginToken(email: string, loginToken: string): Promise { + const user = await User._getUserFromEmail(email); + + // constant time comparison + if (!loginToken?.length || loginToken?.length !== user.loginToken?.length) { + throw new Error("Invalid login token"); + } + const compA = Buffer.from(loginToken); + const compB = Buffer.from(user.loginToken); + if (!crypto.timingSafeEqual(compA, compB)) { + throw new Error("Invalid login token"); + } + if (!user.loginTokenExpireAt || new Date() > user.loginTokenExpireAt) { + throw new Error("Login token expired"); + } + + user.loginToken = undefined; + user.loginTokenExpireAt = undefined; + await user.save(); + + return user; + } + + /** + * Check the login token, and if it's valid, then save the odoo token + * we received in the callback endpoint. + */ + async setOdooToken(odooToken: string) { + if (!odooToken?.length) { + throw new Error("Empty Odoo token"); + } + this.odooToken = odooToken; + await this.save(); + } + + private static async _getUserFromEmail(email: string): Promise { + const result = await pool.query( + ` + SELECT id, + email, + odoo_url, + odoo_token, + login_token, + login_token_expire_at, + translations, + translations_expire_at + FROM users_settings + WHERE email = $1 + `, + [email], + ); + if (result.rows.length === 0) { + return new User(null, email); + } + + const data = result.rows[0]; + return new User( + data.id, + email, + data.odoo_url, + data.odoo_token, + data.login_token, + data.login_token_expire_at, + data.translations, + data.translations_expire_at, + ); + } +} diff --git a/gmail/src/services/app_properties.ts b/gmail/src/services/app_properties.ts deleted file mode 100644 index b7c709b77..000000000 --- a/gmail/src/services/app_properties.ts +++ /dev/null @@ -1,6 +0,0 @@ -export function getOdooServerUrl() { - return PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL"); -} -export function setOdooServerUrl(url: string) { - PropertiesService.getUserProperties().setProperty("ODOO_SERVER_URL", url); -} diff --git a/gmail/src/services/log_email.ts b/gmail/src/services/log_email.ts index 370070843..68608bc34 100644 --- a/gmail/src/services/log_email.ts +++ b/gmail/src/services/log_email.ts @@ -1,20 +1,14 @@ -import { postJsonRpc } from "../utils/http"; -import { escapeHtml } from "../utils/html"; -import { URLS } from "../const"; +import { URLS } from "../consts"; import { Email } from "../models/email"; import { ErrorMessage } from "../models/error_message"; -import { _t } from "../services/translation"; -import { getAccessToken } from "./odoo_auth"; +import { User } from "../models/user"; +import { postJsonRpc } from "../utils/http"; /** * Format the email body before sending it to Odoo. * Add error message at the end of the email, fix some CSS issues,... */ -function _formatEmailBody(email: Email, error: ErrorMessage): string { - let body = email.body; - - body = `${_t("From:")} ${escapeHtml(email.contactEmail)}

${body}`; - +function _formatEmailBody(_t: Function, body: string, error: ErrorMessage): string { if (error.code === "attachments_size_exceeded") { body += `
${_t( "Attachments could not be logged in Odoo because their total size exceeded the allowed maximum.", @@ -27,29 +21,39 @@ function _formatEmailBody(email: Email, error: ErrorMessage): string { /class=\"gmail_chip gmail_drive_chip" style=\"/g, 'class="gmail_chip gmail_drive_chip" style=" min-height: 32px;', ); - - body += `

${_t("Logged from")} ${_t("Gmail Inbox")}`; - return body; } /** * Log the given email body in the chatter of the given record. */ -export function logEmail(recordId: number, recordModel: string, email: Email): ErrorMessage { - const odooAccessToken = getAccessToken(); - const [attachments, error] = email.getAttachments(); - const body = _formatEmailBody(email, error); - const url = PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + URLS.LOG_EMAIL; - - const response = postJsonRpc( - url, - { message: body, res_id: recordId, model: recordModel, attachments: attachments }, - { Authorization: "Bearer " + odooAccessToken }, +export async function logEmail( + _t: Function, + user: User, + recordId: number, + recordModel: string, + email: Email, +): Promise { + const [rawBody, timestamp, [attachments, error]] = await email.getBodyAndAttachments(); + const body = _formatEmailBody(_t, rawBody, error); + + const response = await postJsonRpc( + user.odooUrl + URLS.LOG_EMAIL, + { + body, + res_id: recordId, + model: recordModel, + attachments: attachments, + email_from: email.emailFrom, + subject: email.subject, + timestamp: timestamp, + application_name: _t("Odoo for Gmail"), + }, + { Authorization: "Bearer " + user.odooToken }, ); if (!response) { - error.setError("unknown"); + return new ErrorMessage("unknown"); } return error; diff --git a/gmail/src/services/odoo_auth.ts b/gmail/src/services/odoo_auth.ts index fd8bfda47..9ab0b54f3 100644 --- a/gmail/src/services/odoo_auth.ts +++ b/gmail/src/services/odoo_auth.ts @@ -1,30 +1,7 @@ -import { ODOO_AUTH_URLS } from "../const"; -import { postJsonRpc, encodeQueryData } from "../utils/http"; - -const errorPage = ` - - - -
__ERROR_MESSAGE__
-`; +import { HOST, ODOO_AUTH_URLS } from "../consts"; +import { User } from "../models/user"; +import { encodeQueryData, postJsonRpc } from "../utils/http"; +import { ERROR_PAGE, RAINBOW } from "./pages"; /** * Callback function called during the OAuth authentication process. @@ -33,132 +10,77 @@ const errorPage = ` * We generate a state token (for this function) * 2. The user is redirected to Odoo and enter his login / password * 3. Then the user is redirected to the Google App-Script - * 4. Thanks the the state token, the function "odooAuthCallback" is called with the auth code + * 4. Thanks the state token, the function "odooAuthCallback" is called with the auth code * 5. The auth code is exchanged for an access token with a RPC call */ -function odooAuthCallback(callbackRequest: any) { - Logger.log("Run authcallback"); - const success = callbackRequest.parameter.success; - const authCode = callbackRequest.parameter.auth_code; - +export async function odooAuthCallback(callbackRequest: any) { + const { success, auth_code: authCode, state } = callbackRequest.query; if (success !== "1") { - return HtmlService.createHtmlOutput( - errorPage.replace("__ERROR_MESSAGE__", "Odoo did not return successfully."), - ); + return ERROR_PAGE.replace("__ERROR_MESSAGE__", "Odoo did not return successfully."); } - - Logger.log("Get access token from auth code..."); - - const userProperties = PropertiesService.getUserProperties(); - const odooUrl = userProperties.getProperty("ODOO_SERVER_URL"); - - const response = postJsonRpc(odooUrl + ODOO_AUTH_URLS.CODE_VALIDATION, { - auth_code: authCode, - }); - - if (!response || !response.access_token || !response.access_token.length) { - return HtmlService.createHtmlOutput( - errorPage.replace( - "__ERROR_MESSAGE__", - "The token exchange failed. Maybe your token has expired or your database can not be reached by the Google server." + - "
Contact your administrator or our support.", - ), + const { email, loginToken } = JSON.parse(state); + let response = null; + let user = null; + try { + user = await User.getUserFromLoginToken(email, loginToken); + + console.log("Get access token from auth code..."); + response = await postJsonRpc(user.odooUrl + ODOO_AUTH_URLS.CODE_VALIDATION, { + auth_code: authCode, + }); + if (!response || !response.access_token || !response.access_token.length) { + throw new Error("Odoo exchange failed"); + } + } catch { + return ERROR_PAGE.replace( + "__ERROR_MESSAGE__", + "The token exchange failed. Maybe your token has expired or your database can not be reached by the Google server." + + "
Contact your administrator or our support.", ); } - - const accessToken = response.access_token; - - userProperties.setProperty("ODOO_ACCESS_TOKEN", accessToken); - - return HtmlService.createHtmlOutput("Success !"); + user.setOdooToken(response.access_token); + return RAINBOW; } /** * Generate the URL to redirect the user for the authentication to the Odoo database. * * This URL contains a state and the Odoo database should resend it. - * The Google server use the state code to know which function to execute when the user + * The Google server uses the state code to know which function to execute when the user * is redirected on their server. */ -export function getOdooAuthUrl() { - const userProperties = PropertiesService.getUserProperties(); - const odooUrl = userProperties.getProperty("ODOO_SERVER_URL"); - const scriptId = ScriptApp.getScriptId(); - +export async function getOdooAuthUrl(user: User): Promise { + const odooUrl = user.odooUrl; if (!odooUrl || !odooUrl.length) { throw new Error("Can not retrieve the Odoo database URL."); } - if (!scriptId || !scriptId.length) { - throw new Error("Can not retrieve the script ID."); - } - - const stateToken = ScriptApp.newStateToken().withMethod(odooAuthCallback.name).withTimeout(3600).createToken(); + const loginToken = await user.generateLoginToken(); - const redirectToAddon = `https://script.google.com/macros/d/${scriptId}/usercallback`; - const scope = ODOO_AUTH_URLS.SCOPE; + const redirectToAddon = `${HOST}/auth_callback`; - const url = + return ( odooUrl + ODOO_AUTH_URLS.AUTH_CODE + encodeQueryData({ redirect: redirectToAddon, friendlyname: "Gmail", - state: stateToken, - scope: scope, - }); - - return url; + state: JSON.stringify({ loginToken, email: user.email }), + scope: ODOO_AUTH_URLS.SCOPE, + }) + ); } /** - * Return the access token saved in the user properties. - */ -export const getAccessToken = () => { - const userProperties = PropertiesService.getUserProperties(); - const accessToken = userProperties.getProperty("ODOO_ACCESS_TOKEN"); - if (!accessToken || !accessToken.length) { - return; - } - return accessToken; -}; - -/** - * Reset the access token saved in the user properties. - */ -export const resetAccessToken = () => { - const userProperties = PropertiesService.getUserProperties(); - userProperties.deleteProperty("ODOO_ACCESS_TOKEN"); -}; - -/** - * Make an HTTP request to "/mail_plugin/auth/access_token" (cors="*") on the Odoo - * database to verify that the server is reachable and that the mail plugin module is - * installed. + * Make an HTTP request to the Odoo database to verify that the server + * is reachable and that the mail plugin module is installed. * - * Returns True if the Odoo database is reachable and if the "mail_plugin" module - * is installed, false otherwise. + * Returns the version of the addin that is supported if it's reachable, null otherwise. */ -export const isOdooDatabaseReachable = (odooUrl: string): boolean => { - if (!odooUrl || !odooUrl.length) { - return false; +export async function getSupportedAddinVersion(odooUrl: string): Promise { + if (!odooUrl?.length) { + return null; } - - const response = postJsonRpc( - odooUrl + ODOO_AUTH_URLS.CODE_VALIDATION, - { auth_code: null }, - {}, - { returnRawResponse: true }, - ); - if (!response) { - return false; - } - - const responseCode = response.getResponseCode(); - - if (responseCode > 299 || responseCode < 200) { - return false; - } - - return true; -}; + const response = await postJsonRpc(odooUrl + ODOO_AUTH_URLS.CHECK_VERSION); + return response ? parseInt(response) : null; +} diff --git a/gmail/src/services/odoo_redirection.ts b/gmail/src/services/odoo_redirection.ts new file mode 100644 index 000000000..1771fdfe0 --- /dev/null +++ b/gmail/src/services/odoo_redirection.ts @@ -0,0 +1,5 @@ +import { User } from "../models/user"; + +export function getOdooRecordURL(user: User, model: string, record_id: number) { + return user.odooUrl + `/mail_plugin/redirect_to_record/${model}/?record_id=${record_id}`; +} diff --git a/gmail/src/services/pages.ts b/gmail/src/services/pages.ts new file mode 100644 index 000000000..4b5bdc3ba --- /dev/null +++ b/gmail/src/services/pages.ts @@ -0,0 +1,271 @@ +export const RAINBOW = ` + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+ + + + + +
+
+
You're all set
You can now close this window and connect with Odoo!
+
+
+
+
+
+ +`; + +export const ERROR_PAGE = ` + + + +
__ERROR_MESSAGE__
+`; diff --git a/gmail/src/services/search_records.ts b/gmail/src/services/search_records.ts new file mode 100644 index 000000000..e01c86ab5 --- /dev/null +++ b/gmail/src/services/search_records.ts @@ -0,0 +1,25 @@ +import { URLS } from "../consts"; +import { ErrorMessage } from "../models/error_message"; +import { User } from "../models/user"; +import { postJsonRpc } from "../utils/http"; + +/** + * Search records of the given model. + */ +export async function searchRecords( + user: User, + recordModel: string, + query: string, +): Promise<[any[], number, ErrorMessage]> { + const response = await postJsonRpc( + user.odooUrl + URLS.SEARCH_RECORDS + "/" + recordModel, + { query }, + { Authorization: "Bearer " + user.odooToken }, + ); + + if (!response?.length) { + return [[], 0, new ErrorMessage("unknown", response.error)]; + } + + return [response[0], response[1], new ErrorMessage(null)]; +} diff --git a/gmail/src/services/translation.ts b/gmail/src/services/translation.ts index 663096eba..d8d411f36 100644 --- a/gmail/src/services/translation.ts +++ b/gmail/src/services/translation.ts @@ -1,10 +1,9 @@ +import { URLS } from "../consts"; +import { User } from "../models/user"; import { postJsonRpc } from "../utils/http"; -import { URLS } from "../const"; -import { getAccessToken } from "./odoo_auth"; -import { getOdooServerUrl } from "./app_properties"; /** - * Object which fetchs the translations on the Odoo database, puts them in cache. + * Object which fetch the translations on the Odoo database, puts them in cache. * * Done in a class and not in a simple function so we read only once the cache for all * translations. @@ -12,32 +11,31 @@ import { getOdooServerUrl } from "./app_properties"; export class Translate { translations: Record; - constructor() { - const cache = CacheService.getUserCache(); - const cacheKey = "ODOO_TRANSLATIONS"; - - const translationsStr = cache.get(cacheKey); + constructor(translations?: Record) { + this.translations = translations || {}; + } - const odooServerUrl = getOdooServerUrl(); - const odooAccessToken = getAccessToken(); - if (translationsStr) { - this.translations = JSON.parse(translationsStr); - } else if (odooServerUrl && odooAccessToken) { - Logger.log("Download translations..."); + static async getTranslations(user: User): Promise { + if (!user.odooUrl) { + // The user is not logged yet + const translator = new Translate({}); + return translator._t.bind(translator); + } - this.translations = postJsonRpc( - odooServerUrl + URLS.GET_TRANSLATIONS, + if (!user.translationsExpireAt || new Date() > user.translationsExpireAt) { + user.translations = await postJsonRpc( + user.odooUrl + URLS.GET_TRANSLATIONS, {}, - { Authorization: "Bearer " + odooAccessToken }, + { Authorization: "Bearer " + user.odooToken }, ); - - if (this.translations) { - // Put in the cacher for 6 hours (maximum cache life time) - cache.put(cacheKey, JSON.stringify(this.translations), 21600); - } + // Store the translation for 6 hours + const EXPIRATION_DURATION_MS = 6 * 60 * 60 * 1000; + user.translationsExpireAt = new Date(Date.now() + EXPIRATION_DURATION_MS); + await user.save(); + console.log("Translation fetched"); } - - this.translations = this.translations || {}; + const translator = new Translate(user.translations); + return translator._t.bind(translator); } /** @@ -47,11 +45,11 @@ export class Translate { * (e.g.: "Hello %(name)s") or simple string format (e.g.: "Hello %s"). */ _t(text: string, parameters: any = undefined): string { - let translated = this.translations[text]; + let translated = this.translations.hasOwnProperty(text) ? this.translations[text] : null; if (!translated) { if (this.translations && Object.keys(this.translations).length) { - Logger.log("Translation missing for: " + text); + console.log("Translation missing for: " + text); } translated = text; } @@ -69,21 +67,10 @@ export class Translate { .join("|"), "gi", ); - return translated.replace(re, (key) => parameters[key.substring(2, key.length - 2)] || ""); + return translated.replace( + re, + (key) => parameters[key.substring(2, key.length - 2)] || "", + ); } } } - -const translate = new Translate(); - -// Can be used as a function without reading each time the cache -export function _t(text: string, parameters: any = undefined): string { - return translate._t(text, parameters); -} - -export function clearTranslationCache() { - const cache = CacheService.getUserCache(); - const cacheKey = "ODOO_TRANSLATIONS"; - cache.remove(cacheKey); - translate.translations = {}; -} diff --git a/gmail/src/utils/actions.ts b/gmail/src/utils/actions.ts new file mode 100644 index 000000000..dbe30b075 --- /dev/null +++ b/gmail/src/utils/actions.ts @@ -0,0 +1,209 @@ +/** + * Build the JSON format to execute action (like updating a cart, showing a notification,...) + * + * https://developers.google.com/workspace/add-ons/guides/alternate-runtimes + */ +import jwt from "jsonwebtoken"; +import { HOST } from "../consts"; +import { State } from "../models/state"; +import { User } from "../models/user"; +import { Card } from "./components"; + +/** + * Class used to respond to an event + * (like pushing a card, showing a notification, redirecting to an url...). + */ +export abstract class EventResponse { + abstract build(); +} + +export class Notify extends EventResponse { + message: string; + + constructor(message: string) { + super(); + this.message = message; + } + + build() { + return { renderActions: { action: { notification: { text: this.message } } } }; + } +} + +export class PushCard extends EventResponse { + card: Card; + + constructor(card: Card) { + super(); + this.card = card; + } + + build() { + return { action: { navigations: [{ pushCard: this.card.build() }] } }; + } +} +export class PushToRoot extends EventResponse { + card: Card; + + constructor(card: Card) { + super(); + this.card = card; + } + + build() { + return { action: { navigations: [{ popToRoot: true }, { pushCard: this.card.build() }] } }; + } +} + +export class PopCard extends EventResponse { + build() { + return { action: { navigations: [{ pop: true }] } }; + } +} + +export class PopOneCardAndUpdate extends EventResponse { + card: Card; + + constructor(card: Card) { + super(); + this.card = card; + } + + build() { + return { action: { navigations: [{ pop: true }, { updateCard: this.card.build() }] } }; + } +} + +export class UpdateCard extends EventResponse { + card: Card; + + constructor(card: Card) { + super(); + this.card = card; + } + + build() { + return { action: { navigations: [{ updateCard: this.card.build() }] } }; + } +} + +export class Redirect extends EventResponse { + openLink: OpenLink; + constructor(openLink: OpenLink) { + super(); + this.openLink = openLink; + } + + build() { + return { action: { link: this.openLink.build()["openLink"] } }; + } +} + +/** + * Create an action which will call the given function and pass the state in arguments. + */ +export class ActionCall { + state?: State; + funct: any; + parameters: any; + + constructor(state: State, funct: Function, parameters: any = {}) { + if (!eventHandlers[funct.name]) { + throw new Error(`Event handler not configured: ${funct.name}`); + } + this.state = state; + this.funct = funct; + this.parameters = parameters; + } + + build() { + const payload = { + state: this.state, + arguments: this.parameters, + functionName: this.funct.name, + }; + const token = jwt.sign(payload, process.env.APP_SECRET, { + algorithm: "HS256", + expiresIn: "48h", + }); + + return { + action: { + function: HOST + "/execute_action", + parameters: [ + { + key: "token", + value: token, + }, + ], + }, + }; + } +} + +/** + * Define how the event handlers should be declared. + */ +type EventHandler = ( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +) => EventResponse | Promise; + +const eventHandlers = {}; +/** + * Register the function to be able to call it from its name. + */ +export function registerEventHandler(funct: EventHandler) { + if (!/^on[A-Z][a-zA-Z0-9]+$/.test(funct.name) || eventHandlers.hasOwnProperty(funct.name)) { + throw new Error(`Invalid function name: ${funct.name}`); + } + eventHandlers[funct.name] = funct; +} + +/** + * Get the event handler by the name of the function + * (everything is serialized when the addin call the event). + */ +export function getEventHandler(functionName: string): EventHandler { + if ( + !/^on[A-Z][a-zA-Z0-9]+$/.test(functionName) || + !eventHandlers.hasOwnProperty(functionName) + ) { + throw new Error(`Invalid function name: ${functionName}`); + } + return eventHandlers[functionName]; +} + +export enum OpenLinkOpenAs { + FULL_SIZE = "FULL_SIZE", + OVERLAY = "OVERLAY", +} + +export class OpenLink { + url: string; + openAs: OpenLinkOpenAs; + reloadOnClose: boolean; + + constructor( + url: string, + openAs: OpenLinkOpenAs = OpenLinkOpenAs.FULL_SIZE, + reloadOnClose: boolean = false, + ) { + this.url = url; + this.openAs = openAs; + this.reloadOnClose = reloadOnClose; + } + + build() { + return { + openLink: { + url: this.url, + openAs: this.openAs, + onClose: this.reloadOnClose ? "RELOAD" : "NOTHING", + }, + }; + } +} diff --git a/gmail/src/utils/components.ts b/gmail/src/utils/components.ts new file mode 100644 index 000000000..59c4c73f1 --- /dev/null +++ b/gmail/src/utils/components.ts @@ -0,0 +1,354 @@ +import { HOST } from "../consts"; +import { ActionCall, OpenLink } from "./actions"; + +/** + * Build the JSON format of the components to construct the view. + * + * https://addons.gsuite.google.com/uikit/builder + * https://gw-card-builder.web.app/ + * + * https://developers.google.com/workspace/add-ons/guides/alternate-runtimes + */ +abstract class Component { + abstract build(); +} + +export class CardSection { + header: string; + widgets: Component[]; + + constructor(widgets?: Component[]) { + this.header = ""; + this.widgets = widgets || []; + } + + setHeader(header: string) { + this.header = header; + } + + addWidget(widget: Component) { + this.widgets.push(widget); + } + + build() { + const ret = { + widgets: this.widgets.map((w) => w.build()), + }; + if (this.header?.length) { + ret["header"] = this.header; + } + return ret; + } +} + +export class Card { + sections: CardSection[]; + actions: [string, ActionCall][]; // actions shown in the kebab menu + + constructor(sections?: CardSection[]) { + this.sections = sections || []; + this.actions = []; + } + + addSection(section: CardSection) { + this.sections.push(section); + } + + addAction(label: string, action: ActionCall) { + this.actions.push([label, action]); + } + + build() { + const ret = { + sections: this.sections.map((s) => s.build()), + }; + if (this.actions.length) { + ret["cardActions"] = this.actions.map(([label, action]) => ({ + actionLabel: label, + onClick: action.build(), + })); + } + return ret; + } +} + +export class TextParagraph extends Component { + text: string; + constructor(text: string) { + super(); + this.text = text; + } + build() { + return { + textParagraph: { + text: this.text, + }, + }; + } +} + +export class Button extends Component { + text: string; + onClick: ActionCall | OpenLink; + disabled: boolean; + icon?: string; + iconLabel?: string; + iconCropStyle: ImageCropType; + color?: string; + borderless: boolean; + + constructor( + text: string, + onClick: ActionCall | OpenLink, + color?: string, + disabled: boolean = false, + icon?: string, + iconLabel?: string, + iconCropStyle: ImageCropType = ImageCropType.CIRCLE, + borderless: boolean = false, + ) { + super(); + if (icon?.length && icon.startsWith("/")) { + // Relative URL + icon = `${HOST}${icon}`; + } + this.text = text; + this.onClick = onClick; + this.disabled = disabled; + this.icon = icon; + this.iconLabel = iconLabel; + this.iconCropStyle = iconCropStyle; + this.color = color; + this.borderless = borderless; + } + + build() { + const buttonValues = { + text: this.text, + onClick: this.onClick.build(), + disabled: this.disabled, + }; + if (this.icon) { + buttonValues["icon"] = { + iconUrl: this.icon, + altText: this.iconLabel, + imageType: this.iconCropStyle, + }; + } + if (this.color) { + // Gmail expect the color converted to RGB, each value + // is between 0 and 1 + buttonValues["color"] = { + red: parseInt(this.color.slice(1, 3), 16) / 256, + green: parseInt(this.color.slice(3, 5), 16) / 256, + blue: parseInt(this.color.slice(5, 7), 16) / 256, + }; + } + if (this.borderless) { + buttonValues["color"] = { + red: 1, + green: 1, + blue: 1, + alpha: 1, + }; + } + return { buttonList: { buttons: [buttonValues] } }; + } +} + +export class LinkButton extends Component { + text: string; + onClick: ActionCall | OpenLink; + + constructor(text: string, onClick: ActionCall | OpenLink) { + super(); + this.text = text; + this.onClick = onClick; + } + + build() { + return { + decoratedText: { + text: `${this.text}`, + onClick: this.onClick.build(), + }, + }; + } +} + +/** + * Helper user to build icon button. + */ +export class IconButton extends Button { + constructor( + onClick: ActionCall | OpenLink, + icon?: string, + iconLabel?: string, + iconCropStyle: ImageCropType = ImageCropType.CIRCLE, + ) { + super(undefined, onClick, undefined, false, icon, iconLabel, iconCropStyle); + } +} + +/** + * Show many buttons in the same line. + */ +export class ButtonsList extends Component { + buttons: Button[]; + + constructor(buttons: Button[] = []) { + super(); + this.buttons = buttons; + } + + addButton(button: Button) { + this.buttons.push(button); + } + + build() { + return { + buttonList: { buttons: this.buttons.map((b) => b.build().buttonList.buttons[0]) }, + }; + } +} + +export enum ImageCropType { + SQUARE = "SQUARE", + CIRCLE = "CIRCLE", +} + +export class DecoratedText extends Component { + label: string; + content: string; + icon?: string; + bottomLabel?: string; + button?: Button; + onClick?: ActionCall | OpenLink; + wrap: boolean; + iconLabel?: string; + iconCropStyle: ImageCropType; + + constructor( + label: string, + content: string, + icon: string = undefined, + bottomLabel: string = undefined, + button: Button = undefined, + onClick: ActionCall | OpenLink = undefined, + wrap: boolean = true, + iconLabel: string = undefined, + iconCropStyle: ImageCropType = ImageCropType.CIRCLE, + ) { + super(); + if (icon?.length && icon.startsWith("/")) { + // Relative URL + icon = `${HOST}${icon}`; + } + + this.label = label; + this.content = content; + this.bottomLabel = bottomLabel; + this.button = button; + this.onClick = onClick; + this.wrap = wrap; + this.icon = icon; + this.iconLabel = iconLabel; + this.iconCropStyle = iconCropStyle; + } + build() { + const ret = { + decoratedText: { + text: this.content, + wrapText: this.wrap, + }, + }; + if (this.button) { + ret.decoratedText["button"] = this.button.build().buttonList.buttons[0]; + } + if (this.icon) { + ret.decoratedText["icon"] = { + iconUrl: this.icon, + altText: this.iconLabel, + imageType: this.iconCropStyle, + }; + } + if (this.label) { + ret.decoratedText["topLabel"] = this.label; + } + if (this.bottomLabel) { + ret.decoratedText["bottomLabel"] = this.bottomLabel; + } + if (this.onClick) { + ret.decoratedText["onClick"] = this.onClick.build(); + } + return ret; + } +} + +export class Image extends Component { + url: string; + altText?: string; + onClick?: ActionCall | OpenLink; + + constructor(url: string, altText?: string, onClick?: ActionCall | OpenLink) { + super(); + if (url.startsWith("/")) { + // Relative URL + url = `${HOST}${url}`; + } + this.url = url; + this.altText = altText; + this.onClick = onClick; + } + build() { + const ret = { image: { imageUrl: this.url } }; + if (this.altText) { + ret.image["altText"] = this.altText; + } + if (this.onClick) { + ret.image["onClick"] = this.onClick.build(); + } + return ret; + } +} + +export class TextInput extends Component { + name: string; + label: string; + onChange?: ActionCall; + placeholder?: string; + value?: string; + + constructor( + name: string, + label: string, + onChange?: ActionCall, + placeholder?: string, + value?: string, + ) { + super(); + this.name = name; + this.label = label; + this.onChange = onChange; + this.placeholder = placeholder; + this.value = value; + } + build() { + const ret = { + textInput: { + name: this.name, + label: this.label, + }, + }; + if (this.onChange) { + ret.textInput["onChangeAction"] = this.onChange.build()["action"]; + } + if (this.placeholder) { + ret.textInput["hintText"] = this.placeholder; + } + if (this.value) { + ret.textInput["value"] = this.value; + } + return ret; + } +} diff --git a/gmail/src/utils/db.ts b/gmail/src/utils/db.ts new file mode 100644 index 000000000..e5e6d8a6f --- /dev/null +++ b/gmail/src/utils/db.ts @@ -0,0 +1,17 @@ +import { Pool } from "pg"; +import { PSQL_DB, PSQL_HOST, PSQL_PASS, PSQL_PORT, PSQL_USER } from "../consts"; + +const pool = new Pool({ + user: PSQL_USER, + password: PSQL_PASS, + database: PSQL_DB, + host: PSQL_HOST, + port: PSQL_PORT, +}); + +pool.on("error", (err, client) => { + console.error("Could not connect to the database", err); + process.exit(-1); +}); + +export default pool; diff --git a/gmail/src/utils/format.ts b/gmail/src/utils/format.ts index 23f20089b..1b617cdc5 100644 --- a/gmail/src/utils/format.ts +++ b/gmail/src/utils/format.ts @@ -13,42 +13,11 @@ export function formatUrl(url: string): string { return url.replace(/\/+$/, ""); } -/** - * Return the given string only if it's not null and not empty. - */ -export function isTrue(value: any): string { - if (value && value.length) { - return value; - } -} - -/** - * Return the first element of an array if the array is not null and not empty. - */ -export function first(value: any[]): any { - if (value && value.length) { - return value[0]; - } -} - -/** - * Repeat the given string "n" times. - */ -export function repeat(str: string, n: number) { - let result = ""; - while (n > 0) { - result += str; - n--; - } - return result; -} - -/** - * Truncate the given text to not exceed the given length. - */ -export function truncate(str: string, maxLength: number) { - if (str.length > maxLength) { - return str.substring(0, maxLength - 3) + "..."; - } - return str; +export function htmlEscape(content: string): string { + return content + .replace(/&/g, "&") + .replace(//g, ">") + .replace(/"/g, """) + .replace(/'/g, "'"); } diff --git a/gmail/src/utils/html.ts b/gmail/src/utils/html.ts deleted file mode 100644 index 22be56221..000000000 --- a/gmail/src/utils/html.ts +++ /dev/null @@ -1,9 +0,0 @@ -export function escapeHtml(unsafe: string): string { - unsafe = unsafe || ""; - return unsafe - .replace(/&/g, "&") - .replace(//g, ">") - .replace(/"/g, """) - .replace(/'/g, "'"); -} diff --git a/gmail/src/utils/http.ts b/gmail/src/utils/http.ts index 79a2a34da..186fccefa 100644 --- a/gmail/src/utils/http.ts +++ b/gmail/src/utils/http.ts @@ -1,9 +1,14 @@ -import { State } from "../models/state"; - /** * Make a JSON RPC call with the following parameters. */ -export function postJsonRpc(url: string, data = {}, headers = {}, options: any = {}) { +export async function postJsonRpc(url: string, data = {}, headers = {}) { + for (const key in data) { + // don't send null values + if (data[key] === undefined || data[key] === null) { + data[key] = false; + } + } + // Make a valid "Odoo RPC" call data = { id: 0, @@ -12,74 +17,31 @@ export function postJsonRpc(url: string, data = {}, headers = {}, options: any = params: data, }; - const httpOptions = { - method: "post" as GoogleAppsScript.URL_Fetch.HttpMethod, - contentType: "application/json", - payload: JSON.stringify(data), - headers: headers, - }; - try { - const response = UrlFetchApp.fetch(url, httpOptions); - - if (options.returnRawResponse) { - return response; + const response = await fetch(url, { + method: "POST", + headers: { + ...headers, + "Content-Type": "application/json", + }, + body: JSON.stringify(data), + }); + if (!response.ok) { + throw new Error(`HTTP Error: ${response.status}`); } - const responseCode = response.getResponseCode(); - - if (responseCode > 299 || responseCode < 200) { - return; - } - - const textResponse = response.getContentText("UTF-8"); - const dictResponse = JSON.parse(textResponse); - + const dictResponse = await response.json(); if (!dictResponse.result) { return; } return dictResponse.result; - } catch { + } catch (e) { + console.error(`HTTP Error: ${e}`); return; } } -/** - * Make a JSON RPC call with the following parameters. - * - * Try to first read the response from the cache, if not found, - * make the call and cache the response. - * - * The cache key is based on the URL and the JSON data - * - * Store the result for 6 hours by default (maximum cache duration) - * - * This cache may be needed to make to many HTTP call to an external service (e.g. IAP). - */ -export function postJsonRpcCached(url: string, data = {}, headers = {}, cacheTtl: number = 21600) { - const cache = CacheService.getUserCache(); - - // Max 250 characters, to hash the key to have a fixed length - const cacheKey = - "ODOO_HTTP_CACHE_" + - Utilities.base64Encode(Utilities.computeDigest(Utilities.DigestAlgorithm.SHA_256, JSON.stringify([url, data]))); - - const cachedResponse = cache.get(cacheKey); - - if (cachedResponse) { - return JSON.parse(cachedResponse); - } - - const response = postJsonRpc(url, data, headers); - - if (response) { - cache.put(cacheKey, JSON.stringify(response), cacheTtl); - } - - return response; -} - /** * Take a dictionary and return the URL encoded parameters */ diff --git a/gmail/src/utils/svg.ts b/gmail/src/utils/svg.ts new file mode 100644 index 000000000..397c7825d --- /dev/null +++ b/gmail/src/utils/svg.ts @@ -0,0 +1,33 @@ +import { Resvg } from "@resvg/resvg-js"; +import { Response } from "express"; +import path from "path"; + +/** + * On the 19 December 2025, Google doesn't cache SVG files + * (it doesn't even fetch the route if it ends with `.svg`) + * Because we need to translate the text in some images, + * we dynamically convert the SVG to PNG. + * + * In practice, this route is called once, and then Google caches the PNG. + */ +export function svgToPngResponse(svgContent: NonSharedBuffer, res: Response) { + const fontFiles = [ + path.join(__dirname, "../assets", "GoogleSans.ttf"), + // Manuscript font used by some images + path.join(__dirname, "../assets", "Caveat.ttf"), + ]; + + const resvg = new Resvg(svgContent, { + fitTo: { mode: "width", value: 320 }, + font: { + fontFiles: fontFiles, + loadSystemFonts: false, + // Font used by Gmail + defaultFontFamily: "Google Sans", + }, + }); + + res.set("Content-Type", "image/png"); + res.set("Cache-Control", "public, immutable, max-age=31536000"); + res.send(resvg.render().asPng()); +} diff --git a/gmail/src/views/card_actions.ts b/gmail/src/views/card_actions.ts index d171a5787..d8d462d36 100644 --- a/gmail/src/views/card_actions.ts +++ b/gmail/src/views/card_actions.ts @@ -1,43 +1,19 @@ -import { buildDebugView } from "./debug"; -import { buildView } from "../views/index"; import { State } from "../models/state"; -import { Partner } from "../models/partner"; -import { resetAccessToken } from "../services/odoo_auth"; -import { _t, clearTranslationCache } from "../services/translation"; -import { actionCall } from "./helpers"; -import { pushToRoot } from "./helpers"; +import { User } from "../models/user"; +import { ActionCall, EventResponse, PushToRoot, registerEventHandler } from "../utils/actions"; +import { Card } from "../utils/components"; +import { getLoginMainView } from "../views/login"; +import { onOpenDebugView } from "./debug"; -function onLogout(state: State) { - resetAccessToken(); - clearTranslationCache(); - - const [partner, odooUserCompanies, canCreatePartner, canCreateProject, error] = Partner.enrichPartner( - state.email.contactEmail, - state.email.contactName, - ); - const newState = new State( - partner, - canCreatePartner, - state.email, - odooUserCompanies, - null, - null, - canCreateProject, - error, - ); - return pushToRoot(buildView(newState)); +async function onLogout(state: State, _t: Function, user: User): Promise { + user.odooUrl = undefined; + user.odooToken = undefined; + await user.save(); + return new PushToRoot(await getLoginMainView(user)); } +registerEventHandler(onLogout); -export function buildCardActionsView(state: State, card: Card) { - const canContactOdooDatabase = state.error.canContactOdooDatabase && State.isLogged; - - if (State.isLogged) { - card.addCardAction( - CardService.newCardAction().setText(_t("Logout")).setOnClickAction(actionCall(state, onLogout.name)), - ); - } - - card.addCardAction( - CardService.newCardAction().setText(_t("Debug")).setOnClickAction(actionCall(state, buildDebugView.name)), - ); +export function buildCardActionsView(card: Card, _t: Function) { + card.addAction(_t("Log out"), new ActionCall(undefined, onLogout)); + card.addAction(_t("Debug"), new ActionCall(undefined, onOpenDebugView)); } diff --git a/gmail/src/views/company.ts b/gmail/src/views/company.ts deleted file mode 100644 index 9a13ef072..000000000 --- a/gmail/src/views/company.ts +++ /dev/null @@ -1,231 +0,0 @@ -import { buildView } from "./index"; -import { actionCall, createKeyValueWidget, notify, updateCard } from "./helpers"; -import { SOCIAL_MEDIA_ICONS, UI_ICONS } from "./icons"; -import { URLS } from "../const"; -import { getOdooServerUrl } from "src/services/app_properties"; -import { ErrorMessage } from "../models/error_message"; -import { State } from "../models/state"; -import { Company } from "../models/company"; -import { Partner } from "../models/partner"; -import { _t } from "../services/translation"; - -/** - * Update the application state with the new company created / enriched. - * IT could be necessary to also update the contact if the contact is the company itself. - */ -function _setContactCompany(state: State, company: Company, error: ErrorMessage) { - if (company) { - state.partner.company = company; - if (state.partner.id === company.id) { - // The contact is the same partner as the company - // update his information - state.partner.isCompany = true; - state.partner.image = company.image; - state.partner.phone = company.phone; - state.partner.mobile = company.mobile; - } - } - state.error = error; - return updateCard(buildView(state)); -} - -function onCreateCompany(state: State) { - const [company, error] = Partner.createCompany(state.partner.id); - return _setContactCompany(state, company, error); -} - -function onEnrichCompany(state: State) { - const [company, error] = Partner.enrichCompany(state.partner.company.id); - return _setContactCompany(state, company, error); -} - -function onUnfoldCompanyDescription(state: State) { - state.isCompanyDescriptionUnfolded = true; - return updateCard(buildView(state)); -} - -export function buildCompanyView(state: State, card: Card) { - if (state.partner.company) { - const odooServerUrl = getOdooServerUrl(); - const cids = state.odooCompaniesParameter; - const company = state.partner.company; - - const companySection = CardService.newCardSection().setHeader("" + _t("Company Insights") + ""); - - if (!state.partner.id || state.partner.id !== company.id) { - const companyContent = [company.email, company.phone] - .filter((x) => x) - .map((x) => `${x}`); - - companySection.addWidget( - createKeyValueWidget( - null, - company.name + "
" + companyContent.join("
"), - company.image || UI_ICONS.no_company, - null, - null, - company.id ? odooServerUrl + `/web#id=${company.id}&model=res.partner&view_type=form${cids}` : null, - false, - company.email, - CardService.ImageCropType.CIRCLE, - ), - ); - } - - _addSocialButtons(companySection, company); - - if (company.description) { - const MAX_DESCRIPTION_LENGTH = 70; - if (company.description.length < MAX_DESCRIPTION_LENGTH || state.isCompanyDescriptionUnfolded) { - companySection.addWidget(createKeyValueWidget(_t("Description"), company.description)); - } else { - companySection.addWidget( - createKeyValueWidget( - _t("Description"), - company.description.substring(0, MAX_DESCRIPTION_LENGTH) + - "..." + - "
" + - "" + - _t("Read more") + - "", - null, - null, - null, - actionCall(state, onUnfoldCompanyDescription.name), - ), - ); - } - } - - if (company.address) { - companySection.addWidget( - createKeyValueWidget( - _t("Address"), - company.address, - UI_ICONS.location, - null, - null, - "https://www.google.com/maps/search/" + encodeURIComponent(company.address).replace("/", " "), - ), - ); - } - - if (company.phones) { - companySection.addWidget(createKeyValueWidget(_t("Phones"), company.phones, UI_ICONS.phone)); - } - - if (company.website) { - companySection.addWidget( - createKeyValueWidget(_t("Website"), company.website, UI_ICONS.website, null, null, company.website), - ); - } - - if (company.industry) { - companySection.addWidget(createKeyValueWidget(_t("Industry"), company.industry, UI_ICONS.industry)); - } - - if (company.employees) { - companySection.addWidget( - createKeyValueWidget(_t("Employees"), _t("%s employees", company.employees), UI_ICONS.people), - ); - } - - if (company.foundedYear) { - companySection.addWidget( - createKeyValueWidget(_t("Founded Year"), "" + company.foundedYear, UI_ICONS.foundation), - ); - } - - if (company.tags) { - companySection.addWidget(createKeyValueWidget(_t("Keywords"), company.tags, UI_ICONS.keywords)); - } - - if (company.companyType) { - companySection.addWidget( - createKeyValueWidget(_t("Company Type"), company.companyType, UI_ICONS.company_type), - ); - } - - if (company.annualRevenue) { - companySection.addWidget(createKeyValueWidget(_t("Annual Revenue"), company.annualRevenue, UI_ICONS.money)); - } - - card.addSection(companySection); - - if (!company.isEnriched) { - const enrichSection = CardService.newCardSection(); - enrichSection.addWidget(CardService.newTextParagraph().setText(_t("No insights for this company."))); - if (state.error.canCreateCompany && state.canCreatePartner) { - enrichSection.addWidget( - CardService.newTextButton() - .setText(_t("Enrich Company")) - .setOnClickAction(actionCall(state, onEnrichCompany.name)), - ); - } - card.addSection(enrichSection); - } - } else if (state.partner.id) { - const companySection = CardService.newCardSection().setHeader("" + _t("Company Insights") + ""); - companySection.addWidget(CardService.newTextParagraph().setText(_t("No company attached to this contact."))); - - if (state.error.canCreateCompany && state.canCreatePartner) { - companySection.addWidget( - CardService.newTextButton() - .setText(_t("Create a company")) - .setOnClickAction(actionCall(state, onCreateCompany.name)), - ); - } - card.addSection(companySection); - } -} - -function _addSocialButtons(section: CardSection, company: Company) { - const socialMediaButtons = CardService.newButtonSet(); - - const socialMedias = [ - { - name: "Facebook", - url: "https://facebook.com/", - icon: SOCIAL_MEDIA_ICONS.facebook, - key: "facebook", - }, - { - name: "Twitter", - url: "https://twitter.com/", - icon: SOCIAL_MEDIA_ICONS.twitter, - key: "twitter", - }, - { - name: "LinkedIn", - url: "https://linkedin.com/", - icon: SOCIAL_MEDIA_ICONS.linkedin, - key: "linkedin", - }, - { - name: "Github", - url: "https://github.com/", - icon: SOCIAL_MEDIA_ICONS.github, - key: "github", - }, - { - name: "Crunchbase", - url: "https://crunchbase.com/", - icon: SOCIAL_MEDIA_ICONS.crunchbase, - key: "crunchbase", - }, - ]; - - for (let media of socialMedias) { - const url = company[media.key]; - if (url && url.length) { - socialMediaButtons.addButton( - CardService.newImageButton() - .setAltText(media.name) - .setIconUrl(media.icon) - .setOpenLink(CardService.newOpenLink().setUrl(media.url + url)), - ); - } - } - - section.addWidget(socialMediaButtons); -} diff --git a/gmail/src/views/create_task.ts b/gmail/src/views/create_task.ts index af301468c..71635373c 100644 --- a/gmail/src/views/create_task.ts +++ b/gmail/src/views/create_task.ts @@ -1,158 +1,193 @@ -import { buildView } from "../views/index"; -import { updateCard, pushCard, pushToRoot } from "./helpers"; -import { UI_ICONS } from "./icons"; -import { createKeyValueWidget, actionCall, notify } from "./helpers"; -import { URLS } from "../const"; -import { getOdooServerUrl } from "src/services/app_properties"; -import { ErrorMessage } from "../models/error_message"; import { Project } from "../models/project"; import { State } from "../models/state"; import { Task } from "../models/task"; -import { logEmail } from "../services/log_email"; -import { _t } from "../services/translation"; - -function onSearchProjectClick(state: State, parameters: any, inputs: any) { - const inputQuery = inputs.search_project_query; - const query = (inputQuery && inputQuery.length && inputQuery[0]) || ""; - const [projects, error] = Project.searchProject(query); +import { User } from "../models/user"; +import { + ActionCall, + EventResponse, + Notify, + PopOneCardAndUpdate, + registerEventHandler, + UpdateCard, +} from "../utils/actions"; +import { + Button, + ButtonsList, + Card, + CardSection, + DecoratedText, + Image, + TextInput, + TextParagraph, +} from "../utils/components"; +import { getPartnerView } from "./partner"; + +async function onSearchProjectClick( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const query = formInputs.search_project_query || ""; + const [projects, error] = await Project.searchProject(user, query); + if (error.code) { + return new Notify(error.toString(_t)); + } - state.error = error; state.searchedProjects = projects; - - const createTaskView = buildCreateTaskView(state, query, true); - - // If go back, show again the "Create Project" section, but do not show all old searches - return parameters.hideCreateProjectSection ? updateCard(createTaskView) : pushCard(createTaskView); + return new UpdateCard(getCreateTaskView(state, _t, user, query)); } - -function onCreateProjectClick(state: State, parameters: any, inputs: any) { - const inputQuery = inputs.new_project_name; - const projectName = (inputQuery && inputQuery.length && inputQuery[0]) || ""; - - if (!projectName || !projectName.length) { - return notify(_t("The project name is required")); +registerEventHandler(onSearchProjectClick); + +function onCreateProjectViewClick( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): EventResponse { + return new UpdateCard(getCreateProjectView(state, _t)); +} +registerEventHandler(onCreateProjectViewClick); + +async function onCreateProjectClick( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const projectName = formInputs.new_project_name || ""; + if (!projectName.length) { + return new Notify(_t("The project name is required")); } - const project = Project.createProject(projectName); + const project = await Project.createProject(user, projectName); if (!project) { - return notify(_t("Could not create the project")); + return new Notify(_t("Could not create the project")); } - return onSelectProject(state, { project: project }); + return onSelectProject(state, _t, user, { project: project }, {}); } - -function onSelectProject(state: State, parameters: any) { - const project = Project.fromJson(parameters.project); - const task = Task.createTask(state.partner.id, project.id, state.email.body, state.email.subject); - - if (!task) { - return notify(_t("Could not create the task")); +registerEventHandler(onCreateProjectClick); + +async function onSelectProject( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const project = Project.fromJson(args.project); + const result = await Task.createTask(user, state.partner, project.id, state.email); + + if (!result) { + return new Notify(_t("Could not create the task")); } - task.projectName = project.name; + const [task, partner] = result; + state.partner = partner; state.partner.tasks.push(task); + state.partner.taskCount += 1; - const taskUrl = - PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + - `/web#id=${task.id}&action=project_mail_plugin.project_task_action_form_edit&model=project.task&view_type=form`; - - // Open the URL to the Odoo task and update the card - return CardService.newActionResponseBuilder() - .setOpenLink(CardService.newOpenLink().setUrl(taskUrl)) - .setNavigation(pushToRoot(buildView(state))) - .build(); + return new PopOneCardAndUpdate(getPartnerView(state, _t, user)); } - -export function buildCreateTaskView(state: State, query: string = "", hideCreateProjectSection: boolean = false) { - let noProject = false; - if (!state.searchedProjects) { - // Initiate the search - [state.searchedProjects, state.error] = Project.searchProject(""); - noProject = !state.searchedProjects.length; - } - - const odooServerUrl = getOdooServerUrl(); - const partner = state.partner; - const tasks = partner.tasks; +registerEventHandler(onSelectProject); + +export function getCreateTaskView( + state: State, + _t: Function, + user: User, + query: string = "", + noProject: boolean = false, +): Card { const projects = state.searchedProjects; - const card = CardService.newCardBuilder(); + const card = new Card(); if (!noProject) { - const projectSection = CardService.newCardSection().setHeader( - "" + _t("Create a Task in an existing Project") + "", - ); + const projectSection = new CardSection(); + projectSection.setHeader("" + _t("Create a Task in an existing Project") + ""); projectSection.addWidget( - CardService.newTextInput() - .setFieldName("search_project_query") - .setTitle(_t("Search a Project")) - .setValue(query || "") - .setOnChangeAction( - actionCall(state, onSearchProjectClick.name, { - hideCreateProjectSection: hideCreateProjectSection, - }), - ), + new TextInput( + "search_project_query", + _t("Search a Project"), + new ActionCall(state, onSearchProjectClick), + "", + query || "", + ), ); - projectSection.addWidget( - CardService.newTextButton() - .setText(_t("Search")) - .setOnClickAction( - actionCall(state, onSearchProjectClick.name, { - hideCreateProjectSection: hideCreateProjectSection, - }), - ), + const actionButtonSet = new ButtonsList(); + actionButtonSet.addButton( + new Button(_t("Search"), new ActionCall(state, onSearchProjectClick)), ); + if (state.canCreateProject) { + actionButtonSet.addButton( + new Button( + _t("Create Project"), + new ActionCall(state, onCreateProjectViewClick), + "#875a7b", + ), + ); + } + projectSection.addWidget(actionButtonSet); if (!projects.length) { - projectSection.addWidget(CardService.newTextParagraph().setText(_t("No project found."))); + projectSection.addWidget(new TextParagraph(_t("No project found."))); } for (let project of projects) { - const projectCard = createKeyValueWidget( - null, + const bottomLabel = [project.companyName, project.partnerName, project.stageName]; + const projectCard = new DecoratedText( + undefined, project.name, - null, - project.partnerName, - null, - actionCall(state, onSelectProject.name, { project: project }), + undefined, + bottomLabel.filter((l) => l).join(" - "), + undefined, + new ActionCall(state, onSelectProject, { project: project }), ); projectSection.addWidget(projectCard); } card.addSection(projectSection); - } + } else if (state.canCreateProject) { + return getCreateProjectView(state, _t); + } else { + const noProjectSection = new CardSection(); - if (!hideCreateProjectSection && state.canCreateProject) { - const createProjectSection = CardService.newCardSection().setHeader( - "" + _t("Create a Task in a new Project") + "", - ); - - createProjectSection.addWidget( - CardService.newTextInput().setFieldName("new_project_name").setTitle(_t("Project Name")).setValue(""), - ); + noProjectSection.addWidget(new Image("/assets/empty_folder.svg.png")); - createProjectSection.addWidget( - CardService.newTextButton() - .setText(_t("Create Project & Task")) - .setOnClickAction(actionCall(state, onCreateProjectClick.name)), - ); - card.addSection(createProjectSection); - } else if (noProject) { - const noProjectSection = CardService.newCardSection(); - - noProjectSection.addWidget(CardService.newImage().setImageUrl(UI_ICONS.empty_folder)); - - noProjectSection.addWidget(CardService.newTextParagraph().setText("" + _t("No project") + "")); + noProjectSection.addWidget(new TextParagraph("" + _t("No project") + "")); noProjectSection.addWidget( - CardService.newTextParagraph().setText( - _t("There are no project in your database. Please ask your project manager to create one."), + new TextParagraph( + _t( + "There are no project in your database. Please ask your project manager to create one.", + ), ), ); card.addSection(noProjectSection); } - return card.build(); + return card; +} + +/** + * Card used to create a new project (and to create the task in that project). + */ +export function getCreateProjectView(state: State, _t: Function): Card { + const createProjectSection = new CardSection(); + const card = new Card([createProjectSection]); + + createProjectSection.setHeader("" + _t("Create a Task in a new Project") + ""); + + createProjectSection.addWidget(new TextInput("new_project_name", _t("Project Name"))); + + createProjectSection.addWidget( + new Button(_t("Create Project & Task"), new ActionCall(state, onCreateProjectClick)), + ); + return card; } diff --git a/gmail/src/views/debug.ts b/gmail/src/views/debug.ts index 4576952e8..7a96fd84b 100644 --- a/gmail/src/views/debug.ts +++ b/gmail/src/views/debug.ts @@ -1,30 +1,36 @@ -import { createKeyValueWidget } from "./helpers"; -import { _t, clearTranslationCache } from "../services/translation"; -import { getAccessToken } from "src/services/odoo_auth"; -import { getOdooServerUrl } from "src/services/app_properties"; +import { State } from "../models/state"; +import { User } from "../models/user"; +import { + ActionCall, + EventResponse, + PopCard, + PushCard, + registerEventHandler, +} from "../utils/actions"; +import { Button, Card, CardSection, DecoratedText, TextParagraph } from "../utils/components"; -export function buildDebugView() { - const card = CardService.newCardBuilder(); - const odooServerUrl = getOdooServerUrl(); - const odooAccessToken = getAccessToken(); - - card.setHeader( - CardService.newCardHeader().setTitle(_t("Debug Zone")).setSubtitle(_t("Debug zone for development purpose.")), - ); - - card.addSection(CardService.newCardSection().addWidget(createKeyValueWidget(_t("Odoo Server URL"), odooServerUrl))); - - card.addSection( - CardService.newCardSection().addWidget(createKeyValueWidget(_t("Odoo Access Token"), odooAccessToken)), - ); - - card.addSection( - CardService.newCardSection().addWidget( - CardService.newTextButton() - .setText(_t("Clear Translations Cache")) - .setOnClickAction(CardService.newAction().setFunctionName(clearTranslationCache.name)), - ), - ); +export async function onClearTranslationCache( + state: State, + _t: Function, + user: User, +): Promise { + user.translations = undefined; + user.translationsExpireAt = undefined; + await user.save(); + return new PopCard(); +} +registerEventHandler(onClearTranslationCache); - return card.build(); +export function onOpenDebugView(state: State, _t: Function, user: User): EventResponse { + const section = new CardSection([ + new TextParagraph(_t("Debug zone for development purpose.")), + new DecoratedText(_t("Odoo Server URL"), user.odooUrl), + new DecoratedText(_t("Odoo Access Token"), user.odooToken), + new DecoratedText(_t("Odoo Access Token"), user.odooToken), + new Button(_t("Clear Translations Cache"), new ActionCall(state, onClearTranslationCache)), + ]); + const card = new Card([section]); + section.setHeader(_t("Debug Zone")); + return new PushCard(card); } +registerEventHandler(onOpenDebugView); diff --git a/gmail/src/views/error.ts b/gmail/src/views/error.ts deleted file mode 100644 index cc4af4baa..000000000 --- a/gmail/src/views/error.ts +++ /dev/null @@ -1,63 +0,0 @@ -import { State } from "../models/state"; -import { createKeyValueWidget, actionCall } from "./helpers"; -import { buildView } from "./index"; -import { updateCard } from "./helpers"; -import { UI_ICONS } from "./icons"; -import { _t } from "../services/translation"; -import { buildLoginMainView } from "./login"; - -function onCloseError(state: State) { - state.error.code = null; - return updateCard(buildView(state)); -} - -function _addError(message: string, state: State, icon: string = null): CardSection { - const errorSection = CardService.newCardSection(); - - errorSection.addWidget( - createKeyValueWidget( - null, - message, - icon, - null, - CardService.newImageButton() - .setAltText(_t("Close")) - .setIconUrl(UI_ICONS.close) - .setOnClickAction(actionCall(state, onCloseError.name)), - ), - ); - return errorSection; -} - -export function buildErrorView(state: State, card: Card) { - const error = state.error; - - const ignoredErrors = ["company_created", "company_updated"]; - if (ignoredErrors.indexOf(error.code) >= 0) { - return; - } - - if (error.code === "http_error_odoo") { - const errorSection = _addError(error.message, state); - errorSection.addWidget( - CardService.newTextButton() - .setText(_t("Login")) - .setOnClickAction(CardService.newAction().setFunctionName(buildLoginMainView.name)), - ); - card.addSection(errorSection); - } else if (error.code === "insufficient_credit") { - const errorSection = _addError(error.message, state); - errorSection.addWidget( - CardService.newTextButton() - .setText(_t("Buy new credits")) - .setOpenLink(CardService.newOpenLink().setUrl(error.information)), - ); - card.addSection(errorSection); - } else if (error.code === "missing_data") { - card.addSection(_addError(error.message, state)); - } else { - let errors = [error.message, error.information].filter((x) => x); - const errorMessage = errors.join("\n"); - card.addSection(_addError(errorMessage, state)); - } -} diff --git a/gmail/src/views/helpers.ts b/gmail/src/views/helpers.ts deleted file mode 100644 index db715e840..000000000 --- a/gmail/src/views/helpers.ts +++ /dev/null @@ -1,116 +0,0 @@ -import { UI_ICONS } from "./icons"; -import { State } from "../models/state"; -import { escapeHtml } from "../utils/html"; - -/** - * Remove all cards and push the new one - */ -export function pushToRoot(card: Card) { - return CardService.newNavigation().popToRoot().updateCard(card); -} - -/** - * Remove the last card and push a new one. - */ -export function updateCard(card: Card) { - return CardService.newNavigation().updateCard(card); -} - -/** - * Push a new card on the stack. - */ -export function pushCard(card: Card) { - return CardService.newNavigation().pushCard(card); -} - -/** - * Build a widget "Key / Value / Icon" - * - * If the icon if not a valid URL, take the icon from: - * https://github.com/webdog/octicons-png - */ -export function createKeyValueWidget( - label: string, - content: string, - icon: string = null, - bottomLabel: string = null, - button: Button = null, - action: any = null, - wrap: boolean = true, - iconLabel: string = null, - iconCropStyle: GoogleAppsScript.Card_Service.ImageCropType = CardService.ImageCropType.SQUARE, -) { - const widget = CardService.newDecoratedText().setText(content).setWrapText(true); - if (label && label.length) { - widget.setTopLabel(escapeHtml(label)); - } - - if (bottomLabel) { - widget.setBottomLabel(bottomLabel); - } - - if (button) { - widget.setButton(button); - } - if (action) { - if (typeof action === "string") { - widget.setOpenLink(CardService.newOpenLink().setUrl(action)); - } else { - widget.setOnClickAction(action); - } - } - - if (icon && icon.length) { - const isIconUrl = - icon.indexOf("http://") === 0 || icon.indexOf("https://") === 0 || icon.indexOf("data:image/") === 0; - if (!isIconUrl) { - throw new Error("Invalid icon URL"); - } - - widget.setStartIcon( - CardService.newIconImage() - .setIconUrl(icon) - .setImageCropType(iconCropStyle) - .setAltText(escapeHtml(iconLabel || label)), - ); - } - - widget.setWrapText(wrap); - - return widget; -} - -function _handleActionCall(event) { - const functionName = event.parameters.functionName; - const state = State.fromJson(event.parameters.state); - const parameters = JSON.parse(event.parameters.parameters); - const inputs = event.formInputs; - return eval(functionName)(state, parameters, inputs); -} - -/** - * Create an action which will call the given function and pass the state in arguments. - * - * This is necessary because event handlers can call only function and all arguments - * must be strings. Therefor we serialized the state and other arguments to clean the code - * and to be able to access to it in the event handlers. - */ -export function actionCall(state: State, functionName: string, parameters: any = {}) { - return CardService.newAction() - .setFunctionName(_handleActionCall.name) - .setParameters({ - functionName: functionName, - state: state.toJson(), - parameters: JSON.stringify(parameters), - }); -} - -export function notify(message: string) { - return CardService.newActionResponseBuilder() - .setNotification(CardService.newNotification().setText(message)) - .build(); -} - -export function openUrl(url: string) { - return CardService.newActionResponseBuilder().setOpenLink(CardService.newOpenLink().setUrl(url)).build(); -} diff --git a/gmail/src/views/icons.ts b/gmail/src/views/icons.ts deleted file mode 100644 index ef95aecf1..000000000 --- a/gmail/src/views/icons.ts +++ /dev/null @@ -1,91 +0,0 @@ -// Icon come from https://www.iconfinder.com/ -// Store as PNG 64x64 - -export const SOCIAL_MEDIA_ICONS = { - facebook: - "", - twitter: - "", - github: - "", - linkedin: - "", - crunchbase: - "", -}; - -export const UI_ICONS = { - person: - "", - phone: - "", - home: - "", - people: - "", - project: - "", - work: - "", - money: - "", - interrogation: - "", - industry: - "", - twitter: - "", - timezone: - "", - keywords: - "", - company_type: - "", - email: - "", - odoo: - "", - foundation: - "", - location: - "", - search: - "", - website: - "", - no_result: - "", - save_in_odoo: - "", - open_in_odoo: - "", - email_in_odoo: - "", - email_logged: - "", - reload: - "", - close: - "", - check: - "", - no_company: - "", - empty_folder: - "", -}; - -export const IMAGES_LOGIN = { - main_image: - "", - email: - "", - crm: - "", - project: - "", - search: - "", - ticket: - "", -}; diff --git a/gmail/src/views/index.ts b/gmail/src/views/index.ts deleted file mode 100644 index 8e7f0825b..000000000 --- a/gmail/src/views/index.ts +++ /dev/null @@ -1,35 +0,0 @@ -import { buildPartnerView } from "./partner"; -import { buildErrorView } from "./error"; -import { buildCompanyView } from "./company"; -import { buildLoginMainView } from "./login"; -import { buildCardActionsView } from "./card_actions"; -import { State } from "../models/state"; -import { actionCall } from "./helpers"; -import { _t } from "../services/translation"; - -export function buildView(state: State) { - const card = CardService.newCardBuilder(); - - if (state.error.code) { - buildErrorView(state, card); - } - - buildPartnerView(state, card); - - buildCompanyView(state, card); - - buildCardActionsView(state, card); - - if (!State.isLogged) { - card.setFixedFooter( - CardService.newFixedFooter().setPrimaryButton( - CardService.newTextButton() - .setText(_t("Login")) - .setBackgroundColor("#00A09D") - .setOnClickAction(actionCall(state, buildLoginMainView.name)), - ), - ); - } - - return card.build(); -} diff --git a/gmail/src/views/leads.ts b/gmail/src/views/leads.ts index dea362508..0ea63c901 100644 --- a/gmail/src/views/leads.ts +++ b/gmail/src/views/leads.ts @@ -1,122 +1,170 @@ -import { buildView } from "../views/index"; -import { pushCard, updateCard, createKeyValueWidget, actionCall, notify, openUrl } from "./helpers"; -import { URLS } from "../const"; -import { getOdooServerUrl } from "src/services/app_properties"; -import { UI_ICONS } from "./icons"; -import { logEmail } from "../services/log_email"; -import { _t } from "../services/translation"; import { Lead } from "../models/lead"; import { State } from "../models/state"; +import { User } from "../models/user"; +import { logEmail } from "../services/log_email"; +import { getOdooRecordURL } from "../services/odoo_redirection"; +import { + ActionCall, + EventResponse, + Notify, + OpenLink, + PushCard, + registerEventHandler, + UpdateCard, +} from "../utils/actions"; +import { + Button, + Card, + CardSection, + DecoratedText, + IconButton, + LinkButton, +} from "../utils/components"; +import { getPartnerView } from "./partner"; +import { getSearchRecordView } from "./search_records"; -function onLogEmailOnLead(state: State, parameters: any) { - const leadId = parameters.leadId; +async function onLogEmailOnLead( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const leadId = args.leadId; - if (State.checkLoggingState(state.email.messageId, "leads", leadId)) { - state.error = logEmail(leadId, "crm.lead", state.email); - if (!state.error.code) { - State.setLoggingState(state.email.messageId, "leads", leadId); - } - return updateCard(buildView(state)); + const error = await logEmail(_t, user, leadId, "crm.lead", state.email); + if (error.code) { + return new Notify(error.toString(_t)); } - return notify(_t("Email already logged on the lead")); -} -function onEmailAlreradyLoggedOnLead(state: State) { - return notify(_t("Email already logged on the lead")); + await state.email.setLoggingState(user, "crm.lead", leadId); + return new UpdateCard(getPartnerView(state, _t, user)); } +registerEventHandler(onLogEmailOnLead); -function onCreateLead(state: State) { - const leadId = Lead.createLead(state.partner.id, state.email.body, state.email.subject); +function onEmailAlreradyLoggedOnLead( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): EventResponse { + return new Notify(_t("Email already logged on the opportunity")); +} +registerEventHandler(onEmailAlreradyLoggedOnLead); - if (!leadId) { - return notify(_t("Could not create the lead")); +async function onCreateLead( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const result = await Lead.createLead(user, state.partner, state.email); + if (!result) { + return new Notify(_t("Could not create the opportunity")); } - const cids = state.odooCompaniesParameter; - const leadUrl = - PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + - `/web#id=${leadId}&action=crm_mail_plugin.crm_lead_action_form_edit&model=crm.lead&view_type=form${cids}`; + const [lead, partner] = result; + state.partner = partner; + state.partner.leads.push(lead); + state.partner.leadCount += 1; + return new UpdateCard(getPartnerView(state, _t, user)); +} +registerEventHandler(onCreateLead); - return openUrl(leadUrl); +function onSearchLeadsClick( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): EventResponse { + return new PushCard( + getSearchRecordView( + state, + _t, + "crm.lead", + _t("Opportunities"), + _t("Log the email on the opportunity"), + _t("Email already logged on the opportunity"), + "revenuesDescription", + "", + true, + state.partner.leads, + ), + ); } +registerEventHandler(onSearchLeadsClick); -export function buildLeadsView(state: State, card: Card) { - const odooServerUrl = getOdooServerUrl(); +export function buildLeadsView(state: State, _t: Function, user: User, card: Card) { const partner = state.partner; - const leads = partner.leads; - - if (!leads) { + if (!partner.leads) { // CRM module is not installed // otherwise leads should be at least an empty array return; } - const loggingState = State.getLoggingState(state.email.messageId); + const leads = [...partner.leads].splice(0, 5); - const leadsSection = CardService.newCardSection().setHeader( - "" + _t("Opportunities (%s)", leads.length) + "", + const leadsSection = new CardSection(); + + const searchButton = new IconButton( + new ActionCall(state, onSearchLeadsClick), + "/assets/search.png", + _t("Search Opportunities"), ); - const cids = state.odooCompaniesParameter; - if (state.partner.id) { - leadsSection.addWidget( - CardService.newTextButton().setText(_t("Create")).setOnClickAction(actionCall(state, onCreateLead.name)), - ); + const title = partner.leadCount + ? _t("Opportunities (%s)", partner.leadCount) + : _t("Opportunities"); + const widget = new DecoratedText( + "", + "" + title + "", + undefined, + undefined, + searchButton, + ); + + leadsSection.addWidget(widget); - for (let lead of leads) { - let leadRevenuesDescription; - if (lead.recurringRevenue) { - leadRevenuesDescription = _t( - "%(expected_revenue)s + %(recurring_revenue)s %(recurring_plan)s at %(probability)s%", - { - expected_revenue: lead.expectedRevenue, - probability: lead.probability, - recurring_revenue: lead.recurringRevenue, - recurring_plan: lead.recurringPlan, - }, - ); - } else { - leadRevenuesDescription = _t("%(expected_revenue)s at %(probability)s%", { - expected_revenue: lead.expectedRevenue, - probability: lead.probability, - }); - } - - let leadButton = null; - if (loggingState["leads"].indexOf(lead.id) >= 0) { - leadButton = CardService.newImageButton() - .setAltText(_t("Email already logged on the lead")) - .setIconUrl(UI_ICONS.email_logged) - .setOnClickAction(actionCall(state, onEmailAlreradyLoggedOnLead.name)); - } else { - leadButton = CardService.newImageButton() - .setAltText(_t("Log the email on the lead")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction( - actionCall(state, onLogEmailOnLead.name, { - leadId: lead.id, - }), - ); - } - - leadsSection.addWidget( - createKeyValueWidget( - null, - lead.name, - null, - leadRevenuesDescription, - leadButton, - odooServerUrl + `/web#id=${lead.id}&model=crm.lead&view_type=form${cids}`, - ), + const createButton = new Button(_t("New"), new ActionCall(state, onCreateLead)); + leadsSection.addWidget(createButton); + + for (let lead of leads) { + let leadButton = null; + if (state.email.checkLoggingState("crm.lead", lead.id)) { + leadButton = new IconButton( + new ActionCall(state, onEmailAlreradyLoggedOnLead), + "/assets/email_logged.png", + _t("Email already logged on the opportunity"), + ); + } else { + leadButton = new IconButton( + new ActionCall(state, onLogEmailOnLead, { + leadId: lead.id, + }), + "/assets/email_in_odoo.png", + _t("Log the email on the opportunity"), ); } - } else if (state.canCreatePartner) { - leadsSection.addWidget(CardService.newTextParagraph().setText(_t("Save Contact to create new Opportunities."))); - } else { + + leadsSection.addWidget( + new DecoratedText( + "", + lead.name, + undefined, + lead.revenuesDescription, + leadButton, + new OpenLink(getOdooRecordURL(user, "crm.lead", lead.id)), + ), + ); + } + + if (leads.length < partner.leadCount) { leadsSection.addWidget( - CardService.newTextParagraph().setText(_t("You can only create opportunities for existing customers.")), + new LinkButton(_t("Show all"), new ActionCall(state, onSearchLeadsClick)), ); } card.addSection(leadsSection); - return card; } diff --git a/gmail/src/views/login.ts b/gmail/src/views/login.ts index 574432c08..fdcc40ca7 100644 --- a/gmail/src/views/login.ts +++ b/gmail/src/views/login.ts @@ -1,102 +1,89 @@ -import { formatUrl, repeat } from "../utils/format"; -import { notify, createKeyValueWidget } from "./helpers"; import { State } from "../models/state"; -import { IMAGES_LOGIN } from "./icons"; -import { isOdooDatabaseReachable } from "../services/odoo_auth"; -import { _t, clearTranslationCache } from "../services/translation"; -import { setOdooServerUrl } from "src/services/app_properties"; - -function onNextLogin(event) { - const validatedUrl = formatUrl(event.formInput.odooServerUrl); +import { User } from "../models/user"; +import { getOdooAuthUrl, getSupportedAddinVersion } from "../services/odoo_auth"; +import { + ActionCall, + EventResponse, + Notify, + OpenLink, + OpenLinkOpenAs, + Redirect, + registerEventHandler, +} from "../utils/actions"; +import { Card, CardSection, Image, TextInput } from "../utils/components"; +import { formatUrl } from "../utils/format"; +/** + * Initiate the authentication process, and redirect to the Odoo database. + */ +async function onNextLogin( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + let validatedUrl = formatUrl(formInputs.odooServerUrl); if (!validatedUrl) { - return notify("Invalid URL"); + return new Notify("Invalid URL"); } - if (!/^https:\/\/([^\/?]*\.)?odoo\.com(\/|$)/.test(validatedUrl)) { - return notify("The URL must be a subdomain of odoo.com"); + if (validatedUrl.endsWith("/odoo")) { + validatedUrl = validatedUrl.slice(0, -5); + } else if (validatedUrl.endsWith("/odoo/web")) { + validatedUrl = validatedUrl.slice(0, -9); + } else if (validatedUrl.endsWith("/web")) { + validatedUrl = validatedUrl.slice(0, -4); } - clearTranslationCache(); + user.odooUrl = formInputs.odooServerUrl; + await user.save(); - setOdooServerUrl(validatedUrl); + const version = await getSupportedAddinVersion(validatedUrl); + if (!version) { + return new Notify("Could not connect to your database."); + } - if (!isOdooDatabaseReachable(validatedUrl)) { - return notify( - "Could not connect to your database. Make sure the module is installed in Odoo (Settings > General Settings > Integrations > Mail Plugins)", + if (version !== 2) { + return new Notify( + "This addin version required Odoo 19.1 or a newer version, please install an older addin version.", ); } - - return CardService.newActionResponseBuilder() - .setOpenLink( - CardService.newOpenLink() - .setUrl(State.odooLoginUrl) - .setOpenAs(CardService.OpenAs.OVERLAY) - .setOnClose(CardService.OnClose.RELOAD), - ) - .build(); + const odooLoginUrl = await getOdooAuthUrl(user); + return new Redirect(new OpenLink(odooLoginUrl, OpenLinkOpenAs.OVERLAY, true)); } +registerEventHandler(onNextLogin); -export function buildLoginMainView() { - const card = CardService.newCardBuilder(); - - // Trick to make large centered button - const invisibleChar = "⠀"; - - const faqUrl = "https://www.odoo.com/documentation/master/applications/productivity/mail_plugins.html"; - - card.addSection( - CardService.newCardSection() - .addWidget( - CardService.newImage().setAltText("Connect to your Odoo database").setImageUrl(IMAGES_LOGIN.main_image), - ) - .addWidget( - CardService.newTextInput() - .setFieldName("odooServerUrl") - .setTitle("Database URL") - .setHint("e.g. company.odoo.com") - .setValue(PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") || ""), - ) - .addWidget( - CardService.newTextButton() - .setText(repeat(invisibleChar, 12) + "Login" + repeat(invisibleChar, 12)) - .setTextButtonStyle(CardService.TextButtonStyle.FILLED) - .setBackgroundColor("#00A09D") - .setOnClickAction(CardService.newAction().setFunctionName(onNextLogin.name)), - ) - .addWidget(CardService.newTextParagraph().setText(repeat(invisibleChar, 13) + "OR")) - .addWidget( - CardService.newTextButton() - .setText(repeat(invisibleChar, 11) + " Sign Up" + repeat(invisibleChar, 11)) - .setOpenLink( - CardService.newOpenLink().setUrl( - "https://www.odoo.com/trial?selected_app=mail_plugin:crm:helpdesk:project", - ), - ), - ) - .addWidget( - createKeyValueWidget(null, "Create leads from emails sent to your email address.", IMAGES_LOGIN.email), - ) - .addWidget( - createKeyValueWidget( - null, - "Create tickets from emails sent to your email address.", - IMAGES_LOGIN.ticket, +export async function getLoginMainView(user: User) { + return new Card([ + new CardSection([ + new Image("/assets/login_header.svg.png", "Connect to your Odoo database"), + new TextInput( + "odooServerUrl", + "Connect to...", + new ActionCall(undefined, onNextLogin), + "e.g. company.odoo.com", + user.odooUrl, + ), + new Image( + "/render_button/875a7b/ffffff/Login", + "Login", + new ActionCall(undefined, onNextLogin), + ), + new Image( + "/render_button/e7e9ed/1e1e1e/Sign%20Up", + "Sign Up", + new OpenLink( + "https://www.odoo.com/trial?selected_app=mail_plugin:crm:helpdesk:project", ), - ) - .addWidget(createKeyValueWidget(null, "Centralize Prospects' emails into CRM.", IMAGES_LOGIN.crm)) - .addWidget( - createKeyValueWidget( - null, - "Generate Tasks from emails sent to your email address in any Odoo project.", - IMAGES_LOGIN.project, + ), + new Image( + "/render_button/ffffff/2f9e44/FAQ", + "FAQ", + new OpenLink( + "https://www.odoo.com/documentation/master/applications/productivity/mail_plugins.html", ), - ) - .addWidget(createKeyValueWidget(null, "Search and store insights on your contacts.", IMAGES_LOGIN.search)) - .addWidget( - CardService.newTextParagraph().setText(repeat(invisibleChar, 13) + `FAQ`), ), - ); - - return card.build(); + ]), + ]); } diff --git a/gmail/src/views/partner.ts b/gmail/src/views/partner.ts index 3b0b3da71..7c60ccaaf 100644 --- a/gmail/src/views/partner.ts +++ b/gmail/src/views/partner.ts @@ -1,129 +1,81 @@ -import { buildView } from "./index"; +import { Partner } from "../models/partner"; +import { State } from "../models/state"; +import { User } from "../models/user"; +import { + ActionCall, + EventResponse, + Notify, + registerEventHandler, + UpdateCard, +} from "../utils/actions"; +import { Card, CardSection, DecoratedText } from "../utils/components"; +import { buildCardActionsView } from "./card_actions"; import { buildLeadsView } from "./leads"; +import { getPartnerActionButtons } from "./partner_actions"; import { buildTasksView } from "./tasks"; import { buildTicketsView } from "./tickets"; -import { buildPartnerActionView } from "./partner_actions"; -import { updateCard } from "./helpers"; -import { UI_ICONS } from "./icons"; -import { createKeyValueWidget, actionCall, notify } from "./helpers"; -import { URLS } from "../const"; -import { getOdooServerUrl } from "src/services/app_properties"; -import { State } from "../models/state"; -import { Partner } from "../models/partner"; -import { ErrorMessage } from "../models/error_message"; -import { logEmail } from "../services/log_email"; -import { _t } from "../services/translation"; -import { buildLoginMainView } from "./login"; - -function onLogEmail(state: State) { - const partnerId = state.partner.id; - if (!partnerId) { - throw new Error(_t("This contact does not exist in the Odoo database.")); - } - - if (State.checkLoggingState(state.email.messageId, "partners", partnerId)) { - state.error = logEmail(partnerId, "res.partner", state.email); - if (!state.error.code) { - State.setLoggingState(state.email.messageId, "partners", partnerId); - } - return updateCard(buildView(state)); - } - return notify(_t("Email already logged on the contact")); -} +export async function onReloadPartner( + state: State, + _t: Function, + user: User, +): Promise { + const values = await Partner.getPartner( + user, + state.partner.name, + state.partner.email, + state.partner.id, + ); -function onSavePartner(state: State) { - const partnerValues = { - name: state.partner.name, - email: state.partner.email, - company: state.partner.company && state.partner.company.id, - }; + [state.partner, state.canCreatePartner, state.canCreateProject] = values; - const partnerId = Partner.savePartner(partnerValues); - if (partnerId) { - state.partner.id = partnerId; - state.searchedPartners = null; - state.error = new ErrorMessage(); - return updateCard(buildView(state)); - } else { - return notify(_t("Can not save the contact")); + if (values[3].code) { + return new Notify(values[3].toString(_t)); } + return new UpdateCard(getPartnerView(state, _t, user)); } +registerEventHandler(onReloadPartner); -export function onEmailAlreadyLogged(state: State) { - return notify(_t("Email already logged on the contact")); -} +export function getPartnerView(state: State, _t: Function, user: User): Card { + const section = new CardSection(); + const card = new Card([section]); + buildCardActionsView(card, _t); + card.addAction(_t("Refresh"), new ActionCall(state, onReloadPartner)); -export function buildPartnerView(state: State, card: Card) { const partner = state.partner; - const odooServerUrl = getOdooServerUrl(); - const canContactOdooDatabase = state.error.canContactOdooDatabase && State.isLogged; - - const loggingState = State.getLoggingState(state.email.messageId); - const isEmailLogged = partner.id && loggingState["partners"].indexOf(partner.id) >= 0; - const partnerSection = CardService.newCardSection().setHeader("" + _t("Contact") + ""); - - let partnerButton = null; - if (canContactOdooDatabase && !partner.id) { - partnerButton = state.canCreatePartner - ? CardService.newImageButton() - .setAltText(_t("Save in Odoo")) - .setIconUrl(UI_ICONS.save_in_odoo) - .setOnClickAction(actionCall(state, onSavePartner.name)) - : null; - } else if (canContactOdooDatabase && !isEmailLogged) { - partnerButton = partner.isWriteable - ? CardService.newImageButton() - .setAltText(_t("Log email")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction(actionCall(state, onLogEmail.name)) - : null; - } else if (canContactOdooDatabase && isEmailLogged) { - partnerButton = CardService.newImageButton() - .setAltText(_t("Email already logged on the contact")) - .setIconUrl(UI_ICONS.email_logged) - .setOnClickAction(actionCall(state, onEmailAlreadyLogged.name)); - } else if (!State.isLogged) { - // button "Log the email" but it redirects to the login page - partnerButton = CardService.newImageButton() - .setAltText(_t("Log email")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction(actionCall(state, buildLoginMainView.name)); - } + section.setHeader("" + _t("Contact Details") + ""); - const partnerContent = [partner.email, partner.phone] + let partnerContent = [ + partner.companyName && `🏢 ${partner.companyName}`, + partner.email && `✉️ ${partner.email}`, + partner.phone && `📞 ${partner.phone}`, + ] .filter((x) => x) - .map((x) => `${x}`); - const cids = state.odooCompaniesParameter; + .map((x) => `${x}`) + .join("
"); + if (!partner.id) { + partnerContent = _t("New Person"); + } - const partnerCard = createKeyValueWidget( + const partnerCard = new DecoratedText( + null, + partner.name || partner.email || "", + partner.getImage(), + partnerContent.length ? partnerContent : null, null, - partner.name + "
" + partnerContent.join("
"), - partner.image || (partner.isCompany ? UI_ICONS.no_company : UI_ICONS.person), null, - partnerButton, - partner.id - ? odooServerUrl + `/web#id=${partner.id}&model=res.partner&view_type=form${cids}` - : canContactOdooDatabase - ? null - : actionCall(state, buildLoginMainView.name), false, partner.email, - CardService.ImageCropType.CIRCLE, ); - partnerSection.addWidget(partnerCard); + section.addWidget(partnerCard); - buildPartnerActionView(state, partnerSection); + section.addWidget(getPartnerActionButtons(state, _t, user)); - card.addSection(partnerSection); - - if (canContactOdooDatabase) { - buildLeadsView(state, card); - buildTicketsView(state, card); - buildTasksView(state, card); - } + buildLeadsView(state, _t, user, card); + buildTicketsView(state, _t, user, card); + buildTasksView(state, _t, user, card); return card; } diff --git a/gmail/src/views/partner_actions.ts b/gmail/src/views/partner_actions.ts index f284f4c1a..6144d422d 100644 --- a/gmail/src/views/partner_actions.ts +++ b/gmail/src/views/partner_actions.ts @@ -1,69 +1,106 @@ -import { buildView } from "../views/index"; -import { buildSearchPartnerView } from "./search_partner"; -import { UI_ICONS } from "./icons"; -import { State } from "../models/state"; import { Partner } from "../models/partner"; -import { actionCall } from "./helpers"; -import { updateCard } from "./helpers"; -import { _t } from "../services/translation"; -import { buildLoginMainView } from "./login"; +import { State } from "../models/state"; +import { User } from "../models/user"; +import { logEmail } from "../services/log_email"; +import { getOdooRecordURL } from "../services/odoo_redirection"; +import { + ActionCall, + EventResponse, + Notify, + OpenLink, + PushCard, + registerEventHandler, + UpdateCard, +} from "../utils/actions"; +import { Button, ButtonsList, IconButton } from "../utils/components"; +import { getPartnerView } from "./partner"; +import { getSearchPartnerView } from "./search_partner"; + +async function onLogEmail(state: State, _t: Function, user: User): Promise { + const partnerId = state.partner.id; -function onSearchPartner(state: State) { - if (!state.searchedPartners) { - const [partners, error] = Partner.searchPartner(state.partner.email); - state.searchedPartners = partners; + if (!partnerId) { + throw new Error(_t("This contact does not exist in the Odoo database.")); } - return buildSearchPartnerView(state, state.partner.email, true); + const error = await logEmail(_t, user, partnerId, "res.partner", state.email); + if (error.code) { + return new Notify(error.toString(_t)); + } + state.email.setLoggingState(user, "res.partner", partnerId); + return new UpdateCard(getPartnerView(state, _t, user)); } +registerEventHandler(onLogEmail); -function onReloadPartner(state: State) { - [ - state.partner, - state.odooUserCompanies, - state.canCreatePartner, - state.canCreateProject, - state.error, - ] = Partner.getPartner(state.partner.email, state.partner.name, state.partner.id); +async function onSavePartner(state: State, _t: Function, user: User): Promise { + const partner = await Partner.savePartner(user, state.partner); + if (partner) { + state.partner = partner; + state.partner.isWritable = true; + state.searchedPartners = null; + return new UpdateCard(getPartnerView(state, _t, user)); + } + return new Notify(_t("Can not save the contact")); +} +registerEventHandler(onSavePartner); - return updateCard(buildView(state)); +export function onEmailAlreadyLoggedContact(state: State, _t: Function, user: User): EventResponse { + return new Notify(_t("Email already logged on the contact")); } +registerEventHandler(onEmailAlreadyLoggedContact); -export function buildPartnerActionView(state: State, partnerSection: CardSection) { - const isLogged = State.isLogged; - const canContactOdooDatabase = state.error.canContactOdooDatabase && isLogged; +async function onSearchPartner(state: State, _t: Function, user: User): Promise { + state.searchedPartners = []; + return new PushCard(await getSearchPartnerView(state, _t, user, state.partner.email, true)); +} +registerEventHandler(onSearchPartner); - if (canContactOdooDatabase) { - const actionButtonSet = CardService.newButtonSet(); +export function getPartnerActionButtons(state: State, _t: Function, user: User): ButtonsList { + const actionButtonSet = new ButtonsList(); - if (state.partner.id) { - actionButtonSet.addButton( - CardService.newImageButton() - .setAltText(_t("Refresh")) - .setIconUrl(UI_ICONS.reload) - .setOnClickAction(actionCall(state, onReloadPartner.name)), - ); - } + const isEmailLogged = + state.partner.id && state.email.checkLoggingState("res.partner", state.partner.id); + if (!state.partner.id && state.canCreatePartner) { actionButtonSet.addButton( - CardService.newImageButton() - .setAltText(_t("Search contact")) - .setIconUrl(UI_ICONS.search) - .setOnClickAction(actionCall(state, onSearchPartner.name)), + new Button(_t("Add to Odoo"), new ActionCall(state, onSavePartner), "#875a7b"), ); - - partnerSection.addWidget(actionButtonSet); - } else if (!isLogged) { - // add button but it redirects to the login page - const actionButtonSet = CardService.newButtonSet(); - + } + if (state.partner.id) { actionButtonSet.addButton( - CardService.newImageButton() - .setAltText(_t("Search contact")) - .setIconUrl(UI_ICONS.search) - .setOnClickAction(actionCall(state, buildLoginMainView.name)), + new Button( + _t("View in Odoo"), + new OpenLink(getOdooRecordURL(user, "res.partner", state.partner.id)), + "#875a7b", + ), ); - - partnerSection.addWidget(actionButtonSet); } + if (state.partner.id && !isEmailLogged && state.partner.isWritable) { + actionButtonSet.addButton( + new IconButton( + new ActionCall(state, onLogEmail), + "/assets/email_in_odoo.png", + _t("Log email"), + ), + ); + } + if (state.partner.id && isEmailLogged) { + actionButtonSet.addButton( + new IconButton( + new ActionCall(state, onEmailAlreadyLoggedContact), + "/assets/email_logged.png", + _t("Email already logged on the contact"), + ), + ); + } + + actionButtonSet.addButton( + new IconButton( + new ActionCall(state, onSearchPartner), + "/assets/search.png", + _t("Search contact"), + ), + ); + + return actionButtonSet; } diff --git a/gmail/src/views/search_partner.ts b/gmail/src/views/search_partner.ts index 38aa65855..7a4db33e1 100644 --- a/gmail/src/views/search_partner.ts +++ b/gmail/src/views/search_partner.ts @@ -1,64 +1,118 @@ -import { logEmail } from "../services/log_email"; -import { _t } from "../services/translation"; -import { Partner } from "../models/partner"; import { ErrorMessage } from "../models/error_message"; -import { createKeyValueWidget, actionCall, pushCard, updateCard, notify } from "./helpers"; -import { buildView } from "./index"; +import { Partner } from "../models/partner"; import { State } from "../models/state"; -import { SOCIAL_MEDIA_ICONS, UI_ICONS } from "./icons"; -import { onEmailAlreadyLogged } from "./partner"; - -function onSearchPartnerClick(state: State, parameters: any, inputs: any) { - const inputQuery = inputs.search_partner_query; - const query = (inputQuery && inputQuery.length && inputQuery[0]) || ""; - const [partners, error] = query && query.length ? Partner.searchPartner(query) : [[], new ErrorMessage()]; +import { User } from "../models/user"; +import { logEmail } from "../services/log_email"; +import { + ActionCall, + EventResponse, + Notify, + PushCard, + registerEventHandler, + UpdateCard, +} from "../utils/actions"; +import { + Button, + Card, + CardSection, + DecoratedText, + IconButton, + Image, + TextInput, + TextParagraph, +} from "../utils/components"; +import { buildCardActionsView } from "./card_actions"; +import { getPartnerView } from "./partner"; +import { onEmailAlreadyLoggedContact } from "./partner_actions"; + +async function onSearchPartnerClick( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const query = formInputs.search_partner_query || ""; + const [partners, error] = + query && query.length ? await Partner.searchPartner(user, query) : [[], new ErrorMessage()]; + if (error.code) { + return new Notify(error.toString(_t)); + } state.searchedPartners = partners; - return updateCard(buildSearchPartnerView(state, query)); + const card = await getSearchPartnerView(state, _t, user, query); + return args.fixCard ? new PushCard(card) : new UpdateCard(card); } -function onLogEmailPartner(state: State, parameters: any) { - const partnerId = parameters.partnerId; +registerEventHandler(onSearchPartnerClick); + +async function onLogEmailPartner( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const partnerId = args.partnerId; if (!partnerId) { throw new Error(_t("This contact does not exist in the Odoo database.")); } - if (State.checkLoggingState(state.email.messageId, "partners", partnerId)) { - state.error = logEmail(partnerId, "res.partner", state.email); - if (!state.error.code) { - State.setLoggingState(state.email.messageId, "partners", partnerId); - } - return updateCard(buildSearchPartnerView(state, parameters.query)); + const error = await logEmail(_t, user, partnerId, "res.partner", state.email); + if (error.code) { + return new Notify(error.toString(_t)); } - return notify(_t("Email already logged on the contact")); + await state.email.setLoggingState(user, "res.partner", partnerId); + return new UpdateCard(await getSearchPartnerView(state, _t, user, args.query)); } - -function onOpenPartner(state: State, parameters: any) { - const partner = parameters.partner; - const [newPartner, odooUserCompanies, canCreatePartner, canCreateProject, error] = Partner.getPartner( - partner.email, +registerEventHandler(onLogEmailPartner); + +async function onOpenPartner( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const partner = Partner.fromJson(args.partner); + const [newPartner, canCreatePartner, canCreateProject, error] = await Partner.getPartner( + user, partner.name, + partner.email, partner.id, ); + if (error.code) { + return new Notify(error.toString(_t)); + } const newState = new State( newPartner, canCreatePartner, state.email, - odooUserCompanies, null, null, canCreateProject, - error, ); - return pushCard(buildView(newState)); + return new PushCard(getPartnerView(newState, _t, user)); } - -export function buildSearchPartnerView(state: State, query: string, initialSearch: boolean = false) { - const loggingState = State.getLoggingState(state.email.messageId); - - const card = CardService.newCardBuilder(); - let partners = (state.searchedPartners || []).filter((partner) => partner.id); +registerEventHandler(onOpenPartner); + +export async function getSearchPartnerView( + state: State, + _t: Function, + user: User, + query: string, + initialSearch: boolean = false, + header: string = "", + noLogIcon: boolean = false, + fixCard: boolean = false, +): Promise { + const searchSection = new CardSection(); + const card = new Card([searchSection]); + + buildCardActionsView(card, _t); + + let partners = state.searchedPartners || []; let searchValue = query; if (initialSearch && partners.length <= 1) { @@ -66,63 +120,66 @@ export function buildSearchPartnerView(state: State, query: string, initialSearc searchValue = ""; } - const searchSection = CardService.newCardSection(); - searchSection.addWidget( - CardService.newTextInput() - .setFieldName("search_partner_query") - .setTitle(_t("Search contact")) - .setValue(searchValue) - .setOnChangeAction(actionCall(state, onSearchPartnerClick.name)), + new TextInput( + "search_partner_query", + _t("Search contact"), + new ActionCall(state, onSearchPartnerClick, { fixCard }), + "", + searchValue, + ), ); searchSection.addWidget( - CardService.newTextButton() - .setText(_t("Search")) - .setOnClickAction(actionCall(state, onSearchPartnerClick.name)), + new Button(_t("Search"), new ActionCall(state, onSearchPartnerClick, { fixCard })), ); + if (header?.length) { + searchSection.addWidget(new TextParagraph(`${header}`)); + } + for (let partner of partners) { - const partnerCard = CardService.newDecoratedText() - .setText(partner.name) - .setWrapText(true) - .setOnClickAction(actionCall(state, onOpenPartner.name, { partner: partner })) - .setStartIcon( - CardService.newIconImage() - .setIconUrl(partner.image || (partner.isCompany ? UI_ICONS.no_company : UI_ICONS.person)) - .setImageCropType(CardService.ImageCropType.CIRCLE), - ); - - if (partner.isWriteable) { - partnerCard.setButton( - loggingState["partners"].indexOf(partner.id) < 0 - ? CardService.newImageButton() - .setAltText(_t("Log email")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction( - actionCall(state, onLogEmailPartner.name, { - partnerId: partner.id, - query: query, - }), - ) - : CardService.newImageButton() - .setAltText(_t("Email already logged on the contact")) - .setIconUrl(UI_ICONS.email_logged) - .setOnClickAction(actionCall(state, onEmailAlreadyLogged.name)), - ); - } + let button; + let bottomLabel; if (partner.email) { - partnerCard.setBottomLabel(partner.email); + bottomLabel = partner.id ? partner.email : _t("New Person"); + } + + if (partner.isWritable && !noLogIcon) { + button = !state.email.checkLoggingState("res.partner", partner.id) + ? new IconButton( + new ActionCall(state, onLogEmailPartner, { + partnerId: partner.id, + query: query, + }), + "/assets/email_in_odoo.png", + _t("Log email"), + ) + : new IconButton( + new ActionCall(state, onEmailAlreadyLoggedContact), + "/assets/email_logged.png", + _t("Email already logged on the contact"), + ); } + const partnerCard = new DecoratedText( + undefined, + partner.name, + partner.getImage(), + bottomLabel, + button, + new ActionCall(state, onOpenPartner, { partner }), + true, + ); searchSection.addWidget(partnerCard); } if ((!partners || !partners.length) && !initialSearch) { - searchSection.addWidget(CardService.newTextParagraph().setText(_t("No contact found."))); + const title = encodeURIComponent(_t("No record found.")); + const subTitle = encodeURIComponent(_t("Try using different keywords.")); + searchSection.addWidget(new Image(`/render_search_no_result/${title}/${subTitle}`)); } - card.addSection(searchSection); - return card.build(); + return card; } diff --git a/gmail/src/views/search_records.ts b/gmail/src/views/search_records.ts new file mode 100644 index 000000000..73f09c4ec --- /dev/null +++ b/gmail/src/views/search_records.ts @@ -0,0 +1,202 @@ +import { State } from "../models/state"; +import { User } from "../models/user"; +import { logEmail } from "../services/log_email"; +import { getOdooRecordURL } from "../services/odoo_redirection"; +import { searchRecords } from "../services/search_records"; +import { + ActionCall, + EventResponse, + Notify, + OpenLink, + Redirect, + registerEventHandler, + UpdateCard, +} from "../utils/actions"; +import { + Button, + Card, + CardSection, + DecoratedText, + IconButton, + Image, + TextInput, +} from "../utils/components"; + +async function onSearchRecordClick( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const model = args.model; + const modelDescription = args.modelDescription; + const fieldInfo = args.fieldInfo; + const query = formInputs.query || ""; + + const [records, totalCount, error] = await searchRecords(user, model, query); + if (error.code) { + return new Notify(error.toString(_t)); + } + return new UpdateCard( + getSearchRecordView( + state, + _t, + model, + modelDescription, + args.emailLogMessage, + args.emailAlreadyLoggedMessage, + fieldInfo, + query, + false, + records, + totalCount, + ), + ); +} +registerEventHandler(onSearchRecordClick); + +async function onLogEmailRecord( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const model = args.model; + const modelDescription = args.modelDescription; + const fieldInfo = args.fieldInfo; + const recordId = args.recordId; + const records = args.records; + const totalCount = args.totalCount; + + const error = await logEmail(_t, user, recordId, model, state.email); + if (error.code) { + return new Notify(error.toString(_t)); + } + state.email.setLoggingState(user, model, recordId); + return new UpdateCard( + getSearchRecordView( + state, + _t, + model, + modelDescription, + args.emailLogMessage, + args.emailAlreadyLoggedMessage, + fieldInfo, + args.query, + false, + records, + totalCount, + ), + ); +} +registerEventHandler(onLogEmailRecord); + +function onOpenRecord( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): EventResponse { + const model = args.model; + const recordId = args.recordId; + return new Redirect(new OpenLink(getOdooRecordURL(user, model, recordId))); +} +registerEventHandler(onOpenRecord); + +function onEmailAlreadyLoggedOnRecord( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): EventResponse { + return new Notify(args.emailAlreadyLoggedMessage); +} +registerEventHandler(onEmailAlreadyLoggedOnRecord); + +export function getSearchRecordView( + state: State, + _t: Function, + model: string, + modelDescription: string, + emailLogMessage: string, + emailAlreadyLoggedMessage: string, + fieldInfo: string = "", + query: string = "", + initialSearch: boolean = false, + records: any[] = [], + totalCount: number = 0, +): Card { + const searchSection = new CardSection(); + const card = new Card([searchSection]); + let searchValue = query; + + const baseArgs = { + model, + modelDescription, + fieldInfo, + records, + totalCount, + emailAlreadyLoggedMessage, + emailLogMessage, + }; + + searchSection.addWidget( + new TextInput( + "query", + _t("Search %s", modelDescription), + new ActionCall(state, onSearchRecordClick, baseArgs), + "", + searchValue, + ), + ); + + searchSection.addWidget( + new Button(_t("Search"), new ActionCall(state, onSearchRecordClick, baseArgs)), + ); + + for (let record of records) { + const bottomLabel = fieldInfo?.length && record[fieldInfo] ? record[fieldInfo] : undefined; + + const button = !state.email.checkLoggingState(model, record.id) + ? new IconButton( + new ActionCall(state, onLogEmailRecord, { + ...baseArgs, + recordId: record.id, + query, + }), + "/assets/email_in_odoo.png", + emailLogMessage, + ) + : new IconButton( + new ActionCall(state, onEmailAlreadyLoggedOnRecord, { + emailAlreadyLoggedMessage, + }), + "/assets/email_logged.png", + emailAlreadyLoggedMessage, + ); + + const recordCard = new DecoratedText( + "", + record.name, + undefined, + undefined, + button, + new ActionCall(state, onOpenRecord, { model, recordId: record.id }), + true, + ); + + searchSection.addWidget(recordCard); + } + + if ((!records || !records.length) && !initialSearch) { + const title = encodeURIComponent(_t("No record found.")); + const subTitle = encodeURIComponent(_t("Try using different keywords.")); + searchSection.addWidget(new Image(`/render_search_no_result/${title}/${subTitle}`)); + } + + return card; +} diff --git a/gmail/src/views/tasks.ts b/gmail/src/views/tasks.ts index 4acb78a86..89ff114a8 100644 --- a/gmail/src/views/tasks.ts +++ b/gmail/src/views/tasks.ts @@ -1,91 +1,168 @@ -import { buildView } from "../views/index"; -import { buildCreateTaskView } from "../views/create_task"; -import { updateCard } from "./helpers"; -import { UI_ICONS } from "./icons"; -import { createKeyValueWidget, actionCall, notify } from "./helpers"; -import { URLS } from "../const"; -import { getOdooServerUrl } from "src/services/app_properties"; +import { Project } from "../models/project"; import { State } from "../models/state"; +import { User } from "../models/user"; import { logEmail } from "../services/log_email"; -import { _t } from "../services/translation"; -import { truncate } from "../utils/format"; +import { getOdooRecordURL } from "../services/odoo_redirection"; +import { + ActionCall, + EventResponse, + Notify, + OpenLink, + PushCard, + registerEventHandler, + UpdateCard, +} from "../utils/actions"; +import { + Button, + Card, + CardSection, + DecoratedText, + IconButton, + LinkButton, +} from "../utils/components"; +import { getCreateTaskView } from "./create_task"; +import { getPartnerView } from "./partner"; +import { getSearchRecordView } from "./search_records"; -function onCreateTask(state: State) { - return buildCreateTaskView(state); +async function onCreateTask( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + let noProject = false; + if (!state.searchedProjects) { + // Initiate the search + const [searchedProjects, error] = await Project.searchProject(user, ""); + if (error.code) { + return new Notify(error.toString(_t)); + } + + state.searchedProjects = searchedProjects; + noProject = !state.searchedProjects.length; + } + return new PushCard(getCreateTaskView(state, _t, user, "", noProject)); } +registerEventHandler(onCreateTask); -function onLogEmailOnTask(state: State, parameters: any) { - const taskId = parameters.taskId; +function onSearchTasksClick( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): EventResponse { + return new PushCard( + getSearchRecordView( + state, + _t, + "project.task", + _t("Tasks"), + _t("Log the email on the task"), + _t("Email already logged on the task"), + "projectName", + "", + true, + state.partner.tasks, + ), + ); +} +registerEventHandler(onSearchTasksClick); - if (State.checkLoggingState(state.email.messageId, "tasks", taskId)) { - logEmail(taskId, "project.task", state.email); - if (!state.error.code) { - State.setLoggingState(state.email.messageId, "tasks", taskId); - } - return updateCard(buildView(state)); +async function onLogEmailOnTask( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const taskId = args.taskId; + + const error = await logEmail(_t, user, taskId, "project.task", state.email); + if (error.code) { + return new Notify(error.toString(_t)); } - return notify(_t("Email already logged on the task")); + state.email.setLoggingState(user, "project.task", taskId); + return new UpdateCard(getPartnerView(state, _t, user)); } +registerEventHandler(onLogEmailOnTask); -function onEmailAlreradyLoggedOnTask() { - return notify(_t("Email already logged on the task")); +function onEmailAlreadyLoggedOnTask( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): EventResponse { + return new Notify(_t("Email already logged on the task")); } +registerEventHandler(onEmailAlreadyLoggedOnTask); -export function buildTasksView(state: State, card: Card) { - const odooServerUrl = getOdooServerUrl(); +export function buildTasksView(state: State, _t: Function, user: User, card: Card) { const partner = state.partner; - const tasks = partner.tasks; - - if (!tasks) { + if (!partner.tasks) { return; } - const loggingState = State.getLoggingState(state.email.messageId); - const tasksSection = CardService.newCardSection().setHeader("" + _t("Tasks (%s)", tasks.length) + ""); - const cids = state.odooCompaniesParameter; + const tasks = [...partner.tasks].splice(0, 5); - if (state.partner.id) { - tasksSection.addWidget( - CardService.newTextButton().setText(_t("Create")).setOnClickAction(actionCall(state, onCreateTask.name)), - ); + const tasksSection = new CardSection(); - for (let task of tasks) { - let taskButton = null; - if (loggingState["tasks"].indexOf(task.id) >= 0) { - taskButton = CardService.newImageButton() - .setAltText(_t("Email already logged on the task")) - .setIconUrl(UI_ICONS.email_logged) - .setOnClickAction(actionCall(state, onEmailAlreradyLoggedOnTask.name)); - } else { - taskButton = CardService.newImageButton() - .setAltText(_t("Log the email on the task")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction( - actionCall(state, onLogEmailOnTask.name, { - taskId: task.id, - }), - ); - } - - tasksSection.addWidget( - createKeyValueWidget( - task.projectName, - truncate(task.name, 35), - null, - null, - taskButton, - odooServerUrl + `/web#id=${task.id}&model=project.task&view_type=form${cids}`, - ), + const searchButton = new IconButton( + new ActionCall(state, onSearchTasksClick), + "/assets/search.png", + _t("Search Tasks"), + ); + + const title = partner.taskCount ? _t("Tasks (%s)", partner.taskCount) : _t("Tasks"); + const widget = new DecoratedText( + "", + "" + title + "", + undefined, + undefined, + searchButton, + ); + tasksSection.addWidget(widget); + + const createButton = new Button(_t("New"), new ActionCall(state, onCreateTask)); + tasksSection.addWidget(createButton); + + for (let task of tasks) { + let taskButton = null; + if (state.email.checkLoggingState("project.task", task.id)) { + taskButton = new IconButton( + new ActionCall(state, onEmailAlreadyLoggedOnTask), + "/assets/email_logged.png", + _t("Email already logged on the task"), + ); + } else { + taskButton = new IconButton( + new ActionCall(state, onLogEmailOnTask, { + taskId: task.id, + }), + "/assets/email_in_odoo.png", + _t("Log the email on the task"), ); } - } else if (state.canCreatePartner) { - tasksSection.addWidget(CardService.newTextParagraph().setText(_t("Save the contact to create new tasks."))); - } else { + + tasksSection.addWidget( + new DecoratedText( + "", + task.name, + undefined, + task.projectName, + taskButton, + new OpenLink(getOdooRecordURL(user, "project.task", task.id)), + ), + ); + } + + if (tasks.length < partner.taskCount) { tasksSection.addWidget( - CardService.newTextParagraph().setText(_t("The Contact needs to exist to create Task.")), + new LinkButton(_t("Show all"), new ActionCall(state, onSearchTasksClick)), ); } card.addSection(tasksSection); - return card; } diff --git a/gmail/src/views/tickets.ts b/gmail/src/views/tickets.ts index 225cff886..8fea2d41d 100644 --- a/gmail/src/views/tickets.ts +++ b/gmail/src/views/tickets.ts @@ -1,104 +1,168 @@ -import { buildView } from "../views/index"; -import { updateCard } from "./helpers"; -import { UI_ICONS } from "./icons"; -import { createKeyValueWidget, actionCall, notify, openUrl } from "./helpers"; -import { URLS } from "../const"; -import { getOdooServerUrl } from "src/services/app_properties"; import { State } from "../models/state"; import { Ticket } from "../models/ticket"; +import { User } from "../models/user"; import { logEmail } from "../services/log_email"; -import { _t } from "../services/translation"; - -function onCreateTicket(state: State) { - const ticketId = Ticket.createTicket(state.partner.id, state.email.body, state.email.subject); - - if (!ticketId) { - return notify(_t("Could not create the ticket")); +import { getOdooRecordURL } from "../services/odoo_redirection"; +import { + ActionCall, + EventResponse, + Notify, + OpenLink, + PushCard, + registerEventHandler, + UpdateCard, +} from "../utils/actions"; +import { + Button, + Card, + CardSection, + DecoratedText, + IconButton, + LinkButton, +} from "../utils/components"; +import { getPartnerView } from "./partner"; +import { getSearchRecordView } from "./search_records"; + +async function onCreateTicket( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const result = await Ticket.createTicket(user, state.partner, state.email); + + if (!result) { + return new Notify(_t("Could not create the ticket")); } - const cids = state.odooCompaniesParameter; - - const ticketUrl = - PropertiesService.getUserProperties().getProperty("ODOO_SERVER_URL") + - `/web#id=${ticketId}&action=helpdesk_mail_plugin.helpdesk_ticket_action_form_edit&model=helpdesk.ticket&view_type=form${cids}`; - - return openUrl(ticketUrl); + const [ticket, partner] = result; + state.partner = partner; + state.partner.tickets.push(ticket); + state.partner.ticketCount += 1; + return new UpdateCard(getPartnerView(state, _t, user)); } - -function onLogEmailOnTicket(state: State, parameters: any) { - const ticketId = parameters.ticketId; - - if (State.checkLoggingState(state.email.messageId, "tickets", ticketId)) { - state.error = logEmail(ticketId, "helpdesk.ticket", state.email); - if (!state.error.code) { - State.setLoggingState(state.email.messageId, "tickets", ticketId); - } - return updateCard(buildView(state)); - } - return notify(_t("Email already logged on the ticket")); +registerEventHandler(onCreateTicket); + +function onSearchTicketsClick( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): EventResponse { + return new PushCard( + getSearchRecordView( + state, + _t, + "helpdesk.ticket", + _t("Tickets"), + _t("Log the email on the ticket"), + _t("Email already logged on the ticket"), + "", + "", + true, + state.partner.tickets, + ), + ); } +registerEventHandler(onSearchTicketsClick); + +async function onLogEmailOnTicket( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): Promise { + const ticketId = args.ticketId; + const error = await logEmail(_t, user, ticketId, "helpdesk.ticket", state.email); + if (error.code) { + return new Notify(error.toString(_t)); + } -function onEmailAlreradyLoggedOnTicket() { - return notify(_t("Email already logged on the ticket")); + state.email.setLoggingState(user, "helpdesk.ticket", ticketId); + return new UpdateCard(getPartnerView(state, _t, user)); } +registerEventHandler(onLogEmailOnTicket); + +function onEmailAlreadyLoggedOnTicket( + state: State, + _t: Function, + user: User, + args: Record, + formInputs: Record, +): EventResponse { + return new Notify(_t("Email already logged on the ticket")); +} +registerEventHandler(onEmailAlreadyLoggedOnTicket); -export function buildTicketsView(state: State, card: Card) { - const odooServerUrl = getOdooServerUrl(); +export function buildTicketsView(state: State, _t: Function, user: User, card: Card) { const partner = state.partner; - const tickets = partner.tickets; - - if (!tickets) { + if (!partner.tickets) { + // Helpdesk not installed + // (otherwise we would have an empty array) return; } - const loggingState = State.getLoggingState(state.email.messageId); - - const ticketsSection = CardService.newCardSection().setHeader("" + _t("Tickets (%s)", tickets.length) + ""); + const tickets = [...partner.tickets].splice(0, 5); + + const ticketsSection = new CardSection(); + + const searchButton = new IconButton( + new ActionCall(state, onSearchTicketsClick), + "/assets/search.png", + _t("Search Tickets"), + ); + + const title = partner.ticketCount ? _t("Tickets (%s)", partner.ticketCount) : _t("Tickets"); + const widget = new DecoratedText( + "", + "" + title + "", + undefined, + undefined, + searchButton, + ); + ticketsSection.addWidget(widget); + + const createButton = new Button(_t("New"), new ActionCall(state, onCreateTicket)); + ticketsSection.addWidget(createButton); + + for (let ticket of tickets) { + let ticketButton = null; + if (state.email.checkLoggingState("helpdesk.ticket", ticket.id)) { + ticketButton = new IconButton( + new ActionCall(state, onEmailAlreadyLoggedOnTicket), + "/assets/email_logged.png", + _t("Email already logged on the ticket"), + ); + } else { + ticketButton = new IconButton( + new ActionCall(state, onLogEmailOnTicket, { + ticketId: ticket.id, + }), + "/assets/email_in_odoo.png", + _t("Log the email on the ticket"), + ); + } - if (state.partner.id) { ticketsSection.addWidget( - CardService.newTextButton().setText(_t("Create")).setOnClickAction(actionCall(state, onCreateTicket.name)), + new DecoratedText( + "", + ticket.name, + undefined, + ticket.stageName, + ticketButton, + new OpenLink(getOdooRecordURL(user, "helpdesk.ticket", ticket.id)), + ), ); + } - const cids = state.odooCompaniesParameter; - - for (let ticket of tickets) { - let ticketButton = null; - if (loggingState["tickets"].indexOf(ticket.id) >= 0) { - ticketButton = CardService.newImageButton() - .setAltText(_t("Email already logged on the ticket")) - .setIconUrl(UI_ICONS.email_logged) - .setOnClickAction(actionCall(state, onEmailAlreradyLoggedOnTicket.name)); - } else { - ticketButton = CardService.newImageButton() - .setAltText(_t("Log the email on the ticket")) - .setIconUrl(UI_ICONS.email_in_odoo) - .setOnClickAction( - actionCall(state, "onLogEmailOnTicket", { - ticketId: ticket.id, - }), - ); - } - - ticketsSection.addWidget( - createKeyValueWidget( - null, - ticket.name, - null, - null, - ticketButton, - odooServerUrl + `/web#id=${ticket.id}&model=helpdesk.ticket&view_type=form${cids}`, - ), - ); - } - } else if (state.canCreatePartner) { - ticketsSection.addWidget(CardService.newTextParagraph().setText(_t("Save the contact to create new tickets."))); - } else { + if (tickets.length < partner.ticketCount) { ticketsSection.addWidget( - CardService.newTextParagraph().setText(_t("The Contact needs to exist to create Ticket.")), + new LinkButton(_t("Show all"), new ActionCall(state, onSearchTicketsClick)), ); } card.addSection(ticketsSection); - return card; } diff --git a/gmail/tsconfig.json b/gmail/tsconfig.json index 63e4e22fc..624a162db 100644 --- a/gmail/tsconfig.json +++ b/gmail/tsconfig.json @@ -1,12 +1,14 @@ { "compilerOptions": { - "outDir": "./build", - "baseUrl": ".", - "strictNullChecks": false, - "noImplicitThis": true, - "noEmitOnError": true, - "target": "ES5", - "lib": ["dom", "es6", "scripthost", "es2017"] + "target": "ES2020", + "module": "commonjs", + "strict": false, + "noImplicitAny": false, + "esModuleInterop": true, + "allowSyntheticDefaultImports": true, + "skipLibCheck": true, + "noEmitOnError": false, + "outDir": "dist" }, - "include": ["src/*", "src/**/*"] + "include": ["src"] }