Files
sh d3a72473e9 nodejs: add and fix windows (#6581)
* simplex-chat-nodejs: adjust binding.gyp for windows

* simplex-chat-nodejs: different library linkage for windows

* simplex-chat-nodejs: remove non-moving GC in Windows

"non-moving GC is broken on windows with GHC 9.4-9.6.3" from:

https://github.com/simplex-chat/simplex-chat/blob/master/apps/multiplatform/common/src/commonMain/cpp/desktop/simplex-api.c#L11-L17

* ci: add windows to release-nodejs-libs

* simplex-chat-nodejs: same curl flags for dll.def download
2026-01-15 16:36:35 +00:00

427 lines
13 KiB
C++

#include <napi.h>
#include <sstream>
#include <string>
#include <functional>
#include <cstdlib>
#include "simplex.h"
namespace simplex {
using namespace Napi;
void haskell_init() {
#ifdef _WIN32
// non-moving GC is broken on windows with GHC 9.4-9.6.3
int argc = 5;
const char *argv[] = {
"simplex",
"+RTS", // requires `hs_init_with_rtsopts`
"-A64m", // chunk size for new allocations
"-H64m", // initial heap size
"--install-signal-handlers=no",
nullptr};
#else
int argc = 6;
const char *argv[] = {
"simplex",
"+RTS", // requires `hs_init_with_rtsopts`
"-A64m", // chunk size for new allocations
"-H64m", // initial heap size
"-xn", // non-moving GC
"--install-signal-handlers=no",
nullptr};
#endif
char **pargv = const_cast<char **>(argv);
hs_init_with_rtsopts(&argc, &pargv);
}
class ResultAsyncWorker : public AsyncWorker {
public:
using ExecuteFn = std::function<void(ResultAsyncWorker*)>;
using ResultProcessor = std::function<void(ResultAsyncWorker*, Napi::Env)>;
ResultAsyncWorker(Function& callback, ExecuteFn execute_fn, ResultProcessor result_processor = nullptr)
: AsyncWorker(callback), execute_fn_(std::move(execute_fn)), result_processor_(std::move(result_processor)) {}
void Execute() override {
execute_fn_(this);
}
void OnOK() override {
HandleScope scope(Env());
if (result_processor_) {
result_processor_(this, Env());
} else {
Callback().Call({Env().Null(), String::New(Env(), result_)});
}
}
void OnError(const Error& e) override {
HandleScope scope(Env());
Callback().Call({e.Value(), Env().Undefined()});
}
void SetResult(std::string result) {
result_ = std::move(result);
}
void SetWorkerError(const std::string& msg) {
SetError(msg);
}
const std::string& GetStringResult() const {
return result_;
}
void SetCtrl(uintptr_t ctrl) {
ctrl_ = ctrl;
}
uintptr_t GetCtrl() const {
return ctrl_;
}
protected:
std::string result_;
uintptr_t ctrl_ = 0;
private:
ExecuteFn execute_fn_;
ResultProcessor result_processor_;
};
class BinaryAsyncWorker : public AsyncWorker {
public:
using ExecuteFn = std::function<void(BinaryAsyncWorker*)>;
BinaryAsyncWorker(Function& callback, ExecuteFn execute_fn)
: AsyncWorker(callback), execute_fn_(std::move(execute_fn)) {}
void Execute() override {
execute_fn_(this);
}
void OnOK() override {
HandleScope scope(Env());
if (original_buf == nullptr || binary_len == 0) {
Callback().Call({Env().Null(), Env().Undefined()});
return;
}
char* data_ptr = original_buf + 5;
auto finalizer = [](Napi::Env env, char* finalize_data, char* orig) {
free(orig);
};
Napi::Buffer<char> buffer = Napi::Buffer<char>::New(Env(), data_ptr, binary_len, finalizer, original_buf);
Callback().Call({Env().Null(), buffer});
}
void OnError(const Error& e) override {
HandleScope scope(Env());
Callback().Call({e.Value(), Env().Undefined()});
}
void SetWorkerError(const std::string& msg) {
SetError(msg);
}
char* original_buf = nullptr;
size_t binary_len = 0;
private:
ExecuteFn execute_fn_;
};
// Helper for converting chat_ctrl pointer to BigInt
Napi::BigInt ToChatCtrlBigInt(Napi::Env env, uintptr_t ctrl) {
return Napi::BigInt::New(env, static_cast<uint64_t>(ctrl));
}
// Helper for converting BigInt to chat_ctrl pointer
chat_ctrl FromChatCtrlBigInt(const Napi::Value& value) {
Napi::Env env = value.Env();
if (!value.IsBigInt()) {
Napi::TypeError::New(env, "Expected BigInt for ctrl").ThrowAsJavaScriptException();
return nullptr;
}
Napi::BigInt big = value.As<Napi::BigInt>();
bool lossless;
uint64_t val = big.Uint64Value(&lossless);
if (!lossless) {
Napi::TypeError::New(env, "BigInt too large for ctrl").ThrowAsJavaScriptException();
return nullptr;
}
return reinterpret_cast<chat_ctrl>(val);
}
// Helper for handling common C result patterns (no empty check)
void HandleCResult(ResultAsyncWorker* worker, char* c_res, const std::string& func_name) {
if (c_res == nullptr) {
worker->SetWorkerError(func_name + " failed");
return;
}
std::string res = c_res;
free(c_res);
worker->SetResult(res);
}
Napi::Promise CreatePromiseAndCallback(Env env, Function& cb_out) {
Promise::Deferred deferred = Promise::Deferred::New(env);
cb_out = Function::New(env, [deferred](const CallbackInfo& args) {
if (!args[0].IsNull() && !args[0].IsUndefined()) {
deferred.Reject(args[0]);
} else {
deferred.Resolve(args[1]);
}
});
return deferred.Promise();
}
// Common result processors
ResultAsyncWorker::ResultProcessor MigrateResultProcessor() {
return [](ResultAsyncWorker* worker, Napi::Env env) {
Napi::Array arr = Napi::Array::New(env, 2);
arr.Set(0u, ToChatCtrlBigInt(env, worker->GetCtrl()));
arr.Set(1u, Napi::String::New(env, worker->GetStringResult()));
worker->Callback().Call({env.Null(), arr});
};
}
// Refactored functions using common patterns
Value ChatMigrateInit(const CallbackInfo& args) {
Env env = args.Env();
if (args.Length() < 3 || !args[0].IsString() || !args[1].IsString() || !args[2].IsString()) {
TypeError::New(env, "Expected three string arguments").ThrowAsJavaScriptException();
return env.Undefined();
}
std::string path = args[0].As<String>().Utf8Value();
std::string key = args[1].As<String>().Utf8Value();
std::string confirm = args[2].As<String>().Utf8Value();
Function cb;
Promise promise = CreatePromiseAndCallback(env, cb);
auto execute_fn = [path, key, confirm](ResultAsyncWorker* worker) {
chat_ctrl ctrl = nullptr;
char* c_res = chat_migrate_init(path.c_str(), key.c_str(), confirm.c_str(), &ctrl);
worker->SetCtrl(reinterpret_cast<uintptr_t>(ctrl));
HandleCResult(worker, c_res, "chat_migrate_init");
};
ResultAsyncWorker* worker = new ResultAsyncWorker(cb, std::move(execute_fn), MigrateResultProcessor());
worker->Queue();
return promise;
}
Value ChatCloseStore(const CallbackInfo& args) {
Env env = args.Env();
if (args.Length() < 1 || !args[0].IsBigInt()) {
TypeError::New(env, "Expected bigint (ctrl)").ThrowAsJavaScriptException();
return env.Undefined();
}
chat_ctrl ctrl = FromChatCtrlBigInt(args[0]);
Function cb;
Promise promise = CreatePromiseAndCallback(env, cb);
auto execute_fn = [ctrl](ResultAsyncWorker* worker) {
char* c_res = chat_close_store(ctrl);
HandleCResult(worker, c_res, "chat_close_store");
};
ResultAsyncWorker* worker = new ResultAsyncWorker(cb, std::move(execute_fn));
worker->Queue();
return promise;
}
Value ChatSendCmd(const CallbackInfo& args) {
Env env = args.Env();
if (args.Length() < 2 || !args[0].IsBigInt() || !args[1].IsString()) {
TypeError::New(env, "Expected bigint (ctrl) and string (cmd)").ThrowAsJavaScriptException();
return env.Undefined();
}
chat_ctrl ctrl = FromChatCtrlBigInt(args[0]);
std::string cmd = args[1].As<String>().Utf8Value();
Function cb;
Promise promise = CreatePromiseAndCallback(env, cb);
auto execute_fn = [ctrl, cmd](ResultAsyncWorker* worker) {
char* c_res = chat_send_cmd(ctrl, cmd.c_str());
HandleCResult(worker, c_res, "chat_send_cmd");
};
ResultAsyncWorker* worker = new ResultAsyncWorker(cb, std::move(execute_fn));
worker->Queue();
return promise;
}
Value ChatRecvMsgWait(const CallbackInfo& args) {
Env env = args.Env();
if (args.Length() < 2 || !args[0].IsBigInt() || !args[1].IsNumber()) {
TypeError::New(env, "Expected bigint (ctrl), number (wait)").ThrowAsJavaScriptException();
return env.Undefined();
}
chat_ctrl ctrl = FromChatCtrlBigInt(args[0]);
int wait = static_cast<int>(args[1].As<Number>().Int32Value());
Function cb;
Promise promise = CreatePromiseAndCallback(env, cb);
auto execute_fn = [ctrl, wait](ResultAsyncWorker* worker) {
char* c_res = chat_recv_msg_wait(ctrl, wait);
HandleCResult(worker, c_res, "chat_recv_msg_wait");
};
ResultAsyncWorker* worker = new ResultAsyncWorker(cb, std::move(execute_fn));
worker->Queue();
return promise;
}
Value ChatWriteFile(const CallbackInfo& args) {
Env env = args.Env();
if (args.Length() < 3 || !args[0].IsBigInt() || !args[1].IsString() || !args[2].IsArrayBuffer()) {
TypeError::New(env, "Expected bigint (ctrl), string (path), ArrayBuffer").ThrowAsJavaScriptException();
return env.Undefined();
}
chat_ctrl ctrl = FromChatCtrlBigInt(args[0]);
std::string path = args[1].As<String>().Utf8Value();
ArrayBuffer ab = args[2].As<ArrayBuffer>();
char* data = static_cast<char*>(ab.Data());
size_t len = ab.ByteLength();
Function cb;
Promise promise = CreatePromiseAndCallback(env, cb);
auto execute_fn = [ctrl, path, ab, data, len](ResultAsyncWorker* worker) {
(void)ab; // to keep ArrayBuffer alive
char* c_res = chat_write_file(ctrl, path.c_str(), data, static_cast<int>(len));
HandleCResult(worker, c_res, "chat_write_file");
};
ResultAsyncWorker* worker = new ResultAsyncWorker(cb, std::move(execute_fn));
worker->Queue();
return promise;
}
Value ChatReadFile(const CallbackInfo& args) {
Env env = args.Env();
if (args.Length() < 3 || !args[0].IsString() || !args[1].IsString() || !args[2].IsString()) {
TypeError::New(env, "Expected three strings (path, key, nonce)").ThrowAsJavaScriptException();
return env.Undefined();
}
std::string path = args[0].As<String>().Utf8Value();
std::string key = args[1].As<String>().Utf8Value();
std::string nonce = args[2].As<String>().Utf8Value();
Function cb;
Promise promise = CreatePromiseAndCallback(env, cb);
auto execute_fn = [path, key, nonce](BinaryAsyncWorker* worker) {
char* buf = chat_read_file(path.c_str(), key.c_str(), nonce.c_str());
if (buf == nullptr) {
worker->SetWorkerError("chat_read_file failed");
return;
}
char status = buf[0];
if (status == 1) {
std::string err = buf + 1;
free(buf);
worker->SetWorkerError(err);
return;
} else if (status == 0) {
uint32_t len = *(uint32_t*)(buf + 1);
worker->original_buf = buf;
worker->binary_len = len;
} else {
free(buf);
worker->SetWorkerError("Unexpected status from chat_read_file");
return;
}
};
BinaryAsyncWorker* worker = new BinaryAsyncWorker(cb, std::move(execute_fn));
worker->Queue();
return promise;
}
Value ChatEncryptFile(const CallbackInfo& args) {
Env env = args.Env();
if (args.Length() < 3 || !args[0].IsBigInt() || !args[1].IsString() || !args[2].IsString()) {
TypeError::New(env, "Expected bigint (ctrl), two strings (fromPath, toPath)").ThrowAsJavaScriptException();
return env.Undefined();
}
chat_ctrl ctrl = FromChatCtrlBigInt(args[0]);
std::string fromPath = args[1].As<String>().Utf8Value();
std::string toPath = args[2].As<String>().Utf8Value();
Function cb;
Promise promise = CreatePromiseAndCallback(env, cb);
auto execute_fn = [ctrl, fromPath, toPath](ResultAsyncWorker* worker) {
char* c_res = chat_encrypt_file(ctrl, fromPath.c_str(), toPath.c_str());
HandleCResult(worker, c_res, "chat_encrypt_file");
};
ResultAsyncWorker* worker = new ResultAsyncWorker(cb, std::move(execute_fn));
worker->Queue();
return promise;
}
Value ChatDecryptFile(const CallbackInfo& args) {
Env env = args.Env();
if (args.Length() < 4 || !args[0].IsString() || !args[1].IsString() || !args[2].IsString() || !args[3].IsString()) {
TypeError::New(env, "Expected four strings (fromPath, key, nonce, toPath)").ThrowAsJavaScriptException();
return env.Undefined();
}
std::string fromPath = args[0].As<String>().Utf8Value();
std::string key = args[1].As<String>().Utf8Value();
std::string nonce = args[2].As<String>().Utf8Value();
std::string toPath = args[3].As<String>().Utf8Value();
Function cb;
Promise promise = CreatePromiseAndCallback(env, cb);
auto execute_fn = [fromPath, key, nonce, toPath](ResultAsyncWorker* worker) {
char* c_res = chat_decrypt_file(fromPath.c_str(), key.c_str(), nonce.c_str(), toPath.c_str());
HandleCResult(worker, c_res, "chat_decrypt_file");
};
ResultAsyncWorker* worker = new ResultAsyncWorker(cb, std::move(execute_fn));
worker->Queue();
return promise;
}
Object Init(Env env, Object exports) {
haskell_init();
exports.Set("chat_migrate_init", Function::New(env, ChatMigrateInit));
exports.Set("chat_close_store", Function::New(env, ChatCloseStore));
exports.Set("chat_send_cmd", Function::New(env, ChatSendCmd));
exports.Set("chat_recv_msg_wait", Function::New(env, ChatRecvMsgWait));
exports.Set("chat_write_file", Function::New(env, ChatWriteFile));
exports.Set("chat_read_file", Function::New(env, ChatReadFile));
exports.Set("chat_encrypt_file", Function::New(env, ChatEncryptFile));
exports.Set("chat_decrypt_file", Function::New(env, ChatDecryptFile));
return exports;
}
NODE_API_MODULE(simplex, Init)
}