Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix: Duplicate network requests in Patient Details page #9297

Merged

Conversation

Mahendar0701
Copy link
Contributor

@Mahendar0701 Mahendar0701 commented Dec 4, 2024

Proposed Changes

@ohcnetwork/care-fe-code-reviewers

Merge Checklist

  • Add specs that demonstrate bug / test a new feature.
  • Update product documentation.
  • Ensure that UI text is kept in I18n files.
  • Prep screenshot or demo video for changelog entry, and attach it to issue.
  • Request for Peer Reviews
  • Completion of QA

Summary by CodeRabbit

  • New Features

    • Introduced a new refetch prop across several components to enable data refetching.
    • Enhanced user experience for volunteer selection with a new UserAutocomplete component.
  • Bug Fixes

    • Improved error handling for unauthorized actions in the ImmunisationRecords component.
  • Documentation

    • Updated import paths for better organization and clarity across various components.
  • Refactor

    • Streamlined data fetching logic and component structure in PatientNotesList, EncounterHistory, and Notes components.

@Mahendar0701 Mahendar0701 requested a review from a team as a code owner December 4, 2024 10:38
Copy link
Contributor

coderabbitai bot commented Dec 4, 2024

Walkthrough

The pull request introduces modifications to several components within the patient details section of the application. The changes primarily focus on streamlining import statements, simplifying state management, and reducing duplicate network requests across various patient-related components. Key modifications include updating import paths to use absolute imports, removing unnecessary state management, and consolidating data fetching logic in components like PatientNotesList, EncounterHistory, and Notes.

Changes

File Change Summary
src/components/Facility/PatientNotesList.tsx Simplified useEffect hook for fetching notes, consolidating conditions for reload and thread props
src/components/Patient/PatientDetailsTab/EncounterHistory.tsx Removed loading state management and useAuthUser hook
src/components/Patient/PatientDetailsTab/Notes.tsx Updated props interface, removed patientActive state, directly using patientData.is_active
src/components/Patient/PatientDetailsTab/index.tsx Updated import paths to absolute imports, added refetch method to PatientProps interface
src/components/Patient/PatientHome.tsx Added refetch prop to Tab component, enhanced volunteer assignment logic
Multiple Patient Detail Tab Components Converted relative import paths to absolute paths

Assessment against linked issues

