minimum viable (excluding auth)

This commit is contained in:
simon987 2019-03-09 09:20:51 -05:00
parent 5a7f3316e6
commit 6048cfbebc
11 changed files with 812 additions and 0 deletions

2
.gitignore vendored
View File

@ -10,3 +10,5 @@
# Output of the go coverage tool, specifically when used with LiteIDE
*.out
.idea/

136
api/api.go Normal file
View File

@ -0,0 +1,136 @@
package api
import (
"encoding/json"
"github.com/buaazp/fasthttprouter"
"github.com/fasthttp/websocket"
"github.com/jinzhu/gorm"
_ "github.com/jinzhu/gorm/dialects/postgres"
"github.com/sirupsen/logrus"
"github.com/valyala/fasthttp"
"os"
"path/filepath"
)
var WorkDir, _ = filepath.Abs("./data/")
type Info struct {
Name string `json:"name"`
Version string `json:"version"`
}
var info = Info{
Name: "ws_bucket",
Version: "1.0",
}
var motd = WebsocketMotd{
Info: info,
Motd: "Hello, world",
}
type WebApi struct {
server fasthttp.Server
db *gorm.DB
MotdMessage *websocket.PreparedMessage
}
func Index(ctx *fasthttp.RequestCtx) {
Json(info, ctx)
}
func Json(object interface{}, ctx *fasthttp.RequestCtx) {
resp, err := json.Marshal(object)
if err != nil {
panic(err)
}
ctx.Response.Header.Set("Content-Type", "application/json")
_, err = ctx.Write(resp)
if err != nil {
panic(err)
}
}
func LogRequestMiddleware(h fasthttp.RequestHandler) fasthttp.RequestHandler {
return fasthttp.RequestHandler(func(ctx *fasthttp.RequestCtx) {
logrus.WithFields(logrus.Fields{
"path": string(ctx.Path()),
"header": ctx.Request.Header.String(),
}).Trace(string(ctx.Method()))
h(ctx)
})
}
func New(db *gorm.DB) *WebApi {
api := &WebApi{}
logrus.SetLevel(getLogLevel())
router := fasthttprouter.New()
router.GET("/", LogRequestMiddleware(Index))
router.POST("/client", LogRequestMiddleware(api.CreateClient))
router.POST("/slot", LogRequestMiddleware(api.AllocateUploadSlot))
router.GET("/slot", LogRequestMiddleware(api.ReadUploadSlot))
router.GET("/upload", LogRequestMiddleware(api.Upload))
api.server = fasthttp.Server{
Handler: router.Handler,
Name: "ws_bucket",
}
api.db = db
db.AutoMigrate(&Client{})
db.AutoMigrate(&UploadSlot{})
api.setupMotd()
return api
}
func (api *WebApi) setupMotd() {
var data []byte
data, _ = json.Marshal(motd)
motdMsg, _ := websocket.NewPreparedMessage(websocket.TextMessage, data)
api.MotdMessage = motdMsg
}
func (api *WebApi) Run() {
address := GetServerAddress()
logrus.WithFields(logrus.Fields{
"addr": address,
}).Info("Starting web server")
err := api.server.ListenAndServe(address)
if err != nil {
logrus.Fatalf("Error in ListenAndServe: %s", err)
}
}
func GetServerAddress() string {
serverAddress := os.Getenv("WS_BUCKET_ADDR")
if serverAddress == "" {
serverAddress = "0.0.0.0:3020"
}
return serverAddress
}
func getLogLevel() logrus.Level {
levelStr := os.Getenv("WS_BUCKET_LOGLEVEL")
if levelStr == "" {
return logrus.TraceLevel
} else {
level, err := logrus.ParseLevel(levelStr)
if err != nil {
panic(err)
}
return level
}
}

62
api/auth.go Normal file
View File

