mirror of
https://github.com/cwinfo/matterbridge.git
synced 2025-01-14 06:16:28 +00:00
2f33fe86f5
* Update dependencies and build to go1.22 * Fix api changes wrt to dependencies * Update golangci config
258 lines
10 KiB
Go
258 lines
10 KiB
Go
// Copyright (c) 2021 Tulir Asokan
|
|
//
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
package whatsmeow
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"net/http"
|
|
|
|
waBinary "go.mau.fi/whatsmeow/binary"
|
|
)
|
|
|
|
// Miscellaneous errors
|
|
var (
|
|
ErrNoSession = errors.New("can't encrypt message for device: no signal session established")
|
|
ErrIQTimedOut = errors.New("info query timed out")
|
|
ErrNotConnected = errors.New("websocket not connected")
|
|
ErrNotLoggedIn = errors.New("the store doesn't contain a device JID")
|
|
ErrMessageTimedOut = errors.New("timed out waiting for message send response")
|
|
|
|
ErrAlreadyConnected = errors.New("websocket is already connected")
|
|
|
|
ErrQRAlreadyConnected = errors.New("GetQRChannel must be called before connecting")
|
|
ErrQRStoreContainsID = errors.New("GetQRChannel can only be called when there's no user ID in the client's Store")
|
|
|
|
ErrNoPushName = errors.New("can't send presence without PushName set")
|
|
|
|
ErrNoPrivacyToken = errors.New("no privacy token stored")
|
|
|
|
ErrAppStateUpdate = errors.New("server returned error updating app state")
|
|
)
|
|
|
|
// Errors that happen while confirming device pairing
|
|
var (
|
|
ErrPairInvalidDeviceIdentityHMAC = errors.New("invalid device identity HMAC in pair success message")
|
|
ErrPairInvalidDeviceSignature = errors.New("invalid device signature in pair success message")
|
|
ErrPairRejectedLocally = errors.New("local PrePairCallback rejected pairing")
|
|
)
|
|
|
|
// PairProtoError is included in an events.PairError if the pairing failed due to a protobuf error.
|
|
type PairProtoError struct {
|
|
Message string
|
|
ProtoErr error
|
|
}
|
|
|
|
func (err *PairProtoError) Error() string {
|
|
return fmt.Sprintf("%s: %v", err.Message, err.ProtoErr)
|
|
}
|
|
|
|
func (err *PairProtoError) Unwrap() error {
|
|
return err.ProtoErr
|
|
}
|
|
|
|
// PairDatabaseError is included in an events.PairError if the pairing failed due to being unable to save the credentials to the device store.
|
|
type PairDatabaseError struct {
|
|
Message string
|
|
DBErr error
|
|
}
|
|
|
|
func (err *PairDatabaseError) Error() string {
|
|
return fmt.Sprintf("%s: %v", err.Message, err.DBErr)
|
|
}
|
|
|
|
func (err *PairDatabaseError) Unwrap() error {
|
|
return err.DBErr
|
|
}
|
|
|
|
var (
|
|
// ErrProfilePictureUnauthorized is returned by GetProfilePictureInfo when trying to get the profile picture of a user
|
|
// whose privacy settings prevent you from seeing their profile picture (status code 401).
|
|
ErrProfilePictureUnauthorized = errors.New("the user has hidden their profile picture from you")
|
|
// ErrProfilePictureNotSet is returned by GetProfilePictureInfo when the given user or group doesn't have a profile
|
|
// picture (status code 404).
|
|
ErrProfilePictureNotSet = errors.New("that user or group does not have a profile picture")
|
|
// ErrGroupInviteLinkUnauthorized is returned by GetGroupInviteLink if you don't have the permission to get the link (status code 401).
|
|
ErrGroupInviteLinkUnauthorized = errors.New("you don't have the permission to get the group's invite link")
|
|
// ErrNotInGroup is returned by group info getting methods if you're not in the group (status code 403).
|
|
ErrNotInGroup = errors.New("you're not participating in that group")
|
|
// ErrGroupNotFound is returned by group info getting methods if the group doesn't exist (status code 404).
|
|
ErrGroupNotFound = errors.New("that group does not exist")
|
|
// ErrInviteLinkInvalid is returned by methods that use group invite links if the invite link is malformed.
|
|
ErrInviteLinkInvalid = errors.New("that group invite link is not valid")
|
|
// ErrInviteLinkRevoked is returned by methods that use group invite links if the invite link was valid, but has been revoked and can no longer be used.
|
|
ErrInviteLinkRevoked = errors.New("that group invite link has been revoked")
|
|
// ErrBusinessMessageLinkNotFound is returned by ResolveBusinessMessageLink if the link doesn't exist or has been revoked.
|
|
ErrBusinessMessageLinkNotFound = errors.New("that business message link does not exist or has been revoked")
|
|
// ErrContactQRLinkNotFound is returned by ResolveContactQRLink if the link doesn't exist or has been revoked.
|
|
ErrContactQRLinkNotFound = errors.New("that contact QR link does not exist or has been revoked")
|
|
// ErrInvalidImageFormat is returned by SetGroupPhoto if the given photo is not in the correct format.
|
|
ErrInvalidImageFormat = errors.New("the given data is not a valid image")
|
|
// ErrMediaNotAvailableOnPhone is returned by DecryptMediaRetryNotification if the given event contains error code 2.
|
|
ErrMediaNotAvailableOnPhone = errors.New("media no longer available on phone")
|
|
// ErrUnknownMediaRetryError is returned by DecryptMediaRetryNotification if the given event contains an unknown error code.
|
|
ErrUnknownMediaRetryError = errors.New("unknown media retry error")
|
|
// ErrInvalidDisappearingTimer is returned by SetDisappearingTimer if the given timer is not one of the allowed values.
|
|
ErrInvalidDisappearingTimer = errors.New("invalid disappearing timer provided")
|
|
)
|
|
|
|
// Some errors that Client.SendMessage can return
|
|
var (
|
|
ErrBroadcastListUnsupported = errors.New("sending to non-status broadcast lists is not yet supported")
|
|
ErrUnknownServer = errors.New("can't send message to unknown server")
|
|
ErrRecipientADJID = errors.New("message recipient must be a user JID with no device part")
|
|
ErrServerReturnedError = errors.New("server returned error")
|
|
)
|
|
|
|
type DownloadHTTPError struct {
|
|
*http.Response
|
|
}
|
|
|
|
func (dhe DownloadHTTPError) Error() string {
|
|
return fmt.Sprintf("download failed with status code %d", dhe.StatusCode)
|
|
}
|
|
|
|
func (dhe DownloadHTTPError) Is(other error) bool {
|
|
var otherDHE DownloadHTTPError
|
|
return errors.As(other, &otherDHE) && dhe.StatusCode == otherDHE.StatusCode
|
|
}
|
|
|
|
// Some errors that Client.Download can return
|
|
var (
|
|
ErrMediaDownloadFailedWith403 = DownloadHTTPError{Response: &http.Response{StatusCode: 403}}
|
|
ErrMediaDownloadFailedWith404 = DownloadHTTPError{Response: &http.Response{StatusCode: 404}}
|
|
ErrMediaDownloadFailedWith410 = DownloadHTTPError{Response: &http.Response{StatusCode: 410}}
|
|
ErrNoURLPresent = errors.New("no url present")
|
|
ErrFileLengthMismatch = errors.New("file length does not match")
|
|
ErrTooShortFile = errors.New("file too short")
|
|
ErrInvalidMediaHMAC = errors.New("invalid media hmac")
|
|
ErrInvalidMediaEncSHA256 = errors.New("hash of media ciphertext doesn't match")
|
|
ErrInvalidMediaSHA256 = errors.New("hash of media plaintext doesn't match")
|
|
ErrUnknownMediaType = errors.New("unknown media type")
|
|
ErrNothingDownloadableFound = errors.New("didn't find any attachments in message")
|
|
)
|
|
|
|
var (
|
|
ErrOriginalMessageSecretNotFound = errors.New("original message secret key not found")
|
|
ErrNotEncryptedReactionMessage = errors.New("given message isn't an encrypted reaction message")
|
|
ErrNotPollUpdateMessage = errors.New("given message isn't a poll update message")
|
|
)
|
|
|
|
type wrappedIQError struct {
|
|
HumanError error
|
|
IQError error
|
|
}
|
|
|
|
func (err *wrappedIQError) Error() string {
|
|
return err.HumanError.Error()
|
|
}
|
|
|
|
func (err *wrappedIQError) Is(other error) bool {
|
|
return errors.Is(other, err.HumanError)
|
|
}
|
|
|
|
func (err *wrappedIQError) Unwrap() error {
|
|
return err.IQError
|
|
}
|
|
|
|
func wrapIQError(human, iq error) error {
|
|
return &wrappedIQError{human, iq}
|
|
}
|
|
|
|
// IQError is a generic error container for info queries
|
|
type IQError struct {
|
|
Code int
|
|
Text string
|
|
ErrorNode *waBinary.Node
|
|
RawNode *waBinary.Node
|
|
}
|
|
|
|
// Common errors returned by info queries for use with errors.Is
|
|
var (
|
|
ErrIQBadRequest error = &IQError{Code: 400, Text: "bad-request"}
|
|
ErrIQNotAuthorized error = &IQError{Code: 401, Text: "not-authorized"}
|
|
ErrIQForbidden error = &IQError{Code: 403, Text: "forbidden"}
|
|
ErrIQNotFound error = &IQError{Code: 404, Text: "item-not-found"}
|
|
ErrIQNotAllowed error = &IQError{Code: 405, Text: "not-allowed"}
|
|
ErrIQNotAcceptable error = &IQError{Code: 406, Text: "not-acceptable"}
|
|
ErrIQGone error = &IQError{Code: 410, Text: "gone"}
|
|
ErrIQResourceLimit error = &IQError{Code: 419, Text: "resource-limit"}
|
|
ErrIQLocked error = &IQError{Code: 423, Text: "locked"}
|
|
ErrIQInternalServerError error = &IQError{Code: 500, Text: "internal-server-error"}
|
|
ErrIQServiceUnavailable error = &IQError{Code: 503, Text: "service-unavailable"}
|
|
ErrIQPartialServerError error = &IQError{Code: 530, Text: "partial-server-error"}
|
|
)
|
|
|
|
func parseIQError(node *waBinary.Node) error {
|
|
var err IQError
|
|
err.RawNode = node
|
|
val, ok := node.GetOptionalChildByTag("error")
|
|
if ok {
|
|
err.ErrorNode = &val
|
|
ag := val.AttrGetter()
|
|
err.Code = ag.OptionalInt("code")
|
|
err.Text = ag.OptionalString("text")
|
|
}
|
|
return &err
|
|
}
|
|
|
|
func (iqe *IQError) Error() string {
|
|
if iqe.Code == 0 {
|
|
if iqe.ErrorNode != nil {
|
|
return fmt.Sprintf("info query returned unknown error: %s", iqe.ErrorNode.XMLString())
|
|
} else if iqe.RawNode != nil {
|
|
return fmt.Sprintf("info query returned unexpected response: %s", iqe.RawNode.XMLString())
|
|
} else {
|
|
return "unknown info query error"
|
|
}
|
|
}
|
|
return fmt.Sprintf("info query returned status %d: %s", iqe.Code, iqe.Text)
|
|
}
|
|
|
|
func (iqe *IQError) Is(other error) bool {
|
|
otherIQE, ok := other.(*IQError)
|
|
if !ok {
|
|
return false
|
|
} else if iqe.Code != 0 && otherIQE.Code != 0 {
|
|
return otherIQE.Code == iqe.Code && otherIQE.Text == iqe.Text
|
|
} else if iqe.ErrorNode != nil && otherIQE.ErrorNode != nil {
|
|
return iqe.ErrorNode.XMLString() == otherIQE.ErrorNode.XMLString()
|
|
} else {
|
|
return false
|
|
}
|
|
}
|
|
|
|
// ElementMissingError is returned by various functions that parse XML elements when a required element is missing.
|
|
type ElementMissingError struct {
|
|
Tag string
|
|
In string
|
|
}
|
|
|
|
func (eme *ElementMissingError) Error() string {
|
|
return fmt.Sprintf("missing <%s> element in %s", eme.Tag, eme.In)
|
|
}
|
|
|
|
var ErrIQDisconnected = &DisconnectedError{Action: "info query"}
|
|
|
|
// DisconnectedError is returned if the websocket disconnects before an info query or other request gets a response.
|
|
type DisconnectedError struct {
|
|
Action string
|
|
Node *waBinary.Node
|
|
}
|
|
|
|
func (err *DisconnectedError) Error() string {
|
|
return fmt.Sprintf("websocket disconnected before %s returned response", err.Action)
|
|
}
|
|
|
|
func (err *DisconnectedError) Is(other error) bool {
|
|
otherDisc, ok := other.(*DisconnectedError)
|
|
if !ok {
|
|
return false
|
|
}
|
|
return otherDisc.Action == err.Action
|
|
}
|