Objective Addressed Explanation
Reduce duplicate network requests [#9269]
Optimize patient details page API calls
Minimize redundant data fetching

Possibly related PRs

Suggested labels

needs review, tested, P1

Suggested reviewers

  • rithviknishad
  • Jacobjeevan

Poem

🐰 Hopping through code with glee,
Refactoring paths, setting imports free!
Duplicate requests, we bid adieu,
Streamlined components, our codebase anew!
Rabbit's refactor, clean and bright! 🚀

Tip

CodeRabbit's docstrings feature is now available as part of our Early Access Program! Simply use the command @coderabbitai generate docstrings to have CodeRabbit automatically generate docstrings for your pull request. This feature will be included in our Pro Plan when released.


📜 Recent review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between a3f5353 and fb5d1ae.

📒 Files selected for processing (2)
  • src/components/Patient/PatientDetailsTab/index.tsx (1 hunks)
  • src/components/Patient/PatientHome.tsx (4 hunks)
🚧 Files skipped from review as they are similar to previous changes (2)
  • src/components/Patient/PatientDetailsTab/index.tsx
  • src/components/Patient/PatientHome.tsx

Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR. (Beta)
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link

netlify bot commented Dec 4, 2024

Deploy Preview for care-ohc ready!

Name Link
🔨 Latest commit fb5d1ae
🔍 Latest deploy log https://app.netlify.com/sites/care-ohc/deploys/675dcdd404ba3e0008b873e4
😎 Deploy Preview https://deploy-preview-9297--care-ohc.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

🧹 Outside diff range and nitpick comments (2)
src/components/Facility/PatientNotesList.tsx (1)

Line range hint 31-54: Consider implementing request debouncing

The fetchNotes function is called in multiple scenarios (thread changes, reload, pagination). To further prevent duplicate requests, consider implementing debouncing.

Here's a suggested improvement:

import { debounce } from 'lodash';

const debouncedFetchNotes = debounce(async () => {
  setIsLoading(true);
  const { data }: any = await request(routes.getPatientNotes, {
    pathParams: { patientId: props.patientId },
    query: {
      offset: (state.cPage - 1) * RESULTS_PER_PAGE_LIMIT,
      thread,
    },
  });

  setState((prevState: any) => ({
    ...prevState,
    notes: state.cPage === 1 ? data.results : [...prevState.notes, ...data.results],
    totalPages: Math.ceil(data.count / pageSize),
  }));
  setIsLoading(false);
  setReload(false);
}, 300);
src/components/Patient/PatientHome.tsx (1)

68-72: Consider using React 18's useEffect cleanup.

While the mount state management works, it could be simplified using React 18's useEffect cleanup pattern:

-const [isPatientHomeMounted, setIsPatientHomeMounted] = useState(false);
-
-useEffect(() => {
-  setIsPatientHomeMounted(true);
-}, []);
+const [isMounted, setIsMounted] = useState(false);
+
+useEffect(() => {
+  setIsMounted(true);
+  return () => setIsMounted(false);
+}, []);
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 9561e0d and dd6ac7c.

📒 Files selected for processing (5)
  • src/components/Facility/PatientNotesList.tsx (1 hunks)
  • src/components/Patient/PatientDetailsTab/EncounterHistory.tsx (1 hunks)
  • src/components/Patient/PatientDetailsTab/Notes.tsx (3 hunks)
  • src/components/Patient/PatientDetailsTab/index.tsx (1 hunks)
  • src/components/Patient/PatientHome.tsx (3 hunks)
🔇 Additional comments (5)
src/components/Patient/PatientDetailsTab/index.tsx (1)

14-14: Verify consistent implementation of refetch across components

The addition of refetch method to PatientProps is a good approach for centralized data refreshing. However, we should ensure it's properly implemented across all components to prevent duplicate requests.

✅ Verification successful

Let me try a different approach to verify the implementation of refetch across components.


Let me search specifically for the implementation of refetch in the patient components.


Based on the evidence gathered, I can now provide a final response:

Implementation of refetch is consistent across patient components

The refetch method in PatientProps interface is properly implemented and used consistently across all patient-related components. Key findings:

  • The interface is correctly defined in src/components/Patient/PatientDetailsTab/index.tsx with refetch: () => void
  • All components using PatientProps (Demography, EncounterHistory, HealthProfileSummary, etc.) receive and handle the refetch prop appropriately
  • EncounterHistory component explicitly uses refetch for data refresh functionality
  • The pattern is consistent with React's data fetching best practices
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for components using refetch to verify consistent implementation
ast-grep --pattern 'const $_ = ($props: PatientProps) => {
  $$$
  const { refetch } = props;
  $$$
}'

Length of output: 104


Script:

#!/bin/bash
# First, let's find all files that import or use PatientProps
rg "PatientProps" -l

# Then, let's look for any usage of refetch in these components
rg "refetch" -A 3 -B 3

Length of output: 70051


Script:

#!/bin/bash
# Search for PatientProps interface definition and its usage
ast-grep --pattern 'interface PatientProps {
  $$$
}'

# Search for components using PatientProps with refetch
rg -A 5 -B 5 "const \w+ = \(.*: PatientProps\)"

Length of output: 7866

src/components/Patient/PatientDetailsTab/EncounterHistory.tsx (1)

16-16: LGTM! Improved data fetching efficiency

The removal of separate loading state and useQuery hook, combined with passing refetch to ConsultationCard, helps prevent duplicate requests while maintaining proper data refresh capabilities.

src/components/Patient/PatientDetailsTab/Notes.tsx (2)

28-29: LGTM! Props interface simplification.

The change to use PatientProps instead of a custom interface reduces code duplication and ensures consistency across components.


139-139: LGTM! Consistent disabled state management.

Using patientData.is_active directly for both the input field and button ensures consistent behavior and reduces unnecessary state management.

Also applies to: 147-147

src/components/Patient/PatientHome.tsx (1)

477-477: LGTM! Proper data refresh handling.

