Skip to content

Commit bf5f948

Browse files
committed
Make /patch/{id} the URL for a patch
Previously we'd include the ID of the commitfest in the URL of the patch. In 9f12a5e we introduced a stable URL for patches that would redirect to the one for the latest commitfest. This starts to use that URL as the valid only URL for a patch (with the previous URL redirecting to this one). The reasoning behind this is that the old approach resulted in N different URLs for each patch, which all showed the exact same patch information. The only difference between all these URLs would be the breadcrumb at the top of the page. The only benefit of that approach is that if you're on an old commitfest, and click a link there, then the breadcrumb will bring you back to where you came from. Since people rarely have a reason to browse closed commitfests, the that benefit seems pretty small. Especially because people can just as well press their browser back button, in that case. The problems that these N links cause seem much more impactful to most users: 1. If you click an old link to a cf entry (e.g. one in the email archives), then the breadcrumb will contain some arbitrarily old commitfest. It seems much more useful to have the breadcrumb show the commitfest that the patch is currently active in (or got committed/rejected in). 2. Places that use the stable URLs require an extra round-trip to actually get to the patch page. 3. It's a bit confusing that old pages of a patch still get updated with all the new information, i.e. why have all these pages if they contain the exact same content. 4. Problem 3 is generally also bad for Search Engine Optimization (SEO), for now we don't care much about that though. Finally this also changes the links on the patch page itself for each of the commitfests that a patch has been part of. Those links were already rather useless, since all they effectively did was change the breadcrumb. But with this new commit, they wouldn't even do that anymore, and simply redirect to the current page. So now they start pointing to the commitfest itself, which seems more useful behaviour anyway.
1 parent b4dff24 commit bf5f948

File tree

5 files changed

+46
-35
lines changed

5 files changed

+46
-35
lines changed

pgcommitfest/commitfest/models.py

+3
Original file line numberDiff line numberDiff line change
@@ -122,6 +122,9 @@ class Patch(models.Model, DiffableModel):
122122
'reviewers': 'reviewers_string',
123123
}
124124

125+
def current_commitfest(self):
126+
return self.commitfests.order_by('-startdate').first()
127+
125128
# Some accessors
126129
@property
127130
def authors_string(self):

pgcommitfest/commitfest/templates/commitfest.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -85,7 +85,7 @@ <h3>{{p.is_open|yesno:"Active patches,Closed patches"}}</h3>
8585
{%endifchanged%}
8686
{%endif%}
8787
<tr>
88-
<td><a href="{{p.id}}/">{{p.name}}</a></td>
88+
<td><a href="/patch/{{p.id}}/">{{p.name}}</a></td>
8989
<td>{{p.id}}</td>
9090
<td><span class="label label-{{p.status|patchstatuslabel}}">{{p.status|patchstatusstring}}</span></td>
9191
<td>{%if p.targetversion%}<span class="label label-default">{{p.targetversion}}</span>{%endif%}</td>

pgcommitfest/commitfest/templates/patch.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -68,7 +68,7 @@
6868
<tr>
6969
<th>Status</th>
7070
<td>{%for c in patch_commitfests %}
71-
<div style="margin-bottom: 3px;"><a href="/{{c.commitfest.id}}/{{patch.id}}/">{{c.commitfest}}</a>: <span class="label label-{{c.status|patchstatuslabel}}">{{c.statusstring}}</span></div>
71+
<div style="margin-bottom: 3px;"><a href="/{{c.commitfest.id}}/">{{c.commitfest}}</a>: <span class="label label-{{c.status|patchstatuslabel}}">{{c.statusstring}}</span></div>
7272
{%endfor%}
7373
</td>
7474
</tr>

pgcommitfest/commitfest/views.py

+31-23
Original file line numberDiff line numberDiff line change
@@ -320,17 +320,18 @@ def global_search(request):
320320
})
321321

322322

