woodpecker/server/ws.go

205 lines
3.9 KiB
Go
Raw Normal View History

2015-04-08 22:43:59 +00:00
package server
import (
2015-05-06 07:56:06 +00:00
"bufio"
"net/http"
2015-04-30 02:57:43 +00:00
"strconv"
2015-04-08 22:43:59 +00:00
"time"
"github.com/drone/drone/eventbus"
2015-04-25 00:06:46 +00:00
log "github.com/Sirupsen/logrus"
2015-04-08 22:43:59 +00:00
"github.com/gin-gonic/gin"
"github.com/gorilla/websocket"
2015-05-06 07:56:06 +00:00
// "github.com/koding/websocketproxy"
2015-04-08 22:43:59 +00:00
)
const (
// Time allowed to write the message to the client.
writeWait = 10 * time.Second
// Time allowed to read the next pong message from the client.
pongWait = 60 * time.Second
// Send pings to client with this period. Must be less than pongWait.
pingPeriod = (pongWait * 9) / 10
)
var upgrader = websocket.Upgrader{
ReadBufferSize: 1024,
WriteBufferSize: 1024,
CheckOrigin: func(r *http.Request) bool { return true },
2015-04-08 22:43:59 +00:00
}
// GetRepoEvents will upgrade the connection to a Websocket and will stream
// event updates to the browser.
func GetRepoEvents(c *gin.Context) {
bus := ToBus(c)
repo := ToRepo(c)
2015-04-25 00:06:46 +00:00
2015-04-08 22:43:59 +00:00
// upgrade the websocket
ws, err := upgrader.Upgrade(c.Writer, c.Request, nil)
if err != nil {
c.Fail(400, err)
return
}
ticker := time.NewTicker(pingPeriod)
eventc := make(chan *eventbus.Event)
2015-04-08 22:43:59 +00:00
bus.Subscribe(eventc)
defer func() {
bus.Unsubscribe(eventc)
ticker.Stop()
ws.Close()
close(eventc)
log.Infof("closed websocket")
2015-04-08 22:43:59 +00:00
}()
go func() {
for {
select {
2015-05-06 03:22:02 +00:00
case <-c.Writer.CloseNotify():
ws.Close()
2015-05-06 03:22:02 +00:00
return
2015-04-08 22:43:59 +00:00
case event := <-eventc:
if event == nil {
log.Infof("closed websocket")
ws.Close()
return
}
if event.Kind == eventbus.EventRepo && event.Name == repo.FullName {
ws.WriteMessage(websocket.TextMessage, event.Msg)
2015-05-06 03:22:02 +00:00
break
2015-04-08 22:43:59 +00:00
}
case <-ticker.C:
ws.SetWriteDeadline(time.Now().Add(writeWait))
err := ws.WriteMessage(websocket.PingMessage, []byte{})
if err != nil {
log.Infof("closed websocket")
2015-04-08 22:43:59 +00:00
ws.Close()
return
}
}
}
}()
readWebsocket(ws)
}
2015-04-30 02:57:43 +00:00
func GetStream(c *gin.Context) {
store := ToDatastore(c)
2015-04-30 02:57:43 +00:00
repo := ToRepo(c)
2015-05-06 07:56:06 +00:00
runner := ToRunner(c)
commitseq, _ := strconv.Atoi(c.Params.ByName("build"))
buildseq, _ := strconv.Atoi(c.Params.ByName("number"))
2015-04-30 02:57:43 +00:00
2015-05-06 07:56:06 +00:00
// agent, err := store.BuildAgent(repo.FullName, build)
// if err != nil {
// c.Fail(404, err)
// return
// }
commit, err := store.CommitSeq(repo, commitseq)
if err != nil {
c.Fail(404, err)
return
}
build, err := store.BuildSeq(commit, buildseq)
if err != nil {
c.Fail(404, err)
return
}
rc, err := runner.Logs(build)
2015-04-30 02:57:43 +00:00
if err != nil {
c.Fail(404, err)
return
}
2015-05-06 07:56:06 +00:00
// upgrade the websocket
ws, err := upgrader.Upgrade(c.Writer, c.Request, nil)
2015-04-30 02:57:43 +00:00
if err != nil {
2015-05-06 07:56:06 +00:00
c.Fail(400, err)
2015-04-30 02:57:43 +00:00
return
}
2015-05-06 07:56:06 +00:00
var ticker = time.NewTicker(pingPeriod)
var out = make(chan []byte)
defer func() {
log.Infof("closed stdout websocket")
ticker.Stop()
rc.Close()
ws.Close()
}()
go func() {
for {
select {
case <-c.Writer.CloseNotify():
rc.Close()
ws.Close()
return
case line := <-out:
ws.WriteMessage(websocket.TextMessage, line)
case <-ticker.C:
ws.SetWriteDeadline(time.Now().Add(writeWait))
err := ws.WriteMessage(websocket.PingMessage, []byte{})
if err != nil {
rc.Close()
ws.Close()
return
}
}
}
}()
go func() {
rd := bufio.NewReader(rc)
for {
str, err := rd.ReadBytes('\n')
if err != nil {
break
}
if len(str) == 0 {
break
}
out <- str
}
rc.Close()
ws.Close()
}()
readWebsocket(ws)
// url_, err := url.Parse("ws://" + agent.Addr)
// if err != nil {
// c.Fail(500, err)
// return
// }
// url_.Path = fmt.Sprintf("/stream/%s/%v/%v", repo.FullName, build, task)
// proxy := websocketproxy.NewProxy(url_)
// proxy.ServeHTTP(c.Writer, c.Request)
// log.Debugf("closed websocket")
2015-04-30 02:57:43 +00:00
}
2015-04-08 22:43:59 +00:00
// readWebsocket will block while reading the websocket data
func readWebsocket(ws *websocket.Conn) {
defer ws.Close()
ws.SetReadLimit(512)
ws.SetReadDeadline(time.Now().Add(pongWait))
ws.SetPongHandler(func(string) error {
ws.SetReadDeadline(time.Now().Add(pongWait))
return nil
})
for {
_, _, err := ws.ReadMessage()
if err != nil {
break
}
}
}