mirror of
https://github.com/matrix-construct/construct
synced 2024-10-31 19:08:59 +01:00
411 lines
7.8 KiB
C++
411 lines
7.8 KiB
C++
// The Construct
|
|
//
|
|
// Copyright (C) The Construct Developers, Authors & Contributors
|
|
// Copyright (C) 2016-2023 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.
|
|
|
|
ircd::m::user::keys::send::send(const m::user::keys &user_keys,
|
|
const string_view room_id)
|
|
try
|
|
{
|
|
const auto &user_id
|
|
{
|
|
user_keys.user_room.user.user_id
|
|
};
|
|
|
|
const unique_mutable_buffer keys_buf[2]
|
|
{
|
|
{ 4_KiB },
|
|
{ 4_KiB },
|
|
};
|
|
|
|
json::stack keys[2]
|
|
{
|
|
{ keys_buf[0] },
|
|
{ keys_buf[1] },
|
|
};
|
|
|
|
// master
|
|
{
|
|
json::stack::object object{keys[0]};
|
|
user_keys.cross_master(object);
|
|
}
|
|
|
|
// self
|
|
{
|
|
json::stack::object object{keys[1]};
|
|
user_keys.cross_self(object);
|
|
}
|
|
|
|
json::iov event, content;
|
|
const json::iov::push push[]
|
|
{
|
|
{ event, { "type", "m.signing_key_update" } },
|
|
{ event, { "sender", user_id } },
|
|
{ content, { "master_key", keys[0].completed() } },
|
|
{ content, { "self_signing_key", keys[1].completed() } },
|
|
{ content, { "user_id", user_id } },
|
|
};
|
|
|
|
// For diagnostic purposes; usually not defined.
|
|
const json::iov::push push_room_id
|
|
{
|
|
event, m::valid(m::id::ROOM, room_id),
|
|
{
|
|
"room_id", [&room_id]
|
|
{
|
|
return room_id;
|
|
}
|
|
}
|
|
};
|
|
|
|
m::vm::copts opts;
|
|
opts.edu = true;
|
|
opts.prop_mask.reset();
|
|
opts.prop_mask.set("origin");
|
|
opts.notify_clients = false;
|
|
m::vm::eval
|
|
{
|
|
event, content, opts
|
|
};
|
|
}
|
|
catch(const ctx::interrupted &)
|
|
{
|
|
throw;
|
|
}
|
|
catch(const std::exception &e)
|
|
{
|
|
log::error
|
|
{
|
|
m::log, "Sending m.signing_key_update for %s :%s",
|
|
string_view{user_keys.user_room.user.user_id},
|
|
e.what(),
|
|
};
|
|
}
|
|
|
|
void
|
|
ircd::m::user::keys::update(const m::signing_key_update &sku)
|
|
const
|
|
{
|
|
const m::user::id &user_id
|
|
{
|
|
json::get<"user_id"_>(sku)
|
|
};
|
|
|
|
const m::user::room room
|
|
{
|
|
user_id
|
|
};
|
|
|
|
const json::object &msk
|
|
{
|
|
json::get<"master_key"_>(sku)
|
|
};
|
|
|
|
const auto cross_master_id
|
|
{
|
|
json::get<"master_key"_>(sku)?
|
|
m::send(room, user_id, "ircd.cross_signing.master", "", msk):
|
|
m::event::id::buf{}
|
|
};
|
|
|
|
const json::object &ssk
|
|
{
|
|
json::get<"self_signing_key"_>(sku)
|
|
};
|
|
|
|
const auto cross_self_id
|
|
{
|
|
ssk?
|
|
m::send(room, user_id, "ircd.cross_signing.self", "", ssk):
|
|
m::event::id::buf{}
|
|
};
|
|
|
|
const json::object &usk
|
|
{
|
|
json::get<"user_signing_key"_>(sku)
|
|
};
|
|
|
|
const auto cross_user_id
|
|
{
|
|
usk && my(user_id)?
|
|
m::send(room, user_id, "ircd.cross_signing.user", "", usk):
|
|
m::event::id::buf{}
|
|
};
|
|
}
|
|
|
|
bool
|
|
ircd::m::user::keys::claim(json::stack::object &object,
|
|
const string_view &device_id,
|
|
const string_view &algorithm)
|
|
const
|
|
{
|
|
const fmt::bsprintf<m::event::TYPE_MAX_SIZE> type
|
|
{
|
|
"ircd.device.one_time_key|%s",
|
|
algorithm
|
|
};
|
|
|
|
const m::room::type events
|
|
{
|
|
user_room, type, { -1UL, -1L }, true
|
|
};
|
|
|
|
return !events.for_each([this, &object, &device_id]
|
|
(const string_view &type, const auto &, const m::event::idx &event_idx)
|
|
{
|
|
if(m::redacted(event_idx))
|
|
return true;
|
|
|
|
const bool match
|
|
{
|
|
m::query(std::nothrow, event_idx, "state_key", [&device_id]
|
|
(const string_view &state_key) noexcept
|
|
{
|
|
return state_key == device_id;
|
|
})
|
|
};
|
|
|
|
if(!match)
|
|
return true;
|
|
|
|
const auto algorithm
|
|
{
|
|
split(type, '|').second
|
|
};
|
|
|
|
const bool fetched
|
|
{
|
|
m::get(std::nothrow, event_idx, "content", [&object, &algorithm]
|
|
(const json::object &content)
|
|
{
|
|
json::stack::member
|
|
{
|
|
object, algorithm, json::object
|
|
{
|
|
content[""] // ircd.device.* quirk
|
|
}
|
|
};
|
|
})
|
|
};
|
|
|
|
if(!fetched)
|
|
return true;
|
|
|
|
const auto event_id
|
|
{
|
|
m::event_id(event_idx)
|
|
};
|
|
|
|
const auto redact_id
|
|
{
|
|
m::redact(user_room, user_room.user, event_id, "claimed")
|
|
};
|
|
|
|
return false;
|
|
});
|
|
}
|
|
|
|
void
|
|
ircd::m::user::keys::device(json::stack::object &out,
|
|
const string_view &device_id)
|
|
const
|
|
{
|
|
const m::user::devices devices
|
|
{
|
|
user_room.user
|
|
};
|
|
|
|
devices.get(std::nothrow, device_id, "keys", [this, &out, &device_id]
|
|
(const auto &, const json::object &device_keys)
|
|
{
|
|
const auto &user_id
|
|
{
|
|
user_room.user.user_id
|
|
};
|
|
|
|
for(const auto &member : device_keys)
|
|
if(member.first != "signatures")
|
|
json::stack::member
|
|
{
|
|
out, member
|
|
};
|
|
|
|
json::stack::object sigs
|
|
{
|
|
out, "signatures"
|
|
};
|
|
|
|
json::stack::object user_sigs
|
|
{
|
|
sigs, user_id
|
|
};
|
|
|
|
attach_sigs(user_sigs, device_keys, user_id);
|
|
const m::room::state state
|
|
{
|
|
user_room
|
|
};
|
|
|
|
state.for_each("ircd.keys.signatures", [this, &user_sigs, &user_id, &device_id]
|
|
(const string_view &, const string_view &state_key, const auto &event_idx)
|
|
{
|
|
const auto &[target, source]
|
|
{
|
|
unmake_sigs_state_key(state_key)
|
|
};
|
|
|
|
if(target && target != device_id)
|
|
return true;
|
|
|
|
attach_sigs(user_sigs, event_idx, user_id);
|
|
return true;
|
|
});
|
|
});
|
|
}
|
|
|
|
bool
|
|
ircd::m::user::keys::append_keys(json::stack::object &out,
|
|
const event::idx &event_idx,
|
|
const user::id &user_id)
|
|
const
|
|
{
|
|
return m::get(std::nothrow, event_idx, "content", [this, &out, &user_id]
|
|
(const json::object &device_keys)
|
|
{
|
|
append_keys(out, device_keys, user_id);
|
|
});
|
|
}
|
|
|
|
void
|
|
ircd::m::user::keys::append_keys(json::stack::object &out,
|
|
const json::object &device_keys,
|
|
const user::id &user_id)
|
|
const
|
|
{
|
|
for(const auto &member : device_keys)
|
|
if(member.first != "signatures")
|
|
json::stack::member
|
|
{
|
|
out, member
|
|
};
|
|
|
|
json::stack::object sigs
|
|
{
|
|
out, "signatures"
|
|
};
|
|
|
|
json::stack::object user_sigs
|
|
{
|
|
sigs, user_id
|
|
};
|
|
|
|
attach_sigs(user_sigs, device_keys, user_id);
|
|
const json::object device_keys_keys
|
|
{
|
|
device_keys["keys"]
|
|
};
|
|
|
|
const m::room::state state
|
|
{
|
|
user_room
|
|
};
|
|
|
|
state.for_each("ircd.keys.signatures", [this, &user_sigs, &user_id, &device_keys_keys]
|
|
(const string_view &, const string_view &state_key, const auto &event_idx)
|
|
{
|
|
for(const auto &[key_id_, key] : device_keys_keys)
|
|
{
|
|
const auto &key_id
|
|
{
|
|
split(key_id_, ':').second
|
|
};
|
|
|
|
const auto &[target, source]
|
|
{
|
|
unmake_sigs_state_key(state_key)
|
|
};
|
|
|
|
if(target != key_id)
|
|
continue;
|
|
|
|
attach_sigs(user_sigs, event_idx, user_id);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
}
|
|
|
|
bool
|
|
ircd::m::user::keys::attach_sigs(json::stack::object &user_sigs,
|
|
const event::idx &event_idx,
|
|
const user::id &user_id)
|
|
const
|
|
{
|
|
return m::get(std::nothrow, event_idx, "content", [this, &user_sigs, &user_id]
|
|
(const json::object &device_sigs)
|
|
{
|
|
attach_sigs(user_sigs, device_sigs, user_id);
|
|
});
|
|
}
|
|
|
|
void
|
|
ircd::m::user::keys::attach_sigs(json::stack::object &user_sigs,
|
|
const json::object &device_sigs,
|
|
const user::id &user_id)
|
|
const
|
|
{
|
|
const json::object device_sigs_sigs
|
|
{
|
|
device_sigs["signatures"]
|
|
};
|
|
|
|
const json::object device_sigs_user_sigs
|
|
{
|
|
device_sigs_sigs[user_id]
|
|
};
|
|
|
|
for(const auto &member : device_sigs_user_sigs)
|
|
json::stack::member
|
|
{
|
|
user_sigs, member
|
|
};
|
|
}
|
|
|
|
//
|
|
// user::keys::sigs
|
|
//
|
|
|
|
std::tuple<ircd::string_view, ircd::string_view>
|
|
ircd::m::user::keys::unmake_sigs_state_key(const string_view &state_key)
|
|
noexcept
|
|
{
|
|
const auto &[tgt, src]
|
|
{
|
|
rsplit(state_key, '%')
|
|
};
|
|
|
|
return std::tuple
|
|
{
|
|
tgt, src
|
|
};
|
|
}
|
|
|
|
ircd::string_view
|
|
ircd::m::user::keys::make_sigs_state_key(const mutable_buffer &buf,
|
|
const string_view &tgt,
|
|
const string_view &src)
|
|
{
|
|
return fmt::sprintf
|
|
{
|
|
buf, "%s%s",
|
|
string_view{tgt},
|
|
tgt != src && src?
|
|
string_view{src}:
|
|
string_view{},
|
|
};
|
|
}
|