mirror of
https://github.com/golang/go
synced 2024-11-22 03:44:39 -07:00
dashboard: more descriptive logging, ui tweaks, show better auth error
R=golang-dev, dsymonds CC=golang-dev https://golang.org/cl/5505050
This commit is contained in:
parent
1dfe3d1f6e
commit
9f0e39b992
@ -9,6 +9,7 @@ import (
|
|||||||
"appengine/datastore"
|
"appengine/datastore"
|
||||||
"bytes"
|
"bytes"
|
||||||
"compress/gzip"
|
"compress/gzip"
|
||||||
|
"crypto/hmac"
|
||||||
"crypto/sha1"
|
"crypto/sha1"
|
||||||
"fmt"
|
"fmt"
|
||||||
"http"
|
"http"
|
||||||
@ -18,7 +19,14 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
const commitsPerPage = 20
|
var defaultPackages = []*Package{
|
||||||
|
&Package{Name: "Go"},
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
commitsPerPage = 30
|
||||||
|
maxDatastoreStringLen = 500
|
||||||
|
)
|
||||||
|
|
||||||
// A Package describes a package that is listed on the dashboard.
|
// A Package describes a package that is listed on the dashboard.
|
||||||
type Package struct {
|
type Package struct {
|
||||||
@ -111,11 +119,13 @@ func (c *Commit) Valid() os.Error {
|
|||||||
// It must be called from inside a datastore transaction.
|
// It must be called from inside a datastore transaction.
|
||||||
func (com *Commit) AddResult(c appengine.Context, r *Result) os.Error {
|
func (com *Commit) AddResult(c appengine.Context, r *Result) os.Error {
|
||||||
if err := datastore.Get(c, com.Key(c), com); err != nil {
|
if err := datastore.Get(c, com.Key(c), com); err != nil {
|
||||||
return err
|
return fmt.Errorf("getting Commit: %v", err)
|
||||||
}
|
}
|
||||||
com.ResultData = append(com.ResultData, r.Data())
|
com.ResultData = append(com.ResultData, r.Data())
|
||||||
_, err := datastore.Put(c, com.Key(c), com)
|
if _, err := datastore.Put(c, com.Key(c), com); err != nil {
|
||||||
return err
|
return fmt.Errorf("putting Commit: %v", err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Result returns the build Result for this Commit for the given builder/goHash.
|
// Result returns the build Result for this Commit for the given builder/goHash.
|
||||||
@ -267,7 +277,7 @@ func commitHandler(r *http.Request) (interface{}, os.Error) {
|
|||||||
com.PackagePath = r.FormValue("packagePath")
|
com.PackagePath = r.FormValue("packagePath")
|
||||||
com.Hash = r.FormValue("hash")
|
com.Hash = r.FormValue("hash")
|
||||||
if err := datastore.Get(c, com.Key(c), com); err != nil {
|
if err := datastore.Get(c, com.Key(c), com); err != nil {
|
||||||
return nil, err
|
return nil, fmt.Errorf("getting Commit: %v", err)
|
||||||
}
|
}
|
||||||
return com, nil
|
return com, nil
|
||||||
}
|
}
|
||||||
@ -278,10 +288,13 @@ func commitHandler(r *http.Request) (interface{}, os.Error) {
|
|||||||
// POST request
|
// POST request
|
||||||
defer r.Body.Close()
|
defer r.Body.Close()
|
||||||
if err := json.NewDecoder(r.Body).Decode(com); err != nil {
|
if err := json.NewDecoder(r.Body).Decode(com); err != nil {
|
||||||
return nil, err
|
return nil, fmt.Errorf("decoding Body: %v", err)
|
||||||
|
}
|
||||||
|
if len(com.Desc) > maxDatastoreStringLen {
|
||||||
|
com.Desc = com.Desc[:maxDatastoreStringLen]
|
||||||
}
|
}
|
||||||
if err := com.Valid(); err != nil {
|
if err := com.Valid(); err != nil {
|
||||||
return nil, err
|
return nil, fmt.Errorf("validating Commit: %v", err)
|
||||||
}
|
}
|
||||||
tx := func(c appengine.Context) os.Error {
|
tx := func(c appengine.Context) os.Error {
|
||||||
return addCommit(c, com)
|
return addCommit(c, com)
|
||||||
@ -292,21 +305,24 @@ func commitHandler(r *http.Request) (interface{}, os.Error) {
|
|||||||
// addCommit adds the Commit entity to the datastore and updates the tip Tag.
|
// addCommit adds the Commit entity to the datastore and updates the tip Tag.
|
||||||
// It must be run inside a datastore transaction.
|
// It must be run inside a datastore transaction.
|
||||||
func addCommit(c appengine.Context, com *Commit) os.Error {
|
func addCommit(c appengine.Context, com *Commit) os.Error {
|
||||||
// if this commit is already in the datastore, do nothing
|
|
||||||
var tc Commit // temp value so we don't clobber com
|
var tc Commit // temp value so we don't clobber com
|
||||||
err := datastore.Get(c, com.Key(c), &tc)
|
err := datastore.Get(c, com.Key(c), &tc)
|
||||||
if err != datastore.ErrNoSuchEntity {
|
if err != datastore.ErrNoSuchEntity {
|
||||||
return err
|
// if this commit is already in the datastore, do nothing
|
||||||
|
if err == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return fmt.Errorf("getting Commit: %v", err)
|
||||||
}
|
}
|
||||||
// get the next commit number
|
// get the next commit number
|
||||||
p, err := GetPackage(c, com.PackagePath)
|
p, err := GetPackage(c, com.PackagePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return fmt.Errorf("GetPackage: %v", err)
|
||||||
}
|
}
|
||||||
com.Num = p.NextNum
|
com.Num = p.NextNum
|
||||||
p.NextNum++
|
p.NextNum++
|
||||||
if _, err := datastore.Put(c, p.Key(c), p); err != nil {
|
if _, err := datastore.Put(c, p.Key(c), p); err != nil {
|
||||||
return err
|
return fmt.Errorf("putting Package: %v", err)
|
||||||
}
|
}
|
||||||
// if this isn't the first Commit test the parent commit exists
|
// if this isn't the first Commit test the parent commit exists
|
||||||
if com.Num > 0 {
|
if com.Num > 0 {
|
||||||
@ -315,7 +331,7 @@ func addCommit(c appengine.Context, com *Commit) os.Error {
|
|||||||
Ancestor(p.Key(c)).
|
Ancestor(p.Key(c)).
|
||||||
Count(c)
|
Count(c)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return fmt.Errorf("testing for parent Commit: %v", err)
|
||||||
}
|
}
|
||||||
if n == 0 {
|
if n == 0 {
|
||||||
return os.NewError("parent commit not found")
|
return os.NewError("parent commit not found")
|
||||||
@ -325,12 +341,14 @@ func addCommit(c appengine.Context, com *Commit) os.Error {
|
|||||||
if p.Path == "" {
|
if p.Path == "" {
|
||||||
t := &Tag{Kind: "tip", Hash: com.Hash}
|
t := &Tag{Kind: "tip", Hash: com.Hash}
|
||||||
if _, err = datastore.Put(c, t.Key(c), t); err != nil {
|
if _, err = datastore.Put(c, t.Key(c), t); err != nil {
|
||||||
return err
|
return fmt.Errorf("putting Tag: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// put the Commit
|
// put the Commit
|
||||||
_, err = datastore.Put(c, com.Key(c), com)
|
if _, err = datastore.Put(c, com.Key(c), com); err != nil {
|
||||||
return err
|
return fmt.Errorf("putting Commit: %v", err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// tagHandler records a new tag. It reads a JSON-encoded Tag value from the
|
// tagHandler records a new tag. It reads a JSON-encoded Tag value from the
|
||||||
@ -458,31 +476,34 @@ func resultHandler(r *http.Request) (interface{}, os.Error) {
|
|||||||
res := new(Result)
|
res := new(Result)
|
||||||
defer r.Body.Close()
|
defer r.Body.Close()
|
||||||
if err := json.NewDecoder(r.Body).Decode(res); err != nil {
|
if err := json.NewDecoder(r.Body).Decode(res); err != nil {
|
||||||
return nil, err
|
return nil, fmt.Errorf("decoding Body: %v", err)
|
||||||
}
|
}
|
||||||
if err := res.Valid(); err != nil {
|
if err := res.Valid(); err != nil {
|
||||||
return nil, err
|
return nil, fmt.Errorf("validating Result: %v", err)
|
||||||
}
|
}
|
||||||
// store the Log text if supplied
|
// store the Log text if supplied
|
||||||
if len(res.Log) > 0 {
|
if len(res.Log) > 0 {
|
||||||
hash, err := PutLog(c, res.Log)
|
hash, err := PutLog(c, res.Log)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, fmt.Errorf("putting Log: %v", err)
|
||||||
}
|
}
|
||||||
res.LogHash = hash
|
res.LogHash = hash
|
||||||
}
|
}
|
||||||
tx := func(c appengine.Context) os.Error {
|
tx := func(c appengine.Context) os.Error {
|
||||||
// check Package exists
|
// check Package exists
|
||||||
if _, err := GetPackage(c, res.PackagePath); err != nil {
|
if _, err := GetPackage(c, res.PackagePath); err != nil {
|
||||||
return err
|
return fmt.Errorf("GetPackage: %v", err)
|
||||||
}
|
}
|
||||||
// put Result
|
// put Result
|
||||||
if _, err := datastore.Put(c, res.Key(c), res); err != nil {
|
if _, err := datastore.Put(c, res.Key(c), res); err != nil {
|
||||||
return err
|
return fmt.Errorf("putting Result: %v", err)
|
||||||
}
|
}
|
||||||
// add Result to Commit
|
// add Result to Commit
|
||||||
com := &Commit{PackagePath: res.PackagePath, Hash: res.Hash}
|
com := &Commit{PackagePath: res.PackagePath, Hash: res.Hash}
|
||||||
return com.AddResult(c, res)
|
if err := com.AddResult(c, res); err != nil {
|
||||||
|
return fmt.Errorf("AddResult: %v", err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
return nil, datastore.RunInTransaction(c, tx, nil)
|
return nil, datastore.RunInTransaction(c, tx, nil)
|
||||||
}
|
}
|
||||||
@ -527,47 +548,54 @@ func (e errBadMethod) String() string {
|
|||||||
// supplied key and builder query parameters.
|
// supplied key and builder query parameters.
|
||||||
func AuthHandler(h dashHandler) http.HandlerFunc {
|
func AuthHandler(h dashHandler) http.HandlerFunc {
|
||||||
return func(w http.ResponseWriter, r *http.Request) {
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
c := appengine.NewContext(r)
|
||||||
|
|
||||||
// Put the URL Query values into r.Form to avoid parsing the
|
// Put the URL Query values into r.Form to avoid parsing the
|
||||||
// request body when calling r.FormValue.
|
// request body when calling r.FormValue.
|
||||||
r.Form = r.URL.Query()
|
r.Form = r.URL.Query()
|
||||||
|
|
||||||
|
var err os.Error
|
||||||
|
var resp interface{}
|
||||||
|
|
||||||
// Validate key query parameter for POST requests only.
|
// Validate key query parameter for POST requests only.
|
||||||
key := r.FormValue("key")
|
key := r.FormValue("key")
|
||||||
if r.Method == "POST" && key != secretKey &&
|
if r.Method == "POST" && key != secretKey && !appengine.IsDevAppServer() {
|
||||||
!appengine.IsDevAppServer() {
|
h := hmac.NewMD5([]byte(secretKey))
|
||||||
h := sha1.New()
|
h.Write([]byte(r.FormValue("builder")))
|
||||||
h.Write([]byte(r.FormValue("builder") + secretKey))
|
|
||||||
if key != fmt.Sprintf("%x", h.Sum()) {
|
if key != fmt.Sprintf("%x", h.Sum()) {
|
||||||
logErr(w, r, os.NewError("invalid key"))
|
err = os.NewError("invalid key: " + key)
|
||||||
return
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Call the original HandlerFunc and return the response.
|
// Call the original HandlerFunc and return the response.
|
||||||
c := appengine.NewContext(r)
|
if err == nil {
|
||||||
resp, err := h(r)
|
resp, err = h(r)
|
||||||
dashResp := dashResponse{Response: resp}
|
}
|
||||||
|
|
||||||
|
// Write JSON response.
|
||||||
|
dashResp := &dashResponse{Response: resp}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.Errorf("%v", err)
|
c.Errorf("%v", err)
|
||||||
dashResp.Error = err.String()
|
dashResp.Error = err.String()
|
||||||
}
|
}
|
||||||
w.Header().Set("Content-Type", "application/json")
|
w.Header().Set("Content-Type", "application/json")
|
||||||
if err = json.NewEncoder(w).Encode(dashResp); err != nil {
|
if err = json.NewEncoder(w).Encode(dashResp); err != nil {
|
||||||
c.Criticalf("%v", err)
|
c.Criticalf("encoding response: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func initHandler(w http.ResponseWriter, r *http.Request) {
|
func initHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
// TODO(adg): devise a better way of bootstrapping new packages
|
// TODO(adg): devise a better way of bootstrapping new packages
|
||||||
var pkgs = []*Package{
|
|
||||||
&Package{Name: "Go"},
|
|
||||||
&Package{Name: "Test", Path: "code.google.com/p/go.test"},
|
|
||||||
}
|
|
||||||
c := appengine.NewContext(r)
|
c := appengine.NewContext(r)
|
||||||
for _, p := range pkgs {
|
for _, p := range defaultPackages {
|
||||||
_, err := datastore.Put(c, p.Key(c), p)
|
if err := datastore.Get(c, p.Key(c), new(Package)); err == nil {
|
||||||
if err != nil {
|
continue
|
||||||
|
} else if err != datastore.ErrNoSuchEntity {
|
||||||
|
logErr(w, r, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if _, err := datastore.Put(c, p.Key(c), p); err != nil {
|
||||||
logErr(w, r, err)
|
logErr(w, r, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -31,13 +31,13 @@
|
|||||||
}
|
}
|
||||||
.build .hash {
|
.build .hash {
|
||||||
font-family: monospace;
|
font-family: monospace;
|
||||||
|
font-size: 9pt;
|
||||||
}
|
}
|
||||||
.build .result {
|
.build .result {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
width: 50px;
|
width: 50px;
|
||||||
}
|
}
|
||||||
.build .time {
|
.build .time {
|
||||||
font-family: monospace;
|
|
||||||
color: #666;
|
color: #666;
|
||||||
}
|
}
|
||||||
.build .descr, .build .time, .build .user {
|
.build .descr, .build .time, .build .user {
|
||||||
@ -63,8 +63,6 @@
|
|||||||
|
|
||||||
<h1>Go Build Status</h1>
|
<h1>Go Build Status</h1>
|
||||||
|
|
||||||
<h2>Go</h2>
|
|
||||||
|
|
||||||
{{if $.Commits}}
|
{{if $.Commits}}
|
||||||
|
|
||||||
<table class="build">
|
<table class="build">
|
||||||
@ -91,7 +89,7 @@
|
|||||||
</td>
|
</td>
|
||||||
{{end}}
|
{{end}}
|
||||||
<td class="user">{{shortUser .User}}</td>
|
<td class="user">{{shortUser .User}}</td>
|
||||||
<td class="time">{{.Time.Time.Format "02 Jan 2006 15:04"}}</td>
|
<td class="time">{{.Time.Time.Format "Mon 02 Jan 15:04"}}</td>
|
||||||
<td class="desc">{{shortDesc .Desc}}</td>
|
<td class="desc">{{shortDesc .Desc}}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{{end}}
|
{{end}}
|
||||||
@ -109,6 +107,7 @@
|
|||||||
<p>No commits to display. Hm.</p>
|
<p>No commits to display. Hm.</p>
|
||||||
{{end}}
|
{{end}}
|
||||||
|
|
||||||
|
{{if $.TipState}}
|
||||||
<h2>Other packages</h2>
|
<h2>Other packages</h2>
|
||||||
|
|
||||||
<table class="packages">
|
<table class="packages">
|
||||||
@ -144,6 +143,7 @@
|
|||||||
</tr>
|
</tr>
|
||||||
{{end}}
|
{{end}}
|
||||||
</table>
|
</table>
|
||||||
|
{{end}}
|
||||||
|
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
Loading…
Reference in New Issue
Block a user