@ -0,0 +1,62 @@
package api
import (
"encoding/json"
"github.com/sirupsen/logrus"
"github.com/valyala/fasthttp"
"math/rand"
)
func (api *WebApi) CreateClient(ctx *fasthttp.RequestCtx) {
//TODO: auth
req := &CreateClientRequest{}
err := json.Unmarshal(ctx.Request.Body(), req)
if err != nil {
ctx.Response.SetStatusCode(400)
Json(CreateClientResponse{
Ok: false,
}, ctx)
return
}
if !req.IsValid() {
ctx.Response.SetStatusCode(400)
Json(CreateClientResponse{
Ok: false,
}, ctx)
return
}
client := api.createClient(req)
Json(CreateClientResponse{
Ok: true,
Secret: client.Secret,
}, ctx)
}
func (api *WebApi) createClient(req *CreateClientRequest) *Client {
client := &Client{
Alias: req.Alias,
Secret: genSecret(),
}
api.db.Create(client)
logrus.WithFields(logrus.Fields{
"client": client,
}).Info("Created client")
return client
}
func genSecret() string {
bytes := make([]byte, 32)
for i := 0; i < 32; i++ {
bytes[i] = byte(48 + rand.Intn(122-48))
}
return string(bytes)
}

71
api/models.go Normal file
View File

@ -0,0 +1,71 @@
package api
import (
"path/filepath"
"strings"
)
type GenericResponse struct {
Ok bool `json:"ok"`
}
type CreateClientRequest struct {
Alias string `json:"alias"`
}
func (req *CreateClientRequest) IsValid() bool {
return len(req.Alias) > 3
}
type CreateClientResponse struct {
Ok bool `json:"ok"`
Secret string `json:"secret,omitempty"`
}
type Client struct {
ID int64
Alias string `json:"alias"`
Secret string `json:"secret"`
}
type AllocateUploadSlotRequest struct {
Token string `json:"token"`
MaxSize int64 `json:"max_size"`
FileName string `json:"file_name"`
}
func (req *AllocateUploadSlotRequest) IsValid() bool {
if len(req.Token) < 3 {
return false
}
if len(req.FileName) <= 0 {
return false
}
path := filepath.Join(WorkDir, req.FileName)
pathAbs, err := filepath.Abs(path)
if err != nil {
return false
}
if !strings.HasPrefix(pathAbs, WorkDir) {
return false
}
if req.MaxSize < 0 {
return false
}
return true
}
type UploadSlot struct {
MaxSize int64 `json:"max_size"`
Token string `gorm:"primary_key",json:"token"`
FileName string `json:"file_name"`
}
type WebsocketMotd struct {
Info Info `json:"info"`
Motd string `json:"motd"`
}

180
api/slot.go Normal file
View File

