#include "base_object-inl.h" #include "env-inl.h" #include "memory_tracker-inl.h" #include "node.h" #include "node_external_reference.h" #include "node_internals.h" #include "node_v8_platform-inl.h" #include "tracing/agent.h" #include "util-inl.h" #include #include namespace node { class ExternalReferenceRegistry; using v8::Array; using v8::Context; using v8::Function; using v8::FunctionCallbackInfo; using v8::FunctionTemplate; using v8::Isolate; using v8::Local; using v8::NewStringType; using v8::Object; using v8::String; using v8::Value; class NodeCategorySet : public BaseObject { public: static void Initialize(Local target, Local unused, Local context, void* priv); static void RegisterExternalReferences(ExternalReferenceRegistry* registry); static void New(const FunctionCallbackInfo& args); static void Enable(const FunctionCallbackInfo& args); static void Disable(const FunctionCallbackInfo& args); const std::set& GetCategories() const { return categories_; } void MemoryInfo(MemoryTracker* tracker) const override { tracker->TrackField("categories", categories_); } SET_MEMORY_INFO_NAME(NodeCategorySet) SET_SELF_SIZE(NodeCategorySet) private: NodeCategorySet(Environment* env, Local wrap, std::set&& categories) : BaseObject(env, wrap), categories_(std::move(categories)) { MakeWeak(); } bool enabled_ = false; const std::set categories_; }; void NodeCategorySet::New(const FunctionCallbackInfo& args) { Environment* env = Environment::GetCurrent(args); std::set categories; CHECK(args[0]->IsArray()); Local cats = args[0].As(); for (size_t n = 0; n < cats->Length(); n++) { Local category; if (!cats->Get(env->context(), n).ToLocal(&category)) return; Utf8Value val(env->isolate(), category); if (!*val) return; categories.emplace(*val); } CHECK_NOT_NULL(GetTracingAgentWriter()); new NodeCategorySet(env, args.This(), std::move(categories)); } void NodeCategorySet::Enable(const FunctionCallbackInfo& args) { NodeCategorySet* category_set; ASSIGN_OR_RETURN_UNWRAP(&category_set, args.Holder()); CHECK_NOT_NULL(category_set); const auto& categories = category_set->GetCategories(); if (!category_set->enabled_ && !categories.empty()) { // Starts the Tracing Agent if it wasn't started already (e.g. through // a command line flag.) StartTracingAgent(); GetTracingAgentWriter()->Enable(categories); category_set->enabled_ = true; } } void NodeCategorySet::Disable(const FunctionCallbackInfo& args) { NodeCategorySet* category_set; ASSIGN_OR_RETURN_UNWRAP(&category_set, args.Holder()); CHECK_NOT_NULL(category_set); const auto& categories = category_set->GetCategories(); if (category_set->enabled_ && !categories.empty()) { GetTracingAgentWriter()->Disable(categories); category_set->enabled_ = false; } } void GetEnabledCategories(const FunctionCallbackInfo& args) { Environment* env = Environment::GetCurrent(args); std::string categories = GetTracingAgentWriter()->agent()->GetEnabledCategories(); if (!categories.empty()) { args.GetReturnValue().Set( String::NewFromUtf8(env->isolate(), categories.c_str(), NewStringType::kNormal, categories.size()).ToLocalChecked()); } } static void SetTraceCategoryStateUpdateHandler( const FunctionCallbackInfo& args) { Environment* env = Environment::GetCurrent(args); CHECK(args[0]->IsFunction()); env->set_trace_category_state_function(args[0].As()); } void NodeCategorySet::Initialize(Local target, Local unused, Local context, void* priv) { Environment* env = Environment::GetCurrent(context); Isolate* isolate = env->isolate(); SetMethod(context, target, "getEnabledCategories", GetEnabledCategories); SetMethod(context, target, "setTraceCategoryStateUpdateHandler", SetTraceCategoryStateUpdateHandler); Local category_set = NewFunctionTemplate(isolate, NodeCategorySet::New); category_set->InstanceTemplate()->SetInternalFieldCount( NodeCategorySet::kInternalFieldCount); category_set->Inherit(BaseObject::GetConstructorTemplate(env)); SetProtoMethod(isolate, category_set, "enable", NodeCategorySet::Enable); SetProtoMethod(isolate, category_set, "disable", NodeCategorySet::Disable); SetConstructorFunction(context, target, "CategorySet", category_set); Local isTraceCategoryEnabled = FIXED_ONE_BYTE_STRING(env->isolate(), "isTraceCategoryEnabled"); Local trace = FIXED_ONE_BYTE_STRING(env->isolate(), "trace"); // Grab the trace and isTraceCategoryEnabled intrinsics from the binding // object and expose those to our binding layer. Local binding = context->GetExtrasBindingObject(); target->Set(context, isTraceCategoryEnabled, binding->Get(context, isTraceCategoryEnabled).ToLocalChecked()) .Check(); target->Set(context, trace, binding->Get(context, trace).ToLocalChecked()).Check(); } void NodeCategorySet::RegisterExternalReferences( ExternalReferenceRegistry* registry) { registry->Register(GetEnabledCategories); registry->Register(SetTraceCategoryStateUpdateHandler); registry->Register(NodeCategorySet::New); registry->Register(NodeCategorySet::Enable); registry->Register(NodeCategorySet::Disable); } } // namespace node NODE_BINDING_CONTEXT_AWARE_INTERNAL(trace_events, node::NodeCategorySet::Initialize) NODE_BINDING_EXTERNAL_REFERENCE( trace_events, node::NodeCategorySet::RegisterExternalReferences)