2016-02-10 14:31:53 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
2016-02-13 22:49:48 +00:00
|
|
|
"encoding/json"
|
2016-02-10 14:31:53 +00:00
|
|
|
"log"
|
|
|
|
"strconv"
|
2016-02-13 22:49:48 +00:00
|
|
|
"strings"
|
2016-02-10 14:31:53 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/boltdb/bolt"
|
2016-02-10 16:27:15 +00:00
|
|
|
"github.com/hamcha/clessy/tg"
|
2016-02-10 14:31:53 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
MessageTypeText int = 0
|
|
|
|
MessageTypeAudio int = 1
|
|
|
|
MessageTypePhoto int = 2
|
|
|
|
MessageTypeSticker int = 3
|
|
|
|
MessageTypeVideo int = 4
|
|
|
|
MessageTypeVoice int = 5
|
|
|
|
MessageTypeContact int = 6
|
|
|
|
MessageTypeLocation int = 7
|
2016-02-10 16:27:15 +00:00
|
|
|
MessageTypeDocument int = 8
|
|
|
|
MessageTypeMax int = 9
|
2016-02-10 14:31:53 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type Stats struct {
|
2016-02-12 16:11:33 +00:00
|
|
|
ByUserCount map[string]uint64
|
|
|
|
ByWeekday [7]uint64
|
|
|
|
ByHour [24]uint64
|
|
|
|
ByType [MessageTypeMax]uint64
|
2016-02-13 20:58:05 +00:00
|
|
|
ByDay map[string]uint64
|
2016-02-12 16:11:33 +00:00
|
|
|
TodayDate time.Time
|
|
|
|
Today uint64
|
|
|
|
TotalCount uint64
|
2016-02-10 14:31:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var stats Stats
|
|
|
|
|
2016-02-13 22:49:48 +00:00
|
|
|
type UserCount map[string]uint64
|
|
|
|
|
|
|
|
var words map[string]UserCount
|
|
|
|
|
2016-02-10 14:31:53 +00:00
|
|
|
func MakeUint(bval []byte, bucketName string, key string) uint64 {
|
|
|
|
if bval != nil {
|
|
|
|
intval, bts := binary.Uvarint(bval)
|
|
|
|
if bts > 0 {
|
|
|
|
return intval
|
|
|
|
} else {
|
|
|
|
log.Printf("[%s] Value of key \"%s\" is NaN: %v\r\n", bucketName, key, bval)
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log.Printf("[%s] Key \"%s\" does not exist, set to 0\n", bucketName, key)
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-10 16:27:15 +00:00
|
|
|
func PutUint(value uint64) []byte {
|
|
|
|
bytes := make([]byte, 10)
|
|
|
|
n := binary.PutUvarint(bytes, value)
|
|
|
|
return bytes[:n]
|
|
|
|
}
|
|
|
|
|
2016-02-10 14:31:53 +00:00
|
|
|
func loadStats() {
|
|
|
|
// Load today
|
|
|
|
stats.TodayDate = time.Now()
|
|
|
|
|
2016-02-10 16:27:15 +00:00
|
|
|
err := db.Update(func(tx *bolt.Tx) error {
|
2016-02-10 14:31:53 +00:00
|
|
|
b, err := tx.CreateBucketIfNotExists([]byte("global"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load total messages counter
|
2016-02-12 16:11:33 +00:00
|
|
|
stats.TotalCount = MakeUint(b.Get([]byte("count")), "global", "count")
|
2016-02-10 16:27:15 +00:00
|
|
|
|
2016-02-10 14:31:53 +00:00
|
|
|
// Load hour counters
|
|
|
|
b, err = tx.CreateBucketIfNotExists([]byte("hour"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < 24; i++ {
|
2016-02-12 16:11:33 +00:00
|
|
|
stats.ByHour[i] = MakeUint(b.Get([]byte{byte(i)}), "hour", strconv.Itoa(i))
|
2016-02-10 14:31:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Load weekday counters
|
|
|
|
b, err = tx.CreateBucketIfNotExists([]byte("weekday"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < 7; i++ {
|
2016-02-12 16:11:33 +00:00
|
|
|
stats.ByWeekday[i] = MakeUint(b.Get([]byte{byte(i)}), "weekday", strconv.Itoa(i))
|
2016-02-10 14:31:53 +00:00
|
|
|
}
|
|
|
|
|
2016-02-13 20:58:05 +00:00
|
|
|
// Load day counters
|
2016-02-13 20:59:39 +00:00
|
|
|
stats.ByDay = make(map[string]uint64)
|
2016-02-10 14:31:53 +00:00
|
|
|
b, err = tx.CreateBucketIfNotExists([]byte("date"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2016-02-13 20:58:05 +00:00
|
|
|
b.ForEach(func(day, messages []byte) error {
|
|
|
|
stats.ByDay[string(day)] = MakeUint(messages, "date", string(day))
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
2016-02-10 14:31:53 +00:00
|
|
|
todayKey := stats.TodayDate.Format("2006-1-2")
|
2016-02-12 16:11:33 +00:00
|
|
|
stats.Today = MakeUint(b.Get([]byte(todayKey)), "date", todayKey)
|
2016-02-10 14:31:53 +00:00
|
|
|
|
|
|
|
// Load user counters
|
|
|
|
stats.ByUserCount = make(map[string]uint64)
|
|
|
|
b, err = tx.CreateBucketIfNotExists([]byte("users-count"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.ForEach(func(user, messages []byte) error {
|
|
|
|
stats.ByUserCount[string(user)] = MakeUint(messages, "users-count", string(user))
|
2016-02-10 16:27:15 +00:00
|
|
|
return nil
|
2016-02-10 14:31:53 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
// Load type counters
|
|
|
|
b, err = tx.CreateBucketIfNotExists([]byte("types"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for i := 0; i < MessageTypeMax; i++ {
|
2016-02-12 16:11:33 +00:00
|
|
|
stats.ByType[i] = MakeUint(b.Get([]byte{byte(i)}), "types", strconv.Itoa(i))
|
2016-02-10 14:31:53 +00:00
|
|
|
}
|
|
|
|
|
2016-02-13 22:49:48 +00:00
|
|
|
// Load dictionary
|
|
|
|
b, err = tx.CreateBucketIfNotExists([]byte("words"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
words = make(map[string]UserCount)
|
|
|
|
b.ForEach(func(word, ucount []byte) error {
|
|
|
|
var val UserCount
|
|
|
|
err := json.Unmarshal(ucount, &val)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
words[string(word)] = val
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
2016-02-10 14:31:53 +00:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
assert(err)
|
|
|
|
}
|
2016-02-10 16:27:15 +00:00
|
|
|
|
|
|
|
func updateDate() {
|
2016-02-13 20:58:05 +00:00
|
|
|
dateKey := stats.TodayDate.Format("2006-1-2")
|
2016-02-10 16:27:15 +00:00
|
|
|
err := db.Update(func(tx *bolt.Tx) error {
|
|
|
|
b := tx.Bucket([]byte("date"))
|
2016-02-13 20:58:05 +00:00
|
|
|
err := b.Put([]byte(dateKey), PutUint(stats.Today))
|
2016-02-10 16:27:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Println("[updateDate] Couldn't save last day stats: " + err.Error())
|
|
|
|
}
|
2016-02-13 20:58:05 +00:00
|
|
|
stats.ByDay[dateKey] = stats.Today
|
2016-02-10 16:27:15 +00:00
|
|
|
stats.TodayDate = time.Now()
|
|
|
|
stats.Today = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func updateMean(currentMean, meanCount, newValue uint64) uint64 {
|
|
|
|
return ((currentMean * meanCount) + newValue) / (meanCount + 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func updateStats(message tg.APIMessage) {
|
|
|
|
//
|
|
|
|
// Local update
|
|
|
|
//
|
|
|
|
|
|
|
|
// DB Update flags
|
|
|
|
updatetype := 0
|
|
|
|
|
|
|
|
// Update total count
|
|
|
|
stats.TotalCount++
|
|
|
|
|
|
|
|
// Update individual user's count
|
|
|
|
username := message.User.Username
|
|
|
|
val, exists := stats.ByUserCount[username]
|
|
|
|
if !exists {
|
|
|
|
val = 0
|
|
|
|
}
|
|
|
|
stats.ByUserCount[username] = val + 1
|
|
|
|
|
|
|
|
// Update time counters
|
|
|
|
now := time.Now()
|
|
|
|
hour := now.Hour()
|
|
|
|
wday := now.Weekday()
|
|
|
|
stats.ByHour[hour]++
|
|
|
|
stats.ByWeekday[wday]++
|
|
|
|
|
|
|
|
// Check for day reset
|
|
|
|
if now.Day() != stats.TodayDate.Day() {
|
|
|
|
updateDate()
|
|
|
|
}
|
|
|
|
stats.Today++
|
|
|
|
|
|
|
|
// Text message
|
|
|
|
if message.Text != nil {
|
|
|
|
stats.ByType[MessageTypeText]++
|
|
|
|
updatetype = MessageTypeText
|
2016-02-13 22:49:48 +00:00
|
|
|
|
|
|
|
// Process words
|
|
|
|
processWords(message)
|
2016-02-10 16:27:15 +00:00
|
|
|
}
|
|
|
|
// Audio message
|
|
|
|
if message.Audio != nil {
|
|
|
|
stats.ByType[MessageTypeAudio]++
|
|
|
|
updatetype = MessageTypeAudio
|
|
|
|
}
|
|
|
|
// Photo
|
|
|
|
if message.Photo != nil {
|
|
|
|
stats.ByType[MessageTypePhoto]++
|
|
|
|
updatetype = MessageTypePhoto
|
|
|
|
}
|
|
|
|
// Sticker
|
|
|
|
if message.Sticker != nil {
|
|
|
|
stats.ByType[MessageTypeSticker]++
|
|
|
|
updatetype = MessageTypeSticker
|
|
|
|
}
|
|
|
|
// Video
|
|
|
|
if message.Video != nil {
|
|
|
|
stats.ByType[MessageTypeVideo]++
|
|
|
|
updatetype = MessageTypeVideo
|
|
|
|
}
|
|
|
|
// Voice message
|
|
|
|
if message.Voice != nil {
|
|
|
|
stats.ByType[MessageTypeVoice]++
|
|
|
|
updatetype = MessageTypeVoice
|
|
|
|
}
|
|
|
|
// Contact
|
|
|
|
if message.Contact != nil {
|
|
|
|
stats.ByType[MessageTypeContact]++
|
|
|
|
updatetype = MessageTypeContact
|
|
|
|
}
|
|
|
|
// Location
|
|
|
|
if message.Location != nil {
|
|
|
|
stats.ByType[MessageTypeLocation]++
|
|
|
|
updatetype = MessageTypeLocation
|
|
|
|
}
|
|
|
|
// Document
|
|
|
|
if message.Document != nil {
|
|
|
|
stats.ByType[MessageTypeDocument]++
|
|
|
|
updatetype = MessageTypeDocument
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// DB Update
|
|
|
|
//
|
|
|
|
|
|
|
|
err := db.Update(func(tx *bolt.Tx) error {
|
|
|
|
// Update total counters
|
|
|
|
b := tx.Bucket([]byte("global"))
|
|
|
|
|
|
|
|
err := b.Put([]byte("count"), PutUint(stats.TotalCount))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update time counters
|
|
|
|
b = tx.Bucket([]byte("hour"))
|
|
|
|
err = b.Put([]byte{byte(hour)}, PutUint(stats.ByHour[hour]))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
b = tx.Bucket([]byte("weekday"))
|
2016-02-11 15:45:42 +00:00
|
|
|
err = b.Put([]byte{byte(wday)}, PutUint(stats.ByWeekday[wday]))
|
2016-02-10 16:27:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
b = tx.Bucket([]byte("date"))
|
|
|
|
todayKey := stats.TodayDate.Format("2006-1-2")
|
|
|
|
err = b.Put([]byte(todayKey), PutUint(stats.Today))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update user counters
|
|
|
|
b = tx.Bucket([]byte("users-count"))
|
|
|
|
err = b.Put([]byte(username), PutUint(stats.ByUserCount[username]))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update type counter
|
|
|
|
b = tx.Bucket([]byte("types"))
|
|
|
|
err = b.Put([]byte{byte(updatetype)}, PutUint(stats.ByType[updatetype]))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Println("[updateStats] Got error while updating DB: " + err.Error())
|
|
|
|
}
|
|
|
|
}
|
2016-02-13 22:49:48 +00:00
|
|
|
|
|
|
|
func processWords(message tg.APIMessage) {
|
|
|
|
if len(*(message).Text) < 3 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
wordList := strings.Split(*(message.Text), " ")
|
|
|
|
err := db.Update(func(tx *bolt.Tx) error {
|
|
|
|
b := tx.Bucket([]byte("words"))
|
|
|
|
for _, word := range wordList {
|
2016-02-13 23:07:17 +00:00
|
|
|
if len(word) < 3 {
|
2016-02-13 23:07:50 +00:00
|
|
|
continue
|
2016-02-13 23:07:17 +00:00
|
|
|
}
|
|
|
|
|
2016-02-13 23:14:51 +00:00
|
|
|
word = strings.ToLower(word)
|
|
|
|
|
|
|
|
if strings.HasPrefix(word, "http") {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2016-02-13 23:05:32 +00:00
|
|
|
word = strings.Trim(word, " ?!.,:;/-_()[]{}'\"+=*^\n")
|
2016-02-13 22:59:24 +00:00
|
|
|
count, ok := words[word]
|
|
|
|
if !ok {
|
2016-02-13 22:56:43 +00:00
|
|
|
count = make(UserCount)
|
2016-02-13 22:49:48 +00:00
|
|
|
}
|
|
|
|
val, ok := count[message.User.Username]
|
|
|
|
if !ok {
|
|
|
|
val = 0
|
|
|
|
}
|
|
|
|
count[message.User.Username] = val + 1
|
2016-02-13 23:36:18 +00:00
|
|
|
words[word] = count
|
2016-02-13 22:49:48 +00:00
|
|
|
|
|
|
|
j, err := json.Marshal(count)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.Put([]byte(word), j)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Println("[processWords] Error encountered: " + err.Error())
|
|
|
|
}
|
|
|
|
}
|