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

POC of the auto unsubscribe functionality #376

Open
wants to merge 1 commit into
base: main
Choose a base branch
from
Open

Conversation

elie222
Copy link
Owner

@elie222 elie222 commented Mar 26, 2025

Hasn't been tested. Reference point to start the task yourself.

Summary by CodeRabbit

  • New Features
    • The unsubscribe process now requires both a web address and email for improved validation.
    • Bulk unsubscribe functionality has been enhanced to automatically process unsubscribe links when present.
    • Unsubscribe handling now leverages user authentication for secure and reliable processing.
    • A new optional configuration setting for the unsubscribe service endpoint has been introduced.
Copy link

vercel bot commented Mar 26, 2025

The latest updates on your projects. Learn more about Vercel for Git ↗︎

Name Status Preview Updated (UTC)
inbox-zero ✅ Ready (Inspect) Visit Preview Mar 26, 2025 9:45am
Copy link
Contributor

coderabbitai bot commented Mar 26, 2025

Walkthrough

This pull request updates the unsubscribe functionality across both backend and frontend components. The primary change is modifying the function signature of the unsubscribe process to accept an object with both url and email properties. Additionally, the server-side validation schema and route handler are updated to accommodate the new parameter. On the client side, a new unsubscribe action is integrated into a hook, and an optional environment variable for the unsubscribe API URL is introduced. A new utility function, unsubscribeAction, is added to handle authenticated API calls for unsubscription.

Changes

File(s) Summary
apps/unsubscriber/src/main.ts
apps/unsubscriber/src/server.ts
Updated the unsubscribe functionality to use an object parameter with url and email; modified the request schema and route handler in the server.
apps/web/.../hooks.ts Enhanced the useUnsubscribe hook to check for lastUnsubscribeLink and invoke the new unsubscribeAction accordingly.
apps/web/env.ts Added a new optional environment variable UNSUBSCRIBER_API_URL validated as a string.
apps/web/utils/actions/unsubscriber.ts Introduced the unsubscribeAction function with authentication, API request handling, and instrumentation.

Sequence Diagram(s)

sequenceDiagram
    participant HTTP Request
    participant Server
    participant AutoUnsubscribe
    HTTP Request->>Server: POST /unsubscribe with {url, email}
    Server->>AutoUnsubscribe: autoUnsubscribe({url, email})
    AutoUnsubscribe-->>Server: returns Boolean
    Server-->>HTTP Request: Response (success/failure)
Loading
sequenceDiagram
    participant Client
    participant UnsubscribeAction
    participant AuthService
    participant API
    Client->>UnsubscribeAction: Invoke unsubscribeAction with URL
    UnsubscribeAction->>AuthService: Retrieve user session
    AuthService-->>UnsubscribeAction: Return session (with email)
    UnsubscribeAction->>API: Send fetch request with email & URL
    API-->>UnsubscribeAction: Return response status
    UnsubscribeAction-->>Client: Return success/error message
Loading

Possibly related PRs

Poem

I'm a happy little rabbit, leaping through the code,
With url and email now neatly in a node.
The server hums, the client sings,
As logic flows and simplicity rings.
A boundless field of code unfurls—
My whiskers twitch with cheer for these changes, little pearls!
🐇✨

Warning

There were issues while running some tools. Please review the errors and either fix the tool’s configuration or disable the tool if it’s a critical failure.

🔧 ESLint

If the error stems from missing dependencies, add them to the package.json file. For unrecoverable errors (e.g., due to private dependencies), disable the tool in the CodeRabbit configuration.

apps/unsubscriber/src/main.ts

