mirror of
https://github.com/42wim/matterbridge.git
synced 2024-11-25 05:29:40 +01:00
1d50da4b1c
* nctalk: add message deletion support Signed-off-by: Gary Kim <gary@garykim.dev> * nctalk: seperate out deletion and sending logic Signed-off-by: Gary Kim <gary@garykim.dev> * nctalk: update library to v0.2.0 Signed-off-by: Gary Kim <gary@garykim.dev> * Rename functions to be clearer Signed-off-by: Gary Kim <gary@garykim.dev> * Update to go-nc-talk v0.2.1 Signed-off-by: Gary Kim <gary@garykim.dev> * Update to go-nc-talk v0.2.2 Signed-off-by: Gary Kim <gary@garykim.dev> * Make deletions easier to debug Signed-off-by: Gary Kim <gary@garykim.dev>
322 lines
13 KiB
Go
322 lines
13 KiB
Go
// Copyright (c) 2020 Gary Kim <gary@garykim.dev>, All Rights Reserved
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package user
|
|
|
|
import (
|
|
"crypto/tls"
|
|
"encoding/json"
|
|
"errors"
|
|
"strings"
|
|
|
|
"github.com/monaco-io/request"
|
|
|
|
"gomod.garykim.dev/nc-talk/constants"
|
|
"gomod.garykim.dev/nc-talk/ocs"
|
|
)
|
|
|
|
const (
|
|
ocsCapabilitiesEndpoint = "/ocs/v2.php/cloud/capabilities"
|
|
ocsRoomsv2Endpoint = "/ocs/v2.php/apps/spreed/api/v2/room"
|
|
ocsRoomsv4Endpoint = "/ocs/v2.php/apps/spreed/api/v4/room"
|
|
)
|
|
|
|
var (
|
|
// ErrUserIsNil is returned when a function is called with an nil user.
|
|
ErrUserIsNil = errors.New("user is nil")
|
|
|
|
// ErrCannotDownloadFile is returned when a function cannot download the requested file
|
|
ErrCannotDownloadFile = errors.New("cannot download file")
|
|
)
|
|
|
|
// TalkUser represents a user of Nextcloud Talk
|
|
type TalkUser struct {
|
|
User string
|
|
Pass string
|
|
NextcloudURL string
|
|
Config *TalkUserConfig
|
|
capabilities *Capabilities
|
|
}
|
|
|
|
// TalkUserConfig is configuration options for TalkUsers
|
|
type TalkUserConfig struct {
|
|
TLSConfig *tls.Config
|
|
}
|
|
|
|
// Capabilities describes the capabilities that the Nextcloud Talk instance is capable of. Visit https://nextcloud-talk.readthedocs.io/en/latest/capabilities/ for more info.
|
|
type Capabilities struct {
|
|
AttachmentsFolder string `ocscapability:"config => attachments => folder"`
|
|
Audio bool `ocscapability:"audio"`
|
|
Video bool `ocscapability:"video"`
|
|
Chat bool `ocscapability:"chat"`
|
|
GuestSignaling bool `ocscapability:"guest-signaling"`
|
|
EmptyGroupRoom bool `ocscapability:"empty-group-room"`
|
|
GuestDisplayNames bool `ocscapability:"guest-display-names"`
|
|
MultiRoomUsers bool `ocscapability:"multi-room-users"`
|
|
ChatV2 bool `ocscapability:"chat-v2"`
|
|
Favorites bool `ocscapability:"favorites"`
|
|
LastRoomActivity bool `ocscapability:"last-room-activity"`
|
|
NoPing bool `ocscapability:"no-ping"`
|
|
SystemMessages bool `ocscapability:"system-messages"`
|
|
MentionFlag bool `ocscapability:"mention-flag"`
|
|
InCallFlags bool `ocscapability:"in-call-flags"`
|
|
InviteByMail bool `ocscapability:"invite-by-mail"`
|
|
NotificationLevels bool `ocscapability:"notification-levels"`
|
|
InviteGroupsAndMails bool `ocscapability:"invite-groups-and-mails"`
|
|
LockedOneToOneRooms bool `ocscapability:"locked-one-to-one-rooms"`
|
|
ReadOnlyRooms bool `ocscapability:"read-only-rooms"`
|
|
ChatReadMarker bool `ocscapability:"chat-read-marker"`
|
|
WebinaryLobby bool `ocscapability:"webinary-lobby"`
|
|
StartCallFlag bool `ocscapability:"start-call-flag"`
|
|
ChatReplies bool `ocscapability:"chat-replies"`
|
|
CirclesSupport bool `ocscapability:"circles-support"`
|
|
AttachmentsAllowed bool `ocscapability:"config => attachments => allowed"`
|
|
ConversationsCanCreate bool `ocscapability:"config => conversations => can-create"`
|
|
ForceMute bool `ocscapability:"force-mute"`
|
|
ConversationV2 bool `ocscapability:"conversation-v2"`
|
|
ChatReferenceID bool `ocscapability:"chat-reference-id"`
|
|
ConversationV3 bool `ocscapability:"conversation-v3"`
|
|
ConversationV4 bool `ocscapability:"conversation-v4"`
|
|
SIPSupport bool `ocscapability:"sip-support"`
|
|
ChatReadStatus bool `ocscapability:"chat-read-status"`
|
|
ListableRooms bool `ocscapability:"listable-rooms"`
|
|
PhonebookSearch bool `ocscapability:"phonebook-search"`
|
|
RaiseHand bool `ocscapability:"raise-hand"`
|
|
RoomDescription bool `ocscapability:"room-description"`
|
|
DeleteMessages bool `ocscapability:"delete-messages"`
|
|
RichObjectSharing bool `ocscapability:"rich-object-sharing"`
|
|
ConversationCallFlags bool `ocscapability:"conversation-call-flags"`
|
|
GeoLocationSharing bool `ocscapability:"geo-location-sharing"`
|
|
ReadPrivacyConfig bool `ocscapability:"config => chat => read-privacy"`
|
|
SignalingV3 bool `ocscapability:"signaling-v3"`
|
|
TempUserAvatarAPI bool `ocscapability:"temp-user-avatar-api"`
|
|
MaxGifSizeConfig int `ocscapability:"config => previews => max-gif-size"`
|
|
ChatMaxLength int `ocscapability:"config => chat => max-length"`
|
|
}
|
|
|
|
// RoomInfo contains information about a room
|
|
type RoomInfo struct {
|
|
Token string `json:"token"`
|
|
Name string `json:"name"`
|
|
DisplayName string `json:"displayName"`
|
|
SessionID string `json:"sessionId"`
|
|
ObjectType string `json:"objectType"`
|
|
ObjectID string `json:"objectId"`
|
|
Type int `json:"type"`
|
|
ParticipantType int `json:"participantType"`
|
|
ParticipantFlags int `json:"participantFlags"`
|
|
ReadOnly int `json:"readOnly"`
|
|
LastPing int `json:"lastPing"`
|
|
LastActivity int `json:"lastActivity"`
|
|
NotificationLevel int `json:"notificationLevel"`
|
|
LobbyState int `json:"lobbyState"`
|
|
LobbyTimer int `json:"lobbyTimer"`
|
|
UnreadMessages int `json:"unreadMessages"`
|
|
LastReadMessage int `json:"lastReadMessage"`
|
|
HasPassword bool `json:"hasPassword"`
|
|
HasCall bool `json:"hasCall"`
|
|
CanStartCall bool `json:"canStartCall"`
|
|
CanDeleteConversation bool `json:"canDeleteConversation"`
|
|
CanLeaveConversation bool `json:"canLeaveConversation"`
|
|
IsFavorite bool `json:"isFavorite"`
|
|
UnreadMention bool `json:"unreadMention"`
|
|
LastMessage *ocs.TalkRoomMessageData `json:"lastMessage"`
|
|
}
|
|
|
|
// NewUser returns a TalkUser instance
|
|
// The url should be the full URL of the Nextcloud instance (e.g. https://cloud.mydomain.me)
|
|
func NewUser(url string, username string, password string, config *TalkUserConfig) (*TalkUser, error) {
|
|
return &TalkUser{
|
|
NextcloudURL: strings.TrimSuffix(url, "/"),
|
|
User: username,
|
|
Pass: password,
|
|
Config: config,
|
|
}, nil
|
|
}
|
|
|
|
// RequestClient returns a monaco-io that is preconfigured to make OCS API calls
|
|
func (t *TalkUser) RequestClient(client request.Client) *request.Client {
|
|
if client.Header == nil {
|
|
client.Header = make(map[string]string)
|
|
}
|
|
if client.Header["OCS-APIRequest"] == "" {
|
|
client.Header["OCS-APIRequest"] = "true"
|
|
}
|
|
if client.Header["Accept"] == "" {
|
|
client.Header["Accept"] = "application/json"
|
|
}
|
|
client.BasicAuth = request.BasicAuth{
|
|
Username: t.User,
|
|
Password: t.Pass,
|
|
}
|
|
|
|
// Set Nextcloud URL if there is no host
|
|
if !strings.HasPrefix(client.URL, t.NextcloudURL) {
|
|
if strings.HasPrefix(client.URL, "/") {
|
|
client.URL = t.NextcloudURL + client.URL
|
|
} else {
|
|
client.URL = t.NextcloudURL + "/" + client.URL
|
|
}
|
|
}
|
|
|
|
// Set TLS Config
|
|
if t.Config != nil {
|
|
client.TLSConfig = t.Config.TLSConfig
|
|
}
|
|
|
|
return &client
|
|
}
|
|
|
|
// GetRooms returns a list of all rooms the user is in
|
|
func (t *TalkUser) GetRooms() (*[]RoomInfo, error) {
|
|
endpoint := ocsRoomsv2Endpoint
|
|
capabilities, err := t.Capabilities()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if capabilities.ConversationV4 {
|
|
endpoint = ocsRoomsv4Endpoint
|
|
}
|
|
|
|
client := t.RequestClient(request.Client{
|
|
URL: endpoint,
|
|
})
|
|
res, err := client.Do()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var roomsRequest struct {
|
|
OCS struct {
|
|
Data []RoomInfo `json:"data"`
|
|
} `json:"ocs"`
|
|
}
|
|
|
|
err = json.Unmarshal(res.Data, &roomsRequest)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &roomsRequest.OCS.Data, nil
|
|
}
|
|
|
|
// Capabilities returns an instance of Capabilities that describes what the Nextcloud Talk instance supports
|
|
func (t *TalkUser) Capabilities() (*Capabilities, error) {
|
|
if t.capabilities != nil {
|
|
return t.capabilities, nil
|
|
}
|
|
|
|
client := t.RequestClient(request.Client{
|
|
URL: ocsCapabilitiesEndpoint,
|
|
})
|
|
res, err := client.Do()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
capabilitiesRequest := &struct {
|
|
Ocs ocs.Capabilities `json:"ocs"`
|
|
}{}
|
|
|
|
err = json.Unmarshal(res.Data, capabilitiesRequest)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
sc := capabilitiesRequest.Ocs.Data.Capabilities.SpreedCapabilities
|
|
|
|
tr := &Capabilities{
|
|
Audio: sliceContains(sc.Features, "audio"),
|
|
Video: sliceContains(sc.Features, "video"),
|
|
Chat: sliceContains(sc.Features, "chat"),
|
|
GuestSignaling: sliceContains(sc.Features, "guest-signaling"),
|
|
EmptyGroupRoom: sliceContains(sc.Features, "empty-group-room"),
|
|
GuestDisplayNames: sliceContains(sc.Features, "guest-display-names"),
|
|
MultiRoomUsers: sliceContains(sc.Features, "multi-room-users"),
|
|
ChatV2: sliceContains(sc.Features, "chat-v2"),
|
|
Favorites: sliceContains(sc.Features, "favorites"),
|
|
LastRoomActivity: sliceContains(sc.Features, "last-room-activity"),
|
|
NoPing: sliceContains(sc.Features, "no-ping"),
|
|
SystemMessages: sliceContains(sc.Features, "system-messages"),
|
|
MentionFlag: sliceContains(sc.Features, "mention-flag"),
|
|
InCallFlags: sliceContains(sc.Features, "in-call-flags"),
|
|
InviteByMail: sliceContains(sc.Features, "invite-by-mail"),
|
|
NotificationLevels: sliceContains(sc.Features, "notification-levels"),
|
|
InviteGroupsAndMails: sliceContains(sc.Features, "invite-groups-and-mails"),
|
|
LockedOneToOneRooms: sliceContains(sc.Features, "locked-one-to-one-rooms"),
|
|
ReadOnlyRooms: sliceContains(sc.Features, "read-only-rooms"),
|
|
ChatReadMarker: sliceContains(sc.Features, "chat-read-marker"),
|
|
WebinaryLobby: sliceContains(sc.Features, "webinary-lobby"),
|
|
StartCallFlag: sliceContains(sc.Features, "start-call-flag"),
|
|
ChatReplies: sliceContains(sc.Features, "chat-replies"),
|
|
CirclesSupport: sliceContains(sc.Features, "circles-support"),
|
|
AttachmentsAllowed: sc.Config.Attachments.Allowed,
|
|
AttachmentsFolder: sc.Config.Attachments.Folder,
|
|
ConversationsCanCreate: sc.Config.Conversations.CanCreate,
|
|
ForceMute: sliceContains(sc.Features, "force-mute"),
|
|
ConversationV2: sliceContains(sc.Features, "conversation-v2"),
|
|
ChatReferenceID: sliceContains(sc.Features, "chat-reference-id"),
|
|
ChatMaxLength: sc.Config.Chat.MaxLength,
|
|
ConversationV3: sliceContains(sc.Features, "conversation-v3"),
|
|
ConversationV4: sliceContains(sc.Features, "conversation-v4"),
|
|
SIPSupport: sliceContains(sc.Features, "sip-support"),
|
|
ChatReadStatus: sliceContains(sc.Features, "chat-read-status"),
|
|
ListableRooms: sliceContains(sc.Features, "listable-rooms"),
|
|
PhonebookSearch: sliceContains(sc.Features, "phonebook-search"),
|
|
RaiseHand: sliceContains(sc.Features, "raise-hand"),
|
|
RoomDescription: sliceContains(sc.Features, "room-description"),
|
|
ReadPrivacyConfig: sc.Config.Chat.ReadPrivacy != 0,
|
|
MaxGifSizeConfig: sc.Config.Previews.MaxGifSize,
|
|
DeleteMessages: sliceContains(sc.Features, "delete-messages"),
|
|
RichObjectSharing: sliceContains(sc.Features, "rich-object-sharing"),
|
|
ConversationCallFlags: sliceContains(sc.Features, "conversation-call-flags"),
|
|
GeoLocationSharing: sliceContains(sc.Features, "geo-location-sharing"),
|
|
SignalingV3: sliceContains(sc.Features, "signaling-v3"),
|
|
TempUserAvatarAPI: sliceContains(sc.Features, "temp-user-avatar-api"),
|
|
}
|
|
|
|
t.capabilities = tr
|
|
return tr, nil
|
|
}
|
|
|
|
// sliceContains does the slice contain the string
|
|
func sliceContains(s []string, search string) bool {
|
|
for _, n := range s {
|
|
if n == search {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// DownloadFile downloads the file at the given path
|
|
//
|
|
// Meant to be used with rich object string's path.
|
|
func (t *TalkUser) DownloadFile(path string) (data *[]byte, err error) {
|
|
url := t.NextcloudURL + constants.RemoteDavEndpoint(t.User, "files") + path
|
|
c := t.RequestClient(request.Client{
|
|
URL: url,
|
|
})
|
|
res, err := c.Do()
|
|
if err != nil {
|
|
return
|
|
}
|
|
if res.StatusCode() != 200 {
|
|
err = ErrCannotDownloadFile
|
|
return
|
|
}
|
|
data = &res.Data
|
|
return
|
|
}
|