@ -0,0 +1,180 @@
package api
import (
"encoding/json"
"github.com/fasthttp/websocket"
"github.com/sirupsen/logrus"
"github.com/valyala/fasthttp"
"io"
"os"
"path/filepath"
"sync"
)
const WsBufferSize = 4096
var Mutexes sync.Map
var upgrader = websocket.FastHTTPUpgrader{
ReadBufferSize: WsBufferSize,
WriteBufferSize: WsBufferSize,
EnableCompression: true,
}
func (api *WebApi) AllocateUploadSlot(ctx *fasthttp.RequestCtx) {
req := &AllocateUploadSlotRequest{}
err := json.Unmarshal(ctx.Request.Body(), req)
if err != nil {
ctx.Response.SetStatusCode(400)
Json(GenericResponse{
Ok: false,
}, ctx)
return
}
if !req.IsValid() {
ctx.Response.SetStatusCode(400)
Json(CreateClientResponse{
Ok: false,
}, ctx)
return
}
api.allocateUploadSlot(req)
Json(CreateClientResponse{
Ok: true,
}, ctx)
}
func (api *WebApi) Upload(ctx *fasthttp.RequestCtx) {
token := string(ctx.Request.Header.Peek("X-Upload-Token"))
slot := UploadSlot{}
err := api.db.Where("token=?", token).First(&slot).Error
if err != nil {
ctx.Response.Header.SetStatusCode(400)
logrus.WithFields(logrus.Fields{
"token": token,
}).Warning("Upload slot not found")
return
}
logrus.WithFields(logrus.Fields{
"slot": slot,
}).Info("Upgrading connection")
err = upgrader.Upgrade(ctx, func(ws *websocket.Conn) {
defer ws.Close()
err := ws.WritePreparedMessage(api.MotdMessage)
if err != nil {
panic(err)
}
mt, reader, err := ws.NextReader()
if err != nil {
panic(err)
}
if mt != websocket.BinaryMessage {
return
}
mu, _ := Mutexes.LoadOrStore(slot.Token, &sync.RWMutex{})
mu.(*sync.RWMutex).Lock()
path := filepath.Join(WorkDir, slot.FileName)
fp, err := os.OpenFile(path, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0600)
if err != nil {
panic(err)
}
buf := make([]byte, WsBufferSize)
totalRead := int64(0)
for totalRead < slot.MaxSize {
read, err := reader.Read(buf)
var toWrite int
if totalRead+int64(read) > slot.MaxSize {
toWrite = int(slot.MaxSize - totalRead)
} else {
toWrite = read
}
_, _ = fp.Write(buf[:toWrite])
if err == io.EOF {
break
}
totalRead += int64(read)
}
logrus.WithFields(logrus.Fields{
"totalRead": totalRead,
}).Info("Finished reading")
err = fp.Close()
if err != nil {
panic(err)
}
mu.(*sync.RWMutex).Unlock()
})
if err != nil {
panic(err)
}
}
func (api *WebApi) ReadUploadSlot(ctx *fasthttp.RequestCtx) {
tokenStr := string(ctx.Request.Header.Peek("X-Upload-Token"))
slot := UploadSlot{}
err := api.db.Where("token=?", tokenStr).First(&slot).Error
if err != nil {
ctx.Response.Header.SetStatusCode(404)
logrus.WithFields(logrus.Fields{
"token": tokenStr,
}).Warning("Upload slot not found")
return
}
logrus.WithFields(logrus.Fields{
"slot": slot,
}).Info("Reading")
path := filepath.Join(WorkDir, slot.FileName)
mu, _ := Mutexes.LoadOrStore(slot.Token, &sync.RWMutex{})
mu.(*sync.RWMutex).RLock()
fp, err := os.OpenFile(path, os.O_RDONLY, 0600)
if err != nil {
panic(err)
}
buf := make([]byte, WsBufferSize)
response := ctx.Response.BodyWriter()
for {
read, err := fp.Read(buf)
_, _ = response.Write(buf[:read])
if err == io.EOF {
break
}
if err != nil {
panic(err)
}
}
mu.(*sync.RWMutex).RUnlock()
}
func (api *WebApi) allocateUploadSlot(req *AllocateUploadSlotRequest) {
slot := &UploadSlot{
MaxSize: req.MaxSize,
FileName: req.FileName,
Token: req.Token,
}
logrus.WithFields(logrus.Fields{
"slot": slot,
}).Info("Allocated new upload slot")
api.db.Create(slot)
}

17
main.go Normal file
View File

@ -0,0 +1,17 @@
package main
import (
"github.com/jinzhu/gorm"
"github.com/simon987/ws_bucket/api"
)
func main() {
db, err := gorm.Open("postgres", "host=localhost user=ws_bucket dbname=ws_bucket password=ws_bucket sslmode=disable")
if err != nil {
panic(err)
}
a := api.New(db)
a.Run()
}

24
test/auth_test.go Normal file
View File

@ -0,0 +1,24 @@
package test
import (
"github.com/simon987/ws_bucket/api"
"testing"
)
func TestCreateClient(t *testing.T) {
r := createClient(api.CreateClientRequest{
Alias: "testcreateclient",
})
if r.Ok != true {
t.Error()
}
}
func createClient(request api.CreateClientRequest) (ar *api.CreateClientResponse) {
resp := Post("/client", request)
UnmarshalResponse(resp, &ar)
return
}

