2018-05-16 10:46:14 +00:00
|
|
|
package common
|
|
|
|
|
2018-12-14 04:08:53 +00:00
|
|
|
import (
|
2022-02-21 03:32:53 +00:00
|
|
|
"database/sql"
|
|
|
|
"time"
|
2018-12-14 04:08:53 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
qgen "github.com/Azareal/Gosora/query_gen"
|
2018-12-14 04:08:53 +00:00
|
|
|
)
|
2018-05-16 10:46:14 +00:00
|
|
|
|
|
|
|
var RegLogs RegLogStore
|
2018-12-17 04:58:55 +00:00
|
|
|
var LoginLogs LoginLogStore
|
2018-05-16 10:46:14 +00:00
|
|
|
|
|
|
|
type RegLogItem struct {
|
2022-02-21 03:32:53 +00:00
|
|
|
ID int
|
|
|
|
Username string
|
|
|
|
Email string
|
|
|
|
FailureReason string
|
|
|
|
Success bool
|
|
|
|
IP string
|
|
|
|
DoneAt string
|
2018-05-16 10:46:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type RegLogStmts struct {
|
2022-02-21 03:32:53 +00:00
|
|
|
update *sql.Stmt
|
|
|
|
create *sql.Stmt
|
2018-05-16 10:46:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var regLogStmts RegLogStmts
|
|
|
|
|
|
|
|
func init() {
|
2022-02-21 03:32:53 +00:00
|
|
|
DbInits.Add(func(acc *qgen.Accumulator) error {
|
|
|
|
rl := "registration_logs"
|
|
|
|
regLogStmts = RegLogStmts{
|
|
|
|
update: acc.Update(rl).Set("username=?,email=?,failureReason=?,success=?,doneAt=?").Where("rlid=?").Prepare(),
|
|
|
|
create: acc.Insert(rl).Columns("username,email,failureReason,success,ipaddress,doneAt").Fields("?,?,?,?,?,UTC_TIMESTAMP()").Prepare(),
|
|
|
|
}
|
|
|
|
return acc.FirstError()
|
|
|
|
})
|
2018-05-16 10:46:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Reload this item in the store, probably doesn't matter right now, but it might when we start caching this stuff in memory
|
|
|
|
// ! Retroactive updates of date are not permitted for integrity reasons
|
2021-04-27 10:20:26 +00:00
|
|
|
// TODO: Do we even use this anymore or can we just make the logs immutable (except for deletes) for simplicity sake?
|
2019-08-31 22:34:43 +00:00
|
|
|
func (l *RegLogItem) Commit() error {
|
2022-02-21 03:32:53 +00:00
|
|
|
_, e := regLogStmts.update.Exec(l.Username, l.Email, l.FailureReason, l.Success, l.DoneAt, l.ID)
|
|
|
|
return e
|
2018-05-16 10:46:14 +00:00
|
|
|
}
|
|
|
|
|
2021-04-27 10:20:26 +00:00
|
|
|
func (l *RegLogItem) Create() (id int, e error) {
|
2022-02-21 03:32:53 +00:00
|
|
|
id, e = Createf(regLogStmts.create, l.Username, l.Email, l.FailureReason, l.Success, l.IP)
|
|
|
|
l.ID = id
|
|
|
|
return l.ID, e
|
2018-05-16 10:46:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type RegLogStore interface {
|
2022-02-21 03:32:53 +00:00
|
|
|
Count() (count int)
|
|
|
|
GetOffset(offset, perPage int) (logs []RegLogItem, err error)
|
|
|
|
Purge() error
|
2021-04-27 10:20:26 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
DeleteOlderThanDays(days int) error
|
2018-05-16 10:46:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type SQLRegLogStore struct {
|
2022-02-21 03:32:53 +00:00
|
|
|
count *sql.Stmt
|
|
|
|
getOffset *sql.Stmt
|
|
|
|
purge *sql.Stmt
|
2021-04-27 10:20:26 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
deleteOlderThanDays *sql.Stmt
|
2018-05-16 10:46:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewRegLogStore(acc *qgen.Accumulator) (*SQLRegLogStore, error) {
|
2022-02-21 03:32:53 +00:00
|
|
|
rl := "registration_logs"
|
|
|
|
return &SQLRegLogStore{
|
|
|
|
count: acc.Count(rl).Prepare(),
|
|
|
|
getOffset: acc.Select(rl).Columns("rlid,username,email,failureReason,success,ipaddress,doneAt").Orderby("doneAt DESC").Limit("?,?").Prepare(),
|
|
|
|
purge: acc.Purge(rl),
|
2021-04-27 10:20:26 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
deleteOlderThanDays: acc.Delete(rl).DateOlderThanQ("doneAt", "day").Prepare(),
|
|
|
|
}, acc.FirstError()
|
2018-05-16 10:46:14 +00:00
|
|
|
}
|
|
|
|
|
2019-06-01 12:31:48 +00:00
|
|
|
func (s *SQLRegLogStore) Count() (count int) {
|
2022-02-21 03:32:53 +00:00
|
|
|
return Count(s.count)
|
2018-05-16 10:46:14 +00:00
|
|
|
}
|
|
|
|
|
2021-04-27 10:20:26 +00:00
|
|
|
func (s *SQLRegLogStore) GetOffset(offset, perPage int) (logs []RegLogItem, e error) {
|
2022-02-21 03:32:53 +00:00
|
|
|
rows, e := s.getOffset.Query(offset, perPage)
|
|
|
|
if e != nil {
|
|
|
|
return logs, e
|
|
|
|
}
|
|
|
|
defer rows.Close()
|
|
|
|
|
|
|
|
for rows.Next() {
|
|
|
|
var l RegLogItem
|
|
|
|
var doneAt time.Time
|
|
|
|
e := rows.Scan(&l.ID, &l.Username, &l.Email, &l.FailureReason, &l.Success, &l.IP, &doneAt)
|
|
|
|
if e != nil {
|
|
|
|
return logs, e
|
|
|
|
}
|
|
|
|
l.DoneAt = doneAt.Format("2006-01-02 15:04:05")
|
|
|
|
logs = append(logs, l)
|
|
|
|
}
|
|
|
|
return logs, rows.Err()
|
2018-05-16 10:46:14 +00:00
|
|
|
}
|
2018-12-17 04:58:55 +00:00
|
|
|
|
2021-04-27 10:20:26 +00:00
|
|
|
func (s *SQLRegLogStore) DeleteOlderThanDays(days int) error {
|
2022-02-21 03:32:53 +00:00
|
|
|
_, e := s.deleteOlderThanDays.Exec(days)
|
|
|
|
return e
|
2021-04-27 10:20:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Delete all registration logs
|
|
|
|
func (s *SQLRegLogStore) Purge() error {
|
2022-02-21 03:32:53 +00:00
|
|
|
_, e := s.purge.Exec()
|
|
|
|
return e
|
2021-04-27 10:20:26 +00:00
|
|
|
}
|
|
|
|
|
2018-12-17 04:58:55 +00:00
|
|
|
type LoginLogItem struct {
|
2022-02-21 03:32:53 +00:00
|
|
|
ID int
|
|
|
|
UID int
|
|
|
|
Success bool
|
|
|
|
IP string
|
|
|
|
DoneAt string
|
2018-12-17 04:58:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type LoginLogStmts struct {
|
2022-02-21 03:32:53 +00:00
|
|
|
update *sql.Stmt
|
|
|
|
create *sql.Stmt
|
2018-12-17 04:58:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var loginLogStmts LoginLogStmts
|
|
|
|
|
|
|
|
func init() {
|
2022-02-21 03:32:53 +00:00
|
|
|
DbInits.Add(func(acc *qgen.Accumulator) error {
|
|
|
|
ll := "login_logs"
|
|
|
|
loginLogStmts = LoginLogStmts{
|
|
|
|
update: acc.Update(ll).Set("uid=?,success=?,doneAt=?").Where("lid=?").Prepare(),
|
|
|
|
create: acc.Insert(ll).Columns("uid,success,ipaddress,doneAt").Fields("?,?,?,UTC_TIMESTAMP()").Prepare(),
|
|
|
|
}
|
|
|
|
return acc.FirstError()
|
|
|
|
})
|
2018-12-17 04:58:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Reload this item in the store, probably doesn't matter right now, but it might when we start caching this stuff in memory
|
|
|
|
// ! Retroactive updates of date are not permitted for integrity reasons
|
2019-08-31 22:34:43 +00:00
|
|
|
func (l *LoginLogItem) Commit() error {
|
2022-02-21 03:32:53 +00:00
|
|
|
_, e := loginLogStmts.update.Exec(l.UID, l.Success, l.DoneAt, l.ID)
|
|
|
|
return e
|
2018-12-17 04:58:55 +00:00
|
|
|
}
|
|
|
|
|
2021-04-27 10:20:26 +00:00
|
|
|
func (l *LoginLogItem) Create() (id int, e error) {
|
2022-02-21 03:32:53 +00:00
|
|
|
res, e := loginLogStmts.create.Exec(l.UID, l.Success, l.IP)
|
|
|
|
if e != nil {
|
|
|
|
return 0, e
|
|
|
|
}
|
|
|
|
id64, e := res.LastInsertId()
|
|
|
|
l.ID = int(id64)
|
|
|
|
return l.ID, e
|
2018-12-17 04:58:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type LoginLogStore interface {
|
2022-02-21 03:32:53 +00:00
|
|
|
Count() (count int)
|
|
|
|
CountUser(uid int) (count int)
|
|
|
|
GetOffset(uid, offset, perPage int) (logs []LoginLogItem, err error)
|
|
|
|
Purge() error
|
2021-04-27 10:20:26 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
DeleteOlderThanDays(days int) error
|
2018-12-17 04:58:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type SQLLoginLogStore struct {
|
2022-02-21 03:32:53 +00:00
|
|
|
count *sql.Stmt
|
|
|
|
countForUser *sql.Stmt
|
|
|
|
getOffsetByUser *sql.Stmt
|
|
|
|
purge *sql.Stmt
|
2021-04-27 10:20:26 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
deleteOlderThanDays *sql.Stmt
|
2018-12-17 04:58:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewLoginLogStore(acc *qgen.Accumulator) (*SQLLoginLogStore, error) {
|
2022-02-21 03:32:53 +00:00
|
|
|
ll := "login_logs"
|
|
|
|
return &SQLLoginLogStore{
|
|
|
|
count: acc.Count(ll).Prepare(),
|
|
|
|
countForUser: acc.Count(ll).Where("uid=?").Prepare(),
|
|
|
|
getOffsetByUser: acc.Select(ll).Columns("lid,success,ipaddress,doneAt").Where("uid=?").Orderby("doneAt DESC").Limit("?,?").Prepare(),
|
|
|
|
purge: acc.Purge(ll),
|
2021-04-27 10:20:26 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
deleteOlderThanDays: acc.Delete(ll).DateOlderThanQ("doneAt", "day").Prepare(),
|
|
|
|
}, acc.FirstError()
|
2018-12-17 04:58:55 +00:00
|
|
|
}
|
|
|
|
|
2019-06-01 12:31:48 +00:00
|
|
|
func (s *SQLLoginLogStore) Count() (count int) {
|
2022-02-21 03:32:53 +00:00
|
|
|
return Count(s.count)
|
2018-12-17 04:58:55 +00:00
|
|
|
}
|
|
|
|
|
2019-06-01 12:31:48 +00:00
|
|
|
func (s *SQLLoginLogStore) CountUser(uid int) (count int) {
|
2022-02-21 03:32:53 +00:00
|
|
|
return Countf(s.countForUser, uid)
|
2019-02-24 01:29:06 +00:00
|
|
|
}
|
|
|
|
|
2021-04-27 10:20:26 +00:00
|
|
|
func (s *SQLLoginLogStore) GetOffset(uid, offset, perPage int) (logs []LoginLogItem, e error) {
|
2022-02-21 03:32:53 +00:00
|
|
|
rows, e := s.getOffsetByUser.Query(uid, offset, perPage)
|
|
|
|
if e != nil {
|
|
|
|
return logs, e
|
|
|
|
}
|
|
|
|
defer rows.Close()
|
|
|
|
|
|
|
|
for rows.Next() {
|
|
|
|
l := LoginLogItem{UID: uid}
|
|
|
|
var doneAt time.Time
|
|
|
|
e := rows.Scan(&l.ID, &l.Success, &l.IP, &doneAt)
|
|
|
|
if e != nil {
|
|
|
|
return logs, e
|
|
|
|
}
|
|
|
|
l.DoneAt = doneAt.Format("2006-01-02 15:04:05")
|
|
|
|
logs = append(logs, l)
|
|
|
|
}
|
|
|
|
return logs, rows.Err()
|
2018-12-17 04:58:55 +00:00
|
|
|
}
|
2021-04-27 10:20:26 +00:00
|
|
|
|
|
|
|
func (s *SQLLoginLogStore) DeleteOlderThanDays(days int) error {
|
2022-02-21 03:32:53 +00:00
|
|
|
_, e := s.deleteOlderThanDays.Exec(days)
|
|
|
|
return e
|
2021-04-27 10:20:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Delete all login logs
|
|
|
|
func (s *SQLLoginLogStore) Purge() error {
|
2022-02-21 03:32:53 +00:00
|
|
|
_, e := s.purge.Exec()
|
|
|
|
return e
|
2021-04-27 10:20:26 +00:00
|
|
|
}
|