323-
def patch_redirect(request, patchid):
324-
last_commitfest = PatchOnCommitFest.objects.select_related('commitfest').filter(patch_id=patchid).order_by('-commitfest__startdate').first()
325-
if not last_commitfest:
326-
raise Http404("Patch not found")
327-
return HttpResponseRedirect(f'/{last_commitfest.commitfest_id}/{patchid}/')
323+
def patch_legacy_redirect(request, cfid, patchid):
324+
# Previously we would include the commitfest id in the URL. This is no
325+
# longer the case.
326+
return HttpResponseRedirect(f'/patch/{patchid}/')
328327

329328

330-
def patch(request, cfid, patchid):
331-
cf = get_object_or_404(CommitFest, pk=cfid)
332-
patch = get_object_or_404(Patch.objects.select_related(), pk=patchid, commitfests=cf)
333-
patch_commitfests = PatchOnCommitFest.objects.select_related('commitfest').filter(patch=patch).order_by('-commitfest__startdate')
329+
def patch(request, patchid):
330+
patch = get_object_or_404(Patch.objects.select_related(), pk=patchid)
331+
cf = patch.current_commitfest()
332+
333+
patch_commitfests = PatchOnCommitFest.objects.select_related('commitfest').filter(patch=patch).order_by('-commitfest__startdate').all()
334+
334335
committers = Committer.objects.filter(active=True).order_by('user__last_name', 'user__first_name')
335336

336337
cfbot_branch = getattr(patch, 'cfbot_branch', None)
@@ -373,9 +374,9 @@ def patch(request, cfid, patchid):
373374

374375
@login_required
375376
@transaction.atomic
376-
def patchform(request, cfid, patchid):
377-
cf = get_object_or_404(CommitFest, pk=cfid)
378-
patch = get_object_or_404(Patch, pk=patchid, commitfests=cf)
377+
def patchform(request, patchid):
378+
patch = get_object_or_404(Patch, pk=patchid)
379+
cf = patch.current_commitfest()
379380

380381
prevreviewers = list(patch.reviewers.all())
381382
prevauthors = list(patch.authors.all())
@@ -465,9 +466,9 @@ def _review_status_string(reviewstatus):
465466

466467
@login_required
467468
@transaction.atomic
468-
def comment(request, cfid, patchid, what):
469-
cf = get_object_or_404(CommitFest, pk=cfid)
469+
def comment(request, patchid, what):
470470
patch = get_object_or_404(Patch, pk=patchid)
471+
cf = patch.current_commitfest()
471472
poc = get_object_or_404(PatchOnCommitFest, patch=patch, commitfest=cf)
472473
is_review = (what == 'review')
473474

@@ -562,8 +563,10 @@ def comment(request, cfid, patchid, what):
562563

563564
@login_required
564565
@transaction.atomic
565-
def status(request, cfid, patchid, status):
566-
poc = get_object_or_404(PatchOnCommitFest.objects.select_related(), commitfest__id=cfid, patch__id=patchid)
566+
def status(request, patchid, status):
567+
patch = get_object_or_404(Patch.objects.select_related(), pk=patchid)
568+
cf = patch.current_commitfest()
569+
poc = get_object_or_404(PatchOnCommitFest.objects.select_related(), commitfest__id=cf.id, patch__id=patchid)
567570

568571
if poc.is_closed:
569572
# We allow modification of patches in closed CFs *only* if it's the
@@ -597,8 +600,10 @@ def status(request, cfid, patchid, status):
597600

598601
@login_required
599602
@transaction.atomic
600-
def close(request, cfid, patchid, status):
601-
poc = get_object_or_404(PatchOnCommitFest.objects.select_related(), commitfest__id=cfid, patch__id=patchid)
603+
def close(request, patchid, status):
604+
patch = get_object_or_404(Patch.objects.select_related(), pk=patchid)
605+
cf = patch.current_commitfest()
606+
poc = get_object_or_404(PatchOnCommitFest.objects.select_related(), commitfest__id=cf.id, patch__id=patchid)
602607

603608
if poc.is_closed:
604609
# We allow modification of patches in closed CFs *only* if it's the
@@ -695,8 +700,7 @@ def close(request, cfid, patchid, status):
695700

696701
@login_required
697702
@transaction.atomic
698-
def reviewer(request, cfid, patchid, status):
699-
get_object_or_404(CommitFest, pk=cfid)
703+
def reviewer(request, patchid, status):
700704
patch = get_object_or_404(Patch, pk=patchid)
701705

