-
Notifications
You must be signed in to change notification settings - Fork 0
/
sessions.go
131 lines (122 loc) · 2.95 KB
/
sessions.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
package db
import (
"context"
"errors"
"sync"
"time"
"github.com/dgraph-io/badger/v3"
"github.com/gernest/yukio/pkg/models"
"github.com/golang/protobuf/ptypes"
"google.golang.org/protobuf/proto"
)
var sequences = &sync.Map{}
const LeaseSize = uint64(1000)
var sessionIndex *badger.Sequence
func GetSessionID(ctx context.Context) (uint64, error) {
if sessionIndex != nil {
return sessionIndex.Next()
}
var err error
sessionIndex, err = GetStore(ctx).GetSequence(SessionLease, LeaseSize)
if err != nil {
return 0, err
}
return sessionIndex.Next()
}
func resetSession(s *models.Session, e *models.Event) *models.Session {
var pageview int32
if e.Name == "pageview" {
pageview = 1
}
s.Sign = 1
s.Hostname = e.Hostname
s.Domain = e.Domain
s.EntryPage = e.Pathname
s.ExitPage = e.Pathname
s.IsBounce = true
s.PageViews = pageview
s.Events = 1
s.Referrer = e.Referrer
s.ReferrerSource = e.ReferrerSource
s.UtmMedium = e.UtmMedium
s.UtmSource = e.UtmSource
s.UtmCampaign = e.UtmCampaign
s.CountryCode = e.CountryCode
s.ScreenSize = e.ScreenSize
s.OperatingSystem = e.OperatingSystem
s.OperatingSystemVersion = e.OperatingSystemVersion
s.Browser = e.Browser
s.BrowserVersion = e.BrowserVersion
s.Timestamp = e.Timestamp
s.Start = e.Timestamp
return s
}
func CreateSessionFromEvent(ctx context.Context, s *models.Session, event *models.Event) error {
id, err := GetSessionID(ctx)
if err != nil {
return err
}
s.Id = id
event.SessionId = id
resetSession(s, event)
k := gk().SessionID(event.UserId, event.Domain)
defer pk(k)
v, err := proto.Marshal(s)
if err != nil {
return err
}
return GetStore(ctx).Update(func(txn *badger.Txn) error {
return txn.Set(k.Bytes(), v)
})
}
func SaveSession(ctx context.Context, event *models.Event, sessionWindow time.Duration) (ss *models.Session, err error) {
var os models.Session
defer func() {
if err == nil {
ss = &os
}
}()
k := gk().SessionID(event.UserId, event.Domain)
defer pk(k)
err = GetStore(ctx).View(func(txn *badger.Txn) error {
x, err := txn.Get(k.Bytes())
if err != nil {
return err
}
return x.Value(func(val []byte) error {
return proto.Unmarshal(val, &os)
})
})
if err != nil {
if errors.Is(err, badger.ErrKeyNotFound) {
// This is new event with no session
err = CreateSessionFromEvent(ctx, &os, event)
}
return
}
osts, _ := ptypes.Timestamp(os.Timestamp)
evts, _ := ptypes.Timestamp(event.Timestamp)
active := evts.Sub(osts) < sessionWindow
if active {
var pageView int32
if event.Name == "pageview" {
pageView++
}
os.Timestamp = event.Timestamp
os.ExitPage = event.Pathname
startTS, _ := ptypes.Timestamp(os.Start)
os.Duration = ptypes.DurationProto(evts.Sub(startTS))
os.Events++
os.PageViews++
err = GetStore(ctx).Update(func(txn *badger.Txn) error {
v, err := proto.Marshal(&os)
if err != nil {
return err
}
return txn.Set(k.Bytes(), v)
})
return
}
err = CreateSessionFromEvent(ctx, &os, event)
return
}