JSON API new spec with deref

master
Bel LaPointe 2020-07-22 20:40:44 -06:00
parent 4d667e7b11
commit 710e20d6e0
8 changed files with 86 additions and 46 deletions

View File

@ -5,6 +5,9 @@ import "local/args"
type Config struct { type Config struct {
Port int Port int
DBURI string DBURI string
Database string
FilePrefix string
FileRoot string
} }
func New() Config { func New() Config {
@ -12,6 +15,9 @@ func New() Config {
as.Append(args.INT, "p", "port to listen on", 18114) as.Append(args.INT, "p", "port to listen on", 18114)
as.Append(args.STRING, "dburi", "database uri", "mongodb://localhost:27017") as.Append(args.STRING, "dburi", "database uri", "mongodb://localhost:27017")
as.Append(args.STRING, "fileprefix", "path prefix for file service", "/__files__")
as.Append(args.STRING, "fileroot", "path to file hosting root", "/tmp/")
as.Append(args.STRING, "database", "database name to use", "db")
if err := as.Parse(); err != nil { if err := as.Parse(); err != nil {
panic(err) panic(err)
@ -20,5 +26,8 @@ func New() Config {
return Config{ return Config{
Port: as.GetInt("p"), Port: as.GetInt("p"),
DBURI: as.GetString("dburi"), DBURI: as.GetString("dburi"),
FilePrefix: as.GetString("fileprefix"),
FileRoot: as.GetString("fileroot"),
Database: as.GetString("database"),
} }
} }

View File

@ -12,13 +12,14 @@ const (
) )
type One struct { type One struct {
Name string `bson:"_id,omitempty"` Name string `bson:"_id,omitempty" json:"name,omitempty"`
Type string `bson:"type,omitempty"` Type string `bson:"type,omitempty" json:"type,omitempty"`
Title string `bson:"title,omitempty"` Title string `bson:"title,omitempty" json:"title,omitempty"`
Image string `bson:"image,omitempty"` Image string `bson:"image,omitempty" json:"image,omitempty"`
Text string `bson:"text,omitempty"` Text string `bson:"text,omitempty" json:"text,omitempty"`
Modified int64 `bson:"modified,omitempty"` Relationship string `bson:"relationship,omitempty" json:"relationship,omitempty"`
Connections []Peer `bson:"connections,omitempty"` Modified int64 `bson:"modified,omitempty" json:"modified,omitempty"`
Connections []One `bson:"connections,omitempty" json:"connections,omitempty"`
} }
func (o One) Query() One { func (o One) Query() One {

View File

@ -1,6 +0,0 @@
package entity
type Peer struct {
Name string `bson:"_id,omitempty"`
Relationship string `bson:"relationship,omitempty"`
}

View File

@ -43,3 +43,7 @@ func (g Graph) Insert(ctx context.Context, one entity.One) error {
func (g Graph) Update(ctx context.Context, one entity.One, modify interface{}) error { func (g Graph) Update(ctx context.Context, one entity.One, modify interface{}) error {
return g.mongo.Update(ctx, one, modify) return g.mongo.Update(ctx, one, modify)
} }
func (g Graph) Delete(ctx context.Context, filter interface{}) error {
return g.mongo.Delete(ctx, filter)
}

View File

@ -34,7 +34,7 @@ func TestIntegration(t *testing.T) {
randomOne(), randomOne(),
randomOne(), randomOne(),
} }
ones[0].Connections = []entity.Peer{entity.Peer{Name: ones[2].Name, Relationship: ":("}} ones[0].Connections = []entity.One{entity.One{Name: ones[2].Name, Relationship: ":("}}
t.Run("graph.Insert(...)", func(t *testing.T) { t.Run("graph.Insert(...)", func(t *testing.T) {
for _, one := range ones { for _, one := range ones {
@ -91,7 +91,7 @@ func TestIntegration(t *testing.T) {
}) })
t.Run("graph.Update(foo, ++...); graph.Update(foo, --if :()", func(t *testing.T) { t.Run("graph.Update(foo, ++...); graph.Update(foo, --if :()", func(t *testing.T) {
err := graph.Update(ctx, ones[0].Query(), operator.Set{entity.Connections, []entity.Peer{entity.Peer{Name: "hello", Relationship: ":("}}}) err := graph.Update(ctx, ones[0].Query(), operator.Set{entity.Connections, []entity.One{entity.One{Name: "hello", Relationship: ":("}}})
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
@ -139,6 +139,6 @@ func randomOne() entity.One {
Image: "/path/to.jpg", Image: "/path/to.jpg",
Text: "tee hee xd", Text: "tee hee xd",
Modified: time.Now().UnixNano(), Modified: time.Now().UnixNano(),
Connections: []entity.Peer{}, Connections: []entity.One{},
} }
} }

View File

@ -34,7 +34,7 @@ func NewMongo() Mongo {
} }
return Mongo{ return Mongo{
client: c, client: c,
db: "db", db: config.New().Database,
col: "col", col: "col",
} }
} }

View File

@ -5,6 +5,7 @@ import (
"fmt" "fmt"
"local/whodunit/config" "local/whodunit/config"
"local/whodunit/storage" "local/whodunit/storage"
"local/whodunit/storage/entity"
"log" "log"
"net/http" "net/http"
) )
@ -33,16 +34,30 @@ func foo(g storage.Graph) http.Handler {
} }
func who(g storage.Graph, w http.ResponseWriter, r *http.Request) error { func who(g storage.Graph, w http.ResponseWriter, r *http.Request) error {
results := make(map[string]storage.One) ids := r.URL.Query()["id"]
for _, id := range r.URL.Query()["id"] { _, verbose := r.URL.Query()["v"]
results := make(map[string]entity.One)
for i := 0; i < len(ids); i++ {
id := ids[i]
ones, err := g.List(r.Context(), id) ones, err := g.List(r.Context(), id)
if err != nil { if err != nil {
return err return err
} }
if len(ones) != 1 { if len(ones) != 1 {
ones = append(ones, storage.One{}) ones = append(ones, entity.One{})
}
one := ones[0]
results[id] = one
if verbose {
ones, err := g.List(r.Context(), one.Peers()...)
if err != nil {
return err
}
for i, one := range ones {
one.Connections = nil
results[id].Connections[i] = one
}
} }
results[id] = ones[0]
} }
log.Println("results:", results) log.Println("results:", results)
return json.NewEncoder(w).Encode(results) return json.NewEncoder(w).Encode(results)

View File

@ -6,39 +6,25 @@ import (
"fmt" "fmt"
"local/whodunit/config" "local/whodunit/config"
"local/whodunit/storage" "local/whodunit/storage"
"local/whodunit/storage/entity"
"net/http" "net/http"
"os" "os"
"testing" "testing"
"time" "time"
"github.com/google/uuid"
) )
func TestHtml(t *testing.T) { func TestHtml(t *testing.T) {
if len(os.Getenv("INTEGRATION")) > 0 { if len(os.Getenv("INTEGRATION")) == 0 {
t.Logf("skipping because $INTEGRATION unset") t.Logf("skipping because $INTEGRATION unset")
return return
} }
os.Args = os.Args[:1] os.Args = os.Args[:1]
g := storage.NewGraph()
ones := []storage.One{
storage.One{
ID: "A",
Know: []storage.One{storage.One{}},
},
storage.One{
ID: "B",
Know: []storage.One{storage.One{}},
},
}
ones[0].Know[0] = ones[1]
ones[0].Know[0].Know = nil
ones[0].Know[0].Relation = ":)"
ones[1].Know[0] = ones[0]
ones[1].Know[0].Know = nil
ones[1].Know[0].Relation = ":("
g.Insert(context.TODO(), ones[0]) g := storage.NewGraph()
g.Insert(context.TODO(), ones[1]) ones := fillDB(t, g)
go func() { go func() {
if err := Html(g); err != nil { if err := Html(g); err != nil {
@ -46,7 +32,7 @@ func TestHtml(t *testing.T) {
} }
}() }()
time.Sleep(time.Millisecond * 250) time.Sleep(time.Millisecond * 250)
resp, err := http.Get(fmt.Sprintf("http://localhost:%d/who?id=A&id=B", config.New().Port)) resp, err := http.Get(fmt.Sprintf("http://localhost:%d/who?id=%s&v", config.New().Port, ones[len(ones)-1].Name))
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
@ -61,3 +47,34 @@ func TestHtml(t *testing.T) {
} }
t.Logf("\n%s\n", b) t.Logf("\n%s\n", b)
} }
func fillDB(t *testing.T, g storage.Graph) []entity.One {
ones := make([]entity.One, 5)
for i := range ones {
ones[i] = randomOne()
if i > 0 {
ones[i].Connections = []entity.One{entity.One{
Name: ones[i-1].Name,
Relationship: ":D",
}}
}
}
for i := range ones {
if err := g.Insert(context.TODO(), ones[i]); err != nil {
t.Fatal(err)
}
}
return ones
}
func randomOne() entity.One {
return entity.One{
Name: "name-" + uuid.New().String()[:5],
Type: "type-" + uuid.New().String()[:5],
Title: "titl-" + uuid.New().String()[:5],
Image: "imge-" + uuid.New().String()[:5],
Text: "text-" + uuid.New().String()[:5],
Modified: time.Now().UnixNano(),
Connections: []entity.One{},
}
}