2017-08-18 12:19:13 -06:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 Charybdis Development Team
|
|
|
|
* Copyright (C) 2017 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
using namespace ircd;
|
|
|
|
|
2017-09-08 02:32:49 -07:00
|
|
|
const database::descriptor events_event_id_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"event_id",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
10.1
|
|
|
|
The id of event.
|
|
|
|
|
|
|
|
10.4
|
|
|
|
MUST NOT exceed 255 bytes.
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id. This is redundant data but we have to have it for now.
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-08-18 12:19:13 -06:00
|
|
|
const database::descriptor events_type_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"type",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
10.1
|
|
|
|
The type of event. This SHOULD be namespaced similar to Java package naming conventions
|
|
|
|
e.g. 'com.example.subdomain.event.type'.
|
|
|
|
|
|
|
|
10.4
|
|
|
|
MUST NOT exceed 255 bytes.
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_content_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"content",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
10.1
|
|
|
|
The fields in this object will vary depending on the type of event. When interacting
|
|
|
|
with the REST API, this is the HTTP body.
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
Since events must not exceed 65 KB the maximum size for the content is the remaining
|
|
|
|
space after all the other fields for the event are rendered.
|
|
|
|
|
|
|
|
key is event_id
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_room_id_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"room_id",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
10.2 (apropos room events)
|
|
|
|
Required. The ID of the room associated with this event.
|
|
|
|
|
|
|
|
10.4
|
|
|
|
MUST NOT exceed 255 bytes.
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_sender_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"sender",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
10.2 (apropos room events)
|
|
|
|
Required. Contains the fully-qualified ID of the user who sent this event.
|
|
|
|
|
|
|
|
10.4
|
|
|
|
MUST NOT exceed 255 bytes.
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_state_key_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"state_key",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
10.3 (apropos room state events)
|
|
|
|
A unique key which defines the overwriting semantics for this piece of room state.
|
|
|
|
This value is often a zero-length string. The presence of this key makes this event a
|
|
|
|
State Event. The key MUST NOT start with '_'.
|
|
|
|
|
|
|
|
10.4
|
|
|
|
MUST NOT exceed 255 bytes.
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-10-25 09:47:03 -07:00
|
|
|
const database::descriptor events_origin_descriptor
|
2017-08-18 12:19:13 -06:00
|
|
|
{
|
|
|
|
// name
|
2017-10-25 09:47:03 -07:00
|
|
|
"origin",
|
2017-08-18 12:19:13 -06:00
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
FEDERATION 4.1
|
2017-10-25 09:47:03 -07:00
|
|
|
DNS name of homeserver that created this PDU
|
2017-08-18 12:19:13 -06:00
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
2017-10-25 09:47:03 -07:00
|
|
|
typeid(string_view), typeid(string_view)
|
2017-08-18 12:19:13 -06:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-10-25 09:47:03 -07:00
|
|
|
const database::descriptor events_origin_server_ts_descriptor
|
2017-09-08 02:32:49 -07:00
|
|
|
{
|
|
|
|
// name
|
2017-10-25 09:47:03 -07:00
|
|
|
"origin_server_ts",
|
2017-09-08 02:32:49 -07:00
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
2017-10-25 09:47:03 -07:00
|
|
|
FEDERATION 4.1
|
|
|
|
Timestamp in milliseconds on origin homeserver when this PDU was created.
|
2017-09-08 02:32:49 -07:00
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id
|
2017-10-25 09:47:03 -07:00
|
|
|
value is a machine integer (binary)
|
|
|
|
|
|
|
|
TODO: consider unsigned rather than time_t because of millisecond precision
|
2017-09-08 02:32:49 -07:00
|
|
|
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
2017-10-25 09:47:03 -07:00
|
|
|
typeid(string_view), typeid(time_t)
|
2017-09-08 02:32:49 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_unsigned_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"unsigned",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id
|
|
|
|
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_signatures_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"signatures",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id
|
|
|
|
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-10-03 04:10:26 -07:00
|
|
|
const database::descriptor events_auth_events_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"auth_events",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id.
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_depth_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"depth",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id value is long integer
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(int64_t)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_hashes_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"hashes",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id.
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_membership_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"membership",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id.
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_prev_events_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"prev_events",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id.
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor events_prev_state_descriptor
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"prev_state",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
### developer note:
|
|
|
|
key is event_id.
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-09-24 18:05:42 -07:00
|
|
|
/// prefix transform for event_id suffixes
|
|
|
|
///
|
|
|
|
/// This transform expects a concatenation ending with an event_id which means
|
|
|
|
/// the prefix can be the same for multiple event_id's; therefor we can find
|
|
|
|
/// or iterate "event_id in X" where X is some key like a room_id
|
|
|
|
///
|
|
|
|
const ircd::db::prefix_transform event_id_in
|
|
|
|
{
|
|
|
|
"event_id in",
|
|
|
|
[](const string_view &key)
|
|
|
|
{
|
|
|
|
return key.find('$') != key.npos;
|
|
|
|
},
|
|
|
|
[](const string_view &key)
|
|
|
|
{
|
|
|
|
return rsplit(key, '$').first;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor event_id_in_sender
|
2017-09-08 02:32:49 -07:00
|
|
|
{
|
|
|
|
// name
|
2017-09-24 18:05:42 -07:00
|
|
|
"event_id in sender",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### developer note:
|
|
|
|
|
|
|
|
key is "@sender$event_id"
|
|
|
|
the prefix transform is in effect. this column indexes events by
|
|
|
|
sender offering an iterable bound of the index prefixed by sender
|
|
|
|
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
},
|
|
|
|
|
|
|
|
// options
|
|
|
|
{},
|
|
|
|
|
|
|
|
// comparator
|
|
|
|
{},
|
|
|
|
|
|
|
|
// prefix transform
|
|
|
|
event_id_in,
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor event_id_in_room_id
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"event_id in room_id",
|
2017-09-08 02:32:49 -07:00
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### developer note:
|
|
|
|
|
|
|
|
key is "!room_id$event_id"
|
|
|
|
the prefix transform is in effect. this column indexes events by
|
|
|
|
room_id offering an iterable bound of the index prefixed by room_id
|
|
|
|
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
},
|
|
|
|
|
|
|
|
// options
|
|
|
|
{},
|
|
|
|
|
2017-09-24 18:05:42 -07:00
|
|
|
// comparator - sorts from highest to lowest
|
2017-10-25 09:47:03 -07:00
|
|
|
{}, //ircd::db::reverse_cmp_string_view{},
|
2017-09-24 18:05:42 -07:00
|
|
|
|
|
|
|
// prefix transform
|
|
|
|
event_id_in,
|
|
|
|
};
|
|
|
|
|
|
|
|
/// prefix transform for room_id
|
|
|
|
///
|
|
|
|
/// This transform expects a concatenation ending with a room_id which means
|
|
|
|
/// the prefix can be the same for multiple room_id's; therefor we can find
|
|
|
|
/// or iterate "room_id in X" where X is some repeated prefix
|
|
|
|
///
|
|
|
|
const ircd::db::prefix_transform room_id_in
|
|
|
|
{
|
|
|
|
"room_id in",
|
|
|
|
[](const string_view &key)
|
|
|
|
{
|
|
|
|
return key.find('!') != key.npos;
|
|
|
|
},
|
|
|
|
[](const string_view &key)
|
|
|
|
{
|
|
|
|
return rsplit(key, '!').first;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor event_id_for_room_id_in_type
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"event_id for room_id in type",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### developer note:
|
|
|
|
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
},
|
|
|
|
|
|
|
|
// options
|
|
|
|
{},
|
|
|
|
|
2017-09-08 02:32:49 -07:00
|
|
|
// comparator
|
|
|
|
{},
|
|
|
|
|
|
|
|
// prefix transform
|
2017-09-24 18:05:42 -07:00
|
|
|
room_id_in,
|
|
|
|
};
|
|
|
|
|
2017-09-25 21:42:07 -07:00
|
|
|
const database::descriptor event_id_for_room_id_in_sender
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"event_id for room_id in sender",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### developer note:
|
|
|
|
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
},
|
|
|
|
|
|
|
|
// options
|
|
|
|
{},
|
|
|
|
|
|
|
|
// comparator
|
|
|
|
{},
|
|
|
|
|
|
|
|
// prefix transform
|
|
|
|
room_id_in,
|
|
|
|
};
|
|
|
|
|
2017-09-24 18:05:42 -07:00
|
|
|
/// prefix transform for type,state_key in room_id
|
|
|
|
///
|
|
|
|
/// This transform is special for concatenating room_id with type and state_key
|
|
|
|
/// in that order with prefix being the room_id (this may change to room_id+
|
|
|
|
/// type
|
|
|
|
///
|
|
|
|
const ircd::db::prefix_transform type_state_key_in_room_id
|
|
|
|
{
|
|
|
|
"type,state_key in room_id",
|
|
|
|
[](const string_view &key)
|
2017-09-08 02:32:49 -07:00
|
|
|
{
|
2017-09-24 18:05:42 -07:00
|
|
|
return key.find("..") != key.npos;
|
|
|
|
},
|
|
|
|
[](const string_view &key)
|
|
|
|
{
|
|
|
|
return split(key, "..").first;
|
2017-09-08 02:32:49 -07:00
|
|
|
}
|
2017-09-24 18:05:42 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
const database::descriptor event_id_for_type_state_key_in_room_id
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"event_id for type,state_key in room_id",
|
|
|
|
|
|
|
|
// explanation
|
|
|
|
R"(### developer note:
|
|
|
|
|
|
|
|
)",
|
2017-09-08 02:32:49 -07:00
|
|
|
|
2017-09-24 18:05:42 -07:00
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
},
|
|
|
|
|
|
|
|
// options
|
|
|
|
{},
|
2017-09-08 02:32:49 -07:00
|
|
|
|
2017-09-24 18:05:42 -07:00
|
|
|
// comparator
|
|
|
|
{},
|
2017-09-08 02:32:49 -07:00
|
|
|
|
2017-09-24 18:05:42 -07:00
|
|
|
// prefix transform
|
|
|
|
type_state_key_in_room_id
|
2017-09-08 02:32:49 -07:00
|
|
|
};
|
2017-11-15 17:48:25 -08:00
|
|
|
/*
|
|
|
|
const database::descriptor event_id_timeline_prev
|
|
|
|
{
|
|
|
|
// name
|
|
|
|
"event_id_timeline_prev",
|
2017-09-08 02:32:49 -07:00
|
|
|
|
2017-11-15 17:48:25 -08:00
|
|
|
// explanation
|
|
|
|
R"(### protocol note:
|
|
|
|
|
|
|
|
)",
|
|
|
|
|
|
|
|
// typing (key, value)
|
|
|
|
{
|
|
|
|
typeid(string_view), typeid(string_view)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
*/
|
2017-08-18 12:19:13 -06:00
|
|
|
const database::description events_description
|
|
|
|
{
|
|
|
|
{ "default" },
|
2017-09-08 02:32:49 -07:00
|
|
|
events_event_id_descriptor,
|
2017-08-18 12:19:13 -06:00
|
|
|
events_type_descriptor,
|
|
|
|
events_content_descriptor,
|
|
|
|
events_room_id_descriptor,
|
|
|
|
events_sender_descriptor,
|
|
|
|
events_state_key_descriptor,
|
|
|
|
events_origin_descriptor,
|
|
|
|
events_origin_server_ts_descriptor,
|
2017-09-08 02:32:49 -07:00
|
|
|
events_unsigned_descriptor,
|
|
|
|
events_signatures_descriptor,
|
2017-10-03 04:10:26 -07:00
|
|
|
events_auth_events_descriptor,
|
|
|
|
events_depth_descriptor,
|
|
|
|
events_hashes_descriptor,
|
|
|
|
events_membership_descriptor,
|
|
|
|
events_prev_events_descriptor,
|
|
|
|
events_prev_state_descriptor,
|
2017-09-24 18:05:42 -07:00
|
|
|
event_id_in_sender,
|
|
|
|
event_id_in_room_id,
|
|
|
|
event_id_for_room_id_in_type,
|
2017-09-25 21:42:07 -07:00
|
|
|
event_id_for_room_id_in_sender,
|
2017-09-24 18:05:42 -07:00
|
|
|
event_id_for_type_state_key_in_room_id,
|
2017-11-15 17:48:25 -08:00
|
|
|
// event_id_timeline_prev,
|
|
|
|
// event_id_timeline_next,
|
|
|
|
// event_id_for_type_timeline_prev,
|
2017-08-18 12:19:13 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
std::shared_ptr<database> events_database
|
|
|
|
{
|
|
|
|
std::make_shared<database>("events"s, ""s, events_description)
|
|
|
|
};
|
|
|
|
|
|
|
|
mapi::header IRCD_MODULE
|
|
|
|
{
|
|
|
|
"Hosts the 'events' database"
|
|
|
|
};
|