Skip to content

Commit

Permalink
src,agents: add support for source code collection
Browse files Browse the repository at this point in the history
The runtime now stores the `scriptId` and `url` or `path` of all the
loaded cjs and esm modules.
A new grpc command has been added to retrieve the source code for any
`scriptId`-`url|path` pair by reading the source code from file.
In case the specific `scriptId`-`url|path` pair wasn't stored, it
returns an error.
In the rare case where a ESM is not loaded from a file URL, the code
would be stored on loading time.

PR-URL: #240
Reviewed-By: Rafael Gonzaga <rafael.nunu@hotmail.com>
  • Loading branch information
santigimeno committed Jan 3, 2025
1 parent ad3d41d commit a02d20b
Show file tree
Hide file tree
Showing 31 changed files with 2,678 additions and 53 deletions.
2 changes: 2 additions & 0 deletions agents/grpc/proto/command.proto
Original file line number Diff line number Diff line change
Expand Up @@ -7,13 +7,15 @@ syntax = "proto3";

import "profile.proto";
import "reconfigure.proto";
import "source_code.proto";

package grpcagent;

message CommandArgs {
oneof args {
ReconfigureBody reconfigure = 1;
ProfileArgs profile = 2;
SourceCodeArgs source_code = 3;
}
}

Expand Down
2 changes: 2 additions & 0 deletions agents/grpc/proto/nsolid_service.proto
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import "info.proto";
import "metrics.proto";
import "packages.proto";
import "reconfigure.proto";
import "source_code.proto";
import "startup_times.proto";

package grpcagent;
Expand All @@ -22,6 +23,7 @@ service NSolidService {
rpc ExportBlockedLoop (BlockedLoopEvent) returns (EventResponse) {}
rpc ExportUnblockedLoop (UnblockedLoopEvent) returns (EventResponse) {}
rpc ExportReconfigure (ReconfigureEvent) returns (EventResponse) {}
rpc ExportSourceCode (SourceCodeEvent) returns (EventResponse) {}
rpc ExportStartupTimes (StartupTimesEvent) returns (EventResponse) {}
}

Expand Down
19 changes: 19 additions & 0 deletions agents/grpc/proto/source_code.proto
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
syntax = "proto3";

import "common.proto";

package grpcagent;

message SourceCodeArgs {
int64 thread_id = 1;
int32 script_id = 2;
string path = 3;
}

message SourceCodeEvent {
CommonResponse common = 1;
int64 thread_id = 2;
string path = 3;
string code = 4;
bool is_esm = 5;
}
52 changes: 52 additions & 0 deletions agents/grpc/src/grpc_agent.cc
Original file line number Diff line number Diff line change
Expand Up @@ -1439,6 +1439,8 @@ void GrpcAgent::handle_command_request(CommandRequestStor&& req) {
start_heap_sampling(request);
} else if (cmd == "snapshot") {
start_heap_snapshot(request);
} else if (cmd == "source_code") {
send_source_code_event(request);
} else if (cmd == "startup_times") {
send_startup_times_event(request.requestid().c_str());
} else {
Expand Down Expand Up @@ -1724,6 +1726,56 @@ void GrpcAgent::send_reconfigure_event(const char* req_id) {
});
}

void GrpcAgent::send_source_code_event(const grpcagent::CommandRequest& req) {
const grpcagent::SourceCodeArgs& args = req.args().source_code();
uint64_t thread_id = args.thread_id();
int script_id = args.script_id();
const std::string path = args.path();

ArenaOptions arena_options;
arena_options.initial_block_size = 1024;
arena_options.max_block_size = 65536;
std::unique_ptr<Arena> arena{new Arena{arena_options}};

auto source_code_event =
Arena::Create<grpcagent::SourceCodeEvent>(arena.get());
PopulateCommon(source_code_event->mutable_common(),
"source_code",
req.requestid().c_str());

source_code_event->set_thread_id(thread_id);
source_code_event->set_path(path);


auto envinst_sp = EnvInst::GetInst(thread_id);
if (envinst_sp == nullptr) {
Debug("Error getting EnvInst for thread: %ld\n", thread_id);
PopulateError(source_code_event->mutable_common(),
ErrorType::EThreadGoneError);
} else {
int r = envinst_sp->GetSourceCode(script_id,
path,
source_code_event->mutable_code());
if (r != 0) {
Debug("Error reading file from: %s. Error: %d.\n", path.c_str(), r);
PopulateError(source_code_event->mutable_common(),
ErrorType::ESourceCodeFileError);
}
}

auto context = GrpcClient::MakeClientContext(agent_id_, saas_);

GrpcClient::DelegateAsyncExport(
nsolid_service_stub_.get(), std::move(context), std::move(arena),
std::move(*source_code_event),
[](::grpc::Status,
std::unique_ptr<Arena>&&,
const grpcagent::SourceCodeEvent& info_event,
grpcagent::EventResponse*) {
return true;
});
}

