jlelse
/
kis3
Archived
1
Fork 0
This repository has been archived on 2021-01-04. You can view files and clone it, but cannot push or open issues or pull requests.
kis3/database.go

232 lines
6.9 KiB
Go

package main
import (
"database/sql"
"fmt"
"github.com/gobuffalo/packr/v2"
_ "github.com/mattn/go-sqlite3"
"github.com/mssola/user_agent"
"github.com/rubenv/sql-migrate"
"net/url"
"os"
"path/filepath"
"strings"
)
type Database struct {
sqlDB *sql.DB
}
func initDatabase() (database *Database, e error) {
database = &Database{}
if _, err := os.Stat(appConfig.DbPath); os.IsNotExist(err) {
_ = os.MkdirAll(filepath.Dir(appConfig.DbPath), os.ModePerm)
}
database.sqlDB, e = sql.Open("sqlite3", appConfig.DbPath)
if e != nil {
return
}
e = migrateDatabase(database.sqlDB)
return
}
func migrateDatabase(database *sql.DB) (e error) {
migrations := &migrate.PackrMigrationSource{
Box: packr.New("migrations", "migrations"),
}
_, e = migrate.Exec(database, "sqlite3", migrations, migrate.Up)
return
}
// Tracking
func (db *Database) trackView(urlString string, ref string, ua string) {
if len(urlString) == 0 {
// Don't track empty urls
return
}
if ref != "" {
// Clean referrer and just keep the hostname for more privacy
parsedRef, _ := url.Parse(ref)
ref = parsedRef.Hostname()
}
if ua != "" {
// Parse Useragent
uaName, uaVersion := user_agent.New(ua).Browser()
ua = uaName + " " + uaVersion
}
_, e := db.sqlDB.Exec("insert into views(url, ref, useragent) values(:url, :ref, :ua)", sql.Named("url", urlString), sql.Named("ref", ref), sql.Named("ua", ua))
if e != nil {
fmt.Println("Inserting into DB failed:", e)
}
}
// Requesting
type View int
const (
PAGES View = iota + 1
REFERRERS
USERAGENTS
USERAGENTNAMES
HOURS
DAYS
WEEKS
MONTHS
ALLHOURS
ALLDAYS
)
type ViewsRequest struct {
view View
from string
to string
url string
ref string
ua string
orderrow string
order string
}
type RequestResultRow struct {
First string `json:"first"`
Second int `json:"second"`
}
func (db *Database) request(request *ViewsRequest) (resultRows []*RequestResultRow, e error) {
statement, parameters := request.buildStatement()
namedArgs := make([]interface{}, len(parameters))
for i, v := range parameters {
namedArgs[i] = v
}
rows, e := db.sqlDB.Query(statement, namedArgs...)
if e != nil {
return
} else {
resultRows = []*RequestResultRow{}
for rows.Next() {
var first string
var second int
e = rows.Scan(&first, &second)
if e != nil {
_ = rows.Close()
return
}
if first == "" {
first = "Undefined"
}
resultRows = append(resultRows, &RequestResultRow{
First: first,
Second: second,
})
}
return
}
}
func (request *ViewsRequest) buildStatement() (statement string, parameters []sql.NamedArg) {
filters, parameters := request.buildFilter()
if len(filters) > 0 {
filters = " where " + filters + " "
} else {
filters = " "
}
orderrow := "first"
order := "ASC"
if request.orderrow == "second" {
orderrow = "second"
}
if request.order == "DESC" {
order = "DESC"
}
orderstatement := " ORDER BY " + orderrow + " " + order
switch request.view {
case PAGES:
statement = "SELECT url as first, count(*) as second from views" + filters + "group by first" + orderstatement + ";"
case REFERRERS:
statement = "SELECT ref as first, count(*) as second from views" + filters + "group by first" + orderstatement + ";"
case USERAGENTS:
statement = "SELECT useragent as first, count(*) as second from views" + filters + "group by first" + orderstatement + ";"
case USERAGENTNAMES:
statement = "SELECT substr(useragent, 1, pos-1) as first, COUNT(*) as second from (SELECT *, instr(useragent,' ') AS pos FROM views)" + filters + "group by first" + orderstatement + ";"
case ALLHOURS:
statement = "WITH RECURSIVE hours(hour) AS ( VALUES (datetime(strftime('%Y-%m-%dT%H:00', (SELECT min(time) from views" + filters + "), 'localtime'))) UNION ALL SELECT datetime(hour, '+1 hour') FROM hours WHERE hour <= strftime('%Y-%m-%d %H', (SELECT max(time) from views" + filters + "), 'localtime') ) SELECT strftime('%Y-%m-%d %H', hours.hour) as first, COUNT(time) as second FROM hours LEFT OUTER JOIN (SELECT time from views" + filters + ") ON strftime('%Y-%m-%d %H', hours.hour) = strftime('%Y-%m-%d %H', time, 'localtime') GROUP BY first" + orderstatement + ";"
case ALLDAYS:
statement = "WITH RECURSIVE days(day) AS ( VALUES (datetime((SELECT min(time) from views" + filters + "), 'localtime', 'start of day')) UNION ALL SELECT datetime(day, '+1 day') FROM days WHERE day <= date((SELECT max(time) from views" + filters + "), 'localtime') ) SELECT strftime('%Y-%m-%d', days.day) as first, COUNT(time) as second FROM days LEFT OUTER JOIN (SELECT time from views" + filters + ") ON strftime('%Y-%m-%d', days.day) = strftime('%Y-%m-%d', time, 'localtime') GROUP BY first" + orderstatement + ";"
case HOURS, DAYS, WEEKS, MONTHS:
format := ""
switch request.view {
case HOURS:
format = "%Y-%m-%d %H"
case DAYS:
format = "%Y-%m-%d"
case WEEKS:
format = "%Y-%W"
case MONTHS:
format = "%Y-%m"
}
statement = "SELECT strftime('" + format + "', time, 'localtime') as first, count(*) as second from views" + filters + "group by first" + orderstatement + ";"
}
return
}
// Request filters
func (request *ViewsRequest) buildFilter() (filters string, parameters []sql.NamedArg) {
parameters = []sql.NamedArg{}
var allFilters []string
for _, filter := range []string{
request.buildDateTimeFilter(&parameters),
request.buildUrlFilter(&parameters),
request.buildRefFilter(&parameters),
request.buildUseragentFilter(&parameters),
} {
if len(filter) > 0 {
allFilters = append(allFilters, filter)
}
}
filters = strings.Join(allFilters, " and ")
return
}
func (request *ViewsRequest) buildDateTimeFilter(namedArg *[]sql.NamedArg) (dateTimeFilter string) {
selector := "datetime(time, 'localtime')"
if len(request.from) > 0 && len(request.to) > 0 {
*namedArg = append(*namedArg, sql.Named("from", request.from))
*namedArg = append(*namedArg, sql.Named("to", request.to))
dateTimeFilter = selector + " between :from and :to"
} else if len(request.from) > 0 {
*namedArg = append(*namedArg, sql.Named("from", request.from))
dateTimeFilter = selector + " >= :from"
} else if len(request.to) > 0 {
*namedArg = append(*namedArg, sql.Named("to", request.to))
dateTimeFilter = selector + " <= :to"
}
return
}
func (request *ViewsRequest) buildUrlFilter(namedArg *[]sql.NamedArg) (urlFilter string) {
if len(request.url) > 0 {
*namedArg = append(*namedArg, sql.Named("url", "%"+request.url+"%"))
urlFilter = "url like :url"
}
return
}
func (request *ViewsRequest) buildRefFilter(namedArg *[]sql.NamedArg) (refFilter string) {
if len(request.ref) > 0 {
*namedArg = append(*namedArg, sql.Named("ref", "%"+request.ref+"%"))
refFilter = "ref like :ref"
}
return
}
func (request *ViewsRequest) buildUseragentFilter(namedArg *[]sql.NamedArg) (refFilter string) {
if len(request.ua) > 0 {
*namedArg = append(*namedArg, sql.Named("ua", "%"+request.ua+"%"))
refFilter = "useragent like :ua"
}
return
}