2024-05-09 23:53:33 +04:00
|
|
|
package sqlite
|
|
|
|
|
|
|
|
import (
|
|
|
|
"database/sql"
|
|
|
|
"dwelling-radio/internal/statistics"
|
|
|
|
_ "embed"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
_ "github.com/mattn/go-sqlite3"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
const dbDateFormat = "2006-01-02 15:04:05.999"
|
|
|
|
|
|
|
|
var (
|
|
|
|
//go:embed queries/schema.sql
|
|
|
|
querySchema string
|
|
|
|
|
|
|
|
//go:embed queries/song_add.sql
|
|
|
|
querySongAdd string
|
|
|
|
|
|
|
|
//go:embed queries/history_add.sql
|
|
|
|
queryHistoryAdd string
|
|
|
|
|
|
|
|
//go:embed queries/last_n_songs.sql
|
|
|
|
queryLastNSongs string
|
|
|
|
//go:embed queries/most_popular_songs.sql
|
|
|
|
queryMostPopularSongs string
|
|
|
|
//go:embed queries/most_simultaneous_listeners.sql
|
|
|
|
queryMostSimultaneousListeners string
|
|
|
|
)
|
|
|
|
|
|
|
|
type SQLiteStatistics struct {
|
2024-12-31 00:24:29 +04:00
|
|
|
statistics.SqlStatistics
|
2024-05-09 23:53:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
func New(path string) (statistics.Statistics, error) {
|
|
|
|
db, err := sql.Open("sqlite3", fmt.Sprintf("file:%s?_journal=WAL&_mutex=full", path))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2024-05-15 23:51:10 +04:00
|
|
|
stats := &SQLiteStatistics{
|
2024-12-31 00:24:29 +04:00
|
|
|
SqlStatistics: statistics.SqlStatistics{
|
2024-05-15 23:51:10 +04:00
|
|
|
Db: db, DbDateFormat: dbDateFormat}}
|
2024-05-09 23:53:33 +04:00
|
|
|
|
2024-05-15 23:51:10 +04:00
|
|
|
db.Exec("PRAGMA foreign_keys = ON;")
|
2024-05-09 23:53:33 +04:00
|
|
|
|
2024-05-15 23:51:10 +04:00
|
|
|
_, err = db.Exec(querySchema)
|
2024-05-09 23:53:33 +04:00
|
|
|
if err != nil {
|
2024-05-15 23:51:10 +04:00
|
|
|
return nil, errors.Wrap(err,
|
|
|
|
statistics.ErrPrepareStmt{Name: "initial schema"}.Error())
|
2024-05-09 23:53:33 +04:00
|
|
|
}
|
|
|
|
|
2024-12-31 00:24:29 +04:00
|
|
|
stats.SqlStatistics.StmtSongAdd, err = db.Prepare(querySongAdd)
|
2024-05-09 23:53:33 +04:00
|
|
|
if err != nil {
|
2024-05-15 23:51:10 +04:00
|
|
|
return nil, errors.Wrap(err,
|
|
|
|
statistics.ErrPrepareStmt{Name: "song_add"}.Error())
|
2024-05-09 23:53:33 +04:00
|
|
|
}
|
|
|
|
|
2024-12-31 00:24:29 +04:00
|
|
|
stats.SqlStatistics.StmtHistoryAdd, err = db.Prepare(queryHistoryAdd)
|
2024-05-09 23:53:33 +04:00
|
|
|
if err != nil {
|
2024-05-15 23:51:10 +04:00
|
|
|
return nil, errors.Wrap(err,
|
|
|
|
statistics.ErrPrepareStmt{Name: "history_add"}.Error())
|
2024-05-09 23:53:33 +04:00
|
|
|
}
|
|
|
|
|
2024-12-31 00:24:29 +04:00
|
|
|
stats.SqlStatistics.StmtLastNSongs, err = db.Prepare(queryLastNSongs)
|
2024-05-09 23:53:33 +04:00
|
|
|
if err != nil {
|
2024-05-15 23:51:10 +04:00
|
|
|
return nil, errors.Wrap(err,
|
|
|
|
statistics.ErrPrepareStmt{Name: "last N songs"}.Error())
|
2024-05-09 23:53:33 +04:00
|
|
|
}
|
|
|
|
|
2024-12-31 00:24:29 +04:00
|
|
|
stats.SqlStatistics.StmtMostPopularSongs, err = db.Prepare(queryMostPopularSongs)
|
2024-05-15 23:51:10 +04:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err,
|
|
|
|
statistics.ErrPrepareStmt{Name: "most popular song"}.Error())
|
2024-05-09 23:53:33 +04:00
|
|
|
}
|
|
|
|
|
2024-12-31 00:24:29 +04:00
|
|
|
stats.SqlStatistics.StmtMostSimultaneousListeners, err = db.Prepare(queryMostSimultaneousListeners)
|
2024-05-15 23:51:10 +04:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err,
|
|
|
|
statistics.ErrPrepareStmt{Name: "most simultaneous listeners"}.Error())
|
2024-05-09 23:53:33 +04:00
|
|
|
}
|
|
|
|
|
2024-05-15 23:51:10 +04:00
|
|
|
return stats, nil
|
2024-05-09 23:53:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *SQLiteStatistics) Close() error {
|
2024-12-31 00:24:29 +04:00
|
|
|
return s.SqlStatistics.Close()
|
2024-05-09 23:53:33 +04:00
|
|
|
}
|