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: 488] - Mark thread as read to update email status to read #499

Closed
wants to merge 4 commits into from

Conversation

recursive-0
Copy link

@recursive-0 recursive-0 commented Mar 21, 2025

Description

Threads weren't marked as read previously even after reading all the messages under the email. This was happening because we weren't marking all the messages under the email as read. I have added logic to remove "UNREAD" labels from all the messages to mark the email as read.

Type of Change

  • 🐛 Bug fix (non-breaking change which fixes an issue)

Areas Affected

  • User Interface/Experience
  • Development Workflow

Testing Done

Describe the tests you've done:

  • Manual testing performed

Security Considerations

None

Checklist

  • I have read the CONTRIBUTING document
  • My code follows the project's style guidelines
  • I have performed a self-review of my code
  • I have commented my code, particularly in complex areas
  • My changes generate no new warnings
  • I have added tests that prove my fix/feature works
  • Any dependent changes are merged and published

Additional Notes

I couldn't see an option to mark emails as read in batches so currently I'm just picking up the first email and its messages and marking them as read. Once we have "select all" and "mark as read" feature up, we can extend the logic to get messages for all the email IDs and mark as read.

Screenshots/Recordings

screen-capture.47.webm

By submitting this pull request, I confirm that my contribution is made under the terms of the project's license.

Summary by CodeRabbit

  • New Features

    • Introduced batch processing for Gmail thread modifications, allowing users to apply label changes to multiple threads simultaneously.
  • Refactor

    • Improved the email read status update process for enhanced reliability and consistency, ensuring that when you mark an email as read, all related messages update correctly for a smoother experience.
  • Style

    • Enhanced formatting of the itemContent prop in the MailList component for better readability without changing functionality.
    • Updated whitespace in the id properties of the categories array for consistency.

Copy link

vercel bot commented Mar 21, 2025

@Lokendra-sinh is attempting to deploy a commit to the Zero Team on Vercel.

A member of the Team first needs to authorize it.

Copy link
Contributor

coderabbitai bot commented Mar 21, 2025

Walkthrough

The pull request introduces a new function, processMultipleThreads, in the Google driver module of the mail application, allowing batch processing of Gmail thread label modifications. The markAsRead and markAsUnread methods have been refactored to utilize this new function, enhancing modularity and incorporating error handling. Additionally, formatting changes were made to the MailList component for improved readability, and whitespace adjustments were applied to the id properties in the BulkSelectActions function.

Changes

File(s) Change Summary
apps/mail/app/api/driver/google.ts Introduced processMultipleThreads function for batch label modifications; refactored markAsRead and markAsUnread to use this function; added error handling.
apps/mail/components/mail/mail-list.tsx Modified itemContent prop formatting in MailList component from single-line to multi-line for improved readability.
apps/mail/components/mail/mail.tsx Updated whitespace in id properties of categories array in BulkSelectActions function to include a space after the colon.

Possibly related PRs

  • Draft emails #390: The changes in the main PR, specifically the modifications to the markAsRead and markAsUnread methods in google.ts, are related to the changes in the retrieved PR, which also involves the markAsRead function in mail.ts, as both PRs modify similar functionalities related to email management.
  • Parse From header according to RFC 2822 #472: The changes in the main PR are related to the modifications in the google.ts file, specifically in how the sender variable is handled, as both PRs involve updates to the same file and affect the email processing logic.
  • fix: threads with unread messages where wrongly flagged on mail list #480: The changes in the main PR, specifically the introduction of the processMultipleThreads function and the refactoring of the markAsRead and markAsUnread methods, are related to the modifications in the retrieved PR that enhance message parsing by incorporating label IDs, as both involve handling Gmail message labels and their processing.

Suggested reviewers

  • MrgSub
  • nizzyabi

Poem

I’m a rabbit hopping through the code,
With a skip and a jump down the road.
Threads revealed a secret path to tread,
Marking messages read where error fears once spread.
In lively lines, our changes now glow—
A whimsical dance in code, ready to go!
🐇✨

