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
|
package svc // import "github.com/getwtxt/getwtxt/svc"
import (
"bytes"
"html/template"
"io/ioutil"
"os"
"sync"
"time"
)
// RemoteRegistries holds a list of remote registries to
// periodically scrape for new users. The remote registries
// must have been added via POST like a user.
type RemoteRegistries struct {
Mu sync.RWMutex
List []string
}
type staticAssets struct {
mu sync.RWMutex
index []byte
indexMod time.Time
css []byte
cssMod time.Time
}
func initTemplates() *template.Template {
confObj.Mu.RLock()
assetsDir := confObj.AssetsDir
confObj.Mu.RUnlock()
return template.Must(template.ParseFiles(assetsDir + "/tmpl/index.html"))
}
func cacheTimer() bool {
confObj.Mu.RLock()
answer := time.Since(confObj.LastCache) > confObj.CacheInterval
confObj.Mu.RUnlock()
return answer
}
// Launched by init as a coroutine to watch
// for the update intervals to pass.
func cacheAndPush() {
for {
if cacheTimer() {
refreshCache()
}
if dbTimer() {
errLog("Error pushing cache to database: ", pushDB())
}
time.Sleep(1000 * time.Millisecond)
}
}
func refreshCache() {
// This clusterfuck of mutex read locks is
// necessary to avoid deadlock. This mess
// also avoids a panic that would occur
// should twtxtCache be written to during
// this loop.
twtxtCache.Mu.RLock()
for k := range twtxtCache.Users {
twtxtCache.Mu.RUnlock()
errLog("", twtxtCache.UpdateUser(k))
twtxtCache.Mu.RLock()
}
twtxtCache.Mu.RUnlock()
remoteRegistries.Mu.RLock()
for _, v := range remoteRegistries.List {
errLog("Error refreshing local copy of remote registry data: ", twtxtCache.CrawlRemoteRegistry(v))
}
remoteRegistries.Mu.RUnlock()
confObj.Mu.Lock()
confObj.LastCache = time.Now()
confObj.Mu.Unlock()
}
// pingAssets checks if the local static assets
// need to be re-cached. If they do, they are
// pulled back into memory from disk.
func pingAssets() {
confObj.Mu.RLock()
assetsDir := confObj.AssetsDir
confObj.Mu.RUnlock()
cssStat, err := os.Stat(assetsDir + "/style.css")
errLog("", err)
indexStat, err := os.Stat(assetsDir + "/tmpl/index.html")
errLog("", err)
staticCache.mu.RLock()
indexMod := staticCache.indexMod
cssMod := staticCache.cssMod
staticCache.mu.RUnlock()
if !indexMod.Equal(indexStat.ModTime()) {
tmpls = initTemplates()
var b []byte
buf := bytes.NewBuffer(b)
confObj.Mu.RLock()
errLog("", tmpls.ExecuteTemplate(buf, "index.html", confObj.Instance))
confObj.Mu.RUnlock()
staticCache.mu.Lock()
staticCache.index = buf.Bytes()
staticCache.indexMod = indexStat.ModTime()
staticCache.mu.Unlock()
}
if !cssMod.Equal(cssStat.ModTime()) {
css, err := ioutil.ReadFile(assetsDir + "/style.css")
errLog("", err)
staticCache.mu.Lock()
staticCache.css = css
staticCache.cssMod = cssStat.ModTime()
staticCache.mu.Unlock()
}
}
|