2018-11-29 23:53:43 +01:00
|
|
|
package matterclient
|
|
|
|
|
|
|
|
import (
|
2018-12-05 23:40:55 +00:00
|
|
|
"crypto/md5" //nolint:gosec
|
2018-11-29 23:53:43 +01:00
|
|
|
"crypto/tls"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/http/cookiejar"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/gorilla/websocket"
|
|
|
|
"github.com/jpillora/backoff"
|
|
|
|
"github.com/mattermost/mattermost-server/model"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (m *MMClient) doLogin(firstConnection bool, b *backoff.Backoff) error {
|
|
|
|
var resp *model.Response
|
|
|
|
var appErr *model.AppError
|
|
|
|
var logmsg = "trying login"
|
|
|
|
var err error
|
|
|
|
for {
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debugf("%s %s %s %s", logmsg, m.Credentials.Team, m.Credentials.Login, m.Credentials.Server)
|
2018-11-29 23:53:43 +01:00
|
|
|
if m.Credentials.Token != "" {
|
|
|
|
resp, err = m.doLoginToken()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
m.User, resp = m.Client.Login(m.Credentials.Login, m.Credentials.Pass)
|
|
|
|
}
|
|
|
|
appErr = resp.Error
|
|
|
|
if appErr != nil {
|
|
|
|
d := b.Duration()
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debug(appErr.DetailedError)
|
2018-11-29 23:53:43 +01:00
|
|
|
if firstConnection {
|
|
|
|
if appErr.Message == "" {
|
|
|
|
return errors.New(appErr.DetailedError)
|
|
|
|
}
|
|
|
|
return errors.New(appErr.Message)
|
|
|
|
}
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debugf("LOGIN: %s, reconnecting in %s", appErr, d)
|
2018-11-29 23:53:43 +01:00
|
|
|
time.Sleep(d)
|
|
|
|
logmsg = "retrying login"
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
// reset timer
|
|
|
|
b.Reset()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MMClient) doLoginToken() (*model.Response, error) {
|
|
|
|
var resp *model.Response
|
|
|
|
var logmsg = "trying login"
|
|
|
|
m.Client.AuthType = model.HEADER_BEARER
|
|
|
|
m.Client.AuthToken = m.Credentials.Token
|
|
|
|
if m.Credentials.CookieToken {
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debugf(logmsg + " with cookie (MMAUTH) token")
|
2018-11-29 23:53:43 +01:00
|
|
|
m.Client.HttpClient.Jar = m.createCookieJar(m.Credentials.Token)
|
|
|
|
} else {
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debugf(logmsg + " with personal token")
|
2018-11-29 23:53:43 +01:00
|
|
|
}
|
|
|
|
m.User, resp = m.Client.GetMe("")
|
|
|
|
if resp.Error != nil {
|
|
|
|
return resp, resp.Error
|
|
|
|
}
|
|
|
|
if m.User == nil {
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Errorf("LOGIN TOKEN: %s is invalid", m.Credentials.Pass)
|
2018-11-29 23:53:43 +01:00
|
|
|
return resp, errors.New("invalid token")
|
|
|
|
}
|
|
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MMClient) handleLoginToken() error {
|
|
|
|
switch {
|
|
|
|
case strings.Contains(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN):
|
|
|
|
token := strings.Split(m.Credentials.Pass, model.SESSION_COOKIE_TOKEN+"=")
|
|
|
|
if len(token) != 2 {
|
|
|
|
return errors.New("incorrect MMAUTHTOKEN. valid input is MMAUTHTOKEN=yourtoken")
|
|
|
|
}
|
|
|
|
m.Credentials.Token = token[1]
|
|
|
|
m.Credentials.CookieToken = true
|
|
|
|
case strings.Contains(m.Credentials.Pass, "token="):
|
|
|
|
token := strings.Split(m.Credentials.Pass, "token=")
|
|
|
|
if len(token) != 2 {
|
|
|
|
return errors.New("incorrect personal token. valid input is token=yourtoken")
|
|
|
|
}
|
|
|
|
m.Credentials.Token = token[1]
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MMClient) initClient(firstConnection bool, b *backoff.Backoff) error {
|
|
|
|
uriScheme := "https://"
|
|
|
|
if m.NoTLS {
|
|
|
|
uriScheme = "http://"
|
|
|
|
}
|
|
|
|
// login to mattermost
|
|
|
|
m.Client = model.NewAPIv4Client(uriScheme + m.Credentials.Server)
|
2018-12-05 23:40:55 +00:00
|
|
|
m.Client.HttpClient.Transport = &http.Transport{
|
|
|
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: m.SkipTLSVerify}, //nolint:gosec
|
|
|
|
Proxy: http.ProxyFromEnvironment,
|
|
|
|
}
|
2018-11-29 23:53:43 +01:00
|
|
|
m.Client.HttpClient.Timeout = time.Second * 10
|
|
|
|
|
|
|
|
// handle MMAUTHTOKEN and personal token
|
|
|
|
if err := m.handleLoginToken(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if server alive, retry until
|
|
|
|
if err := m.serverAlive(firstConnection, b); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// initialize user and teams
|
|
|
|
func (m *MMClient) initUser() error {
|
|
|
|
m.Lock()
|
|
|
|
defer m.Unlock()
|
|
|
|
// we only load all team data on initial login.
|
|
|
|
// all other updates are for channels from our (primary) team only.
|
2019-02-23 22:51:27 +01:00
|
|
|
//m.logger.Debug("initUser(): loading all team data")
|
2018-11-29 23:53:43 +01:00
|
|
|
teams, resp := m.Client.GetTeamsForUser(m.User.Id, "")
|
|
|
|
if resp.Error != nil {
|
|
|
|
return resp.Error
|
|
|
|
}
|
|
|
|
for _, team := range teams {
|
2019-04-20 23:06:06 +02:00
|
|
|
idx := 0
|
|
|
|
max := 200
|
|
|
|
usermap := make(map[string]*model.User)
|
|
|
|
mmusers, resp := m.Client.GetUsersInTeam(team.Id, idx, max, "")
|
2018-11-29 23:53:43 +01:00
|
|
|
if resp.Error != nil {
|
|
|
|
return errors.New(resp.Error.DetailedError)
|
|
|
|
}
|
2019-04-20 23:06:06 +02:00
|
|
|
for len(mmusers) > 0 {
|
|
|
|
for _, user := range mmusers {
|
|
|
|
usermap[user.Id] = user
|
|
|
|
}
|
|
|
|
mmusers, resp = m.Client.GetUsersInTeam(team.Id, idx, max, "")
|
|
|
|
if resp.Error != nil {
|
|
|
|
return errors.New(resp.Error.DetailedError)
|
|
|
|
}
|
|
|
|
idx++
|
|
|
|
time.Sleep(time.Millisecond * 200)
|
2018-11-29 23:53:43 +01:00
|
|
|
}
|
2019-04-20 23:06:06 +02:00
|
|
|
m.logger.Infof("found %d users in team %s", len(usermap), team.Name)
|
2018-11-29 23:53:43 +01:00
|
|
|
|
|
|
|
t := &Team{Team: team, Users: usermap, Id: team.Id}
|
|
|
|
|
|
|
|
mmchannels, resp := m.Client.GetChannelsForTeamForUser(team.Id, m.User.Id, "")
|
|
|
|
if resp.Error != nil {
|
|
|
|
return resp.Error
|
|
|
|
}
|
|
|
|
t.Channels = mmchannels
|
|
|
|
mmchannels, resp = m.Client.GetPublicChannelsForTeam(team.Id, 0, 5000, "")
|
|
|
|
if resp.Error != nil {
|
|
|
|
return resp.Error
|
|
|
|
}
|
|
|
|
t.MoreChannels = mmchannels
|
|
|
|
m.OtherTeams = append(m.OtherTeams, t)
|
|
|
|
if team.Name == m.Credentials.Team {
|
|
|
|
m.Team = t
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debugf("initUser(): found our team %s (id: %s)", team.Name, team.Id)
|
2018-11-29 23:53:43 +01:00
|
|
|
}
|
|
|
|
// add all users
|
|
|
|
for k, v := range t.Users {
|
|
|
|
m.Users[k] = v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MMClient) serverAlive(firstConnection bool, b *backoff.Backoff) error {
|
|
|
|
defer b.Reset()
|
|
|
|
for {
|
|
|
|
d := b.Duration()
|
|
|
|
// bogus call to get the serverversion
|
|
|
|
_, resp := m.Client.Logout()
|
|
|
|
if resp.Error != nil {
|
|
|
|
return fmt.Errorf("%#v", resp.Error.Error())
|
|
|
|
}
|
2019-06-16 17:23:50 +03:00
|
|
|
if firstConnection && !m.SkipVersionCheck && !supportedVersion(resp.ServerVersion) {
|
2018-11-29 23:53:43 +01:00
|
|
|
return fmt.Errorf("unsupported mattermost version: %s", resp.ServerVersion)
|
|
|
|
}
|
2019-06-16 17:23:50 +03:00
|
|
|
if !m.SkipVersionCheck {
|
|
|
|
m.ServerVersion = resp.ServerVersion
|
|
|
|
if m.ServerVersion == "" {
|
|
|
|
m.logger.Debugf("Server not up yet, reconnecting in %s", d)
|
|
|
|
time.Sleep(d)
|
|
|
|
} else {
|
|
|
|
m.logger.Infof("Found version %s", m.ServerVersion)
|
|
|
|
return nil
|
|
|
|
}
|
2018-11-29 23:53:43 +01:00
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MMClient) wsConnect() {
|
|
|
|
b := &backoff.Backoff{
|
|
|
|
Min: time.Second,
|
|
|
|
Max: 5 * time.Minute,
|
|
|
|
Jitter: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
m.WsConnected = false
|
|
|
|
wsScheme := "wss://"
|
|
|
|
if m.NoTLS {
|
|
|
|
wsScheme = "ws://"
|
|
|
|
}
|
|
|
|
|
|
|
|
// setup websocket connection
|
|
|
|
wsurl := wsScheme + m.Credentials.Server + model.API_URL_SUFFIX_V4 + "/websocket"
|
|
|
|
header := http.Header{}
|
|
|
|
header.Set(model.HEADER_AUTH, "BEARER "+m.Client.AuthToken)
|
|
|
|
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debugf("WsClient: making connection: %s", wsurl)
|
2018-11-29 23:53:43 +01:00
|
|
|
for {
|
2018-12-05 23:40:55 +00:00
|
|
|
wsDialer := &websocket.Dialer{
|
|
|
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: m.SkipTLSVerify}, //nolint:gosec
|
|
|
|
Proxy: http.ProxyFromEnvironment,
|
|
|
|
}
|
2018-11-29 23:53:43 +01:00
|
|
|
var err error
|
|
|
|
m.WsClient, _, err = wsDialer.Dial(wsurl, header)
|
|
|
|
if err != nil {
|
|
|
|
d := b.Duration()
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debugf("WSS: %s, reconnecting in %s", err, d)
|
2018-11-29 23:53:43 +01:00
|
|
|
time.Sleep(d)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debug("WsClient: connected")
|
2018-11-29 23:53:43 +01:00
|
|
|
m.WsSequence = 1
|
|
|
|
m.WsPingChan = make(chan *model.WebSocketResponse)
|
|
|
|
// only start to parse WS messages when login is completely done
|
|
|
|
m.WsConnected = true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MMClient) createCookieJar(token string) *cookiejar.Jar {
|
|
|
|
var cookies []*http.Cookie
|
|
|
|
jar, _ := cookiejar.New(nil)
|
|
|
|
firstCookie := &http.Cookie{
|
|
|
|
Name: "MMAUTHTOKEN",
|
|
|
|
Value: token,
|
|
|
|
Path: "/",
|
|
|
|
Domain: m.Credentials.Server,
|
|
|
|
}
|
|
|
|
cookies = append(cookies, firstCookie)
|
|
|
|
cookieURL, _ := url.Parse("https://" + m.Credentials.Server)
|
|
|
|
jar.SetCookies(cookieURL, cookies)
|
|
|
|
return jar
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MMClient) checkAlive() error {
|
|
|
|
// check if session still is valid
|
|
|
|
_, resp := m.Client.GetMe("")
|
|
|
|
if resp.Error != nil {
|
|
|
|
return resp.Error
|
|
|
|
}
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debug("WS PING")
|
2018-11-29 23:53:43 +01:00
|
|
|
return m.sendWSRequest("ping", nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MMClient) sendWSRequest(action string, data map[string]interface{}) error {
|
|
|
|
req := &model.WebSocketRequest{}
|
|
|
|
req.Seq = m.WsSequence
|
|
|
|
req.Action = action
|
|
|
|
req.Data = data
|
|
|
|
m.WsSequence++
|
2019-02-23 22:51:27 +01:00
|
|
|
m.logger.Debugf("sendWsRequest %#v", req)
|
2018-12-08 16:04:10 +00:00
|
|
|
return m.WsClient.WriteJSON(req)
|
2018-11-29 23:53:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func supportedVersion(version string) bool {
|
|
|
|
if strings.HasPrefix(version, "3.8.0") ||
|
|
|
|
strings.HasPrefix(version, "3.9.0") ||
|
|
|
|
strings.HasPrefix(version, "3.10.0") ||
|
|
|
|
strings.HasPrefix(version, "4.") ||
|
|
|
|
strings.HasPrefix(version, "5.") {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func digestString(s string) string {
|
2018-12-05 23:40:55 +00:00
|
|
|
return fmt.Sprintf("%x", md5.Sum([]byte(s))) //nolint:gosec
|
2018-11-29 23:53:43 +01:00
|
|
|
}
|