aboutsummaryrefslogtreecommitdiff
path: root/lib/uri.h
blob: 270766dd529292ccadfae36e1f08b9a0f507f964 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
#pragma once

#include "quotient_common.h"

#include <QtCore/QUrl>
#include <QtCore/QUrlQuery>

namespace Quotient {

/*! \brief A wrapper around a Matrix URI or identifier
 *
 * This class encapsulates a Matrix resource identifier, passed in either of
 * 3 forms: a plain Matrix identifier (sigil, localpart, serverpart or, for
 * modern event ids, sigil and base64 hash); an MSC2312 URI (aka matrix: URI);
 * or a matrix.to URL. The input can be either encoded (serverparts with
 * punycode, the rest with percent-encoding) or unencoded (in this case it is
 * the caller's responsibility to resolve all possible ambiguities).
 *
 * The class provides functions to check the validity of the identifier,
 * its type, and obtain components, also in either unencoded (for displaying)
 * or encoded (for APIs) form.
 */
class Uri : private QUrl {
    Q_GADGET
public:
    enum Type : char {
        Invalid = char(-1),
        Empty = 0x0,
        UserId = '@',
        RoomId = '!',
        RoomAlias = '#',
        Group = '+',
        BareEventId = '$', // https://github.com/matrix-org/matrix-doc/pull/2644
        NonMatrix = ':'
    };
    Q_ENUM(Type)
    enum SecondaryType : char { NoSecondaryId = 0x0, EventId = '$' };
    Q_ENUM(SecondaryType)

    enum UriForm : short { CanonicalUri, MatrixToUri };
    Q_ENUM(UriForm)

    /// Construct an empty Matrix URI
    Uri() = default;
    /*! \brief Decode a user input string to a Matrix identifier
     *
     * Accepts plain Matrix ids, MSC2312 URIs (aka matrix: URIs) and
     * matrix.to URLs. In case of URIs/URLs, it uses QUrl's TolerantMode
     * parser to decode common mistakes/irregularities (see QUrl documentation
     * for more details).
     */
    Uri(const QString& uriOrId);

    /// Construct a Matrix URI from components
    explicit Uri(QByteArray primaryId, QByteArray secondaryId = {},
                 QString query = {});
    /// Construct a Matrix URI from matrix.to or MSC2312 (matrix:) URI
    explicit Uri(QUrl url);

    static Uri fromUserInput(const QString& uriOrId);
    static Uri fromUrl(QUrl url);

    /// Get the primary type of the Matrix URI (user id, room id or alias)
    /*! Note that this does not include an event as a separate type, since
     * events can only be addressed inside of rooms, which, in turn, are
     * addressed either by id or alias. If you need to check whether the URI
     * is specifically an event URI, use secondaryType() instead.
     */
    Q_INVOKABLE Type type() const;
    Q_INVOKABLE SecondaryType secondaryType() const;
    Q_INVOKABLE QUrl toUrl(UriForm form = CanonicalUri) const;
    Q_INVOKABLE QString primaryId() const;
    Q_INVOKABLE QString secondaryId() const;
    Q_INVOKABLE QString action() const;
    Q_INVOKABLE void setAction(const QString& newAction);
    Q_INVOKABLE QStringList viaServers() const;
    Q_INVOKABLE bool isValid() const;
    using QUrl::path, QUrl::query, QUrl::fragment;
    using QUrl::isEmpty, QUrl::toDisplayString;

private:
    Type primaryType_ = Empty;
};

}