- Replace nav inline newsletter form with Subscribe CTA link per wireframe - Remove newsletter form from footer; add Connect section with social/RSS links - Fix honeypot inputs using hidden attribute (inline style blocked by CSP) - Add available_tags to HomePage.get_context for Explore Topics section - Add data-comment-form attribute to main comment form for reliable locating - Seed approved comment in E2E content for reply flow testing - Expand test_comments.py: moderation message, not-immediately-visible, missing fields, reply form visible, reply submission - Make COMMENT_RATE_LIMIT_PER_MINUTE configurable; set 100 in dev to prevent E2E test exhaustion; update rate limit unit test with override_settings - Update newsletter/home E2E tests to reflect nav form removal - Update unit test to assert no nav/footer newsletter forms Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
164 lines
4.9 KiB
Python
164 lines
4.9 KiB
Python
import pytest
|
|
from taggit.models import Tag
|
|
|
|
from apps.blog.models import ArticleIndexPage, ArticlePage
|
|
from apps.blog.tests.factories import AuthorFactory
|
|
from apps.comments.models import Comment
|
|
|
|
|
|
@pytest.mark.django_db
|
|
def test_homepage_render(client, home_page):
|
|
resp = client.get("/")
|
|
assert resp.status_code == 200
|
|
|
|
|
|
@pytest.mark.django_db
|
|
def test_article_index_pagination_and_tag_filter(client, home_page):
|
|
index = ArticleIndexPage(title="Articles", slug="articles")
|
|
home_page.add_child(instance=index)
|
|
author = AuthorFactory()
|
|
for n in range(14):
|
|
article = ArticlePage(
|
|
title=f"A{n}",
|
|
slug=f"a{n}",
|
|
author=author,
|
|
summary="summary",
|
|
body=[("rich_text", "<p>body</p>")],
|
|
)
|
|
index.add_child(instance=article)
|
|
article.save_revision().publish()
|
|
|
|
resp = client.get("/articles/?page=2")
|
|
assert resp.status_code == 200
|
|
assert resp.context["articles"].number == 2
|
|
assert "Pagination" in resp.content.decode()
|
|
|
|
|
|
@pytest.mark.django_db
|
|
def test_article_page_related_context(client, home_page):
|
|
index = ArticleIndexPage(title="Articles", slug="articles")
|
|
home_page.add_child(instance=index)
|
|
author = AuthorFactory()
|
|
main = ArticlePage(
|
|
title="Main",
|
|
slug="main",
|
|
author=author,
|
|
summary="summary",
|
|
body=[("rich_text", "<p>body</p>")],
|
|
)
|
|
index.add_child(instance=main)
|
|
main.save_revision().publish()
|
|
|
|
related = ArticlePage(
|
|
title="Related",
|
|
slug="related",
|
|
author=author,
|
|
summary="summary",
|
|
body=[("rich_text", "<p>body</p>")],
|
|
)
|
|
index.add_child(instance=related)
|
|
related.save_revision().publish()
|
|
|
|
resp = client.get("/articles/main/")
|
|
assert resp.status_code == 200
|
|
assert "related_articles" in resp.context
|
|
|
|
|
|
@pytest.mark.django_db
|
|
def test_newsletter_forms_render_in_nav_and_footer(client, home_page):
|
|
resp = client.get("/")
|
|
html = resp.content.decode()
|
|
assert resp.status_code == 200
|
|
# Nav has a Subscribe CTA link (no inline form — wireframe spec)
|
|
assert 'href="#newsletter"' in html
|
|
# Footer has Connect section with social/RSS links (no newsletter form)
|
|
assert "Connect" in html
|
|
assert 'name="source" value="nav"' not in html
|
|
assert 'name="source" value="footer"' not in html
|
|
|
|
|
|
@pytest.mark.django_db
|
|
def test_article_page_renders_share_links_and_newsletter_form(client, home_page):
|
|
index = ArticleIndexPage(title="Articles", slug="articles")
|
|
home_page.add_child(instance=index)
|
|
author = AuthorFactory()
|
|
article = ArticlePage(
|
|
title="Main",
|
|
slug="main",
|
|
author=author,
|
|
summary="summary",
|
|
body=[("rich_text", "<p>body</p>")],
|
|
)
|
|
index.add_child(instance=article)
|
|
article.save_revision().publish()
|
|
|
|
resp = client.get("/articles/main/")
|
|
html = resp.content.decode()
|
|
assert resp.status_code == 200
|
|
assert "Share on X" in html
|
|
assert "Share on LinkedIn" in html
|
|
assert 'data-copy-link' in html
|
|
assert 'name="source" value="article"' in html
|
|
|
|
|
|
@pytest.mark.django_db
|
|
def test_article_page_renders_approved_comments_and_reply_form(client, home_page):
|
|
index = ArticleIndexPage(title="Articles", slug="articles")
|
|
home_page.add_child(instance=index)
|
|
author = AuthorFactory()
|
|
article = ArticlePage(
|
|
title="Main",
|
|
slug="main",
|
|
author=author,
|
|
summary="summary",
|
|
body=[("rich_text", "<p>body</p>")],
|
|
)
|
|
index.add_child(instance=article)
|
|
article.save_revision().publish()
|
|
comment = Comment.objects.create(
|
|
article=article,
|
|
author_name="A",
|
|
author_email="a@example.com",
|
|
body="Top level",
|
|
is_approved=True,
|
|
)
|
|
Comment.objects.create(
|
|
article=article,
|
|
parent=comment,
|
|
author_name="B",
|
|
author_email="b@example.com",
|
|
body="Reply",
|
|
is_approved=True,
|
|
)
|
|
|
|
resp = client.get("/articles/main/")
|
|
html = resp.content.decode()
|
|
assert resp.status_code == 200
|
|
assert "Top level" in html
|
|
assert "Reply" in html
|
|
assert f'name="parent_id" value="{comment.id}"' in html
|
|
|
|
|
|
@pytest.mark.django_db
|
|
def test_article_index_renders_tag_filter_controls(client, home_page):
|
|
index = ArticleIndexPage(title="Articles", slug="articles")
|
|
home_page.add_child(instance=index)
|
|
author = AuthorFactory()
|
|
article = ArticlePage(
|
|
title="Main",
|
|
slug="main",
|
|
author=author,
|
|
summary="summary",
|
|
body=[("rich_text", "<p>body</p>")],
|
|
)
|
|
index.add_child(instance=article)
|
|
article.save_revision().publish()
|
|
tag = Tag.objects.create(name="TagOne", slug="tag-one")
|
|
article.tags.add(tag)
|
|
article.save_revision().publish()
|
|
|
|
resp = client.get("/articles/")
|
|
html = resp.content.decode()
|
|
assert resp.status_code == 200
|
|
assert "/articles/?tag=tag-one" in html
|