Initial dumb servers
commit
1f679f4c06
|
|
@ -0,0 +1,15 @@
|
||||||
|
.*.go
|
||||||
|
vendor
|
||||||
|
*.swp
|
||||||
|
*.swo
|
||||||
|
testdata
|
||||||
|
server/server
|
||||||
|
client/client
|
||||||
|
*.db
|
||||||
|
*.ldb
|
||||||
|
*.log
|
||||||
|
*.bak
|
||||||
|
CURRENT
|
||||||
|
LOCK
|
||||||
|
LOG
|
||||||
|
MANIFEST-*
|
||||||
|
|
@ -0,0 +1,34 @@
|
||||||
|
package config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"local/storage"
|
||||||
|
"sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
var config Config
|
||||||
|
var lock = &sync.RWMutex{}
|
||||||
|
|
||||||
|
type Config struct {
|
||||||
|
db string
|
||||||
|
DB storage.DB
|
||||||
|
Port string
|
||||||
|
Addr string
|
||||||
|
Username string
|
||||||
|
Password string
|
||||||
|
}
|
||||||
|
|
||||||
|
func Values() Config {
|
||||||
|
lock.RLock()
|
||||||
|
defer lock.RUnlock()
|
||||||
|
return config
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c Config) String() string {
|
||||||
|
return fmt.Sprintf(
|
||||||
|
"port:%v db:%v addr:%v user:*** pass:***",
|
||||||
|
c.Port,
|
||||||
|
c.db,
|
||||||
|
c.Addr,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,28 @@
|
||||||
|
package config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"local/storage"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
func New() error {
|
||||||
|
config = Config{
|
||||||
|
db: orEnv(storage.MAP.String(), "DB", "DATABASE"),
|
||||||
|
Addr: orEnv("", "ADDR", "FILE"),
|
||||||
|
Username: orEnv("", "USER", "USERNAME"),
|
||||||
|
Password: orEnv("", "PASS", "PASSWORD"),
|
||||||
|
Port: orEnv("21412", "PORT", "LISTEN"),
|
||||||
|
}
|
||||||
|
DB, err := storage.New(storage.TypeFromString(config.db), config.Addr, config.Username, config.Password)
|
||||||
|
config.DB = DB
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func orEnv(def, key string, keys ...string) string {
|
||||||
|
for _, key := range append(keys, key) {
|
||||||
|
if v, ok := os.LookupEnv(key); ok {
|
||||||
|
return v
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,37 @@
|
||||||
|
package config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestNew(t *testing.T) {
|
||||||
|
invalidEnv := "&&&&&&"
|
||||||
|
keys := []string{
|
||||||
|
"DB",
|
||||||
|
"DATABASE",
|
||||||
|
"ADDR",
|
||||||
|
"PATH",
|
||||||
|
"USER",
|
||||||
|
"USERNAME",
|
||||||
|
"PASS",
|
||||||
|
"PASSWORD",
|
||||||
|
}
|
||||||
|
was := make(map[string]string)
|
||||||
|
for _, k := range keys {
|
||||||
|
v, ok := os.LookupEnv(k)
|
||||||
|
if !ok {
|
||||||
|
v = invalidEnv
|
||||||
|
}
|
||||||
|
was[k] = v
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
for k, v := range was {
|
||||||
|
if v != invalidEnv {
|
||||||
|
os.Setenv(k, v)
|
||||||
|
} else {
|
||||||
|
os.Unsetenv(k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,24 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"local/dynamodb/server/config"
|
||||||
|
"local/dynamodb/server/serve"
|
||||||
|
"log"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
if err := config.New(); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Printf("config: %s", config.Values().String())
|
||||||
|
|
||||||
|
server := serve.New()
|
||||||
|
if err := server.Routes(); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := server.Run(); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,20 @@
|
||||||
|
package serve
|
||||||
|
|
||||||
|
import (
|
||||||
|
"local/dynamodb/server/config"
|
||||||
|
"local/s2sa/s2sa/server/router"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Server struct {
|
||||||
|
port string
|
||||||
|
router *router.Router
|
||||||
|
}
|
||||||
|
|
||||||
|
func New() *Server {
|
||||||
|
config := config.Values()
|
||||||
|
return &Server{
|
||||||
|
port: ":" + strings.TrimPrefix(config.Port, ":"),
|
||||||
|
router: router.New(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,11 @@
|
||||||
|
package serve
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestNew(t *testing.T) {
|
||||||
|
os.Setenv("DB", "MAP")
|
||||||
|
New()
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,62 @@
|
||||||
|
package serve
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"local/dynamodb/server/config"
|
||||||
|
"local/s2sa/s2sa/server/router"
|
||||||
|
"local/storage"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (s *Server) Routes() error {
|
||||||
|
if err := s.router.Add("/"+router.Wildcard, s.CatchAll); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) CatchAll(w http.ResponseWriter, r *http.Request) {
|
||||||
|
foo := http.NotFound
|
||||||
|
switch strings.ToLower(r.Method) {
|
||||||
|
case "get":
|
||||||
|
foo = s.get
|
||||||
|
case "put":
|
||||||
|
foo = s.put
|
||||||
|
case "delete":
|
||||||
|
foo = s.put
|
||||||
|
}
|
||||||
|
foo(w, r)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) get(w http.ResponseWriter, r *http.Request) {
|
||||||
|
config := config.Values()
|
||||||
|
key := strings.Split(r.URL.Path, "/")[1]
|
||||||
|
if v, err := config.DB.Get(key); err == storage.ErrNotFound {
|
||||||
|
w.WriteHeader(http.StatusNotFound)
|
||||||
|
return
|
||||||
|
} else if err != nil {
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
w.Write(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) put(w http.ResponseWriter, r *http.Request) {
|
||||||
|
config := config.Values()
|
||||||
|
key := strings.Split(r.URL.Path, "/")[1]
|
||||||
|
if r == nil || r.Body == nil {
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
value, err := ioutil.ReadAll(r.Body)
|
||||||
|
if err != nil {
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if err := config.DB.Set(key, value); err != nil {
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,137 @@
|
||||||
|
package serve
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"local/dynamodb/server/config"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
var validKey = "key"
|
||||||
|
var validValue = "value"
|
||||||
|
|
||||||
|
func TestRoutes(t *testing.T) {
|
||||||
|
os.Setenv("DB", "MAP")
|
||||||
|
config.New()
|
||||||
|
s := New()
|
||||||
|
if err := s.Routes(); err != nil {
|
||||||
|
t.Fatalf("cannot routes(): %v", err)
|
||||||
|
}
|
||||||
|
config.Values().DB.Set(validKey, []byte("anything"))
|
||||||
|
|
||||||
|
cases := []struct {
|
||||||
|
path string
|
||||||
|
method string
|
||||||
|
status int
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
path: "/" + validKey,
|
||||||
|
method: "GET",
|
||||||
|
status: http.StatusOK,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "/" + validKey,
|
||||||
|
method: "PUT",
|
||||||
|
status: http.StatusOK,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "/" + validKey,
|
||||||
|
method: "DELETE",
|
||||||
|
status: http.StatusOK,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "/" + validKey,
|
||||||
|
method: "POST",
|
||||||
|
status: http.StatusNotFound,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "/",
|
||||||
|
method: "GET",
|
||||||
|
status: http.StatusNotFound,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "/",
|
||||||
|
method: "PUT",
|
||||||
|
status: http.StatusNotFound,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "/",
|
||||||
|
method: "DELETE",
|
||||||
|
status: http.StatusNotFound,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "/",
|
||||||
|
method: "POST",
|
||||||
|
status: http.StatusNotFound,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "",
|
||||||
|
method: "GET",
|
||||||
|
status: http.StatusNotFound,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "",
|
||||||
|
method: "PUT",
|
||||||
|
status: http.StatusNotFound,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "",
|
||||||
|
method: "DELETE",
|
||||||
|
status: http.StatusNotFound,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "",
|
||||||
|
method: "POST",
|
||||||
|
status: http.StatusNotFound,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, c := range cases {
|
||||||
|
//t.Logf("CASE %v", c)
|
||||||
|
r, err := http.NewRequest(c.method, c.path, strings.NewReader(validValue))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("err making request: %v", err)
|
||||||
|
}
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
s.ServeHTTP(w, r)
|
||||||
|
if w.Code != c.status {
|
||||||
|
t.Errorf("wrong status for %q on %q: %v, want %v", c.method, c.path, w.Code, c.status)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPutGet(t *testing.T) {
|
||||||
|
os.Setenv("DB", "MAP")
|
||||||
|
config.New()
|
||||||
|
s := New()
|
||||||
|
if err := s.Routes(); err != nil {
|
||||||
|
t.Errorf("cannot routes(): %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
r, err := http.NewRequest("PUT", "/"+validKey, strings.NewReader(validValue))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("err making put request: %v", err)
|
||||||
|
}
|
||||||
|
s.put(w, r)
|
||||||
|
if w.Code != http.StatusOK {
|
||||||
|
t.Fatalf("err status on put: %v", w.Code)
|
||||||
|
}
|
||||||
|
|
||||||
|
w = httptest.NewRecorder()
|
||||||
|
r, err = http.NewRequest("GET", "/"+validKey, nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("err making get request: %v", err)
|
||||||
|
}
|
||||||
|
s.get(w, r)
|
||||||
|
if w.Code != http.StatusOK {
|
||||||
|
t.Fatalf("err status on get: %v", w.Code)
|
||||||
|
} else if v, err := ioutil.ReadAll(w.Body); err != nil {
|
||||||
|
t.Fatalf("err reading response body on get: %v", err)
|
||||||
|
} else if string(v) != validValue {
|
||||||
|
t.Fatalf("wrong response body on get: %v", string(v))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,13 @@
|
||||||
|
package serve
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (s *Server) Run() error {
|
||||||
|
return http.ListenAndServe(s.port, s)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||||
|
s.router.ServeHTTP(w, r)
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue