// 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
}