Prototype of DirStore and some routes for working with buckets
parent
8d16099c10
commit
915bd5ee65
@ -1,3 +1,132 @@
|
|||||||
package database
|
package database
|
||||||
|
|
||||||
type Database interface{}
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"git.phc.dm.unipi.it/phc/storage/store"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Database interface {
|
||||||
|
Buckets() ([]string, error)
|
||||||
|
CreateBucket(bucket string, options ...any) error
|
||||||
|
|
||||||
|
Bucket(bucket string) (store.Store, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type bucketInfo struct {
|
||||||
|
Name string `json:"name"`
|
||||||
|
Path string `json:"path"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type jsonDB struct {
|
||||||
|
file string
|
||||||
|
|
||||||
|
BucketsInfo map[string]bucketInfo `json:"buckets"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewJSON(file string) Database {
|
||||||
|
db := &jsonDB{
|
||||||
|
file: file,
|
||||||
|
BucketsInfo: map[string]bucketInfo{},
|
||||||
|
}
|
||||||
|
|
||||||
|
err := db.load()
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return db
|
||||||
|
}
|
||||||
|
|
||||||
|
func (db *jsonDB) setup() error {
|
||||||
|
if _, err := os.Stat(db.file); !os.IsNotExist(err) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Printf("missing %q, creating empty database file", db.file)
|
||||||
|
|
||||||
|
f, err := os.Create(db.file)
|
||||||
|
defer f.Close()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
enc := json.NewEncoder(f)
|
||||||
|
enc.SetIndent("", " ")
|
||||||
|
if err := enc.Encode(&db); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (db *jsonDB) store() error {
|
||||||
|
f, err := os.Create(db.file)
|
||||||
|
defer f.Close()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
enc := json.NewEncoder(f)
|
||||||
|
enc.SetIndent("", " ")
|
||||||
|
if err := enc.Encode(&db); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (db *jsonDB) load() error {
|
||||||
|
if err := db.setup(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := os.Open(db.file)
|
||||||
|
defer f.Close()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := json.NewDecoder(f).Decode(&db); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (db *jsonDB) Buckets() ([]string, error) {
|
||||||
|
db.load()
|
||||||
|
|
||||||
|
buckets := make([]string, 0, len(db.BucketsInfo))
|
||||||
|
|
||||||
|
for _, b := range db.BucketsInfo {
|
||||||
|
buckets = append(buckets, b.Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
return buckets, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (db *jsonDB) CreateBucket(bucket string, options ...any) error {
|
||||||
|
db.load()
|
||||||
|
defer db.store()
|
||||||
|
|
||||||
|
bi := bucketInfo{Name: bucket, Path: bucket + "/"}
|
||||||
|
if len(options) > 0 {
|
||||||
|
bi.Path = options[0].(string)
|
||||||
|
}
|
||||||
|
|
||||||
|
db.BucketsInfo[bucket] = bi
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (db *jsonDB) Bucket(bucket string) (store.Store, error) {
|
||||||
|
b := db.BucketsInfo[bucket]
|
||||||
|
|
||||||
|
return &store.DirStore{
|
||||||
|
BaseDir: b.Path,
|
||||||
|
Prefix: 2,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
@ -1,7 +0,0 @@
|
|||||||
package database
|
|
||||||
|
|
||||||
type memDB struct {}
|
|
||||||
|
|
||||||
func NewInMemoryDB() Database {
|
|
||||||
return &memDB{}
|
|
||||||
}
|
|
@ -1,19 +0,0 @@
|
|||||||
package database
|
|
||||||
|
|
||||||
import "database/sql"
|
|
||||||
|
|
||||||
// Uncomment if actually using SQLite, the first build will be fairly slow
|
|
||||||
// import _ "github.com/mattn/go-sqlite3"
|
|
||||||
|
|
||||||
type sqliteDatabase struct {
|
|
||||||
Db *sql.DB
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewSQLite(filename string) (Database, error) {
|
|
||||||
db, err := sql.Open("sqlite3", filename)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return &sqliteDatabase{db}, nil
|
|
||||||
}
|
|
@ -1,9 +1,94 @@
|
|||||||
package routes
|
package routes
|
||||||
|
|
||||||
import "github.com/gofiber/fiber/v2"
|
import (
|
||||||
|
"bytes"
|
||||||
|
"log"
|
||||||
|
|
||||||
|
"github.com/gofiber/fiber/v2"
|
||||||
|
)
|
||||||
|
|
||||||
func (r *Router) Api(api fiber.Router) {
|
func (r *Router) Api(api fiber.Router) {
|
||||||
api.Get("/status", func(c *fiber.Ctx) error {
|
api.Get("/status", func(c *fiber.Ctx) error {
|
||||||
return c.JSON("ok")
|
return c.JSON("ok")
|
||||||
})
|
})
|
||||||
|
|
||||||
|
api.Get("/buckets", func(c *fiber.Ctx) error {
|
||||||
|
buckets, err := r.Database.Buckets()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.JSON(buckets)
|
||||||
|
})
|
||||||
|
|
||||||
|
api.Post("/buckets", func(c *fiber.Ctx) error {
|
||||||
|
var req struct {
|
||||||
|
Bucket string `json:"bucket"`
|
||||||
|
Path string `json:"path"`
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Printf("%v", string(c.Body()))
|
||||||
|
|
||||||
|
if err := c.BodyParser(&req); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
opts := []any{}
|
||||||
|
if req.Path != "" {
|
||||||
|
opts = append(opts, req.Path)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := r.Database.CreateBucket(req.Bucket, opts...); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.JSON("ok")
|
||||||
|
})
|
||||||
|
|
||||||
|
api.Post("/buckets/:bucket", func(c *fiber.Ctx) error {
|
||||||
|
bucket := c.Params("bucket")
|
||||||
|
|
||||||
|
b, err := r.Database.Bucket(bucket)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
ff, err := c.FormFile("file")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
mf, err := ff.Open()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
id, err := b.Create(mf)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.JSON(fiber.Map{
|
||||||
|
"bucket": bucket,
|
||||||
|
"id": id,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
api.Get("/buckets/:bucket/:id", func(c *fiber.Ctx) error {
|
||||||
|
bucket := c.Params("bucket")
|
||||||
|
id := c.Params("id")
|
||||||
|
|
||||||
|
buf := &bytes.Buffer{}
|
||||||
|
|
||||||
|
b, err := r.Database.Bucket(bucket)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := b.Read(id, buf); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.SendStream(buf)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,144 @@
|
|||||||
|
package store
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
|
||||||
|
"git.phc.dm.unipi.it/phc/storage/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Store interface {
|
||||||
|
// Create a new object by reading from the given io.Reader and returns its new id
|
||||||
|
Create(r io.Reader) (string, error)
|
||||||
|
|
||||||
|
// Read the object identified by "id" into the given io.Writer
|
||||||
|
Read(id string, w io.Writer) error
|
||||||
|
|
||||||
|
// Update the object identified by "id" by reading from the given io.Reader
|
||||||
|
Update(id string, r io.Reader) error
|
||||||
|
|
||||||
|
// Delete the object with the given id
|
||||||
|
Delete(id string) error
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// In Memory Byte Store
|
||||||
|
//
|
||||||
|
|
||||||
|
type memStore struct {
|
||||||
|
objects map[string][]byte
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewMemStore() Store {
|
||||||
|
return &memStore{
|
||||||
|
objects: map[string][]byte{},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a new object by reading from the given io.Reader and returns its new id
|
||||||
|
func (m *memStore) Create(r io.Reader) (string, error) {
|
||||||
|
id := utils.GenerateRandomString(16)
|
||||||
|
|
||||||
|
data, err := io.ReadAll(r)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
m.objects[id] = data
|
||||||
|
return id, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read the object identified by "id" into the given io.Writer
|
||||||
|
func (m *memStore) Read(id string, w io.Writer) error {
|
||||||
|
data, found := m.objects[id]
|
||||||
|
if !found {
|
||||||
|
return fmt.Errorf("object with id %q not found", id)
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := w.Write(data); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the object identified by "id" by reading from the given io.Reader
|
||||||
|
func (m *memStore) Update(id string, r io.Reader) error {
|
||||||
|
data, err := io.ReadAll(r)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
m.objects[id] = data
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete the object with the given id
|
||||||
|
func (m *memStore) Delete(id string) error {
|
||||||
|
if _, found := m.objects[id]; !found {
|
||||||
|
return fmt.Errorf("object with id %q not found", id)
|
||||||
|
}
|
||||||
|
|
||||||
|
delete(m.objects, id)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Dir Store
|
||||||
|
//
|
||||||
|
|
||||||
|
type DirStore struct {
|
||||||
|
// BaseDir is the root folder of this Store
|
||||||
|
BaseDir string
|
||||||
|
|
||||||
|
// Prefix is the number of letters to use for the first layer of folders
|
||||||
|
Prefix int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *DirStore) split(id string) (prefix, rest string) {
|
||||||
|
return id[:d.Prefix], id[d.Prefix:]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *DirStore) Create(r io.Reader) (string, error) {
|
||||||
|
id := utils.GenerateRandomString(16)
|
||||||
|
prefix, rest := d.split(id)
|
||||||
|
|
||||||
|
os.MkdirAll(path.Join(d.BaseDir, prefix), os.ModePerm)
|
||||||
|
|
||||||
|
f, err := os.Create(path.Join(d.BaseDir, prefix, rest))
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := io.Copy(f, r); err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
return id, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *DirStore) Read(id string, w io.Writer) error {
|
||||||
|
prefix, rest := d.split(id)
|
||||||
|
|
||||||
|
f, err := os.Open(path.Join(d.BaseDir, prefix, rest))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := io.Copy(w, f); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *DirStore) Update(id string, r io.Reader) error {
|
||||||
|
panic("TODO: Not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *DirStore) Delete(id string) error {
|
||||||
|
panic("TODO: Not implemented")
|
||||||
|
}
|
@ -0,0 +1,23 @@
|
|||||||
|
package utils
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/hex"
|
||||||
|
"math/rand"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
const alphabet = "1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
rand.Seed(time.Now().UnixNano())
|
||||||
|
}
|
||||||
|
|
||||||
|
func GenerateRandomString(n int) string {
|
||||||
|
b := make([]byte, n/2)
|
||||||
|
|
||||||
|
if _, err := rand.Read(b); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return hex.EncodeToString(b)
|
||||||
|
}
|
Loading…
Reference in New Issue