-
Notifications
You must be signed in to change notification settings - Fork 23
/
Copy pathsettings.go
187 lines (163 loc) · 5.71 KB
/
settings.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
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
package botsfw
import (
"context"
"fmt"
"github.com/bots-go-framework/bots-fw-store/botsfwmodels"
"github.com/bots-go-framework/bots-fw/botsfwconst"
"github.com/dal-go/dalgo/dal"
"github.com/dal-go/dalgo/record"
"github.com/strongo/i18n"
"os"
"strings"
)
type DbGetter = func(ctx context.Context) (db dal.DB, err error)
type AppUserGetter = func(
ctx context.Context,
tx dal.ReadSession,
botID string,
appUserID string,
) (
appUser record.DataWithID[string, botsfwmodels.AppUserData],
err error,
)
// BotSettings keeps parameters of a bot that are static and are not changed in runtime
type BotSettings struct {
// Platform is a platform that bot is running on
// E.g.: Telegram, Viber, Facebook Messenger, WhatsApp, etc.
Platform botsfwconst.Platform
// Env is an environment where bot is running
// E.g.: Production/Live, Local/Dev, Staging, etc.
Env string
// Profile is a bot profile that defines bot's behavior
// It includes commands router and some other settings
// More in BotProfile documentation.
Profile BotProfile
// Code is a human-readable ID of a bot.
// When displayed it is usually prefixed with @.
// For example:
// - @listus_bot for https://t.me/listus_bot
Code string
// ID is a bot-platform ID of a bot. For example, it could be a GUID.
// Not all platforms use it. For example Telegram doesn't use it.
ID string
// Token is used to authenticate bot with a platform when it is not responding to a webhook
// but calling platform APIs directly.
Token string
// PaymentToken is used to process payments on bot platform
PaymentToken string
// PaymentTestToken is used to process test payments on bot platform
PaymentTestToken string
// VerifyToken is used by Facebook Messenger - TODO: Document how it is used and add a link to Facebook docs
VerifyToken string
// GAToken is Google Analytics token - TODO: Refactor tu support multiple or move out
GAToken string
// Locale is a default locale for a bot.
// While a bot profile can support multiple locales a bot can be dedicated to a specific country/language
Locale i18n.Locale
// GetDatabase returns connection to a database assigned to a bot.
// You can use same database for multiple bots
// but if you need you can use different databases for different bots.
// It's up to bots creator how to map bots to a database.
// In most cases a single DB is used for all bots.
GetDatabase DbGetter
getAppUser AppUserGetter
}
func (v BotSettings) GetAppUserByID(ctx context.Context, tx dal.ReadSession, appUserID string) (appUser record.DataWithID[string, botsfwmodels.AppUserData], err error) {
return v.getAppUser(ctx, tx, v.Code, appUserID)
}
// NewBotSettings configures bot application
func NewBotSettings(
platform botsfwconst.Platform,
environment string,
profile BotProfile,
code, id, token, gaToken string,
locale i18n.Locale,
getDatabase DbGetter,
getAppUser AppUserGetter,
) BotSettings {
if platform == "" {
panic("NewBotSettings: missing required parameter: platform")
}
if profile == nil {
panic("NewBotSettings: missing required parameter: profile")
}
if code == "" {
panic("NewBotSettings: missing required parameter: code")
}
if token == "" {
envVarKey := fmt.Sprintf("%s_BOT_TOKEN_%s", strings.ToUpper(string(platform)), strings.ToUpper(code))
token = os.Getenv(envVarKey)
if token == "" {
panic("NewBotSettings: missing required parameter 'token' and no environment variable " + envVarKey)
}
}
if gaToken == "" {
envVarKey := fmt.Sprintf("%s_GA_TOKEN_%s", strings.ToUpper(string(platform)), strings.ToUpper(code))
gaToken = os.Getenv(envVarKey)
}
if locale.Code5 == "" {
panic("NewBotSettings: missing required parameter: Locale.Code5")
}
return BotSettings{
Platform: platform,
Profile: profile,
Code: code,
ID: id,
Env: environment,
Token: token,
GAToken: gaToken,
Locale: locale,
GetDatabase: getDatabase,
getAppUser: getAppUser,
}
}
// SettingsProvider returns settings per different keys (ID, code, API token, Locale)
type BotSettingsProvider func(ctx context.Context) BotSettingsBy
// SettingsBy keeps settings per different keys (ID, code, API token, Locale)
// TODO: Decide if it should have map[string]*BotSettings instead of map[string]BotSettings
type BotSettingsBy struct {
// ByCode keeps settings by bot code - it is a human-readable ID of a bot
ByCode map[string]*BotSettings
// ByID keeps settings by bot ID - it is a machine-readable ID of a bot.
ByID map[string]*BotSettings
ByProfile map[string][]*BotSettings
}
// NewBotSettingsBy create settings per different keys (ID, code, API token, Locale)
func NewBotSettingsBy(bots ...BotSettings) (settingsBy BotSettingsBy) {
count := len(bots)
if count == 0 {
panic("NewBotSettingsBy: missing required parameter: bots")
}
settingsBy = BotSettingsBy{
ByCode: make(map[string]*BotSettings, count),
ByID: make(map[string]*BotSettings, count),
ByProfile: make(map[string][]*BotSettings),
}
processBotSettings := func(i int, bot BotSettings) {
if bot.Code == "" {
panic(fmt.Sprintf("Bot with empty code at index %v", i))
}
if _, ok := settingsBy.ByCode[bot.Code]; ok {
panic(fmt.Sprintf("Bot with duplicate code: %v", bot.Code))
} else {
settingsBy.ByCode[bot.Code] = &bot
}
if bot.ID != "" {
if _, ok := settingsBy.ByID[bot.ID]; ok {
panic(fmt.Sprintf("Bot with duplicate ID: %v", bot.ID))
} else {
settingsBy.ByID[bot.ID] = &bot
}
}
profileID := bot.Profile.ID()
if profileBots, ok := settingsBy.ByProfile[profileID]; ok {
settingsBy.ByProfile[profileID] = append(profileBots, &bot)
} else {
settingsBy.ByProfile[profileID] = []*BotSettings{&bot}
}
}
for i, bot := range bots {
processBotSettings(i, bot)
}
return settingsBy
}