mirror of
https://github.com/matrix-construct/construct
synced 2024-11-18 07:50:57 +01:00
218 lines
4.9 KiB
C++
218 lines
4.9 KiB
C++
/*
|
|
* charybdis: 21st Century IRC++d
|
|
*
|
|
* Copyright (C) 2016 Charybdis Development Team
|
|
* Copyright (C) 2016 Jason Volk <jason@zemos.net>
|
|
*
|
|
* Permission to use, copy, modify, and/or distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice is present in all copies.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
|
|
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
*/
|
|
|
|
#pragma once
|
|
#define HAVE_IRCD_M_ID_H
|
|
|
|
namespace ircd::m
|
|
{
|
|
IRCD_M_EXCEPTION(error, INVALID_MXID, http::BAD_REQUEST)
|
|
IRCD_M_EXCEPTION(INVALID_MXID, BAD_SIGIL, http::BAD_REQUEST)
|
|
|
|
IRCD_OVERLOAD(generate)
|
|
|
|
struct id;
|
|
}
|
|
|
|
//
|
|
// Interface to a string representing an mxid. The m::id itself is just a
|
|
// string_view over some existing data. m::id::buf is an m::id with an
|
|
// internal array providing the buffer.
|
|
//
|
|
struct ircd::m::id
|
|
:string_view
|
|
{
|
|
struct event;
|
|
struct user;
|
|
struct room;
|
|
struct alias;
|
|
|
|
template<class T, size_t SIZE = 256> struct buf;
|
|
|
|
struct parser;
|
|
struct generator;
|
|
enum sigil :char;
|
|
|
|
protected:
|
|
struct parser static const parser;
|
|
|
|
public:
|
|
// Extract elements
|
|
string_view local() const { return split(*this, ':').first; }
|
|
string_view host() const { return split(*this, ':').second; }
|
|
string_view name() const { return lstrip(local(), at(0)); }
|
|
string_view hostname() const;
|
|
uint16_t hostport() const;
|
|
|
|
IRCD_USING_OVERLOAD(generate, m::generate);
|
|
|
|
id(const enum sigil &, const mutable_buffer &, const generate_t &, const string_view &host);
|
|
id(const enum sigil &, const mutable_buffer &, const string_view &name, const string_view &host);
|
|
id(const enum sigil &, const mutable_buffer &, const string_view &id);
|
|
id(const enum sigil &, const string_view &id);
|
|
id(const string_view &id);
|
|
id() = default;
|
|
};
|
|
|
|
enum ircd::m::id::sigil
|
|
:char
|
|
{
|
|
EVENT = '$',
|
|
USER = '@',
|
|
ROOM = '!',
|
|
ALIAS = '#',
|
|
};
|
|
|
|
namespace ircd::m
|
|
{
|
|
id::sigil sigil(const char &c);
|
|
id::sigil sigil(const string_view &id);
|
|
const char *reflect(const id::sigil &);
|
|
|
|
// Checks
|
|
bool valid_sigil(const char &c);
|
|
bool valid_sigil(const string_view &id);
|
|
bool valid(const id::sigil &, const string_view &) noexcept;
|
|
bool valid_local(const id::sigil &, const string_view &); // Local part is valid
|
|
void validate(const id::sigil &, const string_view &); // valid() | throws
|
|
}
|
|
|
|
//
|
|
// convenience typedefs
|
|
//
|
|
|
|
struct ircd::m::id::event
|
|
:ircd::m::id
|
|
{
|
|
using buf = m::id::buf<event>;
|
|
template<class... args> event(args&&... a) :m::id{EVENT, std::forward<args>(a)...} {}
|
|
event() = default;
|
|
};
|
|
|
|
struct ircd::m::id::user
|
|
:ircd::m::id
|
|
{
|
|
using buf = m::id::buf<user>;
|
|
template<class... args> user(args&&... a) :m::id{USER, std::forward<args>(a)...} {}
|
|
user() = default;
|
|
};
|
|
|
|
struct ircd::m::id::room
|
|
:ircd::m::id
|
|
{
|
|
using buf = m::id::buf<room>;
|
|
template<class... args> room(args&&... a) :m::id{ROOM, std::forward<args>(a)...} {}
|
|
room() = default;
|
|
};
|
|
|
|
struct ircd::m::id::alias
|
|
:ircd::m::id
|
|
{
|
|
using buf = m::id::buf<alias>;
|
|
template<class... args> alias(args&&... a) :m::id{ALIAS, std::forward<args>(a)...} {}
|
|
alias() = default;
|
|
};
|
|
|
|
/// ID object backed by an internal buffer of wost-case size.
|
|
///
|
|
template<class T,
|
|
size_t MAX = 256>
|
|
struct ircd::m::id::buf
|
|
:T
|
|
{
|
|
static constexpr const size_t SIZE
|
|
{
|
|
MAX
|
|
};
|
|
|
|
private:
|
|
fixed_buffer<mutable_buffer, SIZE> b;
|
|
|
|
public:
|
|
operator const fixed_buffer<mutable_buffer, SIZE> &() const
|
|
{
|
|
return b;
|
|
}
|
|
|
|
operator mutable_buffer()
|
|
{
|
|
return b;
|
|
}
|
|
|
|
template<class... args>
|
|
buf(args&&... a)
|
|
:T{b, std::forward<args>(a)...}
|
|
{}
|
|
|
|
buf() = default;
|
|
|
|
buf(buf &&other) noexcept
|
|
:T{b, std::move(other)}
|
|
{}
|
|
|
|
buf(const buf &other)
|
|
:T{b, other}
|
|
{}
|
|
|
|
buf &operator=(const buf &other)
|
|
{
|
|
static_cast<T &>(*this) = T{b, other};
|
|
return *this;
|
|
}
|
|
|
|
buf &operator=(buf &&other) noexcept
|
|
{
|
|
static_cast<T &>(*this) = T{b, std::move(other)};
|
|
return *this;
|
|
}
|
|
|
|
buf &operator=(const string_view &s)
|
|
{
|
|
static_cast<T &>(*this) = T{b, s};
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
inline uint16_t
|
|
ircd::m::id::hostport()
|
|
const try
|
|
{
|
|
const auto port
|
|
{
|
|
split(host(), ':').second
|
|
};
|
|
|
|
return port? lex_cast<uint16_t>(port) : 8448;
|
|
}
|
|
catch(const std::exception &e)
|
|
{
|
|
return 8448;
|
|
}
|
|
|
|
inline ircd::string_view
|
|
ircd::m::id::hostname()
|
|
const
|
|
{
|
|
return split(host(), ':').first;
|
|
}
|