void GrpcAgent::send_startup_times_event(const char* req_id) {
ArenaOptions arena_options;
arena_options.initial_block_size = 1024;
Expand Down
2 changes: 2 additions & 0 deletions agents/grpc/src/grpc_agent.h
Original file line number Diff line number Diff line change
Expand Up @@ -267,6 +267,8 @@ class GrpcAgent: public std::enable_shared_from_this<GrpcAgent>,

void send_reconfigure_event(const char* req_id);

void send_source_code_event(const grpcagent::CommandRequest& req);

void send_startup_times_event(const char* req_id);

void send_unblocked_loop_event(BlockedLoopStor&& stor);
Expand Down
19 changes: 19 additions & 0 deletions agents/grpc/src/grpc_client.cc
Original file line number Diff line number Diff line change
Expand Up @@ -273,6 +273,25 @@ int GrpcClient::DelegateAsyncExport(
}


int GrpcClient::DelegateAsyncExport(
NSolidService::StubInterface* stub,
std::unique_ptr<ClientContext>&& context,
std::unique_ptr<Arena>&& arena,
grpcagent::SourceCodeEvent&& event,
std::function<bool(Status,
std::unique_ptr<Arena> &&,
const grpcagent::SourceCodeEvent&,
grpcagent::EventResponse*)>&& result_callback) noexcept {
return InternalDelegateAsyncExport<grpcagent::SourceCodeEvent>(
stub,
&NSolidService::StubInterface::async_interface::ExportSourceCode,
std::move(context),
std::move(arena),
std::move(event),
std::move(result_callback));
}


int GrpcClient::DelegateAsyncExport(
NSolidService::StubInterface* stub,
std::unique_ptr<ClientContext>&& context,
Expand Down
10 changes: 10 additions & 0 deletions agents/grpc/src/grpc_client.h
Original file line number Diff line number Diff line change
Expand Up @@ -113,6 +113,16 @@ class GrpcClient {
const grpcagent::ReconfigureEvent&,
grpcagent::EventResponse*)>&& result_callback) noexcept;

static int DelegateAsyncExport(
grpcagent::NSolidService::StubInterface* stub,
std::unique_ptr<::grpc::ClientContext>&& context,
std::unique_ptr<google::protobuf::Arena>&& arena,
grpcagent::SourceCodeEvent&& event,
std::function<bool(::grpc::Status,
std::unique_ptr<google::protobuf::Arena> &&,
const grpcagent::SourceCodeEvent&,
grpcagent::EventResponse*)>&& result_callback) noexcept;

static int DelegateAsyncExport(
grpcagent::NSolidService::StubInterface* stub,
std::unique_ptr<::grpc::ClientContext>&& context,
Expand Down
3 changes: 2 additions & 1 deletion agents/grpc/src/grpc_errors.h
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,8 @@
X(EProfSnapshotError, 500, "Profile/Snapshot creation failure", 1003) \
X(ESnapshotDisabled, 500, "Heap Snapshots disabled", 1004) \
X(ENoMemory, 500, "Internal Runtime Error", 1005) \
X(ENotAvailable, 404, "Resource not available", 1006)
X(ENotAvailable, 404, "Resource not available", 1006) \
X(ESourceCodeFileError, 500, "Internal Runtime Error", 1007)

namespace node {
namespace nsolid {
Expand Down
97 changes: 84 additions & 13 deletions agents/grpc/src/proto/command.pb.cc

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading

0 comments on commit a02d20b

Please sign in to comment.