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

Comprehensive Newsletter functionality #3939

Open
wants to merge 14 commits into
base: main
Choose a base branch
from

Conversation

tsu-ki
Copy link
Contributor

@tsu-ki tsu-ki commented Mar 14, 2025

Closes: #3725

WIP, currently requires testing. I've created newsletter modals and incorporated various functions which provides users with functionality mentioned below:

  1. User Functionality:
  • Subscribe to newsletters
  • Manage their subscription preferences
  • View past newsletters
  • Receive personalised newsletters based on their preferences
  • Unsubscribe if they wish
  1. The admin interface allows site administrators to:
  • Create and publish newsletters
  • Send newsletters to subscribers
  • Manage subscriber lists
  • Track newsletter views and engagement

Implementation includes proper subscription management, confirmation emails, and preference controls.

Summary by CodeRabbit

  • New Features

    • Launched a comprehensive newsletter system with dedicated pages for viewing, subscribing, confirming, unsubscribing, and managing preferences.
    • Introduced engaging UI enhancements including a subscription banner, intuitive sidebar navigation, and detailed newsletter display.
    • Upgraded email communications with modern, responsive confirmation and newsletter update templates.
    • Enhanced administrative tools for efficient newsletter and subscriber management.
    • Added a markdown filter for improved content formatting in newsletters.
  • Bug Fixes

    • Resolved issues related to user activity logging during newsletter interactions.
  • Documentation

    • Updated templates and context processors to improve user experience regarding newsletter subscriptions and management.

@tsu-ki
Copy link
Contributor Author

tsu-ki commented Mar 15, 2025

@DonnieBLT sir, please review the newsletter functionality I've implemented:

Screen.Recording.2025-03-15.at.3.32.13.PM.mov

@DonnieBLT
Copy link
Collaborator

@tsu-ki sure, can you please resolve the conflicts and fix the video / screenshot?

@OWASP-BLT OWASP-BLT deleted a comment from github-actions bot Mar 15, 2025
@DonnieBLT DonnieBLT marked this pull request as ready for review March 15, 2025 10:00
Copy link
Contributor

coderabbitai bot commented Mar 15, 2025

Walkthrough

This change set introduces an extensive newsletter feature. It adds new Django models for newsletters and newsletter subscribers along with associated migrations. New view functions, URL routes, and templates support newsletter subscription, confirmation, unsubscription, preference management, and detail display. A management command for sending newsletters is implemented, and new admin classes enable management via the admin interface. Additionally, relevant settings, context processors, custom template filters, and dependency updates are applied to integrate Markdown rendering and newsletter functionality within the project.

Changes

File(s) Change Summary
blt/settings.py
blt/urls.py
Modified environment variable handling and added a new context processor for newsletters; introduced multiple newsletter URL patterns and imported include in URLs, removing an old management commands route.
website/admin.py
website/management/commands/send_newsletter.py
Introduced NewsletterAdmin and NewsletterSubscriberAdmin classes in the admin; added a management command to send newsletters, including methods to handle email sending and error logging.
website/models.py
website/migrations/0233_ & 0234_ **
Added Newsletter and NewsletterSubscriber models with their methods and properties; created migrations for these models and added a token_created_at field to NewsletterSubscriber.
website/templates/base.html
website/templates/includes/sidenav.html
website/templates/newsletter/
Updated base and sidenav templates to include a newsletter subscription banner and navigation link; added new newsletter templates for detail, email (confirmation and newsletter), home, preferences, and subscription pages.
website/views/core.py
website/views/user.py
website/templatetags/custom_filters.py
Enhanced view functions by adding a newsletter context processor and various newsletter-related functions (home, detail, subscribe, confirm, unsubscribe, preferences, resend confirmation); added a markdown_filter for Markdown conversion in templates.
pyproject.toml Downgraded django-gravatar2 version and added new dependencies: environ, google-auth, django-markdown-deux, bleach, django-anymail, and django-ckeditor.

Sequence Diagram(s)

sequenceDiagram
    participant U as User
    participant FE as Frontend (subscribe.html)
    participant VS as Newsletter Subscribe View
    participant NE as send_confirmation_email
    participant ES as Email Server
    participant EC as User Email Client

    U->>FE: Submit subscription form
    FE->>VS: POST subscription data
    VS->>NE: Create/update subscriber and send confirmation
    NE->>ES: Send confirmation email (confirmation_email.html)
    ES->>EC: Deliver confirmation email
    EC->>U: User receives email
    U->>VS: Clicks confirmation link (newsletter_confirm view)
    VS->>VS: Validate token and update subscription status
Loading
sequenceDiagram
    participant A as Admin
    participant CMD as send_newsletter Command
    participant NL as Newsletter Model
    participant NS as NewsletterSubscriber Model
    participant ES as Email Server

    A->>CMD: Run send_newsletter command
    CMD->>NL: Retrieve newsletter(s) pending send
    CMD->>NS: Retrieve active and confirmed subscribers
    loop For each subscriber
        CMD->>ES: Send newsletter email (using EmailMultiAlternatives)
    end
    CMD->>NL: Mark newsletter as sent
Loading

📜 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 40c14bd and 803feff.

