Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 16 additions & 0 deletions scripts/dev/release-dates/AGENTS.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
- Write the script in pure Python with no dependencies.
- Work backward from the provided Release Date, which we prefer to be on a Wednesday.
- Curation Team Review starts on the Monday prior. After the date, put "(allocate 5 full days)".
- Code Freeze starts on the Thursday prior.
- Core PR Last Call starts on the Thursday prior.
- Community PR Last Call starts on the Thursday prior.
- Start starts three months before the Release Date, on a Thursday. It's ok to put "??" for the sprint number.

Here's some example output:

- Start: Sprint ??, 2026-03-26
- Community PR Last Call: 2026-05-21
- Core PR Last Call: 2026-05-28
- Code Freeze: 2026-06-04
- Curation Team Review: 2026-06-08 (allocate 5 full days)
- Release Date: 2026-06-17
66 changes: 66 additions & 0 deletions scripts/dev/release-dates/generate_release_dates.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
#!/usr/bin/env python3

"""Generate release milestone dates from a release date."""

from __future__ import annotations

import sys
from datetime import date, timedelta


USAGE = "Usage: generate_release_dates.py YYYY-MM-DD"


def parse_release_date(value: str) -> date:
try:
return date.fromisoformat(value)
except ValueError as exc:
raise SystemExit(f"Invalid date '{value}'. Expected YYYY-MM-DD.") from exc


def previous_thursday(reference: date) -> date:
days_since_thursday = (reference.weekday() - 3) % 7
return reference - timedelta(days=days_since_thursday or 7)


def build_schedule(release_date: date) -> list[tuple[str, str]]:
code_freeze = previous_thursday(release_date - timedelta(days=7))
core_pr_last_call = code_freeze - timedelta(days=7)
community_pr_last_call = core_pr_last_call - timedelta(days=7)
curation_team_review = code_freeze + timedelta(days=4)
start = community_pr_last_call - timedelta(days=56)

return [
("Start", f"Sprint ??, {start.isoformat()}"),
("Community PR Last Call", community_pr_last_call.isoformat()),
("Core PR Last Call", core_pr_last_call.isoformat()),
("Code Freeze", code_freeze.isoformat()),
(
"Curation Team Review",
f"{curation_team_review.isoformat()} (allocate 5 full days)",
),
("Release Date", release_date.isoformat()),
]


def main(argv: list[str]) -> int:
if len(argv) != 2 or argv[1] in {"-h", "--help"}:
print(USAGE, file=sys.stderr if len(argv) != 2 else sys.stdout)
return 1 if len(argv) != 2 else 0

release_date = parse_release_date(argv[1])

if release_date.weekday() != 2:
print(
"Warning: Release Date is usually expected to be a Wednesday.",
file=sys.stderr,
)

for label, value in build_schedule(release_date):
print(f"- {label}: {value}")

return 0


if __name__ == "__main__":
raise SystemExit(main(sys.argv))
Loading