63
test/common.go Normal file
View File

@ -0,0 +1,63 @@
package test
import (
"bytes"
"encoding/json"
"fmt"
"github.com/simon987/ws_bucket/api"
"io/ioutil"
"net/http"
)
func Post(path string, x interface{}) *http.Response {
s := http.Client{}
body, err := json.Marshal(x)
buf := bytes.NewBuffer(body)
req, err := http.NewRequest("POST", "http://"+api.GetServerAddress()+path, buf)
handleErr(err)
//ts := time.Now().Format(time.RFC1123)
//
//mac := hmac.New(crypto.SHA256.New, worker.Secret)
//mac.Write(body)
//mac.Write([]byte(ts))
//sig := hex.EncodeToString(mac.Sum(nil))
//
//req.Header.Add("X-Worker-Id", strconv.FormatInt(worker.Id, 10))
//req.Header.Add("X-Signature", sig)
//req.Header.Add("Timestamp", ts)
r, err := s.Do(req)
handleErr(err)
return r
}
func Get(path string, token string) *http.Response {
s := http.Client{}
req, err := http.NewRequest("GET", "http://"+api.GetServerAddress()+path, nil)
handleErr(err)
req.Header.Set("X-Upload-Token", token)
r, err := s.Do(req)
return r
}
func UnmarshalResponse(r *http.Response, result interface{}) {
data, err := ioutil.ReadAll(r.Body)
fmt.Println(string(data))
err = json.Unmarshal(data, result)
handleErr(err)
}
func handleErr(err error) {
if err != nil {
panic(err)
}
}

25
test/main_test.go Normal file
View File

@ -0,0 +1,25 @@
package test
import (
"github.com/jinzhu/gorm"
_ "github.com/jinzhu/gorm/dialects/sqlite"
"github.com/simon987/ws_bucket/api"
"testing"
"time"
)
func TestMain(m *testing.M) {
//db, err := gorm.Open("postgres", "host=localhost user=ws_bucket dbname=ws_bucket password=ws_bucket sslmode=disable")
db, err := gorm.Open("sqlite3", ":memory:")
if err != nil {
panic(err)
}
a := api.New(db)
go a.Run()
time.Sleep(time.Millisecond * 100)
m.Run()
}

53
test/slot_test.go Normal file
View File

@ -0,0 +1,53 @@
package test
import (
"github.com/simon987/ws_bucket/api"
"testing"
)
func TestAllocateUploadInvalidMaxSize(t *testing.T) {
if allocateUploadSlot(api.AllocateUploadSlotRequest{
FileName: "valid",
Token: "valid",
MaxSize: -1,
}).Ok != false {
t.Error()
}
}
func TestAllocateUploadSlotInvalidToken(t *testing.T) {
if allocateUploadSlot(api.AllocateUploadSlotRequest{
FileName: "valid",
Token: "",
MaxSize: 100,
}).Ok != false {
t.Error()
}
}
func TestAllocateUploadSlotUnsafePath(t *testing.T) {
if allocateUploadSlot(api.AllocateUploadSlotRequest{
FileName: "../test.png",
Token: "valid",
MaxSize: 100,
}).Ok != false {
t.Error()
}
if allocateUploadSlot(api.AllocateUploadSlotRequest{
FileName: "test/../../test.png",
Token: "valid",
MaxSize: 100,
}).Ok != false {
t.Error()
}
}
func allocateUploadSlot(request api.AllocateUploadSlotRequest) (ar *api.GenericResponse) {
resp := Post("/slot", request)
UnmarshalResponse(resp, &ar)
return
}

179
test/upload_test.go Normal file
View File