✨ Finishing Touches
  • 📝 Generate Docstrings

🪧 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.
  • @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
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

🧹 Nitpick comments (3)
apps/mail/app/api/driver/google.ts (3)

180-209: Improved thread handling to properly mark emails as read.

The new implementation correctly addresses the issue where email threads weren't being marked as read by retrieving all messages in a thread and removing the "UNREAD" label from each message. The added error handling is also a good improvement.

There's an indentation inconsistency - the new code uses tabs while the rest of the file uses spaces. Consider standardizing the indentation style:

    markAsRead: async (id: string[]) => {
-			try {
-
-				if (id.length > 0) {
-					const threadResponse = await gmail.users.threads.get({
-						userId: 'me',
-						id: id[0],
-					});
-
-					const thread = threadResponse.data;
-
-					const messageIds =
-						thread.messages
-							?.map((message) => message.id)
-							.filter((id): id is string => id !== null && id !== undefined) || [];
-
-
-					if (messageIds.length > 0) {
-              await gmail.users.messages.batchModify({
-							userId: 'me',
-							requestBody: {
-								ids: messageIds,
-								removeLabelIds: ['UNREAD'],
-							},
-						});
-					}
-				}
-			} catch (error) {
-				console.error('Error marking messages as read:', error);
-        throw error
-			}
+      try {
+        if (id.length > 0) {
+          const threadResponse = await gmail.users.threads.get({
+            userId: 'me',
+            id: id[0],
+          });
+
+          const thread = threadResponse.data;
+
+          const messageIds =
+            thread.messages
+              ?.map((message) => message.id)
+              .filter((id): id is string => id !== null && id !== undefined) || [];
+
+          if (messageIds.length > 0) {
+            await gmail.users.messages.batchModify({
+              userId: 'me',
+              requestBody: {
+                ids: messageIds,
+                removeLabelIds: ['UNREAD'],
+              },
+            });
+          }
+        }
+      } catch (error) {
+        console.error('Error marking messages as read:', error);
+        throw error;
+      }
    },

207-208: Add missing semicolon for consistency.

There's a missing semicolon at the end of the throw error statement. This should be fixed to maintain code style consistency.

-        throw error
+        throw error;

211-219: Consider updating markAsUnread function to mirror markAsRead implementation.

For consistency, the markAsUnread function should also use a try-catch block and possibly use the same thread-based approach to ensure all messages in a thread are marked as unread.

    markAsUnread: async (id: string[]) => {
-      await gmail.users.messages.batchModify({
-        userId: "me",
-        requestBody: {
-          ids: id,
-          addLabelIds: ["UNREAD"],
-        },
-      });
+      try {
+        if (id.length > 0) {
+          const threadResponse = await gmail.users.threads.get({
+            userId: 'me',
+            id: id[0],
+          });
+
+          const thread = threadResponse.data;
+
+          const messageIds =
+            thread.messages
+              ?.map((message) => message.id)
+              .filter((id): id is string => id !== null && id !== undefined) || [];
+
+          if (messageIds.length > 0) {
+            await gmail.users.messages.batchModify({
+              userId: "me",
+              requestBody: {
+                ids: messageIds,
+                addLabelIds: ["UNREAD"],
+              },
+            });
+          }
+        }
+      } catch (error) {
+        console.error('Error marking messages as unread:', error);
+        throw error;
+      }
    },
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 8de0716 and a814d02.

📒 Files selected for processing (1)
  • apps/mail/app/api/driver/google.ts (1 hunks)
🔇 Additional comments (1)
apps/mail/app/api/driver/google.ts (1)

190-193: Well-implemented null safety with TypeScript type predicate.

Good use of the TypeScript type predicate (id): id is string => id !== null && id !== undefined to filter out null or undefined values while maintaining type safety.

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

🧹 Nitpick comments (2)
apps/mail/app/api/driver/google.ts (2)