702706
is_reviewer = request.user in patch.reviewers.all()
@@ -715,7 +719,6 @@ def reviewer(request, cfid, patchid, status):
715719
@login_required
716720
@transaction.atomic
717721
def committer(request, cfid, patchid, status):
718-
get_object_or_404(CommitFest, pk=cfid)
719722
patch = get_object_or_404(Patch, pk=patchid)
720723

721724
committer = list(Committer.objects.filter(user=request.user, active=True))
@@ -740,8 +743,7 @@ def committer(request, cfid, patchid, status):
740743

741744
@login_required
742745
@transaction.atomic
743-
def subscribe(request, cfid, patchid, sub):
744-
get_object_or_404(CommitFest, pk=cfid)
746+
def subscribe(request, patchid, sub):
745747
patch = get_object_or_404(Patch, pk=patchid)
746748

747749
if sub == 'un':
@@ -754,6 +756,12 @@ def subscribe(request, cfid, patchid, sub):
754756
return HttpResponseRedirect("../")
755757

756758

759+
def send_patch_email(request, patchid):
760+
patch = get_object_or_404(Patch, pk=patchid)
761+
cf = patch.current_commitfest()
762+
return send_email(request, cf.id)
763+
764+
757765
@login_required
758766
@transaction.atomic
759767
def send_email(request, cfid):

pgcommitfest/urls.py

+10-10
Original file line numberDiff line numberDiff line change
@@ -19,18 +19,18 @@
1919
re_path(r'^(\d+)/$', views.commitfest),
2020
re_path(r'^(open|inprogress|current)/(.*)$', views.redir),
2121
re_path(r'^(?P<cfid>\d+)/activity(?P<rss>\.rss)?/$', views.activity),
22-
re_path(r'^patch/(\d+)/$', views.patch_redirect),
23-
re_path(r'^(\d+)/(\d+)/$', views.patch),
24-
re_path(r'^(\d+)/(\d+)/edit/$', views.patchform),
22+
re_path(r'^(\d+)/(\d+)/$', views.patch_legacy_redirect),
23+
re_path(r'^patch/(\d+)/$', views.patch),
24+
re_path(r'^patch/(\d+)/edit/$', views.patchform),
2525
re_path(r'^(\d+)/new/$', views.newpatch),
26-
re_path(r'^(\d+)/(\d+)/status/(review|author|committer)/$', views.status),
27-
re_path(r'^(\d+)/(\d+)/close/(reject|withdrawn|feedback|committed|next)/$', views.close),
28-
re_path(r'^(\d+)/(\d+)/reviewer/(become|remove)/$', views.reviewer),
29-
re_path(r'^(\d+)/(\d+)/committer/(become|remove)/$', views.committer),
30-
re_path(r'^(\d+)/(\d+)/(un)?subscribe/$', views.subscribe),
31-
re_path(r'^(\d+)/(\d+)/(comment|review)/', views.comment),
26+
re_path(r'^patch/(\d+)/status/(review|author|committer)/$', views.status),
27+
re_path(r'^patch/(\d+)/close/(reject|withdrawn|feedback|committed|next)/$', views.close),
28+
re_path(r'^patch/(\d+)/reviewer/(become|remove)/$', views.reviewer),
29+
re_path(r'^patch/(\d+)/committer/(become|remove)/$', views.committer),
30+
re_path(r'^patch/(\d+)/(un)?subscribe/$', views.subscribe),
31+
re_path(r'^patch/(\d+)/(comment|review)/', views.comment),
3232
re_path(r'^(\d+)/send_email/$', views.send_email),
33-
re_path(r'^(\d+)/\d+/send_email/$', views.send_email),
33+
re_path(r'^patch/(\d+)/send_email/$', views.send_patch_email),
3434
re_path(r'^(\d+)/reports/authorstats/$', reports.authorstats),
3535
re_path(r'^search/$', views.global_search),
3636
re_path(r'^ajax/(\w+)/$', ajax.main),

0 commit comments

Comments
 (0)