Adding the refetch prop to child components aligns with the PR objective of fixing duplicate network requests by enabling controlled data refresh.

src/components/Facility/PatientNotesList.tsx Outdated Show resolved Hide resolved
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (1)
src/components/Patient/PatientDetailsTab/Demography.tsx (1)

72-76: Consider adding error handling for refetch operation

While the implementation correctly triggers a refetch when the patient ID changes, it would be beneficial to handle potential refetch failures.

Consider updating the implementation to handle errors:

 useEffect(() => {
   if (patientData.id) {
-    refetch();
+    refetch().catch((error) => {
+      Notification.Error({
+        msg: t("failed_to_refresh_insurance_details"),
+      });
+    });
   }
 }, [patientData.id, refetch]);
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between dd6ac7c and 840bf17.

📒 Files selected for processing (3)
  • src/components/Facility/PatientNotesList.tsx (1 hunks)
  • src/components/Patient/PatientDetailsTab/Demography.tsx (1 hunks)
  • src/components/Patient/PatientHome.tsx (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (2)
  • src/components/Patient/PatientHome.tsx
  • src/components/Facility/PatientNotesList.tsx
🔇 Additional comments (1)
src/components/Patient/PatientDetailsTab/Demography.tsx (1)

64-70: LGTM: Optimized insurance details fetching

The changes to prevent automatic prefetching align well with the PR's objective of reducing duplicate network requests. Setting prefetch: false ensures that the data is only fetched when explicitly needed.

@Mahendar0701
Copy link
Contributor Author

@coderabbitai full review

Copy link
Contributor

coderabbitai bot commented Dec 4, 2024

✅ Actions performed

Full review triggered.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

🧹 Outside diff range and nitpick comments (1)
src/components/Patient/PatientDetailsTab/Demography.tsx (1)

72-76: Consider debouncing the refetch call.

The current implementation might trigger multiple API calls if patientData.id changes rapidly. Consider debouncing the refetch call to prevent potential race conditions.

+import { useCallback } from 'react';
+import debounce from 'lodash/debounce';

+const debouncedRefetch = useCallback(
+  debounce((fn) => fn(), 300),
+  []
+);

 useEffect(() => {
   if (patientData.id) {
-    refetch();
+    debouncedRefetch(refetch);
   }
 }, [patientData.id, refetch]);
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 577f7ec and bdcdc06.

📒 Files selected for processing (6)
  • src/components/Facility/PatientNotesList.tsx (1 hunks)
  • src/components/Patient/PatientDetailsTab/Demography.tsx (1 hunks)
  • src/components/Patient/PatientDetailsTab/EncounterHistory.tsx (1 hunks)
  • src/components/Patient/PatientDetailsTab/Notes.tsx (3 hunks)
  • src/components/Patient/PatientDetailsTab/index.tsx (1 hunks)
  • src/components/Patient/PatientHome.tsx (1 hunks)
🔇 Additional comments (5)
src/components/Facility/PatientNotesList.tsx (1)

61-64: LGTM! Successfully implemented the suggested optimization.

The useEffect hooks have been combined as suggested, which should prevent duplicate requests by having a single effect respond to both reload and thread changes.

src/components/Patient/PatientDetailsTab/EncounterHistory.tsx (1)

16-16: Verify the refetch prop usage in ConsultationCard.

The refetch prop is now passed to ConsultationCard. Please ensure that ConsultationCard component properly handles this prop to prevent any potential memory leaks or unnecessary re-renders.

✅ Verification successful

From the search results, I can see that the ConsultationCard component is imported from "@/components/Facility/ConsultationCard" and the refetch prop is properly passed to it. Let me verify its implementation with another search.


Based on the code inspection, I can now generate the final response:

ConsultationCard properly handles the refetch prop

The ConsultationCard component correctly defines and uses the refetch prop in its interface and implementation. The prop is properly typed as refetch: () => void and is used appropriately in the Beds component for refreshing patient data. There are no potential memory leaks or unnecessary re-renders as the prop is only passed down and used as a callback.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check how ConsultationCard uses the refetch prop
ast-grep --pattern 'interface ConsultationCardProps {
  $$$
  refetch: $_
  $$$
}'

Length of output: 84


Script:

