Introduces context.Context in API controllers

This commit is contained in:
Deluan 2017-04-03 00:16:21 -04:00
parent dc98b187d9
commit 2861a97a89
5 changed files with 47 additions and 14 deletions

View file

@ -1,6 +1,6 @@
#@IgnoreInspection BashAddShebang #@IgnoreInspection BashAddShebang
BINARY=gosonic BINARY=sonic-server
SOURCES := $(shell find . -name '*.go') SOURCES := $(shell find . -name '*.go')
@ -12,7 +12,7 @@ $(BINARY): $(SOURCES)
.PHONY: clean .PHONY: clean
clean: clean:
rm -f ${BINARY} rm -f ${BINARY}
.PHONY: setup .PHONY: setup
setup: setup:
go get -u github.com/beego/bee # bee command line tool go get -u github.com/beego/bee # bee command line tool

View file

@ -6,9 +6,8 @@ CloudSonic Server
__This is still a work in progress, and has no releases available__ __This is still a work in progress, and has no releases available__
CloudSonic is an application that implements the [Subsonic API](http://www.subsonic.org/pages/api.jsp), but instead of CloudSonic is a music collection server and streamer, optmized to run on cheap VPS servers. It implements the
having its own music library like the original [Subsonic application](http://www.subsonic.org), it interacts directly [Subsonic](http://www.subsonic.org) API
with your iTunes library.
The project's main goals are: The project's main goals are:
@ -17,9 +16,8 @@ The project's main goals are:
[DSub](http://www.subsonic.org/pages/apps.jsp#dsub), [DSub](http://www.subsonic.org/pages/apps.jsp#dsub),
[SubFire](http://www.subsonic.org/pages/apps.jsp#subfire) and [SubFire](http://www.subsonic.org/pages/apps.jsp#subfire) and
[Jamstash](http://www.subsonic.org/pages/apps.jsp#jamstash)) [Jamstash](http://www.subsonic.org/pages/apps.jsp#jamstash))
* Use all metadata from iTunes, so that you can keep using iTunes to manage your music * Import and use all metadata from iTunes, so that you can optionally keep using iTunes to manage your music
* Keep iTunes stats (play counts, last played dates, ratings, etc..) updated, at least on Mac OS X. * Implement Smart Playlists, as iTunes
This allows smart playlists to be used in Subsonic Clients
* Help me learn Go ;) [![Gopher](https://blog.golang.org/favicon.ico)](https://golang.org) * Help me learn Go ;) [![Gopher](https://blog.golang.org/favicon.ico)](https://golang.org)
@ -41,7 +39,7 @@ The server should start listening on port 4533.
### Development Environment ### Development Environment
You will need to install [Go 1.7](https://golang.org/dl/) You will need to install [Go 1.8](https://golang.org/dl/)
Then install dependencies: Then install dependencies:
``` ```

View file

@ -1,6 +1,7 @@
package api package api
import ( import (
"context"
"encoding/xml" "encoding/xml"
"fmt" "fmt"
"strconv" "strconv"
@ -12,7 +13,10 @@ import (
"github.com/cloudsonic/sonic-server/utils" "github.com/cloudsonic/sonic-server/utils"
) )
type BaseAPIController struct{ beego.Controller } type BaseAPIController struct {
beego.Controller
context context.Context
}
func (c *BaseAPIController) NewEmpty() responses.Subsonic { func (c *BaseAPIController) NewEmpty() responses.Subsonic {
return responses.Subsonic{Status: "ok", Version: beego.AppConfig.String("apiVersion")} return responses.Subsonic{Status: "ok", Version: beego.AppConfig.String("apiVersion")}

View file

@ -1,6 +1,7 @@
package api package api
import ( import (
"context"
"crypto/md5" "crypto/md5"
"encoding/hex" "encoding/hex"
"fmt" "fmt"
@ -18,6 +19,7 @@ type ControllerInterface interface {
} }
func Validate(controller BaseAPIController) { func Validate(controller BaseAPIController) {
addNewContext(controller)
if !conf.Sonic.DisableValidation { if !conf.Sonic.DisableValidation {
checkParameters(controller) checkParameters(controller)
authenticate(controller) authenticate(controller)
@ -25,6 +27,20 @@ func Validate(controller BaseAPIController) {
} }
} }
func getData(c BaseAPIController, name string) string {
if v, ok := c.Ctx.Input.GetData(name).(string); ok {
return v
}
return ""
}
func addNewContext(c BaseAPIController) {
ctx := context.Background()
id := getData(c, "requestId")
c.context = context.WithValue(ctx, "requestId", id)
}
func checkParameters(c BaseAPIController) { func checkParameters(c BaseAPIController) {
requiredParameters := []string{"u", "v", "c"} requiredParameters := []string{"u", "v", "c"}
@ -56,10 +72,10 @@ func authenticate(c BaseAPIController) {
pass = string(dec) pass = string(dec)
} }
} }
valid = (pass == password) valid = pass == password
case token != "": case token != "":
t := fmt.Sprintf("%x", md5.Sum([]byte(password+salt))) t := fmt.Sprintf("%x", md5.Sum([]byte(password+salt)))
valid = (t == token) valid = t == token
} }
if user != conf.Sonic.User || !valid { if user != conf.Sonic.User || !valid {

View file

@ -6,8 +6,11 @@ import (
"github.com/astaxie/beego/plugins/cors" "github.com/astaxie/beego/plugins/cors"
"github.com/cloudsonic/sonic-server/api" "github.com/cloudsonic/sonic-server/api"
"github.com/cloudsonic/sonic-server/controllers" "github.com/cloudsonic/sonic-server/controllers"
"github.com/twinj/uuid"
) )
const requestidHeader = "X-Request-Id"
func init() { func init() {
mapEndpoints() mapEndpoints()
mapControllers() mapControllers()
@ -67,12 +70,23 @@ func mapControllers() {
} }
func initFilters() { func initFilters() {
var ValidateRequest = func(ctx *context.Context) { var requestIdFilter = func(ctx *context.Context) {
id := ctx.Input.Header(requestidHeader)
if id == "" {
id = uuid.NewV4().String()
}
ctx.Input.SetData("requestId", id)
}
var validateRequest = func(ctx *context.Context) {
c := api.BaseAPIController{} c := api.BaseAPIController{}
// TODO Find a way to not depend on a controller being passed
c.Ctx = ctx c.Ctx = ctx
c.Data = make(map[interface{}]interface{}) c.Data = make(map[interface{}]interface{})
api.Validate(c) api.Validate(c)
} }
beego.InsertFilter("/rest/*", beego.BeforeRouter, cors.Allow(&cors.Options{ beego.InsertFilter("/rest/*", beego.BeforeRouter, cors.Allow(&cors.Options{
AllowOrigins: []string{"*"}, AllowOrigins: []string{"*"},
AllowMethods: []string{"GET", "POST", "PUT", "DELETE", "OPTIONS"}, AllowMethods: []string{"GET", "POST", "PUT", "DELETE", "OPTIONS"},
@ -81,5 +95,6 @@ func initFilters() {
AllowCredentials: true, AllowCredentials: true,
})) }))
beego.InsertFilter("/rest/*", beego.BeforeRouter, ValidateRequest) beego.InsertFilter("/rest/*", beego.BeforeRouter, requestIdFilter)
beego.InsertFilter("/rest/*", beego.BeforeRouter, validateRequest)
} }