perkeep/website/contributors.go

203 lines
4.8 KiB
Go

package main
import (
"bufio"
"bytes"
"fmt"
"log"
"net/http"
"os"
"os/exec"
"sort"
"strconv"
"strings"
"perkeep.org/internal/osutil"
)
var urlsMap = map[string]author{
"brad@danga.com": {URL: "http://bradfitz.com/", Role: "founder, lead"},
"bslatkin@gmail.com": {URL: "http://www.onebigfluke.com/", Role: "co-founder"},
"mathieu.lonjaret@gmail.com": {
URL: "https://granivo.re/mpl.html",
Role: "has touched almost everything",
Names: []string{"Mathieu Lonjaret"},
},
"zboogs@gmail.com": {URL: "http://www.aaronboodman.com/", Role: "web interface lead"},
"adg@golang.org": {URL: "http://nf.id.au/"},
"dustin@spy.net": {URL: "http://dustin.sallings.org/"},
"dan@erat.org": {URL: "http://www.erat.org/"},
"martine@danga.com": {URL: "http://neugierig.org/"},
"agl@golang.org": {URL: "http://www.imperialviolet.org/"},
"lsimon@commoner.com": {Role: "original publishing UI"},
"s@0x65.net": {URL: "https://0x65.net/"},
}
type author struct {
Names []string
Emails []string
Commits int
Role string
URL string
}
// add merges src's fields into a's.
func (a *author) add(src *author) {
if src == nil {
return
}
a.Emails = append(a.Emails, src.Emails...)
a.Names = append(a.Names, src.Names...)
a.Commits += src.Commits
if src.Role != "" {
a.Role = src.Role
}
if src.URL != "" {
a.URL = src.URL
}
}
type Authors []*author
func (s Authors) Len() int { return len(s) }
func (s Authors) Swap(i, j int) { s[i], s[j] = s[j], s[i] }
func (s Authors) Less(i, j int) bool { return s[i].Commits > s[j].Commits }
func parseLine(l string) (name, email string, commits int, err error) {
t := strings.Split(strings.TrimSpace(l), " ")
if len(t) < 2 {
err = fmt.Errorf("line too short")
return
}
i := strings.LastIndex(t[1], " ")
email = strings.Trim(t[1][i+1:], "<>")
name = t[1][:i]
commits, err = strconv.Atoi(t[0])
return
}
func gitShortlog() *exec.Cmd {
if !*gitContainer {
return exec.Command("/bin/bash", "-c", "git log | git shortlog -sen")
}
args := []string{"run", "--rm"}
if inProd {
args = append(args,
"-v", "/var/camweb:/var/camweb",
"--workdir="+prodSrcDir,
)
} else {
hostRoot, err := osutil.GoPackagePath(prodDomain)
if err != nil {
log.Fatal(err)
}
log.Printf("Using bind root of %q", hostRoot)
args = append(args,
"-v", hostRoot+":"+prodSrcDir,
"--workdir="+prodSrcDir,
)
}
args = append(args, "camlistore/git", "/bin/bash", "-c", "git log | git shortlog -sen")
cmd := exec.Command("docker", args...)
cmd.Stderr = os.Stderr
return cmd
}
func genContribPage() ([]byte, error) {
contribHTML := readTemplate("contributors.html")
// The same committers could've authored commits with different emails/usersnames.
// We index the authors by name and by email to try and merge graphs connected by
// the "same-email" and "same-name" relation into one entity.
byName := make(map[string]*author)
byEmail := make(map[string]*author)
authorMap := make(map[*author]bool)
shortlog := gitShortlog()
shortlogOut, err := shortlog.StdoutPipe()
if err != nil {
return nil, err
}
err = shortlog.Start()
if err != nil {
return nil, fmt.Errorf("couldn't run git shortlog: %v", err)
}
scn := bufio.NewScanner(shortlogOut)
for scn.Scan() {
name, email, commits, err := parseLine(scn.Text())
if err != nil {
return nil, err
}
a := &author{
Emails: []string{email},
Names: []string{name},
Commits: commits,
}
a.add(byName[name])
a.add(byEmail[email])
for _, n := range a.Names {
delete(authorMap, byName[n])
byName[n] = a
}
for _, e := range a.Emails {
delete(authorMap, byEmail[e])
byEmail[e] = a
}
authorMap[a] = true
}
if scn.Err() != nil {
return nil, scn.Err()
}
err = shortlog.Wait()
if err != nil {
return nil, fmt.Errorf("git shortlog failed: %v", err)
}
// Add URLs and roles
for email, m := range urlsMap {
a := byEmail[email]
if a != nil {
a.add(&m)
}
if len(m.Names) > 0 {
a.Names = []string{m.Names[0]}
}
}
authors := Authors{}
for a, _ := range authorMap {
authors = append(authors, a)
}
sort.Sort(authors)
b := &bytes.Buffer{}
err = contribHTML.Execute(b, authors)
return b.Bytes(), err
}
// contribHandler returns a handler that serves the generated contributors page,
// or the static file handler if it couldn't run git for any reason.
func contribHandler() http.HandlerFunc {
c, err := genContribPage()
if err != nil {
log.Printf("Couldn't generate contributors page: %v", err)
log.Printf("Using static contributors page")
return mainHandler
}
title := ""
if m := h1TitlePattern.FindSubmatch(c); len(m) > 1 {
title = string(m[1])
}
return func(w http.ResponseWriter, r *http.Request) {
servePage(w, r, pageParams{
title: title,
content: c,
})
}
}