From ca617a120571a4c9a0c4c610e6bae06a7e904630 Mon Sep 17 00:00:00 2001 From: Aakash Singh Date: Fri, 4 Aug 2023 20:32:56 +0530 Subject: [PATCH 1/5] cleanup fireRequests and remove axios --- package-lock.json | 46 +- package.json | 1 - src/App.tsx | 31 +- src/Common/hooks/useMSEplayer.ts | 13 +- .../Assets/AssetType/ONVIFCamera.tsx | 11 +- .../CriticalCare__API.tsx | 6 +- .../Facility/Consultations/LiveFeed.tsx | 8 +- .../Facility/CoverImageEditModal.tsx | 8 +- src/Components/Patient/FileUpload.tsx | 21 +- src/Components/Patient/UpdateStatusDialog.tsx | 51 +- src/Redux/actions.tsx | 8 +- src/Redux/api.tsx | 65 ++- src/Redux/fireRequest.tsx | 437 +++++++----------- 13 files changed, 269 insertions(+), 437 deletions(-) diff --git a/package-lock.json b/package-lock.json index f0fd1245629..02e7114b71a 100644 --- a/package-lock.json +++ b/package-lock.json @@ -22,7 +22,6 @@ "@sentry/browser": "^7.57.0", "@storybook/builder-vite": "^7.0.26", "@yaireo/ui-range": "^2.1.15", - "axios": "^1.4.0", "browser-image-compression": "^2.0.2", "cross-env": "^7.0.3", "date-fns": "^2.30.0", @@ -6667,29 +6666,6 @@ "node": ">=4" } }, - "node_modules/axios": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/axios/-/axios-1.4.0.tgz", - "integrity": "sha512-S4XCWMEmzvo64T9GfvQDOXgYRDJ/wsSZc7Jvdgx5u1sd0JwsuPLqb3SYmusag+edF6ziyMensPVqLTSc1PiSEA==", - "dependencies": { - "follow-redirects": "^1.15.0", - "form-data": "^4.0.0", - "proxy-from-env": "^1.1.0" - } - }, - "node_modules/axios/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/axobject-query": { "version": "3.2.1", "resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-3.2.1.tgz", @@ -9928,25 +9904,6 @@ "node": ">=0.4.0" } }, - "node_modules/follow-redirects": { - "version": "1.15.2", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.2.tgz", - "integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA==", - "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", @@ -17879,7 +17836,8 @@ "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==" + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", + "dev": true }, "node_modules/psl": { "version": "1.9.0", diff --git a/package.json b/package.json index 484d3688a25..500e0ac1d71 100644 --- a/package.json +++ b/package.json @@ -62,7 +62,6 @@ "@sentry/browser": "^7.57.0", "@storybook/builder-vite": "^7.0.26", "@yaireo/ui-range": "^2.1.15", - "axios": "^1.4.0", "browser-image-compression": "^2.0.2", "cross-env": "^7.0.3", "date-fns": "^2.30.0", diff --git a/src/App.tsx b/src/App.tsx index f34da363620..0b0f3740e05 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -11,7 +11,6 @@ import { IConfig } from "./Common/hooks/useConfig"; import { LocalStorageKeys } from "./Common/constants"; import Plausible from "./Components/Common/Plausible"; import SessionRouter from "./Router/SessionRouter"; -import axios from "axios"; import loadable from "@loadable/component"; const Loading = loadable(() => import("./Components/Common/Loading")); @@ -40,23 +39,21 @@ const App: React.FC = () => { const updateRefreshToken = () => { const refresh = localStorage.getItem(LocalStorageKeys.refreshToken); - // const access = localStorage.getItem(LocalStorageKeys.accessToken); - // if (!access && refresh) { - // localStorage.removeItem(LocalStorageKeys.refreshToken); - // document.location.reload(); - // return; - // } - if (!refresh) { - return; - } - axios - .post("/api/v1/auth/token/refresh/", { - refresh, + if (refresh) { + fetch("/api/v1/auth/token/refresh/", { + method: "POST", + headers: { + "Content-Type": "application/json", + Accept: "application/json", + }, + body: JSON.stringify({ refresh }), }) - .then((resp) => { - localStorage.setItem(LocalStorageKeys.accessToken, resp.data.access); - localStorage.setItem(LocalStorageKeys.refreshToken, resp.data.refresh); - }); + .then((res) => res.json()) + .then((resp) => { + localStorage.setItem(LocalStorageKeys.accessToken, resp.access); + localStorage.setItem(LocalStorageKeys.refreshToken, resp.refresh); + }); + } }; useEffect(() => { updateRefreshToken(); diff --git a/src/Common/hooks/useMSEplayer.ts b/src/Common/hooks/useMSEplayer.ts index b14f94a6a43..bb89818775a 100644 --- a/src/Common/hooks/useMSEplayer.ts +++ b/src/Common/hooks/useMSEplayer.ts @@ -1,5 +1,4 @@ import { useEffect, useRef } from "react"; -import axios from "axios"; export interface IAsset { middlewareHostname: string; @@ -59,10 +58,14 @@ const stopStream = (payload: { id: string }, options: IOptions) => { const { id } = payload; ws?.close(); - axios - .post(`https://${middlewareHostname}/stop`, { - id, - }) + fetch(`https://${middlewareHostname}/stop`, { + method: "POST", + headers: { + "Content-Type": "application/json", + Accept: "application/json", + }, + body: JSON.stringify({ id }), + }) .then((res) => options?.onSuccess && options.onSuccess(res)) .catch((err) => options.onError && options.onError(err)); }; diff --git a/src/Components/Assets/AssetType/ONVIFCamera.tsx b/src/Components/Assets/AssetType/ONVIFCamera.tsx index ec11761abf6..b20ed32dd35 100644 --- a/src/Components/Assets/AssetType/ONVIFCamera.tsx +++ b/src/Components/Assets/AssetType/ONVIFCamera.tsx @@ -8,7 +8,6 @@ import { } from "../../../Redux/actions"; import * as Notification from "../../../Utils/Notifications.js"; import { BedModel } from "../../Facility/models"; -import axios from "axios"; import { getCameraConfig } from "../../../Utils/transformUtils"; import CameraConfigure from "../configure/CameraConfigure"; import Loading from "../../Common/Loading"; @@ -110,13 +109,15 @@ const ONVIFCamera = (props: ONVIFCameraProps) => { }; try { setLoadingAddPreset(true); - const presetData = await axios.get( - `https://${facilityMiddlewareHostname}/status?hostname=${config.hostname}&port=${config.port}&username=${config.username}&password=${config.password}` - ); + const preset = await fetch( + `https://${facilityMiddlewareHostname}/preset?hostname=${config.hostname}\ + &port=${config.port}&username=${config.username}&password=${config.password}\ + &preset_name=${newPreset}` + ).then((res) => res.json()); const res: any = await Promise.resolve( dispatch( createAssetBed( - { meta: { ...data, ...presetData.data } }, + { meta: { ...data, ...preset.data } }, assetId, bed?.id as string ) diff --git a/src/Components/CriticalCareRecording/CriticalCare__API.tsx b/src/Components/CriticalCareRecording/CriticalCare__API.tsx index 48b58faa167..423ee4eefd2 100644 --- a/src/Components/CriticalCareRecording/CriticalCare__API.tsx +++ b/src/Components/CriticalCareRecording/CriticalCare__API.tsx @@ -1,4 +1,4 @@ -import { fireRequestV2 } from "../../Redux/fireRequest"; +import { legacyFireRequest } from "../../Redux/fireRequest"; export const loadDailyRound = ( consultationId: string, @@ -6,7 +6,7 @@ export const loadDailyRound = ( successCB: any = () => null, errorCB: any = () => null ) => { - fireRequestV2("getDailyReport", [], {}, successCB, errorCB, { + legacyFireRequest("getDailyReport", [], {}, successCB, errorCB, { consultationId, id, }); @@ -19,7 +19,7 @@ export const updateDailyRound = ( successCB: any = () => null, errorCB: any = () => null ) => { - fireRequestV2("updateDailyRound", [], params, successCB, errorCB, { + legacyFireRequest("updateDailyRound", [], params, successCB, errorCB, { consultationId, id, }); diff --git a/src/Components/Facility/Consultations/LiveFeed.tsx b/src/Components/Facility/Consultations/LiveFeed.tsx index 027cbe2deb5..e2399879791 100644 --- a/src/Components/Facility/Consultations/LiveFeed.tsx +++ b/src/Components/Facility/Consultations/LiveFeed.tsx @@ -14,7 +14,6 @@ import { import { useFeedPTZ } from "../../../Common/hooks/useFeedPTZ"; import * as Notification from "../../../Utils/Notifications.js"; import { FeedCameraPTZHelpButton } from "./Feed"; -import { AxiosError } from "axios"; import { BedSelect } from "../../Common/BedSelect"; import { BedModel } from "../models"; import useWindowDimensions from "../../../Common/hooks/useWindowDimensions"; @@ -93,10 +92,9 @@ const LiveFeed = (props: any) => { setPresets(resp); }, onError: (resp) => { - resp instanceof AxiosError && - Notification.Error({ - msg: "Camera is offline", - }); + Notification.Error({ + msg: "Camera is offline", + }); }, }); diff --git a/src/Components/Facility/CoverImageEditModal.tsx b/src/Components/Facility/CoverImageEditModal.tsx index 988bc288f58..d365ed5aba9 100644 --- a/src/Components/Facility/CoverImageEditModal.tsx +++ b/src/Components/Facility/CoverImageEditModal.tsx @@ -1,4 +1,3 @@ -import axios from "axios"; import { ChangeEventHandler, useCallback, @@ -110,17 +109,18 @@ const CoverImageEditModal = ({ setIsUploading(true); try { - const response = await axios.post( + const response = await fetch( `/api/v1/facility/${facility.id}/cover_image/`, - formData, { + method: "POST", headers: { "Content-Type": "multipart/form-data", Authorization: "Bearer " + localStorage.getItem(LocalStorageKeys.accessToken), }, + body: formData, } - ); + ) if (response.status === 200) { Success({ msg: "Cover image updated." }); window.location.reload(); diff --git a/src/Components/Patient/FileUpload.tsx b/src/Components/Patient/FileUpload.tsx index 2f85cdd7bb7..0a859880257 100644 --- a/src/Components/Patient/FileUpload.tsx +++ b/src/Components/Patient/FileUpload.tsx @@ -1,4 +1,3 @@ -import axios from "axios"; import CircularProgress from "../Common/components/CircularProgress"; import loadable from "@loadable/component"; import React, { useCallback, useState, useEffect, useRef } from "react"; @@ -32,6 +31,7 @@ import { NonReadOnlyUsers } from "../../Utils/AuthorizeFor"; import AuthorizedChild from "../../CAREUI/misc/AuthorizedChild"; import Page from "../Common/components/Page"; import FilePreviewDialog from "../Common/FilePreviewDialog"; +import { uploadFile } from "../../Redux/fireRequest"; const Loading = loadable(() => import("../Common/Loading")); @@ -928,24 +928,10 @@ export const FileUpload = (props: FileUploadProps) => { if (!f) return; const newFile = new File([f], `${internal_name}`); - const config = { - headers: { - "Content-type": contentType, - "Content-disposition": "inline", - }, - onUploadProgress: (progressEvent: any) => { - const percentCompleted = Math.round( - (progressEvent.loaded * 100) / progressEvent.total - ); - setUploadPercent(percentCompleted); - }, - }; return new Promise((resolve, reject) => { - axios - .put(url, newFile, config) + uploadFile(url, newFile, contentType, setUploadPercent) .then(() => { setUploadStarted(false); - // setUploadSuccess(true); setFile(null); setUploadFileName(""); setReload(!reload); @@ -1049,8 +1035,7 @@ export const FileUpload = (props: FileUploadProps) => { }, }; - axios - .put(url, newFile, config) + uploadFile(url, newFile, "audio/mpeg", setUploadPercent) .then(() => { setAudioUploadStarted(false); // setUploadSuccess(true); diff --git a/src/Components/Patient/UpdateStatusDialog.tsx b/src/Components/Patient/UpdateStatusDialog.tsx index 042937e464e..8edbb9effc4 100644 --- a/src/Components/Patient/UpdateStatusDialog.tsx +++ b/src/Components/Patient/UpdateStatusDialog.tsx @@ -1,5 +1,4 @@ import React, { useEffect, useState, useReducer } from "react"; -import axios from "axios"; import { SAMPLE_TEST_STATUS, SAMPLE_TEST_RESULT, @@ -9,7 +8,7 @@ import { SampleTestModel } from "./models"; import * as Notification from "../../Utils/Notifications.js"; import { createUpload, editUpload } from "../../Redux/actions"; import { useDispatch } from "react-redux"; -import { header_content_type, LinearProgressWithLabel } from "./FileUpload"; +import { FileUpload, header_content_type, LinearProgressWithLabel } from "./FileUpload"; import { Submit } from "../Common/components/ButtonV2"; import CareIcon from "../../CAREUI/icons/CareIcon"; import ConfirmDialog from "../Common/ConfirmDialog"; @@ -18,6 +17,7 @@ import { FieldChangeEvent } from "../Form/FormFields/Utils"; import TextFormField from "../Form/FormFields/TextFormField"; import CheckBoxFormField from "../Form/FormFields/CheckBoxFormField"; import { useTranslation } from "react-i18next"; +import { uploadFile } from "../../Redux/fireRequest"; interface Props { sample: SampleTestModel; @@ -105,38 +105,23 @@ const UpdateStatusDialog = (props: Props) => { if (f === undefined) return; const newFile = new File([f], `${internal_name}`); - const config = { - headers: { - "Content-type": contentType, - "Content-disposition": "inline", - }, - onUploadProgress: (progressEvent: any) => { - const percentCompleted = Math.round( - (progressEvent.loaded * 100) / progressEvent.total - ); - setUploadPercent(percentCompleted); - }, - }; - axios - .put(url, newFile, config) - .then(() => { - setUploadStarted(false); - setUploadDone(true); - redux_dispatch( - editUpload( - { upload_completed: true }, - response.data.id, - "SAMPLE_MANAGEMENT", - sample.id?.toString() ?? "" - ) - ); - Notification.Success({ - msg: "File Uploaded Successfully", - }); - }) - .catch(() => { - setUploadStarted(false); + uploadFile(url, newFile, contentType, setUploadPercent).then(() => { + setUploadStarted(false); + setUploadDone(true); + redux_dispatch( + editUpload( + { upload_completed: true }, + response.data.id, + "SAMPLE_MANAGEMENT", + sample.id?.toString() ?? "" + ) + ); + Notification.Success({ + msg: "File Uploaded Successfully", }); + }).catch(() => { + setUploadStarted(false); + }); }; const onFileChange = (e: React.ChangeEvent): any => { diff --git a/src/Redux/actions.tsx b/src/Redux/actions.tsx index ebfb824e6fd..3e2b6825fc4 100644 --- a/src/Redux/actions.tsx +++ b/src/Redux/actions.tsx @@ -3,12 +3,12 @@ import { MedicineAdministrationRecord, Prescription, } from "../Components/Medicine/models"; -import { fireRequest, fireRequestForFiles } from "./fireRequest"; +import { fireRequest } from "./fireRequest"; import { ICreateHealthIdRequest } from "../Components/ABDM/models"; export const getConfig = () => { - return fireRequestForFiles("config"); + return fireRequest("config"); }; // User export const postLogin = (params: object) => { @@ -723,10 +723,6 @@ export const retrieveUpload = (params: object, fileId: string) => { return fireRequest("retrieveUpload", [], params, { fileId: fileId }); }; -export const retrieveUploadFilesURL = (params: object, fileId: string) => { - return fireRequestForFiles("retrieveUpload", [], params, { fileId: fileId }); -}; - export const editUpload = ( params: object, fileId: string, diff --git a/src/Redux/api.tsx b/src/Redux/api.tsx index 86a85ae8a3b..905ca83dc92 100644 --- a/src/Redux/api.tsx +++ b/src/Redux/api.tsx @@ -23,12 +23,13 @@ const routes: Routes = { }, token_refresh: { - path: "/api/v1/auth/token/refresh", + path: "/api/v1/auth/token/refresh/", method: "POST", + noAuth: true, }, token_verify: { - path: "/api/v1/auth/token/verify", + path: "/api/v1/auth/token/verify/", method: "POST", }, @@ -100,7 +101,7 @@ const routes: Routes = { }, updateUser: { - path: "/api/v1/users", + path: "/api/v1/users/", method: "PUT", }, @@ -110,7 +111,7 @@ const routes: Routes = { }, deleteUser: { - path: "/api/v1/users", + path: "/api/v1/users/", method: "DELETE", }, @@ -149,7 +150,7 @@ const routes: Routes = { }, getAllFacilities: { - path: "/api/v1/getallfacilities", + path: "/api/v1/getallfacilities/", }, createFacility: { @@ -166,12 +167,12 @@ const routes: Routes = { }, updateFacility: { - path: "/api/v1/facility", + path: "/api/v1/facility/", method: "PUT", }, partialUpdateFacility: { - path: "/api/v1/facility", + path: "/api/v1/facility/", method: "PATCH", }, @@ -284,7 +285,7 @@ const routes: Routes = { // Download Api deleteFacility: { - path: "/api/v1/facility", + path: "/api/v1/facility/", method: "DELETE", }, @@ -388,17 +389,17 @@ const routes: Routes = { }, updateCapacity: { - path: "/api/v1/facility/{facilityId}/capacity", + path: "/api/v1/facility/{facilityId}/capacity/", method: "PUT", }, updateDoctor: { - path: "/api/v1/facility/{facilityId}/hospital_doctor", + path: "/api/v1/facility/{facilityId}/hospital_doctor/", method: "PUT", }, deleteDoctor: { - path: "/api/v1/facility/{facilityId}/hospital_doctor", + path: "/api/v1/facility/{facilityId}/hospital_doctor/", method: "DELETE", }, @@ -415,16 +416,10 @@ const routes: Routes = { path: "/api/v1/facility/{facilityId}/patient_stats/{id}/", }, - // //Care Center - // createCenter: { - // path: "/api/v1/carecenter/", - // method: 'POST' - // } - // Patient searchPatient: { - path: "/api/v1/patient/search", + path: "/api/v1/patient/search/", }, patientList: { path: "/api/v1/patient/", @@ -480,7 +475,7 @@ const routes: Routes = { }, deleteExternalResult: { - path: "/api/v1/external_result", + path: "/api/v1/external_result/", method: "DELETE", }, @@ -548,7 +543,7 @@ const routes: Routes = { path: "/api/v1/test_sample/", }, getTestSample: { - path: "/api/v1/test_sample", + path: "/api/v1/test_sample/", }, patchSample: { path: "/api/v1/test_sample/{id}/", @@ -564,26 +559,26 @@ const routes: Routes = { method: "POST", }, getInventoryLog: { - path: "/api/v1/facility", + path: "/api/v1/facility/", }, setMinQuantity: { path: "/api/v1/facility/{facilityId}/min_quantity/", method: "POST", }, getMinQuantity: { - path: "/api/v1/facility", + path: "/api/v1/facility/", method: "GET", }, updateMinQuantity: { - path: "/api/v1/facility/{facilityId}/min_quantity/{inventoryId}", + path: "/api/v1/facility/{facilityId}/min_quantity/{inventoryId}/", method: "PATCH", }, getInventorySummary: { - path: "/api/v1/facility", + path: "/api/v1/facility/", method: "GET", }, getItemName: { - path: "/api/v1/items", + path: "/api/v1/items/", method: "GET", }, flagInventoryItem: { @@ -599,7 +594,7 @@ const routes: Routes = { method: "POST", }, dischargeSummaryPreview: { - path: "/api/v1/consultation/{external_id}/preview_discharge_summary", + path: "/api/v1/consultation/{external_id}/preview_discharge_summary/", method: "GET", }, dischargeSummaryEmail: { @@ -622,7 +617,7 @@ const routes: Routes = { method: "GET", }, updateUserDetails: { - path: "/api/v1/users", + path: "/api/v1/users/", method: "PUT", }, @@ -632,11 +627,11 @@ const routes: Routes = { method: "POST", }, updateShift: { - path: "/api/v1/shift", + path: "/api/v1/shift/", method: "PUT", }, deleteShiftRecord: { - path: "/api/v1/shift", + path: "/api/v1/shift/", method: "DELETE", }, listShiftRequests: { @@ -705,7 +700,7 @@ const routes: Routes = { method: "GET", }, listInvestigationGroups: { - path: "/api/v1/investigation/group", + path: "/api/v1/investigation/group/", method: "GET", }, createInvestigation: { @@ -744,11 +739,11 @@ const routes: Routes = { method: "POST", }, updateResource: { - path: "/api/v1/resource", + path: "/api/v1/resource/", method: "PUT", }, deleteResourceRecord: { - path: "/api/v1/resource", + path: "/api/v1/resource/", method: "DELETE", }, listResourceRequests: { @@ -774,7 +769,7 @@ const routes: Routes = { // Assets endpoints listAssets: { - path: "/api/v1/asset", + path: "/api/v1/asset/", method: "GET", }, createAsset: { @@ -813,7 +808,7 @@ const routes: Routes = { method: "GET", }, getAssetTransaction: { - path: "/api/v1/asset_transaction/{id}", + path: "/api/v1/asset_transaction/{id}/", method: "GET", }, @@ -895,7 +890,7 @@ const routes: Routes = { method: "GET", }, getAssetAvailability: { - path: "/api/v1/asset_availability/{id}", + path: "/api/v1/asset_availability/{id}/", method: "GET", }, diff --git a/src/Redux/fireRequest.tsx b/src/Redux/fireRequest.tsx index 9859ea540b6..e863641ff8c 100644 --- a/src/Redux/fireRequest.tsx +++ b/src/Redux/fireRequest.tsx @@ -1,12 +1,9 @@ import * as Notification from "../Utils/Notifications.js"; -import { isEmpty, omitBy } from "lodash"; +import { LocalStorageKeys } from "../Common/constants.js"; +import api from "./api.js"; -import { LocalStorageKeys } from "../Common/constants"; -import api from "./api"; -import axios from "axios"; - -const requestMap: any = api; +const requestMap = api; export const actions = { FETCH_REQUEST: "FETCH_REQUEST", FETCH_REQUEST_SUCCESS: "FETCH_REQUEST_SUCCESS", @@ -14,6 +11,7 @@ export const actions = { SET_DATA: "SET_DATA", }; +// const isRunning: { [key: string]: AbortController } = {}; const isRunning: any = {}; export const setStoreData = (key: string, value: any) => { @@ -47,29 +45,30 @@ export const fetchResponseSuccess = (key: string, data: any) => { }; }; +export interface ResponseWrapper extends Response { + data: any; +} + export const fireRequest = ( key: string, path: any = [], params: any = {}, pathParam?: any, altKey?: string, - suppressNotif?: boolean + suppressNotif: boolean = false ) => { return (dispatch: any) => { // cancel previous api call - if (isRunning[altKey ? altKey : key]) { - isRunning[altKey ? altKey : key].cancel(); + const requestKey = altKey || key; + if (isRunning[requestKey]) { + isRunning[requestKey].abort(); } - isRunning[altKey ? altKey : key] = axios.CancelToken.source(); + const controller = new AbortController(); + isRunning[requestKey] = controller; + // get api url / method const request = Object.assign({}, requestMap[key]); - if (path.length > 0) { - request.path += "/" + path.join("/"); - } - // add trailing slash to path before query paramaters - if (request.path.slice(-1) !== "/" && request.path.indexOf("?") === -1) { - request.path += "/"; - } + if (request.method === undefined || request.method === "GET") { request.method = "GET"; let qString = ""; @@ -82,236 +81,127 @@ export const fireRequest = ( request.path += `?${qString}`; } } + // set dynamic params in the URL if (pathParam) { Object.keys(pathParam).forEach((param: any) => { request.path = request.path.replace(`{${param}}`, pathParam[param]); }); } - - // set authorization header in the request header - const config: any = { - headers: {}, + const headers: { [key: string]: string } = { + "Content-Type": "application/json", + Accept: "application/json", }; - if (!request.noAuth && localStorage.getItem(LocalStorageKeys.accessToken)) { - config.headers["Authorization"] = + if (!request.noAuth) { + headers["Authorization"] = "Bearer " + localStorage.getItem(LocalStorageKeys.accessToken); - } else { - // TODO: get access token } - const axiosApiCall: any = axios.create(config); dispatch(fetchDataRequest(key)); - return axiosApiCall[request.method.toLowerCase()](request.path, { - ...params, - cancelToken: isRunning[altKey ? altKey : key].token, + return fetch(request.path, { + method: request.method, + headers, + credentials: "include", + cache: "default", + redirect: "follow", + signal: controller.signal, + body: request.method === "GET" ? undefined : JSON.stringify(params), }) - .then((response: any) => { - dispatch(fetchResponseSuccess(key, response.data)); - return response; - }) - .catch((error: any) => { - dispatch(fetchDataRequestError(key, error)); - - if (!(suppressNotif ?? false) && error.response) { - // temporarily don't show invalid phone number error on duplicate patient check - if (error.response.status === 400 && key === "searchPatient") { - return; - } - - // deleteUser: 404 is for permission denied - if (error.response.status === 404 && key === "deleteUser") { - Notification.Error({ - msg: "Permission denied!", - }); - return; - } - - // currentUser is ignored because on the first page load - // 403 error is displayed for invalid credential. - if (error.response.status === 403 && key === "currentUser") { - if (localStorage.getItem(LocalStorageKeys.accessToken)) { - localStorage.removeItem(LocalStorageKeys.accessToken); - } - return; - } + .then(async (res: Response) => { + const response: ResponseWrapper = res as ResponseWrapper; + try { + response.data = await response.json(); + } catch (error) { + console.error(error); + response.data = {}; + } - // 400 Bad Request Error - if (error.response.status === 400 || error.response.status === 406) { - Notification.BadRequest({ - errs: error.response.data, - }); - return error.response; - } + if (response.ok) { + dispatch(fetchResponseSuccess(key, response.data)); + return response; + } - // 4xx Errors - if (error.response.status > 400 && error.response.status < 500) { - if (error.response.data && error.response.data.detail) { - if (error.response.data.code === "token_not_valid") { - window.location.href = "/session-expired"; - } - Notification.Error({ - msg: error.response.data.detail, - }); - } else { - Notification.Error({ - msg: "Something went wrong...!", - }); - } - if (error.response.status === 429) { - return error.response; - } - return; - } + dispatch(fetchDataRequestError(key, response)); + if (suppressNotif) { + return response; + } - // 5xx Errors - if (error.response.status >= 500 && error.response.status <= 599) { - Notification.Error({ - msg: "Something went wrong...!", - }); - return; - } - } else { - return error.response; + if (response.status === 400 || response.status === 406) { + Notification.BadRequest({ + errs: response.data, + }); + return response; } - }); - }; -}; -export const fireRequestV2 = ( - key: string, - path: any = [], - params: any = {}, - successCallback: any = () => undefined, - errorCallback: any = () => undefined, - pathParam?: any, - altKey?: string -) => { - // cancel previous api call - if (isRunning[altKey ? altKey : key]) { - isRunning[altKey ? altKey : key].cancel(); - } - isRunning[altKey ? altKey : key] = axios.CancelToken.source(); - // get api url / method - const request = Object.assign({}, requestMap[key]); - if (path.length > 0) { - request.path += "/" + path.join("/"); - } - if (request.method === undefined || request.method === "GET") { - request.method = "GET"; - const qs = new URLSearchParams(omitBy(params, isEmpty)).toString(); - if (qs !== "") { - request.path += `?${qs}`; - } - } - // set dynamic params in the URL - if (pathParam) { - Object.keys(pathParam).forEach((param: any) => { - request.path = request.path.replace(`{${param}}`, pathParam[param]); - }); - } - - // set authorization header in the request header - const config: any = { - headers: {}, - }; - if (!request.noAuth && localStorage.getItem(LocalStorageKeys.accessToken)) { - config.headers["Authorization"] = - "Bearer " + localStorage.getItem(LocalStorageKeys.accessToken); - } - const axiosApiCall: any = axios.create(config); - - fetchDataRequest(key); - return axiosApiCall[request.method.toLowerCase()](request.path, { - ...params, - cancelToken: isRunning[altKey ? altKey : key].token, - }) - .then((response: any) => { - successCallback(response.data); - }) - .catch((error: any) => { - errorCallback(error); - if (error.response) { - // temporarily don't show invalid phone number error on duplicate patient check - if (error.response.status === 400 && key === "searchPatient") { + if (response.status === 403 && key === "currentUser") { + localStorage.removeItem(LocalStorageKeys.accessToken); return; } - // deleteUser: 404 is for permission denied - if (error.response.status === 404 && key === "deleteUser") { + if (response.status === 404 && key === "deleteUser") { Notification.Error({ msg: "Permission denied!", }); + return response; } - // currentUser is ignored because on the first page load - // 403 error is displayed for invalid credential. - if (error.response.status === 403 && key === "currentUser") { - if (localStorage.getItem(LocalStorageKeys.accessToken)) { - localStorage.removeItem(LocalStorageKeys.accessToken); - } + if (response.status === 429) { + return response; } - // 400 Bad Request Error - if (error.response.status === 400 || error.response.status === 406) { - Notification.BadRequest({ - errs: error.response.data, - }); - } - - // 4xx Errors - if (error.response.status > 400 && error.response.status < 500) { - if (error.response.data && error.response.data.detail) { - Notification.Error({ - msg: error.response.data.detail, - }); - } else { - Notification.Error({ - msg: "Something went wrong...!", - }); + if (response.status > 400 && response.status < 500) { + if (response?.data?.code === "token_not_valid") { + window.location.href = "/session-expired"; } - if (error.response.status === 429) { - return error.response; - } - return; - } - - // 5xx Errors - if (error.response.status >= 500 && error.response.status <= 599) { Notification.Error({ - msg: "Something went wrong...!", + msg: response?.data?.detail || "Something went wrong...!", }); - return; } - } - }); + return response; + }) + .catch((error: any) => { + console.error(error); + dispatch(fetchDataRequestError(key, error)); + + return; + }); + }; }; -export const fireRequestForFiles = ( +export const legacyFireRequest = ( key: string, path: any = [], params: any = {}, + successCallback: any = () => undefined, + errorCallback: any = () => undefined, pathParam?: any, altKey?: string ) => { return (dispatch: any) => { // cancel previous api call - if (isRunning[altKey ? altKey : key]) { - isRunning[altKey ? altKey : key].cancel(); + const requestKey = altKey || key; + if (isRunning[requestKey]) { + isRunning[requestKey].abort(); } - isRunning[altKey ? altKey : key] = axios.CancelToken.source(); + const controller = new AbortController(); + isRunning[requestKey] = controller; + // get api url / method const request = Object.assign({}, requestMap[key]); - if (path.length > 0) { - request.path += "/" + path.join("/"); - } + if (request.method === undefined || request.method === "GET") { request.method = "GET"; - const qs = new URLSearchParams(omitBy(params, isEmpty)).toString(); - if (qs !== "") { - request.path += `?${qs}`; + let qString = ""; + Object.keys(params).forEach((param: any) => { + if (params[param] !== undefined && params[param] !== "") { + qString += `${param}=${encodeURIComponent(params[param])}&`; + } + }); + if (qString !== "") { + request.path += `?${qString}`; } } + // set dynamic params in the URL if (pathParam) { Object.keys(pathParam).forEach((param: any) => { @@ -319,80 +209,105 @@ export const fireRequestForFiles = ( }); } - // set authorization header in the request header - const config: any = { - headers: { - "Content-type": "application/pdf", - "Content-disposition": "inline", - }, + const headers: { [key: string]: string } = { + "Content-Type": "application/json", + Accept: "application/json", }; - // Content-Type: application/pdf - // Content-Disposition: inline; filename="filename.pdf" - if (!request.noAuth && localStorage.getItem(LocalStorageKeys.accessToken)) { - config.headers["Authorization"] = + if (!request.noAuth) { + headers["Authorization"] = "Bearer " + localStorage.getItem(LocalStorageKeys.accessToken); } - const axiosApiCall: any = axios.create(config); - dispatch(fetchDataRequest(key)); - return axiosApiCall[request.method.toLowerCase()](request.path, { - ...params, - cancelToken: isRunning[altKey ? altKey : key].token, + fetchDataRequest(key); + return fetch(request.path, { + method: request.method, + headers, + credentials: "include", + cache: "default", + redirect: "follow", + signal: controller.signal, + body: request.method === "POST" ? JSON.stringify(params) : undefined, }) - .then((response: any) => { - dispatch(fetchResponseSuccess(key, response.data)); - return response; - }) - .catch((error: any) => { - dispatch(fetchDataRequestError(key, error)); + .then(async (res: Response) => { + const response: ResponseWrapper = res as ResponseWrapper; + + try { + response.data = await response.json(); + } catch (error) { + console.error(error); + response.data = {}; + } - if (error.response) { - // temporarily don't show invalid phone number error on duplicate patient check - if (error.response.status === 400 && key === "searchPatient") { - return; - } + if (response.ok) { + successCallback(response.data); + return response; + } - // currentUser is ignored because on the first page load - // 403 error is displayed for invalid credential. - if (error.response.status === 403 && key === "currentUser") { - if (localStorage.getItem(LocalStorageKeys.accessToken)) { - localStorage.removeItem(LocalStorageKeys.accessToken); - } - return; - } + errorCallback(response); - // 400 Bad Request Error - if (error.response.status === 400 || error.response.status === 406) { - Notification.BadRequest({ - errs: error.response.data, - }); - return error.response; - } + if (response.status === 400 || response.status === 406) { + Notification.BadRequest({ + errs: response.data, + }); + return response; + } - // 4xx Errors - if (error.response.status > 400 && error.response.status < 500) { - if (error.response.status === 429) { - return error.response; - } else if (error.response.data && error.response.data.detail) { - Notification.Error({ - msg: error.response.data.detail, - }); - } else { - Notification.Error({ - msg: "Something went wrong...!", - }); - } - return; - } + if (response.status === 403 && key === "currentUser") { + localStorage.removeItem(LocalStorageKeys.accessToken); + return; + } + + if (response.status === 404 && key === "deleteUser") { + Notification.Error({ + msg: "Permission denied!", + }); + return response; + } + + if (response.status === 429) { + return response; + } - // 5xx Errors - if (error.response.status >= 500 && error.response.status <= 599) { - Notification.Error({ - msg: "Something went wrong...!", - }); - return; + if (response.status > 400 && response.status < 500) { + if (response?.data?.code === "token_not_valid") { + window.location.href = "/session-expired"; } + Notification.Error({ + msg: response?.data?.detail || "Something went wrong...!", + }); } + return response; + }) + .catch((error: any) => { + console.error(error); + errorCallback(error); + + return; }); }; }; + +export const uploadFile = async ( + url: string, + file: File, + contentType: string, + uploadProgressCB: (progress: number) => void +) => { + const xhr = new XMLHttpRequest(); + return new Promise((resolve, reject) => { + xhr.upload.onprogress = (event) => { + if (event.lengthComputable) { + uploadProgressCB((event.loaded / event.total) * 100); + } + }; + xhr.onloadend = () => { + if (xhr.readyState === 4 && xhr.status === 200){ + return resolve(xhr.response); + } + reject("failed to upload file"); + }; + xhr.open("PUT", url, true); + xhr.setRequestHeader("Content-Type", contentType); + xhr.send(file); + }); +}; From 42b6af5450b2b58f85bc0819ea618e792579bb7b Mon Sep 17 00:00:00 2001 From: rithviknishad Date: Wed, 27 Sep 2023 16:52:29 +0530 Subject: [PATCH 2/5] fix `path` being ignored --- src/Redux/fireRequest.tsx | 21 ++++++++++++++------- 1 file changed, 14 insertions(+), 7 deletions(-) diff --git a/src/Redux/fireRequest.tsx b/src/Redux/fireRequest.tsx index e863641ff8c..f9b378a37b7 100644 --- a/src/Redux/fireRequest.tsx +++ b/src/Redux/fireRequest.tsx @@ -1,9 +1,8 @@ import * as Notification from "../Utils/Notifications.js"; import { LocalStorageKeys } from "../Common/constants.js"; -import api from "./api.js"; +import routes from "./api.js"; -const requestMap = api; export const actions = { FETCH_REQUEST: "FETCH_REQUEST", FETCH_REQUEST_SUCCESS: "FETCH_REQUEST_SUCCESS", @@ -55,7 +54,7 @@ export const fireRequest = ( params: any = {}, pathParam?: any, altKey?: string, - suppressNotif: boolean = false + suppressNotif = false ) => { return (dispatch: any) => { // cancel previous api call @@ -67,7 +66,11 @@ export const fireRequest = ( isRunning[requestKey] = controller; // get api url / method - const request = Object.assign({}, requestMap[key]); + const request = Object.assign({}, (routes as any)[key]); + + if (path.length > 0) { + request.path += "/" + path.join("/"); + } if (request.method === undefined || request.method === "GET") { request.method = "GET"; @@ -177,7 +180,7 @@ export const legacyFireRequest = ( pathParam?: any, altKey?: string ) => { - return (dispatch: any) => { + return () => { // cancel previous api call const requestKey = altKey || key; if (isRunning[requestKey]) { @@ -187,7 +190,11 @@ export const legacyFireRequest = ( isRunning[requestKey] = controller; // get api url / method - const request = Object.assign({}, requestMap[key]); + const request = Object.assign({}, (routes as any)[key]); + + if (path.length > 0) { + request.path += "/" + path.join("/"); + } if (request.method === undefined || request.method === "GET") { request.method = "GET"; @@ -301,7 +308,7 @@ export const uploadFile = async ( } }; xhr.onloadend = () => { - if (xhr.readyState === 4 && xhr.status === 200){ + if (xhr.readyState === 4 && xhr.status === 200) { return resolve(xhr.response); } reject("failed to upload file"); From 7222cb4069da026afdea03f6e5bd186aa6bf56ec Mon Sep 17 00:00:00 2001 From: Aakash Singh Date: Thu, 28 Sep 2023 09:40:50 +0530 Subject: [PATCH 3/5] avoid auth header for password reset --- src/Redux/api.tsx | 1 + 1 file changed, 1 insertion(+) diff --git a/src/Redux/api.tsx b/src/Redux/api.tsx index b0d7df5ef51..2667521950b 100644 --- a/src/Redux/api.tsx +++ b/src/Redux/api.tsx @@ -58,6 +58,7 @@ const routes = { forgotPassword: { path: "/api/v1/password_reset/", method: "POST", + noAuth: true, }, updatePassword: { From c0a4d3860da3e44dae01a1e60535ec084c43036e Mon Sep 17 00:00:00 2001 From: Aakash Singh Date: Thu, 28 Sep 2023 10:05:04 +0530 Subject: [PATCH 4/5] fix duplicate slash in path --- src/Redux/fireRequest.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Redux/fireRequest.tsx b/src/Redux/fireRequest.tsx index f9b378a37b7..51907affab8 100644 --- a/src/Redux/fireRequest.tsx +++ b/src/Redux/fireRequest.tsx @@ -69,7 +69,7 @@ export const fireRequest = ( const request = Object.assign({}, (routes as any)[key]); if (path.length > 0) { - request.path += "/" + path.join("/"); + request.path += path.join("/"); } if (request.method === undefined || request.method === "GET") { @@ -193,7 +193,7 @@ export const legacyFireRequest = ( const request = Object.assign({}, (routes as any)[key]); if (path.length > 0) { - request.path += "/" + path.join("/"); + request.path += path.join("/"); } if (request.method === undefined || request.method === "GET") { From fee12b389207db8f8e2d47346050300d570436ad Mon Sep 17 00:00:00 2001 From: rithviknishad Date: Thu, 28 Sep 2023 17:29:45 +0530 Subject: [PATCH 5/5] fix trailing slash --- src/Redux/api.tsx | 4 ++-- src/Redux/fireRequest.tsx | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/Redux/api.tsx b/src/Redux/api.tsx index 2667521950b..a8ba0068e0b 100644 --- a/src/Redux/api.tsx +++ b/src/Redux/api.tsx @@ -835,11 +835,11 @@ const routes = { method: "GET", }, getAssetService: { - path: "/api/v1/asset/{asset_external_id}/service_records/{external_id}", + path: "/api/v1/asset/{asset_external_id}/service_records/{external_id}/", method: "GET", }, updateAssetService: { - path: "/api/v1/asset/{asset_external_id}/service_records/{external_id}", + path: "/api/v1/asset/{asset_external_id}/service_records/{external_id}/", method: "PUT", }, diff --git a/src/Redux/fireRequest.tsx b/src/Redux/fireRequest.tsx index 51907affab8..d570d8bdcbb 100644 --- a/src/Redux/fireRequest.tsx +++ b/src/Redux/fireRequest.tsx @@ -69,7 +69,7 @@ export const fireRequest = ( const request = Object.assign({}, (routes as any)[key]); if (path.length > 0) { - request.path += path.join("/"); + request.path += path.join("/") + "/"; } if (request.method === undefined || request.method === "GET") { @@ -193,7 +193,7 @@ export const legacyFireRequest = ( const request = Object.assign({}, (routes as any)[key]); if (path.length > 0) { - request.path += path.join("/"); + request.path += path.join("/") + "/"; } if (request.method === undefined || request.method === "GET") {