| // Copyright 2021 The Tint Authors. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| // gerrit-stats gathers statistics about changes made to Tint. |
| package main |
| |
| import ( |
| "context" |
| "flag" |
| "fmt" |
| "net/url" |
| "os" |
| "os/exec" |
| "regexp" |
| "time" |
| |
| "dawn.googlesource.com/dawn/tools/src/auth" |
| "dawn.googlesource.com/dawn/tools/src/dawn" |
| "dawn.googlesource.com/dawn/tools/src/gerrit" |
| "dawn.googlesource.com/dawn/tools/src/git" |
| "go.chromium.org/luci/auth/client/authcli" |
| ) |
| |
| const yyyymmdd = "2006-01-02" |
| |
| var ( |
| repoFlag = flag.String("repo", "dawn", "the project (tint or dawn)") |
| userFlag = flag.String("user", defaultUser(), "user name / email") |
| afterFlag = flag.String("after", "", "start date") |
| beforeFlag = flag.String("before", "", "end date") |
| daysFlag = flag.Int("days", 182, "interval in days (used if --after is not specified)") |
| verboseFlag = flag.Bool("v", false, "verbose mode - lists all the changes") |
| authFlags = authcli.Flags{} |
| ) |
| |
| func defaultUser() string { |
| if gitExe, err := exec.LookPath("git"); err == nil { |
| if g, err := git.New(gitExe); err == nil { |
| if cwd, err := os.Getwd(); err == nil { |
| if r, err := g.Open(cwd); err == nil { |
| if cfg, err := r.Config(nil); err == nil { |
| return cfg["user.email"] |
| } |
| } |
| } |
| } |
| } |
| return "" |
| } |
| |
| func main() { |
| authFlags.Register(flag.CommandLine, auth.DefaultAuthOptions()) |
| |
| flag.Parse() |
| if err := run(); err != nil { |
| fmt.Fprintln(os.Stderr, err) |
| os.Exit(1) |
| } |
| } |
| |
| func run() error { |
| var after, before time.Time |
| var err error |
| user := *userFlag |
| if user == "" { |
| return fmt.Errorf("Missing required 'user' flag") |
| } |
| if *beforeFlag != "" { |
| before, err = time.Parse(yyyymmdd, *beforeFlag) |
| if err != nil { |
| return fmt.Errorf("Couldn't parse before date: %w", err) |
| } |
| } else { |
| before = time.Now().Add(time.Hour * 24) |
| } |
| if *afterFlag != "" { |
| after, err = time.Parse(yyyymmdd, *afterFlag) |
| if err != nil { |
| return fmt.Errorf("Couldn't parse after date: %w", err) |
| } |
| } else { |
| after = before.Add(-time.Hour * time.Duration(24**daysFlag)) |
| } |
| |
| ctx := context.Background() |
| auth, err := authFlags.Options() |
| if err != nil { |
| return err |
| } |
| |
| g, err := gerrit.New(ctx, auth, dawn.GerritURL) |
| if err != nil { |
| return err |
| } |
| |
| submitted, submittedQuery, err := g.QueryChanges( |
| "status:merged", |
| "owner:"+user, |
| "after:"+date(after), |
| "before:"+date(before), |
| "repo:"+*repoFlag) |
| if err != nil { |
| return fmt.Errorf("Query failed: %w", err) |
| } |
| |
| reviewed, reviewQuery, err := g.QueryChanges( |
| "commentby:"+user, |
| "-owner:"+user, |
| "after:"+date(after), |
| "before:"+date(before), |
| "repo:"+*repoFlag) |
| if err != nil { |
| return fmt.Errorf("Query failed: %w", err) |
| } |
| |
| ignorelist := []*regexp.Regexp{ |
| regexp.MustCompile("Revert .*"), |
| } |
| ignore := func(s string) bool { |
| for _, re := range ignorelist { |
| if re.MatchString(s) { |
| return true |
| } |
| } |
| return false |
| } |
| |
| insertions, deletions := 0, 0 |
| for _, change := range submitted { |
| if ignore(change.Subject) { |
| continue |
| } |
| insertions += change.Insertions |
| deletions += change.Deletions |
| } |
| |
| fmt.Printf("Between %v and %v, %v:\n", date(after), date(before), user) |
| fmt.Printf(" Submitted %v changes (LOC: %v+, %v-) \n", len(submitted), insertions, deletions) |
| fmt.Printf(" Reviewed %v changes\n", len(reviewed)) |
| fmt.Printf("\n") |
| |
| if *verboseFlag { |
| fmt.Printf("Submitted changes:\n") |
| for i, change := range submitted { |
| fmt.Printf("%3.1v: %6.v %v (LOC: %v+, %v-)\n", i, change.Number, change.Subject, change.Insertions, change.Deletions) |
| } |
| fmt.Printf("\n") |
| fmt.Printf("Reviewed changes:\n") |
| for i, change := range reviewed { |
| fmt.Printf("%3.1v: %6.v %v (LOC: %v+, %v-)\n", i, change.Number, change.Subject, change.Insertions, change.Deletions) |
| } |
| } |
| |
| fmt.Printf("\n") |
| fmt.Printf("Submitted query: %vq/%v\n", dawn.GerritURL, url.QueryEscape(submittedQuery)) |
| fmt.Printf("Review query: %vq/%v\n", dawn.GerritURL, url.QueryEscape(reviewQuery)) |
| |
| return nil |
| } |
| |
| func today() time.Time { |
| return time.Now() |
| } |
| |
| func date(t time.Time) string { |
| return t.Format(yyyymmdd) |
| } |