blob: b2b35b6b9b27b88ecf26e74afea1c1fbf5f073af [file] [log] [blame]
// Copyright 2019 Google Inc. All rights reserved.
//
// 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.
package main
import (
"bytes"
"crypto/rand"
"encoding/base64"
"encoding/json"
"flag"
"fmt"
"io"
"io/ioutil"
"log"
"net/http"
"net/rpc"
"net/url"
"os"
"path"
"strings"
"time"
"github.com/google/fmtserver"
"github.com/google/slothfs/cookie"
)
var jsonPrefix = []byte(")]}'")
type File struct {
Status string
LinesInserted int `json:"lines_inserted"`
SizeDelta int `json:"size_delta"`
Size int
Content []byte
}
type Change struct {
Files map[string]*File
}
type gerrit struct {
UserAgent string
URL url.URL
client http.Client
basicAuth string
}
func newGerrit(u url.URL) *gerrit {
g := &gerrit{
URL: u,
}
g.client.CheckRedirect = func(req *http.Request, via []*http.Request) error {
g.setRequest(req)
return nil
}
return g
}
func (g *gerrit) setAuth(auth []byte) {
encoded := make([]byte, base64.StdEncoding.EncodedLen(len(auth)))
base64.StdEncoding.Encode(encoded, auth)
g.basicAuth = string(encoded)
}
func (g *gerrit) setRequest(req *http.Request) {
req.Header.Set("User-Agent", g.UserAgent)
req.Header.Set("Authorization", "Basic "+string(g.basicAuth))
}
func (g *gerrit) getPath(p string) ([]byte, error) {
u := g.URL
u.Path = path.Join(u.Path, p)
if strings.HasSuffix(p, "/") && !strings.HasSuffix(u.Path, "/") {
// Ugh.
u.Path += "/"
}
req, err := http.NewRequest("GET", u.String(), nil)
if err != nil {
return nil, err
}
g.setRequest(req)
rep, err := g.client.Do(req)
if err != nil {
return nil, err
}
if rep.StatusCode != 200 {
return nil, fmt.Errorf("Get %s: status %d", u.String(), rep.StatusCode)
}
defer rep.Body.Close()
return ioutil.ReadAll(rep.Body)
}
func (g *gerrit) postPath(p string, contentType string, content []byte) ([]byte, error) {
u := g.URL
u.Path = path.Join(u.Path, p)
if strings.HasSuffix(p, "/") && !strings.HasSuffix(u.Path, "/") {
// Ugh.
u.Path += "/"
}
req, err := http.NewRequest("POST", u.String(), bytes.NewBuffer(content))
if err != nil {
return nil, err
}
g.setRequest(req)
req.Header.Set("Content-Type", contentType)
rep, err := g.client.Do(req)
if err != nil {
return nil, err
}
if rep.StatusCode/100 != 2 {
return nil, fmt.Errorf("Post %s: status %d", u.String(), rep.StatusCode)
}
defer rep.Body.Close()
return ioutil.ReadAll(rep.Body)
}
// GetContent returns the file content from a file in a change.
func (g *gerrit) GetContent(changeID string, revID string, fileID string) ([]byte, error) {
c, err := g.getPath(fmt.Sprintf("changes/%s/revisions/%s/files/%s/content",
url.PathEscape(changeID), revID, url.PathEscape(fileID)))
if err != nil {
return nil, err
}
dest := make([]byte, base64.StdEncoding.DecodedLen(len(c)))
n, err := base64.StdEncoding.Decode(dest, c)
if err != nil {
return nil, err
}
return dest[:n], nil
}
// GetChange returns the Change (including file contents) for a given change.
func (g *gerrit) GetChange(changeID string, revID string) (*Change, error) {
content, err := g.getPath(fmt.Sprintf("changes/%s/revisions/%s/files/",
url.PathEscape(changeID), revID))
if err != nil {
return nil, err
}
content = bytes.TrimPrefix(content, jsonPrefix)
files := map[string]*File{}
if err := json.Unmarshal(content, &files); err != nil {
return nil, err
}
for name := range files {
c, err := g.GetContent(changeID, revID, name)
if err != nil {
return nil, err
}
files[name].Content = c
}
return &Change{files}, nil
}
type CheckerInput struct {
UUID string `json:"uuid"`
Name string `json:"name"`
Description string `json:"description"`
URL string `json:"url"`
Repository string `json:"repository"`
Status string `json:"status"`
Blocking []string `json:"blocking"`
Query string `json:"query"`
}
const timeLayout = "2006-01-02 15:04:05.000000000"
type Timestamp time.Time
func (ts *Timestamp) String() string {
return ((time.Time)(*ts)).String()
}
func (ts *Timestamp) MarshalJSON() ([]byte, error) {
t := (*time.Time)(ts)
return []byte("\"" + t.Format(timeLayout) + "\""), nil
}
func (ts *Timestamp) UnmarshalJSON(b []byte) error {
b = bytes.TrimPrefix(b, []byte{'"'})
b = bytes.TrimSuffix(b, []byte{'"'})
t, err := time.Parse(timeLayout, string(b))
if err != nil {
return err
}
*ts = Timestamp(t)
log.Println("unmarshal", t)
return nil
}
type CheckerInfo struct {
UUID string `json:"uuid"`
Name string
Description string
URL string `json:"url"`
Repository string `json:"repository"`
Status string
Blocking []string `json:"blocking"`
Query string `json:"query"`
Created Timestamp `json:"created"`
Updated Timestamp `json:"updated"`
}
type wrapJar struct {
http.CookieJar
}
func (w *wrapJar) Cookies(u *url.URL) []*http.Cookie {
cs := w.CookieJar.Cookies(u)
// log.Println("cookies for", u, cs)
return cs
}
func (w *wrapJar) SetCookies(u *url.URL, cs []*http.Cookie) {
w.CookieJar.SetCookies(u, cs)
cs = w.CookieJar.Cookies(u)
}
func (g *gerrit) CreateChecker(repo string) (*CheckerInfo, error) {
var uuidRandom [20]byte
rand.Reader.Read(uuidRandom[:])
uuid := fmt.Sprintf("fmt:%x", uuidRandom)
in := CheckerInput{
UUID: uuid,
Name: "fmtserver",
Repository: repo,
Description: "check source code formatting.",
Status: "ENABLED",
// TODO: should list all file extensions in the query?
}
body, err := json.Marshal(&in)
if err != nil {
return nil, err
}
content, err := g.postPath("a/plugins/checks/checkers/", "application/json", body)
if err != nil {
return nil, err
}
if !bytes.HasPrefix(content, jsonPrefix) {
if len(content) > 100 {
content = content[:100]
}
bodyStr := string(content)
return nil, fmt.Errorf("prefix %q not found, got %s", jsonPrefix, bodyStr)
}
content = bytes.TrimPrefix(content, []byte(jsonPrefix))
out := CheckerInfo{}
log.Printf("return value: %s", string(content))
if err := json.Unmarshal(content, &out); err != nil {
return nil, fmt.Errorf("Unmarshal: %v", err)
}
log.Printf("created checker %#v", out)
return &out, nil
}
type gerritChecker struct {
gerrit *gerrit
fmtClient *rpc.Client
}
func (c *gerritChecker) checkChange(changeID string) error {
ch, err := c.gerrit.GetChange(changeID, "current")
if err != nil {
return err
}
req := fmtserver.FormatRequest{}
for n, f := range ch.Files {
req.Files = append(req.Files,
fmtserver.File{
Name: n,
Content: f.Content,
})
}
rep := fmtserver.FormatReply{}
if err := c.fmtClient.Call("Server.Format", &req, &rep); err != nil {
_, ok := err.(rpc.ServerError)
if ok {
return fmt.Errorf("server returned: %s", err)
}
return err
}
for _, f := range rep.Files {
orig := ch.Files[f.Name]
if orig == nil {
return fmt.Errorf("result had unknown file %q", f.Name)
}
if !bytes.Equal(f.Content, orig.Content) {
msg := f.Message
if msg == "" {
msg = "needs formatting"
}
log.Printf("file %s: %s", f.Name, f.Message)
}
}
return nil
}
func main() {
gerritURL := flag.String("gerrit", "", "URL to gerrit host")
addr := flag.String("addr", "", "Address of the fmtserver")
register := flag.Bool("register", false, "Register with the host")
list := flag.Bool("list", false, "List pending checks")
agent := flag.String("agent", "fmtserver", "user-agent for the fmtserver.")
cookieJar := flag.String("cookies", "", "comma separated paths to cURL-style cookie jar file.")
auth := flag.String("auth_file", "", "file containing user:password")
repo := flag.String("repo", "", "the repository (project) name to apply the checker to.")
flag.Parse()
if *gerritURL == "" {
log.Fatal("must set --gerrit")
}
u, err := url.Parse(*gerritURL)
if err != nil {
log.Fatalf("url.Parse: %v", err)
}
g := newGerrit(*u)
if nm := *cookieJar; nm != "" {
jar, err := cookie.NewJar(nm)
if err != nil {
log.Fatal("NewJar(%s): %v", nm, err)
}
if err := cookie.WatchJar(jar, nm); err != nil {
log.Printf("WatchJar: %v", err)
log.Println("continuing despite WatchJar failure", err)
}
g.client.Jar = &wrapJar{jar}
}
g.UserAgent = *agent
if *auth == "" {
log.Fatal("must set --auth_file")
}
if content, err := ioutil.ReadFile(*auth); err != nil {
log.Fatal(err)
} else {
g.setAuth(bytes.TrimSpace(content))
}
// Do a GET first to complete any cookie dance, because POST aren't redirected properly.
if c, err := g.getPath("a/accounts/self"); err != nil {
log.Fatalf("accounts/self: %v", err)
} else {
io.Copy(os.Stdout, bytes.NewBuffer(c))
}
if *list {
if c, err := g.getPath("plugins/checks/checkers/"); err != nil {
log.Fatalf("ListCheckers: %v", err)
} else {
io.Copy(os.Stdout, bytes.NewBuffer(c))
}
os.Exit(0)
}
if *register {
if *repo == "" {
log.Fatalf("need to set --repo")
}
ch, err := g.CreateChecker(*repo)
if err != nil {
log.Fatalf("CreateChecker: %v", err)
}
log.Printf("CreateChecker result: %v", ch)
os.Exit(0)
}
if *addr == "" {
log.Fatal("must set --addr")
}
client, err := rpc.DialHTTP("tcp", *addr)
if err != nil {
log.Fatalf("DialHTTP(%s): %v", *addr, err)
}
gc := gerritChecker{
gerrit: g,
fmtClient: client,
}
if len(flag.Args()) < 1 {
log.Fatal("pass change IDs on command line")
}
for _, a := range flag.Args() {
if err := gc.checkChange(a); err != nil {
log.Printf("change %s: %v", a, err)
}
}
}