2020-10-13 21:36:50 +02:00
|
|
|
package db
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewInMemoryDB a constructor to build a new inMemoryDB.
|
|
|
|
func NewInMemoryDB() *InMemoryDB {
|
|
|
|
return &InMemoryDB{storage: make(map[string]Host), m: sync.Mutex{}}
|
|
|
|
}
|
|
|
|
|
|
|
|
// InMemoryDB a in memory DB implementing the db.HostDB interface.
|
|
|
|
type InMemoryDB struct {
|
|
|
|
storage map[string]Host
|
|
|
|
m sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithCustomStorage allows to put a custom map as DB storage.
|
|
|
|
// Returns a new InMemoryDB.
|
|
|
|
func (db *InMemoryDB) WithCustomStorage(storage map[string]Host) *InMemoryDB {
|
2020-11-11 12:28:11 +01:00
|
|
|
db.m.Lock()
|
|
|
|
defer db.m.Unlock()
|
|
|
|
|
2020-10-13 21:36:50 +02:00
|
|
|
return &InMemoryDB{storage: storage, m: sync.Mutex{}}
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetHosts returns a paginated result of all hosts.
|
|
|
|
// It returns an error if no host was found or all entries are beeing skiped.
|
2020-10-18 15:35:47 +02:00
|
|
|
func (db *InMemoryDB) GetHosts(pagination Pagination) ([]HostInfo, error) {
|
2020-11-11 12:28:11 +01:00
|
|
|
db.m.Lock()
|
|
|
|
defer db.m.Unlock()
|
|
|
|
|
2020-10-13 21:36:50 +02:00
|
|
|
hosts := make([]HostInfo, 0)
|
|
|
|
for _, value := range db.storage {
|
|
|
|
hosts = append(hosts, value.HostInfo)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(hosts) == 0 {
|
|
|
|
return []HostInfo{}, ErrHostsNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
records := len(hosts)
|
2020-10-18 15:35:47 +02:00
|
|
|
if records < pagination.Skip {
|
2020-10-13 21:36:50 +02:00
|
|
|
return []HostInfo{}, ErrAllEntriesSkipped
|
2020-10-18 15:35:47 +02:00
|
|
|
} else if records < (pagination.Skip + pagination.Limit) {
|
2020-11-11 16:21:15 +01:00
|
|
|
hosts = hosts[pagination.Skip:]
|
|
|
|
foundHosts := make([]HostInfo, len(hosts))
|
|
|
|
copy(foundHosts, hosts)
|
|
|
|
return foundHosts, nil
|
2020-10-13 21:36:50 +02:00
|
|
|
}
|
|
|
|
|
2020-11-11 16:21:15 +01:00
|
|
|
hosts = hosts[pagination.Skip:(pagination.Skip + pagination.Limit)]
|
|
|
|
foundHosts := make([]HostInfo, len(hosts))
|
|
|
|
copy(foundHosts, hosts)
|
|
|
|
return foundHosts, nil
|
2020-10-13 21:36:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetHost returns a host with the matching hostname.
|
|
|
|
// If no host could be found it will return an error.
|
|
|
|
func (db *InMemoryDB) GetHost(hostname string) (HostInfo, error) {
|
2020-11-11 12:28:11 +01:00
|
|
|
db.m.Lock()
|
|
|
|
defer db.m.Unlock()
|
|
|
|
|
2020-10-13 21:36:50 +02:00
|
|
|
host, found := db.storage[hostname]
|
|
|
|
if !found {
|
|
|
|
return HostInfo{}, ErrHostNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
return host.HostInfo, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetStatsByHostname gets all Stats in a paginated form from a specific host.
|
|
|
|
// It returns errors if no host is found or if all entries are beeing skiped.
|
2020-10-18 15:35:47 +02:00
|
|
|
func (db *InMemoryDB) GetStatsByHostname(hostname string, pagination Pagination) ([]Stats, error) {
|
2020-11-11 12:28:11 +01:00
|
|
|
db.m.Lock()
|
|
|
|
defer db.m.Unlock()
|
|
|
|
|
2020-10-13 21:36:50 +02:00
|
|
|
host, found := db.storage[hostname]
|
|
|
|
if !found {
|
|
|
|
return []Stats{}, ErrHostNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
records := len(host.Stats)
|
2020-10-18 15:35:47 +02:00
|
|
|
if records < pagination.Skip {
|
2020-10-13 21:36:50 +02:00
|
|
|
return []Stats{}, ErrAllEntriesSkipped
|
2020-10-18 15:35:47 +02:00
|
|
|
} else if records < (pagination.Skip + pagination.Limit) {
|
2020-11-11 16:21:15 +01:00
|
|
|
stats := host.Stats[pagination.Skip:]
|
|
|
|
foundStats := make([]Stats, len(stats))
|
|
|
|
copy(foundStats, stats)
|
|
|
|
return foundStats, nil
|
2020-10-13 21:36:50 +02:00
|
|
|
}
|
|
|
|
|
2020-11-11 16:21:15 +01:00
|
|
|
stats := host.Stats[pagination.Skip:(pagination.Skip + pagination.Limit)]
|
|
|
|
foundStats := make([]Stats, len(stats))
|
|
|
|
copy(foundStats, stats)
|
|
|
|
return foundStats, nil
|
2020-10-13 21:36:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// InsertStats into the DB.
|
|
|
|
// To do so it takes the hostname of the Hostname field and creates a new host inside the DB and/or adds the stat to it.
|
|
|
|
// The HostInfos are also beeing updated.
|
|
|
|
// This implementation won't return an error but its declared to implement the db.HostDB interface.
|
2020-10-18 15:35:47 +02:00
|
|
|
func (db *InMemoryDB) InsertStats(hostname string, stats Stats) error {
|
2020-10-13 21:36:50 +02:00
|
|
|
db.m.Lock()
|
|
|
|
defer db.m.Unlock()
|
|
|
|
|
2020-11-10 19:07:22 +01:00
|
|
|
host, found := db.storage[hostname]
|
2020-10-13 21:36:50 +02:00
|
|
|
if !found {
|
2020-11-10 19:07:22 +01:00
|
|
|
hostInfo := HostInfo{Hostname: hostname, DataPoints: 1, LastInsert: time.Now()}
|
|
|
|
db.storage[hostname] = Host{HostInfo: hostInfo, Stats: []Stats{stats}}
|
2020-10-13 21:36:50 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-11-11 10:25:59 +01:00
|
|
|
host.Stats = db.insertAtBeginning(host.Stats, stats)
|
2020-10-18 15:35:47 +02:00
|
|
|
host.HostInfo.DataPoints++
|
2020-10-13 21:36:50 +02:00
|
|
|
host.HostInfo.LastInsert = time.Now()
|
|
|
|
|
2020-11-10 19:07:22 +01:00
|
|
|
db.storage[hostname] = host
|
2020-10-13 21:36:50 +02:00
|
|
|
return nil
|
|
|
|
}
|
2020-11-11 10:25:59 +01:00
|
|
|
|
|
|
|
func (db *InMemoryDB) insertAtBeginning(stats []Stats, stat Stats) []Stats {
|
|
|
|
stats = append(stats, Stats{})
|
|
|
|
copy(stats[1:], stats)
|
|
|
|
stats[0] = stat
|
|
|
|
|
|
|
|
return stats
|
|
|
|
}
|