102-104: Consider revising early return condition

The current check might allow unnecessary processing when both arrays are non-empty but one has elements and the other is empty.

-  if(addLabelIds.length === 0 || removeLabelIds.length === 0){
+  if(addLabelIds.length === 0 && removeLabelIds.length === 0){
    return
  }

245-257: Fix error message in markAsUnread

The error message incorrectly states "Error marking messages as read" instead of "unread".

-	console.error('Error marking messages as read:', error);
+	console.error('Error marking messages as unread:', error);
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between a814d02 and 1941bd5.

📒 Files selected for processing (3)
  • apps/mail/app/api/driver/google.ts (3 hunks)
  • apps/mail/components/mail/mail-list.tsx (1 hunks)
  • apps/mail/components/mail/mail.tsx (1 hunks)
✅ Files skipped from review due to trivial changes (2)
  • apps/mail/components/mail/mail.tsx
  • apps/mail/components/mail/mail-list.tsx
🔇 Additional comments (4)
apps/mail/app/api/driver/google.ts (4)

38-42: Good optimization: Auth client defined at module level

Moving the OAuth2 client outside the driver function allows it to be reused across different function calls, avoiding unnecessary recreation of the authorization object.


87-91: Well-defined TypeScript interface

Good use of TypeScript interface to define the structure of props for the new function, making the code more maintainable and providing clear type definitions.


93-137: Great implementation of batch processing

This new function effectively implements the batch processing capability suggested in the previous review. I appreciate the thorough input validation, proper error handling, and efficient use of Promise.all for concurrent execution.

The implementation correctly:

  • Retrieves all message IDs from multiple threads
  • Filters out null/undefined values
  • Uses the batchModify endpoint for better performance
  • Includes comprehensive error handling

232-244: Good refactoring of markAsRead

The method has been well refactored to use the new batch processing function with proper error handling.

@recursive-0
Copy link
Author

Fixes #488

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

🧹 Nitpick comments (2)
apps/mail/app/api/driver/google.ts (2)

93-137: Excellent implementation of batch processing for multiple threads.

The function efficiently processes multiple email threads by:

  1. Retrieving message IDs from each thread
  2. Flattening the results into a single array
  3. Using a batch modify operation to update all messages at once

Good job including proper validation and error handling.

Consider clarifying the early return condition on line 102-104. It currently returns if either array is empty, which works for the current use cases but might be confusing:

-  if(addLabelIds.length === 0 || removeLabelIds.length === 0){
+  if(addLabelIds.length === 0 && removeLabelIds.length === 0){
     return
   }

This change would only return if there are no label operations to perform.


254-255: Fix error message in markAsUnread.

The error message incorrectly refers to marking messages as "read" instead of "unread".

-console.error('Error marking messages as read:', error);
+console.error('Error marking messages as unread:', error);
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 1941bd5 and c00ea0c.

📒 Files selected for processing (1)
  • apps/mail/app/api/driver/google.ts (3 hunks)
🧰 Additional context used
🧬 Code Definitions (1)
apps/mail/app/api/driver/google.ts (1)
apps/mail/lib/auth.ts (1) (1)
  • auth (155-158)
🔇 Additional comments (3)
apps/mail/app/api/driver/google.ts (3)

87-91: LGTM! Well-structured interface.

Good practice defining a clear TypeScript interface for the function parameters. This improves code readability and type safety.


233-244: Great refactoring of markAsRead to use batch processing.

The implementation now efficiently handles multiple threads and includes proper error handling.


38-42:

❓ Verification inconclusive

Good optimization by moving auth outside the driver function.

Making auth a module-level constant improves performance by creating a single reusable OAuth2 client instance instead of recreating it on each driver initialization.


🏁 Script executed:

#!/bin/bash
# Check for any potential issues with the shared auth approach
grep -A 5 -B 5 "auth.setCredentials" apps/mail/app/api/driver/google.ts

Length of output: 853


OAuth2 Client Optimization Approved – Verify Concurrency Implications

Great work moving the OAuth2 client instantiation to the module level. This approach avoids recreating the client on every driver initialization, which is a smart performance enhancement. The grep output confirms that calls to auth.setCredentials occur in both the configuration initialization and the getUserInfo method.

  • Confirmed Usage:
    • The instance is shared across calls, with credentials being set in both the if (config.auth) block and in getUserInfo.
  • Verification Note:
    • Although this shared instance is efficient, please double-check that simultaneous or overlapping invocations do not lead to race conditions or unintended overwrites of credentials in a concurrent execution environment.

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

🧹 Nitpick comments (2)
apps/mail/app/api/driver/google.ts (2)

246-257: Error message needs correction in markAsUnread method.

The error message on line 254 incorrectly states "Error marking messages as read" but this is the markAsUnread method.

-				console.error('Error marking messages as read:', error);
+				console.error('Error marking messages as unread:', error);

119-131: Consider adding a success log message.

For better debugging and observability, consider adding a success log message after the batch operation completes.

    await gmail.users.messages.batchModify({
      userId: "me",
      requestBody: {
        ids: allMessagesIds,
        removeLabelIds: removeLabelIds,
        addLabelIds: addLabelIds
      }
    })
+    console.log(`Successfully processed ${allMessagesIds.length} messages from ${threadIds.length} threads`)
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between c00ea0c and c170cbd.

📒 Files selected for processing (1)
  • apps/mail/app/api/driver/google.ts (3 hunks)
🔇 Additional comments (5)
apps/mail/app/api/driver/google.ts (5)

38-42: Good refactoring: Moving OAuth2 client outside the driver function.

Moving the auth instance to module scope improves reusability and allows it to be shared between functions, which is necessary for the new processMultipleThreads function.


87-91: LGTM: Well-structured interface definition.

The ProcessMultipleThreadsProps interface is clear and well-typed, providing good documentation for the function parameters.


93-137: Great implementation of batch thread processing.

This implementation correctly addresses the issue where email threads were not being marked as read. The function:

  1. Takes multiple thread IDs for batch processing
  2. Properly retrieves all messages from each thread
  3. Uses batchModify to efficiently update all messages at once
  4. Includes robust error handling and input validation

This approach is scalable and will work well with the future "select all" feature mentioned in the PR description.


232-244: Good refactoring of markAsRead method.

The refactoring improves code organization by leveraging the new processMultipleThreads function and adds proper error handling.


98-104:

✅ Verification successful

Verify that empty array validation works as expected.

The function returns early if both addLabelIds and removeLabelIds are empty, but it throws an error if threadIds is empty. This logic looks correct, but it's worth verifying that this matches the expected behavior in all cases.


🏁 Script executed:

#!/bin/bash
# Check how the function handles empty arrays in other parts of the codebase

# Check if threadIds are validated elsewhere
rg -A 5 -B 5 "threadIds.*length.*0" --iglob "*.ts"

# Check for similar validation patterns
rg -A 5 -B 5 "(addLabelIds|removeLabelIds).*length.*0" --iglob "*.ts"

Length of output: 1479


Empty array validation logic verified

The script output confirms that the validation checks in apps/mail/app/api/driver/google.ts work as expected. The code correctly throws an error when threadIds is empty and returns early when both addLabelIds and removeLabelIds are empty. No additional adjustments are needed.

Copy link
Collaborator

@ahmetskilinc ahmetskilinc left a comment

Choose a reason for hiding this comment

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

lgtm

Copy link
Collaborator

Choose a reason for hiding this comment

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

revert this, doesnt need to be touched.

Copy link
Collaborator

Choose a reason for hiding this comment

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

revert this, doesnt need to be touched.

@MrgSub
Copy link
Collaborator

MrgSub commented Mar 21, 2025

Done in #502

@MrgSub MrgSub closed this Mar 21, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants