mirror of
https://github.com/AyuGram/AyuGramDesktop.git
synced 2025-06-05 22:54:01 +02:00
Remove stale confcall users from blockchain.
This commit is contained in:
parent
7deb5bfdf2
commit
e3ef870b29
7 changed files with 437 additions and 91 deletions
|
@ -421,21 +421,6 @@ std::shared_ptr<ParticipantVideoParams> ParseVideoParams(
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<TdE2E::ParticipantState> ParseParticipantState(
|
|
||||||
const MTPDgroupCallParticipant &data) {
|
|
||||||
if (!data.vpublic_key() || data.vpeer().type() != mtpc_peerUser) {
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
const auto &v = *data.vpublic_key();
|
|
||||||
const auto userId = data.vpeer().c_peerUser().vuser_id().v;
|
|
||||||
using State = TdE2E::ParticipantState;
|
|
||||||
return std::make_shared<State>(State{
|
|
||||||
.id = uint64(userId),
|
|
||||||
.key = { .a = v.h.h, .b = v.h.l, .c = v.l.h, .d = v.l.l },
|
|
||||||
});
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
GroupCall::LoadPartTask::LoadPartTask(
|
GroupCall::LoadPartTask::LoadPartTask(
|
||||||
base::weak_ptr<GroupCall> call,
|
base::weak_ptr<GroupCall> call,
|
||||||
int64 time,
|
int64 time,
|
||||||
|
@ -643,17 +628,6 @@ GroupCall::GroupCall(
|
||||||
, _listenersHidden(join.rtmp)
|
, _listenersHidden(join.rtmp)
|
||||||
, _rtmp(join.rtmp)
|
, _rtmp(join.rtmp)
|
||||||
, _rtmpVolume(Group::kDefaultVolume) {
|
, _rtmpVolume(Group::kDefaultVolume) {
|
||||||
if (_conferenceCall) {
|
|
||||||
if (!_e2e) {
|
|
||||||
_e2e = std::make_shared<TdE2E::Call>(
|
|
||||||
TdE2E::MakeUserId(_peer->session().user()));
|
|
||||||
}
|
|
||||||
_e2e->subchainRequests(
|
|
||||||
) | rpl::start_with_next([=](TdE2E::Call::SubchainRequest request) {
|
|
||||||
requestSubchainBlocks(request.subchain, request.height);
|
|
||||||
}, _lifetime);
|
|
||||||
}
|
|
||||||
|
|
||||||
_muted.value(
|
_muted.value(
|
||||||
) | rpl::combine_previous(
|
) | rpl::combine_previous(
|
||||||
) | rpl::start_with_next([=](MuteState previous, MuteState state) {
|
) | rpl::start_with_next([=](MuteState previous, MuteState state) {
|
||||||
|
@ -705,6 +679,9 @@ GroupCall::GroupCall(
|
||||||
|
|
||||||
setupMediaDevices();
|
setupMediaDevices();
|
||||||
setupOutgoingVideo();
|
setupOutgoingVideo();
|
||||||
|
if (_conferenceCall) {
|
||||||
|
setupConferenceCall();
|
||||||
|
}
|
||||||
|
|
||||||
if (_id) {
|
if (_id) {
|
||||||
this->join(inputCall);
|
this->join(inputCall);
|
||||||
|
@ -724,6 +701,62 @@ GroupCall::~GroupCall() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GroupCall::setupConferenceCall() {
|
||||||
|
if (!_e2e) {
|
||||||
|
_e2e = std::make_shared<TdE2E::Call>(
|
||||||
|
TdE2E::MakeUserId(_peer->session().user()));
|
||||||
|
}
|
||||||
|
_e2e->subchainRequests(
|
||||||
|
) | rpl::start_with_next([=](TdE2E::Call::SubchainRequest request) {
|
||||||
|
requestSubchainBlocks(request.subchain, request.height);
|
||||||
|
}, _lifetime);
|
||||||
|
_e2e->sendOutboundBlock(
|
||||||
|
) | rpl::start_with_next([=](QByteArray &&block) {
|
||||||
|
sendOutboundBlock(std::move(block));
|
||||||
|
}, _lifetime);
|
||||||
|
|
||||||
|
_conferenceCall->staleParticipantId(
|
||||||
|
) | rpl::start_with_next([=](UserId staleId) {
|
||||||
|
removeConferenceParticipant(staleId);
|
||||||
|
}, _lifetime);
|
||||||
|
_e2e->participantsSetValue(
|
||||||
|
) | rpl::start_with_next([=](const TdE2E::ParticipantsSet &set) {
|
||||||
|
auto users = base::flat_set<UserId>();
|
||||||
|
users.reserve(set.list.size());
|
||||||
|
auto ids = QStringList();
|
||||||
|
for (const auto &id : set.list) {
|
||||||
|
users.emplace(UserId(id.v));
|
||||||
|
ids.push_back('"' + _peer->owner().user(UserId(id.v))->name() + '"');
|
||||||
|
}
|
||||||
|
LOG(("ACCESS: ") + ids.join(", "));
|
||||||
|
_conferenceCall->setParticipantsWithAccess(std::move(users));
|
||||||
|
}, _lifetime);
|
||||||
|
}
|
||||||
|
|
||||||
|
void GroupCall::removeConferenceParticipant(UserId id) {
|
||||||
|
Expects(_e2e != nullptr);
|
||||||
|
|
||||||
|
const auto block = _e2e->makeRemoveBlock(TdE2E::MakeUserId(id));
|
||||||
|
if (block.data.isEmpty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_api.request(MTPphone_DeleteConferenceCallParticipant(
|
||||||
|
inputCall(),
|
||||||
|
_peer->owner().user(id)->input,
|
||||||
|
MTP_bytes(block.data)
|
||||||
|
)).done([=](const MTPUpdates &result) {
|
||||||
|
_peer->session().api().applyUpdates(result);
|
||||||
|
}).fail([=](const MTP::Error &error) {
|
||||||
|
const auto type = error.type();
|
||||||
|
if (type == u"GROUPCALL_FORBIDDEN"_q) {
|
||||||
|
setState(State::Joining);
|
||||||
|
rejoin();
|
||||||
|
} else {
|
||||||
|
LOG(("NOTREMOVED: %1").arg(type));
|
||||||
|
}
|
||||||
|
}).send();
|
||||||
|
}
|
||||||
|
|
||||||
bool GroupCall::isSharingScreen() const {
|
bool GroupCall::isSharingScreen() const {
|
||||||
return _isSharingScreen.current();
|
return _isSharingScreen.current();
|
||||||
}
|
}
|
||||||
|
@ -1492,11 +1525,15 @@ void GroupCall::sendJoinRequest() {
|
||||||
}
|
}
|
||||||
if (_e2e) {
|
if (_e2e) {
|
||||||
_e2e->joined();
|
_e2e->joined();
|
||||||
|
if (!_pendingOutboundBlock.isEmpty()) {
|
||||||
|
sendOutboundBlock(base::take(_pendingOutboundBlock));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}).fail([=](const MTP::Error &error) {
|
}).fail([=](const MTP::Error &error) {
|
||||||
const auto type = error.type();
|
const auto type = error.type();
|
||||||
if (_e2e) {
|
if (_e2e) {
|
||||||
if (type == u"CONF_WRITE_CHAIN_INVALID"_q) {
|
if (type == u"BLOCK_INVALID"_q
|
||||||
|
|| type.startsWith(u"CONF_WRITE_CHAIN_INVALID"_q)) {
|
||||||
refreshLastBlockAndJoin();
|
refreshLastBlockAndJoin();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -1583,6 +1620,29 @@ void GroupCall::requestSubchainBlocks(int subchain, int height) {
|
||||||
}).send();
|
}).send();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GroupCall::sendOutboundBlock(QByteArray block) {
|
||||||
|
_pendingOutboundBlock = QByteArray();
|
||||||
|
_api.request(MTPphone_SendConferenceCallBroadcast(
|
||||||
|
inputCall(),
|
||||||
|
MTP_bytes(block)
|
||||||
|
)).done([=](const MTPUpdates &result) {
|
||||||
|
_peer->session().api().applyUpdates(result);
|
||||||
|
}).fail([=](const MTP::Error &error) {
|
||||||
|
const auto type = error.type();
|
||||||
|
if (type == u"GROUPCALL_FORBIDDEN"_q) {
|
||||||
|
_pendingOutboundBlock = block;
|
||||||
|
setState(State::Joining);
|
||||||
|
rejoin();
|
||||||
|
} else if (type == u"BLOCK_INVALID"_q
|
||||||
|
|| type.startsWith(u"CONF_WRITE_CHAIN_INVALID"_q)) {
|
||||||
|
LOG(("Call Error: Could not broadcast block: %1").arg(type));
|
||||||
|
} else {
|
||||||
|
LOG(("HMM"));
|
||||||
|
sendOutboundBlock(block);
|
||||||
|
}
|
||||||
|
}).send();
|
||||||
|
}
|
||||||
|
|
||||||
void GroupCall::checkNextJoinAction() {
|
void GroupCall::checkNextJoinAction() {
|
||||||
if (_joinState.action != JoinAction::None) {
|
if (_joinState.action != JoinAction::None) {
|
||||||
return;
|
return;
|
||||||
|
@ -1741,8 +1801,7 @@ void GroupCall::applyMeInCallLocally() {
|
||||||
| Flag::f_volume // Without flag the volume is reset to 100%.
|
| Flag::f_volume // Without flag the volume is reset to 100%.
|
||||||
| Flag::f_volume_by_admin // Self volume can only be set by admin.
|
| Flag::f_volume_by_admin // Self volume can only be set by admin.
|
||||||
| ((muted() != MuteState::Active) ? Flag::f_muted : Flag(0))
|
| ((muted() != MuteState::Active) ? Flag::f_muted : Flag(0))
|
||||||
| (raisedHandRating > 0 ? Flag::f_raise_hand_rating : Flag(0))
|
| (raisedHandRating > 0 ? Flag::f_raise_hand_rating : Flag(0));
|
||||||
| (_e2e ? Flag::f_public_key : Flag(0));
|
|
||||||
real->applyLocalUpdate(
|
real->applyLocalUpdate(
|
||||||
MTP_updateGroupCallParticipants(
|
MTP_updateGroupCallParticipants(
|
||||||
inputCall(),
|
inputCall(),
|
||||||
|
@ -1758,10 +1817,7 @@ void GroupCall::applyMeInCallLocally() {
|
||||||
MTPstring(), // Don't update about text in local updates.
|
MTPstring(), // Don't update about text in local updates.
|
||||||
MTP_long(raisedHandRating),
|
MTP_long(raisedHandRating),
|
||||||
MTPGroupCallParticipantVideo(),
|
MTPGroupCallParticipantVideo(),
|
||||||
MTPGroupCallParticipantVideo(),
|
MTPGroupCallParticipantVideo())),
|
||||||
(_e2e
|
|
||||||
? TdE2E::PublicKeyToMTP(_e2e->myKey())
|
|
||||||
: MTPint256()))),
|
|
||||||
MTP_int(0)).c_updateGroupCallParticipants());
|
MTP_int(0)).c_updateGroupCallParticipants());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1792,11 +1848,7 @@ void GroupCall::applyParticipantLocally(
|
||||||
| (participantPeer == joinAs() ? Flag::f_self : Flag(0))
|
| (participantPeer == joinAs() ? Flag::f_self : Flag(0))
|
||||||
| (participant->raisedHandRating
|
| (participant->raisedHandRating
|
||||||
? Flag::f_raise_hand_rating
|
? Flag::f_raise_hand_rating
|
||||||
: Flag(0))
|
: Flag(0));
|
||||||
| (participant->e2eState ? Flag::f_public_key : Flag(0));
|
|
||||||
const auto publicKey = participant->e2eState
|
|
||||||
? participant->e2eState->key
|
|
||||||
: TdE2E::PublicKey();
|
|
||||||
_peer->groupCall()->applyLocalUpdate(
|
_peer->groupCall()->applyLocalUpdate(
|
||||||
MTP_updateGroupCallParticipants(
|
MTP_updateGroupCallParticipants(
|
||||||
inputCall(),
|
inputCall(),
|
||||||
|
@ -1812,10 +1864,7 @@ void GroupCall::applyParticipantLocally(
|
||||||
MTPstring(), // Don't update about text in local updates.
|
MTPstring(), // Don't update about text in local updates.
|
||||||
MTP_long(participant->raisedHandRating),
|
MTP_long(participant->raisedHandRating),
|
||||||
MTPGroupCallParticipantVideo(),
|
MTPGroupCallParticipantVideo(),
|
||||||
MTPGroupCallParticipantVideo(),
|
MTPGroupCallParticipantVideo())),
|
||||||
MTP_int256(
|
|
||||||
MTP_int128(publicKey.a, publicKey.b),
|
|
||||||
MTP_int128(publicKey.c, publicKey.d)))),
|
|
||||||
MTP_int(0)).c_updateGroupCallParticipants());
|
MTP_int(0)).c_updateGroupCallParticipants());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -43,7 +43,6 @@ class GroupCall;
|
||||||
} // namespace Data
|
} // namespace Data
|
||||||
|
|
||||||
namespace TdE2E {
|
namespace TdE2E {
|
||||||
struct ParticipantState;
|
|
||||||
class Call;
|
class Call;
|
||||||
} // namespace TdE2E
|
} // namespace TdE2E
|
||||||
|
|
||||||
|
@ -170,8 +169,6 @@ struct ParticipantVideoParams;
|
||||||
const tl::conditional<MTPGroupCallParticipantVideo> &camera,
|
const tl::conditional<MTPGroupCallParticipantVideo> &camera,
|
||||||
const tl::conditional<MTPGroupCallParticipantVideo> &screen,
|
const tl::conditional<MTPGroupCallParticipantVideo> &screen,
|
||||||
const std::shared_ptr<ParticipantVideoParams> &existing);
|
const std::shared_ptr<ParticipantVideoParams> &existing);
|
||||||
[[nodiscard]] std::shared_ptr<TdE2E::ParticipantState> ParseParticipantState(
|
|
||||||
const MTPDgroupCallParticipant &data);
|
|
||||||
|
|
||||||
[[nodiscard]] const std::string &GetCameraEndpoint(
|
[[nodiscard]] const std::string &GetCameraEndpoint(
|
||||||
const std::shared_ptr<ParticipantVideoParams> ¶ms);
|
const std::shared_ptr<ParticipantVideoParams> ¶ms);
|
||||||
|
@ -282,6 +279,7 @@ public:
|
||||||
void startScheduledNow();
|
void startScheduledNow();
|
||||||
void toggleScheduleStartSubscribed(bool subscribed);
|
void toggleScheduleStartSubscribed(bool subscribed);
|
||||||
void setNoiseSuppression(bool enabled);
|
void setNoiseSuppression(bool enabled);
|
||||||
|
void removeConferenceParticipant(UserId userId);
|
||||||
|
|
||||||
bool emitShareScreenError();
|
bool emitShareScreenError();
|
||||||
bool emitShareCameraError();
|
bool emitShareCameraError();
|
||||||
|
@ -548,6 +546,7 @@ private:
|
||||||
void sendJoinRequest();
|
void sendJoinRequest();
|
||||||
void refreshLastBlockAndJoin();
|
void refreshLastBlockAndJoin();
|
||||||
void requestSubchainBlocks(int subchain, int height);
|
void requestSubchainBlocks(int subchain, int height);
|
||||||
|
void sendOutboundBlock(QByteArray block);
|
||||||
|
|
||||||
void audioLevelsUpdated(const tgcalls::GroupLevelsUpdate &data);
|
void audioLevelsUpdated(const tgcalls::GroupLevelsUpdate &data);
|
||||||
void setInstanceConnected(tgcalls::GroupNetworkState networkState);
|
void setInstanceConnected(tgcalls::GroupNetworkState networkState);
|
||||||
|
@ -588,6 +587,7 @@ private:
|
||||||
|
|
||||||
void setupMediaDevices();
|
void setupMediaDevices();
|
||||||
void setupOutgoingVideo();
|
void setupOutgoingVideo();
|
||||||
|
void setupConferenceCall();
|
||||||
void setScreenEndpoint(std::string endpoint);
|
void setScreenEndpoint(std::string endpoint);
|
||||||
void setCameraEndpoint(std::string endpoint);
|
void setCameraEndpoint(std::string endpoint);
|
||||||
void addVideoOutput(const std::string &endpoint, SinkPointer sink);
|
void addVideoOutput(const std::string &endpoint, SinkPointer sink);
|
||||||
|
@ -607,6 +607,7 @@ private:
|
||||||
const not_null<Delegate*> _delegate;
|
const not_null<Delegate*> _delegate;
|
||||||
const std::shared_ptr<Data::GroupCall> _conferenceCall;
|
const std::shared_ptr<Data::GroupCall> _conferenceCall;
|
||||||
std::shared_ptr<TdE2E::Call> _e2e;
|
std::shared_ptr<TdE2E::Call> _e2e;
|
||||||
|
QByteArray _pendingOutboundBlock;
|
||||||
|
|
||||||
not_null<PeerData*> _peer; // Can change in legacy group migration.
|
not_null<PeerData*> _peer; // Can change in legacy group migration.
|
||||||
rpl::event_stream<PeerData*> _peerStream;
|
rpl::event_stream<PeerData*> _peerStream;
|
||||||
|
|
|
@ -27,6 +27,7 @@ constexpr auto kSpeakingAfterActive = crl::time(6000);
|
||||||
constexpr auto kActiveAfterJoined = crl::time(1000);
|
constexpr auto kActiveAfterJoined = crl::time(1000);
|
||||||
constexpr auto kWaitForUpdatesTimeout = 3 * crl::time(1000);
|
constexpr auto kWaitForUpdatesTimeout = 3 * crl::time(1000);
|
||||||
constexpr auto kReloadStaleTimeout = 16 * crl::time(1000);
|
constexpr auto kReloadStaleTimeout = 16 * crl::time(1000);
|
||||||
|
constexpr auto kMaxConferenceMembers = 50;
|
||||||
|
|
||||||
[[nodiscard]] QString ExtractNextOffset(const MTPphone_GroupCall &call) {
|
[[nodiscard]] QString ExtractNextOffset(const MTPphone_GroupCall &call) {
|
||||||
return call.match([&](const MTPDphone_groupCall &data) {
|
return call.match([&](const MTPDphone_groupCall &data) {
|
||||||
|
@ -75,6 +76,30 @@ GroupCall::GroupCall(
|
||||||
, _listenersHidden(rtmp) {
|
, _listenersHidden(rtmp) {
|
||||||
if (_conference) {
|
if (_conference) {
|
||||||
session().data().registerGroupCall(this);
|
session().data().registerGroupCall(this);
|
||||||
|
|
||||||
|
_participantUpdates.events(
|
||||||
|
) | rpl::filter([=](const ParticipantUpdate &update) {
|
||||||
|
return !update.now
|
||||||
|
&& !update.was->peer->isSelf()
|
||||||
|
&& !_participantsWithAccess.current().empty();
|
||||||
|
}) | rpl::start_with_next([=](const ParticipantUpdate &update) {
|
||||||
|
if (const auto id = peerToUser(update.was->peer->id)) {
|
||||||
|
if (_participantsWithAccess.current().contains(id)) {
|
||||||
|
_staleParticipantId.fire_copy(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, _checkStaleLifetime);
|
||||||
|
|
||||||
|
_participantsWithAccess.changes(
|
||||||
|
) | rpl::filter([=](const base::flat_set<UserId> &list) {
|
||||||
|
return !list.empty();
|
||||||
|
}) | rpl::start_with_next([=] {
|
||||||
|
if (_allParticipantsLoaded) {
|
||||||
|
checkStaleParticipants();
|
||||||
|
} else {
|
||||||
|
requestParticipants();
|
||||||
|
}
|
||||||
|
}, _checkStaleLifetime);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -84,6 +109,7 @@ GroupCall::~GroupCall() {
|
||||||
}
|
}
|
||||||
api().request(_unknownParticipantPeersRequestId).cancel();
|
api().request(_unknownParticipantPeersRequestId).cancel();
|
||||||
api().request(_participantsRequestId).cancel();
|
api().request(_participantsRequestId).cancel();
|
||||||
|
api().request(_checkStaleRequestId).cancel();
|
||||||
api().request(_reloadRequestId).cancel();
|
api().request(_reloadRequestId).cancel();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -158,7 +184,7 @@ void GroupCall::requestParticipants() {
|
||||||
: ApplySliceSource::SliceLoaded));
|
: ApplySliceSource::SliceLoaded));
|
||||||
setServerParticipantsCount(data.vcount().v);
|
setServerParticipantsCount(data.vcount().v);
|
||||||
if (data.vparticipants().v.isEmpty()) {
|
if (data.vparticipants().v.isEmpty()) {
|
||||||
_allParticipantsLoaded = true;
|
setParticipantsLoaded();
|
||||||
}
|
}
|
||||||
finishParticipantsSliceRequest();
|
finishParticipantsSliceRequest();
|
||||||
if (reloaded) {
|
if (reloaded) {
|
||||||
|
@ -169,7 +195,7 @@ void GroupCall::requestParticipants() {
|
||||||
_participantsRequestId = 0;
|
_participantsRequestId = 0;
|
||||||
const auto reloaded = processSavedFullCall();
|
const auto reloaded = processSavedFullCall();
|
||||||
setServerParticipantsCount(_participants.size());
|
setServerParticipantsCount(_participants.size());
|
||||||
_allParticipantsLoaded = true;
|
setParticipantsLoaded();
|
||||||
finishParticipantsSliceRequest();
|
finishParticipantsSliceRequest();
|
||||||
if (reloaded) {
|
if (reloaded) {
|
||||||
_participantsReloaded.fire({});
|
_participantsReloaded.fire({});
|
||||||
|
@ -177,6 +203,76 @@ void GroupCall::requestParticipants() {
|
||||||
}).send();
|
}).send();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GroupCall::setParticipantsLoaded() {
|
||||||
|
_allParticipantsLoaded = true;
|
||||||
|
checkStaleParticipants();
|
||||||
|
}
|
||||||
|
|
||||||
|
void GroupCall::checkStaleParticipants() {
|
||||||
|
if (_checkStaleRequestId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const auto &list = _participantsWithAccess.current();
|
||||||
|
if (list.empty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto existing = base::flat_set<UserId>();
|
||||||
|
existing.reserve(_participants.size() + 1);
|
||||||
|
existing.emplace(session().userId());
|
||||||
|
for (const auto &participant : _participants) {
|
||||||
|
if (const auto id = peerToUser(participant.peer->id)) {
|
||||||
|
existing.emplace(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (list.size() > existing.size()) {
|
||||||
|
checkStaleRequest();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (const auto &id : list) {
|
||||||
|
if (!existing.contains(id)) {
|
||||||
|
checkStaleRequest();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void GroupCall::checkStaleRequest() {
|
||||||
|
if (_checkStaleRequestId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_checkStaleRequestId = api().request(MTPphone_GetGroupParticipants(
|
||||||
|
input(),
|
||||||
|
MTP_vector<MTPInputPeer>(), // ids
|
||||||
|
MTP_vector<MTPint>(), // ssrcs
|
||||||
|
MTP_string(QString()),
|
||||||
|
MTP_int(kMaxConferenceMembers)
|
||||||
|
)).done([=](const MTPphone_GroupParticipants &result) {
|
||||||
|
_checkStaleRequestId = 0;
|
||||||
|
const auto &list = _participantsWithAccess.current();
|
||||||
|
if (list.empty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto existing = base::flat_set<UserId>();
|
||||||
|
const auto &data = result.data();
|
||||||
|
existing.reserve(data.vparticipants().v.size() + 1);
|
||||||
|
existing.emplace(session().userId());
|
||||||
|
for (const auto &participant : data.vparticipants().v) {
|
||||||
|
const auto peerId = peerFromMTP(participant.data().vpeer());
|
||||||
|
if (const auto id = peerToUser(peerId)) {
|
||||||
|
existing.emplace(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (const auto &id : list) {
|
||||||
|
if (!existing.contains(id)) {
|
||||||
|
_staleParticipantId.fire_copy(id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}).fail([=] {
|
||||||
|
_checkStaleRequestId = 0;
|
||||||
|
}).send();
|
||||||
|
}
|
||||||
|
|
||||||
bool GroupCall::processSavedFullCall() {
|
bool GroupCall::processSavedFullCall() {
|
||||||
if (!_savedFull) {
|
if (!_savedFull) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -287,6 +383,29 @@ auto GroupCall::participantSpeaking() const
|
||||||
return _participantSpeaking.events();
|
return _participantSpeaking.events();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GroupCall::setParticipantsWithAccess(base::flat_set<UserId> list) {
|
||||||
|
_participantsWithAccess = std::move(list);
|
||||||
|
if (_allParticipantsLoaded) {
|
||||||
|
checkStaleParticipants();
|
||||||
|
} else {
|
||||||
|
requestParticipants();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
auto GroupCall::participantsWithAccessCurrent() const
|
||||||
|
-> const base::flat_set<UserId> & {
|
||||||
|
return _participantsWithAccess.current();
|
||||||
|
}
|
||||||
|
|
||||||
|
auto GroupCall::participantsWithAccessValue() const
|
||||||
|
-> rpl::producer<base::flat_set<UserId>> {
|
||||||
|
return _participantsWithAccess.value();
|
||||||
|
}
|
||||||
|
|
||||||
|
rpl::producer<UserId> GroupCall::staleParticipantId() const {
|
||||||
|
return _staleParticipantId.events();
|
||||||
|
}
|
||||||
|
|
||||||
void GroupCall::enqueueUpdate(const MTPUpdate &update) {
|
void GroupCall::enqueueUpdate(const MTPUpdate &update) {
|
||||||
update.match([&](const MTPDupdateGroupCall &updateData) {
|
update.match([&](const MTPDupdateGroupCall &updateData) {
|
||||||
updateData.vcall().match([&](const MTPDgroupCall &data) {
|
updateData.vcall().match([&](const MTPDgroupCall &data) {
|
||||||
|
@ -471,9 +590,7 @@ void GroupCall::applyEnqueuedUpdate(const MTPUpdate &update) {
|
||||||
}, [](const auto &) {
|
}, [](const auto &) {
|
||||||
Unexpected("Type in GroupCall::applyEnqueuedUpdate.");
|
Unexpected("Type in GroupCall::applyEnqueuedUpdate.");
|
||||||
});
|
});
|
||||||
Core::App().calls().applyGroupCallUpdateChecked(
|
Core::App().calls().applyGroupCallUpdateChecked(&session(), update);
|
||||||
&_peer->session(),
|
|
||||||
update);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void GroupCall::processQueuedUpdates() {
|
void GroupCall::processQueuedUpdates() {
|
||||||
|
@ -634,22 +751,15 @@ void GroupCall::applyParticipantsSlice(
|
||||||
const auto existingVideoParams = (i != end(_participants))
|
const auto existingVideoParams = (i != end(_participants))
|
||||||
? i->videoParams
|
? i->videoParams
|
||||||
: nullptr;
|
: nullptr;
|
||||||
const auto existingState = (i != end(_participants))
|
|
||||||
? i->e2eState
|
|
||||||
: nullptr;
|
|
||||||
auto videoParams = localUpdate
|
auto videoParams = localUpdate
|
||||||
? existingVideoParams
|
? existingVideoParams
|
||||||
: Calls::ParseVideoParams(
|
: Calls::ParseVideoParams(
|
||||||
data.vvideo(),
|
data.vvideo(),
|
||||||
data.vpresentation(),
|
data.vpresentation(),
|
||||||
existingVideoParams);
|
existingVideoParams);
|
||||||
auto e2eState = localUpdate
|
|
||||||
? existingState
|
|
||||||
: Calls::ParseParticipantState(data);
|
|
||||||
const auto value = Participant{
|
const auto value = Participant{
|
||||||
.peer = participantPeer,
|
.peer = participantPeer,
|
||||||
.videoParams = std::move(videoParams),
|
.videoParams = std::move(videoParams),
|
||||||
.e2eState = std::move(e2eState),
|
|
||||||
.date = data.vdate().v,
|
.date = data.vdate().v,
|
||||||
.lastActive = lastActive,
|
.lastActive = lastActive,
|
||||||
.raisedHandRating = raisedHandRating,
|
.raisedHandRating = raisedHandRating,
|
||||||
|
@ -1003,7 +1113,7 @@ bool GroupCall::joinedToTop() const {
|
||||||
}
|
}
|
||||||
|
|
||||||
ApiWrap &GroupCall::api() const {
|
ApiWrap &GroupCall::api() const {
|
||||||
return _peer->session().api();
|
return session().api();
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Data
|
} // namespace Data
|
||||||
|
|
|
@ -38,7 +38,6 @@ struct LastSpokeTimes {
|
||||||
struct GroupCallParticipant {
|
struct GroupCallParticipant {
|
||||||
not_null<PeerData*> peer;
|
not_null<PeerData*> peer;
|
||||||
std::shared_ptr<Calls::ParticipantVideoParams> videoParams;
|
std::shared_ptr<Calls::ParticipantVideoParams> videoParams;
|
||||||
std::shared_ptr<TdE2E::ParticipantState> e2eState;
|
|
||||||
TimeId date = 0;
|
TimeId date = 0;
|
||||||
TimeId lastActive = 0;
|
TimeId lastActive = 0;
|
||||||
uint64 raisedHandRating = 0;
|
uint64 raisedHandRating = 0;
|
||||||
|
@ -146,6 +145,16 @@ public:
|
||||||
[[nodiscard]] auto participantSpeaking() const
|
[[nodiscard]] auto participantSpeaking() const
|
||||||
-> rpl::producer<not_null<Participant*>>;
|
-> rpl::producer<not_null<Participant*>>;
|
||||||
|
|
||||||
|
void setParticipantsWithAccess(base::flat_set<UserId> list);
|
||||||
|
[[nodiscard]] auto participantsWithAccessCurrent() const
|
||||||
|
-> const base::flat_set<UserId> &;
|
||||||
|
[[nodiscard]] auto participantsWithAccessValue() const
|
||||||
|
-> rpl::producer<base::flat_set<UserId>>;
|
||||||
|
[[nodiscard]] rpl::producer<UserId> staleParticipantId() const;
|
||||||
|
void setParticipantsLoaded();
|
||||||
|
void checkStaleParticipants();
|
||||||
|
void checkStaleRequest();
|
||||||
|
|
||||||
void enqueueUpdate(const MTPUpdate &update);
|
void enqueueUpdate(const MTPUpdate &update);
|
||||||
void applyLocalUpdate(
|
void applyLocalUpdate(
|
||||||
const MTPDupdateGroupCallParticipants &update);
|
const MTPDupdateGroupCallParticipants &update);
|
||||||
|
@ -254,6 +263,11 @@ private:
|
||||||
rpl::event_stream<not_null<Participant*>> _participantSpeaking;
|
rpl::event_stream<not_null<Participant*>> _participantSpeaking;
|
||||||
rpl::event_stream<> _participantsReloaded;
|
rpl::event_stream<> _participantsReloaded;
|
||||||
|
|
||||||
|
rpl::variable<base::flat_set<UserId>> _participantsWithAccess;
|
||||||
|
rpl::event_stream<UserId> _staleParticipantId;
|
||||||
|
mtpRequestId _checkStaleRequestId = 0;
|
||||||
|
rpl::lifetime _checkStaleLifetime;
|
||||||
|
|
||||||
bool _joinMuted : 1 = false;
|
bool _joinMuted : 1 = false;
|
||||||
bool _canChangeJoinMuted : 1 = true;
|
bool _canChangeJoinMuted : 1 = true;
|
||||||
bool _allParticipantsLoaded : 1 = false;
|
bool _allParticipantsLoaded : 1 = false;
|
||||||
|
|
|
@ -1347,7 +1347,7 @@ groupCall#d597650c flags:# join_muted:flags.1?true can_change_join_muted:flags.2
|
||||||
inputGroupCall#d8aa840f id:long access_hash:long = InputGroupCall;
|
inputGroupCall#d8aa840f id:long access_hash:long = InputGroupCall;
|
||||||
inputGroupCallSlug#fe06823f slug:string = InputGroupCall;
|
inputGroupCallSlug#fe06823f slug:string = InputGroupCall;
|
||||||
|
|
||||||
groupCallParticipant#23860077 flags:# muted:flags.0?true left:flags.1?true can_self_unmute:flags.2?true just_joined:flags.4?true versioned:flags.5?true min:flags.8?true muted_by_you:flags.9?true volume_by_admin:flags.10?true self:flags.12?true video_joined:flags.15?true peer:Peer date:int active_date:flags.3?int source:int volume:flags.7?int about:flags.11?string raise_hand_rating:flags.13?long video:flags.6?GroupCallParticipantVideo presentation:flags.14?GroupCallParticipantVideo public_key:flags.16?int256 = GroupCallParticipant;
|
groupCallParticipant#eba636fe flags:# muted:flags.0?true left:flags.1?true can_self_unmute:flags.2?true just_joined:flags.4?true versioned:flags.5?true min:flags.8?true muted_by_you:flags.9?true volume_by_admin:flags.10?true self:flags.12?true video_joined:flags.15?true peer:Peer date:int active_date:flags.3?int source:int volume:flags.7?int about:flags.11?string raise_hand_rating:flags.13?long video:flags.6?GroupCallParticipantVideo presentation:flags.14?GroupCallParticipantVideo = GroupCallParticipant;
|
||||||
|
|
||||||
phone.groupCall#9e727aad call:GroupCall participants:Vector<GroupCallParticipant> participants_next_offset:string chats:Vector<Chat> users:Vector<User> = phone.GroupCall;
|
phone.groupCall#9e727aad call:GroupCall participants:Vector<GroupCallParticipant> participants_next_offset:string chats:Vector<Chat> users:Vector<User> = phone.GroupCall;
|
||||||
|
|
||||||
|
|
|
@ -34,6 +34,24 @@ constexpr auto kShortPollChainBlocksWaitFor = crl::time(1000);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] tde2e_api::Slice Slice(const std::vector<uint8_t> &data) {
|
||||||
|
return {
|
||||||
|
reinterpret_cast<const char*>(data.data()),
|
||||||
|
std::string_view::size_type(data.size()),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] ParticipantsSet ParseParticipantsSet(
|
||||||
|
const tde2e_api::CallState &state) {
|
||||||
|
auto result = ParticipantsSet();
|
||||||
|
const auto &list = state.participants;
|
||||||
|
result.list.reserve(list.size());
|
||||||
|
for (const auto &entry : list) {
|
||||||
|
result.list.emplace(UserId{ uint64(entry.user_id) });
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
Call::Call(UserId myUserId)
|
Call::Call(UserId myUserId)
|
||||||
|
@ -48,7 +66,14 @@ Call::Call(UserId myUserId)
|
||||||
memcpy(&_myKey, key.value().data(), sizeof(_myKey));
|
memcpy(&_myKey, key.value().data(), sizeof(_myKey));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Call::~Call() {
|
||||||
|
if (const auto id = libId()) {
|
||||||
|
tde2e_api::call_destroy(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void Call::fail(CallFailure reason) {
|
void Call::fail(CallFailure reason) {
|
||||||
|
_emojiHash = QByteArray();
|
||||||
_failure = reason;
|
_failure = reason;
|
||||||
_failures.fire_copy(reason);
|
_failures.fire_copy(reason);
|
||||||
}
|
}
|
||||||
|
@ -99,20 +124,92 @@ Block Call::makeJoinBlock() {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Block Call::makeRemoveBlock(UserId id) {
|
||||||
|
if (failed() || !_id || id == _myUserId) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
auto state = tde2e_api::call_get_state(libId());
|
||||||
|
if (!state.is_ok()) {
|
||||||
|
LOG_AND_FAIL(state.error(), CallFailure::Unknown);
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
auto found = false;
|
||||||
|
auto updated = state.value();
|
||||||
|
auto &list = updated.participants;
|
||||||
|
for (auto i = begin(list); i != end(list); ++i) {
|
||||||
|
if (uint64(i->user_id) == id.v) {
|
||||||
|
list.erase(i);
|
||||||
|
found = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!found) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
const auto result = tde2e_api::call_create_change_state_block(
|
||||||
|
libId(),
|
||||||
|
updated);
|
||||||
|
if (!result.is_ok()) {
|
||||||
|
LOG_AND_FAIL(result.error(), CallFailure::Unknown);
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
.data = QByteArray::fromStdString(result.value()),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
rpl::producer<ParticipantsSet> Call::participantsSetValue() const {
|
||||||
|
return _participantsSet.value();
|
||||||
|
}
|
||||||
|
|
||||||
void Call::joined() {
|
void Call::joined() {
|
||||||
shortPoll(0);
|
shortPoll(0);
|
||||||
if (_id.v) {
|
if (_id) {
|
||||||
shortPoll(1);
|
shortPoll(1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Call::apply(const Block &last) {
|
void Call::apply(int subchain, const Block &last) {
|
||||||
if (_id.v) {
|
Expects(_id || !subchain);
|
||||||
const auto result = tde2e_api::call_apply_block(
|
|
||||||
std::int64_t(_id.v),
|
auto verification = std::optional<tde2e_api::CallVerificationState>();
|
||||||
|
const auto guard = gsl::finally([&] {
|
||||||
|
if (failed() || !_id) {
|
||||||
|
return;
|
||||||
|
} else if (!verification) {
|
||||||
|
const auto id = libId();
|
||||||
|
auto result = tde2e_api::call_get_verification_state(id);
|
||||||
|
if (!result.is_ok()) {
|
||||||
|
LOG_AND_FAIL(result.error(), CallFailure::Unknown);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
verification = std::move(result.value());
|
||||||
|
}
|
||||||
|
_emojiHash = verification->emoji_hash.has_value()
|
||||||
|
? QByteArray::fromStdString(*verification->emoji_hash)
|
||||||
|
: QByteArray();
|
||||||
|
checkForOutboundMessages();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (subchain) {
|
||||||
|
auto result = tde2e_api::call_receive_inbound_message(
|
||||||
|
libId(),
|
||||||
Slice(last.data));
|
Slice(last.data));
|
||||||
if (!result.is_ok()) {
|
if (!result.is_ok()) {
|
||||||
LOG_AND_FAIL(result.error(), CallFailure::Unknown);
|
LOG_AND_FAIL(result.error(), CallFailure::Unknown);
|
||||||
|
} else {
|
||||||
|
verification = std::move(result.value());
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
} else if (_id) {
|
||||||
|
const auto result = tde2e_api::call_apply_block(
|
||||||
|
libId(),
|
||||||
|
Slice(last.data));
|
||||||
|
if (!result.is_ok()) {
|
||||||
|
LOG_AND_FAIL(result.error(), CallFailure::Unknown);
|
||||||
|
} else {
|
||||||
|
_participantsSet = ParseParticipantsSet(result.value());
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -137,6 +234,26 @@ void Call::apply(const Block &last) {
|
||||||
entry.shortPollTimer.callOnce(kShortPollChainBlocksTimeout);
|
entry.shortPollTimer.callOnce(kShortPollChainBlocksTimeout);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const auto state = tde2e_api::call_get_state(libId());
|
||||||
|
if (!state.is_ok()) {
|
||||||
|
LOG_AND_FAIL(state.error(), CallFailure::Unknown);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_participantsSet = ParseParticipantsSet(state.value());
|
||||||
|
}
|
||||||
|
|
||||||
|
void Call::checkForOutboundMessages() {
|
||||||
|
Expects(_id);
|
||||||
|
|
||||||
|
const auto result = tde2e_api::call_pull_outbound_messages(libId());
|
||||||
|
if (!result.is_ok()) {
|
||||||
|
LOG_AND_FAIL(result.error(), CallFailure::Unknown);
|
||||||
|
return;
|
||||||
|
} else if (!result.value().empty()) {
|
||||||
|
_outboundBlocks.fire(
|
||||||
|
QByteArray::fromStdString(result.value().back()));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Call::apply(
|
void Call::apply(
|
||||||
|
@ -145,7 +262,7 @@ void Call::apply(
|
||||||
const Block &block,
|
const Block &block,
|
||||||
bool fromShortPoll) {
|
bool fromShortPoll) {
|
||||||
Expects(subchain >= 0 && subchain < kSubChainsCount);
|
Expects(subchain >= 0 && subchain < kSubChainsCount);
|
||||||
Expects(_id.v != 0 || !fromShortPoll || !subchain);
|
Expects(_id || !fromShortPoll || !subchain);
|
||||||
|
|
||||||
if (!subchain && index >= _lastBlock0Height) {
|
if (!subchain && index >= _lastBlock0Height) {
|
||||||
_lastBlock0 = block;
|
_lastBlock0 = block;
|
||||||
|
@ -158,7 +275,7 @@ void Call::apply(
|
||||||
auto &entry = _subchains[subchain];
|
auto &entry = _subchains[subchain];
|
||||||
if (!fromShortPoll) {
|
if (!fromShortPoll) {
|
||||||
entry.lastUpdate = crl::now();
|
entry.lastUpdate = crl::now();
|
||||||
if (index > entry.height || (!_id.v && subchain != 0)) {
|
if (index > entry.height || (!_id && subchain != 0)) {
|
||||||
entry.waiting.emplace(index, block);
|
entry.waiting.emplace(index, block);
|
||||||
checkWaitingBlocks(subchain);
|
checkWaitingBlocks(subchain);
|
||||||
return;
|
return;
|
||||||
|
@ -167,8 +284,8 @@ void Call::apply(
|
||||||
|
|
||||||
if (failed()) {
|
if (failed()) {
|
||||||
return;
|
return;
|
||||||
} else if (!_id.v || entry.height == index) {
|
} else if (!_id || entry.height == index) {
|
||||||
apply(block);
|
apply(subchain, block);
|
||||||
}
|
}
|
||||||
entry.height = index + 1;
|
entry.height = index + 1;
|
||||||
checkWaitingBlocks(subchain);
|
checkWaitingBlocks(subchain);
|
||||||
|
@ -182,7 +299,7 @@ void Call::checkWaitingBlocks(int subchain, bool waited) {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto &entry = _subchains[subchain];
|
auto &entry = _subchains[subchain];
|
||||||
if (!_id.v) {
|
if (!_id) {
|
||||||
entry.waitingTimer.callOnce(kShortPollChainBlocksWaitFor);
|
entry.waitingTimer.callOnce(kShortPollChainBlocksWaitFor);
|
||||||
return;
|
return;
|
||||||
} else if (entry.shortPolling) {
|
} else if (entry.shortPolling) {
|
||||||
|
@ -200,12 +317,28 @@ void Call::checkWaitingBlocks(int subchain, bool waited) {
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
} else if (level == entry.height + 1) {
|
} else if (level == entry.height + 1) {
|
||||||
const auto result = tde2e_api::call_apply_block(
|
const auto slice = Slice(waiting.begin()->second.data);
|
||||||
std::int64_t(_id.v),
|
if (subchain) {
|
||||||
Slice(waiting.begin()->second.data));
|
auto result = tde2e_api::call_receive_inbound_message(
|
||||||
if (!result.is_ok()) {
|
libId(),
|
||||||
LOG_AND_FAIL(result.error(), CallFailure::Unknown);
|
slice);
|
||||||
return;
|
if (!result.is_ok()) {
|
||||||
|
LOG_AND_FAIL(result.error(), CallFailure::Unknown);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_emojiHash = result.value().emoji_hash.has_value()
|
||||||
|
? QByteArray::fromStdString(*result.value().emoji_hash)
|
||||||
|
: QByteArray();
|
||||||
|
checkForOutboundMessages();
|
||||||
|
} else {
|
||||||
|
const auto result = tde2e_api::call_apply_block(
|
||||||
|
libId(),
|
||||||
|
slice);
|
||||||
|
if (!result.is_ok()) {
|
||||||
|
LOG_AND_FAIL(result.error(), CallFailure::Unknown);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_participantsSet = ParseParticipantsSet(result.value());
|
||||||
}
|
}
|
||||||
entry.height = level;
|
entry.height = level;
|
||||||
}
|
}
|
||||||
|
@ -221,7 +354,7 @@ void Call::shortPoll(int subchain) {
|
||||||
auto &entry = _subchains[subchain];
|
auto &entry = _subchains[subchain];
|
||||||
entry.waitingTimer.cancel();
|
entry.waitingTimer.cancel();
|
||||||
entry.shortPollTimer.cancel();
|
entry.shortPollTimer.cancel();
|
||||||
if (subchain && !_id.v) {
|
if (subchain && !_id) {
|
||||||
// Not ready.
|
// Not ready.
|
||||||
entry.waitingTimer.callOnce(kShortPollChainBlocksWaitFor);
|
entry.waitingTimer.callOnce(kShortPollChainBlocksWaitFor);
|
||||||
return;
|
return;
|
||||||
|
@ -230,6 +363,10 @@ void Call::shortPoll(int subchain) {
|
||||||
_subchainRequests.fire({ subchain, entry.height });
|
_subchainRequests.fire({ subchain, entry.height });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::int64_t Call::libId() const {
|
||||||
|
return std::int64_t(_id.v);
|
||||||
|
}
|
||||||
|
|
||||||
rpl::producer<Call::SubchainRequest> Call::subchainRequests() const {
|
rpl::producer<Call::SubchainRequest> Call::subchainRequests() const {
|
||||||
return _subchainRequests.events();
|
return _subchainRequests.events();
|
||||||
}
|
}
|
||||||
|
@ -246,6 +383,10 @@ void Call::subchainBlocksRequestFinished(int subchain) {
|
||||||
checkWaitingBlocks(subchain);
|
checkWaitingBlocks(subchain);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
rpl::producer<QByteArray> Call::sendOutboundBlock() const {
|
||||||
|
return _outboundBlocks.events();
|
||||||
|
}
|
||||||
|
|
||||||
std::optional<CallFailure> Call::failed() const {
|
std::optional<CallFailure> Call::failed() const {
|
||||||
return _failure;
|
return _failure;
|
||||||
}
|
}
|
||||||
|
@ -257,13 +398,16 @@ rpl::producer<CallFailure> Call::failures() const {
|
||||||
return _failures.events();
|
return _failures.events();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QByteArray Call::emojiHash() const {
|
||||||
|
return _emojiHash.current();
|
||||||
|
}
|
||||||
|
|
||||||
|
rpl::producer<QByteArray> Call::emojiHashValue() const {
|
||||||
|
return _emojiHash.value();
|
||||||
|
}
|
||||||
|
|
||||||
std::vector<uint8_t> Call::encrypt(const std::vector<uint8_t> &data) const {
|
std::vector<uint8_t> Call::encrypt(const std::vector<uint8_t> &data) const {
|
||||||
const auto result = tde2e_api::call_encrypt(
|
const auto result = tde2e_api::call_encrypt(libId(), Slice(data));
|
||||||
std::int64_t(_id.v),
|
|
||||||
std::string_view{
|
|
||||||
reinterpret_cast<const char*>(data.data()),
|
|
||||||
data.size(),
|
|
||||||
});
|
|
||||||
if (!result.is_ok()) {
|
if (!result.is_ok()) {
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
@ -274,12 +418,7 @@ std::vector<uint8_t> Call::encrypt(const std::vector<uint8_t> &data) const {
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<uint8_t> Call::decrypt(const std::vector<uint8_t> &data) const {
|
std::vector<uint8_t> Call::decrypt(const std::vector<uint8_t> &data) const {
|
||||||
const auto result = tde2e_api::call_decrypt(
|
const auto result = tde2e_api::call_decrypt(libId(), Slice(data));
|
||||||
std::int64_t(_id.v),
|
|
||||||
std::string_view{
|
|
||||||
reinterpret_cast<const char*>(data.data()),
|
|
||||||
data.size(),
|
|
||||||
});
|
|
||||||
if (!result.is_ok()) {
|
if (!result.is_ok()) {
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,10 +8,12 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "base/basic_types.h"
|
#include "base/basic_types.h"
|
||||||
|
#include "base/flat_set.h"
|
||||||
#include "base/timer.h"
|
#include "base/timer.h"
|
||||||
|
|
||||||
#include <rpl/producer.h>
|
|
||||||
#include <rpl/event_stream.h>
|
#include <rpl/event_stream.h>
|
||||||
|
#include <rpl/producer.h>
|
||||||
|
#include <rpl/variable.h>
|
||||||
|
|
||||||
#include <crl/crl_time.h>
|
#include <crl/crl_time.h>
|
||||||
|
|
||||||
|
@ -19,6 +21,9 @@ namespace TdE2E {
|
||||||
|
|
||||||
struct UserId {
|
struct UserId {
|
||||||
uint64 v = 0;
|
uint64 v = 0;
|
||||||
|
|
||||||
|
friend inline constexpr auto operator<=>(UserId, UserId) = default;
|
||||||
|
friend inline constexpr bool operator==(UserId, UserId) = default;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct PrivateKeyId {
|
struct PrivateKeyId {
|
||||||
|
@ -27,6 +32,10 @@ struct PrivateKeyId {
|
||||||
|
|
||||||
struct CallId {
|
struct CallId {
|
||||||
uint64 v = 0;
|
uint64 v = 0;
|
||||||
|
|
||||||
|
explicit operator bool() const {
|
||||||
|
return v != 0;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
struct PublicKey {
|
struct PublicKey {
|
||||||
|
@ -41,6 +50,14 @@ struct ParticipantState {
|
||||||
PublicKey key;
|
PublicKey key;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct ParticipantsSet {
|
||||||
|
base::flat_set<UserId> list;
|
||||||
|
|
||||||
|
friend inline bool operator==(
|
||||||
|
const ParticipantsSet &,
|
||||||
|
const ParticipantsSet &) = default;
|
||||||
|
};
|
||||||
|
|
||||||
struct Block {
|
struct Block {
|
||||||
QByteArray data;
|
QByteArray data;
|
||||||
};
|
};
|
||||||
|
@ -52,6 +69,7 @@ enum class CallFailure {
|
||||||
class Call final {
|
class Call final {
|
||||||
public:
|
public:
|
||||||
explicit Call(UserId myUserId);
|
explicit Call(UserId myUserId);
|
||||||
|
~Call();
|
||||||
|
|
||||||
[[nodiscard]] PublicKey myKey() const;
|
[[nodiscard]] PublicKey myKey() const;
|
||||||
|
|
||||||
|
@ -69,11 +87,19 @@ public:
|
||||||
[[nodiscard]] rpl::producer<SubchainRequest> subchainRequests() const;
|
[[nodiscard]] rpl::producer<SubchainRequest> subchainRequests() const;
|
||||||
void subchainBlocksRequestFinished(int subchain);
|
void subchainBlocksRequestFinished(int subchain);
|
||||||
|
|
||||||
|
[[nodiscard]] rpl::producer<QByteArray> sendOutboundBlock() const;
|
||||||
|
|
||||||
[[nodiscard]] std::optional<CallFailure> failed() const;
|
[[nodiscard]] std::optional<CallFailure> failed() const;
|
||||||
[[nodiscard]] rpl::producer<CallFailure> failures() const;
|
[[nodiscard]] rpl::producer<CallFailure> failures() const;
|
||||||
|
|
||||||
|
[[nodiscard]] QByteArray emojiHash() const;
|
||||||
|
[[nodiscard]] rpl::producer<QByteArray> emojiHashValue() const;
|
||||||
|
|
||||||
void refreshLastBlock0(std::optional<Block> block);
|
void refreshLastBlock0(std::optional<Block> block);
|
||||||
[[nodiscard]] Block makeJoinBlock();
|
[[nodiscard]] Block makeJoinBlock();
|
||||||
|
[[nodiscard]] Block makeRemoveBlock(UserId id);
|
||||||
|
|
||||||
|
[[nodiscard]] rpl::producer<ParticipantsSet> participantsSetValue() const;
|
||||||
|
|
||||||
[[nodiscard]] std::vector<uint8_t> encrypt(
|
[[nodiscard]] std::vector<uint8_t> encrypt(
|
||||||
const std::vector<uint8_t> &data) const;
|
const std::vector<uint8_t> &data) const;
|
||||||
|
@ -92,12 +118,15 @@ private:
|
||||||
int height = 0;
|
int height = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
void apply(const Block &last);
|
void apply(int subchain, const Block &last);
|
||||||
void fail(CallFailure reason);
|
void fail(CallFailure reason);
|
||||||
|
|
||||||
|
void checkForOutboundMessages();
|
||||||
void checkWaitingBlocks(int subchain, bool waited = false);
|
void checkWaitingBlocks(int subchain, bool waited = false);
|
||||||
void shortPoll(int subchain);
|
void shortPoll(int subchain);
|
||||||
|
|
||||||
|
[[nodiscard]] std::int64_t libId() const;
|
||||||
|
|
||||||
CallId _id;
|
CallId _id;
|
||||||
UserId _myUserId;
|
UserId _myUserId;
|
||||||
PrivateKeyId _myKeyId;
|
PrivateKeyId _myKeyId;
|
||||||
|
@ -107,10 +136,14 @@ private:
|
||||||
|
|
||||||
SubChainState _subchains[kSubChainsCount];
|
SubChainState _subchains[kSubChainsCount];
|
||||||
rpl::event_stream<SubchainRequest> _subchainRequests;
|
rpl::event_stream<SubchainRequest> _subchainRequests;
|
||||||
|
rpl::event_stream<QByteArray> _outboundBlocks;
|
||||||
|
|
||||||
std::optional<Block> _lastBlock0;
|
std::optional<Block> _lastBlock0;
|
||||||
int _lastBlock0Height = 0;
|
int _lastBlock0Height = 0;
|
||||||
|
|
||||||
|
rpl::variable<ParticipantsSet> _participantsSet;
|
||||||
|
rpl::variable<QByteArray> _emojiHash;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace TdE2E
|
} // namespace TdE2E
|
||||||
|
|
Loading…
Add table
Reference in a new issue