2018-02-03 18:22:01 -08:00
|
|
|
// Matrix Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
|
|
// Copyright (C) 2016-2018 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. The
|
|
|
|
// full license for this software is available in the LICENSE file.
|
2017-08-23 15:10:28 -06:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
#define HAVE_IRCD_M_EVENT_H
|
|
|
|
|
2017-09-08 02:32:49 -07:00
|
|
|
namespace ircd::m
|
|
|
|
{
|
|
|
|
struct event;
|
2017-11-15 17:48:25 -08:00
|
|
|
|
2018-03-08 16:35:39 -08:00
|
|
|
extern conf::item<size_t> event_max_size;
|
|
|
|
|
2017-11-15 17:48:25 -08:00
|
|
|
bool my(const id::event &);
|
|
|
|
bool my(const event &);
|
|
|
|
|
|
|
|
size_t degree(const event &);
|
2018-03-03 06:03:15 -08:00
|
|
|
string_view membership(const event &);
|
2018-01-20 05:48:39 -08:00
|
|
|
|
2018-03-02 23:56:21 -08:00
|
|
|
bool check_size(std::nothrow_t, const event &);
|
|
|
|
void check_size(const event &);
|
|
|
|
|
2017-11-15 17:48:25 -08:00
|
|
|
std::string pretty(const event &);
|
|
|
|
std::string pretty_oneline(const event &);
|
2017-09-08 02:32:49 -07:00
|
|
|
|
2018-02-02 23:20:26 -08:00
|
|
|
id::event event_id(const event &, id::event::buf &buf, const const_buffer &hash);
|
2018-01-20 05:48:39 -08:00
|
|
|
id::event event_id(const event &, id::event::buf &buf);
|
|
|
|
id::event event_id(const event &);
|
2018-02-27 20:47:19 -08:00
|
|
|
|
|
|
|
// [GET]
|
|
|
|
bool exists(const id::event &);
|
2017-09-08 02:32:49 -07:00
|
|
|
}
|
2017-08-23 15:10:28 -06:00
|
|
|
|
2018-02-05 15:19:20 -08:00
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic ignored "-Wsubobject-linkage"
|
2018-02-08 13:23:01 -08:00
|
|
|
/// The Main Event
|
2017-11-15 17:48:25 -08:00
|
|
|
///
|
|
|
|
/// This json::tuple provides at least all of the legal members of the matrix
|
|
|
|
/// standard event. This is the fundamental building block of the matrix
|
|
|
|
/// system. Rooms are collections of events. Messages between servers are
|
2018-01-25 12:14:48 -08:00
|
|
|
/// passed as bundles of events (or directly).
|
|
|
|
///
|
|
|
|
/// It is better to have 100 functions operate on one data structure than
|
|
|
|
/// to have 10 functions operate on 10 data structures.
|
|
|
|
/// -Alan Perlis
|
2017-11-15 17:48:25 -08:00
|
|
|
///
|
2017-09-08 02:32:49 -07:00
|
|
|
struct ircd::m::event
|
2017-08-25 21:39:36 -07:00
|
|
|
:json::tuple
|
2017-08-23 15:10:28 -06:00
|
|
|
<
|
2017-11-15 17:48:25 -08:00
|
|
|
json::property<name::auth_events, json::array>,
|
2017-09-25 21:42:07 -07:00
|
|
|
json::property<name::content, json::object>,
|
2017-10-03 04:10:26 -07:00
|
|
|
json::property<name::depth, int64_t>,
|
2017-10-25 09:47:03 -07:00
|
|
|
json::property<name::event_id, json::string>,
|
|
|
|
json::property<name::hashes, json::object>,
|
|
|
|
json::property<name::membership, json::string>,
|
|
|
|
json::property<name::origin, json::string>,
|
2017-09-08 03:08:29 -07:00
|
|
|
json::property<name::origin_server_ts, time_t>,
|
2017-10-25 09:47:03 -07:00
|
|
|
json::property<name::prev_events, json::array>,
|
|
|
|
json::property<name::prev_state, json::array>,
|
2018-02-15 14:57:56 -08:00
|
|
|
json::property<name::redacts, json::string>,
|
2017-10-25 09:47:03 -07:00
|
|
|
json::property<name::room_id, json::string>,
|
|
|
|
json::property<name::sender, json::string>,
|
|
|
|
json::property<name::signatures, json::object>,
|
|
|
|
json::property<name::state_key, json::string>,
|
2018-02-08 13:23:01 -08:00
|
|
|
json::property<name::type, json::string>
|
2017-08-23 15:10:28 -06:00
|
|
|
>
|
|
|
|
{
|
2017-10-28 12:34:44 -07:00
|
|
|
struct prev;
|
2018-02-09 11:22:46 -08:00
|
|
|
struct fetch;
|
2018-02-21 13:43:33 -08:00
|
|
|
struct conforms;
|
2017-10-25 09:47:03 -07:00
|
|
|
|
2017-11-30 10:47:41 -08:00
|
|
|
// Common convenience aliases
|
2017-09-24 18:05:42 -07:00
|
|
|
using id = m::id::event;
|
2017-11-30 10:47:41 -08:00
|
|
|
using closure = std::function<void (const event &)>;
|
|
|
|
using closure_bool = std::function<bool (const event &)>;
|
2017-09-24 18:05:42 -07:00
|
|
|
|
2018-03-07 11:54:56 -08:00
|
|
|
static ed25519::sig sign(const m::event &);
|
|
|
|
static ed25519::sig sign(json::iov &event, const json::iov &content);
|
|
|
|
static string_view signatures(const mutable_buffer &, json::iov &event, const json::iov &content);
|
|
|
|
|
|
|
|
static sha256::buf hash(const m::event &);
|
|
|
|
static sha256::buf hash(json::iov &event, const string_view &content);
|
|
|
|
static string_view hashes(const mutable_buffer &, json::iov &event, const string_view &content);
|
|
|
|
|
2017-09-08 02:32:49 -07:00
|
|
|
using super_type::tuple;
|
2018-01-20 05:48:39 -08:00
|
|
|
using super_type::operator=;
|
|
|
|
|
2017-11-15 17:48:25 -08:00
|
|
|
event(const id &, const mutable_buffer &buf);
|
|
|
|
event() = default;
|
2017-08-23 15:10:28 -06:00
|
|
|
};
|
2018-02-05 15:19:20 -08:00
|
|
|
#pragma GCC diagnostic pop
|
2017-10-25 09:47:03 -07:00
|
|
|
|
2017-11-15 17:48:25 -08:00
|
|
|
namespace ircd::m
|
|
|
|
{
|
|
|
|
void for_each(const event::prev &, const std::function<void (const event::id &)> &);
|
|
|
|
size_t degree(const event::prev &);
|
|
|
|
size_t count(const event::prev &);
|
|
|
|
|
|
|
|
std::string pretty(const event::prev &);
|
|
|
|
std::string pretty_oneline(const event::prev &);
|
|
|
|
}
|
|
|
|
|
2018-02-05 15:19:20 -08:00
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic ignored "-Wsubobject-linkage"
|
2017-10-28 12:34:44 -07:00
|
|
|
struct ircd::m::event::prev
|
|
|
|
:json::tuple
|
|
|
|
<
|
|
|
|
json::property<name::auth_events, json::array>,
|
2017-11-15 17:48:25 -08:00
|
|
|
json::property<name::prev_state, json::array>,
|
|
|
|
json::property<name::prev_events, json::array>
|
2017-10-28 12:34:44 -07:00
|
|
|
>
|
|
|
|
{
|
|
|
|
enum cond :int;
|
|
|
|
|
2018-02-10 13:28:22 -08:00
|
|
|
std::tuple<event::id, string_view> auth_events(const uint &idx) const;
|
|
|
|
std::tuple<event::id, string_view> prev_states(const uint &idx) const;
|
|
|
|
std::tuple<event::id, string_view> prev_events(const uint &idx) const;
|
|
|
|
|
|
|
|
event::id auth_event(const uint &idx) const;
|
|
|
|
event::id prev_state(const uint &idx) const;
|
|
|
|
event::id prev_event(const uint &idx) const;
|
|
|
|
|
2017-10-28 12:34:44 -07:00
|
|
|
using super_type::tuple;
|
|
|
|
using super_type::operator=;
|
|
|
|
};
|
2018-02-05 15:19:20 -08:00
|
|
|
#pragma GCC diagnostic pop
|
|
|
|
|
2018-02-09 11:22:46 -08:00
|
|
|
struct ircd::m::event::fetch
|
|
|
|
:event
|
|
|
|
{
|
|
|
|
std::array<db::cell, event::size()> cell;
|
|
|
|
db::row row;
|
|
|
|
|
2018-02-10 00:39:13 -08:00
|
|
|
bool valid(const event::id &) const;
|
|
|
|
|
2018-02-09 11:22:46 -08:00
|
|
|
fetch(const event::id &, std::nothrow_t);
|
|
|
|
fetch(const event::id &);
|
|
|
|
fetch();
|
|
|
|
|
|
|
|
friend bool seek(fetch &, const event::id &, std::nothrow_t);
|
|
|
|
friend void seek(fetch &, const event::id &);
|
|
|
|
};
|
2018-02-20 18:09:39 -08:00
|
|
|
|
2018-03-09 07:11:00 -08:00
|
|
|
/// Device to evaluate the conformity of an event object. This is an 'in vitro'
|
|
|
|
/// or 'pure' evaluation: it determines if the event is reasonably sane enough
|
|
|
|
/// to be evaluated further using only the information in the event itself. It
|
|
|
|
/// requires nothing external and conducts no IO etc..
|
|
|
|
///
|
|
|
|
/// This evaluation does not throw or stop when a check fails: instead it
|
|
|
|
/// collects the failures allowing the user to further determine how to proceed
|
|
|
|
/// at their own discretion.
|
|
|
|
///
|
2018-02-21 13:43:33 -08:00
|
|
|
struct ircd::m::event::conforms
|
2018-02-20 18:09:39 -08:00
|
|
|
{
|
|
|
|
enum code :uint;
|
|
|
|
|
|
|
|
uint64_t report {0};
|
|
|
|
|
|
|
|
bool clean() const;
|
|
|
|
operator bool() const;
|
|
|
|
bool operator!() const;
|
|
|
|
bool has(const uint &code) const;
|
|
|
|
bool has(const code &code) const;
|
|
|
|
string_view string(const mutable_buffer &out) const;
|
|
|
|
|
|
|
|
void set(const code &code);
|
|
|
|
void del(const code &code);
|
2018-02-27 20:55:59 -08:00
|
|
|
void operator|=(const code &) &;
|
2018-02-20 18:09:39 -08:00
|
|
|
|
2018-02-21 13:43:33 -08:00
|
|
|
conforms() = default;
|
|
|
|
conforms(const event &);
|
2018-02-21 14:24:34 -08:00
|
|
|
conforms(const event &, const uint64_t &skip);
|
2018-02-20 18:09:39 -08:00
|
|
|
|
2018-02-21 15:00:02 -08:00
|
|
|
static code reflect(const string_view &);
|
2018-02-20 18:09:39 -08:00
|
|
|
friend string_view reflect(const code &);
|
2018-02-21 13:43:33 -08:00
|
|
|
friend std::ostream &operator<<(std::ostream &, const conforms &);
|
2018-02-20 18:09:39 -08:00
|
|
|
};
|
|
|
|
|
2018-03-09 07:11:00 -08:00
|
|
|
/// Report codes corresponding to the checks conducted by event::conforms.
|
|
|
|
/// Developers: If you add a code here you must also add a string reflection
|
|
|
|
/// in the definition file.
|
|
|
|
///
|
2018-02-21 13:43:33 -08:00
|
|
|
enum ircd::m::event::conforms::code
|
2018-02-20 18:09:39 -08:00
|
|
|
:uint
|
|
|
|
{
|
2018-02-21 14:24:34 -08:00
|
|
|
INVALID_OR_MISSING_EVENT_ID, ///< event_id empty or failed mxid grammar check
|
|
|
|
INVALID_OR_MISSING_ROOM_ID, ///< room_id empty or failed mxid grammar check
|
|
|
|
INVALID_OR_MISSING_SENDER_ID, ///< sender empty or failed mxid grammar check
|
|
|
|
MISSING_TYPE, ///< type empty
|
|
|
|
MISSING_ORIGIN, ///< origin empty
|
|
|
|
INVALID_ORIGIN, ///< origin not a proper domain
|
|
|
|
INVALID_OR_MISSING_REDACTS_ID, ///< for m.room.redaction
|
|
|
|
MISSING_MEMBERSHIP, ///< for m.room.member, membership empty
|
|
|
|
INVALID_MEMBERSHIP, ///< for m.room.member (does not check actual states)
|
|
|
|
MISSING_CONTENT_MEMBERSHIP, ///< for m.room.member, content.membership
|
|
|
|
INVALID_CONTENT_MEMBERSHIP, ///< for m.room.member, content.membership
|
|
|
|
MISSING_PREV_EVENTS, ///< for non-m.room.create, empty prev_events
|
|
|
|
MISSING_PREV_STATE, ///< for state_key'ed, empty prev_state
|
|
|
|
DEPTH_NEGATIVE, ///< depth < 0
|
|
|
|
DEPTH_ZERO, ///< for non-m.room.create, depth=0
|
2018-03-09 07:11:00 -08:00
|
|
|
MISSING_SIGNATURES, ///< no signatures
|
|
|
|
MISSING_ORIGIN_SIGNATURE, ///< no signature for origin
|
|
|
|
MISMATCH_ORIGIN_SENDER, ///< sender mxid host not from origin
|
|
|
|
MISMATCH_ORIGIN_EVENT_ID, ///< event_id mxid host not from origin
|
2018-02-20 18:09:39 -08:00
|
|
|
|
|
|
|
_NUM_
|
|
|
|
};
|