// Aseprite // Copyright (C) 2018-2020 Igara Studio S.A. // Copyright (C) 2018 David Capello // // This program is distributed under the terms of // the End-User License Agreement for Aseprite. #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "app/app.h" #include "app/console.h" #include "app/script/engine.h" #include "app/script/luacpp.h" #include "ui/system.h" #include #include #include namespace app { namespace script { namespace { // additional "enum" value to make message callback simpler #define MESSAGE_TYPE_BINARY (int)ix::WebSocketMessageType::Fragment + 10 int WebSocket_new(lua_State* L) { static std::once_flag f; std::call_once(f, &ix::initNetSystem); auto ws = new ix::WebSocket(); push_ptr(L, ws); if (lua_istable(L, 1)) { lua_getfield(L, 1, "url"); if (const char* s = lua_tostring(L, -1)) { ws->setUrl(s); } lua_pop(L, 1); int type = lua_getfield(L, 1, "onreceive"); if (type == LUA_TFUNCTION) { int onreceiveRef = luaL_ref(L, LUA_REGISTRYINDEX); ws->setOnMessageCallback( [L, ws, onreceiveRef](const ix::WebSocketMessagePtr& msg) { int msgType = (msg->binary ? MESSAGE_TYPE_BINARY : static_cast(msg->type)); std::string msgData = msg->str; ui::execute_from_ui_thread([=]() { lua_rawgeti(L, LUA_REGISTRYINDEX, onreceiveRef); lua_pushinteger(L, msgType); lua_pushlstring(L, msgData.c_str(), msgData.length()); if (lua_pcall(L, 2, 0, 0)) { if (const char* s = lua_tostring(L, -1)) { App::instance()->scriptEngine()->consolePrint(s); ws->stop(); } } }); }); } else { lua_pop(L, 1); } } return 1; } int WebSocket_gc(lua_State* L) { auto ws = get_ptr(L, 1); ws->stop(); delete ws; return 0; } int WebSocket_sendText(lua_State* L) { auto ws = get_ptr(L, 1); std::stringstream data; int argc = lua_gettop(L); for (int i = 2; i <= argc; i++) { size_t bufLen; const char* buf = lua_tolstring(L, i, &bufLen); data.write(buf, bufLen); } lua_pop(L, argc); ws->sendText(data.str()); return 0; } int WebSocket_sendBinary(lua_State* L) { auto ws = get_ptr(L, 1); std::stringstream data; int argc = lua_gettop(L); for (int i = 2; i <= argc; i++) { size_t bufLen; const char* buf = lua_tolstring(L, i, &bufLen); data.write(buf, bufLen); } lua_pop(L, argc); ws->sendBinary(data.str()); return 0; } int WebSocket_connect(lua_State* L) { auto ws = get_ptr(L, 1); lua_pop(L, 1); ws->start(); return 0; } int WebSocket_close(lua_State* L) { auto ws = get_ptr(L, 1); lua_pop(L, 1); ws->stop(); return 0; } int WebSocket_get_url(lua_State* L) { auto ws = get_ptr(L, 1); lua_pop(L, 1); lua_pushstring(L, ws->getUrl().c_str()); return 1; } const luaL_Reg WebSocket_methods[] = { { "__gc", WebSocket_gc }, { "close", WebSocket_close }, { "connect", WebSocket_connect }, { "sendText", WebSocket_sendText }, { "sendBinary", WebSocket_sendBinary }, { nullptr, nullptr } }; const Property WebSocket_properties[] = { { "url", WebSocket_get_url, nullptr }, { nullptr, nullptr, nullptr } }; } // namespace { } using WebSocket = ix::WebSocket; DEF_MTNAME(WebSocket); void register_websocket_class(lua_State* L) { REG_CLASS(L, WebSocket); REG_CLASS_NEW(L, WebSocket); REG_CLASS_PROPERTIES(L, WebSocket); // message type enum lua_newtable(L); lua_pushvalue(L, -1); lua_setglobal(L, "MessageType"); setfield_integer(L, "Text", (int)ix::WebSocketMessageType::Message); setfield_integer(L, "Binary", MESSAGE_TYPE_BINARY); setfield_integer(L, "Open", (int)ix::WebSocketMessageType::Open); setfield_integer(L, "Close", (int)ix::WebSocketMessageType::Close); setfield_integer(L, "Error", (int)ix::WebSocketMessageType::Error); setfield_integer(L, "Ping", (int)ix::WebSocketMessageType::Ping); setfield_integer(L, "Pong", (int)ix::WebSocketMessageType::Pong); setfield_integer(L, "Fragment", (int)ix::WebSocketMessageType::Fragment); lua_pop(L, 1); } } // namespace script } // namespace app