-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathvarnam_handlers.go
146 lines (130 loc) · 4.09 KB
/
varnam_handlers.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
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
package main
import (
"database/sql"
"errors"
"log"
"sync"
"time"
"github.com/golang/groupcache"
_ "github.com/mattn/go-sqlite3"
"github.com/varnamproject/libvarnam-golang"
)
type word struct {
Id int `json:"id"`
Confidence int `json:"confidence"`
Word string `json:"word"`
}
var (
languageChannels map[string]chan *libvarnam.Varnam
channelsCount map[string]int
mutex *sync.Mutex
once sync.Once
schemeDetails = libvarnam.GetAllSchemeDetails()
peers = groupcache.NewHTTPPool("http://localhost")
cacheGroups = make(map[string]*groupcache.Group)
)
func isValidSchemeIdentifier(id string) bool {
for _, scheme := range schemeDetails {
if scheme.Identifier == id {
return true
}
}
return false
}
func initLanguageChannels() {
languageChannels = make(map[string]chan *libvarnam.Varnam)
channelsCount = make(map[string]int)
mutex = &sync.Mutex{}
for _, scheme := range schemeDetails {
languageChannels[scheme.Identifier] = make(chan *libvarnam.Varnam, maxHandleCount)
channelsCount[scheme.Identifier] = maxHandleCount
for i := 0; i < maxHandleCount; i++ {
handle, err := libvarnam.Init(scheme.Identifier)
if err != nil {
panic("Unable to init varnam for language " + scheme.LangCode + "." + err.Error())
}
languageChannels[scheme.Identifier] <- handle
}
}
}
func getOrCreateHandler(schemeIdentifier string, f func(handle *libvarnam.Varnam) (data interface{}, err error)) (data interface{}, err error) {
ch, ok := languageChannels[schemeIdentifier]
if !ok {
return nil, errors.New("Invalid scheme identifier")
}
select {
case handle := <-ch:
data, err = f(handle)
go func() { ch <- handle }()
case <-time.After(800 * time.Millisecond):
var handle *libvarnam.Varnam
handle, err = libvarnam.Init(schemeIdentifier)
if err != nil {
log.Println(err)
return nil, errors.New("Unable to initialize varnam handle")
}
data, err = f(handle)
go sendHandlerToChannel(schemeIdentifier, handle, ch)
}
return
}
func transliterate(schemeIdentifier string, word string) (data interface{}, err error) {
return getOrCreateHandler(schemeIdentifier, func(handle *libvarnam.Varnam) (data interface{}, err error) {
data, err = handle.Transliterate(word)
return
})
}
func getWords(schemeIdentifier string, downloadStart int) ([]*word, error) {
filepath, _ := getOrCreateHandler(schemeIdentifier, func(handle *libvarnam.Varnam) (data interface{}, err error) {
return handle.GetSuggestionsFilePath(), nil
})
db, err := sql.Open("sqlite3", filepath.(string))
if err != nil {
return nil, err
}
defer db.Close()
// Making an index for all learned words so that download is faster
// this needs to be removed when there is more clarity on how learned words needs to be handled
_, err = db.Exec("create index if not exists varnamd_download_only_learned on patterns_content (learned) where learned = 1;")
if err != nil {
return nil, err
}
q := "select id, word, confidence from words where id in (select distinct(word_id) from patterns_content where learned = 1) order by id asc limit ? offset ?;"
rows, err := db.Query(q, downloadPageSize, downloadStart)
if err != nil {
return nil, err
}
defer rows.Close()
var words []*word
for rows.Next() {
var id, confidence int
var _word string
rows.Scan(&id, &_word, &confidence)
words = append(words, &word{Id: id, Confidence: confidence, Word: _word})
}
return words, nil
}
func reveseTransliterate(schemeIdentifier string, word string) (data interface{}, err error) {
return getOrCreateHandler(schemeIdentifier, func(handle *libvarnam.Varnam) (data interface{}, err error) {
data, err = handle.ReverseTransliterate(word)
return
})
}
func sendHandlerToChannel(schemeIdentifier string, handle *libvarnam.Varnam, ch chan *libvarnam.Varnam) {
mutex.Lock()
count := channelsCount[schemeIdentifier]
mutex.Unlock()
if count == maxHandleCount {
log.Printf("Throw away handle")
handle.Destroy()
return
}
select {
case ch <- handle:
mutex.Lock()
count = channelsCount[schemeIdentifier]
channelsCount[schemeIdentifier] = count + 1
mutex.Unlock()
default:
}
}