@ -0,0 +1,179 @@
package test
import (
"bytes"
"fmt"
"github.com/fasthttp/websocket"
"github.com/google/uuid"
"github.com/simon987/ws_bucket/api"
"io/ioutil"
"math"
"net/http"
"net/url"
"testing"
)
func TestWebsocketReturnsMotd(t *testing.T) {
id := uuid.New()
allocateUploadSlot(api.AllocateUploadSlotRequest{
FileName: "testmotd",
MaxSize: 0,
Token: id.String(),
})
c := ws(id.String())
motd := &api.WebsocketMotd{}
err := c.ReadJSON(&motd)
handleErr(err)
if len(motd.Motd) <= 0 {
t.Error()
}
if len(motd.Info.Version) <= 0 {
t.Error()
}
}
func TestWebSocketUploadSmallFile(t *testing.T) {
id := uuid.New()
allocateUploadSlot(api.AllocateUploadSlotRequest{
FileName: "testfile",
Token: id.String(),
MaxSize: math.MaxInt64,
})
c := ws(id.String())
_, _, err := c.ReadMessage()
handleErr(err)
err = c.WriteMessage(websocket.BinaryMessage, []byte("testuploadsmallfile"))
handleErr(err)
err = c.Close()
handleErr(err)
resp := readUploadSlot(id.String())
if bytes.Compare(resp, []byte("testuploadsmallfile")) != 0 {
t.Error()
}
}
func TestWebSocketUploadOverwritesFile(t *testing.T) {
id := uuid.New()
allocateUploadSlot(api.AllocateUploadSlotRequest{
FileName: "testuploadoverwrites",
Token: id.String(),
MaxSize: math.MaxInt64,
})
c := ws(id.String())
_, _, err := c.ReadMessage()
handleErr(err)
err = c.WriteMessage(websocket.BinaryMessage, []byte("testuploadsmallfile"))
handleErr(err)
err = c.Close()
handleErr(err)
c1 := ws(id.String())
_, _, err = c1.ReadMessage()
handleErr(err)
err = c1.WriteMessage(websocket.BinaryMessage, []byte("newvalue"))
handleErr(err)
err = c1.Close()
handleErr(err)
resp := readUploadSlot(id.String())
if bytes.Compare(resp, []byte("newvalue")) != 0 {
t.Error()
}
}
func TestWebSocketUploadLargeFile(t *testing.T) {
id := uuid.New()
allocateUploadSlot(api.AllocateUploadSlotRequest{
FileName: "testlargefile",
Token: id.String(),
MaxSize: math.MaxInt64,
})
c := ws(id.String())
_, _, err := c.ReadMessage()
handleErr(err)
chunk := make([]byte, 100000)
_ = copy(chunk, "test")
_ = c.WriteMessage(websocket.BinaryMessage, chunk)
err = c.Close()
handleErr(err)
resp := readUploadSlot(id.String())
if bytes.Compare(resp, chunk) != 0 {
t.Error()
}
}
func TestWebSocketUploadMaxSize(t *testing.T) {
id := uuid.New()
allocateUploadSlot(api.AllocateUploadSlotRequest{
FileName: "testmaxsize",
Token: id.String(),
MaxSize: 10,
})
c := ws(id.String())
_, _, err := c.ReadMessage()
handleErr(err)
chunk := make([]byte, 100000)
_ = copy(chunk, "test")
_ = c.WriteMessage(websocket.BinaryMessage, chunk)
err = c.Close()
handleErr(err)
resp := readUploadSlot(id.String())
if len(resp) != 10 {
t.Error()
}
}
func readUploadSlot(token string) []byte {
r := Get("/slot", token)
data, err := ioutil.ReadAll(r.Body)
handleErr(err)
return data
}
func ws(slot string) *websocket.Conn {
u := url.URL{Scheme: "ws", Host: "localhost:3021", Path: "/upload"}
fmt.Printf("Connecting to %s", u.String())
header := http.Header{}
header.Add("X-Upload-Token", slot)
c, _, err := websocket.DefaultDialer.Dial(u.String(), header)
handleErr(err)
return c
}