📒 Files selected for processing (1)
  • website/migrations/0234_newslettersubscriber_token_created_at.py (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • website/migrations/0234_newslettersubscriber_token_created_at.py
⏰ Context from checks skipped due to timeout of 90000ms (2)
  • GitHub Check: Run Tests
  • GitHub Check: docker-test

🪧 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: 5

♻️ Duplicate comments (1)
website/views/user.py (1)

1322-1337: ⚠️ Potential issue

Fix security vulnerability in error handling.

The current implementation exposes exception details to users, which could leak sensitive information about your system.

Apply this fix to sanitize error responses:

    try:
        data = json.loads(request.body)
        email = data.get("email")

        if not email:
            return JsonResponse({"success": False, "error": "Email is required"})

        subscriber = get_object_or_404(NewsletterSubscriber, email=email, is_active=True, confirmed=False)
        send_confirmation_email(subscriber)

        return JsonResponse({"success": True})
    except Exception as e:
-        return JsonResponse({"success": False, "error": str(e)})
+        # Log the actual error for debugging purposes
+        logger.error("Error in newsletter_resend_confirmation: %s", str(e))
+        # Return a generic error message to the user
+        return JsonResponse({"success": False, "error": "An error occurred while processing your request."})

This change protects your application from potentially leaking sensitive information while still providing useful logs for debugging.

🧰 Tools
🪛 GitHub Check: CodeQL

[warning] 1337-1337: Information exposure through an exception
Stack trace information flows to this location and may be exposed to an external user.

🧹 Nitpick comments (15)
website/templates/base.html (1)

114-136: Well-implemented conditional newsletter subscription banner

The newsletter subscription banner is properly implemented with conditional display logic that ensures it only appears to non-subscribed users. The design is clean, non-intrusive, and follows the site's styling conventions.

Consider replacing the HTML entity reference &rarr; with either a Unicode arrow character (→) or an SVG icon for better accessibility and to resolve the pipeline warning.

-                            Subscribe <span aria-hidden="true">&rarr;</span>
+                            Subscribe <span aria-hidden="true">→</span>
🧰 Tools
🪛 GitHub Actions: CI/CD Optimized

[warning] 131-131: H023 Do not use entity references.

website/templates/newsletter/email/newsletter_email.html (1)

1-251: Well-structured responsive email template with some accessibility improvements needed

The newsletter email template is well-designed with responsive elements and proper conditional sections. The table-based layout and inline styles are appropriate for email templates to ensure compatibility across email clients.

Add the lang attribute to the HTML tag for better accessibility:

-<html>
+<html lang="en">

While the pipeline flags the use of inline styles, these are actually best practice for email templates to ensure consistent rendering across different email clients. No changes needed for the styling approach.

🧰 Tools
🪛 GitHub Actions: CI/CD Optimized

[warning] 2-2: H005 Html tag should have lang attribute.


[warning] 2-2: H030 Consider adding a meta description.


[warning] 2-2: H031 Consider adding meta keywords.


[warning] 45-45: H021 Inline styles should be avoided.


[warning] 54-54: H021 Inline styles should be avoided.


[warning] 66-66: H021 Inline styles should be avoided.


[warning] 70-70: H021 Inline styles should be avoided.


[warning] 81-81: H021 Inline styles should be avoided.


[warning] 88-88: H021 Inline styles should be avoided.


[warning] 99-99: H021 Inline styles should be avoided.


[warning] 106-106: H021 Inline styles should be avoided.


[warning] 116-116: H021 Inline styles should be avoided.


[warning] 120-120: H021 Inline styles should be avoided.


[warning] 128-128: H021 Inline styles should be avoided.


[warning] 133-133: H021 Inline styles should be avoided.


[warning] 140-140: H021 Inline styles should be avoided.


[warning] 153-153: H021 Inline styles should be avoided.


[warning] 159-159: H021 Inline styles should be avoided.


[warning] 169-169: H021 Inline styles should be avoided.


[warning] 178-178: H021 Inline styles should be avoided.


[warning] 202-202: H021 Inline styles should be avoided.


[warning] 209-209: H021 Inline styles should be avoided.


[warning] 226-226: H021 Inline styles should be avoided.


[warning] 231-231: H021 Inline styles should be avoided.


[warning] 241-241: H021 Inline styles should be avoided.

website/templates/newsletter/subscribe.html (1)

61-66: Consider adding a link to the actual privacy policy document.

The privacy policy section mentions that data will be handled according to the privacy policy, but there is no link to the actual privacy policy document. For transparency and to comply with data protection regulations, consider adding a link to the full privacy policy.

<div class="ml-3 text-sm">
    <label for="privacy" class="font-medium text-gray-700">I agree to the privacy policy</label>
    <p class="text-gray-500">
        We'll handle your data according to our privacy policy and will never share it with third parties.
+       <a href="{% url 'privacy_policy' %}" class="text-blue-600 hover:text-blue-500" target="_blank">View our Privacy Policy</a>.
    </p>
</div>
website/templates/newsletter/email/confirmation_email.html (1)

4-6: Add meta description to fix pipeline warning

The CI pipeline indicates missing meta description and keywords tags. While not critical for email templates, adding them can improve accessibility and SEO when emails are viewed in a browser.

<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
+ <meta name="description" content="Confirm your subscription to the {{ project_name }} newsletter">
+ <meta name="keywords" content="newsletter, subscription, confirmation, {{ project_name }}">
<title>Confirm Your Subscription</title>
website/templates/newsletter/detail.html (1)

73-147: Consider lazy loading for dynamic content sections

The template includes three conditional sections for recent bugs, leaderboard updates, and reported IPs. These sections might cause performance issues if they involve complex database queries. Consider implementing lazy loading or pagination for these sections, especially if they contain large datasets.

website/templates/newsletter/home.html (1)

158-184: Improve user feedback for subscription states

The template has good conditional rendering based on subscription status, but consider adding a mechanism for users to resend confirmation emails if they haven't received one, especially in the section where it shows "Please confirm your subscription by checking your email."

<div class="mb-4 p-3 bg-yellow-50 text-yellow-800 rounded-md">
    <p>Please confirm your subscription by checking your email.</p>
+   <p class="mt-2">
+       <a href="{% url 'newsletter_resend_confirmation' %}" class="text-blue-600 hover:underline">
+           Didn't receive the email? Click here to resend
+       </a>
+   </p>
</div>
website/migrations/0231_newsletter_newslettersubscriber.py (2)

23-23: Consider pre-populating slug field from title

The slug field is currently blank=True, which means it could be left empty. For better URL structure and SEO, consider adding a pre-save signal to automatically generate the slug from the title if it's not provided.

# Add to models.py after the Newsletter model definition
@receiver(pre_save, sender=Newsletter)
def create_newsletter_slug(sender, instance, **kwargs):
    if not instance.slug:
        instance.slug = slugify(instance.title)

59-61: Use a JSON field for subscription preferences

The model currently uses separate boolean fields for different subscription preferences. Consider using a JSONField instead to make it easier to add new preference types in the future without requiring schema migrations.

- ("wants_bug_reports", models.BooleanField(default=True)),
- ("wants_leaderboard_updates", models.BooleanField(default=True)),
- ("wants_security_news", models.BooleanField(default=True)),
+ ("preferences", models.JSONField(default=dict, help_text="Subscriber preferences as JSON")),

Then you would initialize it with:

# Default preferences 
default_preferences = {
    'wants_bug_reports': True,
    'wants_leaderboard_updates': True, 
    'wants_security_news': True
}
website/templates/newsletter/preferences.html (1)

151-151: Avoid HTML entity references in favor of Unicode characters

The HTML entity reference &larr; has triggered a warning in your pipeline: "H023 Do not use entity references." Instead, use Unicode characters directly for better compatibility and cleaner code.

-                   class="text-sm font-medium text-blue-600 hover:text-blue-500">&larr; Back to Newsletters</a>
+                   class="text-sm font-medium text-blue-600 hover:text-blue-500">← Back to Newsletters</a>
🧰 Tools
🪛 GitHub Actions: CI/CD Optimized

[warning] 151-151: H023 Do not use entity references.

website/views/core.py (1)

2772-2794: Newsletter context processor implementation

The newsletter_context_processor function is well-implemented but could benefit from improved error handling in the try-except block.

 try:
     context["latest_newsletter"] = Newsletter.objects.filter(status="published").order_by("-published_at").first()
-except:
+except Exception as e:
+    logger.error(f"Error retrieving latest newsletter: {str(e)}")
     pass

Adding specific error logging would help with debugging if there are issues with the newsletter functionality.

website/management/commands/send_newsletter.py (1)

89-101: Consider refactoring unsubscribe URL generation

The code for generating the unsubscribe URL could be more clearly written to handle the case when subscriber is None.

 context = {
     "newsletter": newsletter,
     "subscriber": subscriber,
-    "unsubscribe_url": settings.DOMAIN_NAME
-    + reverse("newsletter_unsubscribe", args=[subscriber.confirmation_token])
-    if subscriber
-    else "#",
+    "unsubscribe_url": (
+        settings.DOMAIN_NAME + reverse("newsletter_unsubscribe", args=[subscriber.confirmation_token])
+        if subscriber else "#"
+    ),
     "view_in_browser_url": settings.DOMAIN_NAME + newsletter.get_absolute_url(),
     "project_name": settings.PROJECT_NAME,
     "recent_bugs": newsletter.get_recent_bugs(),
     "leaderboard": newsletter.get_leaderboard_updates(),
     "reported_ips": newsletter.get_reported_ips(),
 }
website/views/user.py (1)

1129-1155: LGTM - Solid implementation of the newsletter home view.

The view properly filters, paginates, and renders newsletters while logging user activity.

However, the Activity object creation uses a hardcoded object_id=1 which might not represent a valid object:

-            object_id=1,
+            object_id=featured_newsletter.id if featured_newsletter else None,
website/models.py (3)

2300-2331: Consider slug collision handling for uniqueness.
The Newsletter class is well-defined with comprehensive fields and a suitable meta configuration. However, when generating a slug in the save method, consider looping to handle collisions if the slug already exists. Currently, if two newsletters share the same title, the second save may fail due to the unique=True constraint.

Below is an example approach where you check for existing slugs and append a counter:

 def save(self, *args, **kwargs):
     if not self.slug:
         base_slug = slugify(self.title)
+        unique_slug = base_slug
+        counter = 1
+        while Newsletter.objects.filter(slug=unique_slug).exists():
+            unique_slug = f"{base_slug}-{counter}"
+            counter += 1
+        self.slug = unique_slug

     if self.status == "published" and not self.published_at:
         self.published_at = timezone.now()
     super().save(*args, **kwargs)

2332-2397: Use atomic increments for view counts and evaluate potential indexing.

  1. The increment_view_count method directly reads and writes the view_count field. In a high-traffic scenario, concurrent calls might overwrite increments. Consider using Django’s F expressions to ensure atomic updates:
 def increment_view_count(self):
     """Increment the view count for this newsletter"""
-    self.view_count += 1
-    self.save(update_fields=["view_count"])
+    from django.db.models import F
+    Newsletter.objects.filter(pk=self.pk).update(view_count=F("view_count") + 1)
  1. get_recent_bugs, get_leaderboard_updates, and get_reported_ips may be referencing multiple joins/tables. When these lists grow large, you might introduce performance overhead. Consider adding database-level indexes and caching if usage patterns become heavy. At first glance, these are fine for moderate datasets.

2400-2418: Potential privacy consideration for email storage.
Storing subscriber emails in plain text is common for newsletters, but you might consider hashing or encrypting them at rest if privacy or GDPR concerns arise. Also, you might eventually add an index on confirmation_token if you frequently look up subscribers by token.

📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between d3d226b and b8976d1.

📒 Files selected for processing (16)
  • blt/settings.py (1 hunks)
  • blt/urls.py (3 hunks)
  • website/admin.py (2 hunks)
  • website/management/commands/send_newsletter.py (1 hunks)
  • website/migrations/0231_newsletter_newslettersubscriber.py (1 hunks)
  • website/models.py (3 hunks)
  • website/templates/base.html (1 hunks)
  • website/templates/includes/sidenav.html (1 hunks)
  • website/templates/newsletter/detail.html (1 hunks)
  • website/templates/newsletter/email/confirmation_email.html (1 hunks)
  • website/templates/newsletter/email/newsletter_email.html (1 hunks)
  • website/templates/newsletter/home.html (1 hunks)
  • website/templates/newsletter/preferences.html (1 hunks)
  • website/templates/newsletter/subscribe.html (1 hunks)
  • website/views/core.py (4 hunks)
  • website/views/user.py (6 hunks)
🧰 Additional context used
🪛 GitHub Actions: CI/CD Optimized
website/templates/newsletter/email/newsletter_email.html

[warning] 2-2: H005 Html tag should have lang attribute.


[warning] 2-2: H030 Consider adding a meta description.


[warning] 2-2: H031 Consider adding meta keywords.


[warning] 45-45: H021 Inline styles should be avoided.


[warning] 54-54: H021 Inline styles should be avoided.


[warning] 66-66: H021 Inline styles should be avoided.


[warning] 70-70: H021 Inline styles should be avoided.


[warning] 81-81: H021 Inline styles should be avoided.


[warning] 88-88: H021 Inline styles should be avoided.


[warning] 99-99: H021 Inline styles should be avoided.


[warning] 106-106: H021 Inline styles should be avoided.


[warning] 116-116: H021 Inline styles should be avoided.


[warning] 120-120: H021 Inline styles should be avoided.


[warning] 128-128: H021 Inline styles should be avoided.


[warning] 133-133: H021 Inline styles should be avoided.


[warning] 140-140: H021 Inline styles should be avoided.


[warning] 153-153: H021 Inline styles should be avoided.


[warning] 159-159: H021 Inline styles should be avoided.


[warning] 169-169: H021 Inline styles should be avoided.


[warning] 178-178: H021 Inline styles should be avoided.


[warning] 202-202: H021 Inline styles should be avoided.


[warning] 209-209: H021 Inline styles should be avoided.


[warning] 226-226: H021 Inline styles should be avoided.


[warning] 231-231: H021 Inline styles should be avoided.


[warning] 241-241: H021 Inline styles should be avoided.

website/templates/base.html

[warning] 131-131: H023 Do not use entity references.

website/templates/newsletter/email/confirmation_email.html

[warning] 2-2: H005 Html tag should have lang attribute.


[warning] 2-2: H030 Consider adding a meta description.


[warning] 2-2: H031 Consider adding meta keywords.


[warning] 45-45: H021 Inline styles should be avoided.


[warning] 54-54: H021 Inline styles should be avoided.


[warning] 66-66: H021 Inline styles should be avoided.


[warning] 70-70: H021 Inline styles should be avoided.


[warning] 81-81: H021 Inline styles should be avoided.


[warning] 88-88: H021 Inline styles should be avoided.


[warning] 99-99: H021 Inline styles should be avoided.


[warning] 106-106: H021 Inline styles should be avoided.


[warning] 116-116: H021 Inline styles should be avoided.


[warning] 120-120: H021 Inline styles should be avoided.


[warning] 128-128: H021 Inline styles should be avoided.


[warning] 133-133: H021 Inline styles should be avoided.


[warning] 140-140: H021 Inline styles should be avoided.


[warning] 153-153: H021 Inline styles should be avoided.


[warning] 159-159: H021 Inline styles should be avoided.


[warning] 169-169: H021 Inline styles should be avoided.


[warning] 178-178: H021 Inline styles should be avoided.


[warning] 202-202: H021 Inline styles should be avoided.


[warning] 209-209: H021 Inline styles should be avoided.


[warning] 226-226: H021 Inline styles should be avoided.


[warning] 231-231: H021 Inline styles should be avoided.


[warning] 241-241: H021 Inline styles should be avoided.

website/templates/newsletter/detail.html

[warning] 60-60: H006 Img tag should have height and width attributes.

website/templates/newsletter/preferences.html

[warning] 151-151: H023 Do not use entity references.

website/templates/newsletter/home.html

[warning] 33-33: H006 Img tag should have height and width attributes.

🪛 GitHub Check: CodeQL
website/views/user.py

[warning] 1337-1337: Information exposure through an exception
Stack trace information flows to this location and may be exposed to an external user.

🔇 Additional comments (33)
blt/settings.py (1)

172-172: Properly implemented newsletter context processor

The addition of a newsletter context processor is a clean approach to make newsletter-related data available globally to all templates. This follows Django's recommended pattern for making variables accessible throughout the site.

website/templates/includes/sidenav.html (1)

746-759: Well-integrated newsletter navigation link

The newsletter link has been properly implemented in the sidebar with consistent styling and structure. The SVG icon provides a good visual indicator for the newsletter functionality, and the dynamic URL generation follows best practices.

website/templates/newsletter/email/newsletter_email.html (1)

147-152: Verify the link to individual bugs

The link for bug descriptions is using {{ view_in_browser_url }} for all bugs, which doesn't seem to link to individual bug pages.

Consider whether each bug should link to its specific page instead of the general newsletter page:

-<a href="{{ view_in_browser_url }}">{{ bug.description|truncatechars:100 }}</a>
+<a href="{{ site_url }}{% url 'bug_details' bug.id %}">{{ bug.description|truncatechars:100 }}</a>

This assumes there's a bug_details URL pattern that accepts a bug ID parameter.

website/templates/newsletter/subscribe.html (1)

28-74:

❓ Verification inconclusive

Validate subscription form data server-side

The form includes client-side validation through HTML attributes, but ensure that server-side validation is also implemented in the view function that processes this form to prevent malicious submissions or data bypass.


🏁 Script executed:

#!/bin/bash
# Find the newsletter_subscribe view that handles this form submission
rg -A 10 "def newsletter_subscribe" website/views/

Length of output: 671


Enhance Server-Side Validation for Newsletter Subscription

The newsletter subscription view in website/views/user.py already performs a basic check by confirming that an email is provided. However, please consider enhancing the server-side validation by:

  • Validating the email format (for example, leveraging Django’s built-in validators).
  • Sanitizing the inputs further to mitigate malicious submissions.
  • Ensuring that duplicate subscriptions or other edge cases are adequately handled.

These improvements will better secure the endpoint against potential bypasses and unexpected input.

website/templates/newsletter/email/confirmation_email.html (2)

45-264: Note about inline styles flagged in pipeline warnings

The pipeline is flagging numerous inline styles as warnings. However, for HTML emails, inline styles are actually recommended over external stylesheets for maximum compatibility across email clients. Therefore, these warnings can be safely ignored for this specific template.

🧰 Tools
🪛 GitHub Actions: CI/CD Optimized

[warning] 45-45: H021 Inline styles should be avoided.


[warning] 54-54: H021 Inline styles should be avoided.


[warning] 66-66: H021 Inline styles should be avoided.


[warning] 70-70: H021 Inline styles should be avoided.


[warning] 81-81: H021 Inline styles should be avoided.


[warning] 88-88: H021 Inline styles should be avoided.


[warning] 99-99: H021 Inline styles should be avoided.


[warning] 106-106: H021 Inline styles should be avoided.


[warning] 116-116: H021 Inline styles should be avoided.


[warning] 120-120: H021 Inline styles should be avoided.


[warning] 128-128: H021 Inline styles should be avoided.


[warning] 133-133: H021 Inline styles should be avoided.


[warning] 140-140: H021 Inline styles should be avoided.


[warning] 153-153: H021 Inline styles should be avoided.


[warning] 159-159: H021 Inline styles should be avoided.


[warning] 169-169: H021 Inline styles should be avoided.


[warning] 178-178: H021 Inline styles should be avoided.


[warning] 202-202: H021 Inline styles should be avoided.


[warning] 209-209: H021 Inline styles should be avoided.


[warning] 226-226: H021 Inline styles should be avoided.


[warning] 231-231: H021 Inline styles should be avoided.


[warning] 241-241: H021 Inline styles should be avoided.


138-139:

✅ Verification successful

Ensure proper variable availability for {{ project_name }}

The template uses the {{ project_name }} variable in multiple places. Make sure this variable is properly provided in the context when rendering this template, or consider using a default value with the default filter.

Also applies to: 206-217


🏁 Script executed:

#!/bin/bash
# Check the context data being passed to this template
rg -A 10 "send_confirmation_email|render_to_string.*confirmation_email" website/

Length of output: 4650


Review Comment Resolution: Context Variable Verified

The verification confirms that the {{ project_name }} variable is correctly provided via the context in the send_confirmation_email function (using settings.PROJECT_NAME). No further action is needed. If there's a concern about the variable being unset in some edge cases, consider providing a default value using the template’s default filter; otherwise, this setup is sufficient.

website/templates/newsletter/detail.html (1)

71-71:

❓ Verification inconclusive

Ensure safe Markdown rendering

The template renders Markdown content directly which could potentially lead to XSS vulnerabilities if the content includes malicious HTML. Verify that the markdown filter properly sanitizes the HTML output.


🏁 Script executed:

#!/bin/bash
# Check how the markdown filter is implemented
rg -A 5 "def markdown" website/
# Also check if there's a custom templatetag for markdown
fd -e py markdown website/templatetags/

Length of output: 412


Action Required: Revisit Markdown Filter Behavior for Intended Output
It appears the markdown filter (located in website/utils.py as markdown_to_text) converts markdown into HTML and then strips all HTML tags using BeautifulSoup’s get_text(). This approach effectively prevents XSS vulnerabilities by removing any embedded HTML, but it also removes all styling and formatting from the original markdown. Please verify that returning plain text is the desired behavior for newsletter rendering. If preserving some markdown formatting is intended while still mitigating XSS risks, consider switching to a sanitization strategy (such as using Bleach with an allowlist of safe tags).

  • Location: website/utils.py (the markdown_to_text function)
  • Action: Confirm whether plain text output is acceptable or if a safe HTML rendering (with allowed formatting) is needed
website/templates/newsletter/home.html (1)

65-67:

❓ Verification inconclusive

Verify the security of Markdown rendering

The template renders Markdown content directly. Ensure that the markdown filter properly sanitizes the HTML output to prevent potential XSS vulnerabilities.


🏁 Script executed:

#!/bin/bash
# Check the implementation of the markdown filter
fd -e py markdown website/templatetags/

Length of output: 41


Review update: Verify Markdown Filter Sanitization

The investigation did not reveal any custom implementation of a Markdown filter in the repository (no files under website/templatetags/ defining a markdown filter were found). It appears the filter might be provided by a third-party library. Please manually confirm that the third-party implementation properly sanitizes rendered HTML to prevent XSS vulnerabilities. If sanitization is not handled by the library, consider applying an explicit sanitization step (for instance, using bleach).

  • Location: website/templates/newsletter/home.html (lines 65–67)
  • Action Items:
    • Verify the imported Markdown filter’s sanitization behavior.
    • If needed, implement additional sanitization (e.g., bleach) to secure user-submitted Markdown content.
website/templates/newsletter/preferences.html (6)

169-178: CSRF token in AJAX requests is properly implemented

Good implementation of secure AJAX request with proper CSRF token inclusion in the headers and appropriate Content-Type specification.


31-50: Well-implemented conditional warning for unconfirmed emails

The conditional display of a warning message for unconfirmed email addresses with a resend option is a good user experience touch. The SVG warning icon and clear instructions help guide users through the confirmation process.


51-119: Form structure and content preferences implementation is solid

The form implementation with content preferences is well-structured. The checkbox components are properly implemented with descriptive labels and explanations, making it clear to users what each preference entails.


113-117: Confirm unsubscribe action protects against accidental clicks

Good use of the JavaScript confirm() function to prevent accidental unsubscribes by requiring user confirmation.


123-148: Recent newsletters section provides good historical context

The recent newsletters section with dates and titles gives users a good overview of past content, enhancing the subscription management experience.


156-197: Well-implemented AJAX functionality for resending confirmation emails

The JavaScript code for handling the resend confirmation email functionality is well-implemented. It provides good user feedback by changing button state during the request and showing appropriate success/error messages.

website/admin.py (2)

709-739: Well-structured Newsletter admin configuration

The NewsletterAdmin class is well-organized with appropriate fields for display, filtering, and search. The send_newsletter action provides a convenient way for admins to send newsletters directly from the admin interface.

One suggestion to consider:

 def send_newsletter(self, request, queryset):
-    from django.core.management import call_command
+    # Import at the top of the file instead
     
     count = 0
     for newsletter in queryset:

Moving the import statement to the top of the file would follow Python best practices for imports.


741-777: Comprehensive NewsletterSubscriber admin implementation

The NewsletterSubscriberAdmin class provides all necessary functionality for managing subscribers, including useful admin actions for common tasks like sending confirmation emails or marking subscribers as confirmed/unsubscribed. The implementation is clean and follows Django admin best practices.

website/views/core.py (3)

61-62: Newsletter model imports added correctly

The imports for Newsletter and NewsletterSubscriber models have been properly added to the existing imports list.


1382-1384: Featured newsletter implementation in home view

The addition of the featured newsletter to the home view is straightforward and follows Django conventions by fetching the most recent published newsletter.


1405-1405: Featured newsletter added to template context

The featured newsletter is properly added to the template context for the home view.

website/management/commands/send_newsletter.py (2)

45-80: Well-structured newsletter sending implementation

The send_newsletter method is well-implemented with proper error handling and status updates. It correctly handles test mode, filters for active confirmed subscribers, and tracks successful sends.


81-114: Newsletter email construction is comprehensive

The send_to_subscriber method creates both HTML and plain text versions of the email, which is a good practice for email deliverability. The context includes all necessary information for personalizing the newsletter.

blt/urls.py (2)

1093-1100: Well-structured URL patterns for the newsletter feature.

The URL patterns follow RESTful conventions and include all necessary endpoints for a comprehensive newsletter system. The use of slugs and UUIDs for identification is appropriate.


309-315: Good organization of import statements.

The newsletter view imports are properly included at the end of the existing user views imports, maintaining the code organization pattern of the project.

website/views/user.py (6)

1158-1185: Well-structured newsletter detail view with proper context.

This view correctly retrieves the newsletter, increments view count, logs activity, and provides comprehensive context for the template.


1188-1225: Good handling of newsletter subscription logic.

The subscription flow handles various scenarios appropriately including new subscriptions, reactivations, and already subscribed users.


1228-1244: Effective confirmation email implementation.

The email sending logic is well-structured with appropriate context variables and both HTML and plain text versions.


1247-1258: Clear subscription confirmation flow.

This view correctly handles token validation and subscription confirmation status updates.


1261-1286: Comprehensive unsubscribe functionality with proper logging.

The unsubscribe flow properly validates tokens, updates subscription status, and logs the activity.


1289-1319: User-friendly preference management.

This view intelligently creates a subscription for logged-in users who don't have one yet and provides a clean interface for updating preferences.

website/models.py (4)

24-24: No immediate concerns on the import addition.
This line imports Count and Sum, which are used for data aggregation in Django queries. This looks correct and beneficial for handling leaderboard, bug counts, etc.


2298-2299: No functional code changes.
These lines appear to be blank/new lines, so there's nothing to address here.


2399-2399: No code changes.
This is a blank line, so no action needed.


2419-2465:

✅ Verification successful

Validate settings.DOMAIN_NAME usage for unsubscribe links.
The methods here, like get_unsubscribe_url, rely on settings.DOMAIN_NAME to build URLs. Ensure this setting is always configured in production environments to avoid malformed URLs or errors.

Additionally, the resubscribe flow sets a new confirmation_token; if you rely on that token for user verification, consider marking it unique. Otherwise, collisions are highly unlikely but theoretically possible.

Please confirm whether settings.DOMAIN_NAME is properly set in each deployed environment. You can run the following shell script to inspect references of DOMAIN_NAME usage across the codebase:


🏁 Script executed:

#!/bin/bash
rg -A 3 'DOMAIN_NAME'

Length of output: 2860


Action Required: Confirm Production Configuration and Consider Token Uniqueness

The codebase correctly uses settings.DOMAIN_NAME when constructing unsubscribe links; as shown, it’s defined in blt/settings.py (with a default value of "blt.owasp.org") and referenced consistently across views and models. Please ensure that in each deployed environment the DOMAIN_NAME setting is overridden with the proper value to avoid malformed URLs.

Additionally, the resubscribe method regenerates the confirmation_token using uuid.uuid4(). While UUID collisions are statistically negligible, if this token serves a critical role in user verification (beyond its use in constructing URLs), you might consider adding a unique constraint on the corresponding model field to enforce uniqueness at the database level.

  • Verify that production environment configurations correctly set DOMAIN_NAME.
  • Consider marking the confirmation_token field as unique if it’s relied upon for verification purposes.

@tsu-ki
Copy link
Contributor Author

tsu-ki commented Mar 15, 2025

@DonnieBLT sir, I've uploaded 2 videos, which show the user-side functionality and admin-side functionality. Please check them out and update me if there's anything else I should change. (another video is attached in the top comment)

Screen.Recording.2025-03-15.at.3.39.02.PM.mov

@DonnieBLT
Copy link
Collaborator

@tsu-ki can you please fix the conflicts and address the comments

@tsu-ki
Copy link
Contributor Author

tsu-ki commented Mar 15, 2025

@tsu-ki can you please fix the conflicts and address the comments

@DonnieBLT sir, I'll resolve the conflicts. I've also addressed most of the comments and will push the final code once it's ready, just need to fix linting and some dependency errors.
Another thing I wanted to address was using inline css in email template. As, external css is stripped away during email generation, I've resorted to using it inline. Is that okay?

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: 4

♻️ Duplicate comments (1)
website/management/commands/send_newsletter.py (1)

48-51: ⚠️ Potential issue

Potential error when accessing admin email

When sending a test email, the code directly accesses settings.ADMINS[0][1] without checking if the ADMINS setting is configured or non-empty, which could lead to an IndexError.

if test_mode:
    # Send only to admin email for testing
    self.stdout.write(f"Sending test email for '{newsletter.title}' to admin")
-   self.send_to_subscriber(settings.ADMINS[0][1], newsletter, is_test=True)
+   if settings.ADMINS and len(settings.ADMINS) > 0:
+       self.send_to_subscriber(settings.ADMINS[0][1], newsletter, is_test=True)
+   else:
+       self.stderr.write("No admin email configured. Cannot send test email.")
    return
🧹 Nitpick comments (18)
website/templatetags/custom_filters.py (1)

30-34: Consider security implications when using markdown with user-generated content.

The implementation of the markdown filter is well-structured and follows Django's conventions. However, using mark_safe with potentially user-generated content could lead to XSS vulnerabilities if not properly sanitized upstream.

While the markdown library does provide some sanitization by default, ensure that any user-submitted content being processed through this filter is properly validated or sanitized before reaching this point.

For high-traffic scenarios, consider caching the results of the markdown conversion to improve performance:

@register.filter
@stringfilter
def markdown_filter(value):
    """Converts markdown text to HTML."""
+   # Simple caching mechanism could be implemented here
+   # or use Django's caching framework for more complex scenarios
    return mark_safe(markdown.markdown(value, extensions=["extra", "nl2br", "sane_lists"]))
website/templates/newsletter/email/newsletter_email.html (1)

125-137: Consider specifying max-width for the featured image

While you've set the width to 100% with a max-width of 600px, consider also setting explicit dimensions in the HTML attributes to ensure proper rendering in email clients that might ignore some CSS properties.

<img src="{{ newsletter.featured_image.url }}"
     alt="{{ newsletter.title }}"
-    width="600"
-    height="300"
+    width="600"
+    height="auto"
     style="width: 100%;
            max-width: 600px;
            height: auto;
            display: block;
            border: 0">
blt/settings.py (1)

318-321: Commented-out email debugging configuration

These commented lines provide useful information for local email debugging, but commented code should generally be removed in production. Consider moving this to documentation.

- # use this to debug emails locally
- # python -m smtpd -n -c DebuggingServer localhost:1025
- # if DEBUG:
- # EMAIL_BACKEND = "django.core.mail.backends.console.EmailBackend"
website/templates/newsletter/email/confirmation_email.html (1)

183-198: Secure confirmation link implementation

The confirmation link is properly structured with clear styling and explicit target="_blank" attribute. Consider adding rel="noopener noreferrer" for target="_blank" links as you did in the sidenav.html file.

<a href="{{ confirm_url }}"
   target="_blank"
+  rel="noopener noreferrer"
   style="border: solid 1px #e74c3c;
          border-radius: 5px;
          box-sizing: border-box;
          cursor: pointer;
          display: inline-block;
          font-size: 14px;
          font-weight: bold;
          margin: 0;
          padding: 12px 25px;
          text-decoration: none;
          text-transform: capitalize;
          background-color: #e74c3c;
          border-color: #e74c3c;
          color: #ffffff">Confirm Subscription</a>
pyproject.toml (2)

16-16: Consider pinning dependency version reasoning

You've downgraded django-gravatar2 from a caret version to an exact version 1.4.4. While pinning dependencies can be good for stability, it would be helpful to include a comment explaining why this specific version is required.


69-74:

❓ Verification inconclusive

Review potential dependency conflicts and separation of concerns

The new dependencies appear to support the newsletter functionality, but I notice:

  1. environ (^1.0) may conflict with or duplicate django-environ (^0.12.0) already present at line 42
  2. google-auth dependency might be unnecessary if not directly used by the newsletter feature
  3. Consider organizing related dependencies together in the file for better readability

🏁 Script executed:

#!/bin/bash
# Check for potential environ package conflict
pip show environ django-environ 2>/dev/null | grep -E "^Name|^Version|^Summary"

# Check if google-auth is used in the newsletter code
grep -r "from google.auth" --include="*.py" website/

Length of output: 124


Please verify the dependency usage and organization.

  • The dependency declaration environ = "^1.0" may conflict with the already declared django-environ (line 42). It’s unclear if both are necessary or if one is a duplicate.
  • There is no confirmed usage of google-auth in the newsletter-related code. Please double-check if this dependency is directly required.
  • For better readability, consider grouping related dependencies together in the file.
website/templates/newsletter/preferences.html (3)

57-63: Use ARIA attributes for better accessibility

The disabled email input would benefit from ARIA attributes to improve accessibility. Since this field is disabled, adding aria-label would help screen readers provide better context.

<input type="email"
       name="email"
       id="email"
       value="{{ subscriber.email }}"
       placeholder="Your email address"
       disabled
-      class="block w-full rounded-md border-gray-300 bg-gray-100 shadow-sm text-gray-500 sm:text-sm">
+      class="block w-full rounded-md border-gray-300 bg-gray-100 shadow-sm text-gray-500 sm:text-sm"
+      aria-label="Your email address (not editable)">

114-118: Consider adding confirmation dialog for unsubscribe action

While you have an onclick confirmation for unsubscribing, it would be better to use a more accessible approach with a proper modal dialog that can be navigated with keyboard and is visible to screen readers.


183-194: Missing error details in notification

When an error occurs during confirmation email sending, the error message is generic and doesn't provide specific details from the server. Consider showing the actual error message from the response for better troubleshooting.

.then(data => {
    if (data.success) {
        window.createMessage('Confirmation email has been sent.', 'success');
    } else {
-       window.createMessage('Failed to send confirmation email. Please try again.', 'error');
+       window.createMessage('Failed to send confirmation email: ' + (data.error || 'Unknown error'), 'error');
    }
})
.catch(error => {
-   window.createMessage('An error occurred. Please try again.', 'error');
+   window.createMessage('Network error occurred: ' + error.message, 'error');
})
website/management/commands/send_newsletter.py (2)

53-79: Improve error handling and reporting

The current implementation logs errors but doesn't provide comprehensive reporting. Consider enhancing error handling to:

  1. Categorize errors (e.g., network issues vs. invalid email addresses)
  2. Provide a summary of errors after sending is complete
  3. Allow for retrying failed sends

105-115: Add email headers for better deliverability and tracking

Consider adding standard email headers to improve deliverability and tracking:

  1. List-Unsubscribe header
  2. Message-ID for tracking
  3. Precedence: bulk header to indicate this is a bulk email
  4. Reply-To header for user replies
# Create email message
email_message = EmailMultiAlternatives(
    subject=subject, body=text_content, from_email=settings.DEFAULT_FROM_EMAIL, to=[email]
)

+# Add headers for better deliverability
+if subscriber:
+    email_message.extra_headers = {
+        'List-Unsubscribe': f'<{context["unsubscribe_url"]}>',
+        'Precedence': 'bulk',
+        'Message-ID': f'<newsletter-{newsletter.id}-{subscriber.id}@{settings.DOMAIN_NAME}>'
+    }

email_message.attach_alternative(html_content, "text/html")
blt/urls.py (1)

308-314: Consider organizing related imports

The newsletter-related imports are mixed in with other user module imports. Consider grouping them together at the end of the user view imports or in a separate section for better organization and readability.

website/views/user.py (5)

1158-1186: Optional caching for frequent queries.
You are retrieving recent newsletters, bugs, and leaderboard updates on each request. Consider enabling caching or other optimizations if high traffic makes these queries expensive.


1188-1226: Leverage model methods to avoid duplicating re-subscription logic.
You already have a subscriber.resubscribe() method in the NewsletterSubscriber model. Instead of manually updating fields here, consider calling that method to keep all reactivation steps in one place.

-                subscriber.is_active = True
-                subscriber.confirmation_token = uuid.uuid4()
-                subscriber.save()
-                send_confirmation_email(subscriber)
-                messages.success(request, "Your subscription has been reactivated. Please confirm your email.")
+                subscriber.resubscribe()
+                send_confirmation_email(subscriber)
+                messages.success(
+                    request,
+                    "Your subscription has been reactivated. Please confirm your email."
+                )

1228-1247: Make protocol configurable.
Using a hard-coded "https://" in the confirmation URL may break local or non-HTTPS environments. You could dynamically detect or configure the protocol.


1263-1289: Consider storing an 'unsubscribed_at' timestamp.
Recording when the user unsubscribes can assist in analytics or compliance (e.g., GDPR audits).


1324-1339: Avoid catching all exceptions in a single block.
While handling everything in one except block is convenient, it might mask Http404 separately from other errors. Consider distinguishing user-not-found from system errors.

website/models.py (1)

2301-2399: Use atomic increments to avoid lost updates.
increment_view_count() uses a simple retrieval-then-save approach, which can lose increments if called concurrently. Consider using an F-expression for consistency.

def increment_view_count(self):
-    self.view_count += 1
-    self.save(update_fields=["view_count"])
+    from django.db.models import F
+    Newsletter.objects.filter(pk=self.pk).update(view_count=F('view_count') + 1)
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between b8976d1 and 30d2a16.

📒 Files selected for processing (14)
  • blt/settings.py (4 hunks)
  • blt/urls.py (3 hunks)
  • pyproject.toml (2 hunks)
  • website/management/commands/send_newsletter.py (1 hunks)
  • website/models.py (3 hunks)
  • website/templates/base.html (1 hunks)
  • website/templates/includes/sidenav.html (10 hunks)
  • website/templates/newsletter/detail.html (1 hunks)
  • website/templates/newsletter/email/confirmation_email.html (1 hunks)
  • website/templates/newsletter/email/newsletter_email.html (1 hunks)
  • website/templates/newsletter/home.html (1 hunks)
  • website/templates/newsletter/preferences.html (1 hunks)
  • website/templatetags/custom_filters.py (2 hunks)
  • website/views/user.py (6 hunks)
🚧 Files skipped from review as they are similar to previous changes (3)
  • website/templates/newsletter/detail.html
  • website/templates/newsletter/home.html
  • website/templates/base.html
🔇 Additional comments (18)
website/templatetags/custom_filters.py (1)

4-4: Appropriate imports for markdown functionality.

The added imports for markdown library and stringfilter decorator are correctly implemented to support the new markdown filter functionality.

Also applies to: 7-7

website/templates/includes/sidenav.html (3)

417-429: Good addition of Newsletter section to navigation!

The Newsletter navigation item is well-structured and consistent with the existing sidebar design. The SVG icon provides a nice visual cue for users.


555-555: Security improvement with rel attributes

Adding rel="noopener noreferrer" to external links that use target="_blank" is an excellent security practice that prevents potential exploitation through the window.opener property.

Also applies to: 647-647, 658-658, 669-669, 680-680, 691-691, 702-702


728-728: Accessibility improvement with aria-label

Good addition of the aria-label attribute to the language selector, which improves accessibility for screen readers.

website/templates/newsletter/email/newsletter_email.html (2)

1-261: Well-structured email template with responsive design

This newsletter email template follows email best practices:

  • Proper HTML5 doctype and meta tags
  • Responsive design with media queries
  • Inline CSS (necessary for email clients)
  • Table-based layout for compatibility
  • Conditional sections for dynamic content
  • Footer with unsubscribe option

The structure and styling are appropriate for cross-client email compatibility.


146-146: Potential security consideration with safe filter

The template uses {{ newsletter.content|safe }} which will render HTML content without escaping. This is acceptable since it's an email template and the content comes from administrators, but ensure that only trusted users can create newsletters.

blt/settings.py (3)

99-99: Addition of markdown_deux package

The markdown_deux package has been added to INSTALLED_APPS, which will enable Markdown rendering functionality for newsletters.


173-173: Good implementation of a newsletter context processor

Adding the newsletter context processor to the TEMPLATES configuration will make newsletter subscription data available to all templates, enhancing the user experience by showing subscription status throughout the site.


613-621: Markdown configuration for newsletters

The MARKDOWN_DEUX_STYLES configuration is properly set up with code-friendly and tables support, with safe_mode disabled to allow HTML in the markdown. This is appropriate for trusted content authored by administrators.

website/templates/newsletter/email/confirmation_email.html (1)

1-273: Well-structured subscription confirmation email template

This confirmation email template follows email development best practices with:

  • Proper doctype and meta tags
  • Responsive design with media queries
  • Inline CSS for email client compatibility
  • Clear call-to-action button
  • Informative content and security considerations

The template provides a good user experience for confirming newsletter subscriptions.

blt/urls.py (1)

1091-1098: Order URL patterns from most specific to least specific

The URL patterns for the newsletter functionality should be ordered from most specific to least specific to prevent routing conflicts. The pattern with slug parameter should come last (as you've done) to prevent it from capturing requests meant for other endpoints.

website/views/user.py (4)

4-50: Imports look good.
No concerns with the addition of uuid, ContentType, Paginator, and other model imports for newsletter management.


1127-1128: No impact on functionality.
These are just blank or spacing lines, so no changes are needed.


1249-1261: Newsletter confirmation logic looks correct.
No issues spotted; the straightforward confirmation flow aligns with typical email confirmation patterns.


1291-1322: Auto-confirm preference for logged-in subscribers.
This approach is valid. Just keep in mind that if a user changes emails later, some confirmation flows may be bypassed.

website/models.py (3)

24-24: Newly imported aggregate functions are fine.
The use of Count and Sum aligns with your leaderboard and query logic.


2298-2300: No functional changes.
Blank or spacing lines, no action required.


2400-2465: Newsletter subscriber model appears solid.
The design for subscriptions, preferences, and confirmation tokens is clear and well-structured.

@DonnieBLT
Copy link
Collaborator

@tsu-ki can you please fix the conflicts and address the comments

@DonnieBLT sir, I'll resolve the conflicts. I've also addressed most of the comments and will push the final code once it's ready, just need to fix linting and some dependency errors. Another thing I wanted to address was using inline css in email template. As, external css is stripped away during email generation, I've resorted to using it inline. Is that okay?

It’s ok if it’s inline tailwind

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 (10)
blt/settings.py (1)

318-321: Consider removing or utilizing commented code.

These commented lines about email debugging configuration are informative but may confuse future developers about whether they should be used.

Either remove these comments or convert them to proper documentation (e.g., in a docstring or separate documentation file) explaining how to debug emails locally.

website/management/commands/send_newsletter.py (2)

15-17: Add a descriptive docstring for the Command class.

While the help attribute is helpful, a short docstring explaining usage and behavior would improve clarity for maintainers.

 class Command(BaseCommand):
+    """
+    Management command to send published newsletters to subscribers or to admins for testing.
+    """
     help = "Send published newsletter to subscribers"

45-84: Consider separating subscriber retrieval from sending logic.

Having send_newsletter both retrieve subscribers and handle sending could make testing and code maintenance less straightforward. Extracting subscriber retrieval into a helper method (e.g., get_subscribers()) and passing them to a dedicated sending method may improve clarity.

website/views/user.py (2)

1188-1235: Catch specific exceptions instead of a broad Exception.

The broad except Exception as e: block could obscure underlying errors, making debugging harder. Consider catching DatabaseError, IntegrityError, or ValueError to handle known issues more precisely.

     try:
         ...
-    except Exception as e:
+    except (IntegrityError, ValueError, DatabaseError) as e:
         ...

1368-1384: Refine exception handling.

Similar to the subscription logic, catch narrower exceptions for better clarity.

try:
    data = json.loads(request.body)
    ...
-except Exception as e:
+except (json.JSONDecodeError, NewsletterSubscriber.DoesNotExist) as e:
    ...
website/models.py (5)

2300-2330: Newsletter model structure looks good but could benefit from additional indexes.

The Newsletter model is well designed with appropriate fields for content management, publication status, and tracking. The inclusion of boolean flags for different content sections provides flexibility.

Consider adding an index for the published_at field since it's used for ordering in the Meta class. This would improve performance when retrieving published newsletters, especially as the number of newsletters grows:

class Meta:
    ordering = ["-published_at"]
    verbose_name = "Newsletter"
    verbose_name_plural = "Newsletters"
+   indexes = [
+       models.Index(fields=["-published_at"], name="newsletter_pub_date_idx"),
+   ]

2349-2369: Content retrieval methods are well-organized but could be more robust.

These methods for fetching related content are well-structured, but they don't handle potential database errors or empty result sets explicitly.

Consider adding try/except blocks to handle potential database errors:

def get_recent_bugs(self, limit=5):
    """Get recent bugs for newsletter content"""
    if not self.recent_bugs_section:
        return []
+   try:
        return Issue.objects.filter(status="open").order_by("-created")[:limit]
+   except Exception as e:
+       logger.error(f"Error fetching recent bugs for newsletter {self.id}: {e}")
+       return []

2399-2421: NewsletterSubscriber model handles both user and email-based subscriptions effectively.

The model is well-designed to handle both authenticated users and email-only subscribers. The preference fields provide good granularity for personalization.

Consider adding an index for the email field to improve performance for email-based lookups:

class Meta:
    verbose_name = "Newsletter Subscriber"
    verbose_name_plural = "Newsletter Subscribers"
+   indexes = [
+       models.Index(fields=["email"], name="newsletter_sub_email_idx"),
+       models.Index(fields=["user"], name="newsletter_sub_user_idx"),
+   ]

2448-2478: User and email subscription checking methods need transaction handling.

The static methods for checking subscriptions are useful but should handle database operation atomically.

Consider adding transaction handling to prevent race conditions:

@staticmethod
def user_is_subscribed(user):
    """Check if a user is subscribed to the newsletter"""
    if not user or not user.is_authenticated:
        return False
+   from django.db import transaction
+   with transaction.atomic():
        return NewsletterSubscriber.objects.filter(user=user, is_active=True, confirmed=True).exists()

2479-2487: Unsubscribe URL construction could be improved.

The get_unsubscribe_url method manually constructs URLs, which might break if the URL structure changes.

Use Django's build_absolute_uri method in views instead of manually constructing the URL:

def get_unsubscribe_url(self):
    """Get the unsubscribe URL for this subscriber"""
    from django.conf import settings
    from django.urls import reverse

-   scheme = "https" if not settings.DEBUG else "http"
-   return f"{scheme}://{settings.DOMAIN_NAME}{reverse('newsletter_unsubscribe', args=[self.confirmation_token])}"
+   # Return just the relative URL - let the view handle absolute URL construction
+   return reverse('newsletter_unsubscribe', args=[self.confirmation_token])

Then in your view function or email template:

# In a view or email context:
absolute_unsubscribe_url = request.build_absolute_uri(subscriber.get_unsubscribe_url())
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 30d2a16 and 35ebc9f.

📒 Files selected for processing (6)
  • blt/settings.py (4 hunks)
  • website/management/commands/send_newsletter.py (1 hunks)
  • website/migrations/0232_newslettersubscriber_token_created_at.py (1 hunks)
  • website/models.py (3 hunks)
  • website/views/core.py (4 hunks)
  • website/views/user.py (6 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • website/views/core.py
🔇 Additional comments (21)
blt/settings.py (3)

99-99: Good addition for newsletter formatting capabilities.

The inclusion of the "markdown_deux" package in INSTALLED_APPS provides necessary support for Markdown formatting in newsletters, allowing for rich content creation.


173-173: LGTM - Appropriate context processor for newsletter functionality.

Adding the newsletter context processor ensures that newsletter-related data is available to all templates, which is essential for displaying subscription status and other newsletter information to users.


613-621: Verify security implications of safe_mode: False.

The Markdown configuration with safe_mode: False allows raw HTML in markdown content. While this provides flexibility for newsletter formatting, it could potentially introduce security risks if content from untrusted sources is processed.

If newsletters are only created by administrators, this may be acceptable. However, if there's any user-generated content that gets processed through this markdown renderer, consider using safe_mode: True or implement additional sanitization.

website/migrations/0232_newslettersubscriber_token_created_at.py (1)

1-21: Looks good.

The new migration cleanly adds the token_created_at field, and the approach is standard for tracking token creation timestamps.

website/management/commands/send_newsletter.py (2)

22-45: Implementation looks correct.

The handle method accurately filters newsletters based on newsletter_id or published status, gracefully handles exceptions, and logs helpful messages via stdout/stderr.


84-120: No major issues found.

The send_to_subscriber method is well-structured, handles the test mode subject prefix, and uses a text + HTML email approach. The fallback scheme for unsubscribe URLs is also a good practice.

website/views/user.py (14)

4-4: No issues with adding import uuid.


14-14: No issues with adding content type import.


17-17: No issues with adding paginator import.


28-28: No issues with adding new HTTP decorators.


38-38: No issues with importing Activity.


49-49: No issues with importing Newsletter.


50-50: No issues with importing NewsletterSubscriber.


1127-1128: No new logic here.


1129-1156: Hard-coded object_id=1 is confusing.

A past review noted that referencing a hard-coded object ID can cause confusion and mismatched references. Consider dynamically referencing the relevant newsletter or removing the object_id if it does not represent a real object.

 Activity.objects.create(
     user=request.user,
     action_type="view",
     title="Viewed newsletters",
     content_type=ContentType.objects.get_for_model(Newsletter),
-    object_id=1,
+    object_id=newsletters.first().id if newsletters.exists() else None,
 )

1158-1186: No issues with newsletter_detail.

The function logically increments view count, fetches recent newsletters, and logs user activity with the correct newsletter ID.


1237-1271: Looks good.

send_confirmation_email is well-structured, uses a fallback scheme in DEBUG mode, and logs helpful information.


1273-1305: No issues with newsletter_confirm.

Resending on token expiration is a good user experience. The logic is clear and well-documented in logs.


1307-1333: No issues with newsletter_unsubscribe.

Properly deactivates and logs unsubscribes, providing user feedback and tracking with Activity.


1335-1366: No issues with newsletter_preferences.

Auto-creation and preference toggles are clearly handled.

website/models.py (1)

2335-2347: Slug generation and URL handling follow Django conventions.

The save method correctly handles slug generation and publication timestamp, while the get_absolute_url method follows Django's best practices for URL reversing.

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

♻️ Duplicate comments (3)
website/views/user.py (1)

1428-1443: ⚠️ Potential issue

Prevent exposure of the raw exception message.
Returning str(e) in the JSON response at line 1443 can expose sensitive information. Return a generic error message instead.

 except Exception as e:
-    return JsonResponse({"success": False, "error": str(e)})
+    return JsonResponse({"success": False, "error": "An unexpected error occurred. Please contact support."})
🧰 Tools
🪛 GitHub Check: CodeQL

[warning] 1443-1443: Information exposure through an exception
Stack trace information flows to this location and may be exposed to an external user.

website/models.py (2)

2450-2462: ⚠️ Potential issue

Enhance the format_for_email method to properly handle Markdown conversion

The current implementation doesn't convert Markdown content to HTML before sending emails, which could result in raw Markdown being displayed to users in their email clients.

Apply this improvement to properly convert Markdown to HTML:

def format_for_email(self):
    """Format the newsletter content for email sending"""
    from django.utils.html import strip_tags
+   import markdown

+   # Convert Markdown to HTML
+   html_content = markdown.markdown(self.content)
-   email_body = self.content
+   email_body = html_content
    plain_text = strip_tags(email_body).replace("&nbsp;", " ")

    return {
        "subject": self.email_subject or f"Newsletter: {self.title}",
        "html_content": email_body,
        "plain_text": plain_text,
    }

1083-1083: ⚠️ Potential issue

Ensure the migration for contributor_type includes a default value

Adding the contributor_type field without a default value could cause migration issues for existing records.

Create a migration that safely handles existing records by either:

  1. Making the field nullable (null=True), or
  2. Providing a default value, or
  3. Creating a data migration to populate existing records
🧹 Nitpick comments (5)
website/admin.py (1)

727-742: Consider concurrency for sending newsletters.
Currently, send_newsletter relies on a management command call, which presumably sets email_sent to True within that command. However, if multiple admins trigger the same action concurrently, it could send duplicates. Consider synchronizing or marking email_sent early to avoid duplicates.

website/views/user.py (1)

1248-1295: Reactivating subscriptions logic looks fine; be mindful of potential race conditions.
Each request sets is_active=False on any existing active subscription for the same email, then (re)creates or updates the subscriber. If multiple requests come in at once for the same email, there could be a timing conflict. Consider adding some atomic logic or transaction handling if that becomes a concern.

website/models.py (3)

2414-2439: Consider optimizing database queries in content retrieval methods

The methods for retrieving content (bugs, leaderboard, IPs) perform database queries without limiting their execution context, which could impact performance as the database grows.

Consider adding caching or optimizing these queries:

def get_recent_bugs(self, limit=5):
    """Get recent bugs for newsletter content"""
    if not self.recent_bugs_section:
        return []

+   # Add caching to improve performance
+   cache_key = f"newsletter_{self.id}_recent_bugs"
+   cached_bugs = cache.get(cache_key)
+   if cached_bugs is not None:
+       return cached_bugs
+   
    bugs = Issue.objects.filter(status="open").order_by("-created")[:limit]
+   cache.set(cache_key, bugs, 3600)  # Cache for 1 hour
+   return bugs
-   return Issue.objects.filter(status="open").order_by("-created")[:limit]

Apply similar optimizations to the other content retrieval methods.


2544-2551: Use Django's built-in URL handling for more robust unsubscribe URLs

The current implementation manually constructs URLs, which could be problematic if the site configuration changes.

Consider using Django's build_absolute_uri:

def get_unsubscribe_url(self):
    """Get the unsubscribe URL for this subscriber"""
    from django.conf import settings
    from django.urls import reverse
-
-   scheme = "https" if not settings.DEBUG else "http"
-
-   return f"{scheme}://{settings.DOMAIN_NAME}{reverse('newsletter_unsubscribe', args=[self.confirmation_token])}"
+   
+   from django.contrib.sites.shortcuts import get_current_site
+   from django.http import HttpRequest
+   
+   # Create a dummy request object
+   request = HttpRequest()
+   request.META['SERVER_NAME'] = settings.DOMAIN_NAME
+   request.META['SERVER_PORT'] = '443' if not settings.DEBUG else '80'
+   request.scheme = 'https' if not settings.DEBUG else 'http'
+   
+   url = reverse('newsletter_unsubscribe', args=[self.confirmation_token])
+   return request.build_absolute_uri(url)

2527-2543: Implement a more secure token generation approach

While UUIDs are convenient, for security-sensitive operations like subscription management, consider a more robust approach that includes a signature component.

def refresh_token(self):
    """Generate a new confirmation token and reset the creation timestamp"""
-   self.confirmation_token = uuid.uuid4()
+   # Create a more secure token with a signature component
+   from django.core.signing import TimestampSigner
+   signer = TimestampSigner()
+   token_base = str(uuid.uuid4())
+   signed_token = signer.sign(token_base)
+   # Store just the UUID part but use the signed version in URLs
+   self.confirmation_token = uuid.UUID(token_base)
    self.token_created_at = timezone.now()
    self.save(update_fields=["confirmation_token", "token_created_at"])
    return self

This would require updating the token verification process as well.

📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 35ebc9f and 33d54ee.

📒 Files selected for processing (6)
  • blt/urls.py (3 hunks)
  • pyproject.toml (2 hunks)
  • website/admin.py (2 hunks)
  • website/models.py (3 hunks)
  • website/templates/includes/sidenav.html (10 hunks)
  • website/views/user.py (6 hunks)
🚧 Files skipped from review as they are similar to previous changes (3)
  • pyproject.toml
  • website/templates/includes/sidenav.html
  • blt/urls.py
🧰 Additional context used
🪛 GitHub Check: CodeQL
website/views/user.py

[warning] 1443-1443: Information exposure through an exception
Stack trace information flows to this location and may be exposed to an external user.

⏰ Context from checks skipped due to timeout of 90000ms (3)
  • GitHub Check: Run Tests
  • GitHub Check: Analyze (python)
  • GitHub Check: Analyze (javascript-typescript)
🔇 Additional comments (13)
website/admin.py (3)

46-47: New model imports look good.
No issues observed with importing Newsletter and NewsletterSubscriber.


743-780: NewsletterSubscriberAdmin setup looks solid.
The admin actions (send confirmation, confirm, unsubscribe) are well-structured and straightforward.


711-726:

✅ Verification successful

Ensure created_at field exists in the Newsletter model.
You're using date_hierarchy = "created_at" on the NewsletterAdmin, so make sure the Newsletter model actually defines a created_at field. Otherwise, Django admin will raise an error.


🏁 Script executed:

#!/bin/bash
# Verify the presence of `created_at` field in the Newsletter model
rg "class Newsletter" -A 25

Length of output: 6432


Verification Successful: created_at Field Found
The Newsletter model (in website/models.py) clearly defines the created_at field, so the use of date_hierarchy = "created_at" in NewsletterAdmin is valid. No further changes are required.

website/views/user.py (7)

4-4: New imports and references appear correct.
These additions for UUID, ContentType, Paginator, HTTP decorators, and new models are consistent with the new newsletter functionality.

Also applies to: 14-14, 17-17, 28-28, 38-38, 50-50, 51-51


1189-1216: Use the actual newsletter ID instead of a hard-coded object_id=1.
You’re logging an Activity with object_id=1 at lines 1200-1206, which could cause confusion when viewing usage analytics or linking to a specific newsletter.

-            object_id=1,
+            object_id=9999,  # or dynamically reference a real ID if needed

1218-1246: newsletter_detail logic is well-structured.
Incrementing the view count and logging user activity with the actual newsletter ID is correct. The references to get_recent_bugs(), get_leaderboard_updates(), and get_reported_ips() are presumably implemented in the model.


1297-1331: Confirmation email procedure is robust.
Handles domain name, logging, and fails gracefully if sending fails. Nicely done.


1333-1365: Confirmation token flow is straightforward.
Properly checks for expiration, refreshes token if needed, and sets confirmed. Logic is clear and user-friendly.


1367-1393: newsletter_unsubscribe approach is sound.
Deactivating the subscriber, logging the action, and showing messages for unsubscribed and already-unsubscribed users matches typical patterns.


1395-1426: newsletter_preferences gracefully handles new or existing subscribers.
Auto-creating and confirming for logged-in users is convenient. Logic looks fine.

website/models.py (3)

2364-2396: Best practice implementation for Newsletter model

The Newsletter model has a well-structured design with appropriate fields for content management, tracking, and publication status. The Meta class with proper ordering by published date and clear verbose names enhances usability in the admin interface.


2464-2489: Robust subscriber model with comprehensive status management

The NewsletterSubscriber model provides a complete solution for managing subscribers, including support for both authenticated users and email-only subscribers, confirmation workflow, and preference management.


2513-2526: Well-implemented helper methods for subscription status checking

The static methods for verifying subscription status are well-designed, providing convenient ways to check if a user or email is subscribed.

@tsu-ki tsu-ki requested a review from DonnieBLT March 15, 2025 15:23
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: 7

🧹 Nitpick comments (3)
website/admin.py (1)

743-779: Comprehensive subscriber management implementation

The NewsletterSubscriberAdmin class provides a complete set of tools for managing subscribers including status tracking, bulk actions, and search capabilities. The implementation follows Django admin best practices.

The mark_as_confirmed action doesn't include a verification step, which could allow admins to confirm subscriptions without user consent. Consider adding a warning or confirmation step before bulk confirming subscribers.

def mark_as_confirmed(self, request, queryset):
+    if not queryset.filter(confirmed=False).exists():
+        self.message_user(request, "All selected subscribers are already confirmed.", level='WARNING')
+        return
+    
+    unconfirmed_count = queryset.filter(confirmed=False).count()
+    if request.POST.get('confirmation') != 'yes':
+        self.message_user(
+            request, 
+            f"You are about to confirm {unconfirmed_count} subscriber(s). This should only be done in exceptional cases as it bypasses email confirmation.",
+            level='WARNING'
+        )
+        return

    queryset.update(confirmed=True)
    self.message_user(request, f"{queryset.count()} subscribers marked as confirmed.")

mark_as_confirmed.short_description = "Mark selected subscribers as confirmed"
website/views/user.py (1)

1395-1426: Sound newsletter preferences implementation

The newsletter_preferences view is well-implemented with proper user authentication, creates a subscription if one doesn't exist, and handles form submission correctly. The error handling is also appropriate.

One minor improvement could be to validate if the logged-in user's email matches the subscriber email in case a subscription already exists with a different email.

@login_required
def newsletter_preferences(request):
    """View to update newsletter preferences"""
    try:
        subscriber = NewsletterSubscriber.objects.get(user=request.user)
+        
+        # Check if user's email has changed and update the subscription email
+        if subscriber.email != request.user.email:
+            subscriber.email = request.user.email
+            subscriber.save()
+            messages.info(request, "We've updated your subscription email to match your account email.")
+            
    except NewsletterSubscriber.DoesNotExist:
        # If no subscription exists but user wants to manage preferences,
        # create one with their email from their user account
        subscriber = NewsletterSubscriber(
            user=request.user,
            email=request.user.email,
            name=request.user.get_full_name(),
            confirmed=True,  # Auto-confirm for logged-in users
        )
        subscriber.save()
website/models.py (1)

2452-2463: Improved email formatting with markdown conversion.

The format_for_email method now properly converts Markdown content to HTML using appropriate extensions. This addresses the previous review comment about ensuring Markdown conversion.

One recommendation to consider:

def format_for_email(self):
    """Format newsletter content for sending via email, converting Markdown to HTML."""
    import markdown
    
    # Convert Markdown content to HTML
    html_content = markdown.markdown(self.content, extensions=['extra', 'codehilite'])
    
    # Return formatted content for email
    return {
        'title': self.title,
        'content': html_content,
+       'subject': self.email_subject or f"Newsletter: {self.title}",
        'published_at': self.published_at,
+       'plain_text': html_content.replace("&nbsp;", " ") if html_content else ""
    }

This would add the subject line derivation and plain text conversion for email clients that don't support HTML.

📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 33d54ee and 2d0a93a.

📒 Files selected for processing (4)
  • website/admin.py (2 hunks)
  • website/models.py (5 hunks)
  • website/templates/newsletter/preferences.html (1 hunks)
  • website/views/user.py (6 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • website/templates/newsletter/preferences.html
🧰 Additional context used
🪛 Ruff (0.8.2)
website/models.py

34-34: markdown imported but unused

Remove unused import: markdown

(F401)

⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: Run Tests
🔇 Additional comments (13)
website/admin.py (2)

45-46: Clean import addition

The Newsletter and NewsletterSubscriber models are correctly imported in the models list, maintaining alphabetical order.


711-741:

❓ Verification inconclusive

Well-structured NewsletterAdmin implementation

The NewsletterAdmin class is well-organized with appropriate list_display fields, filters, search options, and field groupings. The send_newsletter action properly uses Django's call_command for executing the newsletter sending functionality.

One security consideration: The code assumes that published newsletters are ready to be sent. Consider adding an additional confirmation step or validation before sending newsletters to subscribers.


🏁 Script executed:

#!/bin/bash
# Check if there's proper validation in the send_newsletter management command
rg -A 10 "send_newsletter" | grep -E "status|check|valid"

Length of output: 389


NewsletterAdmin is well-organized but consider an extra confirmation before dispatching newsletters

The implementation is solid—with clear segregation of display fields, filters, and the action logic. The admin action correctly limits newsletter dispatches by checking that each newsletter’s status is "published" and that it hasn’t been sent yet. Moreover, the send_newsletter management command applies additional filtering (ensuring, for example, that published_at is not in the future).

That said, since the code assumes that a published newsletter is ready to be sent, it’s worth considering an extra confirmation step or a more explicit validation within the admin action to help prevent accidental newsletter dispatches.

  • Verify if adding an interactive confirmation (or further validations) in the admin interface meets your security and operational requirements.
website/views/user.py (2)

4-4: Appropriate import addition

The uuid module is correctly imported and used for generating unique confirmation tokens for newsletter subscriptions.


14-15: Model imports correctly organized

The ContentType model import and Activity/Newsletter/NewsletterSubscriber model imports are properly organized alphabetically in their respective import blocks.

Also applies to: 37-51

website/models.py (9)

24-24: Import of Sum is well-integrated into the model.

The addition of Sum to the imports supports the aggregation functionality in the new get_leaderboard_updates method of the Newsletter model.


34-34: Markdown import is used correctly despite static analysis warning.

The static analysis indicates this import might be unused, but it's actually utilized in the format_for_email method of the Newsletter model to convert markdown content to HTML.

🧰 Tools
🪛 Ruff (0.8.2)

34-34: markdown imported but unused

Remove unused import: markdown

(F401)


657-658: Good addition of related_name to Points model.

Adding related_name="points" to the User ForeignKey allows for cleaner reverse lookups (user.points instead of the default user.points_set) and is used correctly in the Newsletter model's get_leaderboard_updates method.


1084-1084: Migration concern for non-nullable contributor_type field addressed.

I see the previous review already identified that adding the non-nullable contributor_type field without a default value could cause migration issues for existing records. Based on the context, it appears this has been addressed in migration files.


2366-2464: Well-structured Newsletter model with comprehensive functionality.

The Newsletter model is well-designed with appropriate fields for content management, publication status, and email tracking. The methods for content retrieval, view counting, and email formatting are comprehensive.

A couple of suggestions for potential improvements:

  1. Consider caching the results of get_recent_bugs, get_leaderboard_updates, and get_reported_ips if they're called repeatedly in templates or views.
  2. The format_for_email method has been improved with markdown conversion as previously suggested.

2423-2428: Leaderboard aggregation uses points relation correctly.

The get_leaderboard_updates method now correctly uses the related_name "points" that was added to the Points model. This ensures proper aggregation of user scores for the leaderboard section of newsletters.

The previous review comment about this issue has been addressed.


2466-2554: Well-implemented NewsletterSubscriber model with robust subscription management.

The NewsletterSubscriber model provides comprehensive subscription management with:

  • Clear subscription status tracking
  • Token-based confirmation and unsubscription
  • User preference controls
  • Helper methods for subscription status checks

The token expiration check and refresh methods are well-designed security features.

The get_unsubscribe_url method properly constructs URLs with appropriate scheme based on the environment.


2529-2538: Good implementation of token expiration check.

The is_token_expired method appropriately checks if the confirmation token has expired based on a configurable time window. This is a good security practice to limit the validity of tokens.


2546-2553: Secure unsubscribe URL generation.

The get_unsubscribe_url method generates unsubscribe links using tokens rather than user IDs or emails, which is a good security practice. It also correctly handles the scheme (http/https) based on the 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 (1)
website/models.py (1)

2546-2552: Unsubscribe URL generation should handle environments without DOMAIN_NAME setting.

While the implementation works in most cases, consider adding a fallback for environments where DOMAIN_NAME might not be configured.

def get_unsubscribe_url(self):
    """Get the unsubscribe URL for this subscriber"""
    from django.conf import settings
    from django.urls import reverse

    scheme = "https" if not settings.DEBUG else "http"
+   domain = getattr(settings, 'DOMAIN_NAME', 'localhost:8000')

-   return f"{scheme}://{settings.DOMAIN_NAME}{reverse('newsletter_unsubscribe', args=[self.confirmation_token])}"
+   return f"{scheme}://{domain}{reverse('newsletter_unsubscribe', args=[self.confirmation_token])}"
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 2d0a93a and 99b0dd4.

📒 Files selected for processing (1)
  • website/models.py (4 hunks)
🔇 Additional comments (5)
website/models.py (5)

24-24: Appropriate import added for newsletter functionality.

The addition of the Sum aggregation import is correctly implemented and used in the Newsletter model's get_leaderboard_updates method.


2365-2463: Well-structured Newsletter model with comprehensive functionality.

The Newsletter model is well-designed with appropriate fields for content management, publication status tracking, and email distribution. The implementation includes several useful methods for content aggregation and email formatting.

Key strengths:

  • Proper slug generation and publication date handling
  • Content section toggles for customization
  • View count tracking
  • Markdown to HTML conversion for emails

2422-2427: Leaderboard aggregation uses correct related fields.

The get_leaderboard_updates method properly leverages the points related_name defined on line 657 to aggregate user scores. This implementation correctly addresses a previous issue raised in code reviews.


2450-2463: Email formatting method now correctly converts Markdown to HTML.

The format_for_email method properly implements Markdown conversion with appropriate extensions, addressing a previous issue raised in code reviews.


2465-2553: Well-implemented NewsletterSubscriber model with security considerations.

The NewsletterSubscriber model provides a comprehensive solution for managing subscriptions with:

  • Secure UUID-based confirmation tokens
  • Token expiration handling
  • Support for both registered and anonymous users
  • Customizable content preferences
  • Clear status tracking

The implementation follows best practices for subscription management including proper confirmation flow and unsubscribe functionality.

coderabbitai[bot]
coderabbitai bot previously approved these changes Mar 15, 2025
messages.error(request, "Email address is required.")
return redirect("newsletter_subscribe")

if not re.match(r"[^@]+@[^@]+\.[^@]+", email):

Check failure

Code scanning / CodeQL

Polynomial regular expression used on uncontrolled data High

This
regular expression
that depends on a
user-provided value
may run slow on strings with many repetitions of '?'.
data = json.loads(request.body)
email = data.get("email")

if not email or not re.match(r"[^@]+@[^@]+\.[^@]+", email):

Check failure

Code scanning / CodeQL

Polynomial regular expression used on uncontrolled data High

This
regular expression
that depends on a
user-provided value
may run slow on strings with many repetitions of '?'.
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)
website/views/user.py (2)

1275-1371: Review of newsletter_subscribe function (potential regex performance issue).

  1. Rate limiting and XSS safeguards are well handled.
  2. Lines [1301-1302]: The static analysis warns of possible polynomial-time complexity in the email validation regex. Consider replacing re.match(r"[^@]+@[^@]+\.[^@]+", email) with Django’s robust email validator (django.core.validators.validate_email) to avoid edge-case performance problems.

Otherwise, the subscription workflow is solid, with clear logging and user feedback.

🧰 Tools
🪛 GitHub Check: CodeQL

[failure] 1301-1301: Polynomial regular expression used on uncontrolled data
This regular expression that depends on a user-provided value may run slow on strings with many repetitions of '?'.


1571-1627: newsletter_resend_confirmation function (potential regex performance issue).

Similar to newsletter_subscribe, the regex check on line 1587 can be replaced by validate_email to avoid possible slowdowns for maliciously crafted input. The time-sleep approach for consistent timing is a nice measure against enumeration attempts.

🧰 Tools
🪛 GitHub Check: CodeQL

[failure] 1587-1587: Polynomial regular expression used on uncontrolled data
This regular expression that depends on a user-provided value may run slow on strings with many repetitions of '?'.

📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 99b0dd4 and 8a6fbba.

📒 Files selected for processing (3)
  • website/migrations/0233_newsletter_newslettersubscriber.py (1 hunks)
  • website/migrations/0234_newslettersubscriber_token_created_at.py (1 hunks)
  • website/views/user.py (7 hunks)
🧰 Additional context used
🪛 GitHub Check: CodeQL
website/views/user.py

[failure] 1301-1301: Polynomial regular expression used on uncontrolled data
This regular expression that depends on a user-provided value may run slow on strings with many repetitions of '?'.


[failure] 1587-1587: Polynomial regular expression used on uncontrolled data
This regular expression that depends on a user-provided value may run slow on strings with many repetitions of '?'.

⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: Run Tests
🔇 Additional comments (9)
website/migrations/0234_newslettersubscriber_token_created_at.py (1)

14-21: Add field token_created_at to track token creation time.

This addition effectively captures when the confirmation token was created. As an alternative, consider using a dedicated method for resetting the token timestamp—this ensures that you do not inadvertently overwrite the field.

website/migrations/0233_newsletter_newslettersubscriber.py (2)

18-48: Newsletter model design looks solid.

The grouping of fields and the inclusion of metadata (e.g. ordering by published_at, verbose names) provide clarity. This structure should be straightforward to maintain and extend.


49-77: Efficient structure for NewsletterSubscriber model.

The combination of fields (e.g. uuid token, user linkage, boolean preferences) helps manage newsletter subscriptions and preferences cleanly. The unique email constraint prevents duplicates, and the default values seem correct.

website/views/user.py (6)

1196-1222: Review of newsletter_home function.

This function correctly handles paginated newsletters and creates an activity record for authenticated users. Watch out for potential edge cases when no newsletter exists (the code already handles featured_newsletter is None gracefully). Great job!


1225-1273: Review of newsletter_detail function.

  1. Good practice to mask IP addresses for non-staff users.
  2. The usage of increment_view_count() is straightforward, but verify it handles potential concurrency if the site is under heavy load (e.g., atomic increments).

Overall, this function is well-structured and secure for displaying newsletter details.


1373-1421: send_confirmation_email function is well-structured.

  1. Properly retrieves domain using Site.objects.get_current() and logs relevant info.
  2. Good detail in exception handling to avoid silent failures.

No major concerns here.


1423-1468: newsletter_confirm function safely checks token.

  1. Use of try/except around is_token_expired() is cautious.
  2. Logs essential details for debugging.

Implementation aligns well with best practices for token verification.


1470-1537: newsletter_unsubscribe function includes IP-based rate limiting.

  1. Enforcing a limit of 10 attempts per hour is reasonable for preventing abuse.
  2. Token aged beyond 90 days triggers a refresh—a practical approach to avoid invalid unsubscribe links floating around.

Implementation looks good overall.


1539-1569: newsletter_preferences function ensures user-friendly defaults.

By creating a subscription record if none exists, you accommodate both brand-new and existing users. The code is succinct and easy to maintain.

coderabbitai[bot]
coderabbitai bot previously approved these changes Mar 15, 2025
coderabbitai[bot]
coderabbitai bot previously approved these changes Mar 15, 2025
coderabbitai[bot]
coderabbitai bot previously approved these changes Mar 15, 2025
coderabbitai[bot]
coderabbitai bot previously approved these changes Mar 15, 2025
coderabbitai[bot]
coderabbitai bot previously approved these changes Mar 15, 2025
coderabbitai[bot]
coderabbitai bot previously approved these changes Mar 15, 2025
@tsu-ki tsu-ki requested a review from DonnieBLT March 15, 2025 22:06
@tsu-ki
Copy link
Contributor Author

tsu-ki commented Mar 15, 2025

@DonnieBLT sir, I've made the required corrections. I've tried to resolve the failing pre-commit test but not sure why it's not getting resolved. please review the changes, thank you!

@coderabbitai coderabbitai bot mentioned this pull request Mar 16, 2025
@DonnieBLT
Copy link
Collaborator

/giphy nice feature

Copy link
Contributor

Giphy GIF

@DonnieBLT
Copy link
Collaborator

can you please update this

Copy link
Collaborator

@DonnieBLT DonnieBLT left a comment

Choose a reason for hiding this comment

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

please update

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.

update the newsletter with new releases
2 participants