From 07d4d93d3a8f84332af6f61db1ab5931c237f9cd Mon Sep 17 00:00:00 2001
From: aforge
Date: Sat, 14 Mar 2020 01:57:38 -0700
Subject: [PATCH 001/142] Basic Tinode Push Gateway adapter.
---
server/main.go | 1 +
server/push/fcm/payload.go | 268 +++++++++++++++++++++++++++++++
server/push/fcm/push_fcm.go | 290 +++-------------------------------
server/push/tnpg/push_tnpg.go | 115 ++++++++++++++
4 files changed, 410 insertions(+), 264 deletions(-)
create mode 100644 server/push/fcm/payload.go
create mode 100644 server/push/tnpg/push_tnpg.go
diff --git a/server/main.go b/server/main.go
index 203e277d7..fa8876dce 100644
--- a/server/main.go
+++ b/server/main.go
@@ -43,6 +43,7 @@ import (
"github.com/tinode/chat/server/push"
_ "github.com/tinode/chat/server/push/fcm"
_ "github.com/tinode/chat/server/push/stdout"
+ _ "github.com/tinode/chat/server/push/tnpg"
"github.com/tinode/chat/server/store"
diff --git a/server/push/fcm/payload.go b/server/push/fcm/payload.go
new file mode 100644
index 000000000..0751b0863
--- /dev/null
+++ b/server/push/fcm/payload.go
@@ -0,0 +1,268 @@
+package fcm
+
+import (
+ "errors"
+ "log"
+ "strconv"
+ "time"
+
+ fcm "firebase.google.com/go/messaging"
+
+ "github.com/tinode/chat/server/drafty"
+ "github.com/tinode/chat/server/push"
+ "github.com/tinode/chat/server/store"
+ t "github.com/tinode/chat/server/store/types"
+)
+
+// Configuration of AndroidNotification payload.
+type AndroidConfig struct {
+ Enabled bool `json:"enabled,omitempty"`
+ // Common defauls for all push types.
+ androidPayload
+ // Configs for specific push types.
+ Msg androidPayload `json:"msg,omitempty"`
+ Sub androidPayload `json:"msg,omitempty"`
+}
+
+func (ac *AndroidConfig) getTitleLocKey(what string) string {
+ var title string
+ if what == push.ActMsg {
+ title = ac.Msg.TitleLocKey
+ } else if what == push.ActSub {
+ title = ac.Sub.TitleLocKey
+ }
+ if title == "" {
+ title = ac.androidPayload.TitleLocKey
+ }
+ return title
+}
+
+func (ac *AndroidConfig) getTitle(what string) string {
+ var title string
+ if what == push.ActMsg {
+ title = ac.Msg.Title
+ } else if what == push.ActSub {
+ title = ac.Sub.Title
+ }
+ if title == "" {
+ title = ac.androidPayload.Title
+ }
+ return title
+}
+
+func (ac *AndroidConfig) getBodyLocKey(what string) string {
+ var body string
+ if what == push.ActMsg {
+ body = ac.Msg.BodyLocKey
+ } else if what == push.ActSub {
+ body = ac.Sub.BodyLocKey
+ }
+ if body == "" {
+ body = ac.androidPayload.BodyLocKey
+ }
+ return body
+}
+
+func (ac *AndroidConfig) getBody(what string) string {
+ var body string
+ if what == push.ActMsg {
+ body = ac.Msg.Body
+ } else if what == push.ActSub {
+ body = ac.Sub.Body
+ }
+ if body == "" {
+ body = ac.androidPayload.Body
+ }
+ return body
+}
+
+func (ac *AndroidConfig) getIcon(what string) string {
+ var icon string
+ if what == push.ActMsg {
+ icon = ac.Msg.Icon
+ } else if what == push.ActSub {
+ icon = ac.Sub.Icon
+ }
+ if icon == "" {
+ icon = ac.androidPayload.Icon
+ }
+ return icon
+}
+
+func (ac *AndroidConfig) getIconColor(what string) string {
+ var color string
+ if what == push.ActMsg {
+ color = ac.Msg.IconColor
+ } else if what == push.ActSub {
+ color = ac.Sub.IconColor
+ }
+ if color == "" {
+ color = ac.androidPayload.IconColor
+ }
+ return color
+}
+
+// Payload to be sent for a specific notification type.
+type androidPayload struct {
+ TitleLocKey string `json:"title_loc_key,omitempty"`
+ Title string `json:"title,omitempty"`
+ BodyLocKey string `json:"body_loc_key,omitempty"`
+ Body string `json:"body,omitempty"`
+ Icon string `json:"icon,omitempty"`
+ IconColor string `json:"icon_color,omitempty"`
+ ClickAction string `json:"click_action,omitempty"`
+}
+
+type messageData struct {
+ Uid t.Uid
+ DeviceId string
+ Message *fcm.Message
+}
+
+func payloadToData(pl *push.Payload) (map[string]string, error) {
+ if pl == nil {
+ return nil, nil
+ }
+
+ data := make(map[string]string)
+ var err error
+ data["what"] = pl.What
+ if pl.Silent {
+ data["silent"] = "true"
+ }
+ data["topic"] = pl.Topic
+ data["ts"] = pl.Timestamp.Format(time.RFC3339Nano)
+ // Must use "xfrom" because "from" is a reserved word. Google did not bother to document it anywhere.
+ data["xfrom"] = pl.From
+ if pl.What == push.ActMsg {
+ data["seq"] = strconv.Itoa(pl.SeqId)
+ data["mime"] = pl.ContentType
+ data["content"], err = drafty.ToPlainText(pl.Content)
+ if err != nil {
+ return nil, err
+ }
+
+ // Trim long strings to 80 runes.
+ // Check byte length first and don't waste time converting short strings.
+ if len(data["content"]) > maxMessageLength {
+ runes := []rune(data["content"])
+ if len(runes) > maxMessageLength {
+ data["content"] = string(runes[:maxMessageLength]) + "…"
+ }
+ }
+ } else if pl.What == push.ActSub {
+ data["modeWant"] = pl.ModeWant.String()
+ data["modeGiven"] = pl.ModeGiven.String()
+ } else {
+ return nil, errors.New("unknown push type")
+ }
+ return data, nil
+}
+
+// PrepareNotifications creates notification payloads ready to be posted
+// to push notification server for the provided receipt.
+func PrepareNotifications(rcpt *push.Receipt, config *AndroidConfig) []messageData {
+ data, _ := payloadToData(&rcpt.Payload)
+ if data == nil {
+ log.Println("fcm push: could not parse payload")
+ return nil
+ }
+
+ // List of UIDs for querying the database
+ uids := make([]t.Uid, len(rcpt.To))
+ skipDevices := make(map[string]bool)
+ i := 0
+ for uid, to := range rcpt.To {
+ uids[i] = uid
+ i++
+
+ // Some devices were online and received the message. Skip them.
+ for _, deviceID := range to.Devices {
+ skipDevices[deviceID] = true
+ }
+ }
+
+ devices, count, err := store.Devices.GetAll(uids...)
+ if err != nil {
+ log.Println("fcm push: db error", err)
+ return nil
+ }
+ if count == 0 {
+ return nil
+ }
+
+ var titlelc, title, bodylc, body, icon, color string
+ if config.Enabled {
+ titlelc = config.getTitleLocKey(rcpt.Payload.What)
+ title = config.getTitle(rcpt.Payload.What)
+ bodylc = config.getBodyLocKey(rcpt.Payload.What)
+ body = config.getBody(rcpt.Payload.What)
+ if body == "$content" {
+ body = data["content"]
+ }
+ icon = config.getIcon(rcpt.Payload.What)
+ color = config.getIconColor(rcpt.Payload.What)
+ }
+
+ var messages []messageData
+ for uid, devList := range devices {
+ for i := range devList {
+ d := &devList[i]
+ if _, ok := skipDevices[d.DeviceId]; !ok && d.DeviceId != "" {
+ msg := fcm.Message{
+ Token: d.DeviceId,
+ Data: data,
+ }
+
+ if d.Platform == "android" {
+ msg.Android = &fcm.AndroidConfig{
+ Priority: "high",
+ }
+ if config.Enabled {
+ // When this notification type is included and the app is not in the foreground
+ // Android won't wake up the app and won't call FirebaseMessagingService:onMessageReceived.
+ // See dicussion: https://github.com/firebase/quickstart-js/issues/71
+ msg.Android.Notification = &fcm.AndroidNotification{
+ // Android uses Tag value to group notifications together:
+ // show just one notification per topic.
+ Tag: rcpt.Payload.Topic,
+ TitleLocKey: titlelc,
+ Title: title,
+ BodyLocKey: bodylc,
+ Body: body,
+ Icon: icon,
+ Color: color,
+ }
+ }
+ } else if d.Platform == "ios" {
+ // iOS uses Badge to show the total unread message count.
+ badge := rcpt.To[uid].Unread
+ // Need to duplicate these in APNS.Payload.Aps.Alert so
+ // iOS may call NotificationServiceExtension (if present).
+ title := "New message"
+ body := data["content"]
+ msg.APNS = &fcm.APNSConfig{
+ Payload: &fcm.APNSPayload{
+ Aps: &fcm.Aps{
+ Badge: &badge,
+ ContentAvailable: true,
+ MutableContent: true,
+ Sound: "default",
+ Alert: &fcm.ApsAlert{
+ Title: title,
+ Body: body,
+ },
+ },
+ },
+ }
+ msg.Notification = &fcm.Notification{
+ Title: title,
+ Body: body,
+ }
+ }
+ messages = append(messages, messageData{Uid: uid, DeviceId: d.DeviceId, Message: &msg})
+ }
+ }
+ }
+ return messages
+}
diff --git a/server/push/fcm/push_fcm.go b/server/push/fcm/push_fcm.go
index ebb132f0a..6bd940c1d 100644
--- a/server/push/fcm/push_fcm.go
+++ b/server/push/fcm/push_fcm.go
@@ -7,16 +7,12 @@ import (
"encoding/json"
"errors"
"log"
- "strconv"
- "time"
fbase "firebase.google.com/go"
fcm "firebase.google.com/go/messaging"
- "github.com/tinode/chat/server/drafty"
"github.com/tinode/chat/server/push"
"github.com/tinode/chat/server/store"
- t "github.com/tinode/chat/server/store/types"
"golang.org/x/oauth2/google"
"google.golang.org/api/option"
@@ -37,112 +33,13 @@ type Handler struct {
client *fcm.Client
}
-// Configuration of AndroidNotification payload.
-type androidConfig struct {
- Enabled bool `json:"enabled,omitempty"`
- // Common defauls for all push types.
- androidPayload
- // Configs for specific push types.
- Msg androidPayload `json:"msg,omitempty"`
- Sub androidPayload `json:"msg,omitempty"`
-}
-
-func (ac *androidConfig) getTitleLocKey(what string) string {
- var title string
- if what == push.ActMsg {
- title = ac.Msg.TitleLocKey
- } else if what == push.ActSub {
- title = ac.Sub.TitleLocKey
- }
- if title == "" {
- title = ac.androidPayload.TitleLocKey
- }
- return title
-}
-
-func (ac *androidConfig) getTitle(what string) string {
- var title string
- if what == push.ActMsg {
- title = ac.Msg.Title
- } else if what == push.ActSub {
- title = ac.Sub.Title
- }
- if title == "" {
- title = ac.androidPayload.Title
- }
- return title
-}
-
-func (ac *androidConfig) getBodyLocKey(what string) string {
- var body string
- if what == push.ActMsg {
- body = ac.Msg.BodyLocKey
- } else if what == push.ActSub {
- body = ac.Sub.BodyLocKey
- }
- if body == "" {
- body = ac.androidPayload.BodyLocKey
- }
- return body
-}
-
-func (ac *androidConfig) getBody(what string) string {
- var body string
- if what == push.ActMsg {
- body = ac.Msg.Body
- } else if what == push.ActSub {
- body = ac.Sub.Body
- }
- if body == "" {
- body = ac.androidPayload.Body
- }
- return body
-}
-
-func (ac *androidConfig) getIcon(what string) string {
- var icon string
- if what == push.ActMsg {
- icon = ac.Msg.Icon
- } else if what == push.ActSub {
- icon = ac.Sub.Icon
- }
- if icon == "" {
- icon = ac.androidPayload.Icon
- }
- return icon
-}
-
-func (ac *androidConfig) getIconColor(what string) string {
- var color string
- if what == push.ActMsg {
- color = ac.Msg.IconColor
- } else if what == push.ActSub {
- color = ac.Sub.IconColor
- }
- if color == "" {
- color = ac.androidPayload.IconColor
- }
- return color
-}
-
-// Payload to be sent for a specific notification type.
-type androidPayload struct {
- TitleLocKey string `json:"title_loc_key,omitempty"`
- Title string `json:"title,omitempty"`
- BodyLocKey string `json:"body_loc_key,omitempty"`
- Body string `json:"body,omitempty"`
- Icon string `json:"icon,omitempty"`
- IconColor string `json:"icon_color,omitempty"`
- ClickAction string `json:"click_action,omitempty"`
-}
-
type configType struct {
Enabled bool `json:"enabled"`
Buffer int `json:"buffer"`
Credentials json.RawMessage `json:"credentials"`
CredentialsFile string `json:"credentials_file"`
TimeToLive uint `json:"time_to_live,omitempty"`
- Android androidConfig `json:"android,omitempty"`
+ Android AndroidConfig `json:"android,omitempty"`
}
// Init initializes the push handler
@@ -206,178 +103,43 @@ func (Handler) Init(jsonconf string) error {
func sendNotifications(rcpt *push.Receipt, config *configType) {
ctx := context.Background()
-
- data, _ := payloadToData(&rcpt.Payload)
- if data == nil {
- log.Println("fcm push: could not parse payload")
+ messages := PrepareNotifications(rcpt, &config.Android)
+ if messages == nil {
return
}
- // List of UIDs for querying the database
- uids := make([]t.Uid, len(rcpt.To))
- skipDevices := make(map[string]bool)
- i := 0
- for uid, to := range rcpt.To {
- uids[i] = uid
- i++
-
- // Some devices were online and received the message. Skip them.
- for _, deviceID := range to.Devices {
- skipDevices[deviceID] = true
- }
- }
-
- devices, count, err := store.Devices.GetAll(uids...)
- if err != nil {
- log.Println("fcm push: db error", err)
- return
- }
- if count == 0 {
- return
- }
-
- var titlelc, title, bodylc, body, icon, color string
- if config.Android.Enabled {
- titlelc = config.Android.getTitleLocKey(rcpt.Payload.What)
- title = config.Android.getTitle(rcpt.Payload.What)
- bodylc = config.Android.getBodyLocKey(rcpt.Payload.What)
- body = config.Android.getBody(rcpt.Payload.What)
- if body == "$content" {
- body = data["content"]
- }
- icon = config.Android.getIcon(rcpt.Payload.What)
- color = config.Android.getIconColor(rcpt.Payload.What)
- }
-
- for uid, devList := range devices {
- for i := range devList {
- d := &devList[i]
- if _, ok := skipDevices[d.DeviceId]; !ok && d.DeviceId != "" {
- msg := fcm.Message{
- Token: d.DeviceId,
- Data: data,
- }
+ for _, m := range messages {
+ _, err := handler.client.Send(ctx, m.Message)
+ if err != nil {
+ if fcm.IsMessageRateExceeded(err) ||
+ fcm.IsServerUnavailable(err) ||
+ fcm.IsInternal(err) ||
+ fcm.IsUnknown(err) {
+ // Transient errors. Stop sending this batch.
+ log.Println("fcm transient failure", err)
+ return
+ }
- if d.Platform == "android" {
- msg.Android = &fcm.AndroidConfig{
- Priority: "high",
- }
- if config.Android.Enabled {
- // When this notification type is included and the app is not in the foreground
- // Android won't wake up the app and won't call FirebaseMessagingService:onMessageReceived.
- // See dicussion: https://github.com/firebase/quickstart-js/issues/71
- msg.Android.Notification = &fcm.AndroidNotification{
- // Android uses Tag value to group notifications together:
- // show just one notification per topic.
- Tag: rcpt.Payload.Topic,
- TitleLocKey: titlelc,
- Title: title,
- BodyLocKey: bodylc,
- Body: body,
- Icon: icon,
- Color: color,
- }
- }
- } else if d.Platform == "ios" {
- // iOS uses Badge to show the total unread message count.
- badge := rcpt.To[uid].Unread
- // Need to duplicate these in APNS.Payload.Aps.Alert so
- // iOS may call NotificationServiceExtension (if present).
- title := "New message"
- body := data["content"]
- msg.APNS = &fcm.APNSConfig{
- Payload: &fcm.APNSPayload{
- Aps: &fcm.Aps{
- Badge: &badge,
- ContentAvailable: true,
- MutableContent: true,
- Sound: "default",
- Alert: &fcm.ApsAlert{
- Title: title,
- Body: body,
- },
- },
- },
- }
- msg.Notification = &fcm.Notification{
- Title: title,
- Body: body,
- }
- }
+ if fcm.IsMismatchedCredential(err) || fcm.IsInvalidArgument(err) {
+ // Config errors
+ log.Println("fcm push: failed", err)
+ return
+ }
- _, err := handler.client.Send(ctx, &msg)
+ if fcm.IsRegistrationTokenNotRegistered(err) {
+ // Token is no longer valid.
+ log.Println("fcm push: invalid token", err)
+ err = store.Devices.Delete(m.Uid, m.DeviceId)
if err != nil {
- if fcm.IsMessageRateExceeded(err) ||
- fcm.IsServerUnavailable(err) ||
- fcm.IsInternal(err) ||
- fcm.IsUnknown(err) {
- // Transient errors. Stop sending this batch.
- log.Println("fcm transient failure", err)
- return
- }
-
- if fcm.IsMismatchedCredential(err) || fcm.IsInvalidArgument(err) {
- // Config errors
- log.Println("fcm push: failed", err)
- return
- }
-
- if fcm.IsRegistrationTokenNotRegistered(err) {
- // Token is no longer valid.
- log.Println("fcm push: invalid token", err)
- err = store.Devices.Delete(uid, d.DeviceId)
- if err != nil {
- log.Println("fcm push: failed to delete invalid token", err)
- }
- } else {
- log.Println("fcm push:", err)
- }
+ log.Println("fcm push: failed to delete invalid token", err)
}
+ } else {
+ log.Println("fcm push:", err)
}
}
}
}
-func payloadToData(pl *push.Payload) (map[string]string, error) {
- if pl == nil {
- return nil, nil
- }
-
- data := make(map[string]string)
- var err error
- data["what"] = pl.What
- if pl.Silent {
- data["silent"] = "true"
- }
- data["topic"] = pl.Topic
- data["ts"] = pl.Timestamp.Format(time.RFC3339Nano)
- // Must use "xfrom" because "from" is a reserved word. Google did not bother to document it anywhere.
- data["xfrom"] = pl.From
- if pl.What == push.ActMsg {
- data["seq"] = strconv.Itoa(pl.SeqId)
- data["mime"] = pl.ContentType
- data["content"], err = drafty.ToPlainText(pl.Content)
- if err != nil {
- return nil, err
- }
-
- // Trim long strings to 80 runes.
- // Check byte length first and don't waste time converting short strings.
- if len(data["content"]) > maxMessageLength {
- runes := []rune(data["content"])
- if len(runes) > maxMessageLength {
- data["content"] = string(runes[:maxMessageLength]) + "…"
- }
- }
- } else if pl.What == push.ActSub {
- data["modeWant"] = pl.ModeWant.String()
- data["modeGiven"] = pl.ModeGiven.String()
- } else {
- return nil, errors.New("unknown push type")
- }
- return data, nil
-}
-
// IsReady checks if the push handler has been initialized.
func (Handler) IsReady() bool {
return handler.input != nil
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
new file mode 100644
index 000000000..5e7cc19e4
--- /dev/null
+++ b/server/push/tnpg/push_tnpg.go
@@ -0,0 +1,115 @@
+// Package tnpg implements push notification plugin for Tinode Push Gateway.
+package tnpg
+
+import (
+ "bytes"
+ "encoding/json"
+ "errors"
+ "log"
+ "net/http"
+
+ "github.com/tinode/chat/server/push"
+ "github.com/tinode/chat/server/push/fcm"
+)
+
+var handler Handler
+
+type Handler struct {
+ input chan *push.Receipt
+ stop chan bool
+}
+
+type configType struct {
+ Enabled bool `json:"enabled"`
+ Buffer int `json:"buffer"`
+ TargetAddress string `json:"target_address"`
+ AuthToken string `json:"auth_token"`
+ Android fcm.AndroidConfig `json:"android,omitempty"`
+}
+
+// Init initializes the handler
+func (Handler) Init(jsonconf string) error {
+ var config configType
+ if err := json.Unmarshal([]byte(jsonconf), &config); err != nil {
+ return errors.New("failed to parse config: " + err.Error())
+ }
+
+ if !config.Enabled {
+ return nil
+ }
+
+ handler.input = make(chan *push.Receipt, config.Buffer)
+ handler.stop = make(chan bool, 1)
+
+ go func() {
+ for {
+ select {
+ case rcpt := <-handler.input:
+ go sendPushes(rcpt, &config)
+ case <-handler.stop:
+ return
+ }
+ }
+ }()
+
+ return nil
+}
+
+func postMessage(body []byte, config *configType) (int, string, error) {
+ reader := bytes.NewReader(body)
+ req, err := http.NewRequest("POST", config.TargetAddress, reader)
+ if err != nil {
+ return -1, "", err
+ }
+ req.Header.Add("Authorization", config.AuthToken)
+ resp, err := http.DefaultClient.Do(req)
+ if err != nil {
+ return -1, "", err
+ }
+ defer resp.Body.Close()
+ return resp.StatusCode, resp.Status, nil
+}
+
+func sendPushes(rcpt *push.Receipt, config *configType) {
+ messages := fcm.PrepareNotifications(rcpt, &config.Android)
+ if messages == nil {
+ return
+ }
+
+ // TODO:
+ // 1. Send multiple payloads in one request.
+ // 2. Compress payloads.
+ for _, m := range messages {
+ msg, err := json.Marshal(m.Message)
+ if err != nil {
+ log.Println("tnpg push: cannot serialize message", err)
+ return
+ }
+ if code, status, err := postMessage(msg, config); err != nil {
+ log.Println("tnpg push failed:", err)
+ } else if code >= 300 {
+ log.Println("tnpg push rejected:", status, err)
+ break
+ }
+ }
+}
+
+// IsReady checks if the handler is initialized.
+func (Handler) IsReady() bool {
+ return handler.input != nil
+}
+
+// Push returns a channel that the server will use to send messages to.
+// If the adapter blocks, the message will be dropped.
+func (Handler) Push() chan<- *push.Receipt {
+ return handler.input
+}
+
+// Stop terminates the handler's worker and stops sending pushes.
+func (Handler) Stop() {
+ handler.stop <- true
+}
+
+func init() {
+ push.Register("tnpg", &handler)
+}
From d22d3277af524e4f715f43a6702200039d31b4cc Mon Sep 17 00:00:00 2001
From: aforge
Date: Sat, 14 Mar 2020 02:01:23 -0700
Subject: [PATCH 002/142] Formatting.
---
server/push/tnpg/push_tnpg.go | 19 ++++++++++---------
1 file changed, 10 insertions(+), 9 deletions(-)
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index 5e7cc19e4..3fd53e217 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -67,7 +67,7 @@ func postMessage(body []byte, config *configType) (int, string, error) {
return -1, "", err
}
defer resp.Body.Close()
- return resp.StatusCode, resp.Status, nil
+ return resp.StatusCode, resp.Status, nil
}
func sendPushes(rcpt *push.Receipt, config *configType) {
@@ -80,17 +80,18 @@ func sendPushes(rcpt *push.Receipt, config *configType) {
// 1. Send multiple payloads in one request.
// 2. Compress payloads.
for _, m := range messages {
- msg, err := json.Marshal(m.Message)
+ msg, err := json.Marshal(m.Message)
if err != nil {
- log.Println("tnpg push: cannot serialize message", err)
+ log.Println("tnpg push: cannot serialize message", err)
return
}
- if code, status, err := postMessage(msg, config); err != nil {
- log.Println("tnpg push failed:", err)
- } else if code >= 300 {
- log.Println("tnpg push rejected:", status, err)
- break
- }
+ if code, status, err := postMessage(msg, config); err != nil {
+ log.Println("tnpg push failed:", err)
+ break
+ } else if code >= 300 {
+ log.Println("tnpg push rejected:", status, err)
+ break
+ }
}
}
From 17a9f5dcc6eb0f434637e124d3178b8bc4d844e9 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 14 Mar 2020 13:44:51 +0300
Subject: [PATCH 003/142] typo in docker build
---
docker-build.sh | 2 +-
server/tinode.conf | 2 +-
2 files changed, 2 insertions(+), 2 deletions(-)
diff --git a/docker-build.sh b/docker-build.sh
index 5f9e40c15..d7d042627 100755
--- a/docker-build.sh
+++ b/docker-build.sh
@@ -28,7 +28,7 @@ for dbtag in "${dbtags[@]}"
do
if [ "$dbtag" == "alldbs" ]; then
# For alldbs, container name is tinode/tinode.
- name="tiniode/tinode"
+ name="tinode/tinode"
else
# Otherwise, tinode/tinode-$dbtag.
name="tinode/tinode-${dbtag}"
diff --git a/server/tinode.conf b/server/tinode.conf
index 52002af80..9a36c5dab 100644
--- a/server/tinode.conf
+++ b/server/tinode.conf
@@ -154,7 +154,7 @@
// Database configuration
"store_config": {
// XTEA encryption key for user IDs and topic names. 16 random bytes base64-encoded.
- // Generate your own and keep it secret. Otherwise your user IDswill be predictable
+ // Generate your own and keep it secret. Otherwise your user IDs will be predictable
// and it will be easy to spam your users.
"uid_key": "la6YsO+bNX/+XIkOqc5Svw==",
From 9d41906975db65cf47a5e250510391b4404da4f1 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 14 Mar 2020 16:27:17 +0300
Subject: [PATCH 004/142] docker changed URLs
---
docker-release.sh | 12 ++++++------
1 file changed, 6 insertions(+), 6 deletions(-)
diff --git a/docker-release.sh b/docker-release.sh
index f4f8eb516..f706b17ab 100755
--- a/docker-release.sh
+++ b/docker-release.sh
@@ -45,23 +45,23 @@ do
name="$(containerName $dbtag)"
if [ -n "$FULLRELEASE" ]; then
curl -u $user:$pass -i -X DELETE \
- https://cloud.docker.com/v2/repositories/tinode/${name}/tags/latest/
+ https://hub.docker.com/v2/repositories/tinode/${name}/tags/latest/
curl -u $user:$pass -i -X DELETE \
- https://cloud.docker.com/v2/repositories/tinode/${name}/tags/${ver[0]}.${ver[1]}/
+ https://hub.docker.com/v2/repositories/tinode/${name}/tags/${ver[0]}.${ver[1]}/
fi
curl -u $user:$pass -i -X DELETE \
- https://cloud.docker.com/v2/repositories/tinode/${name}/tags/${ver[0]}.${ver[1]}.${ver[2]}/
+ https://hub.docker.com/v2/repositories/tinode/${name}/tags/${ver[0]}.${ver[1]}.${ver[2]}/
done
if [ -n "$FULLRELEASE" ]; then
curl -u $user:$pass -i -X DELETE \
- https://cloud.docker.com/v2/repositories/tinode/chatbot/tags/latest/
+ https://hub.docker.com/v2/repositories/tinode/chatbot/tags/latest/
curl -u $user:$pass -i -X DELETE \
- https://cloud.docker.com/v2/repositories/tinode/chatbot/tags/${ver[0]}.${ver[1]}/
+ https://hub.docker.com/v2/repositories/tinode/chatbot/tags/${ver[0]}.${ver[1]}/
fi
curl -u $user:$pass -i -X DELETE \
- https://cloud.docker.com/v2/repositories/tinode/chatbot/tags/${ver[0]}.${ver[1]}.${ver[2]}/
+ https://hub.docker.com/v2/repositories/tinode/chatbot/tags/${ver[0]}.${ver[1]}.${ver[2]}/
# Deploy images for various DB backends
for dbtag in "${dbtags[@]}"
From f3027f1a6d9931a58c3c86ade4c42f8625a27aba Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 14 Mar 2020 16:27:51 +0300
Subject: [PATCH 005/142] add build stamp to main.go
---
monitoring/exporter/main.go | 10 ++++++++++
1 file changed, 10 insertions(+)
diff --git a/monitoring/exporter/main.go b/monitoring/exporter/main.go
index 45fa3e7fb..c86d7e243 100644
--- a/monitoring/exporter/main.go
+++ b/monitoring/exporter/main.go
@@ -26,6 +26,16 @@ func (l promHTTPLogger) Println(v ...interface{}) {
log.Println(v...)
}
+// Build version number defined by the compiler:
+// -ldflags "-X main.buildstamp=value_to_assign_to_buildstamp"
+// Reported to clients in response to {hi} message.
+// For instance, to define the buildstamp as a timestamp of when the server was built add a
+// flag to compiler command line:
+// -ldflags "-X main.buildstamp=`date -u '+%Y%m%dT%H:%M:%SZ'`"
+// or to set it to git tag:
+// -ldflags "-X main.buildstamp=`git describe --tags`"
+var buildstamp = "undef"
+
func main() {
log.Printf("Tinode metrics exporter.")
From e4c5f5a9195cf874f982e51c1e0854b49814edd8 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 14 Mar 2020 16:28:17 +0300
Subject: [PATCH 006/142] adding build script
---
monitoring/exporter/build.sh | 0
1 file changed, 0 insertions(+), 0 deletions(-)
create mode 100755 monitoring/exporter/build.sh
diff --git a/monitoring/exporter/build.sh b/monitoring/exporter/build.sh
new file mode 100755
index 000000000..e69de29bb
From 66c711c2491654396c04f9e16ce281fe798ae821 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 14 Mar 2020 17:02:28 +0300
Subject: [PATCH 007/142] adding cross-compilation script for exporter
---
monitoring/exporter/build.sh | 66 ++++++++++++++++++++++++++++++++++++
1 file changed, 66 insertions(+)
diff --git a/monitoring/exporter/build.sh b/monitoring/exporter/build.sh
index e69de29bb..a0b4af3b2 100755
--- a/monitoring/exporter/build.sh
+++ b/monitoring/exporter/build.sh
@@ -0,0 +1,66 @@
+#!/bin/bash
+
+# Cross-compiling script using https://github.com/mitchellh/gox
+# This scripts build and archives binaries and supporting files.
+
+# Check if gox is installed. Abort otherwise.
+command -v gox >/dev/null 2>&1 || {
+ echo >&2 "This script requires https://github.com/mitchellh/gox. Please install it before running."; exit 1;
+}
+
+# Supported OSs: darwin windows linux
+goplat=( darwin windows linux )
+# Supported CPU architectures: amd64
+goarc=( amd64 )
+
+for line in $@; do
+ eval "$line"
+done
+
+# Strip 'v' prefix as in v0.16.4 -> 0.16.4.
+version=${tag#?}
+
+if [ -z "$version" ]; then
+ # Get last git tag as release version. Tag looks like 'v.1.2.3', so strip 'v'.
+ version=`git describe --tags`
+ version=${version#?}
+fi
+
+echo "Releasing exporter $version"
+
+GOSRC=${GOPATH}/src/github.com/tinode
+
+pushd ${GOSRC}/chat > /dev/null
+
+# Make sure earlier build is deleted
+rm -f ./releases/${version}/exporter*
+
+for plat in "${goplat[@]}"
+do
+ for arc in "${goarc[@]}"
+ do
+ # Remove previous build
+ rm -f $GOPATH/bin/exporter
+ # Build
+ gox -osarch="${plat}/${arc}" \
+ -ldflags "-s -w -X main.buildstamp=`git describe --tags`" \
+ -output $GOPATH/bin/exporter ./monitoring/exporter > /dev/null
+
+ # Copy binary to release folder for pushing to Github.
+ if [ "$plat" = "windows" ]; then
+ # Copy binaries
+ cp $GOPATH/bin/exporter.exe ./releases/${version}/exporter."${plat}-${arc}".exe
+ else
+ plat2=$plat
+ # Rename 'darwin' tp 'mac'
+ if [ "$plat" = "darwin" ]; then
+ plat2=mac
+ fi
+ # Copy binaries
+ cp $GOPATH/bin/exporter ./releases/${version}/exporter."${plat2}-${arc}"
+ fi
+
+ done
+done
+
+popd > /dev/null
From 90d0e37313b76845c2534eacdd64b170c616fb10 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 14 Mar 2020 19:40:42 +0300
Subject: [PATCH 008/142] possible fix for OperationAborted: A conflicting
conditional operation is currently in progress against this resource. Please
try again.
---
server/media/s3/s3.go | 40 +++++++++++++++++++++-------------------
1 file changed, 21 insertions(+), 19 deletions(-)
diff --git a/server/media/s3/s3.go b/server/media/s3/s3.go
index ab192afc4..8c937b0b9 100644
--- a/server/media/s3/s3.go
+++ b/server/media/s3/s3.go
@@ -95,31 +95,33 @@ func (ah *awshandler) Init(jsconf string) error {
// Check if the bucket exists, create one if not.
_, err = ah.svc.CreateBucket(&s3.CreateBucketInput{Bucket: aws.String(ah.conf.BucketName)})
if err != nil {
+ // Bucket exists or a genuine error.
if aerr, ok := err.(awserr.Error); !ok ||
(aerr.Code() != s3.ErrCodeBucketAlreadyExists &&
aerr.Code() != s3.ErrCodeBucketAlreadyOwnedByYou) {
return err
}
+ } else {
+ // This is a new bucket.
+
+ // The following serves two purposes:
+ // 1. Setup CORS policy to be able to serve media directly from S3.
+ // 2. Verify that the bucket is accessible to the current user.
+ origins := ah.conf.CorsOrigins
+ if len(origins) == 0 {
+ origins = append(origins, "*")
+ }
+ _, err = ah.svc.PutBucketCors(&s3.PutBucketCorsInput{
+ Bucket: aws.String(ah.conf.BucketName),
+ CORSConfiguration: &s3.CORSConfiguration{
+ CORSRules: []*s3.CORSRule{{
+ AllowedMethods: aws.StringSlice([]string{http.MethodGet, http.MethodHead}),
+ AllowedOrigins: aws.StringSlice(origins),
+ AllowedHeaders: aws.StringSlice([]string{"*"}),
+ }},
+ },
+ })
}
-
- // The following serves two purposes:
- // 1. Setup CORS policy to be able to serve media directly from S3.
- // 2. Verify that the bucket is accessible to the current user.
- origins := ah.conf.CorsOrigins
- if len(origins) == 0 {
- origins = append(origins, "*")
- }
- _, err = ah.svc.PutBucketCors(&s3.PutBucketCorsInput{
- Bucket: aws.String(ah.conf.BucketName),
- CORSConfiguration: &s3.CORSConfiguration{
- CORSRules: []*s3.CORSRule{{
- AllowedMethods: aws.StringSlice([]string{http.MethodGet, http.MethodHead}),
- AllowedOrigins: aws.StringSlice(origins),
- AllowedHeaders: aws.StringSlice([]string{"*"}),
- }},
- },
- })
-
return err
}
From 823709fe138d6a956363298801065b5002f0d948 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 14 Mar 2020 19:44:09 +0300
Subject: [PATCH 009/142] gofmt and a required comment with package description
---
server/push/fcm/payload.go | 4 +++-
server/push/fcm/push_fcm.go | 2 ++
2 files changed, 5 insertions(+), 1 deletion(-)
diff --git a/server/push/fcm/payload.go b/server/push/fcm/payload.go
index 0751b0863..0169da397 100644
--- a/server/push/fcm/payload.go
+++ b/server/push/fcm/payload.go
@@ -1,3 +1,5 @@
+// Package fcm is push notification plugin using Google FCM.
+// https://firebase.google.com/docs/cloud-messaging
package fcm
import (
@@ -260,7 +262,7 @@ func PrepareNotifications(rcpt *push.Receipt, config *AndroidConfig) []messageDa
Body: body,
}
}
- messages = append(messages, messageData{Uid: uid, DeviceId: d.DeviceId, Message: &msg})
+ messages = append(messages, messageData{Uid: uid, DeviceId: d.DeviceId, Message: &msg})
}
}
}
diff --git a/server/push/fcm/push_fcm.go b/server/push/fcm/push_fcm.go
index 6bd940c1d..5c8b0d9df 100644
--- a/server/push/fcm/push_fcm.go
+++ b/server/push/fcm/push_fcm.go
@@ -1,5 +1,7 @@
// Package fcm implements push notification plugin for Google FCM backend.
// Push notifications for Android, iOS and web clients are sent through Google's Firebase Cloud Messaging service.
+// Package fcm is push notification plugin using Google FCM.
+// https://firebase.google.com/docs/cloud-messaging
package fcm
import (
From a135268c5f29c3c744ec4049270a18c0672bcbf6 Mon Sep 17 00:00:00 2001
From: aforge
Date: Sat, 14 Mar 2020 21:45:56 -0700
Subject: [PATCH 010/142] TNPG: push all messages in one request and compress
request payloads.
---
server/push/tnpg/push_tnpg.go | 63 ++++++++++++++++++++++-------------
1 file changed, 40 insertions(+), 23 deletions(-)
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index 3fd53e217..53d6052fa 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -3,8 +3,11 @@ package tnpg
import (
"bytes"
+ "compress/gzip"
"encoding/json"
"errors"
+ "fmt"
+ "io"
"log"
"net/http"
@@ -12,6 +15,8 @@ import (
"github.com/tinode/chat/server/push/fcm"
)
+const targetAddress = "https://pushgw.tinode.co/push"
+
var handler Handler
type Handler struct {
@@ -20,11 +25,11 @@ type Handler struct {
}
type configType struct {
- Enabled bool `json:"enabled"`
- Buffer int `json:"buffer"`
- TargetAddress string `json:"target_address"`
- AuthToken string `json:"auth_token"`
- Android fcm.AndroidConfig `json:"android,omitempty"`
+ Enabled bool `json:"enabled"`
+ Buffer int `json:"buffer"`
+ CompressPayloads bool `json:"compress_payloads"`
+ AuthToken string `json:"auth_token"`
+ Android fcm.AndroidConfig `json:"android,omitempty"`
}
// Init initializes the handler
@@ -56,12 +61,27 @@ func (Handler) Init(jsonconf string) error {
}
func postMessage(body []byte, config *configType) (int, string, error) {
- reader := bytes.NewReader(body)
- req, err := http.NewRequest("POST", config.TargetAddress, reader)
+ var reader io.Reader
+ if config.CompressPayloads {
+ var buf bytes.Buffer
+ gz := gzip.NewWriter(&buf)
+ if _, err := gz.Write(body); err != nil {
+ return -1, "", err
+ }
+ gz.Close()
+ reader = &buf
+ } else {
+ reader = bytes.NewReader(body)
+ }
+ req, err := http.NewRequest("POST", targetAddress, reader)
if err != nil {
return -1, "", err
}
req.Header.Add("Authorization", config.AuthToken)
+ req.Header.Set("Content-Type", "application/json; charset=utf-8")
+ if config.CompressPayloads {
+ req.Header.Add("Content-Encoding", "gzip")
+ }
resp, err := http.DefaultClient.Do(req)
if err != nil {
return -1, "", err
@@ -76,22 +96,19 @@ func sendPushes(rcpt *push.Receipt, config *configType) {
return
}
- // TODO:
- // 1. Send multiple payloads in one request.
- // 2. Compress payloads.
- for _, m := range messages {
- msg, err := json.Marshal(m.Message)
- if err != nil {
- log.Println("tnpg push: cannot serialize message", err)
- return
- }
- if code, status, err := postMessage(msg, config); err != nil {
- log.Println("tnpg push failed:", err)
- break
- } else if code >= 300 {
- log.Println("tnpg push rejected:", status, err)
- break
- }
+ messageMap := make(map[string]interface{})
+ for i, m := range messages {
+ messageMap[fmt.Sprintf("message-%d", i)] = m
+ }
+ msgs, err := json.Marshal(messageMap)
+ if err != nil {
+ log.Println("tnpg push: cannot serialize push messages -", err)
+ return
+ }
+ if code, status, err := postMessage(msgs, config); err != nil {
+ log.Println("tnpg push failed:", err)
+ } else if code >= 300 {
+ log.Println("tnpg push rejected:", status, err)
}
}
From 6ec2134a179383e4ad711f0668d2de97a907fe1b Mon Sep 17 00:00:00 2001
From: aforge
Date: Sat, 14 Mar 2020 23:06:44 -0700
Subject: [PATCH 011/142] In TNPG serialized messages directly.
---
server/push/tnpg/push_tnpg.go | 7 +------
1 file changed, 1 insertion(+), 6 deletions(-)
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index 53d6052fa..8c9b2ca8c 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -6,7 +6,6 @@ import (
"compress/gzip"
"encoding/json"
"errors"
- "fmt"
"io"
"log"
"net/http"
@@ -96,11 +95,7 @@ func sendPushes(rcpt *push.Receipt, config *configType) {
return
}
- messageMap := make(map[string]interface{})
- for i, m := range messages {
- messageMap[fmt.Sprintf("message-%d", i)] = m
- }
- msgs, err := json.Marshal(messageMap)
+ msgs, err := json.Marshal(messages)
if err != nil {
log.Println("tnpg push: cannot serialize push messages -", err)
return
From 1db6c4e60b4288a67fcd14847a9a9ee674b29008 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 15 Mar 2020 12:19:30 +0300
Subject: [PATCH 012/142] fix for double routing of {sub topic=new} in cluster
---
server/cluster.go | 17 ++++++++++++++++-
server/datamodel.go | 2 +-
server/push/fcm/payload.go | 2 --
server/session.go | 6 +++---
4 files changed, 20 insertions(+), 7 deletions(-)
diff --git a/server/cluster.go b/server/cluster.go
index b4035b866..056d92899 100644
--- a/server/cluster.go
+++ b/server/cluster.go
@@ -323,7 +323,7 @@ type Cluster struct {
// Master at topic's master node receives C2S messages from topic's proxy nodes.
// The message is treated like it came from a session: find or create a session locally,
-// dispatch the message to it like it came from a normal ws/lp connection.
+// dispatch the message to it like it came from a normal ws/lp/gRPC connection.
// Called by a remote node.
func (c *Cluster) Master(msg *ClusterReq, rejected *bool) error {
// Find the local session associated with the given remote session.
@@ -523,6 +523,21 @@ func (c *Cluster) isRemoteTopic(topic string) bool {
return c.ring.Get(topic) != c.thisNodeName
}
+// genLocalTopicName is just like genTopicName(), but the generated name belongs to the current cluster node.
+func (c *Cluster) genLocalTopicName() string {
+ topic := genTopicName()
+ if c == nil {
+ // Cluster not initialized, all topics are local
+ return topic
+ }
+
+ // FIXME: if cluster is large it may become too inefficient.
+ for c.ring.Get(topic) != c.thisNodeName {
+ topic = genTopicName()
+ }
+ return topic
+}
+
// Returns remote node name where the topic is hosted.
// If the topic is hosted locally, returns an empty string.
func (c *Cluster) nodeNameForTopicIfRemote(topic string) string {
diff --git a/server/datamodel.go b/server/datamodel.go
index 33bf599f6..f1044d475 100644
--- a/server/datamodel.go
+++ b/server/datamodel.go
@@ -307,7 +307,7 @@ type ClientComMessage struct {
// Message ID denormalized
id string
- // Topic denormalized
+ // Un-routable (original) topic name denormalized from XXX.Topic.
topic string
// Sender's UserId as string
from string
diff --git a/server/push/fcm/payload.go b/server/push/fcm/payload.go
index 0169da397..bf8068ca2 100644
--- a/server/push/fcm/payload.go
+++ b/server/push/fcm/payload.go
@@ -1,5 +1,3 @@
-// Package fcm is push notification plugin using Google FCM.
-// https://firebase.google.com/docs/cloud-messaging
package fcm
import (
diff --git a/server/session.go b/server/session.go
index b9f50ee71..b73580943 100644
--- a/server/session.go
+++ b/server/session.go
@@ -403,10 +403,10 @@ func (s *Session) subscribe(msg *ClientComMessage) {
var expanded string
isNewTopic := false
if strings.HasPrefix(msg.topic, "new") {
- // Request to create a new named topic
- expanded = genTopicName()
+ // Request to create a new named topic.
+ // If we are in a cluster, make sure the new topic belongs to the current node.
+ expanded = globals.cluster.genLocalTopicName()
isNewTopic = true
- // msg.topic = expanded
} else {
var resp *ServerComMessage
expanded, resp = s.expandTopicName(msg)
From 1b36ace0956c8ab6c1d2c38b7cf68572b2d24838 Mon Sep 17 00:00:00 2001
From: aforge
Date: Sun, 15 Mar 2020 14:19:53 -0700
Subject: [PATCH 013/142] Send only []fcm.Message to TNPG.
---
server/push/tnpg/push_tnpg.go | 6 +++++-
1 file changed, 5 insertions(+), 1 deletion(-)
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index 8c9b2ca8c..c82dd2a95 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -95,7 +95,11 @@ func sendPushes(rcpt *push.Receipt, config *configType) {
return
}
- msgs, err := json.Marshal(messages)
+ var payloads []interface{}
+ for _, m := range messages {
+ payloads = append(payloads, m.Message)
+ }
+ msgs, err := json.Marshal(payloads)
if err != nil {
log.Println("tnpg push: cannot serialize push messages -", err)
return
From 5657d513753895e497fc221e419b9ab68f350820 Mon Sep 17 00:00:00 2001
From: aforge
Date: Mon, 16 Mar 2020 00:21:08 -0700
Subject: [PATCH 014/142] Apply json.Encoder.Encode directly on push payloads.
---
server/push/tnpg/push_tnpg.go | 24 +++++++-----------------
1 file changed, 7 insertions(+), 17 deletions(-)
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index c82dd2a95..8d4ec2848 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -6,7 +6,6 @@ import (
"compress/gzip"
"encoding/json"
"errors"
- "io"
"log"
"net/http"
@@ -59,20 +58,16 @@ func (Handler) Init(jsonconf string) error {
return nil
}
-func postMessage(body []byte, config *configType) (int, string, error) {
- var reader io.Reader
+func postMessage(body interface{}, config *configType) (int, string, error) {
+ buf := new(bytes.Buffer)
if config.CompressPayloads {
- var buf bytes.Buffer
- gz := gzip.NewWriter(&buf)
- if _, err := gz.Write(body); err != nil {
- return -1, "", err
- }
+ gz := gzip.NewWriter(buf)
+ json.NewEncoder(gz).Encode(body)
gz.Close()
- reader = &buf
} else {
- reader = bytes.NewReader(body)
+ json.NewEncoder(buf).Encode(body)
}
- req, err := http.NewRequest("POST", targetAddress, reader)
+ req, err := http.NewRequest("POST", targetAddress, buf)
if err != nil {
return -1, "", err
}
@@ -99,12 +94,7 @@ func sendPushes(rcpt *push.Receipt, config *configType) {
for _, m := range messages {
payloads = append(payloads, m.Message)
}
- msgs, err := json.Marshal(payloads)
- if err != nil {
- log.Println("tnpg push: cannot serialize push messages -", err)
- return
- }
- if code, status, err := postMessage(msgs, config); err != nil {
+ if code, status, err := postMessage(payloads, config); err != nil {
log.Println("tnpg push failed:", err)
} else if code >= 300 {
log.Println("tnpg push rejected:", status, err)
From 36717d8d7af2f3a74df2a98affd640b7ca21c03b Mon Sep 17 00:00:00 2001
From: aforge
Date: Mon, 16 Mar 2020 18:20:55 -0700
Subject: [PATCH 015/142] TNPG: send pushes in batches.
---
server/push/tnpg/push_tnpg.go | 30 ++++++++++++++++++++++--------
1 file changed, 22 insertions(+), 8 deletions(-)
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index 8d4ec2848..5700a6d1c 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -26,6 +26,7 @@ type configType struct {
Enabled bool `json:"enabled"`
Buffer int `json:"buffer"`
CompressPayloads bool `json:"compress_payloads"`
+ BatchSize int `json:"batch_size"`
AuthToken string `json:"auth_token"`
Android fcm.AndroidConfig `json:"android,omitempty"`
}
@@ -41,6 +42,10 @@ func (Handler) Init(jsonconf string) error {
return nil
}
+ if config.BatchSize <= 0 {
+ return errors.New("push.tnpg.batch_size should be greater than zero.")
+ }
+
handler.input = make(chan *push.Receipt, config.Buffer)
handler.stop = make(chan bool, 1)
@@ -90,14 +95,23 @@ func sendPushes(rcpt *push.Receipt, config *configType) {
return
}
- var payloads []interface{}
- for _, m := range messages {
- payloads = append(payloads, m.Message)
- }
- if code, status, err := postMessage(payloads, config); err != nil {
- log.Println("tnpg push failed:", err)
- } else if code >= 300 {
- log.Println("tnpg push rejected:", status, err)
+ n := len(messages)
+ for i := 0; i < n; i += config.BatchSize {
+ upper := i + config.BatchSize
+ if upper > n {
+ upper = n
+ }
+ var payloads []interface{}
+ for j := i; j < upper; j++ {
+ payloads = append(payloads, messages[j].Message)
+ }
+ if code, status, err := postMessage(payloads, config); err != nil {
+ log.Println("tnpg push failed:", err)
+ break
+ } else if code >= 300 {
+ log.Println("tnpg push rejected:", status, err)
+ break
+ }
}
}
From 277cb639d1dac17a27d0e3a443024c19c6979c38 Mon Sep 17 00:00:00 2001
From: aforge
Date: Mon, 16 Mar 2020 20:36:46 -0700
Subject: [PATCH 016/142] Correct Authorization header and push path in TNPG.
---
server/push/tnpg/push_tnpg.go | 18 ++++++++++++------
1 file changed, 12 insertions(+), 6 deletions(-)
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index 5700a6d1c..c5a95c3c1 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -13,7 +13,7 @@ import (
"github.com/tinode/chat/server/push/fcm"
)
-const targetAddress = "https://pushgw.tinode.co/push"
+const baseTargetAddress = "https://pushgw.tinode.co/push/"
var handler Handler
@@ -23,10 +23,11 @@ type Handler struct {
}
type configType struct {
- Enabled bool `json:"enabled"`
- Buffer int `json:"buffer"`
- CompressPayloads bool `json:"compress_payloads"`
- BatchSize int `json:"batch_size"`
+ Enabled bool `json:"enabled"`
+ Buffer int `json:"buffer"`
+ CompressPayloads bool `json:"compress_payloads"`
+ BatchSize int `json:"batch_size"`
+ User string `json:"user"`
AuthToken string `json:"auth_token"`
Android fcm.AndroidConfig `json:"android,omitempty"`
}
@@ -46,6 +47,10 @@ func (Handler) Init(jsonconf string) error {
return errors.New("push.tnpg.batch_size should be greater than zero.")
}
+ if len(config.User) == 0 {
+ return errors.New("push.tnpg.user not specified.")
+ }
+
handler.input = make(chan *push.Receipt, config.Buffer)
handler.stop = make(chan bool, 1)
@@ -72,11 +77,12 @@ func postMessage(body interface{}, config *configType) (int, string, error) {
} else {
json.NewEncoder(buf).Encode(body)
}
+ targetAddress := baseTargetAddress + config.User
req, err := http.NewRequest("POST", targetAddress, buf)
if err != nil {
return -1, "", err
}
- req.Header.Add("Authorization", config.AuthToken)
+ req.Header.Add("Authorization", "Bearer " + config.AuthToken)
req.Header.Set("Content-Type", "application/json; charset=utf-8")
if config.CompressPayloads {
req.Header.Add("Content-Encoding", "gzip")
From ca7e96e3e07ebf516d36bba36cffa72dcd70d941 Mon Sep 17 00:00:00 2001
From: aforge
Date: Mon, 16 Mar 2020 20:40:20 -0700
Subject: [PATCH 017/142] TNPG: hardcode batch size.
---
server/push/tnpg/push_tnpg.go | 14 ++++++--------
1 file changed, 6 insertions(+), 8 deletions(-)
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index c5a95c3c1..5282bc49c 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -13,7 +13,10 @@ import (
"github.com/tinode/chat/server/push/fcm"
)
-const baseTargetAddress = "https://pushgw.tinode.co/push/"
+const (
+ baseTargetAddress = "https://pushgw.tinode.co/push/"
+ batchSize = 100
+)
var handler Handler
@@ -26,7 +29,6 @@ type configType struct {
Enabled bool `json:"enabled"`
Buffer int `json:"buffer"`
CompressPayloads bool `json:"compress_payloads"`
- BatchSize int `json:"batch_size"`
User string `json:"user"`
AuthToken string `json:"auth_token"`
Android fcm.AndroidConfig `json:"android,omitempty"`
@@ -43,10 +45,6 @@ func (Handler) Init(jsonconf string) error {
return nil
}
- if config.BatchSize <= 0 {
- return errors.New("push.tnpg.batch_size should be greater than zero.")
- }
-
if len(config.User) == 0 {
return errors.New("push.tnpg.user not specified.")
}
@@ -102,8 +100,8 @@ func sendPushes(rcpt *push.Receipt, config *configType) {
}
n := len(messages)
- for i := 0; i < n; i += config.BatchSize {
- upper := i + config.BatchSize
+ for i := 0; i < n; i += batchSize {
+ upper := i + batchSize
if upper > n {
upper = n
}
From 76cad4a7ed6810d46e1fb4015dd439d19f60c682 Mon Sep 17 00:00:00 2001
From: aforge
Date: Tue, 17 Mar 2020 01:19:12 -0700
Subject: [PATCH 018/142] Update README file for metrics exporters.
---
monitoring/exporter/README.md | 63 ++++++++++++++++++++++++++++++-----
1 file changed, 54 insertions(+), 9 deletions(-)
diff --git a/monitoring/exporter/README.md b/monitoring/exporter/README.md
index 38e08389f..ba0238e19 100644
--- a/monitoring/exporter/README.md
+++ b/monitoring/exporter/README.md
@@ -1,18 +1,63 @@
-# Prometheus `expvar` Exporter
+# Tinode Metric Exporter
-This is a [prometheus](https://prometheus.io/) [exporter](https://prometheus.io/docs/instrumenting/exporters/): a service which reads JSON monitoring data exposed by Tinode server using [expvar](https://golang.org/pkg/expvar/) and re-publishes it in [prometheus format](https://prometheus.io/docs/concepts/data_model/).
+This is a simple service which reads JSON monitoring data exposed by Tinode server using [expvar](https://golang.org/pkg/expvar/) and re-publishes it in other formats.
+Currently, supported are:
+* [Prometheus](https://prometheus.io/) [exporter](https://prometheus.io/docs/instrumenting/exporters/) which exports data in [prometheus format](https://prometheus.io/docs/concepts/data_model/).
+Note that the monitoring service is expected to pull/scrape data from Prometheus exporter.
+* [InfluxDB](https://www.influxdata.com/) [exporter](https://docs.influxdata.com/influxdb/v1.7/tools/api/#write-http-endpoint), on the contrary, pushes data to its target backend.
## Usage
-Run this service as
-```
-./prometheus --tinode_addr=http://localhost:6060/stats/expvar \
- --namespace=tinode --listen_at=:6222 --metrics_path=/metrics
-```
+Exporters are expected to run next to (pair with) Tinode servers: one Exporter per one Tinode server, i.e. a single Exporter provides metrics from a single Tinode server.
+Currently, the Exporter is fully configured via command line flags. There are three sets of flags:
+### Common flags
+* `serve_for` specifies which monitoring service the Exporter will gather metrics for.
* `tinode_addr` is the address where the Tinode instance publishes `expvar` data to scrape.
-* `namespace` is a prefix to use for metrics names. If you are monitoring multiple tinode instances you may want to use different namespaces.
* `listen_at` is the hostname to bind to for serving the metrics.
-* `metrics_path` path under which to expose the metrics.
+* `instance` is the Exporter instance name (it may be exported to the upstream backend).
+* `metric_list` is a comma-separated list of metrics to export.
+
+### Prometheus
+* `prom_namespace` is a prefix to use for metrics names. If you are monitoring multiple tinode instances you may want to use different namespaces.
+* `prom_metrics_path` is the path under which to expose the metrics for scraping.
+### InfluxDB
+* `influx_push_addr` is the address of InfluxDB target server where the data gets sent.
+* `influx_db_version` is the version of InfluxDB (only 1.7 and 2.0 are supported).
+* `influx_organization` specifies InfluxDB organization to push metrics as.
+* `influx_bucket` is the name of InfluxDB storage bucket to store data in (used only in InfluxDB 2.0).
+* `influx_auth_token` - InfluxDB authentication token.
+* `influx_push_interval` - InfluxDB push interval in seconds.
+
+## Examples
+Run Prometheus Exporter as
+```
+./exporter \
+ --serve_for=prometheus \
+ --tinode_addr=http://localhost:6060/stats/expvar \
+ --listen_at=:6222 \
+ --instance=exp-0 \
+ --prom_namespace=tinode \
+ --prom_metrics_path=/metrics \
+ --prom_timeout=15
+```
+
+This exporter will serve data at path /metrics, on port 6222.
Once running, configure your Prometheus monitoring installation to collect data from this exporter.
+
+Run InfluxDB Exporter as
+```
+./exporter \
+ --serve_for=influxdb \
+ --tinode_addr=http://localhost:6060/stats/expvar \
+ --listen_at=:6222 \
+ --instance=exp-0 \
+ --influx_push_addr=http://my-influxdb-backend.net/write \
+ --influx_db_version=1.7 \
+ --influx_organization=myOrg \
+ --influx_auth_token=myAuthToken123 \
+ --influx_push_interval=30
+```
+
+This exporter will push the collected metrics to the specified backend once every 30 seconds.
From 9881106c2760bb38c5d7ec242b2daa8a8b35e8ab Mon Sep 17 00:00:00 2001
From: aforge
Date: Tue, 17 Mar 2020 20:56:13 -0700
Subject: [PATCH 019/142] More info in monitoring README.md.
---
monitoring/README.md | 6 +++++-
1 file changed, 5 insertions(+), 1 deletion(-)
diff --git a/monitoring/README.md b/monitoring/README.md
index 4cdd12c6a..cf1867973 100644
--- a/monitoring/README.md
+++ b/monitoring/README.md
@@ -1,3 +1,7 @@
# Monitoring Support
-This directory contains code related to monitoring Tinode server. Only [Prometheus](https://prometheus.io/) is [supported](./prometheus/) at this time.
+This directory contains code related to monitoring Tinode server. Supported monitoring services are
+* [Prometheus](https://prometheus.io/)
+* [InfluxDB](https://www.influxdata.com/)
+
+See [exporter/README](./exporter/README.md) for more details.
From 7a8f8bcc760a8c0b7916c1b4423190095df0b33c Mon Sep 17 00:00:00 2001
From: or-else
Date: Wed, 18 Mar 2020 09:55:41 +0300
Subject: [PATCH 020/142] mention default values
---
monitoring/exporter/README.md | 27 ++++++++++++++-------------
1 file changed, 14 insertions(+), 13 deletions(-)
diff --git a/monitoring/exporter/README.md b/monitoring/exporter/README.md
index ba0238e19..261fd4226 100644
--- a/monitoring/exporter/README.md
+++ b/monitoring/exporter/README.md
@@ -12,23 +12,24 @@ Exporters are expected to run next to (pair with) Tinode servers: one Exporter p
Currently, the Exporter is fully configured via command line flags. There are three sets of flags:
### Common flags
-* `serve_for` specifies which monitoring service the Exporter will gather metrics for.
-* `tinode_addr` is the address where the Tinode instance publishes `expvar` data to scrape.
-* `listen_at` is the hostname to bind to for serving the metrics.
-* `instance` is the Exporter instance name (it may be exported to the upstream backend).
-* `metric_list` is a comma-separated list of metrics to export.
+* `serve_for` specifies which monitoring service the Exporter will gather metrics for; accepted values: `influxdb`, `prometheus`; default: `influxdb`.
+* `tinode_addr` is the address where the Tinode instance publishes `expvar` data to scrape; default: `http://localhost:6060/stats/expvar`.
+* `listen_at` is the hostname to bind to for serving the metrics; default: `:6222`.
+* `instance` is the Exporter instance name (it may be exported to the upstream backend); default: `exporter`.
+* `metric_list` is a comma-separated list of metrics to export; default: `Version,LiveTopics,TotalTopics,LiveSessions,ClusterLeader,TotalClusterNodes,LiveClusterNodes,memstats.Alloc`.
### Prometheus
-* `prom_namespace` is a prefix to use for metrics names. If you are monitoring multiple tinode instances you may want to use different namespaces.
-* `prom_metrics_path` is the path under which to expose the metrics for scraping.
+* `prom_namespace` is a prefix to use for metrics names. If you are monitoring multiple tinode instances you may want to use different namespaces; default: `tinode`.
+* `prom_metrics_path` is the path under which to expose the metrics for scraping; default: `/metrics`.
+* `prom_timeout` is the Tinode connection timeout in seconds in response to Prometheus scrapes; default: `15`.
### InfluxDB
-* `influx_push_addr` is the address of InfluxDB target server where the data gets sent.
-* `influx_db_version` is the version of InfluxDB (only 1.7 and 2.0 are supported).
-* `influx_organization` specifies InfluxDB organization to push metrics as.
-* `influx_bucket` is the name of InfluxDB storage bucket to store data in (used only in InfluxDB 2.0).
-* `influx_auth_token` - InfluxDB authentication token.
-* `influx_push_interval` - InfluxDB push interval in seconds.
+* `influx_push_addr` is the address of InfluxDB target server where the data gets sent; default: `http://localhost:9999/write`.
+* `influx_db_version` is the version of InfluxDB (only 1.7 and 2.0 are supported); default: `1.7`.
+* `influx_organization` specifies InfluxDB organization to push metrics as; default: `test`;
+* `influx_bucket` is the name of InfluxDB storage bucket to store data in (used only in InfluxDB 2.0); default: `test`.
+* `influx_auth_token` - InfluxDB authentication token; no default value.
+* `influx_push_interval` - InfluxDB push interval in seconds; default: `30`.
## Examples
Run Prometheus Exporter as
From e2409157b501902a4ac82b88b817977a3cf27cc9 Mon Sep 17 00:00:00 2001
From: or-else
Date: Wed, 18 Mar 2020 17:37:31 +0300
Subject: [PATCH 021/142] add support for unix sockets
---
server/hdl_grpc.go | 3 +--
server/http.go | 38 ++++++++++++++++++++++++++------------
server/tinode.conf | 11 +++++++----
server/utils.go | 17 +++++++++++++++++
4 files changed, 51 insertions(+), 18 deletions(-)
diff --git a/server/hdl_grpc.go b/server/hdl_grpc.go
index bcdd6e186..cb72a8a10 100644
--- a/server/hdl_grpc.go
+++ b/server/hdl_grpc.go
@@ -13,7 +13,6 @@ import (
"crypto/tls"
"io"
"log"
- "net"
"time"
"github.com/tinode/chat/pbx"
@@ -115,7 +114,7 @@ func serveGrpc(addr string, kaEnabled bool, tlsConf *tls.Config) (*grpc.Server,
return nil, nil
}
- lis, err := net.Listen("tcp", addr)
+ lis, err := netListener(addr)
if err != nil {
return nil, err
}
diff --git a/server/http.go b/server/http.go
index 084ee7866..31c3d93da 100644
--- a/server/http.go
+++ b/server/http.go
@@ -13,6 +13,7 @@ import (
"crypto/tls"
"encoding/base64"
"encoding/json"
+ "errors"
"log"
"net"
"net/http"
@@ -34,7 +35,6 @@ func listenAndServe(addr string, mux *http.ServeMux, tlfConf *tls.Config, stop <
httpdone := make(chan bool)
server := &http.Server{
- Addr: addr,
Handler: mux,
}
@@ -45,24 +45,38 @@ func listenAndServe(addr string, mux *http.ServeMux, tlfConf *tls.Config, stop <
if server.TLSConfig != nil {
// If port is not specified, use default https port (443),
// otherwise it will default to 80
- if server.Addr == "" {
- server.Addr = ":https"
+ if addr == "" {
+ addr = ":https"
}
if globals.tlsRedirectHTTP != "" {
- log.Printf("Redirecting connections from HTTP at [%s] to HTTPS at [%s]",
- globals.tlsRedirectHTTP, server.Addr)
-
- // This is a second HTTP server listenning on a different port.
- go http.ListenAndServe(globals.tlsRedirectHTTP, tlsRedirect(server.Addr))
+ // Serving redirects from a unix socket or to a unix socket makes no sense.
+ if isUnixAddr(globals.tlsRedirectHTTP) || isUnixAddr(addr) {
+ err = errors.New("HTTP to HTTPS redirect: unix sockets not supported.")
+ } else {
+ log.Printf("Redirecting connections from HTTP at [%s] to HTTPS at [%s]",
+ globals.tlsRedirectHTTP, addr)
+
+ // This is a second HTTP server listenning on a different port.
+ go http.ListenAndServe(globals.tlsRedirectHTTP, tlsRedirect(addr))
+ }
}
- log.Printf("Listening for client HTTPS connections on [%s]", server.Addr)
- err = server.ListenAndServeTLS("", "")
+ if err == nil {
+ log.Printf("Listening for client HTTPS connections on [%s]", addr)
+ lis, err := netListener(addr)
+ if err == nil {
+ err = server.ServeTLS(lis, "", "")
+ }
+ }
} else {
- log.Printf("Listening for client HTTP connections on [%s]", server.Addr)
- err = server.ListenAndServe()
+ log.Printf("Listening for client HTTP connections on [%s]", addr)
+ lis, err := netListener(addr)
+ if err == nil {
+ err = server.Serve(lis)
+ }
}
+
if err != nil {
if globals.shuttingDown {
log.Println("HTTP server: stopped")
diff --git a/server/tinode.conf b/server/tinode.conf
index 9a36c5dab..333a80b1e 100644
--- a/server/tinode.conf
+++ b/server/tinode.conf
@@ -1,8 +1,9 @@
// The JSON comments are somewhat brittle. Don't try anything too fancy.
{
- // HTTP(S) address:port to listen on for websocket and long polling clients. Either a
- // numeric value or a canonical name, e.g. ":80" or ":https". May include the host name, e.g.
- // "localhost:80" or "hostname.example.com:https".
+ // HTTP(S) address to listen on for websocket and long polling clients. Either a TCP host:port pair
+ // or a path to Unix socket as "unix:/path/to/socket.sock".
+ // The TCP port is numeric value or a canonical name, e.g. ":80" or ":https". May include the host name,
+ ///e.g. "localhost:80" or "hostname.example.com:https".
// It could be blank: if TLS is not configured it will default to ":80", otherwise to ":443".
// Can be overridden from the command line, see option --listen.
"listen": ":6060",
@@ -17,7 +18,8 @@
// URL path for mounting the directory with static files.
"static_mount": "/",
- // Address:port to listen for gRPC clients. Leave blank to disable gRPC support.
+ // TCP host:port or unix:/path/to/socket to listen for gRPC clients.
+ // Leave blank to disable gRPC support.
// Could be overridden from the command line with --grpc_listen.
"grpc_listen": ":6061",
@@ -85,6 +87,7 @@
"enabled": false,
// Listen for connections on this port and redirect them to HTTPS port.
+ // Cannot be a Unix socket.
"http_redirect": ":80",
// Add Strict-Transport-Security to headers, the value signifies age.
diff --git a/server/utils.go b/server/utils.go
index 122202d78..560f643e4 100644
--- a/server/utils.go
+++ b/server/utils.go
@@ -9,6 +9,7 @@ import (
"errors"
"fmt"
"io"
+ "net"
"path/filepath"
"reflect"
"regexp"
@@ -779,3 +780,19 @@ func offsetToLineAndChar(r io.Reader, offset int64) (int, int, error) {
return lnum, cnum, nil
}
+
+// netListener creates net.Listener for tcp and unix domains:
+// if addr is is in the form "unix:/run/tinode.sock" it's a unix socket, otherwise TCP host:port.
+func netListener(addr string) (net.Listener, error) {
+ addrParts := strings.SplitN(addr, ":", 2)
+ if len(addrParts) == 2 && addrParts[0] == "unix" {
+ return net.Listen("unix", addrParts[1])
+ }
+ return net.Listen("tcp", addr)
+}
+
+// Check if specified address is a unix socket like "unix:/run/tinode.sock".
+func isUnixAddr(addr string) bool {
+ addrParts := strings.SplitN(addr, ":", 2)
+ return len(addrParts) == 2 && addrParts[0] == "unix"
+}
From 7905bf5197d6bff59a7fa9f283eca16de8df7a9f Mon Sep 17 00:00:00 2001
From: or-else
Date: Wed, 18 Mar 2020 17:37:51 +0300
Subject: [PATCH 022/142] better handling of s3 bucket errors
---
server/media/s3/s3.go | 12 +++++++-----
1 file changed, 7 insertions(+), 5 deletions(-)
diff --git a/server/media/s3/s3.go b/server/media/s3/s3.go
index 8c937b0b9..f96e96f0b 100644
--- a/server/media/s3/s3.go
+++ b/server/media/s3/s3.go
@@ -95,11 +95,13 @@ func (ah *awshandler) Init(jsconf string) error {
// Check if the bucket exists, create one if not.
_, err = ah.svc.CreateBucket(&s3.CreateBucketInput{Bucket: aws.String(ah.conf.BucketName)})
if err != nil {
- // Bucket exists or a genuine error.
- if aerr, ok := err.(awserr.Error); !ok ||
- (aerr.Code() != s3.ErrCodeBucketAlreadyExists &&
- aerr.Code() != s3.ErrCodeBucketAlreadyOwnedByYou) {
- return err
+ // Check if bucket already exists or a genuine error.
+ if aerr, ok := err.(awserr.Error); ok {
+ if aerr.Code() == s3.ErrCodeBucketAlreadyExists ||
+ aerr.Code() == s3.ErrCodeBucketAlreadyOwnedByYou {
+ // Clear benign error
+ err = nil
+ }
}
} else {
// This is a new bucket.
From ccb57e168c45f1f3873ab63dc68f0fb7dd32bc51 Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 19 Mar 2020 00:51:32 -0700
Subject: [PATCH 023/142] Exporter Docker configuration.
---
docker/README.md | 30 +++++++++++++++++-
docker/exporter/Dockerfile | 37 +++++++++++++++++++++++
docker/exporter/entrypoint.sh | 57 +++++++++++++++++++++++++++++++++++
3 files changed, 123 insertions(+), 1 deletion(-)
create mode 100644 docker/exporter/Dockerfile
create mode 100755 docker/exporter/entrypoint.sh
diff --git a/docker/README.md b/docker/README.md
index 38f9f1296..263e40d55 100644
--- a/docker/README.md
+++ b/docker/README.md
@@ -126,7 +126,6 @@ See [instructions](../chatbot/python/).
The chatbot password is generated only when the database is initialized or reset. It's saved to `/botdata` directory in the container. If you want to keep the data available between container changes, such as image upgrades, make sure the `/botdata` is a mounted volume (i.e. you always launch the container with `--volume botdata:/botdata` option).
-
## Supported environment variables
You can specify the following environment variables when issuing `docker run` command:
@@ -172,3 +171,32 @@ A convenient way to generate a desired number of random bytes and base64-encode
```
$ openssl rand -base64
```
+
+## Metrics Exporter
+
+See [monitoring/exporter/README](../../monitoring/exporter/README.md) for information on the Exporter.
+Container is also available as a part of the Tinode docker distribution: `tinode/exporter`.
+Run it with
+
+```
+$ docker run -p 6222:6222 -d --name tinode-exporter --network tinode-net \
+ --env SERVE_FOR= \
+ --env TINODE_ADDR= \
+ --env LISTEN_AT=":6222" \
+ ... \
+ tinode/exporter:latest
+```
+
+Available variables:
+| Variable | Type | Default | Function |
+| --- | --- | --- | --- |
+| `SERVE_FOR` | string | `` | Either `prometheus` or `influxdb` |
+| `TINODE_ADDR` | string | `http://localhost/stats/expvar/` | Tinode metrics path |
+| `LISTEN_AT` | string | `:6222` | Exporter web server host and port |
+| `INFLUXDB_VERSION` | string | `1.7` | InfluxDB version (`1.7` or `2.0`) |
+| `INFLUXDB_ORGANIZATION` | string | `org` | InfluxDB organization |
+| `INFLUXDB_PUSH_INTERVAL` | int | `60` | Exporter's metrics push interval in seconds |
+| `INFLUXDB_PUSH_ADDRESS` | string | `https://mon.tinode.co/intake` | InfluxDB backend url |
+| `INFLUXDB_AUTH_TOKEN` | string | `Your-token` | InfluxDB auth token |
+| `PROM_NAMESPACE` | string | `tinode` | Prometheus namespace |
+| `PROM_METRICS_PATH` | string | `/metrics` | Exporter webserver path that Prometheus server scrapes |
diff --git a/docker/exporter/Dockerfile b/docker/exporter/Dockerfile
new file mode 100644
index 000000000..6032f79ae
--- /dev/null
+++ b/docker/exporter/Dockerfile
@@ -0,0 +1,37 @@
+FROM alpine:latest
+
+ARG VERSION=0.16.4
+ENV VERSION=$VERSION
+
+ENV SERVE_FOR=""
+
+ENV TINODE_ADDR=http://localhost/stats/expvar/
+ENV INSTANCE="exporter-instance"
+ENV LISTEN_AT=":6222"
+
+ENV INFLUXDB_VERSION=1.7
+ENV INFLUXDB_ORGANIZATION="org"
+ENV INFLUXDB_PUSH_INTERVAL=60
+ENV INFLUXDB_PUSH_ADDRESS=http://localhost:6222/write
+ENV INFLUXDB_AUTH_TOKEN="Your-token"
+
+ENV PROM_NAMESPACE="tinode"
+ENV PROM_METRICS_PATH="/metrics"
+
+LABEL maintainer="Tinode Team "
+LABEL name="TinodeMetricExporter"
+LABEL version=$VERSION
+
+WORKDIR /opt/tinode
+
+RUN apk add --no-cache bash
+
+# Fetch exporter build from Github.
+ADD https://github.com/tinode/chat/releases/download/v$VERSION/exporter.linux-amd64 ./exporter
+
+COPY entrypoint.sh .
+RUN chmod +x exporter && chmod +x entrypoint.sh
+
+ENTRYPOINT ./entrypoint.sh
+
+EXPOSE 6222
diff --git a/docker/exporter/entrypoint.sh b/docker/exporter/entrypoint.sh
new file mode 100755
index 000000000..1523edb12
--- /dev/null
+++ b/docker/exporter/entrypoint.sh
@@ -0,0 +1,57 @@
+#!/bin/bash
+
+# Check if environment variables (provided as argument list) are set.
+function check_vars() {
+ local varnames=( "$@" )
+ for varname in "${varnames[@]}"
+ do
+ eval value=\$${varname}
+ if [ -z "$value" ] ; then
+ echo "$varname env var must be specified."
+ exit 1
+ fi
+ done
+}
+
+echo "hosts: files dns" > /etc/nsswitch.conf
+
+# Required env vars.
+common_vars=( TINODE_ADDR INSTANCE LISTEN_AT SERVE_FOR )
+
+influx_varnames=( INFLUXDB_VERSION INFLUXDB_ORGANIZATION INFLUXDB_PUSH_INTERVAL \
+ INFLUXDB_PUSH_ADDRESS INFLUXDB_AUTH_TOKEN )
+
+prometheus_varnames=( PROM_NAMESPACE PROM_METRICS_PATH )
+
+check_vars "${common_vars[@]}"
+
+# Common arguments.
+args=("--tinode_addr=${TINODE_ADDR}" "--instance=${INSTANCE}" "--listen_at=${LISTEN_AT}" "--serve_for=${SERVE_FOR}")
+
+# Platform-specific arguments.
+case "$SERVE_FOR" in
+"prometheus")
+ check_vars "${prometheus_varnames[@]}"
+ args+=("--prom_namespace=${PROM_NAMESPACE}" "--prom_metrics_path=${PROM_METRICS_PATH}")
+ if [ ! -z "$PROM_TIMEOUT" ]; then
+ args+=("--prom_timeout=${PROM_TIMEOUT}")
+ fi
+ ;;
+"influxdb")
+ check_vars "${influxdb_varnames[@]}"
+ args+=("--influx_db_version=${INFLUXDB_VERSION}" \
+ "--influx_organization=${INFLUXDB_ORGANIZATION}" \
+ "--influx_push_interval=${INFLUXDB_PUSH_INTERVAL}" \
+ "--influx_push_addr=${INFLUXDB_PUSH_ADDRESS}" \
+ "--influx_auth_token=${INFLUXDB_AUTH_TOKEN}")
+ if [ ! -z "$INFLUXDB_BUCKET" ]; then
+ args+=("--influx_bucket=${INFLUXDB_BUCKET}")
+ fi
+ ;;
+*)
+ echo "\$SERVE_FOR must be set to either 'prometheus' or 'influxdb'"
+ exit 1
+ ;;
+esac
+
+./exporter "${args[@]}"
From 0f7738afa02d45d19d85ba40aa3b09118a78a55e Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 19 Mar 2020 00:57:31 -0700
Subject: [PATCH 024/142] Add tinode/exporter docker to docker build and
release scripts.
---
docker-build.sh | 10 ++++++++++
docker-release.sh | 16 ++++++++++++++++
2 files changed, 26 insertions(+)
diff --git a/docker-build.sh b/docker-build.sh
index d7d042627..7a649ca21 100755
--- a/docker-build.sh
+++ b/docker-build.sh
@@ -53,3 +53,13 @@ if [ -n "$FULLRELEASE" ]; then
fi
docker rmi ${rmitags}
docker build --build-arg VERSION=$tag ${buildtags} docker/chatbot
+
+# Build exporter image
+buildtags="--tag tinode/exporter:${ver[0]}.${ver[1]}.${ver[2]}"
+rmitags="tinode/exporter:${ver[0]}.${ver[1]}.${ver[2]}"
+if [ -n "$FULLRELEASE" ]; then
+ rmitags="${rmitags} tinode/exporter:latest tinode/exporter:${ver[0]}.${ver[1]}"
+ buildtags="${buildtags} --tag tinode/exporter:latest --tag tinode/exporter:${ver[0]}.${ver[1]}"
+fi
+docker rmi ${rmitags}
+docker build --build-arg VERSION=$tag ${buildtags} docker/exporter
diff --git a/docker-release.sh b/docker-release.sh
index f706b17ab..09a0be3d7 100755
--- a/docker-release.sh
+++ b/docker-release.sh
@@ -63,6 +63,15 @@ fi
curl -u $user:$pass -i -X DELETE \
https://hub.docker.com/v2/repositories/tinode/chatbot/tags/${ver[0]}.${ver[1]}.${ver[2]}/
+if [ -n "$FULLRELEASE" ]; then
+ curl -u $user:$pass -i -X DELETE \
+ https://hub.docker.com/v2/repositories/tinode/exporter/tags/latest/
+ curl -u $user:$pass -i -X DELETE \
+ https://hub.docker.com/v2/repositories/tinode/exporter/tags/${ver[0]}.${ver[1]}/
+fi
+curl -u $user:$pass -i -X DELETE \
+ https://hub.docker.com/v2/repositories/tinode/exporter/tags/${ver[0]}.${ver[1]}.${ver[2]}/
+
# Deploy images for various DB backends
for dbtag in "${dbtags[@]}"
do
@@ -82,4 +91,11 @@ if [ -n "$FULLRELEASE" ]; then
fi
docker push tinode/chatbot:"${ver[0]}.${ver[1]}.${ver[2]}"
+# Deploy exporter images
+if [ -n "$FULLRELEASE" ]; then
+ docker push tinode/exporter:latest
+ docker push tinode/exporter:"${ver[0]}.${ver[1]}"
+fi
+docker push tinode/exporter:"${ver[0]}.${ver[1]}.${ver[2]}"
+
docker logout
From d5ddd5537864d3d2325a387032e502659970798d Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 19 Mar 2020 01:00:26 -0700
Subject: [PATCH 025/142] Fix exporter README path.
---
docker/README.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/docker/README.md b/docker/README.md
index 263e40d55..52a920bff 100644
--- a/docker/README.md
+++ b/docker/README.md
@@ -174,7 +174,7 @@ $ openssl rand -base64
## Metrics Exporter
-See [monitoring/exporter/README](../../monitoring/exporter/README.md) for information on the Exporter.
+See [monitoring/exporter/README](../monitoring/exporter/README.md) for information on the Exporter.
Container is also available as a part of the Tinode docker distribution: `tinode/exporter`.
Run it with
From 01b6a1554276caf0ff8251b20a2d0cb84efcf4b8 Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 19 Mar 2020 01:02:20 -0700
Subject: [PATCH 026/142] Formatting.
---
docker/README.md | 10 +++++-----
1 file changed, 5 insertions(+), 5 deletions(-)
diff --git a/docker/README.md b/docker/README.md
index 52a920bff..e7bd79bc7 100644
--- a/docker/README.md
+++ b/docker/README.md
@@ -181,21 +181,21 @@ Run it with
```
$ docker run -p 6222:6222 -d --name tinode-exporter --network tinode-net \
--env SERVE_FOR= \
- --env TINODE_ADDR= \
- --env LISTEN_AT=":6222" \
- ... \
+ --env TINODE_ADDR= \
+ --env LISTEN_AT=":6222" \
+ ... \
tinode/exporter:latest
```
Available variables:
| Variable | Type | Default | Function |
| --- | --- | --- | --- |
-| `SERVE_FOR` | string | `` | Either `prometheus` or `influxdb` |
+| `SERVE_FOR` | string | `` | Monitoring service: `prometheus` or `influxdb` |
| `TINODE_ADDR` | string | `http://localhost/stats/expvar/` | Tinode metrics path |
| `LISTEN_AT` | string | `:6222` | Exporter web server host and port |
| `INFLUXDB_VERSION` | string | `1.7` | InfluxDB version (`1.7` or `2.0`) |
| `INFLUXDB_ORGANIZATION` | string | `org` | InfluxDB organization |
-| `INFLUXDB_PUSH_INTERVAL` | int | `60` | Exporter's metrics push interval in seconds |
+| `INFLUXDB_PUSH_INTERVAL` | int | `60` | Exporter metrics push interval in seconds |
| `INFLUXDB_PUSH_ADDRESS` | string | `https://mon.tinode.co/intake` | InfluxDB backend url |
| `INFLUXDB_AUTH_TOKEN` | string | `Your-token` | InfluxDB auth token |
| `PROM_NAMESPACE` | string | `tinode` | Prometheus namespace |
From aa952f92af670880776fd0351da375d27c32b3cb Mon Sep 17 00:00:00 2001
From: or-else
Date: Thu, 19 Mar 2020 13:39:55 +0300
Subject: [PATCH 027/142] enforce sane minumum push interval + gofmt
---
monitoring/exporter/main.go | 27 ++++++++++++++++++---------
1 file changed, 18 insertions(+), 9 deletions(-)
diff --git a/monitoring/exporter/main.go b/monitoring/exporter/main.go
index c86d7e243..c4fe43833 100644
--- a/monitoring/exporter/main.go
+++ b/monitoring/exporter/main.go
@@ -20,6 +20,11 @@ const (
InfluxDB MonitoringService = 2
)
+const (
+ // Minimum interval between InfluxDB pushes in seconds.
+ minPushInterval = 10
+)
+
type promHTTPLogger struct{}
func (l promHTTPLogger) Println(v ...interface{}) {
@@ -40,16 +45,16 @@ func main() {
log.Printf("Tinode metrics exporter.")
var (
- serveFor = flag.String("serve_for", "influxdb", "Monitoring service to gather metrics for. Available: influxdb, prometheus.")
- tinodeAddr = flag.String("tinode_addr", "http://localhost:6060/stats/expvar", "Address of the Tinode instance to scrape.")
- listenAt = flag.String("listen_at", ":6222", "Host name and port to listen for incoming requests on.")
- metricList = flag.String("metric_list", "Version,LiveTopics,TotalTopics,LiveSessions,ClusterLeader,TotalClusterNodes,LiveClusterNodes,memstats.Alloc", "Comma-separated list of metrics to scrape and export.")
- instance = flag.String("instance", "exporter", "Exporter instance name.")
+ serveFor = flag.String("serve_for", "influxdb", "Monitoring service to gather metrics for. Available: influxdb, prometheus.")
+ tinodeAddr = flag.String("tinode_addr", "http://localhost:6060/stats/expvar", "Address of the Tinode instance to scrape.")
+ listenAt = flag.String("listen_at", ":6222", "Host name and port to listen for incoming requests on.")
+ metricList = flag.String("metric_list", "Version,LiveTopics,TotalTopics,LiveSessions,ClusterLeader,TotalClusterNodes,LiveClusterNodes,memstats.Alloc", "Comma-separated list of metrics to scrape and export.")
+ instance = flag.String("instance", "exporter", "Exporter instance name.")
// Prometheus-specific arguments.
- promNamespace = flag.String("prom_namespace", "tinode", "Prometheus namespace for metrics '_...'")
- promMetricsPath = flag.String("prom_metrics_path", "/metrics", "Path under which to expose metrics for Prometheus scrapes.")
- promTimeout = flag.Int("prom_timeout", 15, "Tinode connection timeout in seconds in response to Prometheus scrapes.")
+ promNamespace = flag.String("prom_namespace", "tinode", "Prometheus namespace for metrics '_...'")
+ promMetricsPath = flag.String("prom_metrics_path", "/metrics", "Path under which to expose metrics for Prometheus scrapes.")
+ promTimeout = flag.Int("prom_timeout", 15, "Tinode connection timeout in seconds in response to Prometheus scrapes.")
// InfluxDB-specific arguments.
influxPushAddr = flag.String("influx_push_addr", "http://localhost:9999/write", "Address of InfluxDB target server where the data gets sent.")
@@ -88,6 +93,10 @@ func main() {
if *influxDBVersion != "1.7" && *influxDBVersion != "2.0" {
log.Fatal("Please, set --influx_db_version to either 1.7 or 2.0")
}
+ if *influxPushInterval > 0 && *influxPushInterval < minPushInterval {
+ *influxPushInterval = minPushInterval
+ log.Println("The --influx_push_interval is too low, reset to", minPushInterval)
+ }
}
// Index page at web root.
@@ -103,7 +112,7 @@ func main() {
w.Write([]byte(`Tinode Exporter
Tinode Exporter
Server type` + *serveFor + `
` + servingPath +
-`Build
+ `Build
` + version.Info() + ` ` + version.BuildContext() + `
`))
})
From b8c5b2a29d8ddf4d8fb8efae5a9f499dae261e52 Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 19 Mar 2020 19:33:29 -0700
Subject: [PATCH 028/142] Address comments.
---
docker/README.md | 4 +---
docker/exporter/Dockerfile | 13 ++++++-------
docker/exporter/entrypoint.sh | 9 ++++++++-
3 files changed, 15 insertions(+), 11 deletions(-)
diff --git a/docker/README.md b/docker/README.md
index e7bd79bc7..8780923af 100644
--- a/docker/README.md
+++ b/docker/README.md
@@ -182,7 +182,6 @@ Run it with
$ docker run -p 6222:6222 -d --name tinode-exporter --network tinode-net \
--env SERVE_FOR= \
--env TINODE_ADDR= \
- --env LISTEN_AT=":6222" \
... \
tinode/exporter:latest
```
@@ -192,11 +191,10 @@ Available variables:
| --- | --- | --- | --- |
| `SERVE_FOR` | string | `` | Monitoring service: `prometheus` or `influxdb` |
| `TINODE_ADDR` | string | `http://localhost/stats/expvar/` | Tinode metrics path |
-| `LISTEN_AT` | string | `:6222` | Exporter web server host and port |
| `INFLUXDB_VERSION` | string | `1.7` | InfluxDB version (`1.7` or `2.0`) |
| `INFLUXDB_ORGANIZATION` | string | `org` | InfluxDB organization |
| `INFLUXDB_PUSH_INTERVAL` | int | `60` | Exporter metrics push interval in seconds |
| `INFLUXDB_PUSH_ADDRESS` | string | `https://mon.tinode.co/intake` | InfluxDB backend url |
-| `INFLUXDB_AUTH_TOKEN` | string | `Your-token` | InfluxDB auth token |
+| `INFLUXDB_AUTH_TOKEN` | string | `` | InfluxDB auth token |
| `PROM_NAMESPACE` | string | `tinode` | Prometheus namespace |
| `PROM_METRICS_PATH` | string | `/metrics` | Exporter webserver path that Prometheus server scrapes |
diff --git a/docker/exporter/Dockerfile b/docker/exporter/Dockerfile
index 6032f79ae..1602021a1 100644
--- a/docker/exporter/Dockerfile
+++ b/docker/exporter/Dockerfile
@@ -3,25 +3,24 @@ FROM alpine:latest
ARG VERSION=0.16.4
ENV VERSION=$VERSION
+LABEL maintainer="Tinode Team "
+LABEL name="TinodeMetricExporter"
+LABEL version=$VERSION
+
ENV SERVE_FOR=""
ENV TINODE_ADDR=http://localhost/stats/expvar/
ENV INSTANCE="exporter-instance"
-ENV LISTEN_AT=":6222"
ENV INFLUXDB_VERSION=1.7
ENV INFLUXDB_ORGANIZATION="org"
ENV INFLUXDB_PUSH_INTERVAL=60
-ENV INFLUXDB_PUSH_ADDRESS=http://localhost:6222/write
-ENV INFLUXDB_AUTH_TOKEN="Your-token"
+ENV INFLUXDB_PUSH_ADDRESS=""
+ENV INFLUXDB_AUTH_TOKEN=""
ENV PROM_NAMESPACE="tinode"
ENV PROM_METRICS_PATH="/metrics"
-LABEL maintainer="Tinode Team "
-LABEL name="TinodeMetricExporter"
-LABEL version=$VERSION
-
WORKDIR /opt/tinode
RUN apk add --no-cache bash
diff --git a/docker/exporter/entrypoint.sh b/docker/exporter/entrypoint.sh
index 1523edb12..6de9940ca 100755
--- a/docker/exporter/entrypoint.sh
+++ b/docker/exporter/entrypoint.sh
@@ -13,10 +13,17 @@ function check_vars() {
done
}
+# Make sure the system uses /etc/hosts when resolving domain names
+# (needed for docker-compose's `extra_hosts` param to work correctly).
+# See https://github.com/gliderlabs/docker-alpine/issues/367,
+# https://github.com/golang/go/issues/35305 for details.
echo "hosts: files dns" > /etc/nsswitch.conf
+# Accept http requests at.
+LISTEN_AT=":6222"
+
# Required env vars.
-common_vars=( TINODE_ADDR INSTANCE LISTEN_AT SERVE_FOR )
+common_vars=( TINODE_ADDR INSTANCE SERVE_FOR )
influx_varnames=( INFLUXDB_VERSION INFLUXDB_ORGANIZATION INFLUXDB_PUSH_INTERVAL \
INFLUXDB_PUSH_ADDRESS INFLUXDB_AUTH_TOKEN )
From aec45ed04d6e52131340cbe7e0f0d51fc96ddf8b Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 19 Mar 2020 20:50:33 -0700
Subject: [PATCH 029/142] Tinode docker consolidation.
---
docker/tinode/Dockerfile | 4 ++
docker/tinode/config.template | 2 +-
docker/tinode/entrypoint.sh | 71 +++++++++++++++++++++++++++++++++--
3 files changed, 73 insertions(+), 4 deletions(-)
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index f1150c27e..6e438a1b2 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -24,11 +24,15 @@ LABEL version=$VERSION
# Alternatively use
# `--build-arg TARGET_DB=mysql` to build for MySQL or
# `--build-arg TARGET_DB=mongodb` to build for MongoDB.
+# `--build-arg TARGET_DB=alldbs` to build a generic Tinode docker image.
ARG TARGET_DB=rethinkdb
ENV TARGET_DB=$TARGET_DB
# Runtime options.
+# Specifies what jobs to run: init-db, tinode or both.
+ENV SERVICES_TO_RUN='both'
+
# An option to reset database.
ENV RESET_DB=false
diff --git a/docker/tinode/config.template b/docker/tinode/config.template
index 872b7d20b..9c13c681a 100644
--- a/docker/tinode/config.template
+++ b/docker/tinode/config.template
@@ -8,7 +8,7 @@
"max_message_size": 4194304,
"max_subscriber_count": 32,
"max_tag_count": 16,
- "expvar": "",
+ "expvar": "/stats/expvar/",
"media": {
"use_handler": "$MEDIA_HANDLER",
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index c6b244bf0..e784cc2b7 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -71,8 +71,56 @@ else
echo "" > $STATIC_DIR/firebase-init.js
fi
-# Initialize the database if it has not been initialized yet or if data reset/upgrade has been requested.
-./init-db --reset=${RESET_DB} --upgrade=${UPGRADE_DB} --config=${CONFIG} --data=${SAMPLE_DATA} | grep "usr;tino;" > /botdata/tino-password
+if [ ! -z "$IOS_UNIV_LINKS_APP_ID" ] ; then
+ # Write config to $STATIC_DIR/apple-app-site-association config file.
+ # See https://developer.apple.com/library/archive/documentation/General/Conceptual/AppSearch/UniversalLinks.html for details.
+ cat > $STATIC_DIR/apple-app-site-association <<- EOM
+{
+ "applinks": {
+ "apps": [],
+ "details": [
+ {
+ "appID": "$IOS_UNIV_LINKS_APP_ID",
+ "paths": [ "*" ]
+ }
+ ]
+ }
+}
+EOM
+fi
+
+run_init_db=false
+run_tinode=false
+case "$SERVICES_TO_RUN" in
+"init-db")
+ run_init_db=true
+ ;;
+"tinode")
+ run_tinode=true
+ ;;
+"both")
+ run_init_db=true
+ run_tinode=true
+ ;;
+*)
+ echo "Invalid val for SERVICES_TO_RUN env var. Can be either 'init-db' or 'tinode' or 'both'."
+ exit 1
+ ;;
+esac
+
+echo "Will run init-db: ${run_init_db}, tinode: ${run_tinode}"
+
+touch /botdata/tino-password
+
+if [ "$run_init_db" == "true" ]; then
+ # Initialize the database if it has not been initialized yet or if data reset/upgrade has been requested.
+ ./init-db --reset=${RESET_DB} --upgrade=${UPGRADE_DB} --config=${CONFIG} | grep "usr;tino;" > /botdata/tino-password
+fi
+
+if [ "$run_tinode" != "true" ]; then
+ # If we don't want to run tinode, we are done.
+ exit 0
+fi
if [ -s /botdata/tino-password ] ; then
# Convert Tino's authentication credentials into a cookie file.
@@ -89,5 +137,22 @@ args=("--config=${CONFIG}" "--static_data=$STATIC_DIR")
if [ ! -z "$CLUSTER_SELF" ] ; then
args+=("--cluster_self=$CLUSTER_SELF")
fi
+if [ ! -z "$PPROF_URL" ] ; then
+ args+=("--pprof_url=$PPROF_URL")
+fi
+
+# Create the log directory (/var/log/tinode-`current timestamp`).
+# And symlink /var/log/tinode-latest to it.
+runid=tinode-`date +%s`
+logdir=/var/log/$runid
+mkdir -p $logdir
+if [ -d /var/log/tinode-latest ]; then
+ rm /var/log/tinode-latest
+fi
+pushd .
+cd /var/log
+ln -s $runid tinode-latest
+popd
+
# Run the tinode server.
-./tinode "${args[@]}" 2> /var/log/tinode.log
+./tinode "${args[@]}" 2> $logdir/tinode.log
From 1c7b04275686ad555bc523b8e933c33229caf149 Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 19 Mar 2020 23:36:35 -0700
Subject: [PATCH 030/142] Example e2e docker-compose setup.
---
docker/e2e/README.md | 1 +
docker/e2e/cluster-docker-compose.yml | 207 +++++++++++++++++++++++
docker/e2e/standalone-docker-compose.yml | 136 +++++++++++++++
3 files changed, 344 insertions(+)
create mode 100644 docker/e2e/README.md
create mode 100644 docker/e2e/cluster-docker-compose.yml
create mode 100644 docker/e2e/standalone-docker-compose.yml
diff --git a/docker/e2e/README.md b/docker/e2e/README.md
new file mode 100644
index 000000000..fa919e9c9
--- /dev/null
+++ b/docker/e2e/README.md
@@ -0,0 +1 @@
+Docker compose for E2E setup.
diff --git a/docker/e2e/cluster-docker-compose.yml b/docker/e2e/cluster-docker-compose.yml
new file mode 100644
index 000000000..db1a1cc4b
--- /dev/null
+++ b/docker/e2e/cluster-docker-compose.yml
@@ -0,0 +1,207 @@
+# Reference configuration for a simple 3-node Tinode cluster.
+# Includes:
+# * Mysql database
+# * 3 Tinode servers
+# * 3 exporters
+
+version: '3.4'
+
+# Base Tinode template.
+x-tinode:
+ &tinode-base
+ depends_on:
+ - mysql
+ image: tinode/tinode:latest
+ restart: always
+
+x-exporter:
+ &exporter-base
+ image: tinode/exporter:latest
+ restart: always
+ networks:
+ internal:
+
+x-tinode-env-vars: &tinode-env-vars
+ "STORE_USE_ADAPTER": "mysql"
+ "DEFAULT_SAMPLE_DATA": ""
+ "FCM_PUSH_ENABLED": "false"
+ # "FCM_API_KEY": ""
+ # "FCM_APP_ID": ""
+ # "FCM_PROJECT_ID": ""
+ # "FCM_SENDER_ID":
+ # "FCM_VAPID_KEY": ""
+ # "IOS_UNIV_LINKS_APP_ID": ""
+ "PPROF_URL": "/pprof"
+ # Run tinode server only.
+ "SERVICES_TO_RUN": "tinode"
+ # You can provide your own tinode config by setting EXT_CONFIG env var and binding your configuration file to
+ # "EXT_CONFIG": "/etc/tinode/tinode.conf"
+
+x-exporter-env-vars: &exporter-env-vars
+ "TINODE_ADDR": "http://tinode.host:18080/stats/expvar/"
+ # Prometheus configuration:
+ "SERVE_FOR": "prometheus"
+ "PROM_NAMESPACE": "tinode"
+ "PROM_METRICS_PATH": "/metrics"
+ # InfluxDB configation:
+ # "SERVE_FOR": "influxdb"
+ # "INFLUXDB_VERSION": 1.7
+ # "INFLUXDB_ORGANIZATION": ""
+ # "INFLUXDB_PUSH_INTERVAL": 30
+ # "INFLUXDB_PUSH_ADDRESS": "https://mon.tinode.co/intake"
+ # "INFLUXDB_AUTH_TOKEN": "abcdef"
+
+services:
+ mysql:
+ image: mysql:5.7
+ container_name: mysql
+ restart: always
+ networks:
+ internal:
+ ipv4_address: 172.19.0.3
+ # Use your own volume.
+ # volumes:
+ # - :/var/lib/mysql
+ environment:
+ - MYSQL_ALLOW_EMPTY_PASSWORD=yes
+ healthcheck:
+ test: ["CMD", "mysqladmin" ,"ping", "-h", "localhost"]
+ timeout: 5s
+ retries: 10
+
+ # Run this only when you need to initialize Tinode database.
+ init-db:
+ << : *tinode-base
+ container_name: init-db
+ hostname: init-db
+ restart: "no"
+ networks:
+ internal:
+ ipv4_address: 172.19.0.2
+ environment:
+ << : *tinode-env-vars
+ "SERVICES_TO_RUN": "init-db"
+ "UPGRADE_DB": "false"
+
+ # Run this only when you need to upgrade Tinode database to a newer version.
+ upgrade-db:
+ << : *tinode-base
+ container_name: upgrade-db
+ hostname: upgrade-db
+ restart: "no"
+ networks:
+ internal:
+ ipv4_address: 172.19.0.2
+ environment:
+ << : *tinode-env-vars
+ "SERVICES_TO_RUN": "init-db"
+ "UPGRADE_DB": "true"
+
+ # Tinode servers.
+ tinode-0:
+ << : *tinode-base
+ container_name: tinode-0
+ hostname: tinode-0
+ networks:
+ internal:
+ ipv4_address: 172.19.0.5
+ # You can mount your volumes as necessary:
+ # volumes:
+ # # E.g. external config.
+ # - :/etc/tinode/tinode.conf
+ # # Logs directory.
+ # - :/var/log
+ ports:
+ - "6060:18080"
+ environment:
+ << : *tinode-env-vars
+ "CLUSTER_SELF": "tinode-0"
+
+ tinode-1:
+ << : *tinode-base
+ container_name: tinode-1
+ hostname: tinode-1
+ networks:
+ internal:
+ ipv4_address: 172.19.0.6
+ # You can mount your volumes as necessary:
+ # volumes:
+ # # E.g. external config.
+ # - :/etc/tinode/tinode.conf
+ # # Logs directory.
+ # - :/var/log
+ ports:
+ - "6061:18080"
+ environment:
+ << : *tinode-env-vars
+ "CLUSTER_SELF": "tinode-1"
+
+ tinode-2:
+ << : *tinode-base
+ container_name: tinode-2
+ hostname: tinode-2
+ networks:
+ internal:
+ ipv4_address: 172.19.0.7
+ # You can mount your volumes as necessary:
+ # volumes:
+ # # E.g. external config.
+ # - :/etc/tinode/tinode.conf
+ # # Logs directory.
+ # - :/var/log
+ ports:
+ - "6062:18080"
+ environment:
+ << : *tinode-env-vars
+ "CLUSTER_SELF": "tinode-2"
+
+ # Monitoring.
+ # Exporters are paired with tinode instances.
+ exporter-0:
+ << : *exporter-base
+ container_name: exporter-0
+ hostname: exporter-0
+ depends_on:
+ - tinode-0
+ ports:
+ - "6222:6222"
+ environment:
+ << : *exporter-env-vars
+ "INSTANCE": "tinode-exp-0"
+ extra_hosts:
+ - "tinode.host:172.19.0.5"
+
+ exporter-1:
+ << : *exporter-base
+ container_name: exporter-1
+ hostname: exporter-1
+ depends_on:
+ - tinode-1
+ ports:
+ - "6223:6222"
+ environment:
+ << : *exporter-env-vars
+ "INSTANCE": "tinode-exp-1"
+ extra_hosts:
+ - "tinode.host:172.19.0.6"
+
+ exporter-2:
+ << : *exporter-base
+ container_name: exporter-2
+ hostname: exporter-2
+ depends_on:
+ - tinode-2
+ ports:
+ - "6224:6222"
+ environment:
+ << : *exporter-env-vars
+ "INSTANCE": "tinode-exp-2"
+ extra_hosts:
+ - "tinode.host:172.19.0.7"
+
+networks:
+ internal:
+ ipam:
+ driver: default
+ config:
+ - subnet: "172.19.0.0/24"
diff --git a/docker/e2e/standalone-docker-compose.yml b/docker/e2e/standalone-docker-compose.yml
new file mode 100644
index 000000000..c4b2e264a
--- /dev/null
+++ b/docker/e2e/standalone-docker-compose.yml
@@ -0,0 +1,136 @@
+# Reference configuration for a simple Tinode server.
+# Includes:
+# * Mysql database
+# * Tinode server
+# * Tinode exporters
+
+version: '3.4'
+
+# Base Tinode template.
+x-tinode:
+ &tinode-base
+ depends_on:
+ - mysql
+ image: tinode/tinode:latest
+ restart: always
+
+x-tinode-env-vars: &tinode-env-vars
+ "STORE_USE_ADAPTER": "mysql"
+ "DEFAULT_SAMPLE_DATA": ""
+ "FCM_PUSH_ENABLED": "false"
+ # "FCM_API_KEY": ""
+ # "FCM_APP_ID": ""
+ # "FCM_PROJECT_ID": ""
+ # "FCM_SENDER_ID":
+ # "FCM_VAPID_KEY": ""
+ # "IOS_UNIV_LINKS_APP_ID": ""
+ "PPROF_URL": "/pprof"
+ # Run tinode server only.
+ "SERVICES_TO_RUN": "tinode"
+ # You can provide your own tinode config by setting EXT_CONFIG env var and binding your configuration file to
+ # "EXT_CONFIG": "/etc/tinode/tinode.conf"
+
+x-exporter-env-vars: &exporter-env-vars
+ "TINODE_ADDR": "http://tinode.host:18080/stats/expvar/"
+ # Prometheus configuration:
+ "SERVE_FOR": "prometheus"
+ "PROM_NAMESPACE": "tinode"
+ "PROM_METRICS_PATH": "/metrics"
+ # InfluxDB configation:
+ # "SERVE_FOR": "influxdb"
+ # "INFLUXDB_VERSION": 1.7
+ # "INFLUXDB_ORGANIZATION": ""
+ # "INFLUXDB_PUSH_INTERVAL": 30
+ # "INFLUXDB_PUSH_ADDRESS": "https://mon.tinode.co/intake"
+ # "INFLUXDB_AUTH_TOKEN": "abcdef"
+
+services:
+ mysql:
+ image: mysql:5.7
+ container_name: mysql
+ restart: always
+ networks:
+ internal:
+ ipv4_address: 172.19.0.3
+ # Use your own volume.
+ # volumes:
+ # - :/var/lib/mysql
+ environment:
+ - MYSQL_ALLOW_EMPTY_PASSWORD=yes
+ healthcheck:
+ test: ["CMD", "mysqladmin" ,"ping", "-h", "localhost"]
+ timeout: 5s
+ retries: 10
+
+ # Run this only when you need to initialize Tinode database.
+ init-db:
+ << : *tinode-base
+ container_name: init-db
+ hostname: init-db
+ restart: "no"
+ networks:
+ internal:
+ ipv4_address: 172.19.0.2
+ environment:
+ << : *tinode-env-vars
+ "SERVICES_TO_RUN": "init-db"
+ "UPGRADE_DB": "false"
+
+ # Run this only when you need to upgrade Tinode database to a newer version.
+ upgrade-db:
+ << : *tinode-base
+ container_name: upgrade-db
+ hostname: upgrade-db
+ restart: "no"
+ networks:
+ internal:
+ ipv4_address: 172.19.0.2
+ environment:
+ << : *tinode-env-vars
+ "SERVICES_TO_RUN": "init-db"
+ "UPGRADE_DB": "true"
+
+ tinode-0:
+ << : *tinode-base
+ container_name: tinode-0
+ hostname: tinode-0
+ networks:
+ internal:
+ ipv4_address: 172.19.0.5
+ # You can mount your volumes as necessary:
+ # volumes:
+ # # E.g. external config.
+ # - :/etc/tinode/tinode.conf
+ # # Logs directory.
+ # - :/var/log
+ ports:
+ - "6060:18080"
+ environment:
+ << : *tinode-env-vars
+
+ # Monitoring.
+ # Exporters are paired with tinode instances.
+ exporter-0:
+ container_name: exporter-0
+ hostname: exporter-0
+ depends_on:
+ - tinode-0
+ image: tinode/exporter:latest
+ restart: always
+ networks:
+ internal:
+ ports:
+ - "6222:6222"
+ environment:
+ << : *exporter-env-vars
+ # Remove?
+ "INSTANCE": "tinode-exp-0"
+ extra_hosts:
+ - "tinode.host:172.19.0.5"
+
+networks:
+ internal:
+ ipam:
+ driver: default
+ config:
+ - subnet: "172.19.0.0/24"
From aa388e55c6dd379ce786b63b3f71ceaa3c371cbe Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 19 Mar 2020 23:46:48 -0700
Subject: [PATCH 031/142] Load sample data and cluster config in tinode.conf
template.
---
docker/tinode/config.template | 16 ++++++++++++++++
docker/tinode/entrypoint.sh | 7 ++++++-
2 files changed, 22 insertions(+), 1 deletion(-)
diff --git a/docker/tinode/config.template b/docker/tinode/config.template
index 9c13c681a..4e47f581c 100644
--- a/docker/tinode/config.template
+++ b/docker/tinode/config.template
@@ -133,6 +133,22 @@
}
],
+ "cluster_config": {
+ "self": "",
+ "nodes": [
+ // Name and TCP address of each node.
+ {"name": "tinode-0", "addr": "tinode-0:12001"},
+ {"name": "tinode-1", "addr": "tinode-1:12002"},
+ {"name": "tinode-2", "addr": "tinode-2:12003"}
+ ],
+ "failover": {
+ "enabled": true,
+ "heartbeat": 100,
+ "vote_after": 8,
+ "node_fail_after": 16
+ }
+ },
+
"plugins": [
{
"enabled": $PLUGIN_PYTHON_CHAT_BOT_ENABLED,
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index e784cc2b7..1c4cf2741 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -113,8 +113,13 @@ echo "Will run init-db: ${run_init_db}, tinode: ${run_tinode}"
touch /botdata/tino-password
if [ "$run_init_db" == "true" ]; then
+ init_args=("--reset=${RESET_DB}" "--upgrade=${UPGRADE_DB}" "--config=${CONFIG}")
+ # Maybe load sample data?
+ if [ ! -z "$SAMPLE_DATA" ] ; then
+ init_args+=("--data=$SAMPLE_DATA")
+ fi
# Initialize the database if it has not been initialized yet or if data reset/upgrade has been requested.
- ./init-db --reset=${RESET_DB} --upgrade=${UPGRADE_DB} --config=${CONFIG} | grep "usr;tino;" > /botdata/tino-password
+ ./init-db "${init_args[@]}" | grep "usr;tino;" > /botdata/tino-password
fi
if [ "$run_tinode" != "true" ]; then
From a6ca8106c2f42e107085409fea9494968d545137 Mon Sep 17 00:00:00 2001
From: aforge
Date: Fri, 20 Mar 2020 00:47:01 -0700
Subject: [PATCH 032/142] Load sample data in standalone.
---
docker/e2e/standalone-docker-compose.yml | 2 ++
1 file changed, 2 insertions(+)
diff --git a/docker/e2e/standalone-docker-compose.yml b/docker/e2e/standalone-docker-compose.yml
index c4b2e264a..ccd15121a 100644
--- a/docker/e2e/standalone-docker-compose.yml
+++ b/docker/e2e/standalone-docker-compose.yml
@@ -74,6 +74,8 @@ services:
environment:
<< : *tinode-env-vars
"SERVICES_TO_RUN": "init-db"
+ # Comment out this line if you don't want sample data loaded.
+ "SAMPLE_DATA": "data.json"
"UPGRADE_DB": "false"
# Run this only when you need to upgrade Tinode database to a newer version.
From 75ecc3b2b9f213ab6d051698c245170e9da8f944 Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 20 Mar 2020 20:32:10 +0300
Subject: [PATCH 033/142] another attempt at fixing memory leak
---
server/cluster.go | 5 +++++
1 file changed, 5 insertions(+)
diff --git a/server/cluster.go b/server/cluster.go
index 056d92899..e9ed017a8 100644
--- a/server/cluster.go
+++ b/server/cluster.go
@@ -22,6 +22,8 @@ const (
defaultClusterReconnect = 200 * time.Millisecond
// Number of replicas in ringhash
clusterHashReplicas = 20
+ // Period for running health check on cluster session: terminate sessions with no subscriptions.
+ clusterSessionCleanup = 5 * time.Second
)
type clusterNodeConfig struct {
@@ -744,6 +746,8 @@ func (sess *Session) rpcWriteLoop() {
sess.unsubAll()
}()
+ heartBeat := time.NewTimer(clusterSessionCleanup)
+
for {
select {
case msg, ok := <-sess.send:
@@ -768,6 +772,7 @@ func (sess *Session) rpcWriteLoop() {
case topic := <-sess.detach:
sess.delSub(topic)
+ case <-heartBeat.C:
// All proxied subsriptions are gone, this session is no longer needed.
if sess.countSub() == 0 {
return
From c28f28feb636351545e672cad2ae6e096e3268ec Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 21 Mar 2020 13:01:24 +0300
Subject: [PATCH 034/142] document some incorrect code
---
server/cluster.go | 9 +++++++++
server/session.go | 3 +++
2 files changed, 12 insertions(+)
diff --git a/server/cluster.go b/server/cluster.go
index e9ed017a8..b167f6285 100644
--- a/server/cluster.go
+++ b/server/cluster.go
@@ -746,6 +746,7 @@ func (sess *Session) rpcWriteLoop() {
sess.unsubAll()
}()
+ // Timer which checks for orphaned nodes.
heartBeat := time.NewTimer(clusterSessionCleanup)
for {
@@ -883,6 +884,14 @@ func (c *Cluster) invalidateRemoteSubs() {
}
}
s.remoteSubsLock.Unlock()
+ // FIXME:
+ // This is problematic for two reasons:
+ // 1. We don't really know if subscription contained in s.remoteSubs actually exists.
+ // We only know that {sub} packet was sent to the remote node and it was delivered.
+ // 2. The {pres what=term} is sent on 'me' topic but we don't know if the session is
+ // subscribed to 'me' topic. The correct way of doing it is to send to those online
+ // in the topic on topic itsef, to those offline on their 'me' topic. In general
+ // the 'presTermDirect' should not exist.
s.presTermDirect(topicsToTerminate)
}
}
diff --git a/server/session.go b/server/session.go
index b73580943..05bd27171 100644
--- a/server/session.go
+++ b/server/session.go
@@ -117,6 +117,7 @@ type Session struct {
subsLock sync.RWMutex
// Map of remote topic subscriptions, indexed by topic name.
+ // It does not contain actual subscriptions but rather "maybe subscriptions".
remoteSubs map[string]*RemoteSubscription
// Synchronizes access to remoteSubs.
remoteSubsLock sync.RWMutex
@@ -432,6 +433,7 @@ func (s *Session) subscribe(msg *ClientComMessage) {
} else {
originalTopic = msg.topic
}
+ // FIXME: we don't really know if subscription was successful.
s.addRemoteSub(expanded, &RemoteSubscription{node: remoteNodeName, originalTopic: originalTopic})
}
} else {
@@ -473,6 +475,7 @@ func (s *Session) leave(msg *ClientComMessage) {
log.Println("s.leave:", err, s.sid)
s.queueOut(ErrClusterUnreachable(msg.id, msg.topic, msg.timestamp))
} else {
+ // FIXME: we don't really know if leave succeeded.
s.delRemoteSub(expanded)
}
} else if !msg.Leave.Unsub {
From f92d1f5fa45f0e08ba8f6e0d68fc368ff0fa3708 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 21 Mar 2020 19:10:09 +0300
Subject: [PATCH 035/142] using go mod for package management
---
go.mod | 28 ++++++
go.sum | 302 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++
2 files changed, 330 insertions(+)
create mode 100644 go.mod
create mode 100644 go.sum
diff --git a/go.mod b/go.mod
new file mode 100644
index 000000000..9c8b84490
--- /dev/null
+++ b/go.mod
@@ -0,0 +1,28 @@
+module github.com/tinode/chat
+
+go 1.14
+
+require (
+ firebase.google.com/go v3.12.0+incompatible
+ github.com/DisposaBoy/JsonConfigReader v0.0.0-20171218180944-5ea4d0ddac55
+ github.com/aws/aws-sdk-go v1.29.29
+ github.com/bitly/go-hostpool v0.1.0 // indirect
+ github.com/bmizerany/assert v0.0.0-20160611221934-b7ed37b82869 // indirect
+ github.com/go-sql-driver/mysql v1.5.0
+ github.com/golang/protobuf v1.3.5
+ github.com/google/go-cmp v0.4.0
+ github.com/gorilla/handlers v1.4.2
+ github.com/gorilla/websocket v1.4.2
+ github.com/jmoiron/sqlx v1.2.0
+ github.com/prometheus/client_golang v1.5.1
+ github.com/prometheus/common v0.9.1
+ github.com/tinode/snowflake v1.0.0
+ go.mongodb.org/mongo-driver v1.3.1
+ golang.org/x/crypto v0.0.0-20200320181102-891825fb96df
+ golang.org/x/net v0.0.0-20200320220750-118fecf932d8
+ golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d
+ golang.org/x/text v0.3.2
+ google.golang.org/api v0.20.0
+ google.golang.org/grpc v1.28.0
+ gopkg.in/rethinkdb/rethinkdb-go.v5 v5.1.0
+)
diff --git a/go.sum b/go.sum
new file mode 100644
index 000000000..c012b5340
--- /dev/null
+++ b/go.sum
@@ -0,0 +1,302 @@
+cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
+cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
+cloud.google.com/go v0.38.0 h1:ROfEUZz+Gh5pa62DJWXSaonyu3StP6EA6lPEXPI6mCo=
+cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU=
+firebase.google.com/go v3.12.0+incompatible h1:q70KCp/J0oOL8kJ8oV2j3646kV4TB8Y5IvxXC0WT1bo=
+firebase.google.com/go v3.12.0+incompatible/go.mod h1:xlah6XbEyW6tbfSklcfe5FHJIwjt8toICdV5Wh9ptHs=
+github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
+github.com/DisposaBoy/JsonConfigReader v0.0.0-20171218180944-5ea4d0ddac55 h1:jbGlDKdzAZ92NzK65hUP98ri0/r50vVVvmZsFP/nIqo=
+github.com/DisposaBoy/JsonConfigReader v0.0.0-20171218180944-5ea4d0ddac55/go.mod h1:GCzqZQHydohgVLSIqRKZeTt8IGb1Y4NaFfim3H40uUI=
+github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
+github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
+github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
+github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
+github.com/aws/aws-sdk-go v1.29.29 h1:4TdSYzXL8bHKu80tzPjO4c0ALw4Fd8qZGqf1aozUcBU=
+github.com/aws/aws-sdk-go v1.29.29/go.mod h1:1KvfttTE3SPKMpo8g2c6jL3ZKfXtFvKscTgahTma5Xg=
+github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
+github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8=
+github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM=
+github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw=
+github.com/bitly/go-hostpool v0.1.0 h1:XKmsF6k5el6xHG3WPJ8U0Ku/ye7njX7W81Ng7O2ioR0=
+github.com/bitly/go-hostpool v0.1.0/go.mod h1:4gOCgp6+NZnVqlKyZ/iBZFTAJKembaVENUpMkpg42fw=
+github.com/bmizerany/assert v0.0.0-20160611221934-b7ed37b82869 h1:DDGfHa7BWjL4YnC6+E63dPcxHo2sUxDIu8g3QgEJdRY=
+github.com/bmizerany/assert v0.0.0-20160611221934-b7ed37b82869/go.mod h1:Ekp36dRnpXw/yCqJaO+ZrUyxD+3VXMFFr56k5XYrpB4=
+github.com/cenkalti/backoff v2.0.0+incompatible h1:5IIPUHhlnUZbcHQsQou5k1Tn58nJkeJL9U+ig5CHJbY=
+github.com/cenkalti/backoff v2.0.0+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM=
+github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
+github.com/cespare/xxhash/v2 v2.1.1 h1:6MnRN8NT7+YBpUIWxHtefFZOKTAPgGjpQSxqLNn0+qY=
+github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
+github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
+github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc=
+github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
+github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
+github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
+github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
+github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
+github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98=
+github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
+github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
+github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
+github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
+github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
+github.com/go-sql-driver/mysql v1.4.0/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w=
+github.com/go-sql-driver/mysql v1.5.0 h1:ozyZYNQW3x3HtqT1jira07DN2PArx2v7/mN66gGcHOs=
+github.com/go-sql-driver/mysql v1.5.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg=
+github.com/go-stack/stack v1.8.0 h1:5SgMzNM5HxrEjV0ww2lTmX6E2Izsfxas4+YHWRs3Lsk=
+github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
+github.com/gobuffalo/attrs v0.0.0-20190224210810-a9411de4debd/go.mod h1:4duuawTqi2wkkpB4ePgWMaai6/Kc6WEz83bhFwpHzj0=
+github.com/gobuffalo/depgen v0.0.0-20190329151759-d478694a28d3/go.mod h1:3STtPUQYuzV0gBVOY3vy6CfMm/ljR4pABfrTeHNLHUY=
+github.com/gobuffalo/depgen v0.1.0/go.mod h1:+ifsuy7fhi15RWncXQQKjWS9JPkdah5sZvtHc2RXGlg=
+github.com/gobuffalo/envy v1.6.15/go.mod h1:n7DRkBerg/aorDM8kbduw5dN3oXGswK5liaSCx4T5NI=
+github.com/gobuffalo/envy v1.7.0/go.mod h1:n7DRkBerg/aorDM8kbduw5dN3oXGswK5liaSCx4T5NI=
+github.com/gobuffalo/flect v0.1.0/go.mod h1:d2ehjJqGOH/Kjqcoz+F7jHTBbmDb38yXA598Hb50EGs=
+github.com/gobuffalo/flect v0.1.1/go.mod h1:8JCgGVbRjJhVgD6399mQr4fx5rRfGKVzFjbj6RE/9UI=
+github.com/gobuffalo/flect v0.1.3/go.mod h1:8JCgGVbRjJhVgD6399mQr4fx5rRfGKVzFjbj6RE/9UI=
+github.com/gobuffalo/genny v0.0.0-20190329151137-27723ad26ef9/go.mod h1:rWs4Z12d1Zbf19rlsn0nurr75KqhYp52EAGGxTbBhNk=
+github.com/gobuffalo/genny v0.0.0-20190403191548-3ca520ef0d9e/go.mod h1:80lIj3kVJWwOrXWWMRzzdhW3DsrdjILVil/SFKBzF28=
+github.com/gobuffalo/genny v0.1.0/go.mod h1:XidbUqzak3lHdS//TPu2OgiFB+51Ur5f7CSnXZ/JDvo=
+github.com/gobuffalo/genny v0.1.1/go.mod h1:5TExbEyY48pfunL4QSXxlDOmdsD44RRq4mVZ0Ex28Xk=
+github.com/gobuffalo/gitgen v0.0.0-20190315122116-cc086187d211/go.mod h1:vEHJk/E9DmhejeLeNt7UVvlSGv3ziL+djtTr3yyzcOw=
+github.com/gobuffalo/gogen v0.0.0-20190315121717-8f38393713f5/go.mod h1:V9QVDIxsgKNZs6L2IYiGR8datgMhB577vzTDqypH360=
+github.com/gobuffalo/gogen v0.1.0/go.mod h1:8NTelM5qd8RZ15VjQTFkAW6qOMx5wBbW4dSCS3BY8gg=
+github.com/gobuffalo/gogen v0.1.1/go.mod h1:y8iBtmHmGc4qa3urIyo1shvOD8JftTtfcKi+71xfDNE=
+github.com/gobuffalo/logger v0.0.0-20190315122211-86e12af44bc2/go.mod h1:QdxcLw541hSGtBnhUc4gaNIXRjiDppFGaDqzbrBd3v8=
+github.com/gobuffalo/mapi v1.0.1/go.mod h1:4VAGh89y6rVOvm5A8fKFxYG+wIW6LO1FMTG9hnKStFc=
+github.com/gobuffalo/mapi v1.0.2/go.mod h1:4VAGh89y6rVOvm5A8fKFxYG+wIW6LO1FMTG9hnKStFc=
+github.com/gobuffalo/packd v0.0.0-20190315124812-a385830c7fc0/go.mod h1:M2Juc+hhDXf/PnmBANFCqx4DM3wRbgDvnVWeG2RIxq4=
+github.com/gobuffalo/packd v0.1.0/go.mod h1:M2Juc+hhDXf/PnmBANFCqx4DM3wRbgDvnVWeG2RIxq4=
+github.com/gobuffalo/packr/v2 v2.0.9/go.mod h1:emmyGweYTm6Kdper+iywB6YK5YzuKchGtJQZ0Odn4pQ=
+github.com/gobuffalo/packr/v2 v2.2.0/go.mod h1:CaAwI0GPIAv+5wKLtv8Afwl+Cm78K/I/VCm/3ptBN+0=
+github.com/gobuffalo/syncx v0.0.0-20190224160051-33c29581e754/go.mod h1:HhnNqWY95UYwwW3uSASeV7vtgYkT2t16hJgV3AEPUpw=
+github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ=
+github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
+github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
+github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
+github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
+github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
+github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
+github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
+github.com/golang/protobuf v1.3.5 h1:F768QJ1E9tib+q5Sc8MkdJi1RxLTbRcTf8LJV56aRls=
+github.com/golang/protobuf v1.3.5/go.mod h1:6O5/vntMXwX2lRkT1hjjk0nAC1IDOTvTlVgjlRvqsdk=
+github.com/golang/snappy v0.0.1 h1:Qgr9rKW7uDUkrbSmQeiDsGa8SjGyCOGtuasMWwvp2P4=
+github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
+github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
+github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
+github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
+github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
+github.com/google/go-cmp v0.4.0 h1:xsAVV57WRhGj6kEIi8ReJzQlHHqcBYCElAvkovg3B/4=
+github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
+github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
+github.com/google/martian v2.1.0+incompatible h1:/CP5g8u/VJHijgedC/Legn3BAbAaWPgecwXBIDzw5no=
+github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs=
+github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
+github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
+github.com/googleapis/gax-go/v2 v2.0.5 h1:sjZBwGj9Jlw33ImPtvFviGYvseOtDM7hkSKB7+Tv3SM=
+github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk=
+github.com/gorilla/handlers v1.4.2 h1:0QniY0USkHQ1RGCLfKxeNHK9bkDHGRYGNDFBCS+YARg=
+github.com/gorilla/handlers v1.4.2/go.mod h1:Qkdc/uu4tH4g6mTK6auzZ766c4CA0Ng8+o/OAirnOIQ=
+github.com/gorilla/websocket v1.4.2 h1:+/TMaTYc4QFitKJxsQ7Yye35DkWvkdLcvGKqM+x0Ufc=
+github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
+github.com/hailocab/go-hostpool v0.0.0-20160125115350-e80d13ce29ed h1:5upAirOpQc1Q53c0bnx2ufif5kANL7bfZWcc6VJWJd8=
+github.com/hailocab/go-hostpool v0.0.0-20160125115350-e80d13ce29ed/go.mod h1:tMWxXQ9wFIaZeTI9F+hmhFiGpFmhOHzyShyFUhRm0H4=
+github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
+github.com/hashicorp/golang-lru v0.5.1 h1:0hERBMJE1eitiLkihrMvRVBYAkpHzc/J3QdDN+dAcgU=
+github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
+github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
+github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af h1:pmfjZENx5imkbgOkpRUYLnmbU7UEFbjtDA2hxJ1ichM=
+github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af/go.mod h1:Nht3zPeWKUH0NzdCt2Blrr5ys8VGpn0CEB0cQHVjt7k=
+github.com/jmoiron/sqlx v1.2.0 h1:41Ip0zITnmWNR/vHV+S4m+VoUivnWY5E4OJfLZjCJMA=
+github.com/jmoiron/sqlx v1.2.0/go.mod h1:1FEQNm3xlJgrMD+FBdI9+xvCksHtbpVBBw5dYhBSsks=
+github.com/joho/godotenv v1.3.0/go.mod h1:7hK45KPybAkOC6peb+G5yklZfMxEjkZhHbwpqxOKXbg=
+github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
+github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
+github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU=
+github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w=
+github.com/karrick/godirwalk v1.8.0/go.mod h1:H5KPZjojv4lE+QYImBI8xVtrBRgYrIVsaRPx4tDPEn4=
+github.com/karrick/godirwalk v1.10.3/go.mod h1:RoGL9dQei4vP9ilrpETWE8CLOZ1kiN0LhBygSwrAsHA=
+github.com/kisielk/errcheck v1.2.0/go.mod h1:/BMXB+zMLi60iA8Vv6Ksmxu/1UDYcXs4uQLJ+jE2L00=
+github.com/klauspost/compress v1.9.5 h1:U+CaK85mrNNb4k8BNOfgJtJ/gr6kswUCFj6miSzVC6M=
+github.com/klauspost/compress v1.9.5/go.mod h1:RyIbtBH6LamlWaDj8nUwkbUhJ87Yi3uG0guNDohfE1A=
+github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
+github.com/konsorten/go-windows-terminal-sequences v1.0.2 h1:DB17ag19krx9CFsz4o3enTrPXyIXCl+2iCXH/aMAp9s=
+github.com/konsorten/go-windows-terminal-sequences v1.0.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
+github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc=
+github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI=
+github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
+github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
+github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE=
+github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
+github.com/lib/pq v1.0.0 h1:X5PMW56eZitiTeO7tKzZxFCSpbFZJtkMMooicw2us9A=
+github.com/lib/pq v1.0.0/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo=
+github.com/markbates/oncer v0.0.0-20181203154359-bf2de49a0be2/go.mod h1:Ld9puTsIW75CHf65OeIOkyKbteujpZVXDpWK6YGZbxE=
+github.com/markbates/safe v1.0.1/go.mod h1:nAqgmRi7cY2nqMc92/bSEeQA+R4OheNU2T1kNSCBdG0=
+github.com/mattn/go-sqlite3 v1.9.0 h1:pDRiWfl+++eC2FEFRy6jXmQlvp4Yh3z1MJKg4UeYM/4=
+github.com/mattn/go-sqlite3 v1.9.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
+github.com/matttproud/golang_protobuf_extensions v1.0.1 h1:4hp9jkHxhMHkqkrB3Ix0jegS5sx/RkqARlsWZ6pIwiU=
+github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
+github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
+github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
+github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
+github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
+github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe/go.mod h1:wL8QJuTMNUDYhXwkmfOly8iTdp5TEcJFWZD2D7SIkUc=
+github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
+github.com/opentracing/opentracing-go v1.0.2 h1:3jA2P6O1F9UOrWVpwrIo17pu01KWvNWg4X946/Y5Zwg=
+github.com/opentracing/opentracing-go v1.0.2/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o=
+github.com/pelletier/go-toml v1.4.0/go.mod h1:PN7xzY2wHTK0K9p34ErDQMlFxa51Fk0OUruD3k1mMwo=
+github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
+github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
+github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
+github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
+github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
+github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
+github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw=
+github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo=
+github.com/prometheus/client_golang v1.5.1 h1:bdHYieyGlH+6OLEk2YQha8THib30KP0/yD0YH9m6xcA=
+github.com/prometheus/client_golang v1.5.1/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU=
+github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
+github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
+github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
+github.com/prometheus/client_model v0.2.0 h1:uq5h0d+GuxiXLJLNABMgp2qUWDPiLvgCzz2dUR+/W/M=
+github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
+github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4=
+github.com/prometheus/common v0.9.1 h1:KOMtN28tlbam3/7ZKEYKHhKoJZYYj3gMH4uc62x7X7U=
+github.com/prometheus/common v0.9.1/go.mod h1:yhUN8i9wzaXS3w1O07YhxHEBxD+W35wd8bs7vj7HSQ4=
+github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk=
+github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
+github.com/prometheus/procfs v0.0.8 h1:+fpWZdT24pJBiqJdAwYBjPSk+5YmQzYNPYzQsdzLkt8=
+github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A=
+github.com/rogpeppe/go-internal v1.1.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
+github.com/rogpeppe/go-internal v1.2.2/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
+github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
+github.com/sirupsen/logrus v1.0.6/go.mod h1:pMByvHTf9Beacp5x1UXfOR9xyW/9antXMhjMPG0dEzc=
+github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
+github.com/sirupsen/logrus v1.4.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
+github.com/sirupsen/logrus v1.4.1/go.mod h1:ni0Sbl8bgC9z8RoU9G6nDWqqs/fq4eDPysMBDgk/93Q=
+github.com/sirupsen/logrus v1.4.2 h1:SPIRibHv4MatM3XXNO2BJeFLZwZ2LvZgfQ5+UNI2im4=
+github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE=
+github.com/spf13/cobra v0.0.3/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ=
+github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
+github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
+github.com/stretchr/objx v0.1.1 h1:2vfRuCMp5sSVIDSqO8oNnWJq7mPa6KVP3iPIwFBuy8A=
+github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
+github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
+github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
+github.com/stretchr/testify v1.4.0 h1:2E4SXV/wtOkTonXsotYi4li6zVWxYlZuYNCXe9XRJyk=
+github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
+github.com/tidwall/pretty v1.0.0 h1:HsD+QiTn7sK6flMKIvNmpqz1qrpP3Ps6jOKIKMooyg4=
+github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk=
+github.com/tinode/snowflake v1.0.0 h1:YciQ9ZKn1TrnvpS8yZErt044XJaxWVtR9aMO9rOZVOE=
+github.com/tinode/snowflake v1.0.0/go.mod h1:5JiaCe3o7QdDeyRcAeZBGVghwRS+ygt2CF/hxmAoptQ=
+github.com/xdg/scram v0.0.0-20180814205039-7eeb5667e42c h1:u40Z8hqBAAQyv+vATcGgV0YCnDjqSL7/q/JyPhhJSPk=
+github.com/xdg/scram v0.0.0-20180814205039-7eeb5667e42c/go.mod h1:lB8K/P019DLNhemzwFU4jHLhdvlE6uDZjXFejJXr49I=
+github.com/xdg/stringprep v0.0.0-20180714160509-73f8eece6fdc h1:n+nNi93yXLkJvKwXNP9d55HC7lGK4H/SRcwB5IaUZLo=
+github.com/xdg/stringprep v0.0.0-20180714160509-73f8eece6fdc/go.mod h1:Jhud4/sHMO4oL310DaZAKk9ZaJ08SJfe+sJh0HrGL1Y=
+go.mongodb.org/mongo-driver v1.3.1 h1:op56IfTQiaY2679w922KVWa3qcHdml2K/Io8ayAOUEQ=
+go.mongodb.org/mongo-driver v1.3.1/go.mod h1:MSWZXKOynuguX+JSvwP8i+58jYCXxbia8HS3gZBapIE=
+go.opencensus.io v0.21.0 h1:mU6zScU4U1YAFPHEHYk+3JC4SY7JxgkqS10ZOSyksNg=
+go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
+golang.org/x/crypto v0.0.0-20180820150726-614d502a4dac/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
+golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
+golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
+golang.org/x/crypto v0.0.0-20190422162423-af44ce270edf/go.mod h1:WFFai1msRO1wXaEeE5yQxYXgSfI8pQAWXbQop6sCtWE=
+golang.org/x/crypto v0.0.0-20190530122614-20be4c3c3ed5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
+golang.org/x/crypto v0.0.0-20200320181102-891825fb96df h1:lDWgvUvNnaTnNBc/dwOty86cFeKoKWbwy2wQj0gIxbU=
+golang.org/x/crypto v0.0.0-20200320181102-891825fb96df/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
+golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
+golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
+golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
+golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
+golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
+golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
+golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
+golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
+golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
+golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
+golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
+golang.org/x/net v0.0.0-20200320220750-118fecf932d8 h1:1+zQlQqEEhUeStBTi653GZAnAuivZq/2hz+Iz+OP7rg=
+golang.org/x/net v0.0.0-20200320220750-118fecf932d8/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
+golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
+golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
+golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
+golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d h1:TzXSXBo42m9gQenoE3b9BGiEpg5IG2JkU5FkPIawgtw=
+golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
+golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20190412183630-56d357773e84/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e h1:vcxGaoTs7kV8m5Np9uUNQin4BrLOthgV7252N8V+FwY=
+golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sys v0.0.0-20180828065106-d99a578cf41b/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/sys v0.0.0-20190403152447-81d4e9dc473e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20190419153524-e8e3143a4f4a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20190531175056-4c3a928424d2/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20200122134326-e047566fdf82 h1:ywK/j/KkyTHcdyYSZNXGjMwgmDSfjglYZ3vStQ/gSCU=
+golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
+golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
+golang.org/x/text v0.3.2 h1:tW2bmiBqwgJj/UpqtC8EpXEZVYOwU0yG4iWbprSVAcs=
+golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
+golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
+golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
+golang.org/x/tools v0.0.0-20181030221726-6c7e314b6563/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
+golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
+golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
+golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
+golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
+golang.org/x/tools v0.0.0-20190329151228-23e29df326fe/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
+golang.org/x/tools v0.0.0-20190416151739-9c9e1878f421/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
+golang.org/x/tools v0.0.0-20190420181800-aa740d480789/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
+golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
+golang.org/x/tools v0.0.0-20190531172133-b3315ee88b7d/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
+golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4=
+golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
+google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
+google.golang.org/api v0.20.0 h1:jz2KixHX7EcCPiQrySzPdnYT7DbINAypCqKZ1Z7GM40=
+google.golang.org/api v0.20.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE=
+google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
+google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
+google.golang.org/appengine v1.5.0 h1:KxkO13IPW4Lslp2bz+KHP2E3gtFlrIGNThxkZQ3g+4c=
+google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
+google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
+google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
+google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
+google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55 h1:gSJIx1SDwno+2ElGhA4+qG2zF97qiUzTM+rQ0klBOcE=
+google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
+google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
+google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg=
+google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY=
+google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk=
+google.golang.org/grpc v1.28.0 h1:bO/TA4OxCOummhSf10siHuG7vJOiwh7SpRpFZDkOgl4=
+google.golang.org/grpc v1.28.0/go.mod h1:rpkK4SK4GF4Ach/+MFLZUBavHOvF2JJB5uozKKal+60=
+gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw=
+gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
+gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
+gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo=
+gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
+gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI=
+gopkg.in/fatih/pool.v2 v2.0.0 h1:xIFeWtxifuQJGk/IEPKsTduEKcKvPmhoiVDGpC40nKg=
+gopkg.in/fatih/pool.v2 v2.0.0/go.mod h1:8xVGeu1/2jr2wm5V9SPuMht2H5AEmf5aFMGSQixtjTY=
+gopkg.in/rethinkdb/rethinkdb-go.v5 v5.1.0 h1:ebhfJGmp8FpPwlgypOCcHhh556+AnUs2Tc48+GC8GDg=
+gopkg.in/rethinkdb/rethinkdb-go.v5 v5.1.0/go.mod h1:x+1XKi70FH0kHCpvPQ78hGBCCxoNdE7sP+kEFdKgN6A=
+gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
+gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
+gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
+gopkg.in/yaml.v2 v2.2.5 h1:ymVxjfMaHvXD8RqPRmzHHsB3VvucivSkIAvJFDI5O3c=
+gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
+honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
+honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
+honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
From 2601d51823425c8d7e90b2ac7ccc8eedfedbf9ef Mon Sep 17 00:00:00 2001
From: aforge
Date: Sat, 21 Mar 2020 23:10:54 -0700
Subject: [PATCH 036/142] Run tinode only if init-db returns success.
---
docker/README.md | 2 +-
docker/tinode/Dockerfile | 5 +--
docker/tinode/entrypoint.sh | 68 ++++++++++++++++---------------------
tinode-db/main.go | 11 +++++-
4 files changed, 43 insertions(+), 43 deletions(-)
diff --git a/docker/README.md b/docker/README.md
index 8780923af..03aac9f31 100644
--- a/docker/README.md
+++ b/docker/README.md
@@ -153,7 +153,7 @@ You can specify the following environment variables when issuing `docker run` co
| `MYSQL_DSN` | string | `'root@tcp(mysql)/tinode'` | MySQL [DSN](https://github.com/go-sql-driver/mysql#dsn-data-source-name). |
| `PLUGIN_PYTHON_CHAT_BOT_ENABLED` | bool | `false` | Enable calling into the plugin provided by Python chatbot |
| `RESET_DB` | bool | `false` | Drop and recreate the database. |
-| `SAMPLE_DATA` | string | _see comment_ | File with sample data to load. Default `data.json` when resetting or generating new DB, none when upgrading. Use `-` to disable |
+| `SAMPLE_DATA` | string | _see comment_ | File with sample data to load. Default `data.json` when resetting or generating new DB, none when upgrading. Use `` (empty string) to disable |
| `SMTP_DOMAINS` | string | | White list of email domains; when non-empty, accept registrations with emails from these domains only (email verification). |
| `SMTP_HOST_URL` | string | `'http://localhost:6060/'` | URL of the host where the webapp is running (email verification). |
| `SMTP_LOGIN` | string | | Optional login to use for authentication with the SMTP server (email verification). If login is missing, `addr-spec` part of `SMTP_SENDER` will be used: e.g. if `SMTP_SENDER` is `'"John Doe" '`, `jdoe@example.com` will be used as login. |
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index 6e438a1b2..9249a99f1 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -40,8 +40,8 @@ ENV RESET_DB=false
ENV UPGRADE_DB=false
# Load sample data to database from data.json.
-ARG DEFAULT_SAMPLE_DATA=data.json
-ENV SAMPLE_DATA=
+ARG SAMPLE_DATA=data.json
+ENV SAMPLE_DATA=$SAMPLE_DATA
# The MySQL DSN connection.
ENV MYSQL_DSN='root@tcp(mysql)/tinode'
@@ -102,6 +102,7 @@ RUN tar -xzf tinode-$TARGET_DB.linux-amd64.tar.gz \
&& rm tinode-$TARGET_DB.linux-amd64.tar.gz
# Copy config template to the container.
+COPY init-db .
COPY config.template .
COPY entrypoint.sh .
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index 1c4cf2741..cf5f716e9 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -50,8 +50,8 @@ else
fi
# Load default sample data when generating or resetting the database.
-if [[ -z "$SAMPLE_DATA" && "$UPGRADE_DB" = "false" ]] ; then
- SAMPLE_DATA="$DEFAULT_SAMPLE_DATA"
+if [[ "$UPGRADE_DB" = "true" ]] ; then
+ SAMPLE_DATA=""
fi
# If push notifications are enabled, generate client-side firebase config file.
@@ -89,42 +89,32 @@ if [ ! -z "$IOS_UNIV_LINKS_APP_ID" ] ; then
EOM
fi
-run_init_db=false
-run_tinode=false
-case "$SERVICES_TO_RUN" in
-"init-db")
- run_init_db=true
- ;;
-"tinode")
- run_tinode=true
- ;;
-"both")
- run_init_db=true
- run_tinode=true
- ;;
-*)
- echo "Invalid val for SERVICES_TO_RUN env var. Can be either 'init-db' or 'tinode' or 'both'."
- exit 1
- ;;
-esac
-
-echo "Will run init-db: ${run_init_db}, tinode: ${run_tinode}"
-
-touch /botdata/tino-password
-
-if [ "$run_init_db" == "true" ]; then
- init_args=("--reset=${RESET_DB}" "--upgrade=${UPGRADE_DB}" "--config=${CONFIG}")
- # Maybe load sample data?
- if [ ! -z "$SAMPLE_DATA" ] ; then
- init_args+=("--data=$SAMPLE_DATA")
- fi
- # Initialize the database if it has not been initialized yet or if data reset/upgrade has been requested.
- ./init-db "${init_args[@]}" | grep "usr;tino;" > /botdata/tino-password
+init_args=("--reset=${RESET_DB}" "--upgrade=${UPGRADE_DB}" "--config=${CONFIG}")
+# Maybe load sample data?
+if [ ! -z "$SAMPLE_DATA" ] ; then
+ init_args+=("--data=$SAMPLE_DATA")
+fi
+init_stdout=./init-db-stdout.txt
+init_stderr=./init-db-stderr.txt
+# Initialize the database if it has not been initialized yet or if data reset/upgrade has been requested.
+./init-db "${init_args[@]}" 1>$init_stdout 2>$init_stderr
+init_result=$?
+cat $init_stderr >&2
+if [ $init_result != 0 ]; then
+ echo "./init-db failed. Quitting."
+ exit 1
+fi
+
+# If sample data was provided, try to find Tino password.
+if [ ! -z "$SAMPLE_DATA" ] ; then
+ grep "usr;tino;" $init_stdout > /botdata/tino-password
fi
-if [ "$run_tinode" != "true" ]; then
- # If we don't want to run tinode, we are done.
- exit 0
+# Check if the last line in the output contains the magic string.
+grep -q "All done" $init_stderr
+if [ $? != 0 ]; then
+ echo "Database could not be set up correctly. Quitting."
+ exit 1
fi
if [ -s /botdata/tino-password ] ; then
@@ -140,10 +130,10 @@ args=("--config=${CONFIG}" "--static_data=$STATIC_DIR")
# Maybe set node name in the cluster.
if [ ! -z "$CLUSTER_SELF" ] ; then
- args+=("--cluster_self=$CLUSTER_SELF")
+ args+=("--cluster_self=$CLUSTER_SELF")
fi
if [ ! -z "$PPROF_URL" ] ; then
- args+=("--pprof_url=$PPROF_URL")
+ args+=("--pprof_url=$PPROF_URL")
fi
# Create the log directory (/var/log/tinode-`current timestamp`).
@@ -152,7 +142,7 @@ runid=tinode-`date +%s`
logdir=/var/log/$runid
mkdir -p $logdir
if [ -d /var/log/tinode-latest ]; then
- rm /var/log/tinode-latest
+ rm /var/log/tinode-latest
fi
pushd .
cd /var/log
diff --git a/tinode-db/main.go b/tinode-db/main.go
index f8ac4686d..bceb4a938 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -225,11 +225,20 @@ func main() {
// Upgrade DB from one version to another.
err = store.UpgradeDb(config.StoreConfig)
if err == nil {
- log.Println("Database successfully upgraded")
+ log.Println("Database successfully upgraded. All done.")
}
} else {
// Reset or create DB
err = store.InitDb(config.StoreConfig, true)
+ if err == nil {
+ var action string
+ if *reset {
+ action = "reset"
+ } else {
+ action = "initialized"
+ }
+ log.Println("Database ", action, ". All done.")
+ }
}
if err != nil {
From ef87515f84ed7228b9c3de81f6f651634a5c61da Mon Sep 17 00:00:00 2001
From: aforge
Date: Sat, 21 Mar 2020 23:17:43 -0700
Subject: [PATCH 037/142] Fix comment.
---
docker/tinode/entrypoint.sh | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index cf5f716e9..381e32058 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -110,7 +110,7 @@ if [ ! -z "$SAMPLE_DATA" ] ; then
grep "usr;tino;" $init_stdout > /botdata/tino-password
fi
-# Check if the last line in the output contains the magic string.
+# Check if the init-db output contains the magic string.
grep -q "All done" $init_stderr
if [ $? != 0 ]; then
echo "Database could not be set up correctly. Quitting."
From c328d98e13a9adb4220edd0bdc70ab0dd5749873 Mon Sep 17 00:00:00 2001
From: aforge
Date: Sun, 22 Mar 2020 00:20:32 -0700
Subject: [PATCH 038/142] Address comments.
---
docker/tinode/Dockerfile | 1 -
docker/tinode/entrypoint.sh | 6 +++---
2 files changed, 3 insertions(+), 4 deletions(-)
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index 9249a99f1..e40ab85b5 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -102,7 +102,6 @@ RUN tar -xzf tinode-$TARGET_DB.linux-amd64.tar.gz \
&& rm tinode-$TARGET_DB.linux-amd64.tar.gz
# Copy config template to the container.
-COPY init-db .
COPY config.template .
COPY entrypoint.sh .
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index 381e32058..ada99ada3 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -49,7 +49,7 @@ else
STATIC_DIR="./static"
fi
-# Load default sample data when generating or resetting the database.
+# Do not load data when upgrading database.
if [[ "$UPGRADE_DB" = "true" ]] ; then
SAMPLE_DATA=""
fi
@@ -100,7 +100,7 @@ init_stderr=./init-db-stderr.txt
./init-db "${init_args[@]}" 1>$init_stdout 2>$init_stderr
init_result=$?
cat $init_stderr >&2
-if [ $init_result != 0 ]; then
+if [ $init_result -ne 0 ]; then
echo "./init-db failed. Quitting."
exit 1
fi
@@ -112,7 +112,7 @@ fi
# Check if the init-db output contains the magic string.
grep -q "All done" $init_stderr
-if [ $? != 0 ]; then
+if [ $? -ne 0 ]; then
echo "Database could not be set up correctly. Quitting."
exit 1
fi
From b3e93f1014cd4f7d55b4f190c3fd924ef5fffc47 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 22 Mar 2020 13:28:09 +0300
Subject: [PATCH 039/142] unnecessary import and a different base image
---
chatbot/python/requirements.txt | 3 +--
docker/chatbot/Dockerfile | 8 ++++----
2 files changed, 5 insertions(+), 6 deletions(-)
diff --git a/chatbot/python/requirements.txt b/chatbot/python/requirements.txt
index e13eafe37..053ec320e 100644
--- a/chatbot/python/requirements.txt
+++ b/chatbot/python/requirements.txt
@@ -1,4 +1,3 @@
futures>=3.2.0; python_version<'3'
grpcio>=1.18.0
-requests>=2.21.0
-tinode-grpc>=0.15.11
+tinode-grpc>=0.16
diff --git a/docker/chatbot/Dockerfile b/docker/chatbot/Dockerfile
index 821892a46..826b10a5a 100644
--- a/docker/chatbot/Dockerfile
+++ b/docker/chatbot/Dockerfile
@@ -1,6 +1,6 @@
# Dockerfile builds an image with a chatbot (Tino) for Tinode.
-FROM python:slim
+FROM python:3.7-alpine
ARG VERSION=0.16
ENV VERSION=$VERSION
@@ -12,15 +12,15 @@ LABEL version=$VERSION
RUN mkdir -p /usr/src/bot
WORKDIR /usr/src/bot
-RUN pip install --no-cache-dir -q tinode-grpc
-
# Get tarball with the chatbot code and data.
ADD https://github.com/tinode/chat/releases/download/v$VERSION/py-chatbot.tar.gz .
# Unpack chatbot, delete archive
RUN tar -xzf py-chatbot.tar.gz \
&& rm py-chatbot.tar.gz
-# Use command line parameter `-e LOGIN_AS=user:password` to login as someone other than Tino.
+RUN pip install --no-cache-dir -r requirements.txt
+
+# Use docker's command line parameter `-e LOGIN_AS=user:password` to login as someone other than Tino.
CMD python chatbot.py --login-basic=${LOGIN_AS} --login-cookie=/botdata/.tn-cookie --host=tinode-srv:16061 > /var/log/chatbot.log
From 6613853dbf4e6735a90a5bec3f91c8679ccdf9ea Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 22 Mar 2020 13:28:26 +0300
Subject: [PATCH 040/142] expanded readme
---
docker/e2e/README.md | 8 +++++++-
1 file changed, 7 insertions(+), 1 deletion(-)
diff --git a/docker/e2e/README.md b/docker/e2e/README.md
index fa919e9c9..d4b98ee66 100644
--- a/docker/e2e/README.md
+++ b/docker/e2e/README.md
@@ -1 +1,7 @@
-Docker compose for E2E setup.
+# Docker compose for end-to-end setup.
+
+These reference docker-compose files will run Tinode either as a cluster or as stand-alone service.
+They both use MySQL backend.
+```
+docker-compose -f up -d
+```
From 99916feeea73c44796bb46214e2d44137553b375 Mon Sep 17 00:00:00 2001
From: aforge
Date: Sun, 22 Mar 2020 17:30:59 -0700
Subject: [PATCH 041/142] Check init-db exit code before proceeding to tinode
in tinode docker.
---
docker/tinode/entrypoint.sh | 14 ++------------
tinode-db/main.go | 10 +++++-----
2 files changed, 7 insertions(+), 17 deletions(-)
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index ada99ada3..d9d017c00 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -95,12 +95,9 @@ if [ ! -z "$SAMPLE_DATA" ] ; then
init_args+=("--data=$SAMPLE_DATA")
fi
init_stdout=./init-db-stdout.txt
-init_stderr=./init-db-stderr.txt
# Initialize the database if it has not been initialized yet or if data reset/upgrade has been requested.
-./init-db "${init_args[@]}" 1>$init_stdout 2>$init_stderr
-init_result=$?
-cat $init_stderr >&2
-if [ $init_result -ne 0 ]; then
+./init-db "${init_args[@]}" 1>$init_stdout
+if [ $? -ne 0 ]; then
echo "./init-db failed. Quitting."
exit 1
fi
@@ -110,13 +107,6 @@ if [ ! -z "$SAMPLE_DATA" ] ; then
grep "usr;tino;" $init_stdout > /botdata/tino-password
fi
-# Check if the init-db output contains the magic string.
-grep -q "All done" $init_stderr
-if [ $? -ne 0 ]; then
- echo "Database could not be set up correctly. Quitting."
- exit 1
-fi
-
if [ -s /botdata/tino-password ] ; then
# Convert Tino's authentication credentials into a cookie file.
# The cookie file is also used to check if database has been initialized.
diff --git a/tinode-db/main.go b/tinode-db/main.go
index bceb4a938..58cce008d 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -177,11 +177,11 @@ func main() {
if *datafile != "" && *datafile != "-" {
raw, err := ioutil.ReadFile(*datafile)
if err != nil {
- log.Fatal("Failed to parse data:", err)
+ log.Fatal("Failed to read sample data file:", err)
}
err = json.Unmarshal(raw, &data)
if err != nil {
- log.Fatal(err)
+ log.Fatal("Failed to parse sample data:", err)
}
}
@@ -218,14 +218,14 @@ func main() {
log.Println("Database reset requested")
} else {
log.Println("Database exists, DB version is correct. All done.")
- return
+ os.Exit(0)
}
if *upgrade {
// Upgrade DB from one version to another.
err = store.UpgradeDb(config.StoreConfig)
if err == nil {
- log.Println("Database successfully upgraded. All done.")
+ log.Println("Database successfully upgraded.")
}
} else {
// Reset or create DB
@@ -237,7 +237,7 @@ func main() {
} else {
action = "initialized"
}
- log.Println("Database ", action, ". All done.")
+ log.Println("Database ", action)
}
}
From 005c18ba5583b87830281cab124e6219ce4b1973 Mon Sep 17 00:00:00 2001
From: aforge
Date: Sun, 22 Mar 2020 20:47:22 -0700
Subject: [PATCH 042/142] Better startup order in E2E setup.
---
docker/e2e/README.md | 8 +++-
...cluster-docker-compose.yml => cluster.yml} | 44 ++++++-------------
...docker-compose.yml => single-instance.yml} | 37 +++-------------
docker/exporter/Dockerfile | 1 +
docker/exporter/entrypoint.sh | 10 +++++
docker/tinode/Dockerfile | 4 ++
docker/tinode/entrypoint.sh | 10 +++++
7 files changed, 51 insertions(+), 63 deletions(-)
rename docker/e2e/{cluster-docker-compose.yml => cluster.yml} (84%)
rename docker/e2e/{standalone-docker-compose.yml => single-instance.yml} (76%)
diff --git a/docker/e2e/README.md b/docker/e2e/README.md
index d4b98ee66..eb63f6098 100644
--- a/docker/e2e/README.md
+++ b/docker/e2e/README.md
@@ -1,7 +1,13 @@
# Docker compose for end-to-end setup.
-These reference docker-compose files will run Tinode either as a cluster or as stand-alone service.
+These reference docker-compose files will run Tinode either as [a single-instance](single-instance.yml) or [a 3-node cluster](cluster.yml) setup.
They both use MySQL backend.
+
```
docker-compose -f up -d
```
+
+By default, this command starts up a mysql instance, Tinode server(s) and Tinode exporter(s).
+Tinode server(s) is(are) configured similar to [Tinode Demo/Sandbox](../../README.md#demosandbox) and
+maps its web port to the host's port 6060 (6061, 6062).
+Tinode exporter(s) serve(s) metrics for Prometheus. Port mapping is 6222 (6223, 6224).
diff --git a/docker/e2e/cluster-docker-compose.yml b/docker/e2e/cluster.yml
similarity index 84%
rename from docker/e2e/cluster-docker-compose.yml
rename to docker/e2e/cluster.yml
index db1a1cc4b..998bf0dd3 100644
--- a/docker/e2e/cluster-docker-compose.yml
+++ b/docker/e2e/cluster.yml
@@ -36,6 +36,7 @@ x-tinode-env-vars: &tinode-env-vars
"SERVICES_TO_RUN": "tinode"
# You can provide your own tinode config by setting EXT_CONFIG env var and binding your configuration file to
# "EXT_CONFIG": "/etc/tinode/tinode.conf"
+ "WAIT_FOR": "mysql:3306"
x-exporter-env-vars: &exporter-env-vars
"TINODE_ADDR": "http://tinode.host:18080/stats/expvar/"
@@ -43,6 +44,7 @@ x-exporter-env-vars: &exporter-env-vars
"SERVE_FOR": "prometheus"
"PROM_NAMESPACE": "tinode"
"PROM_METRICS_PATH": "/metrics"
+ #
# InfluxDB configation:
# "SERVE_FOR": "influxdb"
# "INFLUXDB_VERSION": 1.7
@@ -69,34 +71,6 @@ services:
timeout: 5s
retries: 10
- # Run this only when you need to initialize Tinode database.
- init-db:
- << : *tinode-base
- container_name: init-db
- hostname: init-db
- restart: "no"
- networks:
- internal:
- ipv4_address: 172.19.0.2
- environment:
- << : *tinode-env-vars
- "SERVICES_TO_RUN": "init-db"
- "UPGRADE_DB": "false"
-
- # Run this only when you need to upgrade Tinode database to a newer version.
- upgrade-db:
- << : *tinode-base
- container_name: upgrade-db
- hostname: upgrade-db
- restart: "no"
- networks:
- internal:
- ipv4_address: 172.19.0.2
- environment:
- << : *tinode-env-vars
- "SERVICES_TO_RUN": "init-db"
- "UPGRADE_DB": "true"
-
# Tinode servers.
tinode-0:
<< : *tinode-base
@@ -107,7 +81,7 @@ services:
ipv4_address: 172.19.0.5
# You can mount your volumes as necessary:
# volumes:
- # # E.g. external config.
+ # # E.g. external config (assuming EXT_CONFIG is set).
# - :/etc/tinode/tinode.conf
# # Logs directory.
# - :/var/log
@@ -116,6 +90,9 @@ services:
environment:
<< : *tinode-env-vars
"CLUSTER_SELF": "tinode-0"
+ # Uncomment the lines below if you wish to upgrade or reset the database.
+ # "UPGRADE_DB": "true"
+ # "RESET_DB": "true"
tinode-1:
<< : *tinode-base
@@ -126,7 +103,7 @@ services:
ipv4_address: 172.19.0.6
# You can mount your volumes as necessary:
# volumes:
- # # E.g. external config.
+ # # E.g. external config (assuming EXT_CONFIG is set).
# - :/etc/tinode/tinode.conf
# # Logs directory.
# - :/var/log
@@ -135,6 +112,7 @@ services:
environment:
<< : *tinode-env-vars
"CLUSTER_SELF": "tinode-1"
+ "WAIT_FOR": "tinode-0:18080"
tinode-2:
<< : *tinode-base
@@ -145,7 +123,7 @@ services:
ipv4_address: 172.19.0.7
# You can mount your volumes as necessary:
# volumes:
- # # E.g. external config.
+ # # E.g. external config (assuming EXT_CONFIG is set).
# - :/etc/tinode/tinode.conf
# # Logs directory.
# - :/var/log
@@ -154,6 +132,7 @@ services:
environment:
<< : *tinode-env-vars
"CLUSTER_SELF": "tinode-2"
+ "WAIT_FOR": "tinode-0:18080"
# Monitoring.
# Exporters are paired with tinode instances.
@@ -168,6 +147,7 @@ services:
environment:
<< : *exporter-env-vars
"INSTANCE": "tinode-exp-0"
+ "WAIT_FOR": "tinode-0:18080"
extra_hosts:
- "tinode.host:172.19.0.5"
@@ -182,6 +162,7 @@ services:
environment:
<< : *exporter-env-vars
"INSTANCE": "tinode-exp-1"
+ "WAIT_FOR": "tinode-1:18080"
extra_hosts:
- "tinode.host:172.19.0.6"
@@ -196,6 +177,7 @@ services:
environment:
<< : *exporter-env-vars
"INSTANCE": "tinode-exp-2"
+ "WAIT_FOR": "tinode-2:18080"
extra_hosts:
- "tinode.host:172.19.0.7"
diff --git a/docker/e2e/standalone-docker-compose.yml b/docker/e2e/single-instance.yml
similarity index 76%
rename from docker/e2e/standalone-docker-compose.yml
rename to docker/e2e/single-instance.yml
index ccd15121a..dd05a8b37 100644
--- a/docker/e2e/standalone-docker-compose.yml
+++ b/docker/e2e/single-instance.yml
@@ -29,6 +29,7 @@ x-tinode-env-vars: &tinode-env-vars
"SERVICES_TO_RUN": "tinode"
# You can provide your own tinode config by setting EXT_CONFIG env var and binding your configuration file to
# "EXT_CONFIG": "/etc/tinode/tinode.conf"
+ "WAIT_FOR": "mysql:3306"
x-exporter-env-vars: &exporter-env-vars
"TINODE_ADDR": "http://tinode.host:18080/stats/expvar/"
@@ -62,46 +63,19 @@ services:
timeout: 5s
retries: 10
- # Run this only when you need to initialize Tinode database.
- init-db:
- << : *tinode-base
- container_name: init-db
- hostname: init-db
- restart: "no"
- networks:
- internal:
- ipv4_address: 172.19.0.2
- environment:
- << : *tinode-env-vars
- "SERVICES_TO_RUN": "init-db"
- # Comment out this line if you don't want sample data loaded.
- "SAMPLE_DATA": "data.json"
- "UPGRADE_DB": "false"
-
- # Run this only when you need to upgrade Tinode database to a newer version.
- upgrade-db:
- << : *tinode-base
- container_name: upgrade-db
- hostname: upgrade-db
- restart: "no"
- networks:
- internal:
- ipv4_address: 172.19.0.2
- environment:
- << : *tinode-env-vars
- "SERVICES_TO_RUN": "init-db"
- "UPGRADE_DB": "true"
-
+ # Tinode.
tinode-0:
<< : *tinode-base
container_name: tinode-0
hostname: tinode-0
+ depends_on:
+ - mysql
networks:
internal:
ipv4_address: 172.19.0.5
# You can mount your volumes as necessary:
# volumes:
- # # E.g. external config.
+ # # E.g. external config (assuming EXT_CONFIG is set).
# - :/etc/tinode/tinode.conf
# # Logs directory.
# - :/var/log
@@ -127,6 +101,7 @@ services:
<< : *exporter-env-vars
# Remove?
"INSTANCE": "tinode-exp-0"
+ "WAIT_FOR": "tinode-0:18080"
extra_hosts:
- "tinode.host:172.19.0.5"
diff --git a/docker/exporter/Dockerfile b/docker/exporter/Dockerfile
index 1602021a1..c22909d17 100644
--- a/docker/exporter/Dockerfile
+++ b/docker/exporter/Dockerfile
@@ -8,6 +8,7 @@ LABEL name="TinodeMetricExporter"
LABEL version=$VERSION
ENV SERVE_FOR=""
+ENV WAIT_FOR=""
ENV TINODE_ADDR=http://localhost/stats/expvar/
ENV INSTANCE="exporter-instance"
diff --git a/docker/exporter/entrypoint.sh b/docker/exporter/entrypoint.sh
index 6de9940ca..943c618c3 100755
--- a/docker/exporter/entrypoint.sh
+++ b/docker/exporter/entrypoint.sh
@@ -61,4 +61,14 @@ case "$SERVE_FOR" in
;;
esac
+# Wait for Tinode server if needed.
+if [ ! -z "$WAIT_FOR" ] ; then
+ IFS=':' read -ra TND <<< "$WAIT_FOR"
+ if [ ${#TND[@]} -ne 2 ]; then
+ echo "\$WAIT_FOR (${WAIT_FOR}) env var should be in form HOST:PORT"
+ exit 1
+ fi
+ until nc -z -v -w5 ${TND[0]} ${TND[1]}; do echo "waiting for ${WAIT_FOR}..."; sleep 5; done
+fi
+
./exporter "${args[@]}"
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index 6e438a1b2..7695d13ac 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -33,6 +33,10 @@ ENV TARGET_DB=$TARGET_DB
# Specifies what jobs to run: init-db, tinode or both.
ENV SERVICES_TO_RUN='both'
+# Specifies the database host:port pair to wait for before running Tinode.
+# Ignored if empty.
+ENV WAIT_FOR=
+
# An option to reset database.
ENV RESET_DB=false
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index 1c4cf2741..beb495ea7 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -89,6 +89,16 @@ if [ ! -z "$IOS_UNIV_LINKS_APP_ID" ] ; then
EOM
fi
+# Wait for database if needed.
+if [ ! -z "$WAIT_FOR" ] ; then
+ IFS=':' read -ra DB <<< "$WAIT_FOR"
+ if [ ${#DB[@]} -ne 2 ]; then
+ echo "\$WAIT_FOR (${WAIT_FOR}) env var should be in form HOST:PORT"
+ exit 1
+ fi
+ until nc -z -v -w5 ${DB[0]} ${DB[1]}; do echo "waiting for ${WAIT_FOR}..."; sleep 5; done
+fi
+
run_init_db=false
run_tinode=false
case "$SERVICES_TO_RUN" in
From 77fc309d37b3956d5ae62854ac1379d5ec9f651f Mon Sep 17 00:00:00 2001
From: aforge
Date: Sun, 22 Mar 2020 20:52:01 -0700
Subject: [PATCH 043/142] Add RESET_DB and UPGRADE_DB envvar comments to
single-instance.yml.
---
docker/e2e/single-instance.yml | 3 +++
1 file changed, 3 insertions(+)
diff --git a/docker/e2e/single-instance.yml b/docker/e2e/single-instance.yml
index dd05a8b37..be625aa7d 100644
--- a/docker/e2e/single-instance.yml
+++ b/docker/e2e/single-instance.yml
@@ -83,6 +83,9 @@ services:
- "6060:18080"
environment:
<< : *tinode-env-vars
+ # Uncomment the lines below if you wish to upgrade or reset the database.
+ # "UPGRADE_DB": "true"
+ # "RESET_DB": "true"
# Monitoring.
# Exporters are paired with tinode instances.
From 3f7d6ab729c4e777eaa6be4d4dbe1a7f93868144 Mon Sep 17 00:00:00 2001
From: aforge
Date: Sun, 22 Mar 2020 23:31:39 -0700
Subject: [PATCH 044/142] Use Tinode FCM default values in e2e docker-compose
configs.
---
docker/e2e/cluster.yml | 20 +++++++++++---------
docker/e2e/single-instance.yml | 20 +++++++++++---------
2 files changed, 22 insertions(+), 18 deletions(-)
diff --git a/docker/e2e/cluster.yml b/docker/e2e/cluster.yml
index 998bf0dd3..46982deda 100644
--- a/docker/e2e/cluster.yml
+++ b/docker/e2e/cluster.yml
@@ -24,19 +24,21 @@ x-exporter:
x-tinode-env-vars: &tinode-env-vars
"STORE_USE_ADAPTER": "mysql"
"DEFAULT_SAMPLE_DATA": ""
- "FCM_PUSH_ENABLED": "false"
- # "FCM_API_KEY": ""
- # "FCM_APP_ID": ""
- # "FCM_PROJECT_ID": ""
- # "FCM_SENDER_ID":
- # "FCM_VAPID_KEY": ""
- # "IOS_UNIV_LINKS_APP_ID": ""
"PPROF_URL": "/pprof"
- # Run tinode server only.
- "SERVICES_TO_RUN": "tinode"
# You can provide your own tinode config by setting EXT_CONFIG env var and binding your configuration file to
# "EXT_CONFIG": "/etc/tinode/tinode.conf"
"WAIT_FOR": "mysql:3306"
+ # Web client push notifications.
+ # Modify as appropriate.
+ # To enable Tinode server push notifications, please refer to the "push" section of your tinode.conf.
+ "FCM_PUSH_ENABLED": "true"
+ "FCM_API_KEY": "AIzaSyD6X4ULR-RUsobvs1zZ2bHdJuPz39q2tbQ"
+ "FCM_APP_ID": "1:114126160546:web:aca6ea2981feb81fb44dfb"
+ "FCM_PROJECT_ID": "tinode-1000"
+ "FCM_SENDER_ID": 114126160546
+ "FCM_VAPID_KEY": "BOgQVPOMzIMXUpsYGpbVkZoEBc0ifKY_f2kSU5DNDGYI6i6CoKqqxDd7w7PJ3FaGRBgVGJffldETumOx831jl58"
+ # iOS app universal links configuration.
+ # "IOS_UNIV_LINKS_APP_ID": ""
x-exporter-env-vars: &exporter-env-vars
"TINODE_ADDR": "http://tinode.host:18080/stats/expvar/"
diff --git a/docker/e2e/single-instance.yml b/docker/e2e/single-instance.yml
index be625aa7d..ec5c7d9aa 100644
--- a/docker/e2e/single-instance.yml
+++ b/docker/e2e/single-instance.yml
@@ -17,19 +17,21 @@ x-tinode:
x-tinode-env-vars: &tinode-env-vars
"STORE_USE_ADAPTER": "mysql"
"DEFAULT_SAMPLE_DATA": ""
- "FCM_PUSH_ENABLED": "false"
- # "FCM_API_KEY": ""
- # "FCM_APP_ID": ""
- # "FCM_PROJECT_ID": ""
- # "FCM_SENDER_ID":
- # "FCM_VAPID_KEY": ""
- # "IOS_UNIV_LINKS_APP_ID": ""
"PPROF_URL": "/pprof"
- # Run tinode server only.
- "SERVICES_TO_RUN": "tinode"
# You can provide your own tinode config by setting EXT_CONFIG env var and binding your configuration file to
# "EXT_CONFIG": "/etc/tinode/tinode.conf"
"WAIT_FOR": "mysql:3306"
+ # Web client push notifications.
+ # Modify as appropriate.
+ # To enable Tinode server push notifications, please refer to the "push" section of your tinode.conf.
+ "FCM_PUSH_ENABLED": "true"
+ "FCM_API_KEY": "AIzaSyD6X4ULR-RUsobvs1zZ2bHdJuPz39q2tbQ"
+ "FCM_APP_ID": "1:114126160546:web:aca6ea2981feb81fb44dfb"
+ "FCM_PROJECT_ID": "tinode-1000"
+ "FCM_SENDER_ID": 114126160546
+ "FCM_VAPID_KEY": "BOgQVPOMzIMXUpsYGpbVkZoEBc0ifKY_f2kSU5DNDGYI6i6CoKqqxDd7w7PJ3FaGRBgVGJffldETumOx831jl58"
+ # iOS app universal links configuration.
+ # "IOS_UNIV_LINKS_APP_ID": ""
x-exporter-env-vars: &exporter-env-vars
"TINODE_ADDR": "http://tinode.host:18080/stats/expvar/"
From a3b4f671b00af768b45a63c08e5532f9c126eea4 Mon Sep 17 00:00:00 2001
From: aforge
Date: Mon, 23 Mar 2020 20:03:57 -0700
Subject: [PATCH 045/142] Simplify code a bit.
---
docker/tinode/entrypoint.sh | 33 ++++-----------------------------
1 file changed, 4 insertions(+), 29 deletions(-)
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index d9d017c00..73026d629 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -51,7 +51,7 @@ fi
# Do not load data when upgrading database.
if [[ "$UPGRADE_DB" = "true" ]] ; then
- SAMPLE_DATA=""
+ SAMPLE_DATA=
fi
# If push notifications are enabled, generate client-side firebase config file.
@@ -89,11 +89,7 @@ if [ ! -z "$IOS_UNIV_LINKS_APP_ID" ] ; then
EOM
fi
-init_args=("--reset=${RESET_DB}" "--upgrade=${UPGRADE_DB}" "--config=${CONFIG}")
-# Maybe load sample data?
-if [ ! -z "$SAMPLE_DATA" ] ; then
- init_args+=("--data=$SAMPLE_DATA")
-fi
+init_args=("--reset=${RESET_DB}" "--upgrade=${UPGRADE_DB}" "--config=${CONFIG}" "--data=$SAMPLE_DATA")
init_stdout=./init-db-stdout.txt
# Initialize the database if it has not been initialized yet or if data reset/upgrade has been requested.
./init-db "${init_args[@]}" 1>$init_stdout
@@ -116,28 +112,7 @@ if [ -s /botdata/tino-password ] ; then
./credentials.sh /botdata/.tn-cookie < /botdata/tino-password
fi
-args=("--config=${CONFIG}" "--static_data=$STATIC_DIR")
-
-# Maybe set node name in the cluster.
-if [ ! -z "$CLUSTER_SELF" ] ; then
- args+=("--cluster_self=$CLUSTER_SELF")
-fi
-if [ ! -z "$PPROF_URL" ] ; then
- args+=("--pprof_url=$PPROF_URL")
-fi
-
-# Create the log directory (/var/log/tinode-`current timestamp`).
-# And symlink /var/log/tinode-latest to it.
-runid=tinode-`date +%s`
-logdir=/var/log/$runid
-mkdir -p $logdir
-if [ -d /var/log/tinode-latest ]; then
- rm /var/log/tinode-latest
-fi
-pushd .
-cd /var/log
-ln -s $runid tinode-latest
-popd
+args=("--config=${CONFIG}" "--static_data=$STATIC_DIR" "--cluster_self=$CLUSTER_SELF" "--pprof_url=$PPROF_URL")
# Run the tinode server.
-./tinode "${args[@]}" 2> $logdir/tinode.log
+./tinode "${args[@]}" 2> /var/log/tinode.log
From 29b7c5debbbb645811612d3a5400a08ccad20ed4 Mon Sep 17 00:00:00 2001
From: aforge
Date: Mon, 23 Mar 2020 20:22:03 -0700
Subject: [PATCH 046/142] Append to log file in tinode container.
---
docker/tinode/entrypoint.sh | 2 +-
tinode-db/main.go | 1 +
2 files changed, 2 insertions(+), 1 deletion(-)
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index 73026d629..15c9ca65c 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -115,4 +115,4 @@ fi
args=("--config=${CONFIG}" "--static_data=$STATIC_DIR" "--cluster_self=$CLUSTER_SELF" "--pprof_url=$PPROF_URL")
# Run the tinode server.
-./tinode "${args[@]}" 2> /var/log/tinode.log
+./tinode "${args[@]}" 2>> /var/log/tinode.log
diff --git a/tinode-db/main.go b/tinode-db/main.go
index 58cce008d..71593b902 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -246,4 +246,5 @@ func main() {
}
genDb(&data)
+ os.Exit(0)
}
From edb55c4822d1908bda85b2e35c7f74131e33d906 Mon Sep 17 00:00:00 2001
From: aforge
Date: Mon, 23 Mar 2020 21:14:05 -0700
Subject: [PATCH 047/142] More FCM configuration.
---
docker/e2e/cluster.yml | 6 +++++-
docker/e2e/single-instance.yml | 7 ++++++-
2 files changed, 11 insertions(+), 2 deletions(-)
diff --git a/docker/e2e/cluster.yml b/docker/e2e/cluster.yml
index 46982deda..9219481ed 100644
--- a/docker/e2e/cluster.yml
+++ b/docker/e2e/cluster.yml
@@ -31,7 +31,11 @@ x-tinode-env-vars: &tinode-env-vars
# Web client push notifications.
# Modify as appropriate.
# To enable Tinode server push notifications, please refer to the "push" section of your tinode.conf.
- "FCM_PUSH_ENABLED": "true"
+ "FCM_PUSH_ENABLED": "false"
+ # "FCM_CRED_FILE": ""
+ # "FCM_INCLUDE_ANDROID_NOTIFICATION": false
+ #
+ # FCM Web client configuration.
"FCM_API_KEY": "AIzaSyD6X4ULR-RUsobvs1zZ2bHdJuPz39q2tbQ"
"FCM_APP_ID": "1:114126160546:web:aca6ea2981feb81fb44dfb"
"FCM_PROJECT_ID": "tinode-1000"
diff --git a/docker/e2e/single-instance.yml b/docker/e2e/single-instance.yml
index ec5c7d9aa..4253c0d58 100644
--- a/docker/e2e/single-instance.yml
+++ b/docker/e2e/single-instance.yml
@@ -24,7 +24,12 @@ x-tinode-env-vars: &tinode-env-vars
# Web client push notifications.
# Modify as appropriate.
# To enable Tinode server push notifications, please refer to the "push" section of your tinode.conf.
- "FCM_PUSH_ENABLED": "true"
+ "FCM_PUSH_ENABLED": "false"
+ # FCM specific server configuration.
+ # "FCM_CRED_FILE": ""
+ # "FCM_INCLUDE_ANDROID_NOTIFICATION": false
+ #
+ # FCM Web client configuration.
"FCM_API_KEY": "AIzaSyD6X4ULR-RUsobvs1zZ2bHdJuPz39q2tbQ"
"FCM_APP_ID": "1:114126160546:web:aca6ea2981feb81fb44dfb"
"FCM_PROJECT_ID": "tinode-1000"
From d5f194a5ac3d5a8bb5bb8181687cc4579d138303 Mon Sep 17 00:00:00 2001
From: aforge
Date: Mon, 23 Mar 2020 22:35:38 -0700
Subject: [PATCH 048/142] Add Tinode Push Gateway configuration to Tinode
Docker setups.
---
docker/e2e/cluster.yml | 8 ++++++--
docker/e2e/single-instance.yml | 9 ++++++---
docker/tinode/Dockerfile | 12 ++++++++++++
docker/tinode/config.template | 9 +++++++++
4 files changed, 33 insertions(+), 5 deletions(-)
diff --git a/docker/e2e/cluster.yml b/docker/e2e/cluster.yml
index 9219481ed..22368481d 100644
--- a/docker/e2e/cluster.yml
+++ b/docker/e2e/cluster.yml
@@ -28,9 +28,13 @@ x-tinode-env-vars: &tinode-env-vars
# You can provide your own tinode config by setting EXT_CONFIG env var and binding your configuration file to
# "EXT_CONFIG": "/etc/tinode/tinode.conf"
"WAIT_FOR": "mysql:3306"
- # Web client push notifications.
+ # Push notifications.
# Modify as appropriate.
- # To enable Tinode server push notifications, please refer to the "push" section of your tinode.conf.
+ # Tinode Push Gateway configuration.
+ "TNPG_PUSH_ENABLED": "false"
+ # "TNPG_USER": ""
+ # "TNPG_AUTH_TOKEN": ""
+ # FCM specific server configuration.
"FCM_PUSH_ENABLED": "false"
# "FCM_CRED_FILE": ""
# "FCM_INCLUDE_ANDROID_NOTIFICATION": false
diff --git a/docker/e2e/single-instance.yml b/docker/e2e/single-instance.yml
index 4253c0d58..1d2ab6552 100644
--- a/docker/e2e/single-instance.yml
+++ b/docker/e2e/single-instance.yml
@@ -21,11 +21,14 @@ x-tinode-env-vars: &tinode-env-vars
# You can provide your own tinode config by setting EXT_CONFIG env var and binding your configuration file to
# "EXT_CONFIG": "/etc/tinode/tinode.conf"
"WAIT_FOR": "mysql:3306"
- # Web client push notifications.
+ # Push notifications.
# Modify as appropriate.
- # To enable Tinode server push notifications, please refer to the "push" section of your tinode.conf.
- "FCM_PUSH_ENABLED": "false"
+ # Tinode Push Gateway configuration.
+ "TNPG_PUSH_ENABLED": "false"
+ # "TNPG_USER": ""
+ # "TNPG_AUTH_TOKEN": ""
# FCM specific server configuration.
+ "FCM_PUSH_ENABLED": "false"
# "FCM_CRED_FILE": ""
# "FCM_INCLUDE_ANDROID_NOTIFICATION": false
#
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index bdb451413..a5752bc7a 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -86,6 +86,18 @@ ENV FCM_PUSH_ENABLED=false
# Enable Android-specific notifications by default.
ENV FCM_INCLUDE_ANDROID_NOTIFICATION=true
+# Disable push notifications via Tinode Push Gateway.
+ENV TNPG_PUSH_ENABLED=false
+
+# Tinode Push Gateway authentication token.
+ENV TNPG_AUTH_TOKEN=
+
+# Tinode Push Gateway user name.
+ENV TNPG_USER=
+
+# Enable Tinode Push Gateway request payload compression.
+ENV TNPG_COMPRESS_PAYLOADS=true
+
# Use the target db by default.
# When TARGET_DB is "alldbs", it is the user's responsibility
# to set STORE_USE_ADAPTER to the desired db adapter correctly.
diff --git a/docker/tinode/config.template b/docker/tinode/config.template
index 4e47f581c..0134253fd 100644
--- a/docker/tinode/config.template
+++ b/docker/tinode/config.template
@@ -105,6 +105,15 @@
},
"push": [
+ {
+ "name":"tnpg",
+ "config": {
+ "enabled": $TNPG_PUSH_ENABLED,
+ "auth_token": "$TNPG_AUTH_TOKEN",
+ "user": "$TNPG_USER",
+ "compress_payloads": $TNPG_COMPRESS_PAYLOADS
+ }
+ },
{
"name":"fcm",
"config": {
From 69a296099fe8705a73bda643bd9e161529169b9d Mon Sep 17 00:00:00 2001
From: aforge
Date: Mon, 23 Mar 2020 23:57:55 -0700
Subject: [PATCH 049/142] Push metrics to InfluxDB in e2e all exporters by
default.
---
docker/e2e/cluster.yml | 21 ++++++++++-----------
docker/e2e/single-instance.yml | 20 ++++++++++----------
2 files changed, 20 insertions(+), 21 deletions(-)
diff --git a/docker/e2e/cluster.yml b/docker/e2e/cluster.yml
index 22368481d..3a820c86f 100644
--- a/docker/e2e/cluster.yml
+++ b/docker/e2e/cluster.yml
@@ -50,18 +50,17 @@ x-tinode-env-vars: &tinode-env-vars
x-exporter-env-vars: &exporter-env-vars
"TINODE_ADDR": "http://tinode.host:18080/stats/expvar/"
- # Prometheus configuration:
- "SERVE_FOR": "prometheus"
- "PROM_NAMESPACE": "tinode"
- "PROM_METRICS_PATH": "/metrics"
- #
# InfluxDB configation:
- # "SERVE_FOR": "influxdb"
- # "INFLUXDB_VERSION": 1.7
- # "INFLUXDB_ORGANIZATION": ""
- # "INFLUXDB_PUSH_INTERVAL": 30
- # "INFLUXDB_PUSH_ADDRESS": "https://mon.tinode.co/intake"
- # "INFLUXDB_AUTH_TOKEN": "abcdef"
+ "SERVE_FOR": "influxdb"
+ "INFLUXDB_VERSION": 1.7
+ "INFLUXDB_ORGANIZATION": ""
+ "INFLUXDB_PUSH_INTERVAL": 30
+ "INFLUXDB_PUSH_ADDRESS": "https://mon.tinode.co/intake"
+ "INFLUXDB_AUTH_TOKEN": ""
+ # Prometheus configuration:
+ # "SERVE_FOR": "prometheus"
+ # "PROM_NAMESPACE": "tinode"
+ # "PROM_METRICS_PATH": "/metrics"
services:
mysql:
diff --git a/docker/e2e/single-instance.yml b/docker/e2e/single-instance.yml
index 1d2ab6552..08da0e525 100644
--- a/docker/e2e/single-instance.yml
+++ b/docker/e2e/single-instance.yml
@@ -43,17 +43,17 @@ x-tinode-env-vars: &tinode-env-vars
x-exporter-env-vars: &exporter-env-vars
"TINODE_ADDR": "http://tinode.host:18080/stats/expvar/"
- # Prometheus configuration:
- "SERVE_FOR": "prometheus"
- "PROM_NAMESPACE": "tinode"
- "PROM_METRICS_PATH": "/metrics"
# InfluxDB configation:
- # "SERVE_FOR": "influxdb"
- # "INFLUXDB_VERSION": 1.7
- # "INFLUXDB_ORGANIZATION": ""
- # "INFLUXDB_PUSH_INTERVAL": 30
- # "INFLUXDB_PUSH_ADDRESS": "https://mon.tinode.co/intake"
- # "INFLUXDB_AUTH_TOKEN": "abcdef"
+ "SERVE_FOR": "influxdb"
+ "INFLUXDB_VERSION": 1.7
+ "INFLUXDB_ORGANIZATION": ""
+ "INFLUXDB_PUSH_INTERVAL": 30
+ "INFLUXDB_PUSH_ADDRESS": "https://mon.tinode.co/intake"
+ "INFLUXDB_AUTH_TOKEN": ""
+ # Prometheus configuration:
+ # "SERVE_FOR": "prometheus"
+ # "PROM_NAMESPACE": "tinode"
+ # "PROM_METRICS_PATH": "/metrics"
services:
mysql:
From b0dcff1ff8914e067df807b7b37c1c29c933daa9 Mon Sep 17 00:00:00 2001
From: aforge
Date: Tue, 24 Mar 2020 01:01:43 -0700
Subject: [PATCH 050/142] Rename mysql -> db in e2e configs.
---
docker/e2e/cluster.yml | 4 ++--
docker/e2e/single-instance.yml | 6 ++----
2 files changed, 4 insertions(+), 6 deletions(-)
diff --git a/docker/e2e/cluster.yml b/docker/e2e/cluster.yml
index 3a820c86f..81d0fe292 100644
--- a/docker/e2e/cluster.yml
+++ b/docker/e2e/cluster.yml
@@ -10,7 +10,7 @@ version: '3.4'
x-tinode:
&tinode-base
depends_on:
- - mysql
+ - db
image: tinode/tinode:latest
restart: always
@@ -63,7 +63,7 @@ x-exporter-env-vars: &exporter-env-vars
# "PROM_METRICS_PATH": "/metrics"
services:
- mysql:
+ db:
image: mysql:5.7
container_name: mysql
restart: always
diff --git a/docker/e2e/single-instance.yml b/docker/e2e/single-instance.yml
index 08da0e525..c2393cff7 100644
--- a/docker/e2e/single-instance.yml
+++ b/docker/e2e/single-instance.yml
@@ -10,7 +10,7 @@ version: '3.4'
x-tinode:
&tinode-base
depends_on:
- - mysql
+ - db
image: tinode/tinode:latest
restart: always
@@ -56,7 +56,7 @@ x-exporter-env-vars: &exporter-env-vars
# "PROM_METRICS_PATH": "/metrics"
services:
- mysql:
+ db:
image: mysql:5.7
container_name: mysql
restart: always
@@ -78,8 +78,6 @@ services:
<< : *tinode-base
container_name: tinode-0
hostname: tinode-0
- depends_on:
- - mysql
networks:
internal:
ipv4_address: 172.19.0.5
From 703cfa548eed442e0f50d977250e00fcc79b2895 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 24 Mar 2020 21:24:30 +0300
Subject: [PATCH 051/142] missing dot in macro script
---
tn-cli/sample-macro-script.txt | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/tn-cli/sample-macro-script.txt b/tn-cli/sample-macro-script.txt
index 0d7e7663d..c25595280 100644
--- a/tn-cli/sample-macro-script.txt
+++ b/tn-cli/sample-macro-script.txt
@@ -28,7 +28,7 @@
.must usermod $user.params[user] --name 'Test User 1'
# Change test's anon acs.
-must chacs $user.params[user] --anon=JR
+.must chacs $user.params[user] --anon=JR
# Set test's password to test456.
passwd $user.params[user] -P test456
From c5c9e0c546e989fd8b5fff8694baa861cc872184 Mon Sep 17 00:00:00 2001
From: aforge
Date: Tue, 24 Mar 2020 18:09:19 -0700
Subject: [PATCH 052/142] Handle 'DEL' char in the set command in tn-cli.
---
tn-cli/tn-cli.py | 40 +++++++++++++++++++++++-----------------
1 file changed, 23 insertions(+), 17 deletions(-)
diff --git a/tn-cli/tn-cli.py b/tn-cli/tn-cli.py
index 01e0a7e37..d61b1e669 100644
--- a/tn-cli/tn-cli.py
+++ b/tn-cli/tn-cli.py
@@ -89,25 +89,31 @@ def make_vcard(fn, photofile):
card['fn'] = fn.strip()
if photofile != None:
- try:
- f = open(photofile, 'rb')
- # File extension is used as a file type
- mimetype = mimetypes.guess_type(photofile)
- if mimetype[0]:
- mimetype = mimetype[0].split("/")[1]
- else:
- mimetype = 'jpeg'
- data = base64.b64encode(f.read())
- # python3 fix.
- if type(data) is not str:
- data = data.decode()
+ if photofile == '␡':
+ # Delete the avatar.
card['photo'] = {
- 'data': data,
- 'type': mimetype
+ 'data': photofile
}
- f.close()
- except IOError as err:
- stdoutln("Error opening '" + photofile + "':", err)
+ else:
+ try:
+ f = open(photofile, 'rb')
+ # File extension is used as a file type
+ mimetype = mimetypes.guess_type(photofile)
+ if mimetype[0]:
+ mimetype = mimetype[0].split("/")[1]
+ else:
+ mimetype = 'jpeg'
+ data = base64.b64encode(f.read())
+ # python3 fix.
+ if type(data) is not str:
+ data = data.decode()
+ card['photo'] = {
+ 'data': data,
+ 'type': mimetype
+ }
+ f.close()
+ except IOError as err:
+ stdoutln("Error opening '" + photofile + "':", err)
return card
From 86161de6a1ebf505b6cd6ebb6802b397b3ea1170 Mon Sep 17 00:00:00 2001
From: aforge
Date: Tue, 24 Mar 2020 21:04:56 -0700
Subject: [PATCH 053/142] Simplify e2e configs (remove network).
---
docker/e2e/cluster.yml | 45 +++++++++-------------------------
docker/e2e/single-instance.yml | 23 ++---------------
2 files changed, 14 insertions(+), 54 deletions(-)
diff --git a/docker/e2e/cluster.yml b/docker/e2e/cluster.yml
index 81d0fe292..df782c287 100644
--- a/docker/e2e/cluster.yml
+++ b/docker/e2e/cluster.yml
@@ -18,8 +18,6 @@ x-exporter:
&exporter-base
image: tinode/exporter:latest
restart: always
- networks:
- internal:
x-tinode-env-vars: &tinode-env-vars
"STORE_USE_ADAPTER": "mysql"
@@ -67,9 +65,6 @@ services:
image: mysql:5.7
container_name: mysql
restart: always
- networks:
- internal:
- ipv4_address: 172.19.0.3
# Use your own volume.
# volumes:
# - :/var/lib/mysql
@@ -85,9 +80,6 @@ services:
<< : *tinode-base
container_name: tinode-0
hostname: tinode-0
- networks:
- internal:
- ipv4_address: 172.19.0.5
# You can mount your volumes as necessary:
# volumes:
# # E.g. external config (assuming EXT_CONFIG is set).
@@ -107,9 +99,6 @@ services:
<< : *tinode-base
container_name: tinode-1
hostname: tinode-1
- networks:
- internal:
- ipv4_address: 172.19.0.6
# You can mount your volumes as necessary:
# volumes:
# # E.g. external config (assuming EXT_CONFIG is set).
@@ -127,9 +116,6 @@ services:
<< : *tinode-base
container_name: tinode-2
hostname: tinode-2
- networks:
- internal:
- ipv4_address: 172.19.0.7
# You can mount your volumes as necessary:
# volumes:
# # E.g. external config (assuming EXT_CONFIG is set).
@@ -152,13 +138,13 @@ services:
depends_on:
- tinode-0
ports:
- - "6222:6222"
+ - 6222:6222
+ links:
+ - tinode-0:tinode.host
environment:
<< : *exporter-env-vars
- "INSTANCE": "tinode-exp-0"
+ "INSTANCE": "tinode-0"
"WAIT_FOR": "tinode-0:18080"
- extra_hosts:
- - "tinode.host:172.19.0.5"
exporter-1:
<< : *exporter-base
@@ -167,13 +153,13 @@ services:
depends_on:
- tinode-1
ports:
- - "6223:6222"
+ - 6223:6222
+ links:
+ - tinode-1:tinode.host
environment:
<< : *exporter-env-vars
- "INSTANCE": "tinode-exp-1"
+ "INSTANCE": "tinode-1"
"WAIT_FOR": "tinode-1:18080"
- extra_hosts:
- - "tinode.host:172.19.0.6"
exporter-2:
<< : *exporter-base
@@ -182,17 +168,10 @@ services:
depends_on:
- tinode-2
ports:
- - "6224:6222"
+ - 6224:6222
+ links:
+ - tinode-2:tinode.host
environment:
<< : *exporter-env-vars
- "INSTANCE": "tinode-exp-2"
+ "INSTANCE": "tinode-2"
"WAIT_FOR": "tinode-2:18080"
- extra_hosts:
- - "tinode.host:172.19.0.7"
-
-networks:
- internal:
- ipam:
- driver: default
- config:
- - subnet: "172.19.0.0/24"
diff --git a/docker/e2e/single-instance.yml b/docker/e2e/single-instance.yml
index c2393cff7..bb8ba76e7 100644
--- a/docker/e2e/single-instance.yml
+++ b/docker/e2e/single-instance.yml
@@ -60,9 +60,6 @@ services:
image: mysql:5.7
container_name: mysql
restart: always
- networks:
- internal:
- ipv4_address: 172.19.0.3
# Use your own volume.
# volumes:
# - :/var/lib/mysql
@@ -78,9 +75,6 @@ services:
<< : *tinode-base
container_name: tinode-0
hostname: tinode-0
- networks:
- internal:
- ipv4_address: 172.19.0.5
# You can mount your volumes as necessary:
# volumes:
# # E.g. external config (assuming EXT_CONFIG is set).
@@ -104,21 +98,8 @@ services:
- tinode-0
image: tinode/exporter:latest
restart: always
- networks:
- internal:
- ports:
- - "6222:6222"
+ links:
+ - tinode-0:tinode.host
environment:
<< : *exporter-env-vars
- # Remove?
- "INSTANCE": "tinode-exp-0"
"WAIT_FOR": "tinode-0:18080"
- extra_hosts:
- - "tinode.host:172.19.0.5"
-
-networks:
- internal:
- ipam:
- driver: default
- config:
- - subnet: "172.19.0.0/24"
From 78fc59e88f37a6359874f547a74ff655dd427b1d Mon Sep 17 00:00:00 2001
From: or-else
Date: Wed, 25 Mar 2020 08:53:59 +0300
Subject: [PATCH 054/142] simplify avatar deletion
---
tn-cli/tn-cli.py | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/tn-cli/tn-cli.py b/tn-cli/tn-cli.py
index d61b1e669..6592db142 100644
--- a/tn-cli/tn-cli.py
+++ b/tn-cli/tn-cli.py
@@ -89,10 +89,10 @@ def make_vcard(fn, photofile):
card['fn'] = fn.strip()
if photofile != None:
- if photofile == '␡':
+ if photofile == '':
# Delete the avatar.
card['photo'] = {
- 'data': photofile
+ 'data': '␡'
}
else:
try:
From 5888fcf28241cf4869d7b87b94d7e7eb5c7076e0 Mon Sep 17 00:00:00 2001
From: aforge
Date: Tue, 24 Mar 2020 23:43:25 -0700
Subject: [PATCH 055/142] Fetch correct user data when subscribing to ME on
behalf of another user.
---
server/init_topic.go | 14 +++++++++++++-
1 file changed, 13 insertions(+), 1 deletion(-)
diff --git a/server/init_topic.go b/server/init_topic.go
index d272a1a19..a255dd0c4 100644
--- a/server/init_topic.go
+++ b/server/init_topic.go
@@ -113,7 +113,19 @@ func topicInit(t *Topic, sreg *sessionJoin, h *Hub) {
func initTopicMe(t *Topic, sreg *sessionJoin) error {
t.cat = types.TopicCatMe
- user, err := store.Users.Get(sreg.sess.uid)
+ uid := types.ParseUserId(t.name)
+ if uid.IsZero() {
+ // Log out the session
+ sreg.sess.uid = types.ZeroUid
+ log.Println("malformed topic name (sub me): ", t.name, sreg.sess.sid)
+ return types.ErrMalformed
+ }
+ if uid != sreg.sess.uid && sreg.sess.authLvl != auth.LevelRoot {
+ sreg.sess.uid = types.ZeroUid
+ log.Println("initTopicMe: attempt to subscribe to another account by non-root", sreg.sess)
+ return types.ErrPermissionDenied
+ }
+ user, err := store.Users.Get(uid)
if err != nil {
// Log out the session
sreg.sess.uid = types.ZeroUid
From 57ed84720f46c6df253f6f0d300d0e56d59b87e8 Mon Sep 17 00:00:00 2001
From: aforge
Date: Wed, 25 Mar 2020 01:15:25 -0700
Subject: [PATCH 056/142] Move parsing of msg.from to dispatch from
initTopicMe.
---
server/init_topic.go | 11 -----------
server/session.go | 6 +++++-
2 files changed, 5 insertions(+), 12 deletions(-)
diff --git a/server/init_topic.go b/server/init_topic.go
index a255dd0c4..51ba20c51 100644
--- a/server/init_topic.go
+++ b/server/init_topic.go
@@ -114,17 +114,6 @@ func initTopicMe(t *Topic, sreg *sessionJoin) error {
t.cat = types.TopicCatMe
uid := types.ParseUserId(t.name)
- if uid.IsZero() {
- // Log out the session
- sreg.sess.uid = types.ZeroUid
- log.Println("malformed topic name (sub me): ", t.name, sreg.sess.sid)
- return types.ErrMalformed
- }
- if uid != sreg.sess.uid && sreg.sess.authLvl != auth.LevelRoot {
- sreg.sess.uid = types.ZeroUid
- log.Println("initTopicMe: attempt to subscribe to another account by non-root", sreg.sess)
- return types.ErrPermissionDenied
- }
user, err := store.Users.Get(uid)
if err != nil {
// Log out the session
diff --git a/server/session.go b/server/session.go
index 05bd27171..2e9f1cdad 100644
--- a/server/session.go
+++ b/server/session.go
@@ -283,7 +283,11 @@ func (s *Session) dispatch(msg *ClientComMessage) {
s.queueOut(ErrPermissionDenied("", "", msg.timestamp))
log.Println("s.dispatch: non-root asigned msg.from", s.sid)
return
- }
+ } else if fromUid := types.ParseUserId(msg.from); fromUid.IsZero() {
+ s.queueOut(ErrMalformed("", "", msg.timestamp))
+ log.Println("malformed msg.from: ", msg.from, s.sid)
+ return
+ }
var resp *ServerComMessage
if msg, resp = pluginFireHose(s, msg); resp != nil {
From ea4b294fcacefc05d511a81c975e11a7c624b50c Mon Sep 17 00:00:00 2001
From: aforge
Date: Wed, 25 Mar 2020 01:17:02 -0700
Subject: [PATCH 057/142] formatting
---
server/session.go | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/server/session.go b/server/session.go
index 2e9f1cdad..04fcf4709 100644
--- a/server/session.go
+++ b/server/session.go
@@ -287,7 +287,7 @@ func (s *Session) dispatch(msg *ClientComMessage) {
s.queueOut(ErrMalformed("", "", msg.timestamp))
log.Println("malformed msg.from: ", msg.from, s.sid)
return
- }
+ }
var resp *ServerComMessage
if msg, resp = pluginFireHose(s, msg); resp != nil {
From 5897700b5e495886dda4e704d25fef9bd422e1a8 Mon Sep 17 00:00:00 2001
From: aforge
Date: Wed, 25 Mar 2020 01:32:36 -0700
Subject: [PATCH 058/142] Treat explicitly specified empty string args to
Usermod as not-None.
---
tn-cli/macros.py | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/tn-cli/macros.py b/tn-cli/macros.py
index 179e07576..dc8435f79 100644
--- a/tn-cli/macros.py
+++ b/tn-cli/macros.py
@@ -85,11 +85,11 @@ def expand(self, id, cmd, args):
# Change VCard.
varname = cmd.varname if hasattr(cmd, 'varname') and cmd.varname else '$temp'
set_cmd = '.must ' + varname + ' set me'
- if cmd.name:
+ if cmd.name is not None:
set_cmd += ' --fn="%s"' % cmd.name
- if cmd.avatar:
+ if cmd.avatar is not None:
set_cmd += ' --photo="%s"' % cmd.avatar
- if cmd.comment:
+ if cmd.comment is not None:
set_cmd += ' --private="%s"' % cmd.comment
old_user = tn_globals.DefaultUser if tn_globals.DefaultUser else ''
return ['.use --user %s' % cmd.userid,
From 5a63f7b3a061789e60ee49e190a0068e6d3c0765 Mon Sep 17 00:00:00 2001
From: or-else
Date: Wed, 25 Mar 2020 11:55:36 +0300
Subject: [PATCH 059/142] remove unnecessary variable
---
server/init_topic.go | 3 +--
1 file changed, 1 insertion(+), 2 deletions(-)
diff --git a/server/init_topic.go b/server/init_topic.go
index 51ba20c51..98565b3ce 100644
--- a/server/init_topic.go
+++ b/server/init_topic.go
@@ -113,8 +113,7 @@ func topicInit(t *Topic, sreg *sessionJoin, h *Hub) {
func initTopicMe(t *Topic, sreg *sessionJoin) error {
t.cat = types.TopicCatMe
- uid := types.ParseUserId(t.name)
- user, err := store.Users.Get(uid)
+ user, err := store.Users.Get(types.ParseUserId(t.name))
if err != nil {
// Log out the session
sreg.sess.uid = types.ZeroUid
From baf17f5e7e05bd22a91ae070934c068bcec3a0c4 Mon Sep 17 00:00:00 2001
From: or-else
Date: Wed, 25 Mar 2020 11:59:05 +0300
Subject: [PATCH 060/142] commong logging string prefix
---
server/session.go | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/server/session.go b/server/session.go
index 04fcf4709..6f5dc3bc1 100644
--- a/server/session.go
+++ b/server/session.go
@@ -285,7 +285,7 @@ func (s *Session) dispatch(msg *ClientComMessage) {
return
} else if fromUid := types.ParseUserId(msg.from); fromUid.IsZero() {
s.queueOut(ErrMalformed("", "", msg.timestamp))
- log.Println("malformed msg.from: ", msg.from, s.sid)
+ log.Println("s.dispatch: malformed msg.from: ", msg.from, s.sid)
return
}
From fb30f65108301822f50903a2ba0b3c3c08c1fbc7 Mon Sep 17 00:00:00 2001
From: aforge
Date: Wed, 25 Mar 2020 23:46:50 -0700
Subject: [PATCH 061/142] Add docker-compose overrides for rethinkdb and
mongodb.
---
docker/e2e/README.md | 16 ++++++-
docker/e2e/cluster.mongodb.yml | 53 ++++++++++++++++++++++++
docker/e2e/cluster.rethinkdb.yml | 39 +++++++++++++++++
docker/e2e/cluster.yml | 4 ++
docker/e2e/single-instance.mongodb.yml | 37 +++++++++++++++++
docker/e2e/single-instance.rethinkdb.yml | 23 ++++++++++
docker/e2e/single-instance.yml | 2 +
7 files changed, 172 insertions(+), 2 deletions(-)
create mode 100644 docker/e2e/cluster.mongodb.yml
create mode 100644 docker/e2e/cluster.rethinkdb.yml
create mode 100644 docker/e2e/single-instance.mongodb.yml
create mode 100644 docker/e2e/single-instance.rethinkdb.yml
diff --git a/docker/e2e/README.md b/docker/e2e/README.md
index eb63f6098..7c7b06893 100644
--- a/docker/e2e/README.md
+++ b/docker/e2e/README.md
@@ -1,7 +1,6 @@
# Docker compose for end-to-end setup.
-These reference docker-compose files will run Tinode either as [a single-instance](single-instance.yml) or [a 3-node cluster](cluster.yml) setup.
-They both use MySQL backend.
+These reference docker-compose files will run Tinode with the MySql backend either as [a single-instance](single-instance.yml) or [a 3-node cluster](cluster.yml) setup.
```
docker-compose -f up -d
@@ -11,3 +10,16 @@ By default, this command starts up a mysql instance, Tinode server(s) and Tinode
Tinode server(s) is(are) configured similar to [Tinode Demo/Sandbox](../../README.md#demosandbox) and
maps its web port to the host's port 6060 (6061, 6062).
Tinode exporter(s) serve(s) metrics for Prometheus. Port mapping is 6222 (6223, 6224).
+
+Reference configuration for [RethinkDB 2.4.0](https://hub.docker.com/_/rethinkdb?tab=tags) and [MongoDB 4.2.3](https://hub.docker.com/_/mongo?tab=tags) is also available
+in the form of override files.
+
+Start single-instance setup with:
+```
+docker-compose -f single-instance.yml -f single-instance..yml up -d
+```
+
+And cluster with:
+```
+docker-compose -f cluster.yml -f cluster..yml up -d
+```
diff --git a/docker/e2e/cluster.mongodb.yml b/docker/e2e/cluster.mongodb.yml
new file mode 100644
index 000000000..ab8f0e70c
--- /dev/null
+++ b/docker/e2e/cluster.mongodb.yml
@@ -0,0 +1,53 @@
+version: '3.4'
+
+x-mongodb-tinode-env-vars: &mongodb-tinode-env-vars
+ "STORE_USE_ADAPTER": "mongodb"
+
+x-mongodb-exporter-env-vars: &mongodb-exporter-env-vars
+ "SERVE_FOR": "influxdb"
+
+services:
+ db:
+ image: mongo:4.2.3
+ container_name: mongodb
+ entrypoint: [ "/usr/bin/mongod", "--bind_ip_all", "--replSet", "rs0" ]
+ healthcheck:
+ test: ["CMD", "curl -f http://localhost:28017/ || exit 1"]
+
+ # Initializes MongoDb replicaset.
+ initdb:
+ image: mongo:4.2.3
+ container_name: initdb
+ depends_on:
+ - db
+ command: >
+ bash -c "echo 'Starting replica set initialize';
+ until mongo --host mongodb --eval 'print(\"waited for connection\")'; do sleep 2; done;
+ echo 'Connection finished';
+ echo 'Creating replica set';
+ echo \"rs.initiate({'_id': 'rs0', "members": [ {'_id': 0, 'host': 'mongodb:27017'} ]})\" | mongo --host mongodb"
+
+ tinode-0:
+ environment:
+ << : *mongodb-tinode-env-vars
+ "WAIT_FOR": "mongodb:27017"
+
+ tinode-1:
+ environment:
+ << : *mongodb-tinode-env-vars
+
+ tinode-2:
+ environment:
+ << : *mongodb-tinode-env-vars
+
+ exporter-0:
+ environment:
+ << : *mongodb-exporter-env-vars
+
+ exporter-1:
+ environment:
+ << : *mongodb-exporter-env-vars
+
+ exporter-2:
+ environment:
+ << : *mongodb-exporter-env-vars
diff --git a/docker/e2e/cluster.rethinkdb.yml b/docker/e2e/cluster.rethinkdb.yml
new file mode 100644
index 000000000..d74cea003
--- /dev/null
+++ b/docker/e2e/cluster.rethinkdb.yml
@@ -0,0 +1,39 @@
+version: '3.4'
+
+x-rethinkdb-tinode-env-vars: &rethinkdb-tinode-env-vars
+ "STORE_USE_ADAPTER": "rethinkdb"
+
+x-rethinkdb-exporter-env-vars: &rethinkdb-exporter-env-vars
+ "SERVE_FOR": "influxdb"
+
+services:
+ db:
+ image: rethinkdb:2.4.0
+ container_name: rethinkdb
+ healthcheck:
+ test: ["CMD", "curl -f http://localhost:8080/ || exit 1"]
+
+ tinode-0:
+ environment:
+ << : *rethinkdb-tinode-env-vars
+ "WAIT_FOR": "rethinkdb:8080"
+
+ tinode-1:
+ environment:
+ << : *rethinkdb-tinode-env-vars
+
+ tinode-2:
+ environment:
+ << : *rethinkdb-tinode-env-vars
+
+ exporter-0:
+ environment:
+ << : *rethinkdb-exporter-env-vars
+
+ exporter-1:
+ environment:
+ << : *rethinkdb-exporter-env-vars
+
+ exporter-2:
+ environment:
+ << : *rethinkdb-exporter-env-vars
diff --git a/docker/e2e/cluster.yml b/docker/e2e/cluster.yml
index df782c287..6afbd9f12 100644
--- a/docker/e2e/cluster.yml
+++ b/docker/e2e/cluster.yml
@@ -110,6 +110,8 @@ services:
environment:
<< : *tinode-env-vars
"CLUSTER_SELF": "tinode-1"
+ # Wait for tinode-0, not the database since
+ # we let tinode-0 perform all database initialization and upgrade work.
"WAIT_FOR": "tinode-0:18080"
tinode-2:
@@ -127,6 +129,8 @@ services:
environment:
<< : *tinode-env-vars
"CLUSTER_SELF": "tinode-2"
+ # Wait for tinode-0, not the database since
+ # we let tinode-0 perform all database initialization and upgrade work.
"WAIT_FOR": "tinode-0:18080"
# Monitoring.
diff --git a/docker/e2e/single-instance.mongodb.yml b/docker/e2e/single-instance.mongodb.yml
new file mode 100644
index 000000000..6667e5283
--- /dev/null
+++ b/docker/e2e/single-instance.mongodb.yml
@@ -0,0 +1,37 @@
+version: '3.4'
+
+x-mongodb-tinode-env-vars: &mongodb-tinode-env-vars
+ "STORE_USE_ADAPTER": "mongodb"
+
+x-mongodb-exporter-env-vars: &mongodb-exporter-env-vars
+ "SERVE_FOR": "influxdb"
+
+services:
+ db:
+ image: mongo:4.2.3
+ container_name: mongodb
+ entrypoint: [ "/usr/bin/mongod", "--bind_ip_all", "--replSet", "rs0" ]
+ healthcheck:
+ test: ["CMD", "curl -f http://localhost:28017/ || exit 1"]
+
+ # Initializes MongoDb replicaset.
+ initdb:
+ image: mongo:4.2.3
+ container_name: initdb
+ depends_on:
+ - db
+ command: >
+ bash -c "echo 'Starting replica set initialize';
+ until mongo --host mongodb --eval 'print(\"waited for connection\")'; do sleep 2; done;
+ echo 'Connection finished';
+ echo 'Creating replica set';
+ echo \"rs.initiate({'_id': 'rs0', "members": [ {'_id': 0, 'host': 'mongodb:27017'} ]})\" | mongo --host mongodb"
+
+ tinode-0:
+ environment:
+ << : *mongodb-tinode-env-vars
+ "WAIT_FOR": "mongodb:27017"
+
+ exporter-0:
+ environment:
+ << : *mongodb-exporter-env-vars
diff --git a/docker/e2e/single-instance.rethinkdb.yml b/docker/e2e/single-instance.rethinkdb.yml
new file mode 100644
index 000000000..6c0e7dedd
--- /dev/null
+++ b/docker/e2e/single-instance.rethinkdb.yml
@@ -0,0 +1,23 @@
+version: '3.4'
+
+x-rethinkdb-tinode-env-vars: &rethinkdb-tinode-env-vars
+ "STORE_USE_ADAPTER": "rethinkdb"
+
+x-rethinkdb-exporter-env-vars: &rethinkdb-exporter-env-vars
+ "SERVE_FOR": "influxdb"
+
+services:
+ db:
+ image: rethinkdb:2.4.0
+ container_name: rethinkdb
+ healthcheck:
+ test: ["CMD", "curl -f http://localhost:8080/ || exit 1"]
+
+ tinode-0:
+ environment:
+ << : *rethinkdb-tinode-env-vars
+ "WAIT_FOR": "rethinkdb:8080"
+
+ exporter-0:
+ environment:
+ << : *rethinkdb-exporter-env-vars
diff --git a/docker/e2e/single-instance.yml b/docker/e2e/single-instance.yml
index bb8ba76e7..e7d06602d 100644
--- a/docker/e2e/single-instance.yml
+++ b/docker/e2e/single-instance.yml
@@ -98,6 +98,8 @@ services:
- tinode-0
image: tinode/exporter:latest
restart: always
+ ports:
+ - "6222:6222"
links:
- tinode-0:tinode.host
environment:
From 7efdab243972826a942ad938c11dd545929ac464 Mon Sep 17 00:00:00 2001
From: or-else
Date: Thu, 26 Mar 2020 14:15:18 +0300
Subject: [PATCH 062/142] make chatbot docker more configurable
---
docker/chatbot/Dockerfile | 9 ++++++++-
1 file changed, 8 insertions(+), 1 deletion(-)
diff --git a/docker/chatbot/Dockerfile b/docker/chatbot/Dockerfile
index 826b10a5a..6c1d14a73 100644
--- a/docker/chatbot/Dockerfile
+++ b/docker/chatbot/Dockerfile
@@ -3,6 +3,8 @@
FROM python:3.7-alpine
ARG VERSION=0.16
+ARG LOGIN_AS=
+ARG TINODE_HOST=tinode-srv:6061
ENV VERSION=$VERSION
LABEL maintainer="Tinode Team "
@@ -10,8 +12,13 @@ LABEL name="TinodeChatbot"
LABEL version=$VERSION
RUN mkdir -p /usr/src/bot
+RUN mkdir /etc/botdata
+
WORKDIR /usr/src/bot
+# Volume with login cookie. Not created automatically.
+# VOLUME /etc/botdata
+
# Get tarball with the chatbot code and data.
ADD https://github.com/tinode/chat/releases/download/v$VERSION/py-chatbot.tar.gz .
# Unpack chatbot, delete archive
@@ -22,7 +29,7 @@ RUN pip install --no-cache-dir -r requirements.txt
# Use docker's command line parameter `-e LOGIN_AS=user:password` to login as someone other than Tino.
-CMD python chatbot.py --login-basic=${LOGIN_AS} --login-cookie=/botdata/.tn-cookie --host=tinode-srv:16061 > /var/log/chatbot.log
+CMD python chatbot.py --login-basic=${LOGIN_AS} --login-cookie=/etc/botdata/.tn-cookie --host=${TINODE_HOST} > /var/log/chatbot.log
# Plugin port
EXPOSE 40051
From 83150ad58f849ad28ed4b5f9f87c2138db45da0e Mon Sep 17 00:00:00 2001
From: or-else
Date: Thu, 26 Mar 2020 14:18:57 +0300
Subject: [PATCH 063/142] keep old cookie location in chatbot
---
docker/chatbot/Dockerfile | 5 ++---
docker/tinode/Dockerfile | 8 +-------
docker/tinode/entrypoint.sh | 5 ++---
server/push/tnpg/push_tnpg.go | 32 +++++++++++++-------------------
4 files changed, 18 insertions(+), 32 deletions(-)
diff --git a/docker/chatbot/Dockerfile b/docker/chatbot/Dockerfile
index 6c1d14a73..a790e013c 100644
--- a/docker/chatbot/Dockerfile
+++ b/docker/chatbot/Dockerfile
@@ -12,12 +12,11 @@ LABEL name="TinodeChatbot"
LABEL version=$VERSION
RUN mkdir -p /usr/src/bot
-RUN mkdir /etc/botdata
WORKDIR /usr/src/bot
# Volume with login cookie. Not created automatically.
-# VOLUME /etc/botdata
+# VOLUME /botdata
# Get tarball with the chatbot code and data.
ADD https://github.com/tinode/chat/releases/download/v$VERSION/py-chatbot.tar.gz .
@@ -29,7 +28,7 @@ RUN pip install --no-cache-dir -r requirements.txt
# Use docker's command line parameter `-e LOGIN_AS=user:password` to login as someone other than Tino.
-CMD python chatbot.py --login-basic=${LOGIN_AS} --login-cookie=/etc/botdata/.tn-cookie --host=${TINODE_HOST} > /var/log/chatbot.log
+CMD python chatbot.py --login-basic=${LOGIN_AS} --login-cookie=/botdata/.tn-cookie --host=${TINODE_HOST} > /var/log/chatbot.log
# Plugin port
EXPOSE 40051
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index a5752bc7a..66eff6dd5 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -11,7 +11,7 @@
FROM alpine:latest
-ARG VERSION=0.16.3
+ARG VERSION=0.16.4
ENV VERSION=$VERSION
LABEL maintainer="Tinode Team "
@@ -30,9 +30,6 @@ ENV TARGET_DB=$TARGET_DB
# Runtime options.
-# Specifies what jobs to run: init-db, tinode or both.
-ENV SERVICES_TO_RUN='both'
-
# Specifies the database host:port pair to wait for before running Tinode.
# Ignored if empty.
ENV WAIT_FOR=
@@ -95,9 +92,6 @@ ENV TNPG_AUTH_TOKEN=
# Tinode Push Gateway user name.
ENV TNPG_USER=
-# Enable Tinode Push Gateway request payload compression.
-ENV TNPG_COMPRESS_PAYLOADS=true
-
# Use the target db by default.
# When TARGET_DB is "alldbs", it is the user's responsibility
# to set STORE_USE_ADAPTER to the desired db adapter correctly.
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index 6e1331bc6..cb7c0db39 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -96,10 +96,10 @@ if [ ! -z "$WAIT_FOR" ] ; then
echo "\$WAIT_FOR (${WAIT_FOR}) env var should be in form HOST:PORT"
exit 1
fi
- until nc -z -v -w5 ${DB[0]} ${DB[1]}; do echo "waiting for ${WAIT_FOR}..."; sleep 5; done
+ until nc -z -v -w5 ${DB[0]} ${DB[1]}; do echo "waiting for ${WAIT_FOR}..."; sleep 3; done
fi
-init_args=("--reset=${RESET_DB}" "--upgrade=${UPGRADE_DB}" "--config=${CONFIG}" "--data=$SAMPLE_DATA")
+init_args=("--reset=${RESET_DB}" "--upgrade=${UPGRADE_DB}" "--config=${CONFIG}" "--data=${SAMPLE_DATA}")
init_stdout=./init-db-stdout.txt
# Initialize the database if it has not been initialized yet or if data reset/upgrade has been requested.
./init-db "${init_args[@]}" 1>$init_stdout
@@ -115,7 +115,6 @@ fi
if [ -s /botdata/tino-password ] ; then
# Convert Tino's authentication credentials into a cookie file.
- # The cookie file is also used to check if database has been initialized.
# /botdata/tino-password could be empty if DB was not updated. In such a case the
# /botdata/.tn-cookie will not be modified.
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index 5282bc49c..094af43c9 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -21,17 +21,16 @@ const (
var handler Handler
type Handler struct {
- input chan *push.Receipt
- stop chan bool
+ input chan *push.Receipt
+ stop chan bool
}
type configType struct {
- Enabled bool `json:"enabled"`
- Buffer int `json:"buffer"`
- CompressPayloads bool `json:"compress_payloads"`
- User string `json:"user"`
- AuthToken string `json:"auth_token"`
- Android fcm.AndroidConfig `json:"android,omitempty"`
+ Enabled bool `json:"enabled"`
+ Buffer int `json:"buffer"`
+ User string `json:"user"`
+ AuthToken string `json:"auth_token"`
+ Android fcm.AndroidConfig `json:"android,omitempty"`
}
// Init initializes the handler
@@ -68,23 +67,18 @@ func (Handler) Init(jsonconf string) error {
func postMessage(body interface{}, config *configType) (int, string, error) {
buf := new(bytes.Buffer)
- if config.CompressPayloads {
- gz := gzip.NewWriter(buf)
- json.NewEncoder(gz).Encode(body)
- gz.Close()
- } else {
- json.NewEncoder(buf).Encode(body)
- }
+ gz := gzip.NewWriter(buf)
+ json.NewEncoder(gz).Encode(body)
+ gz.Close()
targetAddress := baseTargetAddress + config.User
req, err := http.NewRequest("POST", targetAddress, buf)
if err != nil {
return -1, "", err
}
- req.Header.Add("Authorization", "Bearer " + config.AuthToken)
+ req.Header.Add("Authorization", "Bearer "+config.AuthToken)
req.Header.Set("Content-Type", "application/json; charset=utf-8")
- if config.CompressPayloads {
- req.Header.Add("Content-Encoding", "gzip")
- }
+ req.Header.Add("Content-Encoding", "gzip")
+
resp, err := http.DefaultClient.Do(req)
if err != nil {
return -1, "", err
From 01f0519bd59973e19253f6eed831fb9bd6597496 Mon Sep 17 00:00:00 2001
From: or-else
Date: Thu, 26 Mar 2020 20:06:29 +0300
Subject: [PATCH 064/142] go back to slim as slpine does not have gcc
---
docker/chatbot/Dockerfile | 5 +++--
1 file changed, 3 insertions(+), 2 deletions(-)
diff --git a/docker/chatbot/Dockerfile b/docker/chatbot/Dockerfile
index a790e013c..93b57cbc7 100644
--- a/docker/chatbot/Dockerfile
+++ b/docker/chatbot/Dockerfile
@@ -1,11 +1,12 @@
# Dockerfile builds an image with a chatbot (Tino) for Tinode.
-FROM python:3.7-alpine
+FROM python:3.7-slim
ARG VERSION=0.16
ARG LOGIN_AS=
ARG TINODE_HOST=tinode-srv:6061
ENV VERSION=$VERSION
+ARG BINVERS=$VERSION
LABEL maintainer="Tinode Team "
LABEL name="TinodeChatbot"
@@ -19,7 +20,7 @@ WORKDIR /usr/src/bot
# VOLUME /botdata
# Get tarball with the chatbot code and data.
-ADD https://github.com/tinode/chat/releases/download/v$VERSION/py-chatbot.tar.gz .
+ADD https://github.com/tinode/chat/releases/download/v${BINVERS}/py-chatbot.tar.gz .
# Unpack chatbot, delete archive
RUN tar -xzf py-chatbot.tar.gz \
&& rm py-chatbot.tar.gz
From 13262c7d64cb454ae5fa8c090a5b8519352dfba8 Mon Sep 17 00:00:00 2001
From: or-else
Date: Thu, 26 Mar 2020 20:09:59 +0300
Subject: [PATCH 065/142] typos
---
server/tinode.conf | 10 +++++-----
1 file changed, 5 insertions(+), 5 deletions(-)
diff --git a/server/tinode.conf b/server/tinode.conf
index 333a80b1e..566a23fb3 100644
--- a/server/tinode.conf
+++ b/server/tinode.conf
@@ -1,8 +1,8 @@
// The JSON comments are somewhat brittle. Don't try anything too fancy.
{
- // HTTP(S) address to listen on for websocket and long polling clients. Either a TCP host:port pair
+ // HTTP(S) address to listen on for websocket and long polling clients. Either a TCP host:port pair
// or a path to Unix socket as "unix:/path/to/socket.sock".
- // The TCP port is numeric value or a canonical name, e.g. ":80" or ":https". May include the host name,
+ // The TCP port is numeric value or a canonical name, e.g. ":80" or ":https". May include the host name,
///e.g. "localhost:80" or "hostname.example.com:https".
// It could be blank: if TLS is not configured it will default to ":80", otherwise to ":443".
// Can be overridden from the command line, see option --listen.
@@ -18,7 +18,7 @@
// URL path for mounting the directory with static files.
"static_mount": "/",
- // TCP host:port or unix:/path/to/socket to listen for gRPC clients.
+ // TCP host:port or unix:/path/to/socket to listen for gRPC clients.
// Leave blank to disable gRPC support.
// Could be overridden from the command line with --grpc_listen.
"grpc_listen": ":6061",
@@ -428,10 +428,10 @@
"account": "C"
},
- // Error code to use in case flugin has failed.
+ // Error code to use in case plugin has failed.
"failure_code": 0,
- // Text of an error message to report in case of plugin falure.
+ // Text of an error message to report in case of plugin failure.
"failure_text": null,
// Address of the plugin.
From 885ce7505c71727920bcf705f1c7aadd62282f9b Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 26 Mar 2020 20:52:14 -0700
Subject: [PATCH 066/142] Rename docker/e2e->docker-compose. Bump version up to
3.7.
---
docker/{e2e => docker-compose}/README.md | 0
.../cluster.mongodb.yml | 17 +-------
.../cluster.rethinkdb.yml} | 13 ++++---
docker/{e2e => docker-compose}/cluster.yml | 2 +-
.../single-instance.mongodb.yml | 9 +----
.../single-instance.rethinkdb.yml | 16 ++++++++
.../single-instance.yml | 2 +-
docker/e2e/cluster.rethinkdb.yml | 39 -------------------
8 files changed, 27 insertions(+), 71 deletions(-)
rename docker/{e2e => docker-compose}/README.md (100%)
rename docker/{e2e => docker-compose}/cluster.mongodb.yml (76%)
rename docker/{e2e/single-instance.rethinkdb.yml => docker-compose/cluster.rethinkdb.yml} (70%)
rename docker/{e2e => docker-compose}/cluster.yml (99%)
rename docker/{e2e => docker-compose}/single-instance.mongodb.yml (84%)
create mode 100644 docker/docker-compose/single-instance.rethinkdb.yml
rename docker/{e2e => docker-compose}/single-instance.yml (99%)
delete mode 100644 docker/e2e/cluster.rethinkdb.yml
diff --git a/docker/e2e/README.md b/docker/docker-compose/README.md
similarity index 100%
rename from docker/e2e/README.md
rename to docker/docker-compose/README.md
diff --git a/docker/e2e/cluster.mongodb.yml b/docker/docker-compose/cluster.mongodb.yml
similarity index 76%
rename from docker/e2e/cluster.mongodb.yml
rename to docker/docker-compose/cluster.mongodb.yml
index ab8f0e70c..229fb0277 100644
--- a/docker/e2e/cluster.mongodb.yml
+++ b/docker/docker-compose/cluster.mongodb.yml
@@ -1,11 +1,8 @@
-version: '3.4'
+version: '3.7'
x-mongodb-tinode-env-vars: &mongodb-tinode-env-vars
"STORE_USE_ADAPTER": "mongodb"
-x-mongodb-exporter-env-vars: &mongodb-exporter-env-vars
- "SERVE_FOR": "influxdb"
-
services:
db:
image: mongo:4.2.3
@@ -39,15 +36,3 @@ services:
tinode-2:
environment:
<< : *mongodb-tinode-env-vars
-
- exporter-0:
- environment:
- << : *mongodb-exporter-env-vars
-
- exporter-1:
- environment:
- << : *mongodb-exporter-env-vars
-
- exporter-2:
- environment:
- << : *mongodb-exporter-env-vars
diff --git a/docker/e2e/single-instance.rethinkdb.yml b/docker/docker-compose/cluster.rethinkdb.yml
similarity index 70%
rename from docker/e2e/single-instance.rethinkdb.yml
rename to docker/docker-compose/cluster.rethinkdb.yml
index 6c0e7dedd..4226ff703 100644
--- a/docker/e2e/single-instance.rethinkdb.yml
+++ b/docker/docker-compose/cluster.rethinkdb.yml
@@ -1,11 +1,8 @@
-version: '3.4'
+version: '3.7'
x-rethinkdb-tinode-env-vars: &rethinkdb-tinode-env-vars
"STORE_USE_ADAPTER": "rethinkdb"
-x-rethinkdb-exporter-env-vars: &rethinkdb-exporter-env-vars
- "SERVE_FOR": "influxdb"
-
services:
db:
image: rethinkdb:2.4.0
@@ -18,6 +15,10 @@ services:
<< : *rethinkdb-tinode-env-vars
"WAIT_FOR": "rethinkdb:8080"
- exporter-0:
+ tinode-1:
environment:
- << : *rethinkdb-exporter-env-vars
+ << : *rethinkdb-tinode-env-vars
+
+ tinode-2:
+ environment:
+ << : *rethinkdb-tinode-env-vars
diff --git a/docker/e2e/cluster.yml b/docker/docker-compose/cluster.yml
similarity index 99%
rename from docker/e2e/cluster.yml
rename to docker/docker-compose/cluster.yml
index 6afbd9f12..97af1a774 100644
--- a/docker/e2e/cluster.yml
+++ b/docker/docker-compose/cluster.yml
@@ -4,7 +4,7 @@
# * 3 Tinode servers
# * 3 exporters
-version: '3.4'
+version: '3.7'
# Base Tinode template.
x-tinode:
diff --git a/docker/e2e/single-instance.mongodb.yml b/docker/docker-compose/single-instance.mongodb.yml
similarity index 84%
rename from docker/e2e/single-instance.mongodb.yml
rename to docker/docker-compose/single-instance.mongodb.yml
index 6667e5283..19d6025ab 100644
--- a/docker/e2e/single-instance.mongodb.yml
+++ b/docker/docker-compose/single-instance.mongodb.yml
@@ -1,11 +1,8 @@
-version: '3.4'
+version: '3.7'
x-mongodb-tinode-env-vars: &mongodb-tinode-env-vars
"STORE_USE_ADAPTER": "mongodb"
-x-mongodb-exporter-env-vars: &mongodb-exporter-env-vars
- "SERVE_FOR": "influxdb"
-
services:
db:
image: mongo:4.2.3
@@ -31,7 +28,3 @@ services:
environment:
<< : *mongodb-tinode-env-vars
"WAIT_FOR": "mongodb:27017"
-
- exporter-0:
- environment:
- << : *mongodb-exporter-env-vars
diff --git a/docker/docker-compose/single-instance.rethinkdb.yml b/docker/docker-compose/single-instance.rethinkdb.yml
new file mode 100644
index 000000000..5dfa6afdc
--- /dev/null
+++ b/docker/docker-compose/single-instance.rethinkdb.yml
@@ -0,0 +1,16 @@
+version: '3.7'
+
+x-rethinkdb-tinode-env-vars: &rethinkdb-tinode-env-vars
+ "STORE_USE_ADAPTER": "rethinkdb"
+
+services:
+ db:
+ image: rethinkdb:2.4.0
+ container_name: rethinkdb
+ healthcheck:
+ test: ["CMD", "curl -f http://localhost:8080/ || exit 1"]
+
+ tinode-0:
+ environment:
+ << : *rethinkdb-tinode-env-vars
+ "WAIT_FOR": "rethinkdb:8080"
diff --git a/docker/e2e/single-instance.yml b/docker/docker-compose/single-instance.yml
similarity index 99%
rename from docker/e2e/single-instance.yml
rename to docker/docker-compose/single-instance.yml
index e7d06602d..10203396e 100644
--- a/docker/e2e/single-instance.yml
+++ b/docker/docker-compose/single-instance.yml
@@ -4,7 +4,7 @@
# * Tinode server
# * Tinode exporters
-version: '3.4'
+version: '3.7'
# Base Tinode template.
x-tinode:
diff --git a/docker/e2e/cluster.rethinkdb.yml b/docker/e2e/cluster.rethinkdb.yml
deleted file mode 100644
index d74cea003..000000000
--- a/docker/e2e/cluster.rethinkdb.yml
+++ /dev/null
@@ -1,39 +0,0 @@
-version: '3.4'
-
-x-rethinkdb-tinode-env-vars: &rethinkdb-tinode-env-vars
- "STORE_USE_ADAPTER": "rethinkdb"
-
-x-rethinkdb-exporter-env-vars: &rethinkdb-exporter-env-vars
- "SERVE_FOR": "influxdb"
-
-services:
- db:
- image: rethinkdb:2.4.0
- container_name: rethinkdb
- healthcheck:
- test: ["CMD", "curl -f http://localhost:8080/ || exit 1"]
-
- tinode-0:
- environment:
- << : *rethinkdb-tinode-env-vars
- "WAIT_FOR": "rethinkdb:8080"
-
- tinode-1:
- environment:
- << : *rethinkdb-tinode-env-vars
-
- tinode-2:
- environment:
- << : *rethinkdb-tinode-env-vars
-
- exporter-0:
- environment:
- << : *rethinkdb-exporter-env-vars
-
- exporter-1:
- environment:
- << : *rethinkdb-exporter-env-vars
-
- exporter-2:
- environment:
- << : *rethinkdb-exporter-env-vars
From a9798d5bbb0ba9750eeea9bec861b84109ef8c11 Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 26 Mar 2020 21:52:04 -0700
Subject: [PATCH 067/142] Make RESET_DB and UPGRADE_DB subsitute vars. Beef up
README.
---
docker/docker-compose/README.md | 48 ++++++++++++++++++++---
docker/docker-compose/cluster.yml | 5 +--
docker/docker-compose/single-instance.yml | 5 +--
3 files changed, 47 insertions(+), 11 deletions(-)
diff --git a/docker/docker-compose/README.md b/docker/docker-compose/README.md
index 7c7b06893..46f880f71 100644
--- a/docker/docker-compose/README.md
+++ b/docker/docker-compose/README.md
@@ -3,7 +3,7 @@
These reference docker-compose files will run Tinode with the MySql backend either as [a single-instance](single-instance.yml) or [a 3-node cluster](cluster.yml) setup.
```
-docker-compose -f up -d
+docker-compose -f [-f ] up -d
```
By default, this command starts up a mysql instance, Tinode server(s) and Tinode exporter(s).
@@ -14,12 +14,50 @@ Tinode exporter(s) serve(s) metrics for Prometheus. Port mapping is 6222 (6223,
Reference configuration for [RethinkDB 2.4.0](https://hub.docker.com/_/rethinkdb?tab=tags) and [MongoDB 4.2.3](https://hub.docker.com/_/mongo?tab=tags) is also available
in the form of override files.
-Start single-instance setup with:
+## Commands
+
+### Full stack
+To bring up the full stack, you can use the following commands:
+* MySql:
+-Single-instance setup: `docker-compose -f single-instance.yml up -d`
+-Cluster: `docker-compose -f cluster.yml up -d`
+* RethinkDb:
+-Single-instance setup: `docker-compose -f single-instance.yml -f single-instance.rethinkdb.yml up -d`
+-Cluster: `docker-compose -f cluster.yml -f cluster.rethinkdb.yml up -d`
+* MongoDb:
+-Single-instance setup: `docker-compose -f single-instance.yml -f single-instance.mongodb.yml up -d`
+-Cluster: `docker-compose -f cluster.yml -f cluster.mongodb.yml up -d`
+
+You can run individual/separate components of the setup by providing their names to the `docker-compose` command.
+E.g. to start the Tinode server in the single-instance MySql setup,
+```
+docker-compose -f single-instance.yml up -d tinode-0
+```
+
+### Database resets and/or version upgrades
+To reset the database or upgrade the database version, you can set `RESET_DB` or `UPGRADE_DB` environment variable to true when starting Tinode with docker-compose.
+E.g. for upgrading the database in MongoDb cluster setup, use:
+```
+UPGRADE_DB=true docker-compose -f cluster.yml -f cluster.mongodb.yml up -d tinode-0
+```
+
+For resetting the database in RethinkDb single-instance setup, run:
+```
+RESET_DB=true docker-compose -f single-instance.yml -f single-instance.rethinkdb.yml up -d tinode-0
+```
+
+## Troubleshooting
+Print out and verify your docker-compose configuration by running:
+```
+docker-compose -f [-f ] config
+```
+
+If the Tinode server(s) are failing, you can print the job's stdout/stderr with:
```
-docker-compose -f single-instance.yml -f single-instance..yml up -d
+docker logs tinode-
```
-And cluster with:
+Additionally, you can examine the jobs `tinode.log` file. To download it from the container, run:
```
-docker-compose -f cluster.yml -f cluster..yml up -d
+docker cp tinode-:/var/log/tinode.log .
```
diff --git a/docker/docker-compose/cluster.yml b/docker/docker-compose/cluster.yml
index 97af1a774..b1fe8395b 100644
--- a/docker/docker-compose/cluster.yml
+++ b/docker/docker-compose/cluster.yml
@@ -91,9 +91,8 @@ services:
environment:
<< : *tinode-env-vars
"CLUSTER_SELF": "tinode-0"
- # Uncomment the lines below if you wish to upgrade or reset the database.
- # "UPGRADE_DB": "true"
- # "RESET_DB": "true"
+ "RESET_DB": ${RESET_DB:-false}
+ "UPGRADE_DB": ${UPGRADE_DB:-false}
tinode-1:
<< : *tinode-base
diff --git a/docker/docker-compose/single-instance.yml b/docker/docker-compose/single-instance.yml
index 10203396e..f01eb98c5 100644
--- a/docker/docker-compose/single-instance.yml
+++ b/docker/docker-compose/single-instance.yml
@@ -85,9 +85,8 @@ services:
- "6060:18080"
environment:
<< : *tinode-env-vars
- # Uncomment the lines below if you wish to upgrade or reset the database.
- # "UPGRADE_DB": "true"
- # "RESET_DB": "true"
+ "RESET_DB": ${RESET_DB:-false}
+ "UPGRADE_DB": ${UPGRADE_DB:-false}
# Monitoring.
# Exporters are paired with tinode instances.
From aa651c5b8b68b39c3eee806178e91583ce7ab979 Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 26 Mar 2020 21:57:13 -0700
Subject: [PATCH 068/142] Fix formatting.
---
docker/docker-compose/README.md | 14 +++++++-------
1 file changed, 7 insertions(+), 7 deletions(-)
diff --git a/docker/docker-compose/README.md b/docker/docker-compose/README.md
index 46f880f71..6d26ff8ca 100644
--- a/docker/docker-compose/README.md
+++ b/docker/docker-compose/README.md
@@ -12,21 +12,21 @@ maps its web port to the host's port 6060 (6061, 6062).
Tinode exporter(s) serve(s) metrics for Prometheus. Port mapping is 6222 (6223, 6224).
Reference configuration for [RethinkDB 2.4.0](https://hub.docker.com/_/rethinkdb?tab=tags) and [MongoDB 4.2.3](https://hub.docker.com/_/mongo?tab=tags) is also available
-in the form of override files.
+in the override files.
## Commands
### Full stack
To bring up the full stack, you can use the following commands:
* MySql:
--Single-instance setup: `docker-compose -f single-instance.yml up -d`
--Cluster: `docker-compose -f cluster.yml up -d`
+ - Single-instance setup: `docker-compose -f single-instance.yml up -d`
+ - Cluster: `docker-compose -f cluster.yml up -d`
* RethinkDb:
--Single-instance setup: `docker-compose -f single-instance.yml -f single-instance.rethinkdb.yml up -d`
--Cluster: `docker-compose -f cluster.yml -f cluster.rethinkdb.yml up -d`
+ - Single-instance setup: `docker-compose -f single-instance.yml -f single-instance.rethinkdb.yml up -d`
+ - Cluster: `docker-compose -f cluster.yml -f cluster.rethinkdb.yml up -d`
* MongoDb:
--Single-instance setup: `docker-compose -f single-instance.yml -f single-instance.mongodb.yml up -d`
--Cluster: `docker-compose -f cluster.yml -f cluster.mongodb.yml up -d`
+ - Single-instance setup: `docker-compose -f single-instance.yml -f single-instance.mongodb.yml up -d`
+ - Cluster: `docker-compose -f cluster.yml -f cluster.mongodb.yml up -d`
You can run individual/separate components of the setup by providing their names to the `docker-compose` command.
E.g. to start the Tinode server in the single-instance MySql setup,
From b5ace68c7465abab4346a37d8ae1943fbda269a8 Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 08:14:14 +0300
Subject: [PATCH 069/142] removed $TNPG_COMPRESS_PAYLOADS as it is always true
---
docker/tinode/config.template | 3 +--
1 file changed, 1 insertion(+), 2 deletions(-)
diff --git a/docker/tinode/config.template b/docker/tinode/config.template
index 0134253fd..826c93375 100644
--- a/docker/tinode/config.template
+++ b/docker/tinode/config.template
@@ -110,8 +110,7 @@
"config": {
"enabled": $TNPG_PUSH_ENABLED,
"auth_token": "$TNPG_AUTH_TOKEN",
- "user": "$TNPG_USER",
- "compress_payloads": $TNPG_COMPRESS_PAYLOADS
+ "user": "$TNPG_USER"
}
},
{
From 0584eba0b847c9ab55f28154a2d1b4a8224d412c Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 08:33:40 +0300
Subject: [PATCH 070/142] typo and comment clarification
---
server/auth/rest/README.md | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/server/auth/rest/README.md b/server/auth/rest/README.md
index ba82f9991..8f71c3e81 100644
--- a/server/auth/rest/README.md
+++ b/server/auth/rest/README.md
@@ -48,11 +48,11 @@ Request and response payloads are formatted as JSON. Some of the request or resp
{
// ServerUrl is the URL of the authentication server to call.
"server_url": "http://127.0.0.1:5000/",
- // Server may create new accounts.
+ // Authentication server is allowed to create new accounts.
"allow_new_accounts": true,
// Use separate endpoints, i.e. add request name to serverUrl path when making requests:
// http://127.0.0.1:5000/add
- "use_separae_endpoints": true
+ "use_separate_endpoints": true
}
```
From d838803fccc30335051343031a81c540dc1713be Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 26 Mar 2020 23:34:23 -0700
Subject: [PATCH 071/142] init-db: do not load sample data on upgrade.
---
tinode-db/main.go | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
diff --git a/tinode-db/main.go b/tinode-db/main.go
index 71593b902..7ec8b2367 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -245,6 +245,8 @@ func main() {
log.Fatal("Failed to init DB:", err)
}
- genDb(&data)
+ if !*upgrade {
+ genDb(&data)
+ }
os.Exit(0)
}
From a6e275cc76d554a887e4896dc1cd45a4f12d4a2d Mon Sep 17 00:00:00 2001
From: aforge
Date: Thu, 26 Mar 2020 23:40:49 -0700
Subject: [PATCH 072/142] Add a logging message to say sample data ignored when
an upgrade was requested.
---
tinode-db/main.go | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
diff --git a/tinode-db/main.go b/tinode-db/main.go
index 7ec8b2367..bc50be9d4 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -247,6 +247,8 @@ func main() {
if !*upgrade {
genDb(&data)
- }
+ } else {
+ log.Println("Sample data was ignored. All done.")
+ }
os.Exit(0)
}
From b65273421a839b30c72e87f6090e8c9ef3ba4193 Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 09:49:23 +0300
Subject: [PATCH 073/142] update REST auth documentation per #401
---
server/auth/rest/README.md | 37 +++++++++++++++++++++++++++----------
1 file changed, 27 insertions(+), 10 deletions(-)
diff --git a/server/auth/rest/README.md b/server/auth/rest/README.md
index 8f71c3e81..0f67b9250 100644
--- a/server/auth/rest/README.md
+++ b/server/auth/rest/README.md
@@ -44,16 +44,33 @@ Request and response payloads are formatted as JSON. Some of the request or resp
## Configuration
-```js
-{
- // ServerUrl is the URL of the authentication server to call.
- "server_url": "http://127.0.0.1:5000/",
- // Authentication server is allowed to create new accounts.
- "allow_new_accounts": true,
- // Use separate endpoints, i.e. add request name to serverUrl path when making requests:
- // http://127.0.0.1:5000/add
- "use_separate_endpoints": true
-}
+Add the following section to the `auth_config` in [tinode.conf](../../tinode.conf):
+
+```json
+...
+"auth_config": {
+ ...
+ "myveryownauth": {
+ // ServerUrl is the URL of the authentication server to call.
+ "server_url": "http://127.0.0.1:5000/",
+ // Authentication server is allowed to create new accounts.
+ "allow_new_accounts": true,
+ // Use separate endpoints, i.e. add request name to serverUrl path when making requests:
+ // http://127.0.0.1:5000/add
+ "use_separate_endpoints": true
+ },
+ ...
+},
+```
+The name `myveryownauth` is completely arbitrary, but your client has to be configured to use it. If you want to use your
+config **instead** of stock `basic` (login-password) authentication, then add a logical renaming:
+```
+...
+"auth_config": {
+ "logical_names": ["myveryownauth:basic", "basic:"],
+ "myveryownauth": { ... },
+},
+...
```
## Request
From a7f0d1f26902b57bf8755cc32bc02dc11775ee72 Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 09:51:18 +0300
Subject: [PATCH 074/142] gofmt
---
tinode-db/main.go | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/tinode-db/main.go b/tinode-db/main.go
index bc50be9d4..78aa09f97 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -249,6 +249,6 @@ func main() {
genDb(&data)
} else {
log.Println("Sample data was ignored. All done.")
- }
+ }
os.Exit(0)
}
From de266fbf0554e3f12403f57724e38c635c0accce Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 09:54:03 +0300
Subject: [PATCH 075/142] markdown formatting
---
server/auth/rest/README.md | 5 +++--
1 file changed, 3 insertions(+), 2 deletions(-)
diff --git a/server/auth/rest/README.md b/server/auth/rest/README.md
index 0f67b9250..e1b60a9e6 100644
--- a/server/auth/rest/README.md
+++ b/server/auth/rest/README.md
@@ -46,7 +46,7 @@ Request and response payloads are formatted as JSON. Some of the request or resp
Add the following section to the `auth_config` in [tinode.conf](../../tinode.conf):
-```json
+```js
...
"auth_config": {
...
@@ -64,11 +64,12 @@ Add the following section to the `auth_config` in [tinode.conf](../../tinode.con
```
The name `myveryownauth` is completely arbitrary, but your client has to be configured to use it. If you want to use your
config **instead** of stock `basic` (login-password) authentication, then add a logical renaming:
-```
+```js
...
"auth_config": {
"logical_names": ["myveryownauth:basic", "basic:"],
"myveryownauth": { ... },
+ ...
},
...
```
From 346d2c79aac5bcce10ac8c24a9f6175144c0d2a4 Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 11:23:18 +0300
Subject: [PATCH 076/142] remove unused DEFAULT_SAMPLE_DATA
---
docker/docker-compose/cluster.yml | 1 -
docker/docker-compose/single-instance.yml | 1 -
2 files changed, 2 deletions(-)
diff --git a/docker/docker-compose/cluster.yml b/docker/docker-compose/cluster.yml
index b1fe8395b..ac1b681c1 100644
--- a/docker/docker-compose/cluster.yml
+++ b/docker/docker-compose/cluster.yml
@@ -21,7 +21,6 @@ x-exporter:
x-tinode-env-vars: &tinode-env-vars
"STORE_USE_ADAPTER": "mysql"
- "DEFAULT_SAMPLE_DATA": ""
"PPROF_URL": "/pprof"
# You can provide your own tinode config by setting EXT_CONFIG env var and binding your configuration file to
# "EXT_CONFIG": "/etc/tinode/tinode.conf"
diff --git a/docker/docker-compose/single-instance.yml b/docker/docker-compose/single-instance.yml
index f01eb98c5..e18a9d873 100644
--- a/docker/docker-compose/single-instance.yml
+++ b/docker/docker-compose/single-instance.yml
@@ -16,7 +16,6 @@ x-tinode:
x-tinode-env-vars: &tinode-env-vars
"STORE_USE_ADAPTER": "mysql"
- "DEFAULT_SAMPLE_DATA": ""
"PPROF_URL": "/pprof"
# You can provide your own tinode config by setting EXT_CONFIG env var and binding your configuration file to
# "EXT_CONFIG": "/etc/tinode/tinode.conf"
From ad4508e68cc9dc2fe46766b29464a421c0feb9b3 Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 11:23:45 +0300
Subject: [PATCH 077/142] consistent use of tabs vs spaces
---
docker/tinode/entrypoint.sh | 23 +++++++++++------------
1 file changed, 11 insertions(+), 12 deletions(-)
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index cb7c0db39..186198b1e 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -76,15 +76,15 @@ if [ ! -z "$IOS_UNIV_LINKS_APP_ID" ] ; then
# See https://developer.apple.com/library/archive/documentation/General/Conceptual/AppSearch/UniversalLinks.html for details.
cat > $STATIC_DIR/apple-app-site-association <<- EOM
{
- "applinks": {
- "apps": [],
- "details": [
- {
- "appID": "$IOS_UNIV_LINKS_APP_ID",
- "paths": [ "*" ]
- }
- ]
- }
+ "applinks": {
+ "apps": [],
+ "details": [
+ {
+ "appID": "$IOS_UNIV_LINKS_APP_ID",
+ "paths": [ "*" ]
+ }
+ ]
+ }
}
EOM
fi
@@ -99,10 +99,9 @@ if [ ! -z "$WAIT_FOR" ] ; then
until nc -z -v -w5 ${DB[0]} ${DB[1]}; do echo "waiting for ${WAIT_FOR}..."; sleep 3; done
fi
-init_args=("--reset=${RESET_DB}" "--upgrade=${UPGRADE_DB}" "--config=${CONFIG}" "--data=${SAMPLE_DATA}")
-init_stdout=./init-db-stdout.txt
# Initialize the database if it has not been initialized yet or if data reset/upgrade has been requested.
-./init-db "${init_args[@]}" 1>$init_stdout
+init_stdout=./init-db-stdout.txt
+./init-db --reset=${RESET_DB} --upgrade=${UPGRADE_DB} --config=${CONFIG} --data=${SAMPLE_DATA} 1>${init_stdout}
if [ $? -ne 0 ]; then
echo "./init-db failed. Quitting."
exit 1
From f4d0401d6a24e07993240b85dc3d5592379bde55 Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 12:04:17 +0300
Subject: [PATCH 078/142] consistent use of tabs vs spaces
---
docker/tinode/entrypoint.sh | 8 ++++----
1 file changed, 4 insertions(+), 4 deletions(-)
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index 186198b1e..143b2ffbd 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -44,9 +44,9 @@ fi
# If external static dir is defined, use it.
# Otherwise, fall back to "./static".
if [ ! -z "$EXT_STATIC_DIR" ] ; then
- STATIC_DIR=$EXT_STATIC_DIR
+ STATIC_DIR=$EXT_STATIC_DIR
else
- STATIC_DIR="./static"
+ STATIC_DIR="./static"
fi
# Do not load data when upgrading database.
@@ -57,7 +57,7 @@ fi
# If push notifications are enabled, generate client-side firebase config file.
if [ ! -z "$FCM_PUSH_ENABLED" ] ; then
# Write client config to $STATIC_DIR/firebase-init.js
- cat > $STATIC_DIR/firebase-init.js <<- EOM
+ cat > $STATIC_DIR/firebase-init.js <<- EOM
const FIREBASE_INIT = {
apiKey: "$FCM_API_KEY",
appId: "$FCM_APP_ID",
@@ -73,7 +73,7 @@ fi
if [ ! -z "$IOS_UNIV_LINKS_APP_ID" ] ; then
# Write config to $STATIC_DIR/apple-app-site-association config file.
- # See https://developer.apple.com/library/archive/documentation/General/Conceptual/AppSearch/UniversalLinks.html for details.
+ # See https://developer.apple.com/library/archive/documentation/General/Conceptual/AppSearch/UniversalLinks.html for details.
cat > $STATIC_DIR/apple-app-site-association <<- EOM
{
"applinks": {
From 208c669c56d0af2bb421797f9dc0dad9eb748acf Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 12:04:58 +0300
Subject: [PATCH 079/142] log that data was ignored only when the data was
provided
---
tinode-db/main.go | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/tinode-db/main.go b/tinode-db/main.go
index 78aa09f97..47076d03e 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -247,7 +247,7 @@ func main() {
if !*upgrade {
genDb(&data)
- } else {
+ } else if len(data.Users) > 0 {
log.Println("Sample data was ignored. All done.")
}
os.Exit(0)
From fb3d4c589be05f1a30a1ad598e69651a7b4f160a Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 12:35:49 +0300
Subject: [PATCH 080/142] defer logging adapter name & version till after
config is read
---
tinode-db/main.go | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/tinode-db/main.go b/tinode-db/main.go
index 47076d03e..78091714c 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -165,7 +165,6 @@ func getPassword(n int) string {
}
func main() {
- log.Println("Initializing", store.GetAdapterName(), store.GetAdapterVersion())
var reset = flag.Bool("reset", false, "force database reset")
var upgrade = flag.Bool("upgrade", false, "perform database version upgrade")
var datafile = flag.String("data", "", "name of file with sample data to load")
@@ -198,6 +197,8 @@ func main() {
err := store.Open(1, config.StoreConfig)
defer store.Close()
+ log.Println("Initializing", store.GetAdapterName(), store.GetAdapterVersion())
+
if err != nil {
if strings.Contains(err.Error(), "Database not initialized") {
log.Println("Database not found. Creating.")
From 1f25f0ca6584e017b90aeff5d3d01115988458a6 Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 13:01:26 +0300
Subject: [PATCH 081/142] log formatting
---
tinode-db/main.go | 22 +++++++++++-----------
1 file changed, 11 insertions(+), 11 deletions(-)
diff --git a/tinode-db/main.go b/tinode-db/main.go
index 78091714c..56e701409 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -176,11 +176,11 @@ func main() {
if *datafile != "" && *datafile != "-" {
raw, err := ioutil.ReadFile(*datafile)
if err != nil {
- log.Fatal("Failed to read sample data file:", err)
+ log.Fatalln("Failed to read sample data file:", err)
}
err = json.Unmarshal(raw, &data)
if err != nil {
- log.Fatal("Failed to parse sample data:", err)
+ log.Fatalln("Failed to parse sample data:", err)
}
}
@@ -189,9 +189,9 @@ func main() {
var config configType
if file, err := os.Open(*conffile); err != nil {
- log.Fatal("Failed to read config file:", err)
+ log.Fatalln("Failed to read config file:", err)
} else if err = json.NewDecoder(jcr.New(file)).Decode(&config); err != nil {
- log.Fatal("Failed to parse config file:", err)
+ log.Fatalln("Failed to parse config file:", err)
}
err := store.Open(1, config.StoreConfig)
@@ -206,14 +206,14 @@ func main() {
msg := "Wrong DB version: expected " + strconv.Itoa(store.GetAdapterVersion()) + ", got " +
strconv.Itoa(store.GetDbVersion()) + "."
if *reset {
- log.Println(msg + " Dropping and recreating the database.")
+ log.Println(msg, "Dropping and recreating the database.")
} else if *upgrade {
- log.Println(msg + " Upgrading the database.")
+ log.Println(msg, "Upgrading the database.")
} else {
- log.Fatal(msg + " Use --reset to reset, --upgrade to upgrade.")
+ log.Fatalln(msg, "Use --reset to reset, --upgrade to upgrade.")
}
} else {
- log.Fatal("Failed to init DB adapter:", err)
+ log.Fatalln("Failed to init DB adapter:", err)
}
} else if *reset {
log.Println("Database reset requested")
@@ -238,18 +238,18 @@ func main() {
} else {
action = "initialized"
}
- log.Println("Database ", action)
+ log.Println("Database", action)
}
}
if err != nil {
- log.Fatal("Failed to init DB:", err)
+ log.Fatalln("Failed to init DB:", err)
}
if !*upgrade {
genDb(&data)
} else if len(data.Users) > 0 {
- log.Println("Sample data was ignored. All done.")
+ log.Println("Sample data ignored. All done.")
}
os.Exit(0)
}
From 7c0c039d7bfa87a5f761af12ec8e0663fc77b900 Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 27 Mar 2020 19:03:28 +0300
Subject: [PATCH 082/142] add NO_DB_INIT param to init-db
---
docker/docker-compose/cluster.yml | 2 ++
docker/tinode/Dockerfile | 13 ++++++++-----
docker/tinode/entrypoint.sh | 8 +++++++-
tinode-db/main.go | 6 +++++-
4 files changed, 22 insertions(+), 7 deletions(-)
diff --git a/docker/docker-compose/cluster.yml b/docker/docker-compose/cluster.yml
index ac1b681c1..d48ac5b69 100644
--- a/docker/docker-compose/cluster.yml
+++ b/docker/docker-compose/cluster.yml
@@ -111,6 +111,7 @@ services:
# Wait for tinode-0, not the database since
# we let tinode-0 perform all database initialization and upgrade work.
"WAIT_FOR": "tinode-0:18080"
+ "NO_DB_INIT": "true"
tinode-2:
<< : *tinode-base
@@ -130,6 +131,7 @@ services:
# Wait for tinode-0, not the database since
# we let tinode-0 perform all database initialization and upgrade work.
"WAIT_FOR": "tinode-0:18080"
+ "NO_DB_INIT": "true"
# Monitoring.
# Exporters are paired with tinode instances.
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index 66eff6dd5..7d498c5a5 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -11,7 +11,7 @@
FROM alpine:latest
-ARG VERSION=0.16.4
+ARG VERSION=0.16
ENV VERSION=$VERSION
LABEL maintainer="Tinode Team "
@@ -40,6 +40,9 @@ ENV RESET_DB=false
# An option to upgrade database.
ENV UPGRADE_DB=false
+# Don't initialize database if it's missing
+ENV NO_DB_INIT=false
+
# Load sample data to database from data.json.
ARG SAMPLE_DATA=data.json
ENV SAMPLE_DATA=$SAMPLE_DATA
@@ -104,6 +107,10 @@ RUN apk update && \
WORKDIR /opt/tinode
+# Copy config template to the container.
+COPY config.template .
+COPY entrypoint.sh .
+
# Get the desired Tinode build.
ADD https://github.com/tinode/chat/releases/download/v$VERSION/tinode-$TARGET_DB.linux-amd64.tar.gz .
@@ -111,10 +118,6 @@ ADD https://github.com/tinode/chat/releases/download/v$VERSION/tinode-$TARGET_DB
RUN tar -xzf tinode-$TARGET_DB.linux-amd64.tar.gz \
&& rm tinode-$TARGET_DB.linux-amd64.tar.gz
-# Copy config template to the container.
-COPY config.template .
-COPY entrypoint.sh .
-
# Create directory for chatbot data.
RUN mkdir /botdata
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index 143b2ffbd..a64dabbd6 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -101,7 +101,13 @@ fi
# Initialize the database if it has not been initialized yet or if data reset/upgrade has been requested.
init_stdout=./init-db-stdout.txt
-./init-db --reset=${RESET_DB} --upgrade=${UPGRADE_DB} --config=${CONFIG} --data=${SAMPLE_DATA} 1>${init_stdout}
+./init-db \
+ --reset=${RESET_DB} \
+ --upgrade=${UPGRADE_DB} \
+ --config=${CONFIG} \
+ --data=${SAMPLE_DATA} \
+ --no_int=${NO_DB_INIT}
+ 1>${init_stdout}
if [ $? -ne 0 ]; then
echo "./init-db failed. Quitting."
exit 1
diff --git a/tinode-db/main.go b/tinode-db/main.go
index 56e701409..8ad153f0d 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -167,6 +167,7 @@ func getPassword(n int) string {
func main() {
var reset = flag.Bool("reset", false, "force database reset")
var upgrade = flag.Bool("upgrade", false, "perform database version upgrade")
+ var noInit = flag.Bool("no_init", false, "check that database exists but don't create if missing")
var datafile = flag.String("data", "", "name of file with sample data to load")
var conffile = flag.String("config", "./tinode.conf", "config of the database connection")
@@ -197,10 +198,13 @@ func main() {
err := store.Open(1, config.StoreConfig)
defer store.Close()
- log.Println("Initializing", store.GetAdapterName(), store.GetAdapterVersion())
+ log.Println("Database", store.GetAdapterName(), store.GetAdapterVersion())
if err != nil {
if strings.Contains(err.Error(), "Database not initialized") {
+ if *noInit {
+ log.Fatalln("Database not found.")
+ }
log.Println("Database not found. Creating.")
} else if strings.Contains(err.Error(), "Invalid database version") {
msg := "Wrong DB version: expected " + strconv.Itoa(store.GetAdapterVersion()) + ", got " +
From 926b60fd816ed10c3cfcfce84c4cd21b1b9881d2 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 28 Mar 2020 08:54:29 +0300
Subject: [PATCH 083/142] use [ syntax for consistency
---
docker/tinode/Dockerfile | 2 +-
docker/tinode/entrypoint.sh | 7 ++++++-
2 files changed, 7 insertions(+), 2 deletions(-)
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index 7d498c5a5..75be85cf6 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -40,7 +40,7 @@ ENV RESET_DB=false
# An option to upgrade database.
ENV UPGRADE_DB=false
-# Don't initialize database if it's missing
+# Option to skip DB initialization when it's missing.
ENV NO_DB_INIT=false
# Load sample data to database from data.json.
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index a64dabbd6..8b3043273 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -15,6 +15,11 @@ else
# Remove the old config.
rm -f working.config
+ # The 'alldbs' is not a valid adapter name.
+ if [ "$TARGET_DB" = "alldbs" ] ; then
+ TARGET_DB=
+ fi
+
# Enable email verification if $SMTP_SERVER is defined.
if [ ! -z "$SMTP_SERVER" ] ; then
EMAIL_VERIFICATION_REQUIRED='"auth"'
@@ -50,7 +55,7 @@ else
fi
# Do not load data when upgrading database.
-if [[ "$UPGRADE_DB" = "true" ]] ; then
+if [ "$UPGRADE_DB" = "true" ] ; then
SAMPLE_DATA=
fi
From d4cf94803c9f8d824d019a7d0bd18367149dab12 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 28 Mar 2020 10:02:46 +0300
Subject: [PATCH 084/142] more PEP 440 compliance in python build script
---
py_grpc/version.py | 4 ++++
1 file changed, 4 insertions(+)
diff --git a/py_grpc/version.py b/py_grpc/version.py
index 6602eb3e1..93f4f8939 100644
--- a/py_grpc/version.py
+++ b/py_grpc/version.py
@@ -10,6 +10,10 @@ def git_version():
line = line[1:]
if '-rc' in line:
line = line.replace('-rc', 'rc')
+ if '-beta' in line:
+ line = line.replace('-beta', 'b')
+ if '-alpha' in line:
+ line = line.replace('-alpha', 'a')
if '-' in line:
parts = line.split('-')
line = parts[0] + '.post' + parts[1]
From 55bcb0363fb5b80c0936d415e1f37fae3136b6c2 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 28 Mar 2020 10:49:57 +0300
Subject: [PATCH 085/142] typo in docker
---
docker/tinode/entrypoint.sh | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index 8b3043273..99c0d067c 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -111,7 +111,7 @@ init_stdout=./init-db-stdout.txt
--upgrade=${UPGRADE_DB} \
--config=${CONFIG} \
--data=${SAMPLE_DATA} \
- --no_int=${NO_DB_INIT}
+ --no_init=${NO_DB_INIT}
1>${init_stdout}
if [ $? -ne 0 ]; then
echo "./init-db failed. Quitting."
From 09fe46fe31937b8fcc6e7cfd674546c6111b2c93 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 28 Mar 2020 12:24:12 +0300
Subject: [PATCH 086/142] docker hub http api is just unusable
---
docker-release.sh | 33 ---------------------------------
1 file changed, 33 deletions(-)
diff --git a/docker-release.sh b/docker-release.sh
index 09a0be3d7..c36e62e31 100755
--- a/docker-release.sh
+++ b/docker-release.sh
@@ -39,39 +39,6 @@ source .dockerhub
# Login to docker hub
docker login -u $user -p $pass
-# Remove earlier builds
-for dbtag in "${dbtags[@]}"
-do
- name="$(containerName $dbtag)"
- if [ -n "$FULLRELEASE" ]; then
- curl -u $user:$pass -i -X DELETE \
- https://hub.docker.com/v2/repositories/tinode/${name}/tags/latest/
-
- curl -u $user:$pass -i -X DELETE \
- https://hub.docker.com/v2/repositories/tinode/${name}/tags/${ver[0]}.${ver[1]}/
- fi
- curl -u $user:$pass -i -X DELETE \
- https://hub.docker.com/v2/repositories/tinode/${name}/tags/${ver[0]}.${ver[1]}.${ver[2]}/
-done
-
-if [ -n "$FULLRELEASE" ]; then
- curl -u $user:$pass -i -X DELETE \
- https://hub.docker.com/v2/repositories/tinode/chatbot/tags/latest/
- curl -u $user:$pass -i -X DELETE \
- https://hub.docker.com/v2/repositories/tinode/chatbot/tags/${ver[0]}.${ver[1]}/
-fi
-curl -u $user:$pass -i -X DELETE \
- https://hub.docker.com/v2/repositories/tinode/chatbot/tags/${ver[0]}.${ver[1]}.${ver[2]}/
-
-if [ -n "$FULLRELEASE" ]; then
- curl -u $user:$pass -i -X DELETE \
- https://hub.docker.com/v2/repositories/tinode/exporter/tags/latest/
- curl -u $user:$pass -i -X DELETE \
- https://hub.docker.com/v2/repositories/tinode/exporter/tags/${ver[0]}.${ver[1]}/
-fi
-curl -u $user:$pass -i -X DELETE \
- https://hub.docker.com/v2/repositories/tinode/exporter/tags/${ver[0]}.${ver[1]}.${ver[2]}/
-
# Deploy images for various DB backends
for dbtag in "${dbtags[@]}"
do
From 23106c2f34feb8f990aec1526773ead0535160c8 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 28 Mar 2020 12:25:35 +0300
Subject: [PATCH 087/142] switch back to using port 6060
---
docker/README.md | 10 +++++-----
docker/docker-compose/cluster.yml | 16 ++++++++--------
docker/tinode/Dockerfile | 15 ++++++++-------
docker/tinode/config.template | 10 +++++-----
4 files changed, 26 insertions(+), 25 deletions(-)
diff --git a/docker/README.md b/docker/README.md
index 03aac9f31..8281dac3a 100644
--- a/docker/README.md
+++ b/docker/README.md
@@ -40,17 +40,17 @@ All images are available at https://hub.docker.com/r/tinode/
1. **RethinkDB**:
```
- $ docker run -p 6060:18080 -d --name tinode-srv --network tinode-net tinode/tinode-rethinkdb:latest
+ $ docker run -p 6060:6060 -d --name tinode-srv --network tinode-net tinode/tinode-rethinkdb:latest
```
2. **MySQL**:
```
- $ docker run -p 6060:18080 -d --name tinode-srv --network tinode-net tinode/tinode-mysql:latest
+ $ docker run -p 6060:6060 -d --name tinode-srv --network tinode-net tinode/tinode-mysql:latest
```
3. **MongoDB**:
```
- $ docker run -p 6060:18080 -d --name tinode-srv --network tinode-net tinode/tinode-mongodb:latest
+ $ docker run -p 6060:6060 -d --name tinode-srv --network tinode-net tinode/tinode-mongodb:latest
```
You can also run Tinode with the `tinode/tinode` image (which has all of the above DB adapters compiled in). You will need to specify the database adapter via `STORE_USE_ADAPTER` environment variable. E.g. for `mysql`, the command line will look like
@@ -76,7 +76,7 @@ All images are available at https://hub.docker.com/r/tinode/
The container comes with a built-in config file which can be customized with values from the environment variables (see [Supported environment variables](#supported_environment_variables) below). If changes are extensive it may be more convenient to replace the built-in config file with a custom one. In that case map the config file located on your host (e.g. `/users/jdoe/new_tinode.conf`) to container (e.g. `/tinode.conf`) using [Docker volumes](https://docs.docker.com/storage/volumes/) `--volume /users/jdoe/new_tinode.conf:/tinode.conf` then instruct the container to use the new config `--env EXT_CONFIG=/tinode.conf`:
```
-$ docker run -p 6060:18080 -d --name tinode-srv --network tinode-net \
+$ docker run -p 6060:6060 -d --name tinode-srv --network tinode-net \
--volume /users/jdoe/new_tinode.conf:/tinode.conf \
--env EXT_CONFIG=/tinode.conf \
tinode/tinode-mysql:latest
@@ -109,7 +109,7 @@ Project ID `myproject-1234`, App ID `1:141421356237:web:abc7de1234fab56cd78abc`,
is `83_Or_So_Random_Looking_Characters`, start the container with the following parameters (using MySQL container as an example):
```
-$ docker run -p 6060:18080 -d --name tinode-srv --network tinode-net \
+$ docker run -p 6060:6060 -d --name tinode-srv --network tinode-net \
-v /Users/jdoe:/fcm \
--env FCM_CRED_FILE=/fcm/myproject-1234-firebase-adminsdk-abc12-abcdef012345.json \
--env FCM_API_KEY=AIRaNdOmX4ULR-X6ranDomzZ2bHdRanDomq2tbQ \
diff --git a/docker/docker-compose/cluster.yml b/docker/docker-compose/cluster.yml
index d48ac5b69..ee6ae8bee 100644
--- a/docker/docker-compose/cluster.yml
+++ b/docker/docker-compose/cluster.yml
@@ -86,7 +86,7 @@ services:
# # Logs directory.
# - :/var/log
ports:
- - "6060:18080"
+ - "6060:6060"
environment:
<< : *tinode-env-vars
"CLUSTER_SELF": "tinode-0"
@@ -104,13 +104,13 @@ services:
# # Logs directory.
# - :/var/log
ports:
- - "6061:18080"
+ - "6061:6060"
environment:
<< : *tinode-env-vars
"CLUSTER_SELF": "tinode-1"
# Wait for tinode-0, not the database since
# we let tinode-0 perform all database initialization and upgrade work.
- "WAIT_FOR": "tinode-0:18080"
+ "WAIT_FOR": "tinode-0:6060"
"NO_DB_INIT": "true"
tinode-2:
@@ -124,13 +124,13 @@ services:
# # Logs directory.
# - :/var/log
ports:
- - "6062:18080"
+ - "6062:6060"
environment:
<< : *tinode-env-vars
"CLUSTER_SELF": "tinode-2"
# Wait for tinode-0, not the database since
# we let tinode-0 perform all database initialization and upgrade work.
- "WAIT_FOR": "tinode-0:18080"
+ "WAIT_FOR": "tinode-0:6060"
"NO_DB_INIT": "true"
# Monitoring.
@@ -148,7 +148,7 @@ services:
environment:
<< : *exporter-env-vars
"INSTANCE": "tinode-0"
- "WAIT_FOR": "tinode-0:18080"
+ "WAIT_FOR": "tinode-0:6060"
exporter-1:
<< : *exporter-base
@@ -163,7 +163,7 @@ services:
environment:
<< : *exporter-env-vars
"INSTANCE": "tinode-1"
- "WAIT_FOR": "tinode-1:18080"
+ "WAIT_FOR": "tinode-1:6060"
exporter-2:
<< : *exporter-base
@@ -178,4 +178,4 @@ services:
environment:
<< : *exporter-env-vars
"INSTANCE": "tinode-2"
- "WAIT_FOR": "tinode-2:18080"
+ "WAIT_FOR": "tinode-2:6060"
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index 75be85cf6..a3228d81f 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -1,9 +1,9 @@
# Docker file builds an image with a tinode chat server.
-# The server exposes port 18080.
-# In order to run the image you have to link it to a running RethinkDB container
-# (assuming it's named 'rethinkdb') and map the port where the tinode server accepts connections:
#
-# $ docker run -p 6060:18080 -d --link rethinkdb \
+# In order to run the image you have to link it to a running database container. For example, to
+# to use RethinkDB (named 'rethinkdb') and map the port where the tinode server accepts connections:
+#
+# $ docker run -p 6060:6060 -d --link rethinkdb \
# --env UID_ENCRYPTION_KEY=base64+encoded+16+bytes= \
# --env API_KEY_SALT=base64+encoded+32+bytes \
# --env AUTH_TOKEN_KEY=base64+encoded+32+bytes \
@@ -13,6 +13,7 @@ FROM alpine:latest
ARG VERSION=0.16
ENV VERSION=$VERSION
+ARG BINVERS=$VERSION
LABEL maintainer="Tinode Team "
LABEL name="TinodeChatServer"
@@ -112,7 +113,7 @@ COPY config.template .
COPY entrypoint.sh .
# Get the desired Tinode build.
-ADD https://github.com/tinode/chat/releases/download/v$VERSION/tinode-$TARGET_DB.linux-amd64.tar.gz .
+ADD https://github.com/tinode/chat/releases/download/v$BINVERS/tinode-$TARGET_DB.linux-amd64.tar.gz .
# Unpack the Tinode archive.
RUN tar -xzf tinode-$TARGET_DB.linux-amd64.tar.gz \
@@ -128,5 +129,5 @@ RUN chmod +x credentials.sh
# Generate config from template and run the server.
ENTRYPOINT ./entrypoint.sh
-# HTTP and gRPC ports
-EXPOSE 18080 16061
+# HTTP, gRPC, cluster ports
+EXPOSE 6060 16060 12000-12002
diff --git a/docker/tinode/config.template b/docker/tinode/config.template
index 826c93375..76c66535c 100644
--- a/docker/tinode/config.template
+++ b/docker/tinode/config.template
@@ -1,9 +1,9 @@
{
- "listen": ":18080",
+ "listen": ":6060",
"api_path": "/",
"cache_control": 39600,
"static_mount": "/",
- "grpc_listen": ":16061",
+ "grpc_listen": ":16060",
"api_key_salt": "$API_KEY_SALT",
"max_message_size": 4194304,
"max_subscriber_count": 32,
@@ -145,9 +145,9 @@
"self": "",
"nodes": [
// Name and TCP address of each node.
- {"name": "tinode-0", "addr": "tinode-0:12001"},
- {"name": "tinode-1", "addr": "tinode-1:12002"},
- {"name": "tinode-2", "addr": "tinode-2:12003"}
+ {"name": "tinode-0", "addr": "tinode-0:12000"},
+ {"name": "tinode-1", "addr": "tinode-1:12001"},
+ {"name": "tinode-2", "addr": "tinode-2:12002"}
],
"failover": {
"enabled": true,
From 075c90d492fe967880994f3d08296da8b5828d86 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 28 Mar 2020 12:29:58 +0300
Subject: [PATCH 088/142] more port updates in various places
---
chatbot/python/chatbot.py | 2 +-
docker/chatbot/Dockerfile | 2 +-
docker/docker-compose/single-instance.yml | 6 +++---
docker/tinode/Dockerfile | 2 +-
server/tinode.conf | 2 +-
5 files changed, 7 insertions(+), 7 deletions(-)
diff --git a/chatbot/python/chatbot.py b/chatbot/python/chatbot.py
index 4be5cb7fc..4db684e35 100644
--- a/chatbot/python/chatbot.py
+++ b/chatbot/python/chatbot.py
@@ -349,7 +349,7 @@ def exit_gracefully(signo, stack_frame):
purpose = "Tino, Tinode's chatbot."
print(purpose)
parser = argparse.ArgumentParser(description=purpose)
- parser.add_argument('--host', default='localhost:6061', help='address of Tinode server gRPC endpoint')
+ parser.add_argument('--host', default='localhost:16060', help='address of Tinode server gRPC endpoint')
parser.add_argument('--ssl', action='store_true', help='use SSL to connect to the server')
parser.add_argument('--ssl-host', help='SSL host name to use instead of default (useful for connecting to localhost)')
parser.add_argument('--listen', default='0.0.0.0:40051', help='address to listen on for incoming Plugin API calls')
diff --git a/docker/chatbot/Dockerfile b/docker/chatbot/Dockerfile
index 93b57cbc7..7b84395be 100644
--- a/docker/chatbot/Dockerfile
+++ b/docker/chatbot/Dockerfile
@@ -4,7 +4,7 @@ FROM python:3.7-slim
ARG VERSION=0.16
ARG LOGIN_AS=
-ARG TINODE_HOST=tinode-srv:6061
+ARG TINODE_HOST=tinode-srv:16060
ENV VERSION=$VERSION
ARG BINVERS=$VERSION
diff --git a/docker/docker-compose/single-instance.yml b/docker/docker-compose/single-instance.yml
index e18a9d873..0245482ee 100644
--- a/docker/docker-compose/single-instance.yml
+++ b/docker/docker-compose/single-instance.yml
@@ -41,7 +41,7 @@ x-tinode-env-vars: &tinode-env-vars
# "IOS_UNIV_LINKS_APP_ID": ""
x-exporter-env-vars: &exporter-env-vars
- "TINODE_ADDR": "http://tinode.host:18080/stats/expvar/"
+ "TINODE_ADDR": "http://tinode.host:6060/stats/expvar/"
# InfluxDB configation:
"SERVE_FOR": "influxdb"
"INFLUXDB_VERSION": 1.7
@@ -81,7 +81,7 @@ services:
# # Logs directory.
# - :/var/log
ports:
- - "6060:18080"
+ - "6060:6060"
environment:
<< : *tinode-env-vars
"RESET_DB": ${RESET_DB:-false}
@@ -102,4 +102,4 @@ services:
- tinode-0:tinode.host
environment:
<< : *exporter-env-vars
- "WAIT_FOR": "tinode-0:18080"
+ "WAIT_FOR": "tinode-0:6060"
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index a3228d81f..147b33c2b 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -130,4 +130,4 @@ RUN chmod +x credentials.sh
ENTRYPOINT ./entrypoint.sh
# HTTP, gRPC, cluster ports
-EXPOSE 6060 16060 12000-12002
+EXPOSE 6060 16060 12000-12003
diff --git a/server/tinode.conf b/server/tinode.conf
index 566a23fb3..f3f8fa696 100644
--- a/server/tinode.conf
+++ b/server/tinode.conf
@@ -21,7 +21,7 @@
// TCP host:port or unix:/path/to/socket to listen for gRPC clients.
// Leave blank to disable gRPC support.
// Could be overridden from the command line with --grpc_listen.
- "grpc_listen": ":6061",
+ "grpc_listen": ":16060",
// Enable handling of gRPC keepalives https://github.com/grpc/grpc/blob/master/doc/keepalive.md
// This sets server's GRPC_ARG_KEEPALIVE_TIME_MS to 60 seconds instead of the default 2 hours.
From 246786469ab78c4f31450dd1444019f4e0d124ed Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 28 Mar 2020 13:26:28 +0300
Subject: [PATCH 089/142] missing backslash
---
docker/tinode/entrypoint.sh | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index 99c0d067c..0fd201dbb 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -111,7 +111,7 @@ init_stdout=./init-db-stdout.txt
--upgrade=${UPGRADE_DB} \
--config=${CONFIG} \
--data=${SAMPLE_DATA} \
- --no_init=${NO_DB_INIT}
+ --no_init=${NO_DB_INIT} \
1>${init_stdout}
if [ $? -ne 0 ]; then
echo "./init-db failed. Quitting."
From 866d51f31dfea9b4978ea385bb6ce21fe1e64496 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 28 Mar 2020 14:50:26 +0300
Subject: [PATCH 090/142] keep chat log beteen restarts
---
docker/chatbot/Dockerfile | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/docker/chatbot/Dockerfile b/docker/chatbot/Dockerfile
index 7b84395be..bb8c5111b 100644
--- a/docker/chatbot/Dockerfile
+++ b/docker/chatbot/Dockerfile
@@ -29,7 +29,7 @@ RUN pip install --no-cache-dir -r requirements.txt
# Use docker's command line parameter `-e LOGIN_AS=user:password` to login as someone other than Tino.
-CMD python chatbot.py --login-basic=${LOGIN_AS} --login-cookie=/botdata/.tn-cookie --host=${TINODE_HOST} > /var/log/chatbot.log
+CMD python chatbot.py --login-basic=${LOGIN_AS} --login-cookie=/botdata/.tn-cookie --host=${TINODE_HOST} >> /var/log/chatbot.log
# Plugin port
EXPOSE 40051
From 96983760d2d0cd3e51efddb84bb9a5708b541d71 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 29 Mar 2020 10:30:29 +0300
Subject: [PATCH 091/142] update port references
---
docker/README.md | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/docker/README.md b/docker/README.md
index 8281dac3a..7436e3900 100644
--- a/docker/README.md
+++ b/docker/README.md
@@ -55,12 +55,12 @@ All images are available at https://hub.docker.com/r/tinode/
You can also run Tinode with the `tinode/tinode` image (which has all of the above DB adapters compiled in). You will need to specify the database adapter via `STORE_USE_ADAPTER` environment variable. E.g. for `mysql`, the command line will look like
```
- $ docker run -p 6060:18080 -d -e STORE_USE_ADAPTER mysql --name tinode-srv --network tinode-net tinode/tinode:latest
+ $ docker run -p 6060:6060 -d -e STORE_USE_ADAPTER mysql --name tinode-srv --network tinode-net tinode/tinode:latest
```
See [below](#supported-environment-variables) for more options.
- The port mapping `-p 6060:18080` tells Docker to map container's port 18080 to host's port 6060 making server accessible at http://localhost:6060/. The container will initialize the database with test data on the first run.
+ The port mapping `-p 5678:1234` tells Docker to map container's port 1234 to host's port 5678 making server accessible at http://localhost:5678/. The container will initialize the database with test data on the first run.
You may replace `:latest` with a different tag. See all all available tags here:
* [MySQL tags](https://hub.docker.com/r/tinode/tinode-mysql/tags/)
From 415014f0db5909113f04d99d0414120dd19bb46a Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 29 Mar 2020 12:50:52 +0300
Subject: [PATCH 092/142] fix chatbot failure to corretly connect to cluster
---
chatbot/python/chatbot.py | 54 ++++++++++++++++++++++++++++++---------
1 file changed, 42 insertions(+), 12 deletions(-)
diff --git a/chatbot/python/chatbot.py b/chatbot/python/chatbot.py
index 4db684e35..211153030 100644
--- a/chatbot/python/chatbot.py
+++ b/chatbot/python/chatbot.py
@@ -20,15 +20,19 @@
import time
import grpc
+from google.protobuf.json_format import MessageToDict
# Import generated grpc modules
from tinode_grpc import pb
from tinode_grpc import pbx
APP_NAME = "Tino-chatbot"
-APP_VERSION = "1.1.3"
+APP_VERSION = "1.2.0"
LIB_VERSION = pkg_resources.get_distribution("tinode_grpc").version
+# Maximum length of string to log. Shorten longer strings.
+MAX_LOG_LEN = 64
+
# User ID of the current user
botUID = None
@@ -42,16 +46,29 @@
def add_future(tid, bundle):
onCompletion[tid] = bundle
+# Shorten long strings for logging.
+class StrLogger(json.JSONEncoder):
+ def default(self, obj):
+ if type(obj) == str and len(obj) > MAX_LOG_LEN:
+ return '<' + len(obj) + ', bytes: ' + obj[:12] + '...' + obj[-12:] + '>'
+ return super(StrLogger, self).default(obj)
+
# Resolve or reject the future
def exec_future(tid, code, text, params):
bundle = onCompletion.get(tid)
if bundle != None:
del onCompletion[tid]
- if code >= 200 and code < 400:
- arg = bundle.get('arg')
- bundle.get('action')(arg, params)
- else:
- print("Error:", code, text)
+ try:
+ if code >= 200 and code < 400:
+ arg = bundle.get('arg')
+ bundle.get('onsuccess')(arg, params)
+ else:
+ print("Error: {} {} ({})".format(code, text, tid))
+ onerror = bundle.get('onerror')
+ if onerror:
+ onerror(bundle.get('arg'), {'code': code, 'text': text})
+ except Exception as err:
+ print("Error handling server response", err)
# List of active subscriptions
subscriptions = {}
@@ -61,6 +78,16 @@ def add_subscription(topic):
def del_subscription(topic):
subscriptions.pop(topic, None)
+def subscription_failed(topic, errcode):
+ if topic == 'me':
+ # Failed 'me' subscription means the bot is disfunctional. Break the loop and retry in a few seconds.
+ client_post(None)
+
+def login_error(unused, errcode):
+ # Check for 409 "already authenticated".
+ if errcode.get('code') != 409:
+ exit(1)
+
def server_version(params):
if params == None:
return
@@ -109,7 +136,7 @@ def client_generate():
msg = queue_out.get()
if msg == None:
return
- # print("out:", msg)
+ print("out: ", json.dumps(MessageToDict(msg), cls=StrLogger))
yield msg
def client_post(msg):
@@ -126,7 +153,7 @@ def client_reset():
def hello():
tid = next_id()
add_future(tid, {
- 'action': lambda unused, params: server_version(params),
+ 'onsuccess': lambda unused, params: server_version(params),
})
return pb.ClientMsg(hi=pb.ClientHi(id=tid, user_agent=APP_NAME + "/" + APP_VERSION + " (" +
platform.system() + "/" + platform.release() + "); gRPC-python/" + LIB_VERSION,
@@ -136,7 +163,8 @@ def login(cookie_file_name, scheme, secret):
tid = next_id()
add_future(tid, {
'arg': cookie_file_name,
- 'action': lambda fname, params: on_login(fname, params),
+ 'onsuccess': lambda fname, params: on_login(fname, params),
+ 'onerror': lambda unused, errcode: login_error(unused, errcode),
})
return pb.ClientMsg(login=pb.ClientLogin(id=tid, scheme=scheme, secret=secret))
@@ -144,7 +172,8 @@ def subscribe(topic):
tid = next_id()
add_future(tid, {
'arg': topic,
- 'action': lambda topicName, unused: add_subscription(topicName),
+ 'onsuccess': lambda topicName, unused: add_subscription(topicName),
+ 'onerror': lambda topicName, errcode: subscription_failed(topicName, errcode),
})
return pb.ClientMsg(sub=pb.ClientSub(id=tid, topic=topic))
@@ -152,7 +181,7 @@ def leave(topic):
tid = next_id()
add_future(tid, {
'arg': topic,
- 'action': lambda topicName, unused: del_subscription(topicName)
+ 'onsuccess': lambda topicName, unused: del_subscription(topicName)
})
return pb.ClientMsg(leave=pb.ClientLeave(id=tid, topic=topic))
@@ -200,7 +229,8 @@ def client_message_loop(stream):
try:
# Read server responses
for msg in stream:
- # print("in:", msg)
+ print("in: ", json.dumps(MessageToDict(msg), cls=StrLogger))
+
if msg.HasField("ctrl"):
# Run code on command completion
exec_future(msg.ctrl.id, msg.ctrl.code, msg.ctrl.text, msg.ctrl.params)
From 2314ccf1719afa85001dc6be8b07f12cb8d6657b Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 29 Mar 2020 13:56:13 +0300
Subject: [PATCH 093/142] check for cluster unreachable error
---
chatbot/python/chatbot.py | 8 ++++++--
1 file changed, 6 insertions(+), 2 deletions(-)
diff --git a/chatbot/python/chatbot.py b/chatbot/python/chatbot.py
index 211153030..243a2c832 100644
--- a/chatbot/python/chatbot.py
+++ b/chatbot/python/chatbot.py
@@ -80,8 +80,12 @@ def del_subscription(topic):
def subscription_failed(topic, errcode):
if topic == 'me':
- # Failed 'me' subscription means the bot is disfunctional. Break the loop and retry in a few seconds.
- client_post(None)
+ # Failed 'me' subscription means the bot is disfunctional.
+ if errcode.get('code') == 502:
+ # Cluster unreachable. Break the loop and retry in a few seconds.
+ client_post(None)
+ else:
+ exit(1)
def login_error(unused, errcode):
# Check for 409 "already authenticated".
From 590e294712f19095febb7a2a33cf55cbaceb848d Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 29 Mar 2020 14:03:56 +0300
Subject: [PATCH 094/142] update grpc port references 6061 -> 16060
---
tn-cli/README.md | 2 +-
tn-cli/tn-cli.py | 4 ++--
2 files changed, 3 insertions(+), 3 deletions(-)
diff --git a/tn-cli/README.md b/tn-cli/README.md
index d296a6782..016a88d3a 100644
--- a/tn-cli/README.md
+++ b/tn-cli/README.md
@@ -22,7 +22,7 @@ where `X.XX.XX` is the version number which must match the server version number
The client takes optional parameters:
- * `--host` is the address of the gRPC server to connect to; default `localhost:6061`.
+ * `--host` is the address of the gRPC server to connect to; default `localhost:16060`.
* `--web-host` is the address of Tinode web server, used for file uploads only; default `localhost:6060`.
* `--ssl` the server requires a secure connection (SSL)
* `--ssl-host` the domain name to use for SNI if different from the `--host` domain name.
diff --git a/tn-cli/tn-cli.py b/tn-cli/tn-cli.py
index 6592db142..2bda8bc3a 100644
--- a/tn-cli/tn-cli.py
+++ b/tn-cli/tn-cli.py
@@ -1032,8 +1032,8 @@ def print_server_params(params):
purpose = "Tinode command line client. Version " + version + "."
parser = argparse.ArgumentParser(description=purpose)
- parser.add_argument('--host', default='localhost:6061', help='address of Tinode gRPC server')
- parser.add_argument('--web-host', default='localhost:6060', help='address of Tinode web server')
+ parser.add_argument('--host', default='localhost:16060', help='address of Tinode gRPC server')
+ parser.add_argument('--web-host', default='localhost:6060', help='address of Tinode web server (for file uploads)')
parser.add_argument('--ssl', action='store_true', help='connect to server over secure connection')
parser.add_argument('--ssl-host', help='SSL host name to use instead of default (useful for connecting to localhost)')
parser.add_argument('--login-basic', help='login using basic authentication username:password')
From 679d4efebde283e7087ad18492a7f1166559107e Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 29 Mar 2020 16:45:05 +0300
Subject: [PATCH 095/142] add timestamp to logs
---
chatbot/python/chatbot.py | 49 +++++++++++++++++++++------------------
1 file changed, 27 insertions(+), 22 deletions(-)
diff --git a/chatbot/python/chatbot.py b/chatbot/python/chatbot.py
index 243a2c832..d70548fb4 100644
--- a/chatbot/python/chatbot.py
+++ b/chatbot/python/chatbot.py
@@ -6,6 +6,7 @@
import argparse
import base64
from concurrent import futures
+from datetime import datetime
import json
import os
import pkg_resources
@@ -42,16 +43,19 @@
# This is needed for gRPC ssl to work correctly.
os.environ["GRPC_SSL_CIPHER_SUITES"] = "HIGH+ECDSA"
+def log(*args):
+ print(datetime.utcnow().strftime('%Y-%m-%d %H:%M:%S.%f')[:-3], *args)
+
# Add bundle for future execution
def add_future(tid, bundle):
onCompletion[tid] = bundle
# Shorten long strings for logging.
-class StrLogger(json.JSONEncoder):
+class JsonHelper(json.JSONEncoder):
def default(self, obj):
if type(obj) == str and len(obj) > MAX_LOG_LEN:
return '<' + len(obj) + ', bytes: ' + obj[:12] + '...' + obj[-12:] + '>'
- return super(StrLogger, self).default(obj)
+ return super(JsonHelper, self).default(obj)
# Resolve or reject the future
def exec_future(tid, code, text, params):
@@ -63,12 +67,12 @@ def exec_future(tid, code, text, params):
arg = bundle.get('arg')
bundle.get('onsuccess')(arg, params)
else:
- print("Error: {} {} ({})".format(code, text, tid))
+ log("Error: {} {} ({})".format(code, text, tid))
onerror = bundle.get('onerror')
if onerror:
onerror(bundle.get('arg'), {'code': code, 'text': text})
except Exception as err:
- print("Error handling server response", err)
+ log("Error handling server response", err)
# List of active subscriptions
subscriptions = {}
@@ -95,7 +99,7 @@ def login_error(unused, errcode):
def server_version(params):
if params == None:
return
- print("Server:", params['build'].decode('ascii'), params['ver'].decode('ascii'))
+ log("Server:", params['build'].decode('ascii'), params['ver'].decode('ascii'))
def next_id():
next_id.tid += 1
@@ -129,7 +133,7 @@ def Account(self, acc_event, context):
else:
action = "unknown"
- print("Account", action, ":", acc_event.user_id, acc_event.public)
+ log("Account", action, ":", acc_event.user_id, acc_event.public)
return pb.Unused()
@@ -140,7 +144,7 @@ def client_generate():
msg = queue_out.get()
if msg == None:
return
- print("out: ", json.dumps(MessageToDict(msg), cls=StrLogger))
+ log("out:", json.dumps(MessageToDict(msg), cls=JsonHelper))
yield msg
def client_post(msg):
@@ -204,12 +208,12 @@ def init_server(listen):
server.add_insecure_port(listen)
server.start()
- print("Plugin server running at '"+listen+"'")
+ log("Plugin server running at '"+listen+"'")
return server
def init_client(addr, schema, secret, cookie_file_name, secure, ssl_host):
- print("Connecting to", "secure" if secure else "", "server at", addr,
+ log("Connecting to", "secure" if secure else "", "server at", addr,
"SNI="+ssl_host if ssl_host else "")
channel = None
@@ -233,14 +237,15 @@ def client_message_loop(stream):
try:
# Read server responses
for msg in stream:
- print("in: ", json.dumps(MessageToDict(msg), cls=StrLogger))
+ log(datetime.utcnow().strftime('%Y-%m-%d %H:%M:%S.%f')[:-3],
+ "in:", json.dumps(MessageToDict(msg), cls=JsonHelper))
if msg.HasField("ctrl"):
# Run code on command completion
exec_future(msg.ctrl.id, msg.ctrl.code, msg.ctrl.text, msg.ctrl.params)
elif msg.HasField("data"):
- # print("message from:", msg.data.from_user_id)
+ # log("message from:", msg.data.from_user_id)
# Protection against the bot talking to self from another session.
if msg.data.from_user_id != botUID:
@@ -253,7 +258,7 @@ def client_message_loop(stream):
client_post(publish(msg.data.topic, next_quote()))
elif msg.HasField("pres"):
- # print("presence:", msg.pres.topic, msg.pres.what)
+ # log("presence:", msg.pres.topic, msg.pres.what)
# Wait for peers to appear online and subscribe to their topics
if msg.pres.topic == 'me':
if (msg.pres.what == pb.ServerPres.ON or msg.pres.what == pb.ServerPres.MSG) \
@@ -267,7 +272,7 @@ def client_message_loop(stream):
pass
except grpc._channel._Rendezvous as err:
- print("Disconnected:", err)
+ log("Disconnected:", err)
def read_auth_cookie(cookie_file_name):
"""Read authentication token from a file"""
@@ -306,7 +311,7 @@ def on_login(cookie_file_name, params):
json.dump(nice, cookie)
cookie.close()
except Exception as err:
- print("Failed to save authentication cookie", err)
+ log("Failed to save authentication cookie", err)
def load_quotes(file_name):
with open(file_name) as f:
@@ -323,25 +328,25 @@ def run(args):
"""Use token to login"""
schema = 'token'
secret = args.login_token.encode('acsii')
- print("Logging in with token", args.login_token)
+ log("Logging in with token", args.login_token)
elif args.login_basic:
"""Use username:password"""
schema = 'basic'
secret = args.login_basic.encode('utf-8')
- print("Logging in with login:password", args.login_basic)
+ log("Logging in with login:password", args.login_basic)
else:
"""Try reading the cookie file"""
try:
schema, secret = read_auth_cookie(args.login_cookie)
- print("Logging in with cookie file", args.login_cookie)
+ log("Logging in with cookie file", args.login_cookie)
except Exception as err:
- print("Failed to read authentication cookie", err)
+ log("Failed to read authentication cookie", err)
if schema:
# Load random quotes from file
- print("Loaded {} quotes".format(load_quotes(args.quotes)))
+ log("Loaded {} quotes".format(load_quotes(args.quotes)))
# Start Plugin server
server = init_server(args.listen)
@@ -351,7 +356,7 @@ def run(args):
# Setup closure for graceful termination
def exit_gracefully(signo, stack_frame):
- print("Terminated with signal", signo)
+ log("Terminated with signal", signo)
server.stop(0)
client.cancel()
sys.exit(0)
@@ -373,7 +378,7 @@ def exit_gracefully(signo, stack_frame):
client.cancel()
else:
- print("Error: authentication scheme not defined")
+ log("Error: authentication scheme not defined")
if __name__ == '__main__':
@@ -381,7 +386,7 @@ def exit_gracefully(signo, stack_frame):
random.seed()
purpose = "Tino, Tinode's chatbot."
- print(purpose)
+ log(purpose)
parser = argparse.ArgumentParser(description=purpose)
parser.add_argument('--host', default='localhost:16060', help='address of Tinode server gRPC endpoint')
parser.add_argument('--ssl', action='store_true', help='use SSL to connect to the server')
From 774980a860e989e1238f63c1493dbd4cc3578f39 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 29 Mar 2020 16:46:06 +0300
Subject: [PATCH 096/142] fix for premature serialization of cluster messages
---
server/cluster.go | 10 +++++-----
server/session.go | 6 ++++++
2 files changed, 11 insertions(+), 5 deletions(-)
diff --git a/server/cluster.go b/server/cluster.go
index b167f6285..08430eb00 100644
--- a/server/cluster.go
+++ b/server/cluster.go
@@ -131,7 +131,8 @@ type ClusterReq struct {
// ClusterResp is a Master to Proxy response message.
type ClusterResp struct {
- Msg []byte
+ // Server message with the response.
+ SrvMsg *ServerComMessage
// Session ID to forward message to, if any.
FromSID string
}
@@ -392,7 +393,7 @@ func (Cluster) Proxy(msg *ClusterResp, unused *bool) error {
// This cluster member received a response from topic owner to be forwarded to a session
// Find appropriate session, send the message to it
if sess := globals.sessionStore.Get(msg.FromSID); sess != nil {
- if !sess.queueOutBytes(msg.Msg) {
+ if !sess.queueOut(msg.SrvMsg) {
log.Println("cluster.Proxy: timeout")
}
} else {
@@ -758,15 +759,14 @@ func (sess *Session) rpcWriteLoop() {
}
// The error is returned if the remote node is down. Which means the remote
// session is also disconnected.
- if err := sess.clnode.respond(&ClusterResp{Msg: msg.([]byte), FromSID: sess.sid}); err != nil {
-
+ if err := sess.clnode.respond(&ClusterResp{SrvMsg: msg.(*ServerComMessage), FromSID: sess.sid}); err != nil {
log.Println("cluster sess.writeRPC: " + err.Error())
return
}
case msg := <-sess.stop:
// Shutdown is requested, don't care if the message is delivered
if msg != nil {
- sess.clnode.respond(&ClusterResp{Msg: msg.([]byte), FromSID: sess.sid})
+ sess.clnode.respond(&ClusterResp{SrvMsg: msg.(*ServerComMessage), FromSID: sess.sid})
}
return
diff --git a/server/session.go b/server/session.go
index 6f5dc3bc1..42169ab0a 100644
--- a/server/session.go
+++ b/server/session.go
@@ -1062,6 +1062,12 @@ func (s *Session) serialize(msg *ServerComMessage) interface{} {
if s.proto == GRPC {
return pbServSerialize(msg)
}
+
+ if s.proto == CLUSTER {
+ // No need to serialize the message to bytes within the cluster.
+ return msg
+ }
+
out, _ := json.Marshal(msg)
return out
}
From 655d4c9f9e5842f79f6693b29d1ab155a1c46e3d Mon Sep 17 00:00:00 2001
From: or-else
Date: Mon, 30 Mar 2020 12:17:31 +0300
Subject: [PATCH 097/142] add gob mappings
---
server/cluster.go | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
diff --git a/server/cluster.go b/server/cluster.go
index 08430eb00..2f4e70071 100644
--- a/server/cluster.go
+++ b/server/cluster.go
@@ -699,6 +699,8 @@ func clusterInit(configString json.RawMessage, self *string) int {
gob.Register([]interface{}{})
gob.Register(map[string]interface{}{})
+ gob.Register(map[string]int)
+ gob.Register(map[string]string)
globals.cluster = &Cluster{
thisNodeName: thisName,
@@ -760,7 +762,7 @@ func (sess *Session) rpcWriteLoop() {
// The error is returned if the remote node is down. Which means the remote
// session is also disconnected.
if err := sess.clnode.respond(&ClusterResp{SrvMsg: msg.(*ServerComMessage), FromSID: sess.sid}); err != nil {
- log.Println("cluster sess.writeRPC: " + err.Error())
+ log.Println("cluster: sess.writeRPC: " + err.Error())
return
}
case msg := <-sess.stop:
From 445027106529f3d3a8bcdfaeac13c310fb6e2180 Mon Sep 17 00:00:00 2001
From: or-else
Date: Mon, 30 Mar 2020 12:18:11 +0300
Subject: [PATCH 098/142] race condition in creating a bucket in a cluster
---
server/media/s3/s3.go | 21 ++++++++++++++++++---
1 file changed, 18 insertions(+), 3 deletions(-)
diff --git a/server/media/s3/s3.go b/server/media/s3/s3.go
index f96e96f0b..2d27d8dd3 100644
--- a/server/media/s3/s3.go
+++ b/server/media/s3/s3.go
@@ -92,13 +92,28 @@ func (ah *awshandler) Init(jsconf string) error {
// Create S3 service client
ah.svc = s3.New(sess)
- // Check if the bucket exists, create one if not.
+ // Check if bucket already exists.
+ _, err = ah.svc.HeadBucket(&s3.HeadBucketInput{Bucket: aws.String(ah.conf.BucketName)})
+ if err == nil {
+ // Bucket exists
+ return nil
+ }
+
+ if aerr, ok := err.(awserr.Error); !ok || aerr.Code() != s3.ErrCodeNoSuchBucket {
+ // Hard error.
+ return err
+ }
+
+ // Bucket does not exist. Create one.
_, err = ah.svc.CreateBucket(&s3.CreateBucketInput{Bucket: aws.String(ah.conf.BucketName)})
if err != nil {
- // Check if bucket already exists or a genuine error.
+ // Check if someone has already created a bucket (possible in a cluster).
if aerr, ok := err.(awserr.Error); ok {
if aerr.Code() == s3.ErrCodeBucketAlreadyExists ||
- aerr.Code() == s3.ErrCodeBucketAlreadyOwnedByYou {
+ aerr.Code() == s3.ErrCodeBucketAlreadyOwnedByYou ||
+ // Someone is already creating this bucket:
+ // OperationAborted: A conflicting conditional operation is currently in progress against this resource.
+ aerr.Code() == "OperationAborted" {
// Clear benign error
err = nil
}
From b8a5500c74af0ffcddbb0d02be8215460d5c17bd Mon Sep 17 00:00:00 2001
From: or-else
Date: Mon, 30 Mar 2020 12:28:40 +0300
Subject: [PATCH 099/142] add reference to google group to github issue
template
---
.github/issue_template.md | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
diff --git a/.github/issue_template.md b/.github/issue_template.md
index 2e08d354c..cbf60e406 100644
--- a/.github/issue_template.md
+++ b/.github/issue_template.md
@@ -1,3 +1,5 @@
+# If you are not reporting a bug or requesting a feature, please post to https://groups.google.com/d/forum/tinode instead.
+
### Subject of the issue
Describe your issue here.
@@ -12,7 +14,7 @@ Describe your issue here.
* platform (Windows, Mac, Linux etc)
* version of tinode server, e.g. `0.15.2-rc3`
* database backend
-
+
#### Client-side
- [ ] TinodeWeb/tinodejs: javascript client
* Browser make and version.
From 00df5718dfc07d5206da93b3b0fd86eb678f62ef Mon Sep 17 00:00:00 2001
From: or-else
Date: Mon, 30 Mar 2020 12:29:03 +0300
Subject: [PATCH 100/142] run-cluster.sh updated to use custom config
---
server/run-cluster.sh | 90 +++++++++++++++++++++++++------------------
1 file changed, 53 insertions(+), 37 deletions(-)
diff --git a/server/run-cluster.sh b/server/run-cluster.sh
index 3b6bf1d5f..22a2b21a6 100755
--- a/server/run-cluster.sh
+++ b/server/run-cluster.sh
@@ -7,44 +7,60 @@ ALL_NODE_NAMES=( one two three )
# Port where the first node will listen for client connections over http
HTTP_BASE_PORT=6080
# Port where the first node will listen for gRPC intra-cluster connections.
-GRPC_BASE_PORT=6090
+GRPC_BASE_PORT=16060
-# Allow for non-default config file to be specifid on the command line like config=file_name
-if [ ! -z "$config" ] ; then
- TINODE_CONF=$config
-else
- TINODE_CONF="./tinode.conf"
-fi
+# Assign command line parameters to variables.
+# for line in $@; do
+# eval "$line"
+#done
-case "$1" in
- start)
- echo 'Running cluster on localhost, ports 6080-6082'
+while [[ $# -gt 0 ]]; do
+ key="$1"
+ shift
+ echo "$key"
+ case "$key" in
+ -c|--config)
+ config=$1
+ shift # value
+ ;;
+ start)
+ if [ ! -z "$config" ] ; then
+ TINODE_CONF=$config
+ else
+ TINODE_CONF="tinode.conf"
+ fi
- HTTP_PORT=$HTTP_BASE_PORT
- GRPC_PORT=$GRPC_BASE_PORT
- for NODE_NAME in "${ALL_NODE_NAMES[@]}"
- do
- # Start the node
- ./server -config=${TINODE_CONF} -cluster_self=$NODE_NAME -listen=:${HTTP_PORT} -grpc_listen=:${GRPC_PORT} &
- # Save PID of the node to a temp file.
- # /var/tmp/ does not requre root access.
- echo $!> "/var/tmp/tinode-${NODE_NAME}.pid"
- # Increment ports for the next node.
- HTTP_PORT=$((HTTP_PORT+1))
- GRPC_PORT=$((GRPC_PORT+1))
- done
- ;;
- stop)
- echo 'Stopping cluster'
+ echo "HTTP ports 6080-6082, gRPC ports 16060-16062, config ${config}"
- for NODE_NAME in "${ALL_NODE_NAMES[@]}"
- do
- # Reda PIDs of running nodes from temp files and kill them.
- kill `cat /var/tmp/tinode-${NODE_NAME}.pid`
- # Clean up: delete temp files.
- rm "/var/tmp/tinode-${NODE_NAME}.pid"
- done
- ;;
- *)
- echo $"Usage: $0 {start|stop} [ config= ]"
-esac
+ HTTP_PORT=$HTTP_BASE_PORT
+ GRPC_PORT=$GRPC_BASE_PORT
+ for NODE_NAME in "${ALL_NODE_NAMES[@]}"
+ do
+ # Start the node
+ ./server -config=${TINODE_CONF} -cluster_self=${NODE_NAME} -listen=:${HTTP_PORT} -grpc_listen=:${GRPC_PORT} &
+ # Save PID of the node to a temp file.
+ # /var/tmp/ does not requre root access.
+ echo $!> "/var/tmp/tinode-${NODE_NAME}.pid"
+ # Increment ports for the next node.
+ HTTP_PORT=$((HTTP_PORT+1))
+ GRPC_PORT=$((GRPC_PORT+1))
+ done
+ exit 0
+ ;;
+ stop)
+ echo 'Stopping cluster'
+
+ for NODE_NAME in "${ALL_NODE_NAMES[@]}"
+ do
+ # Read PIDs of running nodes from temp files and kill them.
+ kill `cat /var/tmp/tinode-${NODE_NAME}.pid`
+ # Clean up: delete temp files.
+ rm "/var/tmp/tinode-${NODE_NAME}.pid"
+ done
+ exit 0
+ ;;
+ *)
+ echo $"Usage: $0 {start|stop} [ --config ]"
+ exit 1
+ esac
+done
From b53c888f76eba085b8d6e0bf1d0bfef758524fcb Mon Sep 17 00:00:00 2001
From: or-else
Date: Mon, 30 Mar 2020 12:29:24 +0300
Subject: [PATCH 101/142] correct gob mapping
---
server/cluster.go | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/server/cluster.go b/server/cluster.go
index 2f4e70071..81032de04 100644
--- a/server/cluster.go
+++ b/server/cluster.go
@@ -699,8 +699,8 @@ func clusterInit(configString json.RawMessage, self *string) int {
gob.Register([]interface{}{})
gob.Register(map[string]interface{}{})
- gob.Register(map[string]int)
- gob.Register(map[string]string)
+ gob.Register(map[string]int{})
+ gob.Register(map[string]string{})
globals.cluster = &Cluster{
thisNodeName: thisName,
From 543883e432fc051a61aca8931e14332391b05a4b Mon Sep 17 00:00:00 2001
From: or-else
Date: Mon, 30 Mar 2020 15:54:08 +0300
Subject: [PATCH 102/142] simplify TNPG & add documentation
---
docs/API.md | 23 ++++++++++++++++------
docs/faq.md | 36 +++++++++++++++++++++++++++++++----
server/push/fcm/payload.go | 4 ++--
server/push/fcm/push_fcm.go | 17 +++++++----------
server/push/tnpg/push_tnpg.go | 19 +++++++++---------
server/tinode.conf | 15 ++++++++++++---
6 files changed, 79 insertions(+), 35 deletions(-)
diff --git a/docs/API.md b/docs/API.md
index 0baddd4e9..45f855104 100644
--- a/docs/API.md
+++ b/docs/API.md
@@ -410,10 +410,6 @@ A user is reported as being online when one or more of user's sessions are attac
An empty `ua=""` _user agent_ is not reported. I.e. if user attaches to `me` with non-empty _user agent_ then does so with an empty one, the change is not reported. An empty _user agent_ may be disallowed in the future.
-## Push Notifications Support
-
-Tinode supports mobile push notifications though compile-time plugins. The channel published by the plugin receives a copy of every data message which was attempted to be delivered. The server supports [Google FCM](https://firebase.google.com/docs/cloud-messaging/) out of the box.
-
## Public and Private Fields
Topics and subscriptions have `public` and `private` fields. Generally, the fields are application-defined. The server does not enforce any particular structure of these fields except for `fnd` topic. At the same time, client software should use the same format for interoperability reasons.
@@ -560,18 +556,33 @@ _Important!_ As a security measure, the client should not send security credenti
## Push Notifications
-Tinode uses compile-time adapters for handling push notifications. The server comes with [Google FCM](https://firebase.google.com/docs/cloud-messaging/) and `stdout` adapters. FCM supports all major mobile platforms except Chinese flavor of Android. Any type of push notifications can be handled by writing an appropriate adapter. The payload of the notification from the FCM adapter is the following:
+Tinode uses compile-time adapters for handling push notifications. The server comes with [Tinode Push Gateway](), [Google FCM](https://firebase.google.com/docs/cloud-messaging/), and `stdout` adapters. Tinode Push Gateway and Google FCM support Android with [Play Services](https://developers.google.com/android/guides/overview) (may not be supported by some Chinese phones), iOS devices and all major web browsers excluding Safari. The `stdout` adapter does not actually send push notifications. It's mostly useful for debugging, testing and logging. Other types of push notifications such as [TPNS](https://intl.cloud.tencent.com/product/tpns) can be handled by writing appropriate adapters.
+
+If you are writing a custom plugin, the notification payload is the following:
```js
{
topic: "grpnG99YhENiQU", // Topic which received the message.
xfrom: "usr2il9suCbuko", // ID of the user who sent the message.
ts: "2019-01-06T18:07:30.038Z", // message timestamp in RFC3339 format.
seq: "1234", // sequential ID of the message (integer value sent as text).
- mime: "text/x-drafty", // message MIME-Type.
+ mime: "text/x-drafty", // optional message MIME-Type.
content: "Lorem ipsum dolor sit amet, consectetur adipisci", // The first 80 characters of the message content as plain text.
}
```
+### Tinode Push Gateway
+
+Tinode Push Gateway (TNPG) is a proprietary Tinode service which sends push notifications on behalf of Tinode. It uses Google FCM on the backend and as such supports the same platforms as FCM. The main advantage of using TNPG over FCM is simplicity of configuration: mobile clients do not need to be recompiled, all is needed is a configuration update on a server.
+
+### Google FCM
+
+[Google FCM](https://firebase.google.com/docs/cloud-messaging/) supports Android with [Play Services](https://developers.google.com/android/guides/overview), iPhone and iPad devices, and all major web browsers excluding Safari. In order to use FCM mobile clients (iOS, Android) must be recompiled with credentials obtained from Google.
+
+### Stdout
+
+The `stdout` adapter is mostly useful for debugging and logging. It writes push payload to `STDOUT` where it can be redirected to file or read by some other process.
+
+
## Messages
A message is a logically associated set of data. Messages are passed as JSON-formatted UTF-8 text.
diff --git a/docs/faq.md b/docs/faq.md
index 05df79e6e..c620fcbb8 100644
--- a/docs/faq.md
+++ b/docs/faq.md
@@ -14,8 +14,36 @@ docker cp name-of-the-container:/var/log/tinode.log ./tinode.log
Alternatively, you can instruct the docker container to save the logs to a directory on the host by mapping a host directory to `/var/log/` in the container. Add `-v /where/to/save/logs:/var/log` to the `docker run` command.
-### Q: How to setup FCM push notifications?
-**A**: Enabling push notifications requires two steps:
+
+### Q: What are the options for enabling push notifications?
+**A**: You can use Tinode Push Gateway (TNPG) or you can use Google FCM:
+ * _Tinode Push Gateway_ requires minimum configuration changes by sending pushes on behalf of Tinode.
+ * _Google FCM_ does not rely on Tinode infrastructure for pushes but requires you to recompile mobile apps (iOS and Android).
+
+### Q: How to setup push notifications with Tinode Push Gateway?
+**A**: Enabling TNPG push notifications requires two steps:
+ * register at console.tinode.co and obtain a TNPG token
+ * configure server with the token
+
+#### Obtain TNPG token
+1. Register at https://console.tinode.co and create an organization.
+2. Get the TPNG token from the _On premise_ section by following the instructions there.
+
+#### Configuring the server
+
+Update the server config [`tinode.conf`](../server/tinode.conf#L384), section `"push"` -> `"name": "tnpg"`:
+```js
+{
+ "enabled": true,
+ "org": "test", // name of the organization you registered at console.tinode.co
+ "token": "SoMe_LonG.RaNDoM-StRiNg.123" // authentication token obtained from console.tinode.co
+}
+```
+Make sure the `fcm` section is disabled `"enabled": false`.
+
+
+### Q: How to setup push notifications with Google FCM?
+**A**: Enabling FCM push notifications requires the following steps:
* enable push sending from the server
* enable receiving pushes in the clients
@@ -30,9 +58,9 @@ Alternatively, you can instruct the docker container to save the logs to a direc
4. Update [TinodeWeb](/tinode/webapp/) config [`firebase-init.js`](https://github.com/tinode/webapp/blob/master/firebase-init.js): update `apiKey`, `messagingSenderId`, `projectId`, `appId`, `messagingVapidKey`. See more info at https://github.com/tinode/webapp/#push_notifications
#### iOS and Android
-1. If you are using an Android client, add `google-services.json` to [Tindroid](/tinode/tindroid/) by following instructions at https://developers.google.com/android/guides/google-services-plugin and recompile the client.
+1. If you are using an Android client, add `google-services.json` to [Tindroid](/tinode/tindroid/) by following instructions at https://developers.google.com/android/guides/google-services-plugin and recompile the client. You may also optionally submit it to Google Play Store.
See more info at https://github.com/tinode/tindroid/#push_notifications
-2. If you are using an iOS client, add `GoogleService-Info.plist` to [Tinodios](/tinode/ios/) by following instructions at https://firebase.google.com/docs/cloud-messaging/ios/client) and recompile the client.
+2. If you are using an iOS client, add `GoogleService-Info.plist` to [Tinodios](/tinode/ios/) by following instructions at https://firebase.google.com/docs/cloud-messaging/ios/client) and recompile the client. You may optionally submit the app to Apple AppStore.
See more info at https://github.com/tinode/ios/#push_notifications
diff --git a/server/push/fcm/payload.go b/server/push/fcm/payload.go
index bf8068ca2..57cb87210 100644
--- a/server/push/fcm/payload.go
+++ b/server/push/fcm/payload.go
@@ -192,7 +192,7 @@ func PrepareNotifications(rcpt *push.Receipt, config *AndroidConfig) []messageDa
}
var titlelc, title, bodylc, body, icon, color string
- if config.Enabled {
+ if config != nil && config.Enabled {
titlelc = config.getTitleLocKey(rcpt.Payload.What)
title = config.getTitle(rcpt.Payload.What)
bodylc = config.getBodyLocKey(rcpt.Payload.What)
@@ -218,7 +218,7 @@ func PrepareNotifications(rcpt *push.Receipt, config *AndroidConfig) []messageDa
msg.Android = &fcm.AndroidConfig{
Priority: "high",
}
- if config.Enabled {
+ if config != nil && config.Enabled {
// When this notification type is included and the app is not in the foreground
// Android won't wake up the app and won't call FirebaseMessagingService:onMessageReceived.
// See dicussion: https://github.com/firebase/quickstart-js/issues/71
diff --git a/server/push/fcm/push_fcm.go b/server/push/fcm/push_fcm.go
index 5c8b0d9df..10f71eca6 100644
--- a/server/push/fcm/push_fcm.go
+++ b/server/push/fcm/push_fcm.go
@@ -22,11 +22,13 @@ import (
var handler Handler
-// Size of the input channel buffer.
-const defaultBuffer = 32
+const (
+ // Size of the input channel buffer.
+ bufferSize = 1024
-// Maximum length of a text message in runes
-const maxMessageLength = 80
+ // Maximum length of a text message in runes
+ maxMessageLength = 80
+)
// Handler represents the push handler; implements push.PushHandler interface.
type Handler struct {
@@ -37,7 +39,6 @@ type Handler struct {
type configType struct {
Enabled bool `json:"enabled"`
- Buffer int `json:"buffer"`
Credentials json.RawMessage `json:"credentials"`
CredentialsFile string `json:"credentials_file"`
TimeToLive uint `json:"time_to_live,omitempty"`
@@ -82,11 +83,7 @@ func (Handler) Init(jsonconf string) error {
return err
}
- if config.Buffer <= 0 {
- config.Buffer = defaultBuffer
- }
-
- handler.input = make(chan *push.Receipt, config.Buffer)
+ handler.input = make(chan *push.Receipt, bufferSize)
handler.stop = make(chan bool, 1)
go func() {
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index 094af43c9..def148273 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -16,6 +16,7 @@ import (
const (
baseTargetAddress = "https://pushgw.tinode.co/push/"
batchSize = 100
+ bufferSize = 1024
)
var handler Handler
@@ -26,11 +27,9 @@ type Handler struct {
}
type configType struct {
- Enabled bool `json:"enabled"`
- Buffer int `json:"buffer"`
- User string `json:"user"`
- AuthToken string `json:"auth_token"`
- Android fcm.AndroidConfig `json:"android,omitempty"`
+ Enabled bool `json:"enabled"`
+ OrgName string `json:"org"`
+ AuthToken string `json:"token"`
}
// Init initializes the handler
@@ -44,11 +43,11 @@ func (Handler) Init(jsonconf string) error {
return nil
}
- if len(config.User) == 0 {
- return errors.New("push.tnpg.user not specified.")
+ if config.OrgName == "" {
+ return errors.New("push.tnpg.org not specified.")
}
- handler.input = make(chan *push.Receipt, config.Buffer)
+ handler.input = make(chan *push.Receipt, bufferSize)
handler.stop = make(chan bool, 1)
go func() {
@@ -70,7 +69,7 @@ func postMessage(body interface{}, config *configType) (int, string, error) {
gz := gzip.NewWriter(buf)
json.NewEncoder(gz).Encode(body)
gz.Close()
- targetAddress := baseTargetAddress + config.User
+ targetAddress := baseTargetAddress + config.OrgName
req, err := http.NewRequest("POST", targetAddress, buf)
if err != nil {
return -1, "", err
@@ -88,7 +87,7 @@ func postMessage(body interface{}, config *configType) (int, string, error) {
}
func sendPushes(rcpt *push.Receipt, config *configType) {
- messages := fcm.PrepareNotifications(rcpt, &config.Android)
+ messages := fcm.PrepareNotifications(rcpt, nil)
if messages == nil {
return
}
diff --git a/server/tinode.conf b/server/tinode.conf
index f3f8fa696..03de86509 100644
--- a/server/tinode.conf
+++ b/server/tinode.conf
@@ -307,9 +307,6 @@
// Disabled. Won't work without the server key anyway. See below.
"enabled": false,
- // Number of notifications to keep before they start to be dropped.
- "buffer": 1024,
-
// Firebase project ID.
"project_id": "your-project-id",
@@ -379,6 +376,18 @@
}
}
}
+ },
+ {
+ // Tinode Push Gateway.
+ "name":"tnpg",
+ "config": {
+ // Disabled. Configure first then enable.
+ "enabled": false,
+ // Name of the organization you registered at console.tinode.co.
+ "org": "test",
+ // Authentication token obtained from console.tinode.co
+ "token": "jwt-security-token-obtained-from-console.tinode.co",
+ }
}
],
From 42bf4a280040b969c593661e29b94ef5ebefa334 Mon Sep 17 00:00:00 2001
From: or-else
Date: Mon, 30 Mar 2020 16:27:24 +0300
Subject: [PATCH 103/142] create post url once
---
docs/faq.md | 4 ++--
server/push/tnpg/push_tnpg.go | 10 ++++++----
2 files changed, 8 insertions(+), 6 deletions(-)
diff --git a/docs/faq.md b/docs/faq.md
index c620fcbb8..253a5d1f9 100644
--- a/docs/faq.md
+++ b/docs/faq.md
@@ -64,10 +64,10 @@ See more info at https://github.com/tinode/tindroid/#push_notifications
See more info at https://github.com/tinode/ios/#push_notifications
-### Q: How can new users be added to Tinode?
+### Q: How to add new users?
**A**: There are three ways to create accounts:
* A user can create a new account using one of the applications (web, Android, iOS).
-* A new account can be created using [tn-cli](../tn-cli/) (`acc` command). The process can be scripted.
+* A new account can be created using [tn-cli](../tn-cli/) (`acc` command or `useradd` macro). The process can be scripted.
* If the user already exists in an external database, the Tinode account can be automatically created on the first login using the [rest authenticator](../server/auth/rest/).
diff --git a/server/push/tnpg/push_tnpg.go b/server/push/tnpg/push_tnpg.go
index def148273..03ce95de4 100644
--- a/server/push/tnpg/push_tnpg.go
+++ b/server/push/tnpg/push_tnpg.go
@@ -22,8 +22,9 @@ const (
var handler Handler
type Handler struct {
- input chan *push.Receipt
- stop chan bool
+ input chan *push.Receipt
+ stop chan bool
+ postUrl string
}
type configType struct {
@@ -47,6 +48,7 @@ func (Handler) Init(jsonconf string) error {
return errors.New("push.tnpg.org not specified.")
}
+ handler.postUrl = baseTargetAddress + config.OrgName
handler.input = make(chan *push.Receipt, bufferSize)
handler.stop = make(chan bool, 1)
@@ -69,8 +71,8 @@ func postMessage(body interface{}, config *configType) (int, string, error) {
gz := gzip.NewWriter(buf)
json.NewEncoder(gz).Encode(body)
gz.Close()
- targetAddress := baseTargetAddress + config.OrgName
- req, err := http.NewRequest("POST", targetAddress, buf)
+
+ req, err := http.NewRequest("POST", handler.postUrl, buf)
if err != nil {
return -1, "", err
}
From 3c0a84004eef898ca8dc883ba85a59e4668fb5de Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 31 Mar 2020 08:50:10 +0300
Subject: [PATCH 104/142] added explicit language to email templates
---
server/templ/email-password-reset-ru.templ | 6 +++---
server/templ/email-validation-ru.templ | 8 ++++----
2 files changed, 7 insertions(+), 7 deletions(-)
diff --git a/server/templ/email-password-reset-ru.templ b/server/templ/email-password-reset-ru.templ
index 731dfefae..411037750 100644
--- a/server/templ/email-password-reset-ru.templ
+++ b/server/templ/email-password-reset-ru.templ
@@ -18,11 +18,11 @@
Вы прислали запрос на изменение пароля для вашего аккаунта Tinode.
Для изменения пароля используйте следующую ссылку. Она действительна в течение 24 часов.
-Кликните для изменения пароля.
+Кликните для изменения пароля.
Если ссылка по какой-то причине не работает, скопируйте следующий URL и вставьте его в адресную строку браузера:
-{{.HostUrl}}#reset?scheme={{.Scheme}}&token={{.Token}}
+{{.HostUrl}}#reset?scheme={{.Scheme}}&token={{.Token}}&hl=RU
{{with .Login}}
@@ -44,7 +44,7 @@
Вы прислали запрос на изменение пароля для вашего аккаунта Tinode ({{.HostUrl}}).
Для изменения пароля используйте следующую ниже. Она действительна в течение 24 часов.
- {{.HostUrl}}#reset?scheme={{.Scheme}}&token={{.Token}}
+ {{.HostUrl}}#reset?scheme={{.Scheme}}&token={{.Token}}&hl=RU
Если ссылка по какой-то вы не можете кликнуть по ссылке выше, скопируйте ее и вставьте его в адресную строку браузера.
diff --git a/server/templ/email-validation-ru.templ b/server/templ/email-validation-ru.templ
index 67875c024..e8aae1c00 100644
--- a/server/templ/email-validation-ru.templ
+++ b/server/templ/email-validation-ru.templ
@@ -16,9 +16,9 @@
Вы получили это сообщение потому, что зарегистрировались в Tinode.
-Кликните здесь чтобы подтвердить
+
Кликните здесь чтобы подтвердить
регистрацию или перейдите по сслыке
-{{.HostUrl}}#cred?method=email
+{{.HostUrl}}#cred?method=email&hl=RU
и введите следующий код:
{{.Code}}
Возможно, вам потребуется ввести логин и пароль.
@@ -37,8 +37,8 @@
Вы получили это сообщение потому, что зарегистрировались в Tinode ({{.HostUrl}}).
-Кликните на {{.HostUrl}}#cred?method=email&code={{.Code}}&token={{.Token}} чтобы подтвердить
-регистрацию или перейдите по сслыке {{.HostUrl}}#cred?what=email">{{.HostUrl}}#cred?method=email
+Кликните на {{.HostUrl}}#cred?method=email&code={{.Code}}&token={{.Token}}&hl=RU чтобы подтвердить
+регистрацию или перейдите по сслыке {{.HostUrl}}#cred?what=email">{{.HostUrl}}#cred?method=email&hl=RU
и введите следующий код:
{{.Code}}
From a136a04093450f1037cb9dc9ec2bf985d29cb62a Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 31 Mar 2020 09:07:22 +0300
Subject: [PATCH 105/142] fix rest auth documentation
---
server/auth/rest/README.md | 11 +++++------
1 file changed, 5 insertions(+), 6 deletions(-)
diff --git a/server/auth/rest/README.md b/server/auth/rest/README.md
index e1b60a9e6..111ff9bb2 100644
--- a/server/auth/rest/README.md
+++ b/server/auth/rest/README.md
@@ -44,13 +44,13 @@ Request and response payloads are formatted as JSON. Some of the request or resp
## Configuration
-Add the following section to the `auth_config` in [tinode.conf](../../tinode.conf):
+Add the following section to the `auth_config` in [tinode.conf](../../tinode.conf):
```js
...
"auth_config": {
...
- "myveryownauth": {
+ "rest": {
// ServerUrl is the URL of the authentication server to call.
"server_url": "http://127.0.0.1:5000/",
// Authentication server is allowed to create new accounts.
@@ -62,13 +62,12 @@ Add the following section to the `auth_config` in [tinode.conf](../../tinode.con
...
},
```
-The name `myveryownauth` is completely arbitrary, but your client has to be configured to use it. If you want to use your
-config **instead** of stock `basic` (login-password) authentication, then add a logical renaming:
+If you want to use your authenticator **instead** of stock `basic` (login-password) authentication, then add a logical renaming:
```js
...
"auth_config": {
- "logical_names": ["myveryownauth:basic", "basic:"],
- "myveryownauth": { ... },
+ "logical_names": ["basic:rest"],
+ "rest": { ... },
...
},
...
From 9833e202ff08061ee6335e7f263c8ac5e382c8cb Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 31 Mar 2020 09:11:45 +0300
Subject: [PATCH 106/142] grammar
---
server/auth/rest/README.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/server/auth/rest/README.md b/server/auth/rest/README.md
index 111ff9bb2..ba88a79a1 100644
--- a/server/auth/rest/README.md
+++ b/server/auth/rest/README.md
@@ -62,7 +62,7 @@ Add the following section to the `auth_config` in [tinode.conf](../../tinode.con
...
},
```
-If you want to use your authenticator **instead** of stock `basic` (login-password) authentication, then add a logical renaming:
+If you want to use your authenticator **instead** of stock `basic` (login-password) authentication, add a logical renaming:
```js
...
"auth_config": {
From 8007b0df39a3d4db80130c8cb8f7bbaab8784086 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 31 Mar 2020 10:20:26 +0300
Subject: [PATCH 107/142] push adapter documentation
---
docs/API.md | 4 ++--
docs/faq.md | 21 +++------------------
server/push/fcm/README.md | 25 +++++++++++++++++++++++++
server/push/stdout/README.md | 4 ++++
server/push/tnpg/README.md | 27 +++++++++++++++++++++++++++
5 files changed, 61 insertions(+), 20 deletions(-)
create mode 100644 server/push/fcm/README.md
create mode 100644 server/push/stdout/README.md
create mode 100644 server/push/tnpg/README.md
diff --git a/docs/API.md b/docs/API.md
index 45f855104..53f6e155c 100644
--- a/docs/API.md
+++ b/docs/API.md
@@ -556,7 +556,7 @@ _Important!_ As a security measure, the client should not send security credenti
## Push Notifications
-Tinode uses compile-time adapters for handling push notifications. The server comes with [Tinode Push Gateway](), [Google FCM](https://firebase.google.com/docs/cloud-messaging/), and `stdout` adapters. Tinode Push Gateway and Google FCM support Android with [Play Services](https://developers.google.com/android/guides/overview) (may not be supported by some Chinese phones), iOS devices and all major web browsers excluding Safari. The `stdout` adapter does not actually send push notifications. It's mostly useful for debugging, testing and logging. Other types of push notifications such as [TPNS](https://intl.cloud.tencent.com/product/tpns) can be handled by writing appropriate adapters.
+Tinode uses compile-time adapters for handling push notifications. The server comes with [Tinode Push Gateway](../server/push/tnpg/), [Google FCM](https://firebase.google.com/docs/cloud-messaging/), and `stdout` adapters. Tinode Push Gateway and Google FCM support Android with [Play Services](https://developers.google.com/android/guides/overview) (may not be supported by some Chinese phones), iOS devices and all major web browsers excluding Safari. The `stdout` adapter does not actually send push notifications. It's mostly useful for debugging, testing and logging. Other types of push notifications such as [TPNS](https://intl.cloud.tencent.com/product/tpns) can be handled by writing appropriate adapters.
If you are writing a custom plugin, the notification payload is the following:
```js
@@ -572,7 +572,7 @@ If you are writing a custom plugin, the notification payload is the following:
### Tinode Push Gateway
-Tinode Push Gateway (TNPG) is a proprietary Tinode service which sends push notifications on behalf of Tinode. It uses Google FCM on the backend and as such supports the same platforms as FCM. The main advantage of using TNPG over FCM is simplicity of configuration: mobile clients do not need to be recompiled, all is needed is a configuration update on a server.
+Tinode Push Gateway (TNPG) is a proprietary Tinode service which sends push notifications on behalf of Tinode. Internally it uses Google FCM and as such supports the same platforms as FCM. The main advantage of using TNPG over FCM is simplicity of configuration: mobile clients do not need to be recompiled, all is needed is a [configuration update](../server/push/tnpg/) on a server.
### Google FCM
diff --git a/docs/faq.md b/docs/faq.md
index 253a5d1f9..02e2ad5f7 100644
--- a/docs/faq.md
+++ b/docs/faq.md
@@ -22,24 +22,9 @@ Alternatively, you can instruct the docker container to save the logs to a direc
### Q: How to setup push notifications with Tinode Push Gateway?
**A**: Enabling TNPG push notifications requires two steps:
- * register at console.tinode.co and obtain a TNPG token
- * configure server with the token
-
-#### Obtain TNPG token
-1. Register at https://console.tinode.co and create an organization.
-2. Get the TPNG token from the _On premise_ section by following the instructions there.
-
-#### Configuring the server
-
-Update the server config [`tinode.conf`](../server/tinode.conf#L384), section `"push"` -> `"name": "tnpg"`:
-```js
-{
- "enabled": true,
- "org": "test", // name of the organization you registered at console.tinode.co
- "token": "SoMe_LonG.RaNDoM-StRiNg.123" // authentication token obtained from console.tinode.co
-}
-```
-Make sure the `fcm` section is disabled `"enabled": false`.
+ * register at console.tinode.co and obtain a TNPG token.
+ * configure server with the token.
+See detailed instructions [here](../server/push/tnpg/).
### Q: How to setup push notifications with Google FCM?
diff --git a/server/push/fcm/README.md b/server/push/fcm/README.md
new file mode 100644
index 000000000..0460895ee
--- /dev/null
+++ b/server/push/fcm/README.md
@@ -0,0 +1,25 @@
+# FCM push adapter
+
+This adapter sends push notifications to mobile clients and web browsers using [Google FCM](https://firebase.google.com/docs/cloud-messaging/). As of the time of this writing it supports Android with [Play Services](https://developers.google.com/android/guides/overview), iOS devices, and all major web browsers [excluding Safari](https://caniuse.com/#feat=push-api).
+
+This adapter requires you to obtain your own credentials from Goole Firebase. If you want to use iOS and Android mobile apps with your service, they must be recompiled with your credentials obtained from Google. If you do not want to recompile mobile clients, consider using TNPG adapter instead.
+
+
+## Configuring FCM adapter
+
+### Server and TinodeWeb
+
+1. Create a project at https://firebase.google.com/ if you have not done so already.
+2. Follow instructions at https://cloud.google.com/iam/docs/creating-managing-service-account-keys to download the credentials file.
+3. Update the server config [`tinode.conf`](../server/tinode.conf#L255), section `"push"` -> `"name": "fcm"`. Do _ONE_ of the following:
+ * _Either_ enter the path to the downloaded credentials file into `"credentials_file"`.
+ * _OR_ copy the file contents to `"credentials"`.
+ Remove the other entry. I.e. if you have updated `"credentials_file"`, remove `"credentials"` and vice versa.
+4. Update [TinodeWeb](/tinode/webapp/) config [`firebase-init.js`](https://github.com/tinode/webapp/blob/master/firebase-init.js): update `apiKey`, `messagingSenderId`, `projectId`, `appId`, `messagingVapidKey`. See more info at https://github.com/tinode/webapp/#push_notifications
+
+### iOS and Android
+
+1. If you are using an Android client, add `google-services.json` to [Tindroid](/tinode/tindroid/) by following instructions at https://developers.google.com/android/guides/google-services-plugin and recompile the client. You may also optionally submit it to Google Play Store.
+See more info at https://github.com/tinode/tindroid/#push_notifications
+2. If you are using an iOS client, add `GoogleService-Info.plist` to [Tinodios](/tinode/ios/) by following instructions at https://firebase.google.com/docs/cloud-messaging/ios/client) and recompile the client. You may optionally submit the app to Apple AppStore.
+See more info at https://github.com/tinode/ios/#push_notifications
diff --git a/server/push/stdout/README.md b/server/push/stdout/README.md
new file mode 100644
index 000000000..c742899d9
--- /dev/null
+++ b/server/push/stdout/README.md
@@ -0,0 +1,4 @@
+# `stdout` push adapter
+
+This is an adapter which logs push notifications to `STDOUT` where they can be redirected to file or processed by some other service.
+This adapter is primarily intended for debugging and logging.
diff --git a/server/push/tnpg/README.md b/server/push/tnpg/README.md
new file mode 100644
index 000000000..b8286c698
--- /dev/null
+++ b/server/push/tnpg/README.md
@@ -0,0 +1,27 @@
+# TNPG: Push Gateway
+
+This is push notifications adapter which communicates with Tinode Push Gateway (TNPG).
+
+TNPG is a proprietary service intended to simplify deployment of on-premise installations.
+Deploying a Tinode server without TNPG requires [configuring Google FCM](../fcm/) with your own credentials including recompiling mobile clients and releasing them to PlayStore and AppStore under your own accounts which is usually time consuming and relatively complex.
+
+TNPG solves this problem by allowing you to send push notifications on behalf of Tinode. Internally it uses Google FCM and as such supports the same platforms as FCM. The main advantage of using TNPG over FCM is simplicity of configuration: mobile clients do not need to be recompiled, all is needed is a configuration update on the server.
+
+## Configuring TNPG adapter
+
+### Obtain TNPG token
+
+1. Register at https://console.tinode.co and create an organization.
+2. Get the TPNG token from the _On premise_ section by following the instructions there.
+
+### Configuring the server
+
+Update the server config [`tinode.conf`](../server/tinode.conf#L384), section `"push"` -> `"name": "tnpg"`:
+```js
+{
+ "enabled": true,
+ "org": "myorg", // name of the organization you registered at console.tinode.co
+ "token": "SoMe_LonG.RaNDoM-StRiNg.12345" // authentication token obtained from console.tinode.co
+}
+```
+Make sure the `fcm` section is disabled `"enabled": false`.
From 4f376a07b1ddd1a4f917218a5aa2a553b5146f6d Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 31 Mar 2020 10:33:01 +0300
Subject: [PATCH 108/142] api TOC and links
---
docs/API.md | 6 ++++--
1 file changed, 4 insertions(+), 2 deletions(-)
diff --git a/docs/API.md b/docs/API.md
index 53f6e155c..3996fe15c 100644
--- a/docs/API.md
+++ b/docs/API.md
@@ -28,7 +28,6 @@
- [`sys` Topic](#sys-topic)
- [Using Server-Issued Message IDs](#using-server-issued-message-ids)
- [User Agent and Presence Notifications](#user-agent-and-presence-notifications)
- - [Push Notifications Support](#push-notifications-support)
- [Public and Private Fields](#public-and-private-fields)
- [Public](#public)
- [Private](#private)
@@ -37,6 +36,9 @@
- [Uploading](#uploading)
- [Downloading](#downloading)
- [Push Notifications](#push-notifications)
+ - [Tinode Push Gateway](#tinode-push-gateway)
+ - [Google FCM](#google-fcm)
+ - [Stdout](#stdout)
- [Messages](#messages)
- [Client to Server Messages](#client-to-server-messages)
- [`{hi}`](#hi)
@@ -576,7 +578,7 @@ Tinode Push Gateway (TNPG) is a proprietary Tinode service which sends push noti
### Google FCM
-[Google FCM](https://firebase.google.com/docs/cloud-messaging/) supports Android with [Play Services](https://developers.google.com/android/guides/overview), iPhone and iPad devices, and all major web browsers excluding Safari. In order to use FCM mobile clients (iOS, Android) must be recompiled with credentials obtained from Google.
+[Google FCM](https://firebase.google.com/docs/cloud-messaging/) supports Android with [Play Services](https://developers.google.com/android/guides/overview), iPhone and iPad devices, and all major web browsers excluding Safari. In order to use FCM mobile clients (iOS, Android) must be recompiled with credentials obtained from Google. See [instructions](../server/push/fcm/) for details.
### Stdout
From 54dfd1918e3fd0ff326e20da43743a0b1f8dff89 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 31 Mar 2020 16:45:15 +0300
Subject: [PATCH 109/142] replace DisposaBoy/JsonConfigReader with
tinode/jsonco
---
go.mod | 2 +-
go.sum | 4 +--
server/db/mongodb/tests/mongo_test.go | 2 +-
server/main.go | 16 +++++-------
server/utils.go | 37 ---------------------------
tinode-db/main.go | 20 ++++++++++++---
6 files changed, 28 insertions(+), 53 deletions(-)
diff --git a/go.mod b/go.mod
index 9c8b84490..5ba3ee849 100644
--- a/go.mod
+++ b/go.mod
@@ -4,7 +4,6 @@ go 1.14
require (
firebase.google.com/go v3.12.0+incompatible
- github.com/DisposaBoy/JsonConfigReader v0.0.0-20171218180944-5ea4d0ddac55
github.com/aws/aws-sdk-go v1.29.29
github.com/bitly/go-hostpool v0.1.0 // indirect
github.com/bmizerany/assert v0.0.0-20160611221934-b7ed37b82869 // indirect
@@ -16,6 +15,7 @@ require (
github.com/jmoiron/sqlx v1.2.0
github.com/prometheus/client_golang v1.5.1
github.com/prometheus/common v0.9.1
+ github.com/tinode/jsonco v1.0.0
github.com/tinode/snowflake v1.0.0
go.mongodb.org/mongo-driver v1.3.1
golang.org/x/crypto v0.0.0-20200320181102-891825fb96df
diff --git a/go.sum b/go.sum
index c012b5340..4cdc9399a 100644
--- a/go.sum
+++ b/go.sum
@@ -5,8 +5,6 @@ cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSR
firebase.google.com/go v3.12.0+incompatible h1:q70KCp/J0oOL8kJ8oV2j3646kV4TB8Y5IvxXC0WT1bo=
firebase.google.com/go v3.12.0+incompatible/go.mod h1:xlah6XbEyW6tbfSklcfe5FHJIwjt8toICdV5Wh9ptHs=
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
-github.com/DisposaBoy/JsonConfigReader v0.0.0-20171218180944-5ea4d0ddac55 h1:jbGlDKdzAZ92NzK65hUP98ri0/r50vVVvmZsFP/nIqo=
-github.com/DisposaBoy/JsonConfigReader v0.0.0-20171218180944-5ea4d0ddac55/go.mod h1:GCzqZQHydohgVLSIqRKZeTt8IGb1Y4NaFfim3H40uUI=
github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
@@ -185,6 +183,8 @@ github.com/stretchr/testify v1.4.0 h1:2E4SXV/wtOkTonXsotYi4li6zVWxYlZuYNCXe9XRJy
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
github.com/tidwall/pretty v1.0.0 h1:HsD+QiTn7sK6flMKIvNmpqz1qrpP3Ps6jOKIKMooyg4=
github.com/tidwall/pretty v1.0.0/go.mod h1:XNkn88O1ChpSDQmQeStsy+sBenx6DDtFZJxhVysOjyk=
+github.com/tinode/jsonco v1.0.0 h1:zVcpjzDvjuA1G+HLrckI5EiiRyq9jgV3x37OQl6e5FE=
+github.com/tinode/jsonco v1.0.0/go.mod h1:Bnavu3302Qfn2pILMNwASkelodgeew3IvDrbdzU84u8=
github.com/tinode/snowflake v1.0.0 h1:YciQ9ZKn1TrnvpS8yZErt044XJaxWVtR9aMO9rOZVOE=
github.com/tinode/snowflake v1.0.0/go.mod h1:5JiaCe3o7QdDeyRcAeZBGVghwRS+ygt2CF/hxmAoptQ=
github.com/xdg/scram v0.0.0-20180814205039-7eeb5667e42c h1:u40Z8hqBAAQyv+vATcGgV0YCnDjqSL7/q/JyPhhJSPk=
diff --git a/server/db/mongodb/tests/mongo_test.go b/server/db/mongodb/tests/mongo_test.go
index acc70c971..4b561209f 100644
--- a/server/db/mongodb/tests/mongo_test.go
+++ b/server/db/mongodb/tests/mongo_test.go
@@ -19,10 +19,10 @@ import (
"testing"
"time"
- jcr "github.com/DisposaBoy/JsonConfigReader"
"github.com/google/go-cmp/cmp"
"github.com/google/go-cmp/cmp/cmpopts"
adapter "github.com/tinode/chat/server/db"
+ jcr "github.com/tinode/jsonco"
b "go.mongodb.org/mongo-driver/bson"
mdb "go.mongodb.org/mongo-driver/mongo"
mdbopts "go.mongodb.org/mongo-driver/mongo/options"
diff --git a/server/main.go b/server/main.go
index fa8876dce..6b1cafbb1 100644
--- a/server/main.go
+++ b/server/main.go
@@ -22,11 +22,11 @@ import (
"strings"
"time"
- // For stripping comments from JSON config
- jcr "github.com/DisposaBoy/JsonConfigReader"
-
gh "github.com/gorilla/handlers"
+ // For stripping comments from JSON config
+ jcr "github.com/tinode/jsonco"
+
// Authenticators
"github.com/tinode/chat/server/auth"
_ "github.com/tinode/chat/server/auth/anon"
@@ -264,17 +264,15 @@ func main() {
if file, err := os.Open(*configfile); err != nil {
log.Fatal("Failed to read config file: ", err)
} else {
- if err = json.NewDecoder(jcr.New(file)).Decode(&config); err != nil {
- // Need to reset file to start in order to convert byte offset to line number and character position.
- // Ignore possible error: can't use it anyway.
- file.Seek(0, 0)
+ jr := jcr.New(file)
+ if err = json.NewDecoder(jr).Decode(&config); err != nil {
switch jerr := err.(type) {
case *json.UnmarshalTypeError:
- lnum, cnum, _ := offsetToLineAndChar(file, jerr.Offset)
+ lnum, cnum, _ := jr.LineAndChar(jerr.Offset)
log.Fatalf("Unmarshall error in config file in %s at %d:%d (offset %d bytes): %s",
jerr.Field, lnum, cnum, jerr.Offset, jerr.Error())
case *json.SyntaxError:
- lnum, cnum, _ := offsetToLineAndChar(file, jerr.Offset)
+ lnum, cnum, _ := jr.LineAndChar(jerr.Offset)
log.Fatalf("Syntax error in config file at %d:%d (offset %d bytes): %s",
lnum, cnum, jerr.Offset, jerr.Error())
default:
diff --git a/server/utils.go b/server/utils.go
index 560f643e4..a002d634c 100644
--- a/server/utils.go
+++ b/server/utils.go
@@ -3,12 +3,10 @@
package main
import (
- "bufio"
"crypto/tls"
"encoding/json"
"errors"
"fmt"
- "io"
"net"
"path/filepath"
"reflect"
@@ -746,41 +744,6 @@ func mergeMaps(dst, src map[string]interface{}) (map[string]interface{}, bool) {
return dst, changed
}
-// Calculate line and character position from byte offset into a file.
-func offsetToLineAndChar(r io.Reader, offset int64) (int, int, error) {
- if offset < 0 {
- return -1, -1, errors.New("offset value cannot be negative")
- }
-
- br := bufio.NewReader(r)
-
- // Count lines and characters.
- lnum := 1
- cnum := 0
- // Number of bytes consumed.
- var count int64
- for {
- ch, size, err := br.ReadRune()
- if err == io.EOF {
- return -1, -1, errors.New("offset value too large")
- }
- count += int64(size)
-
- if ch == '\n' {
- lnum++
- cnum = 0
- } else {
- cnum++
- }
-
- if count >= offset {
- break
- }
- }
-
- return lnum, cnum, nil
-}
-
// netListener creates net.Listener for tcp and unix domains:
// if addr is is in the form "unix:/run/tinode.sock" it's a unix socket, otherwise TCP host:port.
func netListener(addr string) (net.Listener, error) {
diff --git a/tinode-db/main.go b/tinode-db/main.go
index 8ad153f0d..ec003318b 100644
--- a/tinode-db/main.go
+++ b/tinode-db/main.go
@@ -12,11 +12,11 @@ import (
"strings"
"time"
- jcr "github.com/DisposaBoy/JsonConfigReader"
_ "github.com/tinode/chat/server/db/mongodb"
_ "github.com/tinode/chat/server/db/mysql"
_ "github.com/tinode/chat/server/db/rethinkdb"
"github.com/tinode/chat/server/store"
+ jcr "github.com/tinode/jsonco"
)
type configType struct {
@@ -191,8 +191,22 @@ func main() {
var config configType
if file, err := os.Open(*conffile); err != nil {
log.Fatalln("Failed to read config file:", err)
- } else if err = json.NewDecoder(jcr.New(file)).Decode(&config); err != nil {
- log.Fatalln("Failed to parse config file:", err)
+ } else {
+ jr := jcr.New(file)
+ if err = json.NewDecoder(jr).Decode(&config); err != nil {
+ switch jerr := err.(type) {
+ case *json.UnmarshalTypeError:
+ lnum, cnum, _ := jr.LineAndChar(jerr.Offset)
+ log.Fatalf("Unmarshall error in config file in %s at %d:%d (offset %d bytes): %s",
+ jerr.Field, lnum, cnum, jerr.Offset, jerr.Error())
+ case *json.SyntaxError:
+ lnum, cnum, _ := jr.LineAndChar(jerr.Offset)
+ log.Fatalf("Syntax error in config file at %d:%d (offset %d bytes): %s",
+ lnum, cnum, jerr.Offset, jerr.Error())
+ default:
+ log.Fatal("Failed to parse config file: ", err)
+ }
+ }
}
err := store.Open(1, config.StoreConfig)
From af9269f3aaa62254738cf54b71a59db2b4cff983 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 31 Mar 2020 18:16:42 +0300
Subject: [PATCH 110/142] gofmt -s
---
monitoring/exporter/influxdb_exporter.go | 6 +++---
monitoring/exporter/prom_exporter.go | 4 ++--
monitoring/exporter/scraper.go | 4 ++--
server/store/store.go | 2 +-
server/topic.go | 2 +-
5 files changed, 9 insertions(+), 9 deletions(-)
diff --git a/monitoring/exporter/influxdb_exporter.go b/monitoring/exporter/influxdb_exporter.go
index 0d489d365..568172177 100644
--- a/monitoring/exporter/influxdb_exporter.go
+++ b/monitoring/exporter/influxdb_exporter.go
@@ -2,8 +2,8 @@ package main
import (
"bytes"
- "log"
"fmt"
+ "log"
"net/http"
"net/url"
"time"
@@ -20,9 +20,9 @@ type InfluxDBExporter struct {
}
// NewInfluxDBExporter returns an initialized InfluxDB exporter.
-func NewInfluxDBExporter(influxDBVersion, pushBaseAddress, organization, bucket, token, instance string, scraper *Scraper) *InfluxDBExporter{
+func NewInfluxDBExporter(influxDBVersion, pushBaseAddress, organization, bucket, token, instance string, scraper *Scraper) *InfluxDBExporter {
targetAddress := formPushTargetAddress(influxDBVersion, pushBaseAddress, organization, bucket)
- tokenHeader := formAuthorizationHeaderValue(influxDBVersion, token)
+ tokenHeader := formAuthorizationHeaderValue(influxDBVersion, token)
return &InfluxDBExporter{
targetAddress: targetAddress,
organization: organization,
diff --git a/monitoring/exporter/prom_exporter.go b/monitoring/exporter/prom_exporter.go
index 666753521..bb6b2147b 100644
--- a/monitoring/exporter/prom_exporter.go
+++ b/monitoring/exporter/prom_exporter.go
@@ -13,7 +13,7 @@ type PromExporter struct {
timeout time.Duration
namespace string
- scraper *Scraper
+ scraper *Scraper
up *prometheus.Desc
version *prometheus.Desc
@@ -151,7 +151,7 @@ func (e *PromExporter) parseAndUpdate(ch chan<- prometheus.Metric, desc *prometh
return nil
} else {
return err
- }
+ }
}
func firstError(errs ...error) error {
diff --git a/monitoring/exporter/scraper.go b/monitoring/exporter/scraper.go
index 4065a68a9..ebe5a3464 100644
--- a/monitoring/exporter/scraper.go
+++ b/monitoring/exporter/scraper.go
@@ -10,8 +10,8 @@ import (
// Scraper collects metrics from a tinode server.
type Scraper struct {
- address string
- metrics []string
+ address string
+ metrics []string
}
var errKeyNotFound = errors.New("key not found")
diff --git a/server/store/store.go b/server/store/store.go
index d2eb43222..55f3e7ac0 100644
--- a/server/store/store.go
+++ b/server/store/store.go
@@ -352,7 +352,7 @@ func (UsersObjMapper) FindSubs(id types.Uid, required, optional []string) ([]typ
}
allSubs := append(usubs, tsubs...)
- for i, _ := range allSubs {
+ for i := range allSubs {
// Indicate that the returned access modes are not 'N', but rather undefined.
allSubs[i].ModeGiven = types.ModeUnset
allSubs[i].ModeWant = types.ModeUnset
diff --git a/server/topic.go b/server/topic.go
index 7abaec437..803eed962 100644
--- a/server/topic.go
+++ b/server/topic.go
@@ -1943,7 +1943,7 @@ func (t *Topic) replyGetData(sess *Session, asUid types.Uid, id string, req *Msg
// Push the list of messages to the client as {data}.
if messages != nil {
count = len(messages)
- for i, _ := range messages {
+ for i := range messages {
mm := &messages[i]
sess.queueOut(&ServerComMessage{Data: &MsgServerData{
Topic: toriginal,
From 699218304cc68e111cfe07e3395e097ab8b5c850 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 31 Mar 2020 19:08:00 +0300
Subject: [PATCH 111/142] fixes for lint warnings
---
server/auth/rest/auth_rest.go | 2 +-
server/store/types/types.go | 8 ++++++--
2 files changed, 7 insertions(+), 3 deletions(-)
diff --git a/server/auth/rest/auth_rest.go b/server/auth/rest/auth_rest.go
index 7ac2a84d0..359b210a1 100644
--- a/server/auth/rest/auth_rest.go
+++ b/server/auth/rest/auth_rest.go
@@ -1,4 +1,4 @@
-// Package REST provides authentication by calling a separate process over REST API.
+// Package rest provides authentication by calling a separate process over REST API (technically JSON RPC, not REST).
package rest
import (
diff --git a/server/store/types/types.go b/server/store/types/types.go
index 29e769f74..acd3fd604 100644
--- a/server/store/types/types.go
+++ b/server/store/types/types.go
@@ -367,9 +367,13 @@ func (ss StringSlice) Value() (driver.Value, error) {
type ObjState int
const (
- StateOK ObjState = 0
+ // StateOK indicates normal user or topic.
+ StateOK ObjState = 0
+ // StateSuspended indicates suspended user or topic.
StateSuspended ObjState = 10
- StateDeleted ObjState = 20
+ // StateDeleted indicates soft-deleted user or topic.
+ StateDeleted ObjState = 20
+ // StateUndefined indicates state which has not been set explicitly.
StateUndefined ObjState = 30
)
From c27e7d4fa644247ab53b31fcd52c6a678cb59e20 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 31 Mar 2020 21:38:45 +0300
Subject: [PATCH 112/142] removed redundant parenthesis
---
server/topic.go | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/server/topic.go b/server/topic.go
index 803eed962..9a30b30fb 100644
--- a/server/topic.go
+++ b/server/topic.go
@@ -1363,7 +1363,7 @@ func (t *Topic) replyGetDesc(sess *Session, asUid types.Uid, id string, opts *Ms
}
// Check if user requested modified data
- ifUpdated := (opts == nil || opts.IfModifiedSince == nil || opts.IfModifiedSince.Before(t.updated))
+ ifUpdated := opts == nil || opts.IfModifiedSince == nil || opts.IfModifiedSince.Before(t.updated)
desc := &MsgTopicDesc{}
if !ifUpdated {
From a093d402fd01a95879b2088cffb3451b808a4353 Mon Sep 17 00:00:00 2001
From: or-else
Date: Wed, 1 Apr 2020 21:15:04 +0300
Subject: [PATCH 113/142] support for incognito mode
---
server/push/tnpg/README.md | 2 +-
server/topic.go | 33 +++++++++++++++++++++------------
2 files changed, 22 insertions(+), 13 deletions(-)
diff --git a/server/push/tnpg/README.md b/server/push/tnpg/README.md
index b8286c698..542a026e2 100644
--- a/server/push/tnpg/README.md
+++ b/server/push/tnpg/README.md
@@ -14,7 +14,7 @@ TNPG solves this problem by allowing you to send push notifications on behalf of
1. Register at https://console.tinode.co and create an organization.
2. Get the TPNG token from the _On premise_ section by following the instructions there.
-### Configuring the server
+### Configure the server
Update the server config [`tinode.conf`](../server/tinode.conf#L384), section `"push"` -> `"name": "tnpg"`:
```js
diff --git a/server/topic.go b/server/topic.go
index 9a30b30fb..0455876f1 100644
--- a/server/topic.go
+++ b/server/topic.go
@@ -1084,7 +1084,7 @@ func (t *Topic) requestSub(h *Hub, sess *Session, asUid types.Uid, asLvl auth.Le
// If user has not requested a new access mode, provide one by default.
if modeWant == types.ModeUnset {
// If the user has self-banned before, un-self-ban. Otherwise do not make a change.
- if !userData.modeWant.IsJoiner() {
+ if !oldWant.IsJoiner() {
log.Println("No J permissions before")
// Set permissions NO WORSE than default, but possibly better (admin or owner banned himself).
userData.modeWant = userData.modeGiven | t.accessFor(asLvl)
@@ -1137,9 +1137,13 @@ func (t *Topic) requestSub(h *Hub, sess *Session, asUid types.Uid, asLvl auth.Le
}
// If topic is being muted, send "off" notification and disable updates.
- // DO it before applying the new permissions.
+ // Do it before applying the new permissions.
if (oldWant & oldGiven).IsPresencer() && !(userData.modeWant & userData.modeGiven).IsPresencer() {
- t.presSingleUserOffline(asUid, "off+dis", nilPresParams, "", false)
+ if t.cat == types.TopicCatMe {
+ t.presUsersOfInterest("off+dis", t.userAgent)
+ } else {
+ t.presSingleUserOffline(asUid, "off+dis", nilPresParams, "", false)
+ }
}
// Apply changes.
@@ -1401,13 +1405,13 @@ func (t *Topic) replyGetDesc(sess *Session, asUid types.Uid, id string, opts *Ms
Anon: t.accessAnon.String()}
}
- if t.cat != types.TopicCatMe {
- desc.Acs = &MsgAccessMode{
- Want: pud.modeWant.String(),
- Given: pud.modeGiven.String(),
- Mode: (pud.modeGiven & pud.modeWant).String()}
- } else if sess.authLvl == auth.LevelRoot {
- // If 'me' is in memory then user account is invariable not suspended.
+ desc.Acs = &MsgAccessMode{
+ Want: pud.modeWant.String(),
+ Given: pud.modeGiven.String(),
+ Mode: (pud.modeGiven & pud.modeWant).String()}
+
+ if t.cat == types.TopicCatMe && sess.authLvl == auth.LevelRoot {
+ // If 'me' is in memory then user account is invariably not suspended.
desc.State = types.StateOK.String()
}
@@ -2527,8 +2531,13 @@ func (t *Topic) notifySubChange(uid, actor types.Uid, oldWant, oldGiven,
} else if (newWant & newGiven).IsPresencer() && !(oldWant & oldGiven).IsPresencer() {
// Subscription un-muted.
- // Notify subscriber of topic's online status.
- t.presSingleUserOffline(uid, "?unkn+en", nilPresParams, "", false)
+ if t.cat == types.TopicCatMe {
+ // User is visible online now, notify subscribers.
+ t.presUsersOfInterest("on+en", t.userAgent)
+ } else {
+ // Notify subscriber of topic's online status.
+ t.presSingleUserOffline(uid, "?unkn+en", nilPresParams, "", false)
+ }
}
// Notify requester's other sessions that permissions have changed.
From 00cdd35194bf2e0c8246844100ce409e2775fc80 Mon Sep 17 00:00:00 2001
From: or-else
Date: Thu, 2 Apr 2020 09:48:01 +0300
Subject: [PATCH 114/142] incognito mode works as expected
---
server/hdl_longpoll.go | 2 --
server/pres.go | 20 +++++++++++++++++---
server/topic.go | 8 ++++----
3 files changed, 21 insertions(+), 9 deletions(-)
diff --git a/server/hdl_longpoll.go b/server/hdl_longpoll.go
index 7fdeaf50a..926c618fe 100644
--- a/server/hdl_longpoll.go
+++ b/server/hdl_longpoll.go
@@ -91,7 +91,6 @@ func (sess *Session) readOnce(wrt http.ResponseWriter, req *http.Request) (int,
// - if payload exists, process it and close
// - if sid is not empty but there is no session, report an error
func serveLongPoll(wrt http.ResponseWriter, req *http.Request) {
-
now := time.Now().UTC().Round(time.Millisecond)
// Use the lowest common denominator - this is a legacy handler after all (otherwise would use application/json)
@@ -138,7 +137,6 @@ func serveLongPoll(wrt http.ResponseWriter, req *http.Request) {
enc.Encode(pkt)
return
-
}
// Existing session
diff --git a/server/pres.go b/server/pres.go
index 8c2a92d0f..b9d5a55de 100644
--- a/server/pres.go
+++ b/server/pres.go
@@ -198,7 +198,7 @@ func (t *Topic) presProcReq(fromUserID, what string, wantReply bool) string {
}
}
- // log.Println("in-what:", debugWhat, "out-what", what, "from:", fromUserID, "to:", t.name, "reply:", replyAs)
+ // log.Println("out-what", what, "from:", fromUserID, "to:", t.name, "reply:", replyAs)
// If requester's online status has not changed, do not reply, otherwise an endless loop will happen.
// wantReply is needed to ensure unnecessary {pres} is not sent:
@@ -222,11 +222,25 @@ func (t *Topic) presProcReq(fromUserID, what string, wantReply bool) string {
// Case C: user agent change, "ua", ua
// Case D: User updated 'public', "upd"
func (t *Topic) presUsersOfInterest(what, ua string) {
+ parts := strings.Split(what, "+")
+ wantReply := parts[0] == "on"
+ goOffline := len(parts) > 1 && parts[1] == "dis"
+
// Push update to subscriptions
- for topic := range t.perSubs {
+ for topic, psd := range t.perSubs {
globals.hub.route <- &ServerComMessage{
- Pres: &MsgServerPres{Topic: "me", What: what, Src: t.name, UserAgent: ua, WantReply: (what == "on")},
+ Pres: &MsgServerPres{
+ Topic: "me",
+ What: what,
+ Src: t.name,
+ UserAgent: ua,
+ WantReply: wantReply},
rcptto: topic}
+
+ if psd.online && goOffline {
+ psd.online = false
+ t.perSubs[topic] = psd
+ }
}
}
diff --git a/server/topic.go b/server/topic.go
index 0455876f1..ac9334e14 100644
--- a/server/topic.go
+++ b/server/topic.go
@@ -2531,12 +2531,12 @@ func (t *Topic) notifySubChange(uid, actor types.Uid, oldWant, oldGiven,
} else if (newWant & newGiven).IsPresencer() && !(oldWant & oldGiven).IsPresencer() {
// Subscription un-muted.
- if t.cat == types.TopicCatMe {
+ // Notify subscriber of topic's online status.
+ if t.cat == types.TopicCatGrp {
+ t.presSingleUserOffline(uid, "?unkn+en", nilPresParams, "", false)
+ } else if t.cat == types.TopicCatMe {
// User is visible online now, notify subscribers.
t.presUsersOfInterest("on+en", t.userAgent)
- } else {
- // Notify subscriber of topic's online status.
- t.presSingleUserOffline(uid, "?unkn+en", nilPresParams, "", false)
}
}
From 4bde83932a701b60125012c748068d178ea9479d Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 3 Apr 2020 10:14:55 +0300
Subject: [PATCH 115/142] more informative logging, code formatting, readme
restructured
---
monitoring/exporter/README.md | 59 +++++++++++++-----------
monitoring/exporter/influxdb_exporter.go | 27 ++++++++---
monitoring/exporter/main.go | 52 +++++++++++++--------
3 files changed, 87 insertions(+), 51 deletions(-)
diff --git a/monitoring/exporter/README.md b/monitoring/exporter/README.md
index 261fd4226..8a0852727 100644
--- a/monitoring/exporter/README.md
+++ b/monitoring/exporter/README.md
@@ -1,15 +1,17 @@
# Tinode Metric Exporter
-This is a simple service which reads JSON monitoring data exposed by Tinode server using [expvar](https://golang.org/pkg/expvar/) and re-publishes it in other formats.
-Currently, supported are:
-* [Prometheus](https://prometheus.io/) [exporter](https://prometheus.io/docs/instrumenting/exporters/) which exports data in [prometheus format](https://prometheus.io/docs/concepts/data_model/).
-Note that the monitoring service is expected to pull/scrape data from Prometheus exporter.
-* [InfluxDB](https://www.influxdata.com/) [exporter](https://docs.influxdata.com/influxdb/v1.7/tools/api/#write-http-endpoint), on the contrary, pushes data to its target backend.
+This is a simple service which reads JSON monitoring data exposed by Tinode server using [expvar](https://golang.org/pkg/expvar/) and re-publishes it in other formats. Currently the supported formats are:
+
+* [InfluxDB](https://www.influxdata.com/) [exporter](https://docs.influxdata.com/influxdb/v1.7/tools/api/#write-http-endpoint) **pushes** data to its target backend. This is the default mode.
+* [Prometheus](https://prometheus.io/) [exporter](https://prometheus.io/docs/instrumenting/exporters/) which exports data in [prometheus format](https://prometheus.io/docs/concepts/data_model/). The Prometheus monitoring service is expected to **pull/scrape** data from the exporter.
## Usage
-Exporters are expected to run next to (pair with) Tinode servers: one Exporter per one Tinode server, i.e. a single Exporter provides metrics from a single Tinode server.
-Currently, the Exporter is fully configured via command line flags. There are three sets of flags:
+Exporters are intended to run next to (pair with) Tinode servers: one Exporter per one Tinode server, i.e. a single Exporter provides metrics from a single Tinode server.
+
+## Configuration
+
+The exporters are configured by command-line flags:
### Common flags
* `serve_for` specifies which monitoring service the Exporter will gather metrics for; accepted values: `influxdb`, `prometheus`; default: `influxdb`.
@@ -18,11 +20,6 @@ Currently, the Exporter is fully configured via command line flags. There are th
* `instance` is the Exporter instance name (it may be exported to the upstream backend); default: `exporter`.
* `metric_list` is a comma-separated list of metrics to export; default: `Version,LiveTopics,TotalTopics,LiveSessions,ClusterLeader,TotalClusterNodes,LiveClusterNodes,memstats.Alloc`.
-### Prometheus
-* `prom_namespace` is a prefix to use for metrics names. If you are monitoring multiple tinode instances you may want to use different namespaces; default: `tinode`.
-* `prom_metrics_path` is the path under which to expose the metrics for scraping; default: `/metrics`.
-* `prom_timeout` is the Tinode connection timeout in seconds in response to Prometheus scrapes; default: `15`.
-
### InfluxDB
* `influx_push_addr` is the address of InfluxDB target server where the data gets sent; default: `http://localhost:9999/write`.
* `influx_db_version` is the version of InfluxDB (only 1.7 and 2.0 are supported); default: `1.7`.
@@ -31,21 +28,7 @@ Currently, the Exporter is fully configured via command line flags. There are th
* `influx_auth_token` - InfluxDB authentication token; no default value.
* `influx_push_interval` - InfluxDB push interval in seconds; default: `30`.
-## Examples
-Run Prometheus Exporter as
-```
-./exporter \
- --serve_for=prometheus \
- --tinode_addr=http://localhost:6060/stats/expvar \
- --listen_at=:6222 \
- --instance=exp-0 \
- --prom_namespace=tinode \
- --prom_metrics_path=/metrics \
- --prom_timeout=15
-```
-
-This exporter will serve data at path /metrics, on port 6222.
-Once running, configure your Prometheus monitoring installation to collect data from this exporter.
+#### Example
Run InfluxDB Exporter as
```
@@ -62,3 +45,25 @@ Run InfluxDB Exporter as
```
This exporter will push the collected metrics to the specified backend once every 30 seconds.
+
+
+### Prometheus
+* `prom_namespace` is a prefix to use for metrics names. If you are monitoring multiple tinode instances you may want to use different namespaces; default: `tinode`.
+* `prom_metrics_path` is the path under which to expose the metrics for scraping; default: `/metrics`.
+* `prom_timeout` is the Tinode connection timeout in seconds in response to Prometheus scrapes; default: `15`.
+
+#### Example
+Run Prometheus Exporter as
+```
+./exporter \
+ --serve_for=prometheus \
+ --tinode_addr=http://localhost:6060/stats/expvar \
+ --listen_at=:6222 \
+ --instance=exp-0 \
+ --prom_namespace=tinode \
+ --prom_metrics_path=/metrics \
+ --prom_timeout=15
+```
+
+This exporter will serve data at path /metrics, on port 6222.
+Once running, configure your Prometheus monitoring installation to collect data from this exporter.
diff --git a/monitoring/exporter/influxdb_exporter.go b/monitoring/exporter/influxdb_exporter.go
index 568172177..bf657354b 100644
--- a/monitoring/exporter/influxdb_exporter.go
+++ b/monitoring/exporter/influxdb_exporter.go
@@ -3,9 +3,11 @@ package main
import (
"bytes"
"fmt"
+ "io/ioutil"
"log"
"net/http"
"net/url"
+ "strings"
"time"
)
@@ -20,7 +22,9 @@ type InfluxDBExporter struct {
}
// NewInfluxDBExporter returns an initialized InfluxDB exporter.
-func NewInfluxDBExporter(influxDBVersion, pushBaseAddress, organization, bucket, token, instance string, scraper *Scraper) *InfluxDBExporter {
+func NewInfluxDBExporter(influxDBVersion, pushBaseAddress, organization,
+ bucket, token, instance string, scraper *Scraper) *InfluxDBExporter {
+
targetAddress := formPushTargetAddress(influxDBVersion, pushBaseAddress, organization, bucket)
tokenHeader := formAuthorizationHeaderValue(influxDBVersion, token)
return &InfluxDBExporter{
@@ -34,10 +38,10 @@ func NewInfluxDBExporter(influxDBVersion, pushBaseAddress, organization, bucket,
}
// Push scrapes metrics from Tinode server and pushes these metrics to InfluxDB.
-func (e *InfluxDBExporter) Push() (string, error) {
+func (e *InfluxDBExporter) Push() error {
metrics, err := e.scraper.CollectRaw()
if err != nil {
- return "", err
+ return err
}
b := new(bytes.Buffer)
ts := time.Now().UnixNano()
@@ -46,15 +50,26 @@ func (e *InfluxDBExporter) Push() (string, error) {
}
req, err := http.NewRequest("POST", e.targetAddress, b)
if err != nil {
- return "", err
+ return err
}
req.Header.Add("Authorization", e.tokenHeader)
resp, err := http.DefaultClient.Do(req)
if err != nil {
- return "", err
+ return err
}
defer resp.Body.Close()
- return resp.Status, nil
+
+ if resp.StatusCode >= 400 {
+ var body string
+ if rb, err := ioutil.ReadAll(resp.Body); err != nil {
+ body = err.Error()
+ } else {
+ body = strings.TrimSpace(string(rb))
+ }
+
+ return fmt.Errorf("HTTP %s: %s", resp.Status, body)
+ }
+ return nil
}
func formPushTargetAddress(influxDBVersion, baseAddr, organization, bucket string) string {
diff --git a/monitoring/exporter/main.go b/monitoring/exporter/main.go
index c4fe43833..d48da6d21 100644
--- a/monitoring/exporter/main.go
+++ b/monitoring/exporter/main.go
@@ -45,11 +45,17 @@ func main() {
log.Printf("Tinode metrics exporter.")
var (
- serveFor = flag.String("serve_for", "influxdb", "Monitoring service to gather metrics for. Available: influxdb, prometheus.")
- tinodeAddr = flag.String("tinode_addr", "http://localhost:6060/stats/expvar", "Address of the Tinode instance to scrape.")
- listenAt = flag.String("listen_at", ":6222", "Host name and port to listen for incoming requests on.")
- metricList = flag.String("metric_list", "Version,LiveTopics,TotalTopics,LiveSessions,ClusterLeader,TotalClusterNodes,LiveClusterNodes,memstats.Alloc", "Comma-separated list of metrics to scrape and export.")
- instance = flag.String("instance", "exporter", "Exporter instance name.")
+ serveFor = flag.String("serve_for", "influxdb",
+ "Monitoring service to gather metrics for. Available: influxdb, prometheus.")
+ tinodeAddr = flag.String("tinode_addr", "http://localhost:6060/stats/expvar",
+ "Address of the Tinode instance to scrape.")
+ listenAt = flag.String("listen_at", ":6222",
+ "Host name and port to listen for incoming requests on.")
+ metricList = flag.String("metric_list",
+ "Version,LiveTopics,TotalTopics,LiveSessions,ClusterLeader,TotalClusterNodes,LiveClusterNodes,memstats.Alloc",
+ "Comma-separated list of metrics to scrape and export.")
+ instance = flag.String("instance", "exporter",
+ "Exporter instance name.")
// Prometheus-specific arguments.
promNamespace = flag.String("prom_namespace", "tinode", "Prometheus namespace for metrics '_...'")
@@ -57,12 +63,18 @@ func main() {
promTimeout = flag.Int("prom_timeout", 15, "Tinode connection timeout in seconds in response to Prometheus scrapes.")
// InfluxDB-specific arguments.
- influxPushAddr = flag.String("influx_push_addr", "http://localhost:9999/write", "Address of InfluxDB target server where the data gets sent.")
- influxDBVersion = flag.String("influx_db_version", "1.7", "Version of InfluxDB (only 1.7 and 2.0 are supported).")
- influxOrganization = flag.String("influx_organization", "test", "InfluxDB organization to push metrics as.")
- influxBucket = flag.String("influx_bucket", "test", "InfluxDB storage bucket to store data in (used only in InfluxDB 2.0).")
- influxAuthToken = flag.String("influx_auth_token", "", "InfluxDB authentication token.")
- influxPushInterval = flag.Int("influx_push_interval", 30, "InfluxDB push interval in seconds.")
+ influxPushAddr = flag.String("influx_push_addr", "http://localhost:9999/write",
+ "Address of InfluxDB target server where the data gets sent.")
+ influxDBVersion = flag.String("influx_db_version", "1.7",
+ "Version of InfluxDB (only 1.7 and 2.0 are supported).")
+ influxOrganization = flag.String("influx_organization", "test",
+ "InfluxDB organization to push metrics as.")
+ influxBucket = flag.String("influx_bucket", "test",
+ "InfluxDB storage bucket to store data in (used only in InfluxDB 2.0).")
+ influxAuthToken = flag.String("influx_auth_token", "",
+ "InfluxDB authentication token.")
+ influxPushInterval = flag.Int("influx_push_interval", 30,
+ "InfluxDB push interval in seconds.")
)
flag.Parse()
@@ -91,7 +103,7 @@ func main() {
log.Fatal("Must specify --influx_bucket")
}
if *influxDBVersion != "1.7" && *influxDBVersion != "2.0" {
- log.Fatal("Please, set --influx_db_version to either 1.7 or 2.0")
+ log.Fatal("The --influx_db_version must be either 1.7 or 2.0")
}
if *influxPushInterval > 0 && *influxPushInterval < minPushInterval {
*influxPushInterval = minPushInterval
@@ -118,6 +130,9 @@ func main() {
})
metrics := strings.Split(*metricList, ",")
+ for i, m := range metrics {
+ metrics[i] = strings.TrimSpace(m)
+ }
scraper := Scraper{address: *tinodeAddr, metrics: metrics}
var serverTypeString string
// Create exporters.
@@ -141,15 +156,16 @@ func main() {
)
case InfluxDB:
serverTypeString = fmt.Sprintf("%s, version %s", *serveFor, *influxDBVersion)
- influxDBExporter := NewInfluxDBExporter(*influxDBVersion, *influxPushAddr, *influxOrganization, *influxBucket, *influxAuthToken, *instance, &scraper)
+ influxDBExporter := NewInfluxDBExporter(*influxDBVersion, *influxPushAddr, *influxOrganization, *influxBucket,
+ *influxAuthToken, *instance, &scraper)
if *influxPushInterval > 0 {
go func() {
interval := time.Duration(*influxPushInterval) * time.Second
ch := time.Tick(interval)
for {
if _, ok := <-ch; ok {
- if status, err := influxDBExporter.Push(); err != nil {
- log.Printf("InfluxDB push failed (status '%s'), error: %s", status, err.Error())
+ if err := influxDBExporter.Push(); err != nil {
+ log.Println("InfluxDB push failed:", err)
}
} else {
return
@@ -162,10 +178,10 @@ func main() {
// Forces a data push.
http.HandleFunc("/push", func(w http.ResponseWriter, r *http.Request) {
var msg string
- if http_status, err := influxDBExporter.Push(); err == nil {
- msg = "ok - " + http_status
+ if err := influxDBExporter.Push(); err == nil {
+ msg = "HTTP 200 OK"
} else {
- msg = "fail - " + err.Error()
+ msg = err.Error()
}
w.Write([]byte(`Tinode Push
From a4369009cd606831748602663afb4a9ed59da467 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 5 Apr 2020 14:13:11 +0300
Subject: [PATCH 116/142] fix for #409 and probably for #400
---
server/validate/email/validate.go | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/server/validate/email/validate.go b/server/validate/email/validate.go
index 94c489063..60f6480e7 100644
--- a/server/validate/email/validate.go
+++ b/server/validate/email/validate.go
@@ -173,7 +173,7 @@ func (v *validator) Init(jsonconf string) error {
// Optionally resolve paths against the location of this executable file.
v.ValidationTemplFile = resolveTemplatePath(v.ValidationTemplFile)
- v.ResetTemplFile = resolveTemplatePath(v.ValidationTemplFile)
+ v.ResetTemplFile = resolveTemplatePath(v.ResetTemplFile)
// Paths to templates could be templates themselves: they may be language-dependent.
var validationPathTempl, resetPathTempl *textt.Template
From f9a0ae8921c10fbe1433fba967558b1451196ada Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 5 Apr 2020 20:37:30 +0300
Subject: [PATCH 117/142] adding tpng to docker
---
docker/tinode/Dockerfile | 12 +++++++++---
docker/tinode/config.template | 5 ++---
docker/tinode/entrypoint.sh | 6 +++++-
server/tinode.conf | 2 +-
4 files changed, 17 insertions(+), 8 deletions(-)
diff --git a/docker/tinode/Dockerfile b/docker/tinode/Dockerfile
index 147b33c2b..738189391 100644
--- a/docker/tinode/Dockerfile
+++ b/docker/tinode/Dockerfile
@@ -1,6 +1,6 @@
# Docker file builds an image with a tinode chat server.
#
-# In order to run the image you have to link it to a running database container. For example, to
+# In order to run the image you have to link it to a running database container. For example, to
# to use RethinkDB (named 'rethinkdb') and map the port where the tinode server accepts connections:
#
# $ docker run -p 6060:6060 -d --link rethinkdb \
@@ -83,6 +83,12 @@ ENV TLS_ENABLED=false
# Disable push notifications by default.
ENV FCM_PUSH_ENABLED=false
+# Declare FCM-related vars
+ENV FCM_API_KEY=
+ENV FCM_APP_ID=
+ENV FCM_SENDER_ID=
+ENV FCM_PROJECT_ID=
+ENV FCM_VAPID_KEY=
# Enable Android-specific notifications by default.
ENV FCM_INCLUDE_ANDROID_NOTIFICATION=true
@@ -93,8 +99,8 @@ ENV TNPG_PUSH_ENABLED=false
# Tinode Push Gateway authentication token.
ENV TNPG_AUTH_TOKEN=
-# Tinode Push Gateway user name.
-ENV TNPG_USER=
+# Tinode Push Gateway organization name as registered at console.tinode.co
+ENV TNPG_ORG=
# Use the target db by default.
# When TARGET_DB is "alldbs", it is the user's responsibility
diff --git a/docker/tinode/config.template b/docker/tinode/config.template
index 76c66535c..498102f3f 100644
--- a/docker/tinode/config.template
+++ b/docker/tinode/config.template
@@ -109,15 +109,14 @@
"name":"tnpg",
"config": {
"enabled": $TNPG_PUSH_ENABLED,
- "auth_token": "$TNPG_AUTH_TOKEN",
- "user": "$TNPG_USER"
+ "token": "$TNPG_AUTH_TOKEN",
+ "org": "$TNPG_USER"
}
},
{
"name":"fcm",
"config": {
"enabled": $FCM_PUSH_ENABLED,
- "buffer": 1024,
"project_id": "$FCM_PROJECT_ID",
"credentials_file": "$FCM_CRED_FILE",
"time_to_live": 3600,
diff --git a/docker/tinode/entrypoint.sh b/docker/tinode/entrypoint.sh
index 0fd201dbb..fdd970ba2 100644
--- a/docker/tinode/entrypoint.sh
+++ b/docker/tinode/entrypoint.sh
@@ -35,6 +35,10 @@ else
FCM_PUSH_ENABLED=true
fi
+ if [ ! -z "$TNPG_AUTH_TOKEN" ] ; then
+ TNPG_PUSH_ENABLED=true
+ fi
+
# Generate a new 'working.config' from template and environment
while IFS='' read -r line || [[ -n $line ]] ; do
while [[ "$line" =~ (\$[A-Z_][A-Z_0-9]*) ]] ; do
@@ -60,7 +64,7 @@ if [ "$UPGRADE_DB" = "true" ] ; then
fi
# If push notifications are enabled, generate client-side firebase config file.
-if [ ! -z "$FCM_PUSH_ENABLED" ] ; then
+if [ ! -z "$FCM_PUSH_ENABLED" ] || [ ! -z "$TNPG_PUSH_ENABLED" ] ; then
# Write client config to $STATIC_DIR/firebase-init.js
cat > $STATIC_DIR/firebase-init.js <<- EOM
const FIREBASE_INIT = {
diff --git a/server/tinode.conf b/server/tinode.conf
index 03de86509..c0a8a0925 100644
--- a/server/tinode.conf
+++ b/server/tinode.conf
@@ -383,7 +383,7 @@
"config": {
// Disabled. Configure first then enable.
"enabled": false,
- // Name of the organization you registered at console.tinode.co.
+ // Name of the organization you registered at console.tinode.co.
"org": "test",
// Authentication token obtained from console.tinode.co
"token": "jwt-security-token-obtained-from-console.tinode.co",
From 785be0323dcee545c33b390384af7f01e8f9f440 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 7 Apr 2020 09:06:21 +0300
Subject: [PATCH 118/142] update docs per #412
---
docs/API.md | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
diff --git a/docs/API.md b/docs/API.md
index 3996fe15c..a59139e7a 100644
--- a/docs/API.md
+++ b/docs/API.md
@@ -628,6 +628,8 @@ The user agent `ua` is expected to follow [RFC 7231 section 5.5.3](http://tools.
Message `{acc}` creates users or updates `tags` or authentication credentials `scheme` and `secret` of exiting users. To create a new user set `user` to the string `new` optionally followed by any character sequence, e.g. `newr15gsr`. Either authenticated or anonymous session can send an `{acc}` message to create a new user. To update authentication data or validate a credential of the current user leave `user` unset.
+The `{acc}` message **cannot** be used to modify `desc` of an existing user. Update user's `me` topic instead.
+
```js
acc: {
id: "1a2b3", // string, client-provided message id, optional
@@ -656,7 +658,7 @@ acc: {
],
desc: { // object, user initialisation data closely matching that of table
- // initialisation; optional
+ // initialisation; used only when creating an account; optional
defacs: {
auth: "JRWS", // string, default access mode for peer to peer conversations
// between this user and other authenticated users
From f7c8b0ff2b785c93d9a01d6a3aa1f5fea576e5ab Mon Sep 17 00:00:00 2001
From: or-else
Date: Wed, 8 Apr 2020 09:05:53 +0300
Subject: [PATCH 119/142] minor doc correction
---
docs/API.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/docs/API.md b/docs/API.md
index a59139e7a..42770409f 100644
--- a/docs/API.md
+++ b/docs/API.md
@@ -628,7 +628,7 @@ The user agent `ua` is expected to follow [RFC 7231 section 5.5.3](http://tools.
Message `{acc}` creates users or updates `tags` or authentication credentials `scheme` and `secret` of exiting users. To create a new user set `user` to the string `new` optionally followed by any character sequence, e.g. `newr15gsr`. Either authenticated or anonymous session can send an `{acc}` message to create a new user. To update authentication data or validate a credential of the current user leave `user` unset.
-The `{acc}` message **cannot** be used to modify `desc` of an existing user. Update user's `me` topic instead.
+The `{acc}` message **cannot** be used to modify `desc` or `cred` of an existing user. Update user's `me` topic instead.
```js
acc: {
From e0d4b17f84ddc45f7df7bb9dea50eaf555a0aaec Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 10 Apr 2020 18:46:05 +0300
Subject: [PATCH 120/142] workaroud for rethinkdb/rethinkdb-go/issues/486; a
few other bug fixes
---
server/db/rethinkdb/adapter.go | 14 +++++----
server/topic.go | 16 +++++++---
server/user.go | 53 ++++++++++++++++++++++++----------
3 files changed, 59 insertions(+), 24 deletions(-)
diff --git a/server/db/rethinkdb/adapter.go b/server/db/rethinkdb/adapter.go
index 8cb0cf953..1a992c353 100644
--- a/server/db/rethinkdb/adapter.go
+++ b/server/db/rethinkdb/adapter.go
@@ -889,16 +889,20 @@ func (a *adapter) UserUpdateTags(uid t.Uid, add, remove, reset []string) ([]stri
return nil, err
}
- // Get the new tags
- cursor, err := q.Field("Tags").Run(a.conn)
+ // Get the new tags.
+ // Using Pluck instead of Field because of https://github.com/rethinkdb/rethinkdb-go/issues/486
+ cursor, err := q.Pluck("Tags").Run(a.conn)
if err != nil {
return nil, err
}
defer cursor.Close()
- var tags []string
- err = cursor.One(&tags)
- return tags, err
+ var tagsField struct{ Tags []string }
+ err = cursor.One(&tagsField)
+ if err != nil {
+ return nil, err
+ }
+ return tagsField.Tags, nil
}
// UserGetByCred returns user ID for the given validated credential.
diff --git a/server/topic.go b/server/topic.go
index ac9334e14..eebe201d1 100644
--- a/server/topic.go
+++ b/server/topic.go
@@ -2104,7 +2104,7 @@ func (t *Topic) replySetCred(sess *Session, asUid types.Uid, authLevel auth.Leve
_, tags, err = addCreds(asUid, creds, nil, sess.lang, tmpToken)
}
- if err == nil && len(tags) > 0 {
+ if tags != nil {
t.tags = tags
t.presSubsOnline("tags", "", nilPresParams, nilPresFilters, "")
}
@@ -2287,11 +2287,19 @@ func (t *Topic) replyDelCred(h *Hub, sess *Session, asUid types.Uid, authLvl aut
sess.queueOut(ErrPermissionDenied(del.Id, t.original(asUid), now))
return errors.New("del.cred: invalid topic category")
}
+ if del.Cred == nil || del.Cred.Method == "" {
+ sess.queueOut(ErrMalformed(del.Id, t.original(asUid), now))
+ return errors.New("del.cred: missing method")
+ }
tags, err := deleteCred(asUid, authLvl, del.Cred)
- if err == nil {
- t.tags = tags
- t.presSubsOnline("tags", "", nilPresParams, nilPresFilters, "")
+ if tags != nil {
+ // Check if anything has been actuallt removed.
+ _, removed := stringSliceDelta(t.tags, tags)
+ if len(removed) > 0 {
+ t.tags = tags
+ t.presSubsOnline("tags", "", nilPresParams, nilPresFilters, "")
+ }
}
sess.queueOut(decodeStoreError(err, del.Id, del.Topic, now, nil))
return err
diff --git a/server/user.go b/server/user.go
index 19c5e4f3c..651eb2fcf 100644
--- a/server/user.go
+++ b/server/user.go
@@ -311,8 +311,10 @@ func updateUserAuth(msg *ClientComMessage, user *types.User, rec *auth.Rec) erro
}
// Tags may have been changed by authhdl.UpdateRecord, reset them.
- // Can't do much with the error here, so ignoring it.
- store.Users.UpdateTags(user.Uid(), nil, nil, rec.Tags)
+ // Can't do much with the error here, logging it but not returning.
+ if _, err = store.Users.UpdateTags(user.Uid(), nil, nil, rec.Tags); err != nil {
+ log.Println("updateUserAuth tags update failed:", err)
+ }
return nil
}
@@ -322,9 +324,8 @@ func updateUserAuth(msg *ClientComMessage, user *types.User, rec *auth.Rec) erro
// addCreds adds new credentials and re-send validation request for existing ones. It also adds credential-defined
// tags if necessary.
-// Returns methods validated in this call only, full set of tags.
-func addCreds(uid types.Uid, creds []MsgCredClient, tags []string, lang string,
- tmpToken []byte) ([]string, []string, error) {
+// Returns methods validated in this call only. Returns either a full set of tags or nil for tags when tags are unchanged.
+func addCreds(uid types.Uid, creds []MsgCredClient, extraTags []string, lang string, tmpToken []byte) ([]string, []string, error) {
var validated []string
for i := range creds {
cr := &creds[i]
@@ -346,20 +347,27 @@ func addCreds(uid types.Uid, creds []MsgCredClient, tags []string, lang string,
// Generate tags for these confirmed credentials.
if globals.validators[cr.Method].addToTags {
- tags = append(tags, cr.Method+":"+cr.Value)
+ extraTags = append(extraTags, cr.Method+":"+cr.Value)
}
}
}
// Save tags potentially changed by the validator.
- if len(tags) > 0 {
- tags, _ = store.Users.UpdateTags(uid, tags, nil, nil)
+ if len(extraTags) > 0 {
+ if utags, err := store.Users.UpdateTags(uid, extraTags, nil, nil); err != nil {
+ extraTags = utags
+ } else {
+ log.Println("add cred tags update failed:", err)
+ }
+ } else {
+ extraTags = nil
}
- return validated, tags, nil
+ return validated, extraTags, nil
}
// validatedCreds returns the list of validated credentials including those validated in this call.
-// Returns all validated methods including those validated earlier and now, full set of tags.
+// Returns all validated methods including those validated earlier and now.
+// Returns either a full set of tags or nil for tags if tags are unchanged.
func validatedCreds(uid types.Uid, authLvl auth.Level, creds []MsgCredClient, errorOnFail bool) ([]string, []string, error) {
// Check if credential validation is required.
@@ -417,7 +425,14 @@ func validatedCreds(uid types.Uid, authLvl auth.Level, creds []MsgCredClient, er
var tags []string
if len(tagsToAdd) > 0 {
// Save update to tags
- tags, _ = store.Users.UpdateTags(uid, tagsToAdd, nil, nil)
+ if utags, err := store.Users.UpdateTags(uid, tagsToAdd, nil, nil); err == nil {
+ tags = utags
+ } else {
+ log.Println("validated creds tags update failed:", err)
+ tags = nil
+ }
+ } else {
+ tags = nil
}
var validated []string
@@ -429,6 +444,7 @@ func validatedCreds(uid types.Uid, authLvl auth.Level, creds []MsgCredClient, er
}
// deleteCred deletes user's credential.
+// Returns full set of remaining tags or nil if tags are unchanged.
func deleteCred(uid types.Uid, authLvl auth.Level, cred *MsgCredClient) ([]string, error) {
vld := store.GetValidator(cred.Method)
if vld == nil || cred.Value == "" {
@@ -447,7 +463,6 @@ func deleteCred(uid types.Uid, authLvl auth.Level, cred *MsgCredClient) ([]strin
// If credential is required, make sure the method remains validated even after this credential is deleted.
if isRequired {
-
// There could be multiple validated credentials for the same method thus we are getting a map with count
// for each method.
@@ -457,10 +472,10 @@ func deleteCred(uid types.Uid, authLvl auth.Level, cred *MsgCredClient) ([]strin
return nil, err
}
- // Check if it's OK to delete: there is another validated value.
+ // Check if it's OK to delete: there is another validated value or this value is not validated in the first place.
var okTodelete bool
for _, cr := range allCreds {
- if cr.Done && cr.Value != cred.Value {
+ if (cr.Done && cr.Value != cred.Value) || (!cr.Done && cr.Value == cred.Value) {
okTodelete = true
break
}
@@ -482,8 +497,16 @@ func deleteCred(uid types.Uid, authLvl auth.Level, cred *MsgCredClient) ([]strin
var tags []string
if globals.validators[cred.Method].addToTags {
// This error should not be returned to user.
- tags, _ = store.Users.UpdateTags(uid, nil, []string{cred.Method + ":" + cred.Value}, nil)
+ if utags, err := store.Users.UpdateTags(uid, nil, []string{cred.Method + ":" + cred.Value}, nil); err == nil {
+ tags = utags
+ } else {
+ log.Println("delete cred: failed to update tags:", err)
+ tags = nil
+ }
+ } else {
+ tags = nil
}
+
return tags, nil
}
From 16313e8998c4bac3082541a4917b93b859650c43 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sun, 12 Apr 2020 18:34:46 +0300
Subject: [PATCH 121/142] typo in comment
---
server/topic.go | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/server/topic.go b/server/topic.go
index eebe201d1..1758e98c1 100644
--- a/server/topic.go
+++ b/server/topic.go
@@ -2294,7 +2294,7 @@ func (t *Topic) replyDelCred(h *Hub, sess *Session, asUid types.Uid, authLvl aut
tags, err := deleteCred(asUid, authLvl, del.Cred)
if tags != nil {
- // Check if anything has been actuallt removed.
+ // Check if anything has been actually removed.
_, removed := stringSliceDelta(t.tags, tags)
if len(removed) > 0 {
t.tags = tags
From 31f0c29108afb83578887d9e5d70ba3a1dbf68f3 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 14 Apr 2020 10:53:41 +0300
Subject: [PATCH 122/142] allow "acs" notifications even if topic is muted
---
server/topic.go | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/server/topic.go b/server/topic.go
index 1758e98c1..47a93b486 100644
--- a/server/topic.go
+++ b/server/topic.go
@@ -447,8 +447,8 @@ func (t *Topic) run(hub *Hub) {
// Check presence filters
pud := t.perUser[pssd.uid]
- // Send "gone" notification even if the topic is muted.
- if (!(pud.modeGiven & pud.modeWant).IsPresencer() && msg.Pres.What != "gone") ||
+ // Send "gone" and "acs" notifications even if the topic is muted.
+ if (!(pud.modeGiven & pud.modeWant).IsPresencer() && msg.Pres.What != "gone" && msg.Pres.What != "acs") ||
(msg.Pres.FilterIn != 0 && int(pud.modeGiven&pud.modeWant)&msg.Pres.FilterIn == 0) ||
(msg.Pres.FilterOut != 0 && int(pud.modeGiven&pud.modeWant)&msg.Pres.FilterOut != 0) {
continue
From d047cd6a0049e4ec6d92e33aa67a5853cb2a43ca Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 14 Apr 2020 14:08:04 +0300
Subject: [PATCH 123/142] missing state assignment in rdb and mongo adapters
---
server/db/mongodb/adapter.go | 6 ++++--
server/db/rethinkdb/adapter.go | 4 +++-
2 files changed, 7 insertions(+), 3 deletions(-)
diff --git a/server/db/mongodb/adapter.go b/server/db/mongodb/adapter.go
index dcbdfdb15..163256262 100644
--- a/server/db/mongodb/adapter.go
+++ b/server/db/mongodb/adapter.go
@@ -1228,6 +1228,7 @@ func (a *adapter) TopicsForUser(uid t.Uid, keepDeleted bool, opts *t.QueryOpt) (
} else {
topq = append(topq, sub.Topic)
}
+ sub.Private = unmarshalBsonD(sub.Private)
join[sub.Topic] = sub
}
cur.Close(a.ctx)
@@ -1251,9 +1252,9 @@ func (a *adapter) TopicsForUser(uid t.Uid, keepDeleted bool, opts *t.QueryOpt) (
}
sub = join[top.Id]
sub.ObjHeader.MergeTimes(&top.ObjHeader)
- sub.SetSeqId(top.SeqId)
+ sub.SetState(top.State)
sub.SetTouchedAt(top.TouchedAt)
- sub.Private = unmarshalBsonD(sub.Private)
+ sub.SetSeqId(top.SeqId)
if t.GetTopicCat(sub.Topic) == t.TopicCatGrp {
// all done with a grp topic
sub.SetPublic(unmarshalBsonD(top.Public))
@@ -1286,6 +1287,7 @@ func (a *adapter) TopicsForUser(uid t.Uid, keepDeleted bool, opts *t.QueryOpt) (
uid2 := t.ParseUid(usr.Id)
if sub, ok := join[uid.P2PName(uid2)]; ok {
sub.ObjHeader.MergeTimes(&usr.ObjHeader)
+ sub.SetState(usr.State)
sub.SetPublic(unmarshalBsonD(usr.Public))
sub.SetWith(uid2.UserId())
sub.SetDefaultAccess(usr.Access.Auth, usr.Access.Anon)
diff --git a/server/db/rethinkdb/adapter.go b/server/db/rethinkdb/adapter.go
index 1a992c353..fbc65ce27 100644
--- a/server/db/rethinkdb/adapter.go
+++ b/server/db/rethinkdb/adapter.go
@@ -1112,8 +1112,9 @@ func (a *adapter) TopicsForUser(uid t.Uid, keepDeleted bool, opts *t.QueryOpt) (
for cursor.Next(&top) {
sub = join[top.Id]
sub.ObjHeader.MergeTimes(&top.ObjHeader)
- sub.SetSeqId(top.SeqId)
+ sub.SetState(top.State)
sub.SetTouchedAt(top.TouchedAt)
+ sub.SetSeqId(top.SeqId)
if t.GetTopicCat(sub.Topic) == t.TopicCatGrp {
// all done with a grp topic
sub.SetPublic(top.Public)
@@ -1142,6 +1143,7 @@ func (a *adapter) TopicsForUser(uid t.Uid, keepDeleted bool, opts *t.QueryOpt) (
uid2 := t.ParseUid(usr.Id)
if sub, ok := join[uid.P2PName(uid2)]; ok {
sub.ObjHeader.MergeTimes(&usr.ObjHeader)
+ sub.SetState(usr.State)
sub.SetPublic(usr.Public)
sub.SetWith(uid2.UserId())
sub.SetDefaultAccess(usr.Access.Auth, usr.Access.Anon)
From 0a312077d6c19d1305bcaf56fbd58f1662854361 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 14 Apr 2020 18:28:31 +0300
Subject: [PATCH 124/142] don't attempt to generate default mongo _id
---
server/store/types/types.go | 5 ++---
1 file changed, 2 insertions(+), 3 deletions(-)
diff --git a/server/store/types/types.go b/server/store/types/types.go
index acd3fd604..9d209e143 100644
--- a/server/store/types/types.go
+++ b/server/store/types/types.go
@@ -299,9 +299,8 @@ func ParseP2P(p2p string) (uid1, uid2 Uid, err error) {
// ObjHeader is the header shared by all stored objects.
type ObjHeader struct {
// using string to get around rethinkdb's problems with uint64;
- // `bson:"_id"` tag is for mongodb to use as primary key '_id'
- // 'omitempty' causes mongodb automaticaly create "_id" field if field not set explicitly
- Id string `bson:"_id,omitempty"`
+ // `bson:"_id"` tag is for mongodb to use as primary key '_id'.
+ Id string `bson:"_id"`
id Uid
CreatedAt time.Time
UpdatedAt time.Time
From abb2a7b7b51060edf39afe491fa447ac0c7acee9 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 14 Apr 2020 18:33:58 +0300
Subject: [PATCH 125/142] when deleting topic use del.Hard instead of
defaulting to true
---
server/hub.go | 9 +++++----
1 file changed, 5 insertions(+), 4 deletions(-)
diff --git a/server/hub.go b/server/hub.go
index 55f13a74a..7e8ea90db 100644
--- a/server/hub.go
+++ b/server/hub.go
@@ -206,7 +206,8 @@ func (h *Hub) run() {
log.Println("hub: topic's broadcast queue is full", dst.name)
}
}
- } else if (strings.HasPrefix(msg.rcptto, "usr") || strings.HasPrefix(msg.rcptto, "grp")) && globals.cluster.isRemoteTopic(msg.rcptto) {
+ } else if (strings.HasPrefix(msg.rcptto, "usr") || strings.HasPrefix(msg.rcptto, "grp")) &&
+ globals.cluster.isRemoteTopic(msg.rcptto) {
// It is a remote topic.
if err := globals.cluster.routeToTopicIntraCluster(msg.rcptto, msg); err != nil {
log.Printf("hub: routing to '%s' failed", msg.rcptto)
@@ -364,7 +365,7 @@ func (h *Hub) topicUnreg(sess *Session, topic string, msg *ClientComMessage, rea
// Case 1.1.1: requester is the owner or last sub in a p2p topic
t.markPaused(true)
- if err := store.Topics.Delete(topic, true); err != nil {
+ if err := store.Topics.Delete(topic, msg.Del.Hard); err != nil {
t.markPaused(false)
sess.queueOut(ErrUnknown(msg.id, msg.topic, now))
return err
@@ -422,7 +423,7 @@ func (h *Hub) topicUnreg(sess *Session, topic string, msg *ClientComMessage, rea
if tcat == types.TopicCatP2P && len(subs) < 2 {
// This is a P2P topic and fewer than 2 subscriptions, delete the entire topic
- if err := store.Topics.Delete(topic, true); err != nil {
+ if err := store.Topics.Delete(topic, msg.Del.Hard); err != nil {
sess.queueOut(ErrUnknown(msg.id, msg.topic, now))
return err
}
@@ -454,7 +455,7 @@ func (h *Hub) topicUnreg(sess *Session, topic string, msg *ClientComMessage, rea
} else {
// Case 1.2.1.1: owner, delete the group topic from db.
// Only group topics have owners.
- if err := store.Topics.Delete(topic, true); err != nil {
+ if err := store.Topics.Delete(topic, msg.Del.Hard); err != nil {
sess.queueOut(ErrUnknown(msg.id, msg.topic, now))
return err
}
From db431ba8429a54eb5d68daf79682adf0a6419ff6 Mon Sep 17 00:00:00 2001
From: or-else
Date: Tue, 14 Apr 2020 18:42:44 +0300
Subject: [PATCH 126/142] fix for the first crash in #420
---
server/cluster.go | 20 ++++++++++++++++----
server/hub.go | 2 +-
2 files changed, 17 insertions(+), 5 deletions(-)
diff --git a/server/cluster.go b/server/cluster.go
index 81032de04..9bfae44b9 100644
--- a/server/cluster.go
+++ b/server/cluster.go
@@ -408,13 +408,21 @@ func (Cluster) Proxy(msg *ClusterResp, unused *bool) error {
func (c *Cluster) Route(msg *ClusterReq, rejected *bool) error {
*rejected = false
if msg.Signature != c.ring.Signature() {
- log.Println("cluster Route: session signature mismatch", msg.Sess.Sid)
+ sid := ""
+ if msg.Sess != nil {
+ sid = msg.Sess.Sid
+ }
+ log.Println("cluster Route: session signature mismatch", sid)
*rejected = true
return nil
}
if msg.SrvMsg == nil {
+ sid := ""
+ if msg.Sess != nil {
+ sid = msg.Sess.Sid
+ }
// TODO: maybe panic here.
- log.Println("cluster Route: nil server message", msg.Sess.Sid)
+ log.Println("cluster Route: nil server message", sid)
*rejected = true
return nil
}
@@ -606,7 +614,7 @@ func (c *Cluster) routeToTopic(msg *ClientComMessage, topic string, sess *Sessio
}
// Forward server response message to the node that owns topic.
-func (c *Cluster) routeToTopicIntraCluster(topic string, msg *ServerComMessage) error {
+func (c *Cluster) routeToTopicIntraCluster(topic string, msg *ServerComMessage, sess *Session) error {
n := c.nodeForTopic(topic)
if n == nil {
return errors.New("node for topic not found (intra)")
@@ -617,8 +625,12 @@ func (c *Cluster) routeToTopicIntraCluster(topic string, msg *ServerComMessage)
Signature: c.ring.Signature(),
Fingerprint: c.fingerprint,
RcptTo: topic,
- SrvMsg: msg}
+ SrvMsg: msg,
+ }
+ if sess != nil {
+ req.Sess = &ClusterSess{Sid: sess.sid}
+ }
return n.route(req)
}
diff --git a/server/hub.go b/server/hub.go
index 7e8ea90db..6e86720b8 100644
--- a/server/hub.go
+++ b/server/hub.go
@@ -209,7 +209,7 @@ func (h *Hub) run() {
} else if (strings.HasPrefix(msg.rcptto, "usr") || strings.HasPrefix(msg.rcptto, "grp")) &&
globals.cluster.isRemoteTopic(msg.rcptto) {
// It is a remote topic.
- if err := globals.cluster.routeToTopicIntraCluster(msg.rcptto, msg); err != nil {
+ if err := globals.cluster.routeToTopicIntraCluster(msg.rcptto, msg, msg.sess); err != nil {
log.Printf("hub: routing to '%s' failed", msg.rcptto)
}
} else if msg.Pres == nil && msg.Info == nil {
From 00a596440ed1150db25dd91aec14bc306b048c9c Mon Sep 17 00:00:00 2001
From: or-else
Date: Thu, 16 Apr 2020 11:23:29 +0300
Subject: [PATCH 127/142] reply with InfoNoAction if credential being deleted
is not found
---
server/topic.go | 6 +++++-
1 file changed, 5 insertions(+), 1 deletion(-)
diff --git a/server/topic.go b/server/topic.go
index 47a93b486..a209f7b06 100644
--- a/server/topic.go
+++ b/server/topic.go
@@ -2299,9 +2299,13 @@ func (t *Topic) replyDelCred(h *Hub, sess *Session, asUid types.Uid, authLvl aut
if len(removed) > 0 {
t.tags = tags
t.presSubsOnline("tags", "", nilPresParams, nilPresFilters, "")
+ sess.queueOut(decodeStoreError(err, del.Id, del.Topic, now, nil))
+ } else {
+ sess.queueOut(InfoNoAction(del.Id, del.Topic, now))
}
+ } else {
+ sess.queueOut(decodeStoreError(err, del.Id, del.Topic, now, nil))
}
- sess.queueOut(decodeStoreError(err, del.Id, del.Topic, now, nil))
return err
}
From 4da4a465759c56c02f21fae9fd609383ad2c553a Mon Sep 17 00:00:00 2001
From: or-else
Date: Thu, 16 Apr 2020 12:29:18 +0300
Subject: [PATCH 128/142] respond with InfoNoAction to delete requests of
credential is not found
---
server/db/mongodb/adapter.go | 22 ++++++++++++++++++----
server/db/mysql/adapter.go | 31 +++++++++++++++++++++++++------
server/db/rethinkdb/adapter.go | 21 +++++++++++++++++----
server/topic.go | 11 ++++++-----
server/user.go | 4 ++++
5 files changed, 70 insertions(+), 19 deletions(-)
diff --git a/server/db/mongodb/adapter.go b/server/db/mongodb/adapter.go
index 163256262..17a6af9c2 100644
--- a/server/db/mongodb/adapter.go
+++ b/server/db/mongodb/adapter.go
@@ -601,7 +601,7 @@ func (a *adapter) UserDelete(uid t.Uid, hard bool) error {
}
// Delete credentials.
- if err = a.credDel(sc, uid, "", ""); err != nil {
+ if err = a.credDel(sc, uid, "", ""); err != nil && err != t.ErrNotFound {
return err
}
@@ -921,7 +921,12 @@ func (a *adapter) credDel(ctx context.Context, uid t.Uid, method, value string)
filter["value"] = value
}
} else {
- _, err := credCollection.DeleteMany(ctx, filter)
+ res, err := credCollection.DeleteMany(ctx, filter)
+ if err == nil {
+ if res.DeletedCount == 0 {
+ err = t.ErrNotFound
+ }
+ }
return err
}
@@ -930,12 +935,21 @@ func (a *adapter) credDel(ctx context.Context, uid t.Uid, method, value string)
hardDeleteFilter["$or"] = b.A{
b.M{"done": true},
b.M{"retries": 0}}
- if _, err := credCollection.DeleteMany(ctx, hardDeleteFilter); err != nil {
+ res, err := credCollection.DeleteMany(ctx, hardDeleteFilter)
+ if err != nil {
return err
}
+ if res.DeletedCount > 0 {
+ return nil
+ }
// Soft-delete all other values.
- _, err := credCollection.UpdateMany(ctx, filter, b.M{"$set": b.M{"deletedat": t.TimeNow()}})
+ res, err = credCollection.UpdateMany(ctx, filter, b.M{"$set": b.M{"deletedat": t.TimeNow()}})
+ if err == nil {
+ if res.DeletedCount == 0 {
+ err = t.ErrNotFound
+ }
+ }
return err
}
diff --git a/server/db/mysql/adapter.go b/server/db/mysql/adapter.go
index 6505d0350..eb03c7d95 100644
--- a/server/db/mysql/adapter.go
+++ b/server/db/mysql/adapter.go
@@ -905,7 +905,7 @@ func (a *adapter) UserDelete(uid t.Uid, hard bool) error {
}
// Delete all credentials.
- if err = credDel(tx, uid, "", ""); err != nil {
+ if err = credDel(tx, uid, "", ""); err != nil && err != t.ErrNotFound {
return err
}
@@ -2342,8 +2342,10 @@ func deviceDelete(tx *sqlx.Tx, uid t.Uid, deviceID string) error {
res, err = tx.Exec("DELETE FROM devices WHERE userid=? AND hash=?", store.DecodeUid(uid), deviceHasher(deviceID))
}
- if count, _ := res.RowsAffected(); count == 0 && err == nil {
- err = t.ErrNotFound
+ if err == nil {
+ if count, _ := res.RowsAffected(); count == 0 {
+ err = t.ErrNotFound
+ }
}
return err
@@ -2465,19 +2467,36 @@ func credDel(tx *sqlx.Tx, uid t.Uid, method, value string) error {
}
}
+ var err error
+ var res sql.Result
if method == "" {
- _, err := tx.Exec("DELETE FROM credentials"+constraints, args...)
+ // Case 1
+ res, err = tx.Exec("DELETE FROM credentials"+constraints, args...)
+ if err == nil {
+ if count, _ := res.RowsAffected(); count == 0 {
+ err = t.ErrNotFound
+ }
+ }
return err
}
// Case 2.1
- if _, err := tx.Exec("DELETE FROM credentials"+constraints+" AND (done=true OR retries=0)", args...); err != nil {
+ res, err = tx.Exec("DELETE FROM credentials"+constraints+" AND (done=true OR retries=0)", args...)
+ if err != nil {
return err
}
+ if count, _ := res.RowsAffected(); count > 0 {
+ return nil
+ }
// Case 2.2
args = append([]interface{}{t.TimeNow()}, args...)
- _, err := tx.Exec("UPDATE credentials SET deletedat=?"+constraints, args...)
+ res, err = tx.Exec("UPDATE credentials SET deletedat=?"+constraints, args...)
+ if err == nil {
+ if count, _ := res.RowsAffected(); count >= 0 {
+ err = t.ErrNotFound
+ }
+ }
return err
}
diff --git a/server/db/rethinkdb/adapter.go b/server/db/rethinkdb/adapter.go
index fbc65ce27..d3b0be8e3 100644
--- a/server/db/rethinkdb/adapter.go
+++ b/server/db/rethinkdb/adapter.go
@@ -772,7 +772,7 @@ func (a *adapter) UserDelete(uid t.Uid, hard bool) error {
}
// Delete credentials.
- if err = a.CredDel(uid, "", ""); err != nil {
+ if err = a.CredDel(uid, "", ""); err != nil && err != t.ErrNotFound {
return err
}
// And finally delete the user.
@@ -2069,18 +2069,31 @@ func (a *adapter) CredDel(uid t.Uid, method, value string) error {
}
if method == "" {
- _, err := q.Delete().RunWrite(a.conn)
+ res, err := q.Delete().RunWrite(a.conn)
+ if err == nil {
+ if res.Deleted == 0 {
+ err = t.ErrNotFound
+ }
+ }
return err
}
// Hard-delete all confirmed values or values with no attempts at confirmation.
- _, err := q.Filter(rdb.Or(rdb.Row.Field("Done").Eq(true), rdb.Row.Field("Retries").Eq(0))).Delete().RunWrite(a.conn)
+ res, err := q.Filter(rdb.Or(rdb.Row.Field("Done").Eq(true), rdb.Row.Field("Retries").Eq(0))).Delete().RunWrite(a.conn)
if err != nil {
return err
}
+ if res.Deleted > 0 {
+ return nil
+ }
// Soft-delete all other values.
- _, err = q.Update(map[string]interface{}{"DeletedAt": t.TimeNow()}).RunWrite(a.conn)
+ res, err = q.Update(map[string]interface{}{"DeletedAt": t.TimeNow()}).RunWrite(a.conn)
+ if err == nil {
+ if res.Deleted == 0 {
+ err = t.ErrNotFound
+ }
+ }
return err
}
diff --git a/server/topic.go b/server/topic.go
index a209f7b06..ddd53c5ae 100644
--- a/server/topic.go
+++ b/server/topic.go
@@ -2299,13 +2299,14 @@ func (t *Topic) replyDelCred(h *Hub, sess *Session, asUid types.Uid, authLvl aut
if len(removed) > 0 {
t.tags = tags
t.presSubsOnline("tags", "", nilPresParams, nilPresFilters, "")
- sess.queueOut(decodeStoreError(err, del.Id, del.Topic, now, nil))
- } else {
- sess.queueOut(InfoNoAction(del.Id, del.Topic, now))
}
- } else {
- sess.queueOut(decodeStoreError(err, del.Id, del.Topic, now, nil))
+ } else if err == nil {
+ sess.queueOut(InfoNoAction(del.Id, del.Topic, now))
+ return nil
}
+
+ sess.queueOut(decodeStoreError(err, del.Id, del.Topic, now, nil))
+
return err
}
diff --git a/server/user.go b/server/user.go
index 651eb2fcf..b7d853d3f 100644
--- a/server/user.go
+++ b/server/user.go
@@ -490,6 +490,10 @@ func deleteCred(uid types.Uid, authLvl auth.Level, cred *MsgCredClient) ([]strin
// The credential is either not required or more than one credential is validated for the given method.
err := vld.Remove(uid, cred.Value)
if err != nil {
+ if err == types.ErrNotFound {
+ // Credential is not deleted because it's not found
+ err = nil
+ }
return nil, err
}
From 005e53fed9dae84513b49aa2273bd69e34a619e5 Mon Sep 17 00:00:00 2001
From: or-else
Date: Thu, 16 Apr 2020 17:58:05 +0300
Subject: [PATCH 129/142] documentation updates
---
README.md | 8 ++++----
server/push/tnpg/README.md | 9 ++++-----
2 files changed, 8 insertions(+), 9 deletions(-)
diff --git a/README.md b/README.md
index ca8ea4d04..1f7d84f14 100644
--- a/README.md
+++ b/README.md
@@ -75,9 +75,9 @@ When you register a new account you are asked for an email address to send valid
* Persistent message store, paginated message history.
* Javascript bindings with no external dependencies.
* Java bindings (dependencies: [Jackson](https://github.com/FasterXML/jackson), [Java-Websocket](https://github.com/TooTallNate/Java-WebSocket)). Suitable for Android but with no Android SDK dependencies.
-* Websocket, long polling, and [gRPC](https://grpc.io/) over TCP transports.
+* Websocket, long polling, and [gRPC](https://grpc.io/) over TCP or Unix sockets.
* JSON or [protobuf version 3](https://developers.google.com/protocol-buffers/) wire protocols.
-* [TLS](https://en.wikipedia.org/wiki/Transport_Layer_Security) with [Letsencrypt](https://letsencrypt.org/) or conventional certificates.
+* Optional built-in [TLS](https://en.wikipedia.org/wiki/Transport_Layer_Security) with [Letsencrypt](https://letsencrypt.org/) or conventional certificates.
* User search/discovery.
* Rich formatting of messages, markdown-style: \*style\* → **style**.
* Inline images and file attachments.
@@ -86,9 +86,9 @@ When you register a new account you are asked for an email address to send valid
* Support for client-side data caching.
* Ability to block unwanted communication server-side.
* Anonymous users (important for use cases related to tech support over chat).
-* Push notifications using [FCM](https://firebase.google.com/docs/cloud-messaging/).
+* Push notifications using [FCM](https://firebase.google.com/docs/cloud-messaging/) or [TNPG](server/push/tnpg/).
* Storage and out of band transfer of large objects like video files using local file system or Amazon S3.
-* Plugins to extend functionality, for example to enable chatbots.
+* Plugins to extend functionality, for example, to enable chatbots.
### Planned
diff --git a/server/push/tnpg/README.md b/server/push/tnpg/README.md
index 542a026e2..b99384d79 100644
--- a/server/push/tnpg/README.md
+++ b/server/push/tnpg/README.md
@@ -3,9 +3,9 @@
This is push notifications adapter which communicates with Tinode Push Gateway (TNPG).
TNPG is a proprietary service intended to simplify deployment of on-premise installations.
-Deploying a Tinode server without TNPG requires [configuring Google FCM](../fcm/) with your own credentials including recompiling mobile clients and releasing them to PlayStore and AppStore under your own accounts which is usually time consuming and relatively complex.
+Deploying a Tinode server without TNPG requires [configuring Google FCM](../fcm/) with your own credentials, recompiling Android and iOS clients, releasing them to PlayStore and AppStore under your own accounts. It's usually time consuming and relatively complex.
-TNPG solves this problem by allowing you to send push notifications on behalf of Tinode. Internally it uses Google FCM and as such supports the same platforms as FCM. The main advantage of using TNPG over FCM is simplicity of configuration: mobile clients do not need to be recompiled, all is needed is a configuration update on the server.
+TNPG solves this problem by allowing you to send push notifications on behalf of Tinode: you hand a notification over to Tinode, Tinode sends it to client using its own credentials and certificates. Internally it uses [Google FCM](https://firebase.google.com/docs/cloud-messaging/) and as such supports the same platforms as FCM. The main advantage of using TNPG over FCM is simplicity of configuration: mobile clients don't have to be recompiled, all is needed is a configuration update on the server.
## Configuring TNPG adapter
@@ -15,8 +15,7 @@ TNPG solves this problem by allowing you to send push notifications on behalf of
2. Get the TPNG token from the _On premise_ section by following the instructions there.
### Configure the server
-
-Update the server config [`tinode.conf`](../server/tinode.conf#L384), section `"push"` -> `"name": "tnpg"`:
+Update the server config [`tinode.conf`](../../tinode.conf#L384), section `"push"` -> `"name": "tnpg"`:
```js
{
"enabled": true,
@@ -24,4 +23,4 @@ Update the server config [`tinode.conf`](../server/tinode.conf#L384), section `"
"token": "SoMe_LonG.RaNDoM-StRiNg.12345" // authentication token obtained from console.tinode.co
}
```
-Make sure the `fcm` section is disabled `"enabled": false`.
+Make sure the `fcm` section is disabled `"enabled": false` or removed altogether.
From c1a6e4c875170faf188a53f6f0bf8f7f175bc173 Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 17 Apr 2020 15:29:59 +0300
Subject: [PATCH 130/142] mongo adapter fix
---
server/db/mongodb/adapter.go | 10 ++++------
1 file changed, 4 insertions(+), 6 deletions(-)
diff --git a/server/db/mongodb/adapter.go b/server/db/mongodb/adapter.go
index 17a6af9c2..c1ceca89f 100644
--- a/server/db/mongodb/adapter.go
+++ b/server/db/mongodb/adapter.go
@@ -935,18 +935,16 @@ func (a *adapter) credDel(ctx context.Context, uid t.Uid, method, value string)
hardDeleteFilter["$or"] = b.A{
b.M{"done": true},
b.M{"retries": 0}}
- res, err := credCollection.DeleteMany(ctx, hardDeleteFilter)
- if err != nil {
+ if res, err := credCollection.DeleteMany(ctx, hardDeleteFilter); err != nil {
return err
- }
- if res.DeletedCount > 0 {
+ } else if res.DeletedCount > 0 {
return nil
}
// Soft-delete all other values.
- res, err = credCollection.UpdateMany(ctx, filter, b.M{"$set": b.M{"deletedat": t.TimeNow()}})
+ res, err := credCollection.UpdateMany(ctx, filter, b.M{"$set": b.M{"deletedat": t.TimeNow()}})
if err == nil {
- if res.DeletedCount == 0 {
+ if res.ModifiedCount == 0 {
err = t.ErrNotFound
}
}
From 32527a4943a0387745df856df762812eb2f8214e Mon Sep 17 00:00:00 2001
From: or-else
Date: Fri, 17 Apr 2020 18:10:44 +0300
Subject: [PATCH 131/142] add explanation of the file descriptors limit on
linux
---
docs/faq.md | 4 ++++
1 file changed, 4 insertions(+)
diff --git a/docs/faq.md b/docs/faq.md
index 02e2ad5f7..36bfa8c1f 100644
--- a/docs/faq.md
+++ b/docs/faq.md
@@ -68,3 +68,7 @@ USE 'tinode';
UPDATE auth SET authlvl=30 WHERE uname='basic:login-of-the-user-to-make-root';
```
The test database has a stock user `xena` which has root access.
+
+
+### Q: Once the number of connection reaches about 1000 per node, all kinds of problems start. Is this a bug?
+**A**: It is likely not a bug. To ensure good server performance Linux limits the total number of open file descriptors (live network connections, open files) for each process at the kernel level. The default limit is usually 1024. There are other possible restrictions on the number of file descriptors. The problems you are experiencing are likely caused by exceeding one of the OS limits. Please seek assistance of a system administrator.
From 58d0c11498349a7b3574afcf19011d27846b1382 Mon Sep 17 00:00:00 2001
From: or-else
Date: Sat, 18 Apr 2020 10:51:34 +0300
Subject: [PATCH 132/142] doc updates
---
docs/faq.md | 2 +-
docs/ios-account.png | Bin 97075 -> 111280 bytes
2 files changed, 1 insertion(+), 1 deletion(-)
diff --git a/docs/faq.md b/docs/faq.md
index 36bfa8c1f..07f55230e 100644
--- a/docs/faq.md
+++ b/docs/faq.md
@@ -71,4 +71,4 @@ The test database has a stock user `xena` which has root access.
### Q: Once the number of connection reaches about 1000 per node, all kinds of problems start. Is this a bug?
-**A**: It is likely not a bug. To ensure good server performance Linux limits the total number of open file descriptors (live network connections, open files) for each process at the kernel level. The default limit is usually 1024. There are other possible restrictions on the number of file descriptors. The problems you are experiencing are likely caused by exceeding one of the OS limits. Please seek assistance of a system administrator.
+**A**: It is likely not a bug. To ensure good server performance Linux limits the total number of open file descriptors (live network connections, open files) for each process at the kernel level. The default limit is usually 1024. There are other possible restrictions on the number of file descriptors. The problems you are experiencing are likely caused by exceeding one of the Linux-imposed limits. Please seek assistance of a system administrator.
diff --git a/docs/ios-account.png b/docs/ios-account.png
index 8294ca91aca07fec67ad23f0c3b9440b241a8194..fca4ee2b62df61e666e6b26d90078fc7faa3a504 100644
GIT binary patch
literal 111280
zcmdSBWmH^Cw>FA~#)1WR2?Tf7;10pv-JQmRLvVKp5(sXMy9W#I?!n#dHre|<=iD>C
z`}6y8o6%!c*Q!}nYgN^>XHLQu*3`w21ZZnx=gb4-C;Qui
z2V8!uW+Wr|+r-71pG-qekwnDa$&`ehfsKKQOaOs|goMw@#EeHtRQz9Z@GpKc3l|p$
z9!5rYcXtMNRt9?~b4F%vZf-^<7Dg5pdT1Kj^qp@Wy{zun053A_W3^w$02
z2S}=Hj^HI)=eM!JHJj(me<%5se7q3Abq=Cl)=T?m+FvnZV*Vi_@aGWg3@?(~xEcDl
zo0=tBm~hc!L5#$>xQVhtt(R$UtQ;*374ORfN%|f?-nZ#iu>n$bYRFAfk7|sShoUa1
zgumU{NYCE2V@xoF7nPLIn3$RdkBmr?LXQj$4Q)sLEMmm7<&1vHT(Tb)
z@otV~ux)lB+WY$=Wk?O22DWw0D4iHZcCeh79#n>y;}S!!XXG@&r#PNL#>CX8IGqT^
zLseBZ+
z5qN&D8-zB`YXn$WSROP9%6KYYBjtlJW>)38F2G{sDfy(d?e
z-;pcddp}>#muCGAj#q2;(fG@yAe
z+bg(kPLDs+_n|oNxvbp4`{{f@nB*{M#eI!+*`a4N&-b022rl^LDmPvX4U62Kto$g$
zGd6c~i%#~L5KH-k1hv_f{Y`5~PB_UHUa=1S)#>ekl7v7|_COz`zX
z2#UTO<(eSiK2koprHrbw{h?4R+86xxn1E*n<<6SJ6n-rvL|yXmU&YD>6R1K})d?%i
z7;&&!gj`$y+sY3qiw;>Vab<)I}>DXVq5r{%re2=&p4tHVLW?1ac9-DArSxrZW;NezZ
z9v)g)qsqXaQo9vcJ%Uq`xKbqltcSg!;Vt6KeOT5J4qxtdw;AzE94}51gacRXb2+M#
z_NA|dA*KwLzoLFv`AT1i-HV$)Ktkxf5UXdeK=-~`sOSdT6g#Z
zIq=;~Eb#^Q#b*YEoR>B_^&`6DuuM>L{gYd@|FlN3YihM|!({u@p5SY;yu5m$&t~{Q
zPan4a_j4B_dH~bI%`vv;)t}Ts}k&ktfagp1*
zS+ZB1xr*Y{Q&`liBl~x!AwxMD=IlDL(CG4*pcU3bLca?#j^>u%nPZ=O;xB7C(c0Kf
zc%@4pH{OVoAVR>P5ixm
zlzW&Pp>_0}P*}(j`%<#1JGMgcGqo*CreTU>wgeAir
zwxZ^&1<8fAwL6wm
zNraE8#b)g|8IF2oFR|Vb3^68H(3yVGu;!a@O}*0WiVaspig!E8xfW*nLx~`M({XBg
zQHhs9dalh2Wg(*0NsBZqF0MEo<Q6ZuB8LA_C
zB37b}1O>`NN6QYg(Fe`iO2sjig}C0GJcJjpZk8Z_|QVpl_BVt4Xnxf(Gh1zxfT!q
zN$%Sg9+7RW*(d^dNnM(tJS6#YcHbRr^iX8&z;pb#Ia$#j#JnK))f8zVCE|gS-0F`~
zN9THda)!?NlFxw*mdoisqkMXbi?SJ5qwJa+`y$sbhAUCOvRoI6_hNKGa<%eof{*
zeh)hc_gCbclx(Vlflz)x-@reyp@MeSP5(qrv5?$GFg^gYKny*Q(0
z-1ie4)S}DOsHv^xcl5Y~Xk=tT9;gmEm{>?zp}o-zMkn$`;y_&Fg+R@A4@ccZr1Sua
zYqBYUr!$c_SK420@h=CEbE&ezC!|{z_zAXrZtxl|$s9s8Fl@{w7BNKpJPhaKU&v9W
zHEXGaabSG$7tS(3L3dH%dwY1S@QA1%cH&nz%a3Yml*0AhFeeW#^1*u4lP4Mz`g+)M
z)gb+ax#ed1H<$~=v>>-GnYav^gkNy?4x;i-y~m!G0Vb;4GJKs?9JPcn+9S8iep8xE
zP=KQNZY-_ZA11$K`=l==KMaBH+@$zsxmpJ4S1gC@M{CNTeO>;V73FA53$y6g^&v+;
zbdQIcNHBlS9Y5qP%jc&$W#%>x2Ru%bTz7x1s)@_*lu
z$OL*Lf%)~D^&>CS&&U}A3HIz&EE!G
zP$c7dy~^4OVJ>9<`5}UuU6E)=}Pc*M?=|Jg^?-St_yDi}|}W9TZUX
z!hEsvkOi!?tVeVOmzV|Qh88)ZT+TWkzsBHmTl4Bs$H=6}AF#RwwP(N8)L`V6u#hK4
z>`tpKE00s7(fq7!b!(|J{zqE|+iKGQHnInN#yK{^5$l#`UU43N^o_XZ;cgnAQejPm
zeMK6_AExX%AMqY4unyr5pon^ob0M>}7{L^onV2bX{T?&UX$esuAt;S%u_}$A^Dd_ke44S9B(cIB~EFyW35Fl=AD;$iD
z)ukS|TwqKQ?@K8|21
z%8>S6-BH!=i}0$IW=vAjW$4mG^{H??M-#Xb
zh9OYsMq9|IUhRfAz0doz%8hnWGM8b2=!hCl(EZf+Mc1cURVqD_Nk%gI5a#`nOrD(B
zM^d4mx~9lsKGI1w?@gis1=s&^2rxm7QWC-VEqjf!Wf=f_Dpp
zi-JozR3Z&GVy^qlHyxzixjF2gCmXX|9ZPvvDtRhr-N2^L2UTyU?W!~AV;;DC|dyX=K7|iek>-Z
zhf8)AE*d)w3o1>kW$8wr@RihO!0W%PW$h*)O(a(T=JgVw*ZQ7_4ObU+6bHqfv*S!$6yBjDNIqUS@v6_99
zCO~+3vXBSv>_s{RIK-*o(iG6o$&Mgn!|(G|xqJMXqeT;V6<{TBQm;p!Iu%cmuvVbB
z`OmVDK*>=OvC526BW%oj1-a~+1Z8`WmeEAKx*us{A$X}5!6y-quF@V627)?;urJ(flY!)T*BAt|*z$$=Of
z+x)}A_7i%aoQ^-MZj{1$Yz*!7IT}cs-aUnmoW*dm^pJ<9gl5f_0(OK&@uj`nQ^)mQ
zyrSbtcbgU3hmf$Pnj4~N5%J4uO5&%R_N_2;6Huoh$IOClXNv)C2zl(p`lr+Dl{c0Xs9z3$P)T~c$LA_#FVFp{iI;Rx%`K648AN`~6sA}MO!H6fpHFF{UGfSo5Po>!?(b9N+-g{MC!>wGU
zGbir2OQFKn7YUsv5p(8-@QhpQ9Me*%s)*QJm5*__h9wqXpG5mCq0X==2p{p%ran=!>YYiHh`o1%J#MxiR`kV9T@>!`S@Js(@vo6s~I0$3Xp!g9<=_@`cwQDzA8Y+%rJhcSl
z`vU5MGAPf^h~`OBHuR&gQd59vaPJ=C{RbK1qOd7QQe{9zDXMbpRRf^O4XakKsUTne
zV*Pg%tvhwa>Ul`)!hiTFA&lq<2P26NX)>uW6Us&(Wi)IM<2F0D#b+*}_3Mxb$ArKS
z{_fgY*J$La)7I2btCFzF_;?76G%J2dB@S~Eng>E(!=uKG92J2;C$o8}S>NXH-6@IL
zAKg!=0xujg?4*FwgaR+WlB(H}FqMErP6KW!PR^jpzQNq0tA@}v6NwO!}brY|&bu
z0pfFRCbnYXbT_T*ty+Z4Md)eYcFiXtBXeMW6TT+__h4vA+4{w{r=@ty{M@0EiRB3;no5-PKs0BnaF=_>LXR89|_gA6weKzE>j+jERHhK#Wv5OzAGC?`Ao;ZF5pdo37w8c
zJlXM93cjF3X}JBT8NpDJg?M~vFFZzr^tRfOWGib3T&037+|*!O(}J1|BRG
zDoUaa1>e$$Yizz5U<)Lkl>WJQPo}splp`wUqYDt1`g}P2CYh@8oEn`KZ^t={;}8D(
z0v~7WaPkZrr(<%_1zaD~
zOGLR*Ua~L&ctC|L533WMQWX9?{Ev(MyB>^a5QCSHWRdn|8ks!~*K0Vz43A2zI^wwt
zI}P=;vo35a+E