Skip to content

Commit

Permalink
Merge remote-tracking branch 'upstream/main'
Browse files Browse the repository at this point in the history
  • Loading branch information
Cyunong committed Dec 11, 2024
2 parents 4b9c093 + 2b0561e commit af292a8
Show file tree
Hide file tree
Showing 118 changed files with 2,296 additions and 1,488 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -22,11 +22,11 @@

#include <string>
#include <vector>
#include "api/adapter/data/serializable.h"
#include "nlohmann/json.hpp"

namespace hippy::devtools {

class DomainMetas : public Serializable {
class DomainMetas {
public:
DomainMetas() = default;
explicit DomainMetas(uint32_t node_id) : node_id_(node_id) {}
Expand All @@ -44,7 +44,7 @@ class DomainMetas : public Serializable {
inline void SetNodeValue(std::string node_value) { node_value_ = node_value; }
inline void SetChildrenCount(uint64_t children_count) { children_count_ = children_count; }
inline void AddChild(const DomainMetas& meta) { children_.emplace_back(meta); }
std::string Serialize() const override;
nlohmann::json ToJson() const;

private:
uint32_t node_id_;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ namespace hippy::devtools {
class DevtoolsBackendService : public std::enable_shared_from_this<DevtoolsBackendService> {
public:
DevtoolsBackendService(const DevtoolsConfig& devtools_config,
std::shared_ptr<footstone::WorkerManager> worker_manager);
std::shared_ptr<footstone::WorkerManager> worker_manager, std::function<void()> reconnect_handler);

~DevtoolsBackendService();

Expand Down
3 changes: 2 additions & 1 deletion devtools/devtools-backend/include/tunnel/net_channel.h
Original file line number Diff line number Diff line change
Expand Up @@ -32,14 +32,15 @@ constexpr uint8_t kTaskFlag = 210; // message flag
class NetChannel {
public:
using ReceiveDataHandler = std::function<void(const std::string& msg, uint8_t flag)>;
using ReconnectHandler = std::function<void()>;

virtual ~NetChannel() {}

/**
* @brief connect to frontend
* @param handler to receive msg from frontend
*/
virtual void Connect(ReceiveDataHandler handler) = 0;
virtual void Connect(ReceiveDataHandler handler, ReconnectHandler reconnect_handler) = 0;

/**
* @brief send data to frontend
Expand Down
2 changes: 1 addition & 1 deletion devtools/devtools-backend/include/tunnel/tcp/tcp_channel.h
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@ constexpr int32_t kBufferSize = 32 * 1024;
class TcpChannel : public hippy::devtools::NetChannel, public std::enable_shared_from_this<TcpChannel> {
public:
TcpChannel();
void Connect(ReceiveDataHandler handler) override;
void Connect(ReceiveDataHandler handler, ReconnectHandler reconnect_handler) override;
void Send(const std::string& data) override;
void Close(int32_t code, const std::string& reason) override;

Expand Down
2 changes: 1 addition & 1 deletion devtools/devtools-backend/include/tunnel/tunnel_service.h
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ class TunnelService : public std::enable_shared_from_this<TunnelService> {
/**
* @brief connect to frontend
*/
void Connect();
void Connect(std::function<void()> reconnect_handler);

/**
* @brief send data to frontend
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -49,24 +49,28 @@ namespace hippy::devtools {
class WebSocketChannel : public hippy::devtools::NetChannel, public std::enable_shared_from_this<WebSocketChannel> {
public:
explicit WebSocketChannel(const std::string& ws_uri);
void Connect(ReceiveDataHandler handler) override;
void Connect(ReceiveDataHandler handler, ReconnectHandler reconnect_handler) override;
void Send(const std::string& rsp_data) override;
void Close(int32_t code, const std::string& reason) override;

private:
void StartConnect(const std::string& ws_uri);
void HandleSocketInit(const websocketpp::connection_hdl& handle);
void HandleSocketConnectFail(const websocketpp::connection_hdl& handle);
void HandleSocketConnectOpen(const websocketpp::connection_hdl& handle);
void HandleSocketConnectOpen(const websocketpp::connection_hdl& handle, ReconnectHandler reconnect_handler);
void HandleSocketConnectMessage(const websocketpp::connection_hdl& handle, const WSMessagePtr& message_ptr);
void HandleSocketConnectClose(const websocketpp::connection_hdl& handle);
void AttemptReconnect();

WSClient ws_client_;
websocketpp::connection_hdl connection_hdl_;
std::string ws_uri_;
ReceiveDataHandler data_handler_;
WSThread ws_thread_;
std::vector<std::string> unset_messages_{};
bool ws_should_reconnect;
static const int MAX_RECONNECT_ATTEMPTS = 10;
int ws_reconnect_attempts;
};
} // namespace hippy::devtools

Expand Down
88 changes: 19 additions & 69 deletions devtools/devtools-backend/src/api/adapter/data/domain_metas.cc
Original file line number Diff line number Diff line change
Expand Up @@ -36,78 +36,28 @@ constexpr char kLayoutX[] = "x";
constexpr char kLayoutY[] = "y";
constexpr char kStyle[] = "style";

std::string DomainMetas::Serialize() const {
std::string node_str = "{\"";
node_str += kNodeId;
node_str += "\":";
node_str += std::to_string(node_id_);
node_str += ",\"";
node_str += kParentId;
node_str += "\":";
node_str += std::to_string(parent_id_);
node_str += ",\"";
node_str += kRootId;
node_str += "\":";
node_str += std::to_string(root_id_);
node_str += ",\"";
node_str += kClassName;
node_str += "\":\"";
node_str += class_name_;
node_str += "\",\"";
node_str += kNodeName;
node_str += "\":\"";
node_str += node_name_;
node_str += "\",\"";
node_str += kLocalName;
node_str += "\":\"";
node_str += local_name_;
node_str += "\",\"";
node_str += kNodeValue;
node_str += "\":\"";
node_str += node_value_;
node_str += "\",\"";
node_str += kChildNodeCount;
node_str += "\":";
node_str += std::to_string(children_.size());
node_str += ",\"";
node_str += kStyle;
node_str += "\":";
node_str += style_props_;
node_str += ",\"";
node_str += kAttributes;
node_str += "\":";
node_str += total_props_;
node_str += ",\"";
node_str += kLayoutX;
node_str += "\":";
node_str += std::to_string(static_cast<int>(layout_x_));
node_str += ",\"";
node_str += kLayoutY;
node_str += "\":";
node_str += std::to_string(static_cast<int>(layout_y_));
node_str += ",\"";
node_str += kWidth;
node_str += "\":";
node_str += std::to_string(static_cast<int>(width_));
node_str += ",\"";
node_str += kHeight;
node_str += "\":";
node_str += std::to_string(static_cast<int>(height_));
node_str += ",\"";
node_str += kChildNodeCount;
node_str += "\":";
node_str += std::to_string(static_cast<int>(children_count_));
nlohmann::json DomainMetas::ToJson() const {
nlohmann::json jsonObject;
jsonObject[kNodeId] = node_id_;
jsonObject[kParentId] = parent_id_;
jsonObject[kRootId] = root_id_;
jsonObject[kClassName] = class_name_;
jsonObject[kNodeName] = node_name_;
jsonObject[kLocalName] = local_name_;
jsonObject[kNodeValue] = node_value_;
jsonObject[kChildNodeCount] = children_.size();
jsonObject[kStyle] = nlohmann::json::parse(style_props_, nullptr, false);
jsonObject[kAttributes] = nlohmann::json::parse(total_props_, nullptr, false);
jsonObject[kLayoutX] = static_cast<int>(layout_x_);
jsonObject[kLayoutY] = static_cast<int>(layout_y_);
jsonObject[kWidth] = static_cast<int>(width_);
jsonObject[kHeight] = static_cast<int>(height_);
jsonObject[kChildNodeCount] = static_cast<int>(children_count_);
if (!children_.empty()) {
node_str += ",\"children\": [";
for (auto& child : children_) {
node_str += child.Serialize();
node_str += ",";
jsonObject["children"].push_back(child.ToJson());
}
node_str = node_str.substr(0, node_str.length() - 1); // remove last ","
node_str += "]";
}
node_str += "}";
return node_str;
return jsonObject;
}

} // namespace hippy::devtools
5 changes: 3 additions & 2 deletions devtools/devtools-backend/src/api/devtools_backend_service.cc
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,8 @@

namespace hippy::devtools {
DevtoolsBackendService::DevtoolsBackendService(const DevtoolsConfig& devtools_config,
std::shared_ptr<footstone::WorkerManager> worker_manager) {
std::shared_ptr<footstone::WorkerManager> worker_manager,
std::function<void()> reconnect_handler) {
FOOTSTONE_DLOG(INFO) << kDevToolsTag << "DevtoolsBackendService create framework:" << devtools_config.framework
<< ",tunnel:" << devtools_config.tunnel;
auto data_provider = std::make_shared<DataProvider>();
Expand All @@ -38,7 +39,7 @@ DevtoolsBackendService::DevtoolsBackendService(const DevtoolsConfig& devtools_co
domain_dispatch_ = std::make_shared<DomainDispatch>(data_channel_, worker_manager);
domain_dispatch_->RegisterDefaultDomainListener();
tunnel_service_ = std::make_shared<TunnelService>(domain_dispatch_, devtools_config);
tunnel_service_->Connect();
tunnel_service_->Connect(reconnect_handler);
notification_center->runtime_notification = std::make_shared<DefaultRuntimeNotification>(tunnel_service_);

if (devtools_config.framework == Framework::kHippy) {
Expand Down
2 changes: 1 addition & 1 deletion devtools/devtools-backend/src/module/domain/css_domain.cc
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@ void CssDomain::RegisterCallback() {
return;
}
auto response_callback = [callback, provider = self->GetDataProvider()](const DomainMetas& data) {
auto model = CssModel::CreateModel(nlohmann::json::parse(data.Serialize(), nullptr, false));
auto model = CssModel::CreateModel(data.ToJson());
model.SetDataProvider(provider);
if (callback) {
callback(model);
Expand Down
6 changes: 3 additions & 3 deletions devtools/devtools-backend/src/module/domain/dom_domain.cc
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,7 @@ void DomDomain::RegisterCallback() {
auto dom_tree_adapter = self->GetDataProvider()->dom_tree_adapter;
if (dom_tree_adapter) {
auto response_callback = [callback, provider = self->GetDataProvider()](const DomainMetas& data) {
auto model = DomModel::CreateModel(nlohmann::json::parse(data.Serialize(), nullptr, false));
auto model = DomModel::CreateModel(data.ToJson());
model.SetDataProvider(provider);
if (callback) {
callback(model);
Expand Down Expand Up @@ -172,8 +172,8 @@ void DomDomain::GetBoxModel(const DomNodeDataRequest& request) {
}

void DomDomain::GetNodeForLocation(const DomNodeForLocationRequest& request) {
if (!dom_data_call_back_) {
ResponseErrorToFrontend(request.GetId(), kErrorFailCode, "GetNodeForLocation, dom_data_callback is null");
if (!location_for_node_call_back_) {
ResponseErrorToFrontend(request.GetId(), kErrorFailCode, "GetNodeForLocation, location_for_node_call_back is null");
return;
}
if (!request.HasSetXY()) {
Expand Down
3 changes: 2 additions & 1 deletion devtools/devtools-backend/src/tunnel/tcp/tcp_channel.cc
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,8 @@ TcpChannel::TcpChannel() {
frame_codec_ = FrameCodec();
}

void TcpChannel::Connect(ReceiveDataHandler handler) {
void TcpChannel::Connect(ReceiveDataHandler handler, ReconnectHandler reconnect_handler) {
// TODO: reconnect in TCP if needed
frame_codec_.SetEncodeCallback([WEAK_THIS](void *data, int32_t len) {
DEFINE_AND_CHECK_SELF(TcpChannel)
if (self->client_fd_ < 0) {
Expand Down
4 changes: 2 additions & 2 deletions devtools/devtools-backend/src/tunnel/tunnel_service.cc
Original file line number Diff line number Diff line change
Expand Up @@ -38,14 +38,14 @@ TunnelService::TunnelService(std::shared_ptr<DomainDispatch> dispatch, const Dev
channel_ = NetChannel::CreateChannel(devtools_config);
}

void TunnelService::Connect() {
void TunnelService::Connect(std::function<void()> reconnect_handler) {
FOOTSTONE_DLOG(INFO) << kDevToolsTag << "TunnelService, start connect.";
channel_->Connect([WEAK_THIS](const std::string& msg, int flag) {
if (flag == kTaskFlag) {
DEFINE_AND_CHECK_SELF(TunnelService)
self->HandleReceiveData(msg);
}
});
}, reconnect_handler);
dispatch_->SetResponseHandler([WEAK_THIS](const std::string &rsp_data) {
DEFINE_AND_CHECK_SELF(TunnelService)
self->channel_->Send(rsp_data);
Expand Down
40 changes: 33 additions & 7 deletions devtools/devtools-backend/src/tunnel/ws/web_socket_channel.cc
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,8 @@ typedef WSClient::connection_ptr WSConnectionPtr;
namespace hippy::devtools {

WebSocketChannel::WebSocketChannel(const std::string& ws_uri) {
ws_reconnect_attempts = 0;
ws_should_reconnect = true;
ws_uri_ = ws_uri;
ws_client_.clear_access_channels(websocketpp::log::alevel::all);
ws_client_.set_access_channels(websocketpp::log::alevel::fail);
Expand All @@ -43,7 +45,7 @@ WebSocketChannel::WebSocketChannel(const std::string& ws_uri) {
ws_client_.start_perpetual();
}

void WebSocketChannel::Connect(ReceiveDataHandler handler) {
void WebSocketChannel::Connect(ReceiveDataHandler handler, ReconnectHandler reconnect_handler) {
if (ws_uri_.empty()) {
FOOTSTONE_DLOG(ERROR) << kDevToolsTag << "websocket uri is empty, connect error";
return;
Expand All @@ -55,9 +57,9 @@ void WebSocketChannel::Connect(ReceiveDataHandler handler) {
DEFINE_AND_CHECK_SELF(WebSocketChannel)
self->HandleSocketInit(handle);
});
ws_client_.set_open_handler([WEAK_THIS](const websocketpp::connection_hdl& handle) {
ws_client_.set_open_handler([WEAK_THIS, reconnect_handler](const websocketpp::connection_hdl& handle) {
DEFINE_AND_CHECK_SELF(WebSocketChannel)
self->HandleSocketConnectOpen(handle);
self->HandleSocketConnectOpen(handle, reconnect_handler);
});
ws_client_.set_close_handler([WEAK_THIS](const websocketpp::connection_hdl& handle) {
DEFINE_AND_CHECK_SELF(WebSocketChannel)
Expand Down Expand Up @@ -87,6 +89,7 @@ void WebSocketChannel::Send(const std::string& rsp_data) {
}

void WebSocketChannel::Close(int32_t code, const std::string& reason) {
ws_should_reconnect = false;
if (!connection_hdl_.lock()) {
FOOTSTONE_DLOG(ERROR) << kDevToolsTag << "send message error, handler is null";
return;
Expand Down Expand Up @@ -128,7 +131,8 @@ void WebSocketChannel::HandleSocketConnectFail(const websocketpp::connection_hdl
<< ", remote close reason:" << con->get_remote_close_reason().c_str();
}

void WebSocketChannel::HandleSocketConnectOpen(const websocketpp::connection_hdl& handle) {
void WebSocketChannel::HandleSocketConnectOpen(const websocketpp::connection_hdl& handle,
ReconnectHandler reconnect_handler) {
connection_hdl_ = handle.lock();
FOOTSTONE_DLOG(INFO) << kDevToolsTag << "websocket connect open";
if (!connection_hdl_.lock() || unset_messages_.empty()) {
Expand All @@ -139,6 +143,10 @@ void WebSocketChannel::HandleSocketConnectOpen(const websocketpp::connection_hdl
ws_client_.send(connection_hdl_, message, websocketpp::frame::opcode::text, error_code);
}
unset_messages_.clear();
if (0 < ws_reconnect_attempts && ws_reconnect_attempts < MAX_RECONNECT_ATTEMPTS) {
reconnect_handler();
}
ws_reconnect_attempts = 0;
}

void WebSocketChannel::HandleSocketConnectMessage(const websocketpp::connection_hdl& handle,
Expand All @@ -153,14 +161,32 @@ void WebSocketChannel::HandleSocketConnectMessage(const websocketpp::connection_
void WebSocketChannel::HandleSocketConnectClose(const websocketpp::connection_hdl& handle) {
websocketpp::lib::error_code error_code;
auto con = ws_client_.get_con_from_hdl(handle, error_code);
// set handle nullptr when connect fail
data_handler_ = nullptr;
unset_messages_.clear();
FOOTSTONE_DLOG(INFO) << kDevToolsTag << "websocket connect close, state: " << con->get_state()
<< ", error message:" << con->get_ec().message().c_str()
<< ", local close code:" << con->get_local_close_code()
<< ", local close reason: " << con->get_local_close_reason().c_str()
<< ", remote close code:" << con->get_remote_close_code()
<< ", remote close reason:" << con->get_remote_close_reason().c_str();
if (ws_should_reconnect) {
AttemptReconnect();
}
else {
// set handle nullptr when connect fail
data_handler_ = nullptr;
unset_messages_.clear();
}
}

void WebSocketChannel::AttemptReconnect() {
ws_reconnect_attempts++;
if (ws_reconnect_attempts < MAX_RECONNECT_ATTEMPTS) {
FOOTSTONE_DLOG(INFO) << "Attempting to reconnect (" << ws_reconnect_attempts << "/"
<< MAX_RECONNECT_ATTEMPTS << ")...";
StartConnect(ws_uri_);
} else {
ws_should_reconnect = false;
FOOTSTONE_DLOG(INFO) << "Max reconnect attempts reached.";
}
}

} // namespace hippy::devtools
Original file line number Diff line number Diff line change
Expand Up @@ -87,7 +87,8 @@ jint OnCreateDevtools(JNIEnv* j_env,
const string_view ws_url = JniUtils::ToStrView(j_env, j_ws_url);
DevtoolsDataSource::SetFileCacheDir(StringViewUtils::ToStdString(
StringViewUtils::ConvertEncoding(data_dir, string_view::Encoding::Utf8).utf8_value()));
auto devtools_data_source = std::make_shared<hippy::devtools::DevtoolsDataSource>(
auto devtools_data_source = std::make_shared<hippy::devtools::DevtoolsDataSource>();
devtools_data_source->CreateDevtoolsService(
StringViewUtils::ToStdString(StringViewUtils::ConvertEncoding(ws_url, string_view::Encoding::Utf8).utf8_value()),
worker_manager);
uint32_t id = devtools::DevtoolsDataSource::Insert(devtools_data_source);
Expand Down
Loading

0 comments on commit af292a8

Please sign in to comment.