mirror of
https://git.jami.net/savoirfairelinux/jami-daemon.git
synced 2025-08-12 22:09:25 +08:00
329 lines
13 KiB
C++
Executable File
329 lines
13 KiB
C++
Executable File
#pragma once
|
|
|
|
#define V8_STRING_NEW(str) v8::String::NewFromUtf8(v8::Isolate::GetCurrent(), str.data(), v8::String::kNormalString, str.size())
|
|
|
|
#include <uv.h>
|
|
#include <queue>
|
|
#include <functional>
|
|
#include <mutex>
|
|
|
|
using namespace v8;
|
|
|
|
Persistent<Function> accountsChangedCb;
|
|
Persistent<Function> registrationStateChangedCb;
|
|
Persistent<Function> volatileDetailsChangedCb;
|
|
Persistent<Function> incomingAccountMessageCb;
|
|
Persistent<Function> accountMessageStatusChangedCb;
|
|
Persistent<Function> incomingTrustRequestCb;
|
|
Persistent<Function> contactAddedCb;
|
|
Persistent<Function> contactRemovedCb;
|
|
Persistent<Function> exportOnRingEndedCb;
|
|
Persistent<Function> nameRegistrationEndedCb;
|
|
Persistent<Function> knownDevicesChangedCb;
|
|
Persistent<Function> registeredNameFoundCb;
|
|
Persistent<Function> callStateChangedCb;
|
|
Persistent<Function> incomingMessageCb;
|
|
Persistent<Function> incomingCallCb;
|
|
|
|
std::queue<std::function<void() >> pendingSignals;
|
|
std::mutex pendingSignalsLock;
|
|
|
|
uv_async_t signalAsync;
|
|
|
|
Persistent<Function>* getPresistentCb(const std::string &signal) {
|
|
if (signal == "AccountsChanged")
|
|
return &accountsChangedCb;
|
|
else if (signal == "RegistrationStateChanged")
|
|
return ®istrationStateChangedCb;
|
|
else if (signal == "VolatileDetailsChanged")
|
|
return &volatileDetailsChangedCb;
|
|
else if (signal == "IncomingAccountMessage")
|
|
return &incomingAccountMessageCb;
|
|
else if (signal == "AccountMessageStatusChanged")
|
|
return &accountMessageStatusChangedCb;
|
|
else if (signal == "IncomingTrustRequest")
|
|
return &incomingTrustRequestCb;
|
|
else if (signal == "ContactAdded")
|
|
return &contactAddedCb;
|
|
else if (signal == "ContactRemoved")
|
|
return &contactRemovedCb;
|
|
else if (signal == "ExportOnRingEnded")
|
|
return &exportOnRingEndedCb;
|
|
else if (signal == "NameRegistrationEnded")
|
|
return &nameRegistrationEndedCb;
|
|
else if (signal == "KnownDevicesChanged")
|
|
return &knownDevicesChangedCb;
|
|
else if (signal == "RegisteredNameFound")
|
|
return ®isteredNameFoundCb;
|
|
else if (signal == "CallStateChanged")
|
|
return &callStateChangedCb;
|
|
else if (signal == "IncomingMessage")
|
|
return &incomingMessageCb;
|
|
else if (signal == "IncomingCall")
|
|
return &incomingCallCb;
|
|
else return nullptr;
|
|
}
|
|
|
|
void intVectToJsArray(const std::vector<uint8_t>& intVect, const Local<Array>& jsArray) {
|
|
for (unsigned int i = 0; i < intVect.size(); i++)
|
|
jsArray->Set(SWIGV8_INTEGER_NEW_UNS(i), SWIGV8_INTEGER_NEW(intVect[i]));
|
|
}
|
|
|
|
void stringMapToJsMap(const std::map<std::string, std::string>& strmap, const Local<Object> &jsMap) {
|
|
for (auto& kvpair : strmap)
|
|
jsMap->Set(V8_STRING_NEW(std::get<0>(kvpair)), V8_STRING_NEW(std::get<1>(kvpair)));
|
|
}
|
|
|
|
void setCallback(const std::string& signal, Local<Function>& func) {
|
|
if (auto* presistentCb = getPresistentCb(signal)) {
|
|
if (func->IsObject() && func->IsFunction()) {
|
|
presistentCb->Reset(Isolate::GetCurrent(), func);
|
|
} else {
|
|
presistentCb->Reset();
|
|
}
|
|
} else {
|
|
printf("No Signal Associated with Event \'%s\'\n", signal.c_str());
|
|
}
|
|
}
|
|
|
|
void parseCbMap(const Local<Value>& callbackMap) {
|
|
Local<Object> cbAssocArray = callbackMap->ToObject();
|
|
Local<Array> props = cbAssocArray->GetOwnPropertyNames();
|
|
for (uint32_t i = 0; i < props->Length(); ++i) {
|
|
const Local<Value> key_local = props->Get(i);
|
|
std::string key = *String::Utf8Value(key_local);
|
|
Handle<Object> buffer = cbAssocArray->Get(V8_STRING_NEW(key))->ToObject();
|
|
Local<Function> func = Local<Function>::Cast(buffer);
|
|
setCallback(key, func);
|
|
}
|
|
}
|
|
|
|
void handlePendingSignals(uv_async_t* async_data) {
|
|
SWIGV8_HANDLESCOPE();
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
while (not pendingSignals.empty()) {
|
|
pendingSignals.front()();
|
|
pendingSignals.pop();
|
|
}
|
|
}
|
|
|
|
void registrationStateChanged(const std::string& account_id, const std::string& state, int code, const std::string& detail_str) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, state, code, detail_str]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), registrationStateChangedCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), V8_STRING_NEW(state), SWIGV8_INTEGER_NEW(code), V8_STRING_NEW(detail_str)};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 4, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void volatileDetailsChanged(const std::string& account_id, const std::map<std::string, std::string>& details) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, details]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), volatileDetailsChangedCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Object> jsMap = SWIGV8_OBJECT_NEW();
|
|
stringMapToJsMap(details, jsMap);
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), jsMap};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 2, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void accountsChanged() {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), accountsChangedCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Value> callback_args[] = {};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 0, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void contactAdded(const std::string& account_id, const std::string& uri, bool confirmed) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, uri, confirmed]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), contactAddedCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), V8_STRING_NEW(uri), SWIGV8_BOOLEAN_NEW(confirmed)};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 3, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void contactRemoved(const std::string& account_id, const std::string& uri, bool banned) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, uri, banned]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), contactRemovedCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), V8_STRING_NEW(uri), SWIGV8_BOOLEAN_NEW(banned)};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 3, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void exportOnRingEnded(const std::string& account_id, int state, const std::string& pin) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, state, pin]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), exportOnRingEndedCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), SWIGV8_INTEGER_NEW(state), V8_STRING_NEW(pin)};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 3, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void nameRegistrationEnded(const std::string& account_id, int state, const std::string& name) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, state, name]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), nameRegistrationEndedCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), SWIGV8_INTEGER_NEW(state), V8_STRING_NEW(name)};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 3, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void registeredNameFound(const std::string& account_id, int state, const std::string& address, const std::string& name) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, state, address, name]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), registeredNameFoundCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), SWIGV8_INTEGER_NEW(state), V8_STRING_NEW(address), V8_STRING_NEW(name)};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 4, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void accountMessageStatusChanged(const std::string& account_id, uint64_t message_id, const std::string& to, int state) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, message_id, to, state]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), accountMessageStatusChangedCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), SWIGV8_INTEGER_NEW_UNS(message_id), V8_STRING_NEW(to), SWIGV8_INTEGER_NEW(state)};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 4, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void incomingAccountMessage(const std::string& account_id, const std::string& messageId, const std::string& from, const std::map<std::string, std::string>& payloads) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, from, payloads]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), incomingAccountMessageCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Object> jsMap = SWIGV8_OBJECT_NEW();
|
|
stringMapToJsMap(payloads, jsMap);
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), V8_STRING_NEW(from), jsMap};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 3, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void knownDevicesChanged(const std::string& account_id, const std::map<std::string, std::string>& devices) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, devices]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), knownDevicesChangedCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Object> jsMap = SWIGV8_OBJECT_NEW();
|
|
stringMapToJsMap(devices, jsMap);
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), jsMap};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 2, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void incomingTrustRequest(const std::string& account_id, const std::string& from, const std::vector<uint8_t>& payload, time_t received) {
|
|
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, from, payload, received]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), incomingTrustRequestCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Array> jsArray = SWIGV8_ARRAY_NEW(payload.size());
|
|
intVectToJsArray(payload, jsArray);
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), V8_STRING_NEW(from), jsArray, SWIGV8_NUMBER_NEW(received)};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 4, callback_args);
|
|
}
|
|
});
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void callStateChanged(const std::string& call_id, const std::string& state, int detail_code) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([call_id, state, detail_code]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), callStateChangedCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(call_id), V8_STRING_NEW(state), SWIGV8_INTEGER_NEW(detail_code)};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 3, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void incomingMessage(const std::string& id, const std::string& from, const std::map<std::string, std::string>& messages) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([id, from, messages]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), incomingMessageCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Object> jsMap = SWIGV8_OBJECT_NEW();
|
|
stringMapToJsMap(messages, jsMap);
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(id), V8_STRING_NEW(from), jsMap};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 4, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|
|
|
|
void incomingCall(const std::string& account_id, const std::string& call_id, const std::string& from) {
|
|
|
|
std::lock_guard<std::mutex> lock(pendingSignalsLock);
|
|
pendingSignals.emplace([account_id, call_id, from]() {
|
|
Local<Function> func = Local<Function>::New(Isolate::GetCurrent(), incomingCallCb);
|
|
if (!func.IsEmpty()) {
|
|
Local<Value> callback_args[] = {V8_STRING_NEW(account_id), V8_STRING_NEW(call_id), V8_STRING_NEW(from)};
|
|
func->Call(SWIGV8_CURRENT_CONTEXT()->Global(), 3, callback_args);
|
|
}
|
|
});
|
|
|
|
uv_async_send(&signalAsync);
|
|
}
|