// Copyright © 2019 Ettore Di Giacinto // // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation; either version 2 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License along // with this program; if not, see . package pkg import ( "encoding/base64" "fmt" "hash/crc32" "os" "strconv" "sync" "time" "github.com/pkg/errors" storm "github.com/asdine/storm" "github.com/asdine/storm/q" "go.etcd.io/bbolt" ) //var BoltInstance PackageDatabase type BoltDatabase struct { sync.Mutex Path string } func NewBoltDatabase(path string) PackageDatabase { // if BoltInstance == nil { // BoltInstance = &BoltDatabase{Path: path} // } //return BoltInstance, nil return &BoltDatabase{Path: path} } func (db *BoltDatabase) Get(s string) (string, error) { bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return "", err } defer bolt.Close() var str string bolt.Get("solver", s, &str) return str, errors.New("Not implemented") } func (db *BoltDatabase) Set(k, v string) error { bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return err } defer bolt.Close() return bolt.Set("solver", k, v) } func (db *BoltDatabase) Create(v []byte) (string, error) { enc := base64.StdEncoding.EncodeToString(v) crc32q := crc32.MakeTable(0xD5828281) ID := fmt.Sprintf("%08x", crc32.Checksum([]byte(enc), crc32q)) // TODO: Replace with package fingerprint? return ID, db.Set(ID, enc) } func (db *BoltDatabase) Retrieve(ID string) ([]byte, error) { pa, err := db.Get(ID) if err != nil { return nil, err } enc, err := base64.StdEncoding.DecodeString(pa) if err != nil { return nil, err } return enc, nil } func (db *BoltDatabase) FindPackage(tofind Package) (Package, error) { p := &DefaultPackage{} bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return nil, err } defer bolt.Close() err = bolt.Select(q.Eq("Name", tofind.GetName()), q.Eq("Category", tofind.GetCategory()), q.Eq("Version", tofind.GetVersion())).Limit(1).First(p) if err != nil { return nil, err } return p, nil } func (db *BoltDatabase) UpdatePackage(p Package) error { // TODO: Change, but by query we cannot update by ID err := db.RemovePackage(p) if err != nil { return err } _, err = db.CreatePackage(p) if err != nil { return err } return nil } func (db *BoltDatabase) GetPackage(ID string) (Package, error) { p := &DefaultPackage{} bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return nil, err } defer bolt.Close() iid, err := strconv.Atoi(ID) if err != nil { return nil, err } err = bolt.Select(q.Eq("ID", iid)).Limit(1).First(p) //err = bolt.One("id", iid, p) return p, err } func (db *BoltDatabase) GetPackages() []string { ids := []string{} bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return []string{} } defer bolt.Close() // Fetching records one by one (useful when the bucket contains a lot of records) query := bolt.Select() query.Each(new(DefaultPackage), func(record interface{}) error { u := record.(*DefaultPackage) ids = append(ids, strconv.Itoa(u.ID)) return nil }) return ids } func (db *BoltDatabase) GetAllPackages(packages chan Package) error { bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return err } defer bolt.Close() // Fetching records one by one (useful when the bucket contains a lot of records) //query := bolt.Select() var packs []Package err = bolt.All(&packs) if err != nil { return err } for _, r := range packs { packages <- r } return nil // return query.Each(new(DefaultPackage), func(record interface{}) error { // u := record.(*DefaultPackage) // packages <- u // return err // }) } // Encode encodes the package to string. // It returns an ID which can be used to retrieve the package later on. func (db *BoltDatabase) CreatePackage(p Package) (string, error) { bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return "", errors.Wrap(err, "Error opening boltdb "+db.Path) } defer bolt.Close() dp, ok := p.(*DefaultPackage) if !ok { return "", errors.New("Bolt DB support only DefaultPackage type for now") } err = bolt.Save(dp) if err != nil { return "", errors.Wrap(err, "Error saving package to "+db.Path) } return strconv.Itoa(dp.ID), err } func (db *BoltDatabase) Clean() error { db.Lock() defer db.Unlock() return os.RemoveAll(db.Path) } func (db *BoltDatabase) GetPackageFiles(p Package) ([]string, error) { bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return []string{}, errors.Wrap(err, "Error opening boltdb "+db.Path) } defer bolt.Close() files := bolt.From("files") var pf PackageFile err = files.One("PackageFingerprint", p.GetFingerPrint(), &pf) if err != nil { return []string{}, errors.Wrap(err, "While finding files") } return pf.Files, nil } func (db *BoltDatabase) SetPackageFiles(p *PackageFile) error { bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return errors.Wrap(err, "Error opening boltdb "+db.Path) } defer bolt.Close() files := bolt.From("files") return files.Save(p) } func (db *BoltDatabase) RemovePackageFiles(p Package) error { bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return errors.Wrap(err, "Error opening boltdb "+db.Path) } defer bolt.Close() files := bolt.From("files") var pf PackageFile err = files.One("PackageFingerprint", p.GetFingerPrint(), &pf) if err != nil { return errors.Wrap(err, "While finding files") } return files.DeleteStruct(&pf) } func (db *BoltDatabase) RemovePackage(p Package) error { bolt, err := storm.Open(db.Path, storm.BoltOptions(0600, &bbolt.Options{Timeout: 30 * time.Second})) if err != nil { return errors.Wrap(err, "Error opening boltdb "+db.Path) } defer bolt.Close() var found DefaultPackage err = bolt.Select(q.Eq("Name", p.GetName()), q.Eq("Category", p.GetCategory()), q.Eq("Version", p.GetVersion())).Limit(1).Delete(&found) if err != nil { return errors.Wrap(err, "No package found to delete") } return nil }