Oops! Something went wrong! :(

ESLint: 9.23.0

ESLint couldn't find an eslint.config.(js|mjs|cjs) file.

From ESLint v9.0.0, the default configuration file is now eslint.config.js.
If you are using a .eslintrc.* file, please follow the migration guide
to update your configuration file to the new format:

https://eslint.org/docs/latest/use/configure/migration-guide

If you still have problems after following the migration guide, please stop by
https://eslint.org/chat/help to chat with the team.

apps/web/env.ts

Oops! Something went wrong! :(

ESLint: 9.23.0

ESLint couldn't find an eslint.config.(js|mjs|cjs) file.

From ESLint v9.0.0, the default configuration file is now eslint.config.js.
If you are using a .eslintrc.* file, please follow the migration guide
to update your configuration file to the new format:

https://eslint.org/docs/latest/use/configure/migration-guide

If you still have problems after following the migration guide, please stop by
https://eslint.org/chat/help to chat with the team.

apps/web/app/(app)/bulk-unsubscribe/hooks.ts

Oops! Something went wrong! :(

ESLint: 9.23.0

ESLint couldn't find an eslint.config.(js|mjs|cjs) file.

From ESLint v9.0.0, the default configuration file is now eslint.config.js.
If you are using a .eslintrc.* file, please follow the migration guide
to update your configuration file to the new format:

https://eslint.org/docs/latest/use/configure/migration-guide

If you still have problems after following the migration guide, please stop by
https://eslint.org/chat/help to chat with the team.

  • 2 others
✨ Finishing Touches
  • 📝 Generate Docstrings

Thanks for using CodeRabbit! It's free for OSS, and your support helps us grow. If you like it, consider giving us a shout-out.

❤️ 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.
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai plan to trigger planning for file edits and PR creation.
  • @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: 3

🧹 Nitpick comments (3)
apps/unsubscriber/src/server.ts (1)

37-39: Verify error message to reflect both parameters.

The error message only mentions "Invalid URL provided" but now that the schema validates both URL and email, consider updating the error message to reflect both validation requirements.

-      return reply.status(400).send({ error: "Invalid URL provided" });
+      return reply.status(400).send({ error: "Invalid URL or email provided" });
apps/web/utils/actions/unsubscriber.ts (1)

47-49: Add more detailed error handling.

The current error handling only returns a generic "Failed to unsubscribe" message without including any details from the response.

- if (!response.ok) {
-   return { error: "Failed to unsubscribe" };
- }
+ if (!response.ok) {
+   try {
+     const errorData = await response.json();
+     return { error: errorData.message || "Failed to unsubscribe" };
+   } catch (e) {
+     return { error: `Failed to unsubscribe: ${response.status} ${response.statusText}` };
+   }
+ }
apps/web/app/(app)/bulk-unsubscribe/hooks.ts (1)

51-86: POC implementation needs testing and error handling

According to the PR objectives, this is a POC that hasn't been tested. Before merging, consider adding proper testing for this functionality.

Improve the error handling by checking the result of the unsubscribe action:

if (item.lastUnsubscribeLink) {
-  await unsubscribeAction({ url: item.lastUnsubscribeLink });
+  const result = await unsubscribeAction({ url: item.lastUnsubscribeLink });
+  if (result.error) {
+    toast.error(`Failed to unsubscribe: ${result.error}`);
+    return;
+  }
}

Also, consider adding a loading indicator or toast notification to indicate that an unsubscribe action is in progress.

📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 74c3373 and d9e3013.

📒 Files selected for processing (5)
  • apps/unsubscriber/src/main.ts (1 hunks)
  • apps/unsubscriber/src/server.ts (2 hunks)
  • apps/web/app/(app)/bulk-unsubscribe/hooks.ts (3 hunks)
  • apps/web/env.ts (1 hunks)
  • apps/web/utils/actions/unsubscriber.ts (2 hunks)
🧰 Additional context used
🧬 Code Definitions (3)
apps/web/app/(app)/bulk-unsubscribe/hooks.ts (1)
apps/web/utils/actions/unsubscriber.ts (1)
  • unsubscribeAction (35-53)
apps/web/utils/actions/unsubscriber.ts (1)
apps/web/env.ts (1)
  • env (5-215)
apps/unsubscriber/src/server.ts (1)
apps/unsubscriber/src/main.ts (1)
  • autoUnsubscribe (206-276)
🔇 Additional comments (8)
apps/web/env.ts (1)

82-82: Added new environment variable for unsubscriber service.

This new optional environment variable UNSUBSCRIBER_API_URL allows configuring the endpoint for the unsubscribe service, making the system more flexible for different deployment configurations.

apps/unsubscriber/src/server.ts (2)

17-20: Updated schema to include email parameter.

The unsubscribe schema now requires both URL and email fields, which aligns with the updated function signature in the main module.


28-29: Updated function call to use the new parameter structure.

The route handler now correctly destructures both url and email from the request body and passes them as an object to the autoUnsubscribe function, maintaining consistency with the updated function signature.

apps/unsubscriber/src/main.ts (1)

206-212:

❓ Verification inconclusive

Updated function signature to include email parameter.

The autoUnsubscribe function now accepts an object with both url and email properties instead of just a URL string. This change allows for more contextual information to be passed to the function.

While the function signature has been updated to include the email parameter, I notice that the email isn't being used in the function body. This might be intentional as part of the POC, but it's worth confirming whether this is expected behavior.


🏁 Script executed:

#!/bin/bash
# Check if the email parameter is used anywhere in the function body
grep -A 70 "export async function autoUnsubscribe" apps/unsubscriber/src/main.ts | grep -E "email\s*[=:]"

Length of output: 124


Rewritten Review Comment

The updated function signature now includes an email property in addition to url as part of an object parameter for autoUnsubscribe. The grep check confirms that email is declared only in the function’s parameters and isn’t used within its body. This might be intentional for the POC, but please verify whether the omission of any logic using email is by design or if further implementation is expected.

  • Location: apps/unsubscriber/src/main.ts, around lines 206-212
  • Observation: The email parameter appears only in the type declaration, with no ensuing usage inside the function.
apps/web/utils/actions/unsubscriber.ts (1)

7-7: Added env import for API URL configuration.

The env import is required to access the UNSUBSCRIBER_API_URL configuration value.

apps/web/app/(app)/bulk-unsubscribe/hooks.ts (3)

7-10: Import structure updated to include unsubscribeAction

The import now correctly includes the new unsubscribeAction function alongside the existing setNewsletterStatusAction. This is necessary for the auto-unsubscribe functionality being implemented.


82-82: Added lastUnsubscribeLink to dependency array

Correctly added item.lastUnsubscribeLink to the dependency array of the useCallback, ensuring the callback is updated when this property changes.


58-70:

❓ Verification inconclusive

Potential logic flow issue with status updates

The code automatically attempts to unsubscribe when item.status === NewsletterStatus.UNSUBSCRIBED and then sets the status to null. This might be counter-intuitive as you're taking an action on an already unsubscribed newsletter.

Verify if this is the intended behavior. Should the auto-unsubscribe action happen when:

  1. The user is marking a newsletter as unsubscribed (else branch)
  2. Or when they're resetting the status of an already unsubscribed newsletter (current implementation)

🌐 Web query:

What does setting a newsletter status to null mean in the context of email unsubscribe systems?

💡 Result:

Setting a newsletter status to null in email unsubscribe systems typically indicates removing or invalidating the subscription record, but its exact implications depend on the system's architecture and data handling. Here's a breakdown:


Key Implications

  1. Global Unsubscription
    In systems like Salesforce Marketing Cloud[4], setting a status to null may trigger a master unsubscribe, blocking all emails across an organization. This differs from list-specific unsubscribes and affects:

    • Business unit-level sends
    • Enterprise-wide campaigns
    • Automated workflows tied to subscriber status
  2. SPF Authentication and Bounce Handling
    Null sender addresses (e.g., <>) are used for emails where delivery notifications aren’t required (e.g., bounce messages)[1]. However, in unsubscribe systems, null statuses might bypass standard SPF checks if the domain lacks proper HELO configurations[1].

  3. Data View Conflicts

    • _Unsubscribe Data Views track events (e.g., clicks, API calls)[4], while _Subscribers reflects current status. Setting a status to null may disconnect event logs from active records, requiring manual reconciliation.
  4. Technical Risks

    • Errors in Automation: Missing email addresses (TargetEmail = null) disrupt processes like indemnification workflows[5].
    • List-Unsubscribe Headers: Improperly configured null states can break unsubscribe headers, leading to spam complaints[12].

Common Use Cases

  • Bulk Removal: Clearing subscription data during migrations or compliance audits.
  • Opt-Out Enforcement: Resetting users to a "never subscribed" state instead of retaining explicit unsubscription records[2].
  • System Cleanup: Deleting inactive or invalid email addresses from databases[5].

Best Practices

  • Explicit Statuses: Use Unsubscribed, Active, or Blocked instead of null to avoid ambiguity[8].
  • Data Validation: Ensure emails aren’t null before processing unsubscriptions[5].
  • Audit Logs: Track null status changes to troubleshoot delivery failures[4].

In most systems, null is a blunt tool for subscription management. Prefer granular controls (e.g., list-level unsubscribes or global opt-outs) to maintain deliverability and compliance.

Citations:


Action Required: Verify Unsubscribe Flow and Status Update Logic

The current implementation automatically triggers an unsubscribe action for newsletters already marked as unsubscribed. This is immediately followed by updating the status to null, which—according to common practices in email unsubscribe systems—typically serves to remove or invalidate the subscription record. However, this behavior could be counterintuitive if the intent was to perform the unsubscribe action only when changing the status from an active state.

Please verify whether the intended behavior is to:

  • Clean up unsubscribed records: Automatically remove the subscription (by setting the status to null) for newsletters that are already unsubscribed, potentially as a data cleanup mechanism.
  • User-triggered unsubscription: Execute the unsubscribe action only when the user is marking a newsletter as unsubscribed (i.e., via the else branch), rather than performing this action on an already unsubscribed record.

The code in question (located at apps/web/app/(app)/bulk-unsubscribe/hooks.ts, lines 58–70) currently handles the two branches as follows:

  • If item.status === NewsletterStatus.UNSUBSCRIBED:
    • Calls unsubscribeAction if a lastUnsubscribeLink is available.
    • Updates the newsletter status to null (effectively removing the record).
    • Calls mutate() to refresh the state.
  • Else:
    • Calls unsubscribeAndArchive, handling both unsubscription and archival for newsletters not yet marked as unsubscribed.

A brief clarification in the code comments might help future maintainers understand the rationale behind this flow.

Comment on lines +43 to +46
const response = await fetch(
`${env.UNSUBSCRIBER_API_URL}/unsubscribe?url=${url}&email=${session.user.email}`,
);

Copy link
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue

API endpoint inconsistency with server implementation.

The server implementation in apps/unsubscriber/src/server.ts expects the URL and email in the request body, but this client is sending them as query parameters.

The request format should match what the server expects. The server is set up to parse the request body, not query parameters.

Comment on lines +35 to +53
export const unsubscribeAction = withActionInstrumentation(
"unsubscribe",
async (options: { url: string }) => {
const session = await auth();
if (!session?.user.email) return { error: "Not logged in" };

const { url } = options;

const response = await fetch(
`${env.UNSUBSCRIBER_API_URL}/unsubscribe?url=${url}&email=${session.user.email}`,
);

if (!response.ok) {
return { error: "Failed to unsubscribe" };
}

return { success: true };
},
);
Copy link
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue

Implemented new server action for unsubscription.

The new unsubscribeAction server action correctly:

  1. Verifies user authentication
  2. Extracts the URL from options
  3. Makes a request to the unsubscriber API
  4. Handles the response appropriately

The URL parameters in the fetch request are not being encoded, which could lead to issues with special characters in URLs or emails:

- const response = await fetch(
-   `${env.UNSUBSCRIBER_API_URL}/unsubscribe?url=${url}&email=${session.user.email}`,
- );
+ const response = await fetch(
+   `${env.UNSUBSCRIBER_API_URL}/unsubscribe`, {
+     method: 'POST',
+     headers: {
+       'Content-Type': 'application/json',
+     },
+     body: JSON.stringify({ url, email: session.user.email }),
+   }
+ );

This change:

  1. Uses POST with a JSON body instead of query parameters
  2. Avoids URL encoding issues
  3. Aligns with how the server is expecting to receive the data (in the request body)
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
export const unsubscribeAction = withActionInstrumentation(
"unsubscribe",
async (options: { url: string }) => {
const session = await auth();
if (!session?.user.email) return { error: "Not logged in" };
const { url } = options;
const response = await fetch(
`${env.UNSUBSCRIBER_API_URL}/unsubscribe?url=${url}&email=${session.user.email}`,
);
if (!response.ok) {
return { error: "Failed to unsubscribe" };
}
return { success: true };
},
);
export const unsubscribeAction = withActionInstrumentation(
"unsubscribe",
async (options: { url: string }) => {
const session = await auth();
if (!session?.user.email) return { error: "Not logged in" };
const { url } = options;
const response = await fetch(
`${env.UNSUBSCRIBER_API_URL}/unsubscribe`, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({ url, email: session.user.email }),
}
);
if (!response.ok) {
return { error: "Failed to unsubscribe" };
}
return { success: true };
},
);
Comment on lines +60 to +63
if (item.lastUnsubscribeLink) {
await unsubscribeAction({ url: item.lastUnsubscribeLink });
}

Copy link
Contributor

Choose a reason for hiding this comment

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

💡 Verification agent

🧩 Analysis chain

Added auto-unsubscribe functionality using the unsubscribe link

This change adds automatic unsubscription via API when a user has an existing unsubscribe link. However, there's a potential inconsistency between the implementation and expected behavior.

While the code passes only the URL to unsubscribeAction, the AI summary suggests the function should accept both url and email parameters. Let's verify the actual implementation:

Also, there's no handling for potential failures from the unsubscribe API call. Consider adding a check for the response and only setting the newsletter status to null if the unsubscribe action succeeds.


🏁 Script executed:

#!/bin/bash
# Check if unsubscribeAction accepts both url and email or just url
cat apps/web/utils/actions/unsubscriber.ts | grep -A 10 "unsubscribeAction"

Length of output: 451


Action Required: Improve error handling in auto-unsubscribe flow

  • The unsubscribeAction function is correctly implemented to accept only a url parameter, with the email automatically retrieved from the session. There’s no need to pass the email explicitly.
  • However, the current implementation in apps/web/app/(app)/bulk-unsubscribe/hooks.ts does not handle failures from the unsubscribe API call. Consider checking the response from unsubscribeAction and only updating the newsletter status to null if the action indicates success.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
1 participant