2015-08-12 23:04:19 +02:00
|
|
|
// Copyright (C) 2015 The Syncthing Authors.
|
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
2017-02-09 07:52:18 +01:00
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
2015-08-12 23:04:19 +02:00
|
|
|
|
|
|
|
// Checks for authors that are not mentioned in AUTHORS
|
2017-07-07 20:43:26 +00:00
|
|
|
package meta
|
2015-08-12 23:04:19 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"io/ioutil"
|
|
|
|
"os/exec"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
2017-07-07 20:43:26 +00:00
|
|
|
"testing"
|
2015-08-12 23:04:19 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// list of commits that we don't include in our checks; because they are
|
|
|
|
// legacy things that don't check code, are committed with incorrect address,
|
|
|
|
// or for other reasons.
|
|
|
|
var excludeCommits = stringSetFromStrings([]string{
|
|
|
|
"63bd0136fb40a91efaa279cb4b4159d82e8e6904",
|
|
|
|
"4e2feb6fbc791bb8a2daf0ab8efb10775d66343e",
|
|
|
|
"f2459ef3319b2f060dbcdacd0c35a1788a94b8bd",
|
|
|
|
"b61f418bf2d1f7d5a9d7088a20a2a448e5e66801",
|
|
|
|
"a9339d0627fff439879d157c75077f02c9fac61b",
|
|
|
|
"254c63763a3ad42fd82259f1767db526cff94a14",
|
|
|
|
"4b76ec40c07078beaa2c5e250ed7d9bd6276a718",
|
|
|
|
"32a76901a91ff0f663db6f0830e0aedec946e4d0",
|
|
|
|
"3626003f680bad3e63677982576d3a05421e88e9",
|
2015-11-12 08:46:13 +01:00
|
|
|
"342036408e65bd25bb6afbcc705e2e2c013bb01f",
|
2015-11-16 20:48:40 +01:00
|
|
|
"e37cefdbee1c1cd95ad095b5da6d1252723f103b",
|
2016-07-04 14:46:24 +02:00
|
|
|
"bcc5d7c00f52552303b463d43a636f27b7f7e19b",
|
2015-08-12 23:04:19 +02:00
|
|
|
})
|
|
|
|
|
2017-07-07 20:43:26 +00:00
|
|
|
func TestCheckAuthors(t *testing.T) {
|
|
|
|
if testing.Short() {
|
|
|
|
t.Skip("skipping slow test")
|
|
|
|
}
|
2015-08-12 23:04:19 +02:00
|
|
|
|
2017-07-07 20:43:26 +00:00
|
|
|
actual, hashes := actualAuthorEmails(t, ".", "../cmd/", "../lib/", "../gui/", "../test/", "../script/")
|
|
|
|
listed := listedAuthorEmails(t)
|
2015-08-12 23:04:19 +02:00
|
|
|
missing := actual.except(listed)
|
2017-07-07 20:43:26 +00:00
|
|
|
for author := range missing {
|
|
|
|
t.Logf("Missing author: %s", author)
|
|
|
|
for _, hash := range hashes[author] {
|
|
|
|
t.Logf(" in hash: %s", hash)
|
2015-08-12 23:04:19 +02:00
|
|
|
}
|
2017-07-07 20:43:26 +00:00
|
|
|
}
|
|
|
|
if len(missing) > 0 {
|
|
|
|
t.Errorf("Missing %d author(s)", len(missing))
|
2015-08-12 23:04:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// actualAuthorEmails returns the set of author emails found in the actual git
|
|
|
|
// commit log, except those in excluded commits.
|
2017-07-07 20:43:26 +00:00
|
|
|
func actualAuthorEmails(t *testing.T, paths ...string) (stringSet, map[string][]string) {
|
2016-03-01 18:11:06 +01:00
|
|
|
args := append([]string{"log", "--format=%H %ae"}, paths...)
|
|
|
|
cmd := exec.Command("git", args...)
|
2015-08-12 23:04:19 +02:00
|
|
|
bs, err := cmd.Output()
|
|
|
|
if err != nil {
|
2017-07-07 20:43:26 +00:00
|
|
|
t.Fatal("authorEmails:", err)
|
2015-08-12 23:04:19 +02:00
|
|
|
}
|
|
|
|
|
2017-07-07 20:43:26 +00:00
|
|
|
hashes := make(map[string][]string)
|
2015-08-12 23:04:19 +02:00
|
|
|
authors := newStringSet()
|
|
|
|
for _, line := range bytes.Split(bs, []byte{'\n'}) {
|
|
|
|
fields := strings.Fields(string(line))
|
|
|
|
if len(fields) != 2 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
hash, author := fields[0], fields[1]
|
|
|
|
if excludeCommits.has(hash) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2017-07-07 20:43:26 +00:00
|
|
|
if strings.Contains(strings.ToLower(body(t, hash)), "skip-check: authors") {
|
2016-05-08 10:47:57 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2015-08-12 23:04:19 +02:00
|
|
|
authors.add(author)
|
2017-07-07 20:43:26 +00:00
|
|
|
hashes[author] = append(hashes[author], hash)
|
2015-08-12 23:04:19 +02:00
|
|
|
}
|
|
|
|
|
2017-07-07 20:43:26 +00:00
|
|
|
return authors, hashes
|
2015-08-12 23:04:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// listedAuthorEmails returns the set of author emails mentioned in AUTHORS
|
2017-07-07 20:43:26 +00:00
|
|
|
func listedAuthorEmails(t *testing.T) stringSet {
|
|
|
|
bs, err := ioutil.ReadFile("../AUTHORS")
|
2015-08-12 23:04:19 +02:00
|
|
|
if err != nil {
|
2017-07-07 20:43:26 +00:00
|
|
|
t.Fatal("listedAuthorEmails:", err)
|
2015-08-12 23:04:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
emailRe := regexp.MustCompile(`<([^>]+)>`)
|
|
|
|
matches := emailRe.FindAllStringSubmatch(string(bs), -1)
|
|
|
|
|
|
|
|
authors := newStringSet()
|
|
|
|
for _, match := range matches {
|
|
|
|
authors.add(match[1])
|
|
|
|
}
|
|
|
|
return authors
|
|
|
|
}
|
|
|
|
|
2017-07-07 20:43:26 +00:00
|
|
|
func body(t *testing.T, hash string) string {
|
2016-05-08 10:47:57 +00:00
|
|
|
cmd := exec.Command("git", "show", "--pretty=format:%b", "-s", hash)
|
|
|
|
bs, err := cmd.Output()
|
|
|
|
if err != nil {
|
2017-07-07 20:43:26 +00:00
|
|
|
t.Fatal("body:", err)
|
2016-05-08 10:47:57 +00:00
|
|
|
}
|
|
|
|
return string(bs)
|
|
|
|
}
|
|
|
|
|
2015-08-12 23:04:19 +02:00
|
|
|
// A simple string set type
|
|
|
|
|
|
|
|
type stringSet map[string]struct{}
|
|
|
|
|
|
|
|
func newStringSet() stringSet {
|
|
|
|
return make(stringSet)
|
|
|
|
}
|
|
|
|
|
|
|
|
func stringSetFromStrings(ss []string) stringSet {
|
|
|
|
s := newStringSet()
|
|
|
|
for _, e := range ss {
|
|
|
|
s.add(e)
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s stringSet) add(e string) {
|
|
|
|
s[e] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s stringSet) has(e string) bool {
|
|
|
|
_, ok := s[e]
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s stringSet) except(other stringSet) stringSet {
|
|
|
|
diff := newStringSet()
|
|
|
|
for e := range s {
|
|
|
|
if !other.has(e) {
|
|
|
|
diff.add(e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return diff
|
|
|
|
}
|