2018-03-08 03:59:47 +00:00
|
|
|
package counters
|
|
|
|
|
2019-07-28 04:01:33 +00:00
|
|
|
import (
|
|
|
|
"database/sql"
|
|
|
|
|
|
|
|
c "github.com/Azareal/Gosora/common"
|
|
|
|
qgen "github.com/Azareal/Gosora/query_gen"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
2018-03-08 03:59:47 +00:00
|
|
|
|
|
|
|
var LangViewCounter *DefaultLangViewCounter
|
|
|
|
|
|
|
|
var langCodes = []string{
|
|
|
|
"unknown",
|
2018-03-17 08:16:43 +00:00
|
|
|
"none",
|
2018-03-08 03:59:47 +00:00
|
|
|
"af",
|
|
|
|
"ar",
|
|
|
|
"az",
|
|
|
|
"be",
|
|
|
|
"bg",
|
|
|
|
"bs",
|
|
|
|
"ca",
|
|
|
|
"cs",
|
|
|
|
"cy",
|
|
|
|
"da",
|
|
|
|
"de",
|
|
|
|
"dv",
|
|
|
|
"el",
|
|
|
|
"en",
|
|
|
|
"eo",
|
|
|
|
"es",
|
|
|
|
"et",
|
|
|
|
"eu",
|
|
|
|
"fa",
|
|
|
|
"fi",
|
|
|
|
"fo",
|
|
|
|
"fr",
|
|
|
|
"gl",
|
|
|
|
"gu",
|
|
|
|
"he",
|
|
|
|
"hi",
|
|
|
|
"hr",
|
|
|
|
"hu",
|
|
|
|
"hy",
|
|
|
|
"id",
|
|
|
|
"is",
|
|
|
|
"it",
|
|
|
|
"ja",
|
|
|
|
"ka",
|
|
|
|
"kk",
|
|
|
|
"kn",
|
|
|
|
"ko",
|
|
|
|
"kok",
|
|
|
|
"ky",
|
|
|
|
"lt",
|
|
|
|
"lv",
|
|
|
|
"mi",
|
|
|
|
"mk",
|
|
|
|
"mn",
|
|
|
|
"mr",
|
|
|
|
"ms",
|
|
|
|
"mt",
|
|
|
|
"nb",
|
|
|
|
"nl",
|
|
|
|
"nn",
|
|
|
|
"ns",
|
|
|
|
"pa",
|
|
|
|
"pl",
|
|
|
|
"ps",
|
|
|
|
"pt",
|
|
|
|
"qu",
|
|
|
|
"ro",
|
|
|
|
"ru",
|
|
|
|
"sa",
|
|
|
|
"se",
|
|
|
|
"sk",
|
|
|
|
"sl",
|
|
|
|
"sq",
|
|
|
|
"sr",
|
|
|
|
"sv",
|
|
|
|
"sw",
|
|
|
|
"syr",
|
|
|
|
"ta",
|
|
|
|
"te",
|
|
|
|
"th",
|
|
|
|
"tl",
|
|
|
|
"tn",
|
|
|
|
"tr",
|
|
|
|
"tt",
|
|
|
|
"ts",
|
|
|
|
"uk",
|
|
|
|
"ur",
|
|
|
|
"uz",
|
|
|
|
"vi",
|
|
|
|
"xh",
|
|
|
|
"zh",
|
|
|
|
"zu",
|
|
|
|
}
|
|
|
|
|
|
|
|
type DefaultLangViewCounter struct {
|
|
|
|
buckets []*RWMutexCounterBucket //[OSID]count
|
|
|
|
codesToIndices map[string]int
|
|
|
|
|
2019-03-21 22:59:41 +00:00
|
|
|
insert *sql.Stmt
|
|
|
|
}
|
2018-03-08 03:59:47 +00:00
|
|
|
|
2019-03-21 22:59:41 +00:00
|
|
|
func NewDefaultLangViewCounter(acc *qgen.Accumulator) (*DefaultLangViewCounter, error) {
|
2019-07-28 04:01:33 +00:00
|
|
|
langBuckets := make([]*RWMutexCounterBucket, len(langCodes))
|
2018-03-08 03:59:47 +00:00
|
|
|
for bucketID, _ := range langBuckets {
|
|
|
|
langBuckets[bucketID] = &RWMutexCounterBucket{counter: 0}
|
|
|
|
}
|
2019-07-28 04:01:33 +00:00
|
|
|
codesToIndices := make(map[string]int, len(langCodes))
|
2018-03-08 03:59:47 +00:00
|
|
|
for index, code := range langCodes {
|
|
|
|
codesToIndices[code] = index
|
|
|
|
}
|
|
|
|
|
2019-07-28 04:01:33 +00:00
|
|
|
co := &DefaultLangViewCounter{
|
2018-03-08 03:59:47 +00:00
|
|
|
buckets: langBuckets,
|
|
|
|
codesToIndices: codesToIndices,
|
|
|
|
insert: acc.Insert("viewchunks_langs").Columns("count, createdAt, lang").Fields("?,UTC_TIMESTAMP(),?").Prepare(),
|
|
|
|
}
|
|
|
|
|
2019-07-28 04:01:33 +00:00
|
|
|
c.AddScheduledFifteenMinuteTask(co.Tick)
|
|
|
|
//c.AddScheduledSecondTask(co.Tick)
|
|
|
|
c.AddShutdownTask(co.Tick)
|
|
|
|
return co, acc.FirstError()
|
2018-03-08 03:59:47 +00:00
|
|
|
}
|
|
|
|
|
2019-07-28 04:01:33 +00:00
|
|
|
func (co *DefaultLangViewCounter) Tick() error {
|
|
|
|
for id, bucket := range co.buckets {
|
2018-03-08 03:59:47 +00:00
|
|
|
var count int
|
|
|
|
bucket.RLock()
|
|
|
|
count = bucket.counter
|
|
|
|
bucket.counter = 0 // TODO: Add a SetZero method to reduce the amount of duplicate code between the OS and agent counters?
|
|
|
|
bucket.RUnlock()
|
|
|
|
|
2019-07-28 04:01:33 +00:00
|
|
|
err := co.insertChunk(count, id) // TODO: Bulk insert for speed?
|
2018-03-08 03:59:47 +00:00
|
|
|
if err != nil {
|
2019-07-28 04:01:33 +00:00
|
|
|
return errors.Wrap(errors.WithStack(err), "langview counter")
|
2018-03-08 03:59:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-07-28 04:01:33 +00:00
|
|
|
func (co *DefaultLangViewCounter) insertChunk(count int, id int) error {
|
2018-03-08 03:59:47 +00:00
|
|
|
if count == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
2019-07-28 04:01:33 +00:00
|
|
|
langCode := langCodes[id]
|
|
|
|
c.DebugLogf("Inserting a vchunk with a count of %d for lang %s (%d)", count, langCode, id)
|
|
|
|
_, err := co.insert.Exec(count, langCode)
|
2018-03-08 03:59:47 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-07-28 04:01:33 +00:00
|
|
|
func (co *DefaultLangViewCounter) Bump(langCode string) (validCode bool) {
|
2019-02-23 06:29:19 +00:00
|
|
|
validCode = true
|
2019-07-28 04:01:33 +00:00
|
|
|
id, ok := co.codesToIndices[langCode]
|
2018-03-08 03:59:47 +00:00
|
|
|
if !ok {
|
|
|
|
// TODO: Tell the caller that the code's invalid
|
|
|
|
id = 0 // Unknown
|
2019-02-23 06:29:19 +00:00
|
|
|
validCode = false
|
2018-03-08 03:59:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Test this check
|
2019-07-28 04:01:33 +00:00
|
|
|
c.DebugDetail("co.buckets[", id, "]: ", co.buckets[id])
|
|
|
|
if len(co.buckets) <= id || id < 0 {
|
2019-02-23 06:29:19 +00:00
|
|
|
return validCode
|
2018-03-08 03:59:47 +00:00
|
|
|
}
|
2019-07-28 04:01:33 +00:00
|
|
|
co.buckets[id].Lock()
|
|
|
|
co.buckets[id].counter++
|
|
|
|
co.buckets[id].Unlock()
|
2019-02-23 06:29:19 +00:00
|
|
|
|
|
|
|
return validCode
|
2018-03-08 03:59:47 +00:00
|
|
|
}
|