Prototype of DirStore and some routes for working with buckets
parent
8d16099c10
commit
915bd5ee65
@ -1,3 +1,132 @@
|
||||
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
|
||||
|
||||
import "github.com/gofiber/fiber/v2"
|
||||
import (
|
||||
"bytes"
|
||||
"log"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
)
|
||||
|
||||
func (r *Router) Api(api fiber.Router) {
|
||||
api.Get("/status", func(c *fiber.Ctx) error {
|
||||
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