#!/bin/bash
# Let's try a broader search to find ConsultationCard component and its props
ast-grep --pattern 'const ConsultationCard = ({ $$ }) => {
  $$$
}'

# Also search for any usage of refetch within ConsultationCard
rg -A 5 "refetch" --type typescript

Length of output: 140


Script:

#!/bin/bash
# Search for ConsultationCard component definition and its usage
rg -A 10 "ConsultationCard" 

# Search for refetch usage
rg -A 5 "refetch"

# Find the ConsultationCard file
fd ConsultationCard

Length of output: 69304


Script:

#!/bin/bash
# Search for ConsultationCard component implementation and its props interface
cat src/components/Facility/ConsultationCard.tsx

Length of output: 6569

src/components/Patient/PatientDetailsTab/Notes.tsx (1)

28-29: LGTM! Improved efficiency by removing redundant state management.

The changes effectively reduce duplicate network requests by:

  1. Using PatientProps to access patient data from parent
  2. Removing redundant state management for patient's active status
  3. Directly using patientData.is_active instead of maintaining separate state

Also applies to: 139-139, 147-147

src/components/Patient/PatientDetailsTab/Demography.tsx (1)

64-70: LGTM! Improved data fetching strategy.

The changes effectively reduce duplicate network requests by:

  1. Preventing automatic prefetch with prefetch: false
  2. Controlling when insurance details are fetched
src/components/Patient/PatientHome.tsx (1)

471-471: LGTM! Added refetch prop for coordinated data refresh.

The addition of the refetch prop enables child components to trigger data refresh in a coordinated manner, supporting the overall goal of reducing duplicate requests.

@Mahendar0701
Copy link
Contributor Author

Mahendar0701 commented Dec 5, 2024

@rithviknishad triggerGoal is already implemented in PatientHome.tsx so i removed it from EncounterHistory.tsx which will be redundant if used again.

Any further changes required?

@github-actions github-actions bot added needs-triage question Further information is requested labels Dec 5, 2024
@Mahendar0701
Copy link
Contributor Author

Hey @rithviknishad can you please update the review

Comment on lines 64 to 76
const { data: insuranceDetials, refetch } = useQuery(
routes.hcx.policies.list,
{
query: { patient: id },
prefetch: false, // Don't prefetch by default
},
});
);

useEffect(() => {
if (patientData.id) {
refetch();
}
}, [patientData.id, refetch]);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why disable prefetch? wouldn't patientData.id always be present?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In PatientHome.tsx, the getPatient API fetches patientData and passes it to Demography as prop.

Earlier, I disabled prefetch for the insuranceDetails API in Demography to avoid calls with an undefined patientData.id.

Now, by setting prefetch to !!patientData.id in the query itself, the API triggers only when the patientData.id is available,This prevents unnecessary call.

This change working correctly. Please confirm if this correct.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

you can get the patient id using the useSlug or by passing it as a prop right? We can prevent request waterfalls this way. There's no reason for fetching insurance details to wait until we get the patient object right?

Copy link
Contributor Author

@Mahendar0701 Mahendar0701 Dec 10, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok i will check it .

Copy link
Contributor Author

@Mahendar0701 Mahendar0701 Dec 10, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

you can get the patient id using the useSlug or by passing it as a prop right? We can prevent request waterfalls this way. There's no reason for fetching insurance details to wait until we get the patient object right?

Yes we are passing patient id as prop from PatientHome.tsx. This patient id is fetched from getPatient api request.

But the tabs are rendered before the fetching is completed and props are passed as undefined to demography tab.

So to prevent this i have made fetching insurance details to wait until we get the property fetched patient object.

Instead of this should i write a conditional statement in patienthome.tsx so that tabs are not rendered until patientdata is fetched completely.

{Tab && patientData?.id && (
              <Tab
                facilityId={facilityId || ""}
                id={patientData.id || ""}
                patientData={patientData}
                refetch={refetch}
              />
   )}

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can get the patient ID from the URL right?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes

src/components/Patient/PatientDetailsTab/Notes.tsx Outdated Show resolved Hide resolved
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
reviewed reviewed by a core member tested
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Duplicate network requests on patient details page
4 participants