hledger/tools/relnotes.hs

130 lines
5.4 KiB
Haskell
Executable File

#!/usr/bin/env stack
{- stack script --resolver nightly-2024-12-19 --compile
--package process
--package split
-}
-- --verbosity=info
{-
Make/update hledger release notes from the latest release's changelogs.
Run this in the root of the hledger repo, in a release branch.
If you have switched branch recently you may have to kill an unhelpful
ghc environment file first. Eg: rm -f .ghc.environment* && tools/relnotes.hs
This reads the latest release's changes from the main CHANGES.md files,
converts to release notes format, and inserts this into relnotes.md.
If there already were release notes for this release (at top of relnotes),
they will be replaced, except for a manually added "highlights" paragraph if any
(after the release heading, beginning and ending with **), which is preserved.
The release's version is taken from the the first h1 heading in the project CHANGES.md.
It is assumed that all the other changelogs' first section is also for this version (XXX),
and that changelogs and relnotes are non-empty with specific layout and heading formats.
In the end I wrote this in haskell because everything else was harder.
2024-02-08 TODO:
preserve last release's summary
fix headings
unwrap long lines
merge md-issue-refs
-}
{-# OPTIONS_GHC -Wno-x-partial #-}
import Control.Monad
import Data.Char
import Data.List
import Data.List.Split
import System.Exit
import System.IO
import System.Process
import Text.Printf
strToVer = splitOn "."
verToStr = intercalate "."
main = do
-- gather latest release changes & info
(projectChangesHeading, projectChanges) <- changelogFirstSection <$> readFile "CHANGES.md"
(hledgerChangesHeading, hledgerChanges) <- changelogFirstSection <$> readFile "hledger/CHANGES.md"
(hledgerUiChangesHeading, hledgerUiChanges) <- changelogFirstSection <$> readFile "hledger-ui/CHANGES.md"
(hledgerWebChangesHeading, hledgerWebChanges) <- changelogFirstSection <$> readFile "hledger-web/CHANGES.md"
reltags <- lines <$> readProcess "git" ["tag", "--sort=-creatordate", "-l", "[0-9]*"] ""
printf $ "previous release tags: " <> unwords (take 5 reltags) <> " ...\n"
let
[_, ver, date] = words projectChangesHeading
verexists = ver `elem` reltags
printf $ "project CHANGES.md's top heading: " <> projectChangesHeading
printf $ "inferred latest release version and date: " <> intercalate ", " [ver, date] <> "\n"
printf $ "a tag for this release " <> (if verexists then "exists" else "does not yet exist") <> "\n"
let prevvers =
map verToStr $ dropWhile (>=strToVer ver) $ map strToVer $ reltags
printf $ "releases before this one: " <> unwords (take 5 prevvers) <> " ...\n"
when (null prevvers) $ do
printf $ "error: no previous releases found. This expects to run before new release headings are added to changelogs\n"
exitFailure
let prevver = head prevvers
printf $ "previous release: " <> prevver <> "\n"
relauthors <- map (unwords . drop 1 . words) . lines <$> readProcess "git" ["shortlog", "-sn", prevver<>".."<>if verexists then ver else ""] ""
printf $ "this release's authors: " <> intercalate ", " relauthors <> "\n"
-- convert to release notes format
let
newrelnotesheading = printf "## %s hledger-%s\n" date ver
newrelnotesbody = intercalate "\n\n" [
changelogHeadingToRelnotesHeading "hledger" hledgerChangesHeading, hledgerChanges,
changelogHeadingToRelnotesHeading "hledger-ui" hledgerUiChangesHeading, hledgerUiChanges,
changelogHeadingToRelnotesHeading "hledger-web" hledgerWebChangesHeading, hledgerWebChanges,
"### project changes " <> ver <> "\n", projectChanges,
"### credits " <> ver <> "\n", intercalate ",\n" relauthors <> ".\n"
] <> "\n\n"
-- insert or update in relnotes.md
let relnotesfile = "doc/relnotes.md"
(preamble, rnlatestver, rnlatesthighlights, rnlatestbody, rest) <-
relnotesSections <$> readFile' relnotesfile
-- putStrLn $
writeFile relnotesfile $
unlines [
preamble,
newrelnotesheading,
if null rnlatesthighlights then "" else rnlatesthighlights <> "\n",
newrelnotesbody,
if rnlatestver == ver then "" else rnlatestbody,
rest
] <> "\n"
changelogFirstSection alltext =
case uncons $ dropWhile (not.isReleaseHeading) $ lines alltext of
Nothing -> ("","")
Just (l,ls) -> (nl l, strip' $ unlines $ takeWhile (not.isReleaseHeading) ls)
where
isReleaseHeading = isPrefixOf "# "
changelogHeadingToRelnotesHeading pkgname heading =
let [_, ver, _] = words heading
in printf "### %s %s\n" pkgname ver
relnotesSections alltext = (unlines preamble, firstsectionver, firstsectionhighlights, firstsectionbody, rest)
where
isReleaseHeading = isPrefixOf "## "
(preamble, (firstsectionheading:ls)) = span (not.isReleaseHeading) $ lines alltext
firstsectionver = drop 8 $ (!!2) $ words firstsectionheading
ls2 = dropWhile null ls
(firstsectionhighlights, ls3) =
case ls2 of
('*':'*':_):_ -> (unlines $ takeWhile (not.null) ls2, dropWhile (not.null) ls2)
_ -> ("", ls2)
(firstsectionls, restls) = span (not.isReleaseHeading) ls3
firstsectionbody = unlines $ firstsectionheading : firstsectionls
rest = unlines restls
unlines' = intercalate "\n"
lstrip = dropWhile isSpace
rstrip = reverse . lstrip . reverse
strip = lstrip . rstrip
strip' = nl . strip
nl = (<>"\n")