//////////////////////////////////////////////////////////////////////////////// /// DISCLAIMER /// /// Copyright 2014-2016 ArangoDB GmbH, Cologne, Germany /// Copyright 2004-2014 triAGENS GmbH, Cologne, Germany /// /// Licensed under the Apache License, Version 2.0 (the "License"); /// you may not use this file except in compliance with the License. /// You may obtain a copy of the License at /// /// http://www.apache.org/licenses/LICENSE-2.0 /// /// Unless required by applicable law or agreed to in writing, software /// distributed under the License is distributed on an "AS IS" BASIS, /// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. /// See the License for the specific language governing permissions and /// limitations under the License. /// /// Copyright holder is ArangoDB GmbH, Cologne, Germany /// /// @author Jan Steemann //////////////////////////////////////////////////////////////////////////////// #include "RocksDBV8Functions.h" #include "Aql/Functions.h" #include "Basics/Exceptions.h" #include "Basics/Result.h" #include "Cluster/ServerState.h" #include "RocksDBEngine/RocksDBCollection.h" #include "RocksDBEngine/RocksDBCommon.h" #include "RocksDBEngine/RocksDBEngine.h" #include "StorageEngine/EngineSelectorFeature.h" #include "Utils/ExecContext.h" #include "V8/v8-conv.h" #include "V8/v8-globals.h" #include "V8/v8-utils.h" #include "V8/v8-vpack.h" #include "V8Server/v8-collection.h" #include "V8Server/v8-externals.h" #include "VocBase/LogicalCollection.h" #include using namespace arangodb; /// flush the WAL static void JS_FlushWal(v8::FunctionCallbackInfo const& args) { TRI_V8_TRY_CATCH_BEGIN(isolate); v8::Local context = isolate->GetCurrentContext(); v8::HandleScope scope(isolate); bool waitForSync = false; bool waitForCollector = false; bool writeShutdownFile = false; if (args.Length() > 0) { if (args[0]->IsObject()) { v8::Handle obj = args[0]->ToObject(TRI_IGETC).FromMaybe(v8::Local()); if (TRI_HasProperty(context, isolate, obj, "waitForSync")) { waitForSync = TRI_ObjectToBoolean( isolate, obj->Get(TRI_V8_ASCII_STRING(isolate, "waitForSync"))); } if (TRI_HasProperty(context, isolate, obj, "waitForCollector")) { waitForCollector = TRI_ObjectToBoolean( isolate, obj->Get(TRI_V8_ASCII_STRING(isolate, "waitForCollector"))); } if (TRI_HasProperty(context, isolate, obj, "writeShutdownFile")) { writeShutdownFile = TRI_ObjectToBoolean( isolate, obj->Get(TRI_V8_ASCII_STRING(isolate, "writeShutdownFile"))); } } else { waitForSync = TRI_ObjectToBoolean(isolate, args[0]); if (args.Length() > 1) { waitForCollector = TRI_ObjectToBoolean(isolate, args[1]); if (args.Length() > 2) { writeShutdownFile = TRI_ObjectToBoolean(isolate, args[2]); } } } } EngineSelectorFeature::ENGINE->flushWal(waitForSync, waitForCollector, writeShutdownFile); TRI_V8_RETURN_TRUE(); TRI_V8_TRY_CATCH_END } /// this is just a stub static void JS_WaitCollectorWal(v8::FunctionCallbackInfo const& args) { TRI_V8_TRY_CATCH_BEGIN(isolate); v8::HandleScope scope(isolate); if (ServerState::instance()->isCoordinator()) { TRI_V8_THROW_EXCEPTION(TRI_ERROR_NOT_IMPLEMENTED); } // this is just a stub TRI_V8_RETURN_TRUE(); TRI_V8_TRY_CATCH_END } /// this is just a stub static void JS_TransactionsWal(v8::FunctionCallbackInfo const& args) { TRI_V8_TRY_CATCH_BEGIN(isolate); v8::HandleScope scope(isolate); if (ServerState::instance()->isCoordinator()) { TRI_V8_THROW_EXCEPTION(TRI_ERROR_NOT_IMPLEMENTED); } // this is just a stub TRI_V8_RETURN_TRUE(); TRI_V8_TRY_CATCH_END } /// this is just a stub static void JS_PropertiesWal(v8::FunctionCallbackInfo const& args) { TRI_V8_TRY_CATCH_BEGIN(isolate); v8::HandleScope scope(isolate); if (ServerState::instance()->isCoordinator()) { TRI_V8_THROW_EXCEPTION(TRI_ERROR_NOT_IMPLEMENTED); } // this is just a stub TRI_V8_RETURN_TRUE(); TRI_V8_TRY_CATCH_END } static void JS_RecalculateCounts(v8::FunctionCallbackInfo const& args) { TRI_V8_TRY_CATCH_BEGIN(isolate); v8::HandleScope scope(isolate); auto* collection = UnwrapCollection(isolate, args.Holder()); if (!collection) { TRI_V8_THROW_EXCEPTION_INTERNAL("cannot extract collection"); } if (ExecContext::CURRENT != nullptr) { if (!ExecContext::CURRENT->canUseCollection(collection->name(), auth::Level::RW)) { TRI_V8_THROW_EXCEPTION(TRI_ERROR_FORBIDDEN); } } auto* physical = toRocksDBCollection(*collection); v8::Handle result = v8::Number::New(isolate, static_cast(physical->recalculateCounts())); TRI_V8_RETURN(result); TRI_V8_TRY_CATCH_END } static void JS_CompactCollection(v8::FunctionCallbackInfo const& args) { TRI_V8_TRY_CATCH_BEGIN(isolate); v8::HandleScope scope(isolate); auto* collection = UnwrapCollection(isolate, args.Holder()); if (!collection) { TRI_V8_THROW_EXCEPTION_INTERNAL("cannot extract collection"); } auto* physical = toRocksDBCollection(*collection); physical->compact(); TRI_V8_RETURN_UNDEFINED(); TRI_V8_TRY_CATCH_END } static void JS_EstimateCollectionSize(v8::FunctionCallbackInfo const& args) { TRI_V8_TRY_CATCH_BEGIN(isolate); v8::HandleScope scope(isolate); auto* collection = UnwrapCollection(isolate, args.Holder()); if (!collection) { TRI_V8_THROW_EXCEPTION_INTERNAL("cannot extract collection"); } auto* physical = toRocksDBCollection(*collection); VPackBuilder builder; physical->estimateSize(builder); v8::Handle result = TRI_VPackToV8(isolate, builder.slice()); TRI_V8_RETURN(result); TRI_V8_TRY_CATCH_END } static void JS_WaitForEstimatorSync(v8::FunctionCallbackInfo const& args) { TRI_V8_TRY_CATCH_BEGIN(isolate); v8::HandleScope scope(isolate); EngineSelectorFeature::ENGINE->waitForEstimatorSync(std::chrono::seconds(10)); TRI_V8_RETURN_TRUE(); TRI_V8_TRY_CATCH_END } void RocksDBV8Functions::registerResources() { ISOLATE; v8::HandleScope scope(isolate); TRI_GET_GLOBALS(); // patch ArangoCollection object v8::Handle rt = v8::Handle::New(isolate, v8g->VocbaseColTempl); TRI_ASSERT(!rt.IsEmpty()); TRI_AddMethodVocbase(isolate, rt, TRI_V8_ASCII_STRING(isolate, "recalculateCount"), JS_RecalculateCounts, true); TRI_AddMethodVocbase(isolate, rt, TRI_V8_ASCII_STRING(isolate, "compact"), JS_CompactCollection); TRI_AddMethodVocbase(isolate, rt, TRI_V8_ASCII_STRING(isolate, "estimatedSize"), JS_EstimateCollectionSize); // add global WAL handling functions TRI_AddGlobalFunctionVocbase(isolate, TRI_V8_ASCII_STRING(isolate, "WAL_FLUSH"), JS_FlushWal, true); TRI_AddGlobalFunctionVocbase(isolate, TRI_V8_ASCII_STRING(isolate, "WAL_WAITCOLLECTOR"), JS_WaitCollectorWal, true); TRI_AddGlobalFunctionVocbase(isolate, TRI_V8_ASCII_STRING(isolate, "WAL_PROPERTIES"), JS_PropertiesWal, true); TRI_AddGlobalFunctionVocbase(isolate, TRI_V8_ASCII_STRING(isolate, "WAL_TRANSACTIONS"), JS_TransactionsWal, true); TRI_AddGlobalFunctionVocbase(isolate, TRI_V8_ASCII_STRING(isolate, "WAIT_FOR_ESTIMATOR_SYNC"), JS_WaitForEstimatorSync, true); }