[tools] Add 'add-gerrit-hashtags' tool
Parses the CL descriptions and adds missing hashtags to Gerrit changes. Also add ./tools/push-to-gerrit which runs this after pushing the local branch's changes to 'main'. Use this for the VSCode 'push' task. Change-Id: I4c3f5982f6fdc7c1c6ebe770fc7811b1b38795d1 Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/133061 Commit-Queue: Ben Clayton <bclayton@google.com> Kokoro: Ben Clayton <bclayton@google.com> Reviewed-by: Dan Sinclair <dsinclair@chromium.org>
This commit is contained in:
parent
ee8cdb5a36
commit
1a8d0785f2
|
@ -125,17 +125,12 @@
|
||||||
{
|
{
|
||||||
"label": "push",
|
"label": "push",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "git",
|
"command": "./tools/push-to-gerrit",
|
||||||
"args": [
|
|
||||||
"push",
|
|
||||||
"origin",
|
|
||||||
"HEAD:refs/for/main"
|
|
||||||
],
|
|
||||||
"options": {
|
"options": {
|
||||||
"cwd": "${workspaceRoot}"
|
"cwd": "${workspaceRoot}"
|
||||||
},
|
},
|
||||||
"problemMatcher": [],
|
"problemMatcher": [],
|
||||||
}
|
},
|
||||||
],
|
],
|
||||||
"inputs": [
|
"inputs": [
|
||||||
{
|
{
|
||||||
|
|
2
go.mod
2
go.mod
|
@ -3,7 +3,7 @@ module dawn.googlesource.com/dawn
|
||||||
go 1.18
|
go 1.18
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/andygrunwald/go-gerrit v0.0.0-20220427111355-d3e91fbf2db5
|
github.com/andygrunwald/go-gerrit v0.0.0-20230508072829-423d372345aa
|
||||||
github.com/ben-clayton/webidlparser v0.0.0-20210923100217-8ba896ded094
|
github.com/ben-clayton/webidlparser v0.0.0-20210923100217-8ba896ded094
|
||||||
github.com/fatih/color v1.13.0
|
github.com/fatih/color v1.13.0
|
||||||
github.com/google/go-cmp v0.5.9
|
github.com/google/go-cmp v0.5.9
|
||||||
|
|
2
go.sum
2
go.sum
|
@ -8,6 +8,8 @@ cloud.google.com/go/longrunning v0.4.1 h1:v+yFJOfKC3yZdY6ZUI933pIYdhyhV8S3NpWrXW
|
||||||
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
||||||
github.com/andygrunwald/go-gerrit v0.0.0-20220427111355-d3e91fbf2db5 h1:HBlTlvyq4siv4ZK41DebGIX11/9gFBqUF8G64AePjyQ=
|
github.com/andygrunwald/go-gerrit v0.0.0-20220427111355-d3e91fbf2db5 h1:HBlTlvyq4siv4ZK41DebGIX11/9gFBqUF8G64AePjyQ=
|
||||||
github.com/andygrunwald/go-gerrit v0.0.0-20220427111355-d3e91fbf2db5/go.mod h1:aqcjwEnmLLSalFNYR0p2ttnEXOVVRctIzsUMHbEcruU=
|
github.com/andygrunwald/go-gerrit v0.0.0-20220427111355-d3e91fbf2db5/go.mod h1:aqcjwEnmLLSalFNYR0p2ttnEXOVVRctIzsUMHbEcruU=
|
||||||
|
github.com/andygrunwald/go-gerrit v0.0.0-20230508072829-423d372345aa h1:bGSzPoUh/2eduqGEk54TCoB4v81MVi6Hr3+fYQwFrBM=
|
||||||
|
github.com/andygrunwald/go-gerrit v0.0.0-20230508072829-423d372345aa/go.mod h1:SeP12EkHZxEVjuJ2HZET304NBtHGG2X6w2Gzd0QXAZw=
|
||||||
github.com/ben-clayton/webidlparser v0.0.0-20210923100217-8ba896ded094 h1:CTVJdI6oUCRNucMEmoh3c2U88DesoPtefsxKhoZ1WuQ=
|
github.com/ben-clayton/webidlparser v0.0.0-20210923100217-8ba896ded094 h1:CTVJdI6oUCRNucMEmoh3c2U88DesoPtefsxKhoZ1WuQ=
|
||||||
github.com/ben-clayton/webidlparser v0.0.0-20210923100217-8ba896ded094/go.mod h1:bV550SPlMos7UhMprxlm14XTBTpKHSUZ8Q4Id5qQuyw=
|
github.com/ben-clayton/webidlparser v0.0.0-20210923100217-8ba896ded094/go.mod h1:bV550SPlMos7UhMprxlm14XTBTpKHSUZ8Q4Id5qQuyw=
|
||||||
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
|
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
|
||||||
|
|
|
@ -0,0 +1,22 @@
|
||||||
|
#!/usr/bin/env bash
|
||||||
|
# Copyright 2023 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.
|
||||||
|
|
||||||
|
set -e # Fail on any error.
|
||||||
|
|
||||||
|
SCRIPT_DIR="$( cd "$( dirname "${BASH_SOURCE[0]}")" >/dev/null 2>&1 && pwd )"
|
||||||
|
|
||||||
|
git push origin HEAD:refs/for/main
|
||||||
|
|
||||||
|
${SCRIPT_DIR}/run add-gerrit-hashtags
|
|
@ -0,0 +1,168 @@
|
||||||
|
// Copyright 2023 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.
|
||||||
|
|
||||||
|
// add-gerrit-hashtags adds any missing hashtags parsed from the CL description to the Gerrit change.
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"dawn.googlesource.com/dawn/tools/src/container"
|
||||||
|
"dawn.googlesource.com/dawn/tools/src/dawn"
|
||||||
|
"dawn.googlesource.com/dawn/tools/src/gerrit"
|
||||||
|
"dawn.googlesource.com/dawn/tools/src/git"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
toolName = "add-gerrit-hashtags"
|
||||||
|
yyyymmdd = "2006-01-02"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// See https://dawn-review.googlesource.com/new-password for obtaining
|
||||||
|
// username and password for gerrit.
|
||||||
|
gerritUser = flag.String("gerrit-user", "", "gerrit authentication username")
|
||||||
|
gerritPass = flag.String("gerrit-pass", "", "gerrit authentication password")
|
||||||
|
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", 30, "interval in days (used if --after is not specified)")
|
||||||
|
verboseFlag = flag.Bool("v", false, "verbose mode - lists all the changes")
|
||||||
|
dryrunFlag = flag.Bool("dry", false, "dry mode. Don't apply any changes")
|
||||||
|
)
|
||||||
|
|
||||||
|
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() {
|
||||||
|
flag.Usage = func() {
|
||||||
|
out := flag.CommandLine.Output()
|
||||||
|
fmt.Fprintf(out, "%v adds any missing hashtags parsed from the CL description to the Gerrit change.\n", toolName)
|
||||||
|
fmt.Fprintf(out, "\n")
|
||||||
|
flag.PrintDefaults()
|
||||||
|
}
|
||||||
|
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(24 * time.Hour)
|
||||||
|
}
|
||||||
|
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))
|
||||||
|
}
|
||||||
|
|
||||||
|
g, err := gerrit.New(dawn.GerritURL, gerrit.Credentials{
|
||||||
|
Username: *gerritUser, Password: *gerritPass,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
submitted, _, err := g.QueryChanges(
|
||||||
|
"owner:"+user,
|
||||||
|
"after:"+date(after),
|
||||||
|
"before:"+date(before),
|
||||||
|
"repo:"+*repoFlag)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Query failed: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
numUpdated := 0
|
||||||
|
for _, cl := range submitted {
|
||||||
|
expected := parseHashtags(cl.Subject)
|
||||||
|
got := container.NewSet(cl.Hashtags...)
|
||||||
|
if !got.ContainsAll(expected) {
|
||||||
|
toAdd := expected.Clone()
|
||||||
|
toAdd.RemoveAll(got)
|
||||||
|
fmt.Printf("%v: %v missing hashtags: %v\n", cl.Number, cl.Subject, strings.Join(toAdd.List(), ", "))
|
||||||
|
if !*dryrunFlag {
|
||||||
|
if err := g.AddHashtags(cl.ChangeID, toAdd); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
numUpdated++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if numUpdated > 0 {
|
||||||
|
fmt.Println()
|
||||||
|
fmt.Println(numUpdated, "changes updated with new hashtags")
|
||||||
|
} else {
|
||||||
|
fmt.Println("no changes updated")
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var reBracketHashtag = regexp.MustCompile(`\[(\w+)\]`)
|
||||||
|
var reColonHashtag = regexp.MustCompile(`^(\w+):`)
|
||||||
|
|
||||||
|
func parseHashtags(subject string) container.Set[string] {
|
||||||
|
out := container.NewSet[string]()
|
||||||
|
for _, match := range reBracketHashtag.FindAllStringSubmatch(subject, -1) {
|
||||||
|
out.Add(match[1])
|
||||||
|
}
|
||||||
|
if match := reColonHashtag.FindStringSubmatch(subject); len(match) > 1 {
|
||||||
|
out.Add(match[1])
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func today() time.Time {
|
||||||
|
return time.Now()
|
||||||
|
}
|
||||||
|
|
||||||
|
func date(t time.Time) string {
|
||||||
|
return t.Format(yyyymmdd)
|
||||||
|
}
|
|
@ -35,11 +35,11 @@ var (
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
flag.ErrHelp = errors.New("benchdiff is a tool to compare two benchmark results")
|
flag.ErrHelp = errors.New("benchdiff is a tool to compare two benchmark results")
|
||||||
flag.Parse()
|
|
||||||
flag.Usage = func() {
|
flag.Usage = func() {
|
||||||
fmt.Fprintln(os.Stderr, "benchdiff <benchmark-a> <benchmark-b>")
|
fmt.Fprintln(os.Stderr, "benchdiff <benchmark-a> <benchmark-b>")
|
||||||
flag.PrintDefaults()
|
flag.PrintDefaults()
|
||||||
}
|
}
|
||||||
|
flag.Parse()
|
||||||
|
|
||||||
args := flag.Args()
|
args := flag.Args()
|
||||||
if len(args) < 2 {
|
if len(args) < 2 {
|
||||||
|
|
|
@ -116,7 +116,7 @@ func (r *ResultSource) GetResults(ctx context.Context, cfg Config, auth auth.Opt
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
*ps, err = gerrit.LatestPatchest(strconv.Itoa(ps.Change))
|
*ps, err = gerrit.LatestPatchset(strconv.Itoa(ps.Change))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err := fmt.Errorf("failed to find latest patchset of change %v: %w",
|
err := fmt.Errorf("failed to find latest patchset of change %v: %w",
|
||||||
ps.Change, err)
|
ps.Change, err)
|
||||||
|
@ -288,7 +288,7 @@ func LatestCTSRoll(g *gerrit.Gerrit) (gerrit.ChangeInfo, error) {
|
||||||
|
|
||||||
// LatestPatchset returns the most recent patchset for the given change.
|
// LatestPatchset returns the most recent patchset for the given change.
|
||||||
func LatestPatchset(g *gerrit.Gerrit, change int) (gerrit.Patchset, error) {
|
func LatestPatchset(g *gerrit.Gerrit, change int) (gerrit.Patchset, error) {
|
||||||
ps, err := g.LatestPatchest(strconv.Itoa(change))
|
ps, err := g.LatestPatchset(strconv.Itoa(change))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err := fmt.Errorf("failed to find latest patchset of change %v: %w",
|
err := fmt.Errorf("failed to find latest patchset of change %v: %w",
|
||||||
ps.Change, err)
|
ps.Change, err)
|
||||||
|
|
|
@ -25,6 +25,7 @@ import (
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"dawn.googlesource.com/dawn/tools/src/container"
|
||||||
"github.com/andygrunwald/go-gerrit"
|
"github.com/andygrunwald/go-gerrit"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -55,8 +56,8 @@ type Patchset struct {
|
||||||
// ChangeInfo is an alias to gerrit.ChangeInfo
|
// ChangeInfo is an alias to gerrit.ChangeInfo
|
||||||
type ChangeInfo = gerrit.ChangeInfo
|
type ChangeInfo = gerrit.ChangeInfo
|
||||||
|
|
||||||
// LatestPatchest returns the latest Patchset from the ChangeInfo
|
// LatestPatchset returns the latest Patchset from the ChangeInfo
|
||||||
func LatestPatchest(change *ChangeInfo) Patchset {
|
func LatestPatchset(change *ChangeInfo) Patchset {
|
||||||
u, _ := url.Parse(change.URL)
|
u, _ := url.Parse(change.URL)
|
||||||
ps := Patchset{
|
ps := Patchset{
|
||||||
Host: u.Host,
|
Host: u.Host,
|
||||||
|
@ -198,11 +199,11 @@ func (g *Gerrit) EditFiles(changeID, newCommitMsg string, files map[string]strin
|
||||||
return Patchset{}, g.maybeWrapError(err)
|
return Patchset{}, g.maybeWrapError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return g.LatestPatchest(changeID)
|
return g.LatestPatchset(changeID)
|
||||||
}
|
}
|
||||||
|
|
||||||
// LatestPatchest returns the latest patchset for the change.
|
// LatestPatchset returns the latest patchset for the change.
|
||||||
func (g *Gerrit) LatestPatchest(changeID string) (Patchset, error) {
|
func (g *Gerrit) LatestPatchset(changeID string) (Patchset, error) {
|
||||||
change, _, err := g.client.Changes.GetChange(changeID, &gerrit.ChangeOptions{
|
change, _, err := g.client.Changes.GetChange(changeID, &gerrit.ChangeOptions{
|
||||||
AdditionalFields: []string{"CURRENT_REVISION"},
|
AdditionalFields: []string{"CURRENT_REVISION"},
|
||||||
})
|
})
|
||||||
|
@ -218,6 +219,17 @@ func (g *Gerrit) LatestPatchest(changeID string) (Patchset, error) {
|
||||||
return ps, nil
|
return ps, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// AddHashtags adds the given hashtags to the change
|
||||||
|
func (g *Gerrit) AddHashtags(changeID string, tags container.Set[string]) error {
|
||||||
|
_, resp, err := g.client.Changes.SetHashtags(changeID, &gerrit.HashtagsInput{
|
||||||
|
Add: tags.List(),
|
||||||
|
})
|
||||||
|
if err != nil && resp.StatusCode != 409 { // 409: already ready
|
||||||
|
return g.maybeWrapError(err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// CommentSide is an enumerator for specifying which side code-comments should
|
// CommentSide is an enumerator for specifying which side code-comments should
|
||||||
// be shown.
|
// be shown.
|
||||||
type CommentSide int
|
type CommentSide int
|
||||||
|
|
Loading…
Reference in New Issue