diff --git a/README.md b/README.md index b123a48..cd0ff62 100644 --- a/README.md +++ b/README.md @@ -9,23 +9,11 @@

-

- +

@localazy/api-client - npm - -license - - - -[//]: # '' -[//]: # ' schema' -[//]: # '
' -[//]: # ' build' -[//]: # ' coverage' - -

+license +
# 📦 Localazy API Client diff --git a/docs/api-client-reference.md b/docs/api-client-reference.md index 65e8b6f..d04e217 100644 --- a/docs/api-client-reference.md +++ b/docs/api-client-reference.md @@ -43,10 +43,9 @@ Create a new instance of ApiClient and pass in your project token. -| Arguments | Type | Description | -| ----------------- | --------------------------------------------------------------- | :-------------------- | -| options | [`ApiClientOptions`](../src/types/api-client-options.ts) | Api client options. | -| config `optional` | [`CreateAxiosDefaults`](https://axios-http.com/docs/req_config) | Axios request config. | +| Arguments | Type | Description | +| --------- | -------------------------------------------------------- | :------------------ | +| options | [`ApiClientOptions`](../src/types/api-client-options.ts) | Api client options. | | Returns | Type | | ----------- | :------------------------------------- | @@ -69,7 +68,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/projects#list-projects) | Arguments | Type | Description | | ----------------- | -------------------------------------------------------------- | :---------------------------- | | request | [`ProjectsListRequest`](../src/types/projects-list-request.ts) | Projects list request config. | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | Axios request config. | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | Request config. | | Returns | Type | | -------------------- | :----------------------------------- | @@ -93,7 +92,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/projects#list-projects) | Arguments | Type | Description | | ----------------- | -------------------------------------------------------------- | :---------------------------- | | request | [`ProjectsListRequest`](../src/types/projects-list-request.ts) | Projects list request config. | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | Axios request config. | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | Request config. | | Returns | Type | | ------------------ | :----------------------------------- | @@ -114,10 +113,10 @@ List all [files](../src/types/file.ts) in the project. See: [Localazy API Docs](https://localazy.com/docs/api/files#list-files-in-project) -| Arguments | Type | -| ----------------- | -------------------------------------------------------------- | -| request | [`FilesListRequest`](../src/types/files-list-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| Arguments | Type | +| ----------------- | -------------------------------------------------------- | +| request | [`FilesListRequest`](../src/types/files-list-request.ts) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | ----------------- | :----------------------------- | @@ -137,10 +136,10 @@ First [file](../src/types/file.ts) in the project. See: [Localazy API Docs](https://localazy.com/docs/api/files#list-files-in-project) -| Arguments | Type | -| ----------------- | -------------------------------------------------------------- | -| request | [`FilesListRequest`](../src/types/files-list-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| Arguments | Type | +| ----------------- | -------------------------------------------------------- | +| request | [`FilesListRequest`](../src/types/files-list-request.ts) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | --------------- | :----------------------------- | @@ -161,7 +160,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/files#retrieve-a-list-of- | Arguments | Type | | ----------------- | --------------------------------------------------------------- | | request | [`FileListKeysRequest`](../src/types/file-list-keys-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | ---------------- | :--------------------------- | @@ -186,7 +185,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/files#retrieve-a-list-of- | Arguments | Type | | ----------------- | --------------------------------------------------------------- | | request | [`FileListKeysRequest`](../src/types/file-list-keys-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | ------------------------ | :------------------------------------------------ | @@ -218,7 +217,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/files#list-file-content) | Arguments | Type | | ----------------- | -------------------------------------------------------------------- | | request | [`FileGetContentRequest`](../src/types/file-get-contents-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | --------------- | :-------------------------------------------------------------- | @@ -242,10 +241,10 @@ Update [key](../src/types/key.ts). See: [Localazy API Docs](https://localazy.com/docs/api/source-keys#update-source-key) -| Arguments | Type | -| ----------------- | -------------------------------------------------------------- | -| request | [`KeyUpdateRequest`](../src/types/key-update-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| Arguments | Type | +| ----------------- | -------------------------------------------------------- | +| request | [`KeyUpdateRequest`](../src/types/key-update-request.ts) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | | --------------- | @@ -268,10 +267,10 @@ Delete [key](../src/types/key.ts). See: [Localazy API Docs](https://localazy.com/docs/api/source-keys#delete-source-key) -| Arguments | Type | -| ----------------- | -------------------------------------------------------------- | -| request | [`KeyDeleteRequest`](../src/types/key-delete-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| Arguments | Type | +| ----------------- | -------------------------------------------------------- | +| request | [`KeyDeleteRequest`](../src/types/key-delete-request.ts) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | | --------------- | @@ -292,10 +291,10 @@ Import JSON object as source keys. See: [Localazy API Docs](https://localazy.com/docs/api/import#import-content-to-a-project) -| Arguments | Type | -| ----------------- | -------------------------------------------------------------- | -| request | [`ImportJsonRequest`](../src/types/import-json-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| Arguments | Type | +| ----------------- | ---------------------------------------------------------- | +| request | [`ImportJsonRequest`](../src/types/import-json-request.ts) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | --------------- | :----------------------------- | @@ -332,10 +331,10 @@ const file = await api.import.json({ Export translated keys as JSON object. -| Arguments | Type | -| ----------------- | -------------------------------------------------------------- | -| request | [`ExportJsonRequest`](../src/types/export-json-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| Arguments | Type | +| ----------------- | ---------------------------------------------------------- | +| request | [`ExportJsonRequest`](../src/types/export-json-request.ts) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | ------------------- | :-------------------------------------- | @@ -359,9 +358,9 @@ List all [formats](../src/types/format.ts) and related options. See: [Localazy API Docs](https://localazy.com/docs/api/import#list-available-file-types) -| Arguments | Type | -| ----------------- | -------------------------------------------------------------- | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| Arguments | Type | +| ----------------- | ------------------------------------------------- | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | ------------------- | :--------------------------------- | @@ -382,7 +381,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/screenshot-management#lis | Arguments | Type | | ----------------- | -------------------------------------------------------------------- | | request | [`ScreenshotsListRequest`](../src/types/screenshots-list-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | ----------------------- | :----------------------------------------- | @@ -403,7 +402,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/screenshot-management#lis | Arguments | Type | | ----------------- | ----------------------------------------------------------------------------- | | request | [`ScreenshotsListTagsRequest`](../src/types/screenshots-list-tags-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | -------------------------- | :------------------------------------------------ | @@ -424,7 +423,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/screenshot-management#cre | Arguments | Type | | ----------------- | ---------------------------------------------------------------------- | | request | [`ScreenshotCreateRequest`](../src/types/screenshot-create-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | ----------------- | :------------- | @@ -446,7 +445,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/screenshot-management#upd | Arguments | Type | | ----------------- | ------------------------------------------------------------------------------------------ | | request | [`ScreenshotUpdateImageDataRequest`](../src/types/screenshot-update-image-data-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | | --------------- | @@ -469,7 +468,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/screenshot-management#upd | Arguments | Type | | ----------------- | ---------------------------------------------------------------------- | | request | [`ScreenshotUpdateRequest`](../src/types/screenshot-update-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | | --------------- | @@ -493,7 +492,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/screenshot-management#del | Arguments | Type | | ----------------- | ---------------------------------------------------------------------- | | request | [`ScreenshotDeleteRequest`](../src/types/screenshot-delete-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | | --------------- | @@ -517,7 +516,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/glossary#list-all-glossar | Arguments | Type | | ----------------- | -------------------------------------------------------------- | | request | [`GlossaryListRequest`](../src/types/glossary-list-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | --------------------------- | :-------------------------------------------------- | @@ -538,7 +537,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/glossary#get-glossary-ter | Arguments | Type | | ----------------- | -------------------------------------------------------------- | | request | [`GlossaryFindRequest`](../src/types/glossary-find-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | ------------------------- | :-------------------------------------------------- | @@ -560,7 +559,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/glossary#create-new-gloss | Arguments | Type | | ----------------- | ------------------------------------------------------------------ | | request | [`GlossaryCreateRequest`](../src/types/glossary-create-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | ----------------- | :----------------- | @@ -587,7 +586,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/glossary#update-glossary- | Arguments | Type | | ----------------- | ------------------------------------------------------------------ | | request | [`GlossaryUpdateRequest`](../src/types/glossary-update-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | | --------------- | @@ -615,7 +614,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/glossary#delete-glossary- | Arguments | Type | | ----------------- | ------------------------------------------------------------------ | | request | [`GlossaryDeleteRequest`](../src/types/glossary-delete-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | | --------------- | @@ -639,7 +638,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/webhooks-api#list-webhook | Arguments | Type | | ----------------- | -------------------------------------------------------------- | | request | [`WebhooksListRequest`](../src/types/webhooks-list-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | -------------------- | :----------------------------------- | @@ -660,7 +659,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/webhooks-api#update-webho | Arguments | Type | | ----------------- | ------------------------------------------------------------------ | | request | [`WebhooksUpdateRequest`](../src/types/webhooks-update-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | | --------------- | @@ -692,7 +691,7 @@ See: [Localazy API Docs](https://localazy.com/docs/api/webhooks-api#webhook-secr | Arguments | Type | | ----------------- | ------------------------------------------------------------------------- | | request | [`WebhooksGetSecretRequest`](../src/types/webhooks-get-secret-request.ts) | -| config `optional` | [`AxiosRequestConfig`](https://axios-http.com/docs/req_config) | +| config `optional` | [`RequestConfig`](../src/types/request-config.ts) | | Returns | Type | | ------------------------- | :-------------------------------------------------- | diff --git a/package-lock.json b/package-lock.json index dda3cd8..cd0d03e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -10,7 +10,6 @@ "license": "MIT", "dependencies": { "@localazy/languages": "^0.1.6", - "axios": "^1.6.8", "lodash-es": "^4.17.21" }, "devDependencies": { @@ -22,7 +21,6 @@ "@typescript-eslint/parser": "^7.2.0", "@vitest/coverage-v8": "^1.4.0", "@vitest/ui": "^1.4.0", - "axios-mock-adapter": "^1.22.0", "eslint": "^8.57.0", "eslint-config-airbnb": "^19.0.4", "eslint-config-airbnb-base": "^15.0.0", @@ -2521,11 +2519,6 @@ "dev": true, "peer": true }, - "node_modules/asynckit": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", - "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" - }, "node_modules/available-typed-arrays": { "version": "1.0.7", "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", @@ -2551,29 +2544,6 @@ "node": ">=4" } }, - "node_modules/axios": { - "version": "1.6.8", - "resolved": "https://registry.npmjs.org/axios/-/axios-1.6.8.tgz", - "integrity": "sha512-v/ZHtJDU39mDpyBoFVkETcd/uNdxrWRrg3bKpOKzXFA6Bvqopts6ALSMU3y6ijYxbw2B+wPrIv46egTzJXCLGQ==", - "dependencies": { - "follow-redirects": "^1.15.6", - "form-data": "^4.0.0", - "proxy-from-env": "^1.1.0" - } - }, - "node_modules/axios-mock-adapter": { - "version": "1.22.0", - "resolved": "https://registry.npmjs.org/axios-mock-adapter/-/axios-mock-adapter-1.22.0.tgz", - "integrity": "sha512-dmI0KbkyAhntUR05YY96qg2H6gg0XMl2+qTW0xmYg6Up+BFBAJYRLROMXRdDEL06/Wqwa0TJThAYvFtSFdRCZw==", - "dev": true, - "dependencies": { - "fast-deep-equal": "^3.1.3", - "is-buffer": "^2.0.5" - }, - "peerDependencies": { - "axios": ">= 0.17.0" - } - }, "node_modules/axobject-query": { "version": "3.2.1", "resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-3.2.1.tgz", @@ -3075,17 +3045,6 @@ "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true }, - "node_modules/combined-stream": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", - "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", - "dependencies": { - "delayed-stream": "~1.0.0" - }, - "engines": { - "node": ">= 0.8" - } - }, "node_modules/commander": { "version": "10.0.1", "resolved": "https://registry.npmjs.org/commander/-/commander-10.0.1.tgz", @@ -3375,14 +3334,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/delayed-stream": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", - "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", - "engines": { - "node": ">=0.4.0" - } - }, "node_modules/dequal": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", @@ -4660,25 +4611,6 @@ "integrity": "sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==", "dev": true }, - "node_modules/follow-redirects": { - "version": "1.15.6", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.6.tgz", - "integrity": "sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==", - "funding": [ - { - "type": "individual", - "url": "https://github.com/sponsors/RubenVerborgh" - } - ], - "engines": { - "node": ">=4.0" - }, - "peerDependenciesMeta": { - "debug": { - "optional": true - } - } - }, "node_modules/for-each": { "version": "0.3.3", "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", @@ -4704,19 +4636,6 @@ "url": "https://github.com/sponsors/isaacs" } }, - "node_modules/form-data": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", - "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", - "dependencies": { - "asynckit": "^0.4.0", - "combined-stream": "^1.0.8", - "mime-types": "^2.1.12" - }, - "engines": { - "node": ">= 6" - } - }, "node_modules/fs-extra": { "version": "7.0.1", "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.1.tgz", @@ -5346,29 +5265,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/is-buffer": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.5.tgz", - "integrity": "sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "engines": { - "node": ">=4" - } - }, "node_modules/is-callable": { "version": "1.2.7", "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", @@ -6200,25 +6096,6 @@ "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", "dev": true }, - "node_modules/mime-db": { - "version": "1.52.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", - "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", - "engines": { - "node": ">= 0.6" - } - }, - "node_modules/mime-types": { - "version": "2.1.35", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", - "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", - "dependencies": { - "mime-db": "1.52.0" - }, - "engines": { - "node": ">= 0.6" - } - }, "node_modules/mimic-fn": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-4.0.0.tgz", @@ -7023,11 +6900,6 @@ "dev": true, "peer": true }, - "node_modules/proxy-from-env": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", - "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" - }, "node_modules/public-encrypt": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/public-encrypt/-/public-encrypt-4.0.3.tgz", diff --git a/package.json b/package.json index 8416ee8..635ffe1 100644 --- a/package.json +++ b/package.json @@ -68,7 +68,6 @@ }, "dependencies": { "@localazy/languages": "^0.1.6", - "axios": "^1.6.8", "lodash-es": "^4.17.21" }, "devDependencies": { @@ -80,7 +79,6 @@ "@typescript-eslint/parser": "^7.2.0", "@vitest/coverage-v8": "^1.4.0", "@vitest/ui": "^1.4.0", - "axios-mock-adapter": "^1.22.0", "eslint": "^8.57.0", "eslint-config-airbnb": "^19.0.4", "eslint-config-airbnb-base": "^15.0.0", diff --git a/src/api/api-client.ts b/src/api/api-client.ts index cb95480..7988402 100644 --- a/src/api/api-client.ts +++ b/src/api/api-client.ts @@ -1,4 +1,4 @@ -import { CreateAxiosDefaults } from 'axios'; +import { FetchHttpAdapter } from '@/http/fetch-http-adapter'; import { ApiExport } from '@/api/methods/api-export'; import { ApiFiles } from '@/api/methods/api-files'; import { ApiFormats } from '@/api/methods/api-formats'; @@ -8,7 +8,6 @@ import { ApiKeys } from '@/api/methods/api-keys'; import { ApiProjects } from '@/api/methods/api-projects'; import { ApiScreenshots } from '@/api/methods/api-screenshots'; import { ApiWebhooks } from '@/api/methods/api-webhooks'; -import { AxiosHttpAdapter } from '@/http/axios-http-adapter'; import { IHttpAdapter } from '@/http/i-http-adapter'; import { ApiClientOptions } from '@/types/api-client-options'; @@ -33,8 +32,8 @@ export class ApiClient { public screenshots: ApiScreenshots; - constructor(options: ApiClientOptions, config?: CreateAxiosDefaults) { - this.client = new AxiosHttpAdapter(options, config); + constructor(options: ApiClientOptions) { + this.client = new FetchHttpAdapter(options); this.projects = new ApiProjects(this); this.import = new ApiImport(this); diff --git a/src/api/methods/api-export.ts b/src/api/methods/api-export.ts index 00ba397..2375b38 100644 --- a/src/api/methods/api-export.ts +++ b/src/api/methods/api-export.ts @@ -1,5 +1,5 @@ import { Locales } from '@localazy/languages'; -import { AxiosRequestConfig } from 'axios'; +import { RequestConfig } from '@/types/request-config'; import { ApiBase } from '@/api/methods/api-base'; import { ExportJsonRequest } from '@/types/export-json-request'; import { I18nJson } from '@/types/i18n-json'; @@ -11,9 +11,9 @@ export class ApiExport extends ApiBase { * Export translated keys as JSON object. * * @param request Export JSON request config. - * @param config Axios request config. + * @param config Request config. */ - public async json(request: ExportJsonRequest, config?: AxiosRequestConfig): Promise { + public async json(request: ExportJsonRequest, config?: RequestConfig): Promise { const { project, file, langs }: ExportJsonRequest = request; const result: Key[][] = await Promise.all( diff --git a/src/api/methods/api-files.ts b/src/api/methods/api-files.ts index 9f74cde..133d367 100644 --- a/src/api/methods/api-files.ts +++ b/src/api/methods/api-files.ts @@ -1,5 +1,5 @@ -import { AxiosRequestConfig } from 'axios'; import { Blob } from 'node:buffer'; +import { RequestConfig } from '@/types/request-config'; import { ApiBase } from '@/api/methods/api-base'; import { File } from '@/types/file'; import { FileGetContentsRequest } from '@/types/file-get-contents-request'; @@ -13,27 +13,27 @@ export class ApiFiles extends ApiBase { * List all {@link File files} in the project. * * @param request Files list request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/files#list-files-in-project Localazy API Docs} */ - public async list(request: FilesListRequest, config?: AxiosRequestConfig): Promise { + public async list(request: FilesListRequest, config?: RequestConfig): Promise { const { project }: FilesListRequest = request; const projectId: string = ApiBase.getId(project, 'project'); - return this.api.client.get(`/projects/${projectId}/files`, config); + return (await this.api.client.get(`/projects/${projectId}/files`, config)) as Promise; } /** * First {@link File file} in the project. * * @param request Files list request config. - * @param config Axios request config. + * @param config Request config. * @throws Error At least one file must exist, otherwise an error is thrown. * * @see {@link https://localazy.com/docs/api/files#list-files-in-project Localazy API Docs} */ - public async first(request: FilesListRequest, config?: AxiosRequestConfig): Promise { + public async first(request: FilesListRequest, config?: RequestConfig): Promise { const files: File[] = await this.list(request, config); if (files.length === 0) { @@ -47,12 +47,12 @@ export class ApiFiles extends ApiBase { * List all {@link Key keys} for the language in the {@link File file}. * * @param request File list keys request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/files#retrieve-a-list-of-keys-and-translations-from-file * | Localazy API Docs} */ - public async listKeys(request: FileListKeysRequest, config?: AxiosRequestConfig): Promise { + public async listKeys(request: FileListKeysRequest, config?: RequestConfig): Promise { const keys: Key[] = []; let pageResult: KeysPaginated = { keys: [], @@ -72,37 +72,38 @@ export class ApiFiles extends ApiBase { * List all {@link Key keys} for the language in the {@link File file}. Result is paginated. * * @param request File list keys request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/files#retrieve-a-list-of-keys-and-translations-from-file * | Localazy API Docs} */ - public async listKeysPage(request: FileListKeysRequest, config?: AxiosRequestConfig): Promise { + public async listKeysPage(request: FileListKeysRequest, config?: RequestConfig): Promise { const { project, file, lang, ...params }: FileListKeysRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const fileId: string = ApiBase.getId(file, 'file'); - return this.api.client.get(`/projects/${projectId}/files/${fileId}/keys/${lang}`, { ...config, params }); + return (await this.api.client.get(`/projects/${projectId}/files/${fileId}/keys/${lang}`, { + ...config, + params, + })) as KeysPaginated; } /** * Get the contents of the {@link File file}. * * @param request File get contents request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/files#list-file-content Localazy API Docs} */ - public async getContents(request: FileGetContentsRequest, config?: AxiosRequestConfig): Promise { + public async getContents(request: FileGetContentsRequest, config?: RequestConfig): Promise { const { project, file, lang }: FileGetContentsRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const fileId: string = ApiBase.getId(file, 'file'); - const buffer: Uint8Array = await this.api.client.get(`/projects/${projectId}/files/${fileId}/download/${lang}`, { + return (await this.api.client.get(`/projects/${projectId}/files/${fileId}/download/${lang}`, { ...config, - responseType: 'arraybuffer', - }); - - return new Blob([buffer]); + responseType: 'blob', + })) as Blob; } } diff --git a/src/api/methods/api-formats.ts b/src/api/methods/api-formats.ts index 88cc7ec..777db54 100644 --- a/src/api/methods/api-formats.ts +++ b/src/api/methods/api-formats.ts @@ -1,16 +1,16 @@ -import { AxiosRequestConfig } from 'axios'; import { ApiBase } from '@/api/methods/api-base'; import { Format } from '@/types/format'; +import { RequestConfig } from '@/types/request-config'; export class ApiFormats extends ApiBase { /** * List all {@link Format formats} and related options. * - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/import#list-available-file-types Localazy API Docs} */ - public async list(config?: AxiosRequestConfig): Promise { - return this.api.client.get('/import/formats', config); + public async list(config?: RequestConfig): Promise { + return (await this.api.client.get('/import/formats', config)) as Format[]; } } diff --git a/src/api/methods/api-glossary.ts b/src/api/methods/api-glossary.ts index b1c458d..91a9539 100644 --- a/src/api/methods/api-glossary.ts +++ b/src/api/methods/api-glossary.ts @@ -1,4 +1,4 @@ -import { AxiosRequestConfig, AxiosResponseTransformer } from 'axios'; +import { RequestConfig } from '@/types/request-config'; import { ApiBase } from '@/api/methods/api-base'; import { GlossaryCreateRequest } from '@/types/glossary-create-request'; import { GlossaryDeleteRequest } from '@/types/glossary-delete-request'; @@ -12,35 +12,35 @@ export class ApiGlossary extends ApiBase { * List all {@link GlossaryRecord glossary records} in the project. * * @param request Glossary records list request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/glossary#list-all-glossary-terms Localazy API Docs} */ - public async list(request: GlossaryListRequest, config?: AxiosRequestConfig): Promise { + public async list(request: GlossaryListRequest, config?: RequestConfig): Promise { const { project }: GlossaryListRequest = request; const projectId: string = ApiBase.getId(project, 'project'); - const transformResponse: AxiosResponseTransformer = (data: string): GlossaryRecord[] => { - const json: { glossaries: GlossaryRecord[] } = JSON.parse(data); - return json.glossaries; - }; + const response: { glossaries: GlossaryRecord[] } = (await this.api.client.get( + `/projects/${projectId}/glossary`, + config, + )) as { glossaries: GlossaryRecord[] }; - return this.api.client.get(`/projects/${projectId}/glossary`, { transformResponse, ...config }); + return response.glossaries; } /** * Find {@link GlossaryRecord glossary record} specified by `id`. * * @param request Glossary record find request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/glossary#get-glossary-term Localazy API Docs} */ - public async find(request: GlossaryFindRequest, config?: AxiosRequestConfig): Promise { + public async find(request: GlossaryFindRequest, config?: RequestConfig): Promise { const { project, glossaryRecord }: GlossaryFindRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const id: string = typeof glossaryRecord === 'string' ? glossaryRecord : glossaryRecord.id; - return this.api.client.get(`/projects/${projectId}/glossary/${id}`, config); + return (await this.api.client.get(`/projects/${projectId}/glossary/${id}`, config)) as GlossaryRecord; } /** @@ -48,30 +48,31 @@ export class ApiGlossary extends ApiBase { * There is a limit of 1000 glossary records per project. * * @param request Glossary record create request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/glossary#create-new-glossary-term Localazy API Docs} */ - public async create(request: GlossaryCreateRequest, config?: AxiosRequestConfig): Promise { + public async create(request: GlossaryCreateRequest, config?: RequestConfig): Promise { const { project, ...data }: GlossaryCreateRequest = request; const projectId: string = ApiBase.getId(project, 'project'); - const transformResponse: AxiosResponseTransformer = (d: string): string => { - const json: { result: string } = JSON.parse(d); - return json.result; - }; + const response: { result: string } = (await this.api.client.post( + `/projects/${projectId}/glossary`, + data, + config, + )) as { result: string }; - return this.api.client.post(`/projects/${projectId}/glossary`, data, { transformResponse, ...config }); + return response.result; } /** * Update {@link GlossaryRecord glossary record} specified by `id`. * * @param request Glossary record update request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/glossary#update-glossary-term Localazy API Docs} */ - public async update(request: GlossaryUpdateRequest, config?: AxiosRequestConfig): Promise { + public async update(request: GlossaryUpdateRequest, config?: RequestConfig): Promise { const { project, glossaryRecord, ...data }: GlossaryUpdateRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const id: string = typeof glossaryRecord === 'string' ? glossaryRecord : glossaryRecord.id; @@ -83,11 +84,11 @@ export class ApiGlossary extends ApiBase { * Delete {@link GlossaryRecord glossary record} specified by `id`. * * @param request Glossary record delete request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/glossary#delete-glossary-term Localazy API Docs} */ - public async delete(request: GlossaryDeleteRequest, config?: AxiosRequestConfig): Promise { + public async delete(request: GlossaryDeleteRequest, config?: RequestConfig): Promise { const { project, glossaryRecord }: GlossaryDeleteRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const id: string = typeof glossaryRecord === 'string' ? glossaryRecord : glossaryRecord.id; diff --git a/src/api/methods/api-import.ts b/src/api/methods/api-import.ts index d5a36e2..f51ec56 100644 --- a/src/api/methods/api-import.ts +++ b/src/api/methods/api-import.ts @@ -1,4 +1,4 @@ -import { AxiosRequestConfig } from 'axios'; +import { RequestConfig } from '@/types/request-config'; import { ApiBase } from '@/api/methods/api-base'; import { File } from '@/types/file'; import { I18nJson } from '@/types/i18n-json'; @@ -15,17 +15,21 @@ export class ApiImport extends ApiBase { * Import source keys from JSON object. * * @param request Import JSON request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/import#import-content-to-a-project Localazy API Docs} */ - public async json(request: ImportJsonRequest, config?: AxiosRequestConfig): Promise { + public async json(request: ImportJsonRequest, config?: RequestConfig): Promise { const { project, json }: ImportJsonRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const chunks: I18nJson[] = JsonUtils.slice(json); const data: ImportData = importDataFactory(request, chunks); - const { result } = await this.api.client.post(`/projects/${projectId}/import`, data, config); + const { result }: { result: string } = (await this.api.client.post( + `/projects/${projectId}/import`, + data, + config, + )) as { result: string }; await delay(); diff --git a/src/api/methods/api-keys.ts b/src/api/methods/api-keys.ts index 3fd8cc5..882c672 100644 --- a/src/api/methods/api-keys.ts +++ b/src/api/methods/api-keys.ts @@ -1,18 +1,18 @@ -import { AxiosRequestConfig } from 'axios'; import { ApiBase } from '@/api/methods/api-base'; import { KeyDeleteRequest } from '@/types/key-delete-request'; import { KeyUpdateRequest } from '@/types/key-update-request'; +import { RequestConfig } from '@/types/request-config'; export class ApiKeys extends ApiBase { /** * Update {@link Key key}. * * @param request Key update request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/source-keys#update-source-key Localazy API Docs} */ - public async update(request: KeyUpdateRequest, config?: AxiosRequestConfig): Promise { + public async update(request: KeyUpdateRequest, config?: RequestConfig): Promise { const { project, key, ...data }: KeyUpdateRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const keyId: string = ApiBase.getId(key, 'key'); @@ -24,11 +24,11 @@ export class ApiKeys extends ApiBase { * Delete {@link Key key}. * * @param request Key delete request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/source-keys#delete-source-key Localazy API Docs} */ - public async delete(request: KeyDeleteRequest, config?: AxiosRequestConfig): Promise { + public async delete(request: KeyDeleteRequest, config?: RequestConfig): Promise { const { project, key }: KeyDeleteRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const keyId: string = ApiBase.getId(key, 'key'); diff --git a/src/api/methods/api-projects.ts b/src/api/methods/api-projects.ts index 031d639..3367c80 100644 --- a/src/api/methods/api-projects.ts +++ b/src/api/methods/api-projects.ts @@ -1,4 +1,4 @@ -import { AxiosRequestConfig } from 'axios'; +import { RequestConfig } from '@/types/request-config'; import { ApiBase } from '@/api/methods/api-base'; import { Project } from '@/types/project'; import { ProjectsListRequest } from '@/types/projects-list-request'; @@ -8,24 +8,24 @@ export class ApiProjects extends ApiBase { * List all {@link Project projects}. * * @param request Projects list request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/projects#list-projects Localazy API Docs} */ - public async list(request?: ProjectsListRequest, config?: AxiosRequestConfig): Promise { - return this.api.client.get('/projects', { ...config, params: request }); + public async list(request?: ProjectsListRequest, config?: RequestConfig): Promise { + return (await this.api.client.get('/projects', { ...config, params: request })) as Project[]; } /** * First {@link Project project}. * * @param request Projects list request config. - * @param config Axios request config. + * @param config Request config. * @throws Error At least one project must exist, otherwise an error is thrown. * * @see {@link https://localazy.com/docs/api/projects#list-projects Localazy API Docs} */ - public async first(request?: ProjectsListRequest, config?: AxiosRequestConfig): Promise { + public async first(request?: ProjectsListRequest, config?: RequestConfig): Promise { const projects: Project[] = await this.list(request, config); if (projects.length === 0) { diff --git a/src/api/methods/api-screenshots.ts b/src/api/methods/api-screenshots.ts index 4ecd5fa..273bb65 100644 --- a/src/api/methods/api-screenshots.ts +++ b/src/api/methods/api-screenshots.ts @@ -1,4 +1,4 @@ -import { AxiosRequestConfig, AxiosResponseTransformer } from 'axios'; +import { RequestConfig } from '@/types/request-config'; import { ApiBase } from '@/api/methods/api-base'; import { Screenshot } from '@/types/screenshot'; import { ScreenshotCreateRequest } from '@/types/screenshot-create-request'; @@ -14,64 +14,62 @@ export class ApiScreenshots extends ApiBase { * List all {@link Screenshot screenshots} in the project. * * @param request Screenshots list request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/screenshot-management#list-screenshots Localazy API Docs} */ - public async list(request: ScreenshotsListRequest, config?: AxiosRequestConfig): Promise { + public async list(request: ScreenshotsListRequest, config?: RequestConfig): Promise { const { project }: ScreenshotsListRequest = request; const projectId: string = ApiBase.getId(project, 'project'); - return this.api.client.get(`/projects/${projectId}/screenshots`, config); + return (await this.api.client.get(`/projects/${projectId}/screenshots`, config)) as Screenshot[]; } /** * List all {@link ScreenshotTag screenshots tags} in the project. * * @param request Screenshots list tags request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/screenshot-management#list-screenshots-tags Localazy API Docs} */ - public async listTags(request: ScreenshotsListTagsRequest, config?: AxiosRequestConfig): Promise { + public async listTags(request: ScreenshotsListTagsRequest, config?: RequestConfig): Promise { const { project }: ScreenshotsListTagsRequest = request; const projectId: string = ApiBase.getId(project, 'project'); - return this.api.client.get(`/projects/${projectId}/screenshots/tags`, config); + return (await this.api.client.get(`/projects/${projectId}/screenshots/tags`, config)) as ScreenshotTag[]; } /** * Create {@link Screenshot screenshot}. * * @param request Screenshot create request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/screenshot-management#create-a-new-screenshot Localazy API Docs} */ - public async create(request: ScreenshotCreateRequest, config?: AxiosRequestConfig): Promise { + public async create(request: ScreenshotCreateRequest, config?: RequestConfig): Promise { const { project }: ScreenshotCreateRequest = request; const projectId: string = ApiBase.getId(project, 'project'); - const transformResponse: AxiosResponseTransformer = (d: string): string => { - const json: { id: string } = JSON.parse(d); - return json.id; - }; + const response: { id: string } = (await this.api.client.post( + `/projects/${projectId}/screenshots`, + request.encodedData, + config, + )) as { id: string }; - return this.api.client.post(`/projects/${projectId}/screenshots`, request.encodedData, { - transformResponse, - ...config, - }); + return response.id; } /** * Update the image data of {@link Screenshot screenshot}. * * @param request Screenshot update image data request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/screenshot-management#update-the-image-of-an-existing-screenshot * | Localazy API Docs} */ - public async updateImageData(request: ScreenshotUpdateImageDataRequest, config?: AxiosRequestConfig): Promise { + public async updateImageData(request: ScreenshotUpdateImageDataRequest, config?: RequestConfig): Promise { const { project, screenshot }: ScreenshotUpdateImageDataRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const screenshotId: string = ApiBase.getId(screenshot, 'screenshot'); @@ -84,11 +82,11 @@ export class ApiScreenshots extends ApiBase { * Image data are updated with `screenshots.updateImageData`. * * @param request Screenshot update request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/screenshot-management#update-an-existing-screenshot Localazy API Docs} */ - public async update(request: ScreenshotUpdateRequest, config?: AxiosRequestConfig): Promise { + public async update(request: ScreenshotUpdateRequest, config?: RequestConfig): Promise { const { project, screenshot, ...data }: ScreenshotUpdateRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const screenshotId: string = ApiBase.getId(screenshot, 'screenshot'); @@ -100,11 +98,11 @@ export class ApiScreenshots extends ApiBase { * Delete {@link Screenshot screenshot}. * * @param request Screenshot delete request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/screenshot-management#delete-a-screenshot Localazy API Docs} */ - public async delete(request: ScreenshotDeleteRequest, config?: AxiosRequestConfig): Promise { + public async delete(request: ScreenshotDeleteRequest, config?: RequestConfig): Promise { const { project, screenshot }: ScreenshotDeleteRequest = request; const projectId: string = ApiBase.getId(project, 'project'); const screenshotId: string = ApiBase.getId(screenshot, 'screenshot'); diff --git a/src/api/methods/api-webhooks.ts b/src/api/methods/api-webhooks.ts index 1fe7b1c..cd229d5 100644 --- a/src/api/methods/api-webhooks.ts +++ b/src/api/methods/api-webhooks.ts @@ -1,4 +1,4 @@ -import { AxiosRequestConfig, AxiosResponseTransformer } from 'axios'; +import { RequestConfig } from '@/types/request-config'; import { ApiBase } from '@/api/methods/api-base'; import { Webhook } from '@/types/webhook'; import { WebhooksGetSecretRequest } from '@/types/webhooks-get-secret-request'; @@ -11,30 +11,29 @@ export class ApiWebhooks extends ApiBase { * List all {@link Webhook webhooks} in the project. * * @param request Webhooks list request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/webhooks-api#list-webhooks-configuration Localazy API Docs} */ - public async list(request: WebhooksListRequest, config?: AxiosRequestConfig): Promise { + public async list(request: WebhooksListRequest, config?: RequestConfig): Promise { const { project }: WebhooksListRequest = request; const projectId: string = ApiBase.getId(project, 'project'); - const transformResponse: AxiosResponseTransformer = (data: string): Webhook[] => { - const json: { items: Webhook[] } = JSON.parse(data); - return json.items; + const response: { items: Webhook[] } = (await this.api.client.get(`/projects/${projectId}/webhooks`, config)) as { + items: Webhook[]; }; - return this.api.client.get(`/projects/${projectId}/webhooks`, { transformResponse, ...config }); + return response.items; } /** * Update all {@link Webhook webhooks} in the project. * * @param request Webhooks update request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/webhooks-api#update-webhooks-configuration Localazy API Docs} */ - public async update(request: WebhooksUpdateRequest, config?: AxiosRequestConfig): Promise { + public async update(request: WebhooksUpdateRequest, config?: RequestConfig): Promise { const { project, ...data }: WebhooksUpdateRequest = request; const projectId: string = ApiBase.getId(project, 'project'); @@ -47,18 +46,18 @@ export class ApiWebhooks extends ApiBase { * {@link https://localazy.com/docs/api/webhooks-api#security}. * * @param request Webhooks get secret request config. - * @param config Axios request config. + * @param config Request config. * * @see {@link https://localazy.com/docs/api/webhooks-api#webhook-secrets Localazy API Docs} */ - public async getSecret(request: WebhooksGetSecretRequest, config?: AxiosRequestConfig): Promise { + public async getSecret(request: WebhooksGetSecretRequest, config?: RequestConfig): Promise { const { project }: WebhooksGetSecretRequest = request; const projectId: string = ApiBase.getId(project, 'project'); - const transformResponse: AxiosResponseTransformer = (data: string): WebhooksSecret => { - const json: { secret: WebhooksSecret } = JSON.parse(data); - return json.secret; - }; + const response: { secret: string } = (await this.api.client.get( + `/projects/${projectId}/webhooks/secret`, + config, + )) as { secret: string }; - return this.api.client.get(`/projects/${projectId}/webhooks/secret`, { transformResponse, ...config }); + return response.secret; } } diff --git a/src/errors/localazy-error.ts b/src/errors/localazy-error.ts deleted file mode 100644 index 40ef237..0000000 --- a/src/errors/localazy-error.ts +++ /dev/null @@ -1,10 +0,0 @@ -export class LocalazyError extends Error { - public code: number; - - constructor(message: string, code: number, options?: ErrorOptions) { - super(message, options); - - this.name = 'LocalazyError'; - this.code = code; - } -} diff --git a/src/http/axios-http-adapter.ts b/src/http/axios-http-adapter.ts deleted file mode 100644 index 5d4ca54..0000000 --- a/src/http/axios-http-adapter.ts +++ /dev/null @@ -1,59 +0,0 @@ -import axios, { AxiosInstance, AxiosRequestConfig, AxiosResponse, CreateAxiosDefaults } from 'axios'; -import { isNumber, isPlainObject, isString } from 'lodash-es'; -import { LocalazyError } from '@/errors/localazy-error'; -import { IHttpAdapter } from '@/http/i-http-adapter'; -import { ApiClientOptions } from '@/types/api-client-options'; - -export class AxiosHttpAdapter implements IHttpAdapter { - public client: AxiosInstance; - - constructor(options: ApiClientOptions, config?: CreateAxiosDefaults) { - this.client = AxiosHttpAdapter.clientFactory( - options.apiUrl || 'https://api.localazy.com', - options.authToken, - config, - ); - } - - protected static clientFactory(baseURL: string, authToken: string, config?: CreateAxiosDefaults): AxiosInstance { - const client: AxiosInstance = axios.create({ - ...config, - baseURL, - headers: { - common: { - Authorization: `Bearer ${authToken}`, - Accept: `application/json`, - }, - }, - }); - - client.interceptors.response.use( - (response: AxiosResponse) => response.data, - (err: any): Promise => { - const data = err.response?.data; - if (isPlainObject(data) && isNumber(data.code) && isString(data.error)) { - return Promise.reject(new LocalazyError(data.error, data.code, { cause: err })); - } - return Promise.reject(err); - }, - ); - - return client; - } - - post(url: string, data?: D, config?: AxiosRequestConfig): Promise { - return this.client.post(url, data, config); - } - - put(url: string, data?: D, config?: AxiosRequestConfig): Promise { - return this.client.put(url, data, config); - } - - get(url: string, config?: AxiosRequestConfig): Promise { - return this.client.get(url, config); - } - - delete(url: string, config?: AxiosRequestConfig): Promise { - return this.client.delete(url, config); - } -} diff --git a/src/http/fetch-http-adapter.ts b/src/http/fetch-http-adapter.ts new file mode 100644 index 0000000..6002df6 --- /dev/null +++ b/src/http/fetch-http-adapter.ts @@ -0,0 +1,88 @@ +import { IHttpAdapter } from '@/http/i-http-adapter'; +import { ApiClientOptions } from '@/types/api-client-options'; +import { RequestConfig } from '@/types/request-config'; + +export class FetchHttpAdapter implements IHttpAdapter { + protected apiUrl: string; + + protected authToken: string; + + constructor(options: ApiClientOptions) { + this.apiUrl = options.apiUrl || 'https://api.localazy.com'; + this.authToken = options.authToken; + } + + async get(url: string, config?: RequestConfig): Promise { + return this.makeRequest('GET', url, config); + } + + async post(url: string, data: unknown, config?: RequestConfig): Promise { + return this.makeRequest('POST', url, config, data); + } + + async put(url: string, data: unknown, config?: RequestConfig): Promise { + return this.makeRequest('PUT', url, config, data); + } + + async delete(url: string, config?: RequestConfig): Promise { + return this.makeRequest('DELETE', url, config); + } + + protected urlFactory(url: string): string { + return `${this.apiUrl}${url}`; + } + + protected configFactory(method: string, config?: RequestConfig, data?: unknown): RequestInit { + const result: RequestInit = { + method, + headers: { + ...(config?.headers || {}), + Authorization: `Bearer ${this.authToken}`, + Accept: 'application/json', + 'Content-Type': 'application/json', + }, + }; + + if (data) { + result.body = JSON.stringify(data); + } + + return result; + } + + protected async makeRequest( + method: string, + url: string, + config?: RequestConfig, + data?: unknown, + ): Promise { + let resolvedUrl: string = this.urlFactory(url); + if (method === 'GET' && config?.params) { + const params: URLSearchParams = new URLSearchParams(config.params); + resolvedUrl += `?${params.toString()}`; + } + const response: Response = await fetch(resolvedUrl, this.configFactory(method, config, data)); + + const contentType: string | null = response.headers.get('content-type'); + const isBlob: boolean = config?.responseType === 'blob' || false; + const isJson: boolean = contentType?.startsWith('application/json') || false; + + let result: string | object | { error: string } | Blob; + + if (isBlob) { + result = await response.blob(); + } else if (isJson) { + result = (await response.json()) as object; + } else { + result = await response.text(); + } + + if (response.status >= 400) { + // @ts-expect-error property is not defined on string + const text: string = isJson && result.error ? result.error : response.statusText; + throw new Error(`Request failed with status code ${response.status}: ${text}`); + } + + return result; + } +} diff --git a/src/http/i-http-adapter.ts b/src/http/i-http-adapter.ts index f82cb9c..a79dfa3 100644 --- a/src/http/i-http-adapter.ts +++ b/src/http/i-http-adapter.ts @@ -1,11 +1,6 @@ -import { AxiosRequestConfig } from 'axios'; - -export interface IHttpAdapter { - post(url: string, data?: D, config?: AxiosRequestConfig): Promise; - - put(url: string, data?: D, config?: AxiosRequestConfig): Promise; - - get(url: string, config?: AxiosRequestConfig): Promise; - - delete(url: string, config?: AxiosRequestConfig): Promise; +export interface IHttpAdapter { + get: (url: string, config?: T) => Promise; + post: (url: string, data: unknown, config?: T) => Promise; + put: (url: string, data: unknown, config?: T) => Promise; + delete: (url: string, config?: T) => Promise; } diff --git a/src/main.ts b/src/main.ts index f565383..e8157c3 100644 --- a/src/main.ts +++ b/src/main.ts @@ -29,8 +29,7 @@ export * from '@/enums/project-tone'; export * from '@/enums/project-type'; export * from '@/enums/user-role'; export * from '@/enums/webhook-event'; -export * from '@/errors/localazy-error'; -export * from '@/http/axios-http-adapter'; +export * from '@/http/fetch-http-adapter'; export * from '@/http/i-http-adapter'; export * from '@/types/api-client-options'; export * from '@/types/export-json-request'; @@ -69,6 +68,7 @@ export * from '@/types/locales-keys'; export * from '@/types/organization'; export * from '@/types/project'; export * from '@/types/projects-list-request'; +export * from '@/types/request-config'; export * from '@/types/screenshot-create-request'; export * from '@/types/screenshot-delete-request'; export * from '@/types/screenshot-metadata'; diff --git a/src/types/api-client-options.ts b/src/types/api-client-options.ts index c13027e..8da3baf 100644 --- a/src/types/api-client-options.ts +++ b/src/types/api-client-options.ts @@ -1,10 +1,5 @@ -import { CreateAxiosDefaults } from 'axios'; - export type ApiClientOptions = { authToken: string; apiUrl?: string; - - // TODO remove - axios?: CreateAxiosDefaults; }; diff --git a/src/types/request-config.ts b/src/types/request-config.ts new file mode 100644 index 0000000..78d8146 --- /dev/null +++ b/src/types/request-config.ts @@ -0,0 +1,5 @@ +export type RequestConfig = { + headers?: Record; + params?: Record; + responseType?: 'json' | 'text' | 'blob'; +}; diff --git a/tests/fixtures/empty-project/index.ts b/tests/fixtures/empty-project/index.ts index e4139ed..256d434 100644 --- a/tests/fixtures/empty-project/index.ts +++ b/tests/fixtures/empty-project/index.ts @@ -1,18 +1,20 @@ -import { mockAdapter } from '@tests/support/axios-mock-adapter'; +import { fetchMock } from '@tests/support'; import projects from '@tests/fixtures/empty-project/projects.json'; import files from '@tests/fixtures/empty-project/files.json'; +const baseUrl: string = 'https://api.localazy.com'; + export const serverResponses = { projects, files, }; export const mockResponses = (): void => { - mockAdapter.reset(); + fetchMock.reset(); // projects - mockAdapter.onGet('/projects').reply(200, JSON.stringify(serverResponses.projects)); + fetchMock.get(`${baseUrl}/projects`, serverResponses.projects); // files - mockAdapter.onGet('/projects/_a0000000000000000002/files').reply(200, JSON.stringify(serverResponses.files)); + fetchMock.get(`${baseUrl}/projects/_a0000000000000000002/files`, serverResponses.files); }; diff --git a/tests/fixtures/full-project/index.ts b/tests/fixtures/full-project/index.ts index 385499d..d1c0d2f 100644 --- a/tests/fixtures/full-project/index.ts +++ b/tests/fixtures/full-project/index.ts @@ -1,19 +1,22 @@ +import { fetchMock } from '@tests/support'; import fileDownload from '@tests/fixtures/full-project/fileDownload.json'; import fileKeys from '@tests/fixtures/full-project/fileKeys.json'; import files from '@tests/fixtures/full-project/files.json'; import formats from '@tests/fixtures/full-project/formats.json'; import glossary from '@tests/fixtures/full-project/glossary.json'; import projects from '@tests/fixtures/full-project/projects.json'; +import projectsOrgsLangs from '@tests/fixtures/full-project/projectsOrgsLangs.json'; import screenshots from '@tests/fixtures/full-project/screenshots.json'; import screenshotTags from '@tests/fixtures/full-project/screenshotTags.json'; -import { textToUint8Array } from '@tests/support/utils'; import webhooks from '@tests/fixtures/full-project/webhooks.json'; import webhooksSecret from '@tests/fixtures/full-project/webhooksSecret.json'; -import { mockAdapter } from '@tests/support/axios-mock-adapter'; + +const baseUrl: string = 'https://api.localazy.com'; export const serverResponses = { formats, projects, + projectsOrgsLangs, glossary, files, fileKeys, @@ -48,79 +51,79 @@ export const serverResponses = { }; export const mockResponses = (): void => { - mockAdapter.reset(); + fetchMock.reset(); // formats - mockAdapter.onGet('/import/formats').reply(200, JSON.stringify(serverResponses.formats)); + fetchMock.get(`${baseUrl}/import/formats`, serverResponses.formats); // projects - mockAdapter.onGet('/projects').reply(200, JSON.stringify(serverResponses.projects)); + fetchMock.get(`${baseUrl}/projects`, serverResponses.projects); + fetchMock.get(`${baseUrl}/projects?languages=true&organization=true`, serverResponses.projectsOrgsLangs); // glossary - mockAdapter.onGet('/projects/_a0000000000000000001/glossary').reply(200, JSON.stringify(serverResponses.glossary)); - mockAdapter - .onGet('/projects/_a0000000000000000001/glossary/_a0000000000000000001') - .reply(200, JSON.stringify(serverResponses.glossary.glossaries[0])); - mockAdapter.onPost('/projects/_a0000000000000000001/glossary').reply(200, JSON.stringify(serverResponses.resultPost)); - mockAdapter - .onPut('/projects/_a0000000000000000001/glossary/_a0000000000000000001') - .reply(200, JSON.stringify(serverResponses.resultPut)); - mockAdapter - .onDelete('/projects/_a0000000000000000001/glossary/_a0000000000000000001') - .reply(200, JSON.stringify(serverResponses.resultDelete)); + fetchMock.get(`${baseUrl}/projects/_a0000000000000000001/glossary`, serverResponses.glossary); + fetchMock.get( + `${baseUrl}/projects/_a0000000000000000001/glossary/_a0000000000000000001`, + serverResponses.glossary.glossaries[0], + ); + fetchMock.post(`${baseUrl}/projects/_a0000000000000000001/glossary`, serverResponses.resultPost); + fetchMock.put(`${baseUrl}/projects/_a0000000000000000001/glossary/_a0000000000000000001`, serverResponses.resultPut); + fetchMock.delete( + `${baseUrl}/projects/_a0000000000000000001/glossary/_a0000000000000000001`, + serverResponses.resultDelete, + ); // files - mockAdapter.onGet('/projects/_a0000000000000000001/files').reply(200, JSON.stringify(serverResponses.files)); - mockAdapter - .onGet('/projects/_a0000000000000000001/files/_e000000000001/keys/en') - .reply(200, JSON.stringify(serverResponses.fileKeys)); - mockAdapter - .onGet('/projects/_a0000000000000000001/files/_e000000000001/download/en') - .reply(200, textToUint8Array(serverResponses.fileDownload)); + fetchMock.get(`${baseUrl}/projects/_a0000000000000000001/files`, serverResponses.files); + fetchMock.get( + `${baseUrl}/projects/_a0000000000000000001/files/_e000000000001/download/en`, + serverResponses.fileDownload, + ); // keys - mockAdapter - .onPut('/projects/_a0000000000000000001/keys/_a0000000000000000001') - .reply(200, JSON.stringify(serverResponses.resultPut)); - mockAdapter - .onDelete('/projects/_a0000000000000000001/keys/_a0000000000000000001') - .reply(200, JSON.stringify(serverResponses.resultDelete)); + fetchMock.get(`${baseUrl}/projects/_a0000000000000000001/files/_e000000000001/keys/en`, serverResponses.fileKeys); + fetchMock.get( + `${baseUrl}/projects/_a0000000000000000001/files/_e000000000001/keys/en?next=`, + serverResponses.fileKeys, + ); + fetchMock.put(`${baseUrl}/projects/_a0000000000000000001/keys/_a0000000000000000001`, serverResponses.resultPut); + fetchMock.delete( + `${baseUrl}/projects/_a0000000000000000001/keys/_a0000000000000000001`, + serverResponses.resultDelete, + ); // import - mockAdapter.onPost('/projects/_a0000000000000000001/import').reply(200, JSON.stringify(serverResponses.resultPost)); + fetchMock.post(`${baseUrl}/projects/_a0000000000000000001/import`, serverResponses.resultPost); // screenshots - mockAdapter - .onGet('/projects/_a0000000000000000001/screenshots') - .reply(200, JSON.stringify(serverResponses.screenshots)); - mockAdapter - .onGet('/projects/_a0000000000000000001/screenshots/tags') - .reply(200, JSON.stringify(serverResponses.screenshotTags)); - mockAdapter - .onPost('/projects/_a0000000000000000001/screenshots') - .reply(200, JSON.stringify(serverResponses.resultPostScreenshot)); - mockAdapter - .onPost('/projects/_a0000000000000000001/screenshots/_a0000000000000000001') - .reply(200, JSON.stringify(serverResponses.resultPost)); - mockAdapter - .onPut('/projects/_a0000000000000000001/screenshots/_a0000000000000000001') - .reply(200, JSON.stringify(serverResponses.resultPut)); - mockAdapter - .onDelete('/projects/_a0000000000000000001/screenshots/_a0000000000000000001') - .reply(200, JSON.stringify(serverResponses.resultDelete)); + fetchMock.get(`${baseUrl}/projects/_a0000000000000000001/screenshots`, serverResponses.screenshots); + fetchMock.get(`${baseUrl}/projects/_a0000000000000000001/screenshots/tags`, serverResponses.screenshotTags); + fetchMock.post(`${baseUrl}/projects/_a0000000000000000001/screenshots`, serverResponses.resultPostScreenshot); + fetchMock.post( + `${baseUrl}/projects/_a0000000000000000001/screenshots/_a0000000000000000001`, + serverResponses.resultPost, + ); + fetchMock.put( + `${baseUrl}/projects/_a0000000000000000001/screenshots/_a0000000000000000001`, + serverResponses.resultPut, + ); + fetchMock.delete( + `${baseUrl}/projects/_a0000000000000000001/screenshots/_a0000000000000000001`, + serverResponses.resultDelete, + ); // webhooks - mockAdapter.onGet('/projects/_a0000000000000000001/webhooks').reply(200, JSON.stringify(serverResponses.webhooks)); - mockAdapter - .onGet('/projects/_a0000000000000000001/webhooks/secret') - .reply(200, JSON.stringify(serverResponses.webhooksSecret)); - mockAdapter.onPost('/projects/_a0000000000000000001/webhooks').reply(200, JSON.stringify(serverResponses.resultPost)); + fetchMock.get(`${baseUrl}/projects/_a0000000000000000001/webhooks`, serverResponses.webhooks); + fetchMock.get(`${baseUrl}/projects/_a0000000000000000001/webhooks/secret`, serverResponses.webhooksSecret); + fetchMock.post(`${baseUrl}/projects/_a0000000000000000001/webhooks`, serverResponses.resultPost); // errors - mockAdapter - .onPut('/projects/_a0000000000000000001/keys/unknown-key-id') - .reply(400, JSON.stringify(serverResponses.resultInvalidId)); - mockAdapter - .onPut('/projects/_a0000000000000000001/keys/_a1111111111111111111') - .reply(401, JSON.stringify(serverResponses.resultUnauthorized)); + fetchMock.put(`${baseUrl}/projects/_a0000000000000000001/keys/unknown-key-id`, { + status: 400, + body: serverResponses.resultInvalidId, + }); + fetchMock.put(`${baseUrl}/projects/_a0000000000000000001/keys/_a1111111111111111111`, { + status: 401, + body: serverResponses.resultUnauthorized, + }); }; diff --git a/tests/fixtures/full-project/projectsOrgsLangs.json b/tests/fixtures/full-project/projectsOrgsLangs.json new file mode 100644 index 0000000..5a7a5ba --- /dev/null +++ b/tests/fixtures/full-project/projectsOrgsLangs.json @@ -0,0 +1,44 @@ +[ + { + "id": "_a0000000000000000001", + "orgId": "_a000001", + "name": "Test project", + "slug": "test-project", + "image": "", + "url": "https://testing.localazy.com/p/test-project", + "description": "", + "type": "restricted", + "tone": "not_specified", + "role": "owner", + "organization": { + "availableKeys": 50000, + "usedKeys": 17431, + "figma": true, + "screenshotsForFigma": true, + "connectedApps": true, + "releaseTags": true, + "formatConversions": true, + "screenshots": true, + "additionalMt": true, + "mtPretranslate": true, + "webhooks": true + }, + "languages": [ + { + "id": 85, + "code": "en", + "tag": "en", + "name": "English", + "active": 6, + "review": 0, + "current": 6, + "translated": 6, + "sourceChanged": 0, + "needImprovement": 0, + "enabled": true, + "published": true + } + ], + "sourceLanguage": 85 + } +] diff --git a/tests/specs/__snapshots__/glossary.spec.ts.snap b/tests/specs/__snapshots__/glossary.spec.ts.snap deleted file mode 100644 index 0b9f8b4..0000000 --- a/tests/specs/__snapshots__/glossary.spec.ts.snap +++ /dev/null @@ -1,5 +0,0 @@ -// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html - -exports[`Glossary > api.glossary.delete 1`] = `undefined`; - -exports[`Glossary > api.glossary.update 1`] = `"{"description":"Exceptional term description","caseSensitive":true,"translateTerm":true,"term":[{"lang":"en","term":"Exceptional term"}]}"`; diff --git a/tests/specs/__snapshots__/import.spec.ts.snap b/tests/specs/__snapshots__/import.spec.ts.snap deleted file mode 100644 index 2c5d349..0000000 --- a/tests/specs/__snapshots__/import.spec.ts.snap +++ /dev/null @@ -1,3 +0,0 @@ -// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html - -exports[`Import > api.import.json 1`] = `"{"files":[{"name":"en.json","content":{"type":"json","en":{"headers":{"person":"Active person"}}}}]}"`; diff --git a/tests/specs/__snapshots__/keys.spec.ts.snap b/tests/specs/__snapshots__/keys.spec.ts.snap deleted file mode 100644 index f295348..0000000 --- a/tests/specs/__snapshots__/keys.spec.ts.snap +++ /dev/null @@ -1,5 +0,0 @@ -// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html - -exports[`Keys > api.keys.delete 1`] = `undefined`; - -exports[`Keys > api.keys.update 1`] = `"{"comment":"Comment for translators."}"`; diff --git a/tests/specs/__snapshots__/screenshots.spec.ts.snap b/tests/specs/__snapshots__/screenshots.spec.ts.snap deleted file mode 100644 index 9f8283f..0000000 --- a/tests/specs/__snapshots__/screenshots.spec.ts.snap +++ /dev/null @@ -1,9 +0,0 @@ -// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html - -exports[`Screenshots > api.screenshots.create 1`] = `""`; - -exports[`Screenshots > api.screenshots.delete 1`] = `undefined`; - -exports[`Screenshots > api.screenshots.update 1`] = `"{"comment":"Hey! Nice screenshot.","tags":["blue"]}"`; - -exports[`Screenshots > api.screenshots.updateImageData 1`] = `""`; diff --git a/tests/specs/__snapshots__/webhooks.spec.ts.snap b/tests/specs/__snapshots__/webhooks.spec.ts.snap deleted file mode 100644 index 3333289..0000000 --- a/tests/specs/__snapshots__/webhooks.spec.ts.snap +++ /dev/null @@ -1,3 +0,0 @@ -// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html - -exports[`Webhooks > api.webhooks.update 1`] = `"{"items":[{"enabled":true,"customId":"1","description":"This is a test webhook","url":"https://example.com/webhook","events":["comment_added","import_finished","import_finished_empty","project_published","tag_promoted"]}]}"`; diff --git a/tests/specs/errors.spec.ts b/tests/specs/errors.spec.ts index 5a2b8b6..98e4b23 100644 --- a/tests/specs/errors.spec.ts +++ b/tests/specs/errors.spec.ts @@ -9,21 +9,9 @@ describe('Errors', (): void => { const api: ApiClient = getApiClient(); const project: Project = await api.projects.first(); - let name: string | undefined; - let message: string | undefined; - let code: string | undefined; - - try { + await expect(async (): Promise => { await api.keys.update({ project, key: 'unknown-key-id', comment: 'Key comment' }); - } catch (err: any) { - name = err.name; - message = err.message; - code = err.code; - } - - expect(name).toBe('LocalazyError'); - expect(message).toBe('invalid_id'); - expect(code).toBe(400); + }).rejects.toThrowError('Request failed with status code 400: invalid_id'); }); test('Update key with random id', async (): Promise => { @@ -31,21 +19,9 @@ describe('Errors', (): void => { const api: ApiClient = getApiClient(); const project: Project = await api.projects.first(); - let name: string | undefined; - let message: string | undefined; - let code: string | undefined; - - try { + await expect(async (): Promise => { await api.keys.update({ project, key: '_a1111111111111111111', comment: 'Key comment' }); - } catch (err: any) { - name = err.name; - message = err.message; - code = err.code; - } - - expect(name).toBe('LocalazyError'); - expect(message).toBe('unauthorized'); - expect(code).toBe(401); + }).rejects.toThrowError('Request failed with status code 401: unauthorized'); }); test('Get first file in empty project', async (): Promise => { @@ -57,9 +33,7 @@ describe('Errors', (): void => { try { await api.files.first({ project }); - debugger; } catch (err: any) { - debugger; message = err.message; } diff --git a/tests/specs/files.spec.ts b/tests/specs/files.spec.ts index 7834f93..4a30d26 100644 --- a/tests/specs/files.spec.ts +++ b/tests/specs/files.spec.ts @@ -1,6 +1,6 @@ -import { beforeEach, describe, expect, test } from 'vitest'; +import { beforeEach, describe, expect, MockInstance, test, vi } from 'vitest'; import { Blob } from 'node:buffer'; -import { getApiClient } from '@tests/support'; +import { getApiClient, getApiUrl, getToken } from '@tests/support'; import { fullProject } from '@tests/fixtures'; import { ApiClient, @@ -25,25 +25,55 @@ describe('Files', (): void => { }); test('api.files.list', async (): Promise => { + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); const files: File[] = await api.files.list({ project }); expect(files[0].name).toBe('en.json'); expect(files[0].type).toBe('json'); + expect(spy).toHaveBeenCalledWith(`${getApiUrl()}/projects/_a0000000000000000001/files`, { + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'GET', + }); }); test('api.files.first', async (): Promise => { + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); const file: File = await api.files.first({ project }); expect(file.name).toBe('en.json'); expect(file.type).toBe('json'); + expect(spy).toHaveBeenCalledWith(`${getApiUrl()}/projects/_a0000000000000000001/files`, { + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'GET', + }); }); test('api.files.listKeys', async (): Promise => { const file: File = await api.files.first({ project }); const request: FileListKeysRequest = { project, file, lang: Locales.ENGLISH }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); const keys: Key[] = await api.files.listKeys(request); expect(keys[0].value).toBe('My Application'); + expect(spy).toHaveBeenCalledWith( + `${getApiUrl()}/projects/_a0000000000000000001/files/_e000000000001/keys/en?next=`, + { + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'GET', + }, + ); }); test('api.files.listKeysPage', async (): Promise => { @@ -52,7 +82,6 @@ describe('Files', (): void => { project, file, lang: Locales.ENGLISH, - limit: 1, }; const keys: Key[] = []; @@ -71,6 +100,7 @@ describe('Files', (): void => { }); test('api.files.getContents', async (): Promise => { + // const files: File[] = await api.files.list({ project }); const file: File = await api.files.first({ project }); const request: FileGetContentsRequest = { project, file, lang: Locales.ENGLISH }; const blob: Blob = await api.files.getContents(request); diff --git a/tests/specs/glossary.spec.ts b/tests/specs/glossary.spec.ts index 26bf7cc..8462a6c 100644 --- a/tests/specs/glossary.spec.ts +++ b/tests/specs/glossary.spec.ts @@ -1,5 +1,5 @@ -import { beforeEach, describe, expect, test } from 'vitest'; -import { getApiClient, mockAdapter } from '@tests/support'; +import { beforeEach, describe, expect, MockInstance, test, vi } from 'vitest'; +import { getApiClient, getToken } from '@tests/support'; import { fullProject } from '@tests/fixtures'; import { ApiClient, @@ -30,10 +30,22 @@ describe('Glossary', (): void => { test('api.glossary.find', async (): Promise => { const records: GlossaryRecord[] = await api.glossary.list({ project }); + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); const record: GlossaryRecord = await api.glossary.find({ project, glossaryRecord: records[0].id }); expect(record.term[0].term).toBe('Monitor'); expect(record.description).toBe('A screen used for displaying visual output from a computer.'); + expect(spy).toHaveBeenCalledWith( + 'https://api.localazy.com/projects/_a0000000000000000001/glossary/_a0000000000000000001', + { + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'GET', + }, + ); }); test('api.glossary.create', async (): Promise => { @@ -44,9 +56,20 @@ describe('Glossary', (): void => { translateTerm: true, term: [{ lang: 'en', term: 'Exceptional term' }], }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); const recordId: string = await api.glossary.create(request); expect(recordId).toBe(fullProject.serverResponses.resultPost.result); + expect(spy).toHaveBeenCalledWith('https://api.localazy.com/projects/_a0000000000000000001/glossary', { + // eslint-disable-next-line max-len + body: '{"description":"Exceptional term description","caseSensitive":true,"translateTerm":true,"term":[{"lang":"en","term":"Exceptional term"}]}', + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'POST', + }); }); test('api.glossary.update', async (): Promise => { @@ -59,18 +82,40 @@ describe('Glossary', (): void => { translateTerm: true, term: [{ lang: 'en', term: 'Exceptional term' }], }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); await api.glossary.update(request); - expect(mockAdapter.history.put.length).toBe(1); - expect(mockAdapter.history.put[0].data).toMatchSnapshot(); + expect(spy).toHaveBeenCalledWith( + 'https://api.localazy.com/projects/_a0000000000000000001/glossary/_a0000000000000000001', + { + // eslint-disable-next-line max-len + body: '{"description":"Exceptional term description","caseSensitive":true,"translateTerm":true,"term":[{"lang":"en","term":"Exceptional term"}]}', + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'PUT', + }, + ); }); test('api.glossary.delete', async (): Promise => { const records: GlossaryRecord[] = await api.glossary.list({ project }); const request: GlossaryDeleteRequest = { project, glossaryRecord: records[0] }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); await api.glossary.delete(request); - expect(mockAdapter.history.delete.length).toBe(1); - expect(mockAdapter.history.delete[0].data).toMatchSnapshot(); + expect(spy).toHaveBeenCalledWith( + 'https://api.localazy.com/projects/_a0000000000000000001/glossary/_a0000000000000000001', + { + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'DELETE', + }, + ); }); }); diff --git a/tests/specs/import.spec.ts b/tests/specs/import.spec.ts index d5c5cdd..ac6c615 100644 --- a/tests/specs/import.spec.ts +++ b/tests/specs/import.spec.ts @@ -1,5 +1,5 @@ -import { beforeEach, describe, expect, test } from 'vitest'; -import { getApiClient, mockAdapter } from '@tests/support'; +import { beforeEach, describe, expect, MockInstance, test, vi } from 'vitest'; +import { getApiClient, getToken } from '@tests/support'; import { fullProject } from '@tests/fixtures'; import { ApiClient, I18nJson, ImportJsonRequest, Project } from '@/main'; @@ -17,9 +17,17 @@ describe('Import', (): void => { test('api.import.json', async (): Promise => { const json: I18nJson = { en: { headers: { person: 'Active person' } } }; const request: ImportJsonRequest = { project, json, fileOptions: { name: 'en.json' } }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); await api.import.json(request); - expect(mockAdapter.history.post.length).toBe(1); - expect(mockAdapter.history.post[0].data).toMatchSnapshot(); + expect(spy).toHaveBeenCalledWith('https://api.localazy.com/projects/_a0000000000000000001/import', { + body: '{"files":[{"name":"en.json","content":{"type":"json","en":{"headers":{"person":"Active person"}}}}]}', + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'POST', + }); }); }); diff --git a/tests/specs/keys.spec.ts b/tests/specs/keys.spec.ts index e49b9d5..69f7dcb 100644 --- a/tests/specs/keys.spec.ts +++ b/tests/specs/keys.spec.ts @@ -1,5 +1,5 @@ -import { beforeEach, describe, expect, test } from 'vitest'; -import { getApiClient, mockAdapter } from '@tests/support'; +import { beforeEach, describe, expect, MockInstance, test, vi } from 'vitest'; +import { getApiClient, getToken } from '@tests/support'; import { fullProject } from '@tests/fixtures'; import { ApiClient, File, Key, KeyDeleteRequest, KeyUpdateRequest, Locales, Project } from '@/main'; @@ -18,19 +18,40 @@ describe('Keys', (): void => { const file: File = await api.files.first({ project }); const keys: Key[] = await api.files.listKeys({ project, file, lang: Locales.ENGLISH }); const request: KeyUpdateRequest = { project, key: keys[0], comment: 'Comment for translators.' }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); await api.keys.update(request); - expect(mockAdapter.history.put.length).toBe(1); - expect(mockAdapter.history.put[0].data).toMatchSnapshot(); + expect(spy).toHaveBeenCalledWith( + 'https://api.localazy.com/projects/_a0000000000000000001/keys/_a0000000000000000001', + { + body: '{"comment":"Comment for translators."}', + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'PUT', + }, + ); }); test('api.keys.delete', async (): Promise => { const file: File = await api.files.first({ project }); const keys: Key[] = await api.files.listKeys({ project, file, lang: Locales.ENGLISH }); const request: KeyDeleteRequest = { project, key: keys[0] }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); await api.keys.delete(request); - expect(mockAdapter.history.delete.length).toBe(1); - expect(mockAdapter.history.delete[0].data).toMatchSnapshot(); + expect(spy).toHaveBeenCalledWith( + 'https://api.localazy.com/projects/_a0000000000000000001/keys/_a0000000000000000001', + { + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'DELETE', + }, + ); }); }); diff --git a/tests/specs/projects.spec.ts b/tests/specs/projects.spec.ts index 88fece1..babb5b0 100644 --- a/tests/specs/projects.spec.ts +++ b/tests/specs/projects.spec.ts @@ -18,6 +18,12 @@ describe('Projects', (): void => { expect(projects[0].name).toBe('Test project'); }); + test('api.projects.list | organization and languages', async (): Promise => { + const projects: Project[] = await api.projects.list({ languages: true, organization: true }); + + expect(projects[0].name).toBe('Test project'); + }); + test('api.projects.first', async (): Promise => { const project: Project = await api.projects.first(); diff --git a/tests/specs/screenshots.spec.ts b/tests/specs/screenshots.spec.ts index 6f4be10..95fa4fd 100644 --- a/tests/specs/screenshots.spec.ts +++ b/tests/specs/screenshots.spec.ts @@ -1,5 +1,5 @@ -import { beforeEach, describe, expect, test } from 'vitest'; -import { getApiClient, readImageFile, mockAdapter } from '@tests/support'; +import { beforeEach, describe, expect, MockInstance, test, vi } from 'vitest'; +import { getApiClient, readImageFile, getToken } from '@tests/support'; import { fullProject } from '@tests/fixtures'; import { ApiClient, @@ -38,21 +38,40 @@ describe('Screenshots', (): void => { test('api.screenshots.create', async (): Promise => { const encodedData: string = readImageFile('./tests/fixtures/screenshot.png', 'image/png'); const request: ScreenshotCreateRequest = { project, encodedData }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); const screenshotId: string = await api.screenshots.create(request); expect(screenshotId).toBe('_a0000000000000000001'); - expect(mockAdapter.history.post.length).toBe(1); - expect(mockAdapter.history.post[0].data).toMatchSnapshot(); + expect(spy).toHaveBeenCalledWith('https://api.localazy.com/projects/_a0000000000000000001/screenshots', { + body: `"${encodedData}"`, + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'POST', + }); }); test('api.screenshots.updateImageData', async (): Promise => { const encodedData: string = readImageFile('./tests/fixtures/screenshot.png', 'image/png'); const screenshots: Screenshot[] = await api.screenshots.list({ project }); const request: ScreenshotUpdateImageDataRequest = { project, screenshot: screenshots[0], encodedData }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); await api.screenshots.updateImageData(request); - expect(mockAdapter.history.post.length).toBe(1); - expect(mockAdapter.history.post[0].data).toMatchSnapshot(); + expect(spy).toHaveBeenCalledWith( + 'https://api.localazy.com/projects/_a0000000000000000001/screenshots/_a0000000000000000001', + { + body: `"${encodedData}"`, + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'POST', + }, + ); }); test('api.screenshots.update', async (): Promise => { @@ -63,18 +82,39 @@ describe('Screenshots', (): void => { comment: 'Hey! Nice screenshot.', tags: ['blue'], }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); await api.screenshots.update(request); - expect(mockAdapter.history.put.length).toBe(1); - expect(mockAdapter.history.put[0].data).toMatchSnapshot(); + expect(spy).toHaveBeenCalledWith( + 'https://api.localazy.com/projects/_a0000000000000000001/screenshots/_a0000000000000000001', + { + body: '{"comment":"Hey! Nice screenshot.","tags":["blue"]}', + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'PUT', + }, + ); }); test('api.screenshots.delete', async (): Promise => { const screenshots: Screenshot[] = await api.screenshots.list({ project }); const request: ScreenshotDeleteRequest = { project, screenshot: screenshots[0] }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); await api.screenshots.delete(request); - expect(mockAdapter.history.delete.length).toBe(1); - expect(mockAdapter.history.delete[0].data).toMatchSnapshot(); + expect(spy).toHaveBeenCalledWith( + 'https://api.localazy.com/projects/_a0000000000000000001/screenshots/_a0000000000000000001', + { + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'DELETE', + }, + ); }); }); diff --git a/tests/specs/webhooks.spec.ts b/tests/specs/webhooks.spec.ts index 58d061f..73c32aa 100644 --- a/tests/specs/webhooks.spec.ts +++ b/tests/specs/webhooks.spec.ts @@ -1,5 +1,5 @@ -import { beforeEach, describe, expect, test } from 'vitest'; -import { getApiClient, mockAdapter } from '@tests/support'; +import { beforeEach, describe, expect, MockInstance, test, vi } from 'vitest'; +import { getApiClient, getToken } from '@tests/support'; import { fullProject } from '@tests/fixtures'; import { ApiClient, Project, Webhook, WebhooksSecret, WebhooksUpdateRequest } from '@/main'; @@ -33,10 +33,19 @@ describe('Webhooks', (): void => { }, ], }; + const spy: MockInstance = vi.spyOn(globalThis, 'fetch'); await api.webhooks.update(request); - expect(mockAdapter.history.post.length).toBe(1); - expect(mockAdapter.history.post[0].data).toMatchSnapshot(); + expect(spy).toHaveBeenCalledWith('https://api.localazy.com/projects/_a0000000000000000001/webhooks', { + // eslint-disable-next-line max-len + body: '{"items":[{"enabled":true,"customId":"1","description":"This is a test webhook","url":"https://example.com/webhook","events":["comment_added","import_finished","import_finished_empty","project_published","tag_promoted"]}]}', + headers: { + Accept: 'application/json', + Authorization: `Bearer ${getToken()}`, + 'Content-Type': 'application/json', + }, + method: 'POST', + }); }); test('api.webhooks.getSecret', async (): Promise => { diff --git a/tests/support/axios-mock-adapter.ts b/tests/support/axios-mock-adapter.ts deleted file mode 100644 index b5d1312..0000000 --- a/tests/support/axios-mock-adapter.ts +++ /dev/null @@ -1,4 +0,0 @@ -import axios from 'axios'; -import MockAdapter from 'axios-mock-adapter'; - -export const mockAdapter: MockAdapter = new MockAdapter(axios, { onNoMatch: 'throwException' }); diff --git a/tests/support/fetch-mock.ts b/tests/support/fetch-mock.ts new file mode 100644 index 0000000..0a2e792 --- /dev/null +++ b/tests/support/fetch-mock.ts @@ -0,0 +1,3 @@ +import fm from 'fetch-mock'; + +export const fetchMock = fm; diff --git a/tests/support/index.ts b/tests/support/index.ts index 457f729..bece2fa 100644 --- a/tests/support/index.ts +++ b/tests/support/index.ts @@ -1,2 +1,2 @@ -export { mockAdapter } from '@tests/support/axios-mock-adapter'; +export { fetchMock } from '@tests/support/fetch-mock'; export * from '@tests/support/utils'; diff --git a/tests/support/utils.ts b/tests/support/utils.ts index 69a70e7..b8599d1 100644 --- a/tests/support/utils.ts +++ b/tests/support/utils.ts @@ -1,4 +1,3 @@ -import axios from 'axios'; import fs from 'fs'; import { ApiClient } from '@/api/api-client'; @@ -8,16 +7,14 @@ export const readImageFile = (path: string, type: string): string => { return `data:${type};base64,${base64}`; }; -export const downloadImageFile = async (url: string, type: string): Promise => { - const { data } = await axios.get(url, { responseType: 'arraybuffer' }); - const base64: string = Buffer.from(data, 'binary').toString('base64'); - return `data:${type};base64,${base64}`; -}; - export const textToUint8Array = (text: string): Uint8Array => Buffer.from(text, 'utf-8'); +export const getToken = (): string => 'project-token'; + +export const getApiUrl = (): string => 'https://api.localazy.com'; + export const getApiClient = (): ApiClient => new ApiClient({ - apiUrl: 'https://api.localazy.com', - authToken: 'project-token', + apiUrl: getApiUrl(), + authToken: getToken(), });