2015-04-08 15:43:59 -07:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
|
2015-05-17 11:10:43 -07:00
|
|
|
"github.com/drone/drone/pkg/bus"
|
2015-05-17 11:42:56 -07:00
|
|
|
"github.com/drone/drone/pkg/queue"
|
2015-05-17 11:45:09 -07:00
|
|
|
"github.com/drone/drone/pkg/remote"
|
2015-05-17 14:25:04 -07:00
|
|
|
"github.com/drone/drone/pkg/runner"
|
|
|
|
"github.com/drone/drone/pkg/server/session"
|
|
|
|
"github.com/drone/drone/pkg/settings"
|
2015-05-17 11:10:43 -07:00
|
|
|
"github.com/drone/drone/pkg/store"
|
2015-05-17 13:51:42 -07:00
|
|
|
common "github.com/drone/drone/pkg/types"
|
2015-04-08 15:43:59 -07:00
|
|
|
)
|
|
|
|
|
2015-04-22 01:00:15 -07:00
|
|
|
func SetQueue(q queue.Queue) gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
c.Set("queue", q)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func ToQueue(c *gin.Context) queue.Queue {
|
2015-05-16 19:46:12 -07:00
|
|
|
v, ok := c.Get("queue")
|
|
|
|
if !ok {
|
2015-04-22 01:00:15 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return v.(queue.Queue)
|
|
|
|
}
|
|
|
|
|
2015-05-17 11:10:43 -07:00
|
|
|
func SetBus(r bus.Bus) gin.HandlerFunc {
|
2015-04-08 15:43:59 -07:00
|
|
|
return func(c *gin.Context) {
|
2015-05-17 11:10:43 -07:00
|
|
|
c.Set("bus", r)
|
2015-04-08 15:43:59 -07:00
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-17 11:10:43 -07:00
|
|
|
func ToBus(c *gin.Context) bus.Bus {
|
|
|
|
v, ok := c.Get("bus")
|
2015-05-16 19:46:12 -07:00
|
|
|
if !ok {
|
2015-04-08 15:43:59 -07:00
|
|
|
return nil
|
|
|
|
}
|
2015-05-17 11:10:43 -07:00
|
|
|
return v.(bus.Bus)
|
2015-04-08 15:43:59 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func ToRemote(c *gin.Context) remote.Remote {
|
2015-05-16 19:46:12 -07:00
|
|
|
v, ok := c.Get("remote")
|
|
|
|
if !ok {
|
2015-04-08 15:43:59 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return v.(remote.Remote)
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetRemote(r remote.Remote) gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
c.Set("remote", r)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-06 00:56:06 -07:00
|
|
|
func ToRunner(c *gin.Context) runner.Runner {
|
2015-05-16 19:46:12 -07:00
|
|
|
v, ok := c.Get("runner")
|
|
|
|
if !ok {
|
2015-05-06 00:56:06 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return v.(runner.Runner)
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetRunner(r runner.Runner) gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
c.Set("runner", r)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-08 15:43:59 -07:00
|
|
|
func ToSettings(c *gin.Context) *settings.Settings {
|
2015-05-16 19:46:12 -07:00
|
|
|
v, ok := c.Get("settings")
|
|
|
|
if !ok {
|
2015-04-08 15:43:59 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return v.(*settings.Settings)
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetSettings(s *settings.Settings) gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
c.Set("settings", s)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func ToPerm(c *gin.Context) *common.Perm {
|
2015-05-16 19:46:12 -07:00
|
|
|
v, ok := c.Get("perm")
|
|
|
|
if !ok {
|
2015-04-08 15:43:59 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return v.(*common.Perm)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ToUser(c *gin.Context) *common.User {
|
2015-05-16 19:46:12 -07:00
|
|
|
v, ok := c.Get("user")
|
|
|
|
if !ok {
|
2015-04-08 15:43:59 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return v.(*common.User)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ToRepo(c *gin.Context) *common.Repo {
|
2015-05-16 19:46:12 -07:00
|
|
|
v, ok := c.Get("repo")
|
|
|
|
if !ok {
|
2015-04-08 15:43:59 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return v.(*common.Repo)
|
|
|
|
}
|
|
|
|
|
2015-05-17 11:10:43 -07:00
|
|
|
func ToDatastore(c *gin.Context) store.Store {
|
|
|
|
return c.MustGet("datastore").(store.Store)
|
2015-04-08 15:43:59 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func ToSession(c *gin.Context) session.Session {
|
|
|
|
return c.MustGet("session").(session.Session)
|
|
|
|
}
|
|
|
|
|
2015-05-17 11:10:43 -07:00
|
|
|
func SetDatastore(ds store.Store) gin.HandlerFunc {
|
2015-04-08 15:43:59 -07:00
|
|
|
return func(c *gin.Context) {
|
|
|
|
c.Set("datastore", ds)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetSession(s session.Session) gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
c.Set("session", s)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetUser(s session.Session) gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
ds := ToDatastore(c)
|
2015-04-12 21:35:16 -07:00
|
|
|
token := s.GetLogin(c.Request)
|
2015-04-30 14:23:46 -07:00
|
|
|
if token == nil || len(token.Login) == 0 {
|
2015-04-08 15:43:59 -07:00
|
|
|
c.Next()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-11 00:45:31 -07:00
|
|
|
user, err := ds.UserLogin(token.Login)
|
2015-04-08 15:43:59 -07:00
|
|
|
if err == nil {
|
2015-05-11 00:45:31 -07:00
|
|
|
c.Set("user", user)
|
2015-04-08 15:43:59 -07:00
|
|
|
}
|
2015-04-12 22:32:32 -07:00
|
|
|
|
|
|
|
// if session token we can proceed, otherwise
|
|
|
|
// we should validate the token hasn't been revoked
|
2015-04-30 14:23:46 -07:00
|
|
|
switch token.Kind {
|
|
|
|
case common.TokenSess:
|
2015-04-12 22:32:32 -07:00
|
|
|
c.Next()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// to verify the token we fetch from the datastore
|
|
|
|
// and check to see if the token issued date matches
|
|
|
|
// what we found in the jwt (in case the label is re-used)
|
2015-05-11 00:45:31 -07:00
|
|
|
t, err := ds.TokenLabel(user, token.Label)
|
2015-04-12 22:32:32 -07:00
|
|
|
if err != nil || t.Issued != token.Issued {
|
|
|
|
c.AbortWithStatus(403)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Next()
|
2015-04-08 15:43:59 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetRepo() gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
ds := ToDatastore(c)
|
|
|
|
u := ToUser(c)
|
|
|
|
owner := c.Params.ByName("owner")
|
|
|
|
name := c.Params.ByName("name")
|
2015-05-11 00:45:31 -07:00
|
|
|
r, err := ds.RepoName(owner, name)
|
2015-04-08 15:43:59 -07:00
|
|
|
switch {
|
|
|
|
case err != nil && u != nil:
|
2015-04-25 16:43:51 -07:00
|
|
|
c.Fail(404, err)
|
2015-04-08 15:43:59 -07:00
|
|
|
return
|
|
|
|
case err != nil && u == nil:
|
2015-04-25 16:43:51 -07:00
|
|
|
c.Fail(401, err)
|
2015-04-08 15:43:59 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
c.Set("repo", r)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetPerm() gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
remote := ToRemote(c)
|
|
|
|
user := ToUser(c)
|
|
|
|
repo := ToRepo(c)
|
|
|
|
perm := perms(remote, user, repo)
|
|
|
|
c.Set("perm", perm)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func MustUser() gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
u := ToUser(c)
|
|
|
|
if u == nil {
|
|
|
|
c.AbortWithStatus(401)
|
|
|
|
} else {
|
|
|
|
c.Set("user", u)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func MustAdmin() gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
u := ToUser(c)
|
|
|
|
if u == nil {
|
|
|
|
c.AbortWithStatus(401)
|
|
|
|
} else if !u.Admin {
|
|
|
|
c.AbortWithStatus(403)
|
|
|
|
} else {
|
|
|
|
c.Set("user", u)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-30 14:57:53 -07:00
|
|
|
func MustAgent() gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
sess := ToSession(c)
|
|
|
|
token := sess.GetLogin(c.Request)
|
|
|
|
if token == nil {
|
|
|
|
c.AbortWithStatus(401)
|
|
|
|
return
|
|
|
|
} else if token.Kind != common.TokenAgent {
|
|
|
|
c.AbortWithStatus(500)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-08 15:43:59 -07:00
|
|
|
func CheckPull() gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
u := ToUser(c)
|
|
|
|
m := ToPerm(c)
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case u == nil && m == nil:
|
|
|
|
c.AbortWithStatus(401)
|
|
|
|
case u == nil && m.Pull == false:
|
|
|
|
c.AbortWithStatus(401)
|
|
|
|
case u != nil && m.Pull == false:
|
|
|
|
c.AbortWithStatus(404)
|
|
|
|
default:
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func CheckPush() gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
switch c.Request.Method {
|
|
|
|
case "GET", "OPTIONS":
|
|
|
|
c.Next()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
u := ToUser(c)
|
|
|
|
m := ToPerm(c)
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case u == nil && m.Push == false:
|
|
|
|
c.AbortWithStatus(401)
|
|
|
|
case u != nil && m.Push == false:
|
|
|
|
c.AbortWithStatus(404)
|
|
|
|
default:
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetHeaders() gin.HandlerFunc {
|
|
|
|
return func(c *gin.Context) {
|
|
|
|
|
|
|
|
c.Writer.Header().Add("Access-Control-Allow-Origin", "*")
|
|
|
|
c.Writer.Header().Add("X-Frame-Options", "DENY")
|
|
|
|
c.Writer.Header().Add("X-Content-Type-Options", "nosniff")
|
|
|
|
c.Writer.Header().Add("X-XSS-Protection", "1; mode=block")
|
|
|
|
c.Writer.Header().Add("Cache-Control", "no-cache")
|
|
|
|
c.Writer.Header().Add("Cache-Control", "no-store")
|
|
|
|
c.Writer.Header().Add("Cache-Control", "max-age=0")
|
|
|
|
c.Writer.Header().Add("Cache-Control", "must-revalidate")
|
|
|
|
c.Writer.Header().Add("Cache-Control", "value")
|
|
|
|
c.Writer.Header().Set("Last-Modified", time.Now().UTC().Format(http.TimeFormat))
|
|
|
|
c.Writer.Header().Set("Expires", "Thu, 01 Jan 1970 00:00:00 GMT")
|
|
|
|
if c.Request.TLS != nil {
|
|
|
|
c.Writer.Header().Add("Strict-Transport-Security", "max-age=31536000")
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|