From f7ac211a193a5846b30f9c370a31f0391b0ef044 Mon Sep 17 00:00:00 2001 From: Mohammed Nihal <57055998+nihal467@users.noreply.github.com> Date: Thu, 9 Nov 2023 17:57:03 +0530 Subject: [PATCH 1/8] New Cypress Test | Functionality test of Linking Skill to a User | User Tab (#6568) * linking skill to a user * fixed failure * removed duplicate code in asset module * minor-fix-assertion --- cypress/e2e/assets_spec/assets_manage.cy.ts | 17 ------- cypress/e2e/users_spec/user_manage.cy.ts | 49 +++++++++++++++++-- cypress/pageobject/Users/ManageUserPage.ts | 34 +++++++++++++ src/Components/Common/SkillSelect.tsx | 3 ++ src/Components/Form/AutoCompleteAsync.tsx | 3 ++ src/Components/Users/SkillsSlideOver.tsx | 4 +- .../Users/SkillsSlideOverComponents.tsx | 1 + src/Components/Users/UserProfile.tsx | 5 +- 8 files changed, 94 insertions(+), 22 deletions(-) diff --git a/cypress/e2e/assets_spec/assets_manage.cy.ts b/cypress/e2e/assets_spec/assets_manage.cy.ts index 8b440f68761..164bc239253 100644 --- a/cypress/e2e/assets_spec/assets_manage.cy.ts +++ b/cypress/e2e/assets_spec/assets_manage.cy.ts @@ -117,23 +117,6 @@ describe("Asset", () => { assetPage.verifyDeleteStatus(); }); - it("Verify Facility Asset Page Redirection", () => { - cy.visit("/facility"); - assetSearchPage.typeSearchKeyword(fillFacilityName); - assetSearchPage.pressEnter(); - facilityPage.verifyFacilityBadgeContent(fillFacilityName); - facilityPage.visitAlreadyCreatedFacility(); - facilityPage.clickManageFacilityDropdown(); - facilityPage.clickCreateAssetFacilityOption(); - facilityPage.verifyfacilitycreateassetredirection(); - facilityPage.verifyassetfacilitybackredirection(); - facilityPage.clickManageFacilityDropdown(); - facilityPage.clickviewAssetFacilityOption(); - facilityPage.verifyfacilityviewassetredirection(); - assetFilters.assertFacilityText(fillFacilityName); - facilityPage.verifyassetfacilitybackredirection(); - }); - afterEach(() => { cy.saveLocalStorage(); }); diff --git a/cypress/e2e/users_spec/user_manage.cy.ts b/cypress/e2e/users_spec/user_manage.cy.ts index b7e1e8b53e9..db865dc8126 100644 --- a/cypress/e2e/users_spec/user_manage.cy.ts +++ b/cypress/e2e/users_spec/user_manage.cy.ts @@ -10,10 +10,13 @@ describe("Manage User", () => { const usernametolinkfacilitydoc1 = "dummydoctor4"; const usernametolinkfacilitydoc2 = "dummydoctor5"; const usernametolinkfacilitydoc3 = "dummydoctor6"; - const usernametolinkskill = "devdistrictadmin"; + const usernametolinkskill = "devdoctor"; + const usernameforworkinghour = "devdistrictadmin"; const usernamerealname = "Dummy Doctor"; const facilitytolinkusername = "Dummy Shifting Center"; + const facilitytolinkskill = "Dummy Facility 1"; const workinghour = "23"; + const linkedskill = "General Medicine"; before(() => { loginPage.loginAsDisctrictAdmin(); @@ -25,10 +28,50 @@ describe("Manage User", () => { cy.awaitUrl("/users"); }); - it("add working hour for a user and verify its reflection in card and user profile", () => { - // verify mandatory field error and select working hour for a user + it("linking skills for a users and verify its reflection in profile", () => { + // select the district user and select one skill link and verify its profile reflection + userPage.typeInSearchInput(usernameforworkinghour); + userPage.checkUsernameText(usernameforworkinghour); + manageUserPage.clicklinkedskillbutton(); + manageUserPage.typeSkill(linkedskill); + manageUserPage.selectFacilityFromDropdown(linkedskill); + manageUserPage.clickAddSkillButton(); + manageUserPage.clickCloseSlideOver(); + manageUserPage.navigateToProfile(); + manageUserPage.assertSkillInAlreadyLinkedSkills(linkedskill); + }); + + it("linking skills for a doctor users and verify its reflection in doctor connect", () => { + // select a doctor user and link and unlink same skill twice and verify the badge is only shown once in doctor connect userPage.typeInSearchInput(usernametolinkskill); userPage.checkUsernameText(usernametolinkskill); + manageUserPage.clicklinkedskillbutton(); + manageUserPage.typeSkill(linkedskill); + manageUserPage.selectFacilityFromDropdown(linkedskill); + manageUserPage.clickAddSkillButton(); + manageUserPage.clickCloseSlideOver(); + cy.wait(5000); // temporary hack to fix the failure + manageUserPage.clicklinkedskillbutton(); + manageUserPage.assertSkillInAddedUserSkills(linkedskill); + manageUserPage.clickUnlinkSkill(); + manageUserPage.clickSubmit(); + manageUserPage.typeSkill(linkedskill); + manageUserPage.selectFacilityFromDropdown(linkedskill); + manageUserPage.clickAddSkillButton(); + manageUserPage.clickCloseSlideOver(); + // verifying the doctor connect + manageUserPage.navigateToFacility(); + manageUserPage.typeFacilitySearch(facilitytolinkskill); + manageUserPage.assertFacilityInCard(facilitytolinkskill); + manageUserPage.clickFacilityPatients(); + manageUserPage.clickDoctorConnectButton(); + manageUserPage.assertSkillIndoctorconnect(linkedskill); + }); + + it("add working hour for a user and verify its reflection in card and user profile", () => { + // verify mandatory field error and select working hour for a user + userPage.typeInSearchInput(usernameforworkinghour); + userPage.checkUsernameText(usernameforworkinghour); manageUserPage.clicksetaveragehourbutton(); manageUserPage.clearweeklyhourfield(); manageUserPage.clickSubmit(); diff --git a/cypress/pageobject/Users/ManageUserPage.ts b/cypress/pageobject/Users/ManageUserPage.ts index 70e617f7e35..1b340c875ee 100644 --- a/cypress/pageobject/Users/ManageUserPage.ts +++ b/cypress/pageobject/Users/ManageUserPage.ts @@ -103,10 +103,44 @@ export class ManageUserPage { cy.get("#facility-patients").click(); } + clicklinkedskillbutton() { + cy.get("#skills").click(); + } + + clickAddSkillButton() { + cy.intercept("GET", "**/api/v1/skill/**").as("getSkills"); + cy.get("#add-skill-button").click(); + cy.wait("@getSkills").its("response.statusCode").should("eq", 200); + } + + assertSkillInAlreadyLinkedSkills(skillName) { + cy.get("#already-linked-skills") + .contains(skillName) + .should("have.length", 1); + } + + assertSkillIndoctorconnect(skillName) { + cy.get("#doctor-connect-home-doctor") + .contains(skillName) + .should("have.length", 1); + } + + typeSkill(skillName) { + cy.get("#select-skill").click().type(skillName); + } + clickDoctorConnectButton() { cy.get("#doctor-connect-patient-button").click(); } + clickUnlinkSkill() { + cy.get("#unlink-skill").click(); + } + + assertSkillInAddedUserSkills(skillName) { + cy.get("#added-user-skills").should("contain", skillName); + } + assertDoctorConnectVisibility(realName) { cy.get("#doctor-connect-home-doctor").should("contain.text", realName); cy.get("#doctor-connect-remote-doctor").should("contain.text", realName); diff --git a/src/Components/Common/SkillSelect.tsx b/src/Components/Common/SkillSelect.tsx index 5ec97974879..3257971d03c 100644 --- a/src/Components/Common/SkillSelect.tsx +++ b/src/Components/Common/SkillSelect.tsx @@ -5,6 +5,7 @@ import AutoCompleteAsync from "../Form/AutoCompleteAsync"; import { SkillObjectModel } from "../Users/models"; interface SkillSelectProps { + id?: string; name: string; errors?: string | undefined; className?: string; @@ -20,6 +21,7 @@ interface SkillSelectProps { export const SkillSelect = (props: SkillSelectProps) => { const { + id, name, multiple, selected, @@ -64,6 +66,7 @@ export const SkillSelect = (props: SkillSelectProps) => { return ( Promise | undefined; @@ -31,6 +32,7 @@ interface Props { const AutoCompleteAsync = (props: Props) => { const { + id, name, selected, fetchData, @@ -82,6 +84,7 @@ const AutoCompleteAsync = (props: Props) => {
{ {!isLoading && (
{ username={username} /> addSkill(username, selectedSkill)} className="w-6rem" @@ -146,7 +148,7 @@ export default ({ show, setShow, username }: IProps) => {
) : ( -
+
{hasSkills ? ( {skill.skill_object.name}
-
+
{details.skills && details.skills.length ? details.skills?.map((skill: SkillObjectModel) => { return ( From c67ced2d6c2b20803652547a21dbc6a0c59b638c Mon Sep 17 00:00:00 2001 From: Rithvik Nishad Date: Fri, 10 Nov 2023 12:28:31 +0530 Subject: [PATCH 2/8] quick patch: hide not working sort filter (#6577) --- src/Components/Facility/LiveFeedScreen.tsx | 67 ---------------------- 1 file changed, 67 deletions(-) diff --git a/src/Components/Facility/LiveFeedScreen.tsx b/src/Components/Facility/LiveFeedScreen.tsx index b9a2eabdc03..a47d4a0c955 100644 --- a/src/Components/Facility/LiveFeedScreen.tsx +++ b/src/Components/Facility/LiveFeedScreen.tsx @@ -14,9 +14,6 @@ import { AssetData } from "../Assets/AssetTypes"; import { Popover, Transition } from "@headlessui/react"; import { FieldLabel } from "../Form/FormFields/FormField"; import CheckBoxFormField from "../Form/FormFields/CheckBoxFormField"; -import { useTranslation } from "react-i18next"; -import { SortOption } from "../Common/SortDropdown"; -import { SelectFormField } from "../Form/FormFields/SelectFormField"; import LiveFeedTile from "./LiveFeedTile"; import { getCameraConfig } from "../../Utils/transformUtils"; import { getPermittedFacility, listAssets } from "../../Redux/actions"; @@ -24,45 +21,11 @@ import { useDispatch } from "react-redux"; const PER_PAGE_LIMIT = 6; -const SORT_OPTIONS: SortOption[] = [ - { isAscending: true, value: "bed__name" }, - { isAscending: false, value: "-bed__name" }, - { isAscending: false, value: "-created_date" }, - { isAscending: true, value: "created_date" }, -]; - interface Props { facilityId: string; } -const getOrderingList = async ( - facilityId: string, - setOrdering: (order: string) => void -) => { - const orderData = localStorage.getItem("live-feed-order"); - if (orderData) { - const order = JSON.parse(orderData); - const orderValue = order.find((item: any) => item.facility === facilityId); - setOrdering(orderValue.order); - } -}; - -const setOrderingList = async (facilityId: string, order: string) => { - const orderData = localStorage.getItem("live-feed-order") || "[]"; - const orderList = JSON.parse(orderData); - const index = orderList.findIndex( - (item: any) => item.facility === facilityId - ); - if (index !== -1) { - orderList[index].order = order; - } else { - orderList.push({ facility: facilityId, order }); - } - localStorage.setItem("live-feed-order", JSON.stringify(orderList)); -}; - export default function LiveFeedScreen({ facilityId }: Props) { - const { t } = useTranslation(); const dispatch = useDispatch(); const [isFullscreen, setFullscreen] = useFullscreen(); const sidebar = useContext(SidebarShrinkContext); @@ -73,7 +36,6 @@ export default function LiveFeedScreen({ facilityId }: Props) { const { qParams, updateQuery, removeFilter, updatePage } = useFilters({ limit: PER_PAGE_LIMIT, }); - const [ordering, setOrdering] = useState("bed__name"); const [refresh_presets_hash, setRefreshPresetsHash] = useState( Number(new Date()) @@ -88,10 +50,6 @@ export default function LiveFeedScreen({ facilityId }: Props) { }; }, []); - useEffect(() => { - getOrderingList(facilityId, setOrdering); - }, [facilityId]); - useEffect(() => { async function fetchFacilityOrObject() { if (facility) return facility; @@ -112,7 +70,6 @@ export default function LiveFeedScreen({ facilityId }: Props) { asset_class: "ONVIF", facility: facilityId || "", location: qParams.location, - ordering: qParams.ordering || ordering, bed_is_occupied: qParams.bed_is_occupied, }; @@ -204,30 +161,6 @@ export default function LiveFeedScreen({ facilityId }: Props) { )}
- { - updateQuery({ ordering: value }); - setOrderingList(facilityId, value); - }} - options={SORT_OPTIONS} - optionLabel={({ value }) => t("SortOptions." + value)} - optionIcon={({ isAscending }) => ( - - )} - optionValue={({ value }) => value} - labelClassName="text-sm" - errorClassName="hidden" - /> Date: Fri, 10 Nov 2023 14:37:53 +0530 Subject: [PATCH 3/8] Fix clear button for CNS location filter (#6581) * Fix clear button for CNS location filter * Fixes * tooltip --- .../Facility/CentralNursingStation.tsx | 18 ++++++------------ src/Components/Facility/FacilityCard.tsx | 2 +- src/Components/Facility/LiveFeedScreen.tsx | 18 ++++++------------ 3 files changed, 13 insertions(+), 25 deletions(-) diff --git a/src/Components/Facility/CentralNursingStation.tsx b/src/Components/Facility/CentralNursingStation.tsx index 26bed83635b..0eae504399c 100644 --- a/src/Components/Facility/CentralNursingStation.tsx +++ b/src/Components/Facility/CentralNursingStation.tsx @@ -168,11 +168,15 @@ export default function CentralNursingStation({ facilityId }: Props) { Filter by Location -
+
updateQuery({ location })} + setSelected={(location) => { + location + ? updateQuery({ location }) + : removeFilter("location"); + }} selected={qParams.location} showAll={false} multiple={false} @@ -180,16 +184,6 @@ export default function CentralNursingStation({ facilityId }: Props) { errors="" errorClassName="hidden" /> - {qParams.location && ( - removeFilter("location")} - > - Clear - - )}
{ : "button-primary-border bg-primary-100" }`} > - + Live Patients / Total beds {" "} Filter by Location -
+
updateQuery({ location })} + setSelected={(location) => { + location + ? updateQuery({ location }) + : removeFilter("location"); + }} selected={qParams.location} showAll={false} multiple={false} @@ -149,16 +153,6 @@ export default function LiveFeedScreen({ facilityId }: Props) { errors="" errorClassName="hidden" /> - {qParams.location && ( - removeFilter("location")} - > - Clear - - )}
Date: Fri, 10 Nov 2023 16:06:07 +0530 Subject: [PATCH 4/8] Fixed Flaky Cypress Test in the User_Manage File (#6583) * flaky-test-in-usertab * trying out another solution * trying slow approach * revert vite config --- cypress/e2e/shifting_spec/filter.cy.ts | 2 +- cypress/e2e/users_spec/user_manage.cy.ts | 11 +++++++++++ cypress/pageobject/Users/ManageUserPage.ts | 2 ++ package.json | 2 +- 4 files changed, 15 insertions(+), 2 deletions(-) diff --git a/cypress/e2e/shifting_spec/filter.cy.ts b/cypress/e2e/shifting_spec/filter.cy.ts index f1068f7c53a..3790198a4db 100644 --- a/cypress/e2e/shifting_spec/filter.cy.ts +++ b/cypress/e2e/shifting_spec/filter.cy.ts @@ -19,7 +19,7 @@ describe("Shifting section filter", () => { shiftingPage.filterByFacility( "Dummy Shifting", "Dummy Shifting", - "District Admin" + "District" ); shiftingPage.facilityAssignedBadge().should("exist"); diff --git a/cypress/e2e/users_spec/user_manage.cy.ts b/cypress/e2e/users_spec/user_manage.cy.ts index db865dc8126..9fe3aea75f9 100644 --- a/cypress/e2e/users_spec/user_manage.cy.ts +++ b/cypress/e2e/users_spec/user_manage.cy.ts @@ -2,6 +2,7 @@ import { cy, describe, before, beforeEach, it, afterEach } from "local-cypress"; import LoginPage from "../../pageobject/Login/LoginPage"; import { UserPage } from "../../pageobject/Users/UserSearch"; import ManageUserPage from "../../pageobject/Users/ManageUserPage"; +import { UserCreationPage } from "../../pageobject/Users/UserCreation"; describe("Manage User", () => { const loginPage = new LoginPage(); @@ -11,6 +12,7 @@ describe("Manage User", () => { const usernametolinkfacilitydoc2 = "dummydoctor5"; const usernametolinkfacilitydoc3 = "dummydoctor6"; const usernametolinkskill = "devdoctor"; + const userCreationPage = new UserCreationPage(); const usernameforworkinghour = "devdistrictadmin"; const usernamerealname = "Dummy Doctor"; const facilitytolinkusername = "Dummy Shifting Center"; @@ -37,7 +39,16 @@ describe("Manage User", () => { manageUserPage.selectFacilityFromDropdown(linkedskill); manageUserPage.clickAddSkillButton(); manageUserPage.clickCloseSlideOver(); + cy.wait(5000); + manageUserPage.clicklinkedskillbutton(); + manageUserPage.assertSkillInAddedUserSkills(linkedskill); + manageUserPage.clickCloseSlideOver(); + cy.wait(5000); manageUserPage.navigateToProfile(); + userCreationPage.verifyElementContainsText( + "username-profile-details", + usernameforworkinghour + ); manageUserPage.assertSkillInAlreadyLinkedSkills(linkedskill); }); diff --git a/cypress/pageobject/Users/ManageUserPage.ts b/cypress/pageobject/Users/ManageUserPage.ts index 1b340c875ee..b79b96447f0 100644 --- a/cypress/pageobject/Users/ManageUserPage.ts +++ b/cypress/pageobject/Users/ManageUserPage.ts @@ -73,7 +73,9 @@ export class ManageUserPage { } navigateToProfile() { + cy.intercept("GET", "**/api/v1/users/**").as("getUsers"); cy.get("#profilenamelink").click(); + cy.wait("@getUsers").its("response.statusCode").should("eq", 200); } verifyWorkingHours(expectedHours: string) { diff --git a/package.json b/package.json index 1ec0e88f48f..ca3f6920a12 100644 --- a/package.json +++ b/package.json @@ -130,7 +130,7 @@ "@typescript-eslint/parser": "^5.61.0", "@vitejs/plugin-react-swc": "^3.3.2", "autoprefixer": "^10.4.14", - "cypress": "^13.1.0", + "cypress": "^13.5.0", "cypress-localstorage-commands": "^2.2.3", "eslint": "^8.44.0", "eslint-config-prettier": "^8.8.0", From 5b9c61848ddc8b729802e9099fd4ccfa5ad5ea2f Mon Sep 17 00:00:00 2001 From: Rithvik Nishad Date: Fri, 10 Nov 2023 20:35:44 +0530 Subject: [PATCH 5/8] refresh administrations upon archive (#6586) --- .../AdministrationEventCell.tsx | 3 ++ .../AdministrationTableRow.tsx | 36 ++++++++++--------- .../Medicine/PrescrpitionTimeline.tsx | 7 +++- 3 files changed, 29 insertions(+), 17 deletions(-) diff --git a/src/Components/Medicine/MedicineAdministrationSheet/AdministrationEventCell.tsx b/src/Components/Medicine/MedicineAdministrationSheet/AdministrationEventCell.tsx index 9869b677cd2..1bcfc24a80a 100644 --- a/src/Components/Medicine/MedicineAdministrationSheet/AdministrationEventCell.tsx +++ b/src/Components/Medicine/MedicineAdministrationSheet/AdministrationEventCell.tsx @@ -11,12 +11,14 @@ interface Props { administrations: MedicineAdministrationRecord[]; interval: { start: Date; end: Date }; prescription: Prescription; + refetch: () => void; } export default function AdministrationEventCell({ administrations, interval: { start, end }, prescription, + refetch, }: Props) { const [showTimeline, setShowTimeline] = useState(false); // Check if cell belongs to an administered prescription @@ -55,6 +57,7 @@ export default function AdministrationEventCell({ interval={{ start, end }} prescription={prescription} showPrescriptionDetails + onRefetch={refetch} />