mirror of https://gitee.com/bigwinds/arangodb
231 lines
7.3 KiB
C++
231 lines
7.3 KiB
C++
////////////////////////////////////////////////////////////////////////////////
|
|
/// @brief source code loader
|
|
///
|
|
/// @file
|
|
///
|
|
/// DISCLAIMER
|
|
///
|
|
/// Copyright 2014 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 Dr. Frank Celler
|
|
/// @author Copyright 2014, ArangoDB GmbH, Cologne, Germany
|
|
/// @author Copyright 2011-2013, triAGENS GmbH, Cologne, Germany
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include "JSLoader.h"
|
|
|
|
#include "Basics/MutexLocker.h"
|
|
#include "Basics/files.h"
|
|
#include "Basics/logging.h"
|
|
#include "Basics/tri-strings.h"
|
|
#include "Basics/StringUtils.h"
|
|
#include "V8/v8-utils.h"
|
|
|
|
using namespace std;
|
|
using namespace triagens::basics;
|
|
using namespace triagens::arango;
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// --SECTION-- constructors and destructors
|
|
// -----------------------------------------------------------------------------
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
/// @brief constructs a loader
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
JSLoader::JSLoader () {
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// --SECTION-- public methods
|
|
// -----------------------------------------------------------------------------
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
/// @brief executes a named script in the global context
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
v8::Handle<v8::Value> JSLoader::executeGlobalScript (v8::Isolate* isolate,
|
|
v8::Handle<v8::Context> context,
|
|
string const& name) {
|
|
v8::TryCatch tryCatch;
|
|
v8::EscapableHandleScope scope(isolate);
|
|
v8::Handle<v8::Value> result;
|
|
|
|
findScript(name);
|
|
|
|
map<string, string>::iterator i = _scripts.find(name);
|
|
|
|
if (i == _scripts.end()) {
|
|
// correct the path/name
|
|
LOG_ERROR("unknown script '%s'", StringUtils::correctPath(name).c_str());
|
|
return v8::Undefined(isolate);
|
|
}
|
|
|
|
result = TRI_ExecuteJavaScriptString(isolate,
|
|
context,
|
|
TRI_V8_STD_STRING(i->second),
|
|
TRI_V8_STD_STRING(name),
|
|
false);
|
|
|
|
if (tryCatch.HasCaught()) {
|
|
if (tryCatch.CanContinue()) {
|
|
TRI_LogV8Exception(isolate, &tryCatch);/// TODO: could this be the place where we loose the information about parse errors of scripts?
|
|
return v8::Undefined(isolate);
|
|
}
|
|
else {
|
|
TRI_GET_GLOBALS();
|
|
|
|
v8g->_canceled = true;
|
|
}
|
|
}
|
|
return scope.Escape<v8::Value>(result);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
/// @brief loads a named script
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
JSLoader::eState JSLoader::loadScript(v8::Isolate* isolate,
|
|
v8::Handle<v8::Context>& context,
|
|
string const& name) {
|
|
v8::TryCatch tryCatch;
|
|
v8::HandleScope scope(isolate);
|
|
|
|
findScript(name);
|
|
|
|
map<string, string>::iterator i = _scripts.find(name);
|
|
|
|
if (i == _scripts.end()) {
|
|
// correct the path/name
|
|
LOG_ERROR("unknown script '%s'", StringUtils::correctPath(name).c_str());
|
|
return eFailLoad;
|
|
}
|
|
|
|
// Enter the newly created execution environment.
|
|
v8::Context::Scope context_scope(context);
|
|
|
|
TRI_ExecuteJavaScriptString(isolate,
|
|
context,
|
|
TRI_V8_STD_STRING(i->second),
|
|
TRI_V8_STD_STRING(name),
|
|
false);
|
|
|
|
if (tryCatch.HasCaught()) {
|
|
if (tryCatch.CanContinue()) {
|
|
TRI_LogV8Exception(isolate, &tryCatch);
|
|
return eFailExecute;
|
|
}
|
|
else {
|
|
TRI_GET_GLOBALS();
|
|
|
|
v8g->_canceled = true;
|
|
return eFailExecute;
|
|
}
|
|
}
|
|
|
|
return eSuccess;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
/// @brief loads all scripts
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool JSLoader::loadAllScripts (v8::Isolate* isolate,
|
|
v8::Handle<v8::Context>& context) {
|
|
v8::HandleScope scope(isolate);
|
|
|
|
if (_directory.empty()) {
|
|
return true;
|
|
}
|
|
|
|
vector<string> parts = getDirectoryParts();
|
|
|
|
bool result = true;
|
|
|
|
for (size_t i = 0; i < parts.size(); i++) {
|
|
result = result && TRI_ExecuteGlobalJavaScriptDirectory(isolate, parts.at(i).c_str());
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
/// @brief loads a named script
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool JSLoader::executeScript (v8::Isolate* isolate,
|
|
v8::Handle<v8::Context>& context,
|
|
string const& name) {
|
|
v8::TryCatch tryCatch;
|
|
v8::HandleScope scope(isolate);
|
|
|
|
findScript(name);
|
|
|
|
map<string, string>::iterator i = _scripts.find(name);
|
|
|
|
if (i == _scripts.end()) {
|
|
return false;
|
|
}
|
|
|
|
string content = "(function() { " + i->second + "/* end-of-file '" + name + "' */ })()";
|
|
|
|
// Enter the newly created execution environment.
|
|
v8::Context::Scope context_scope(context);
|
|
|
|
TRI_ExecuteJavaScriptString(isolate,
|
|
context,
|
|
TRI_V8_STD_STRING(content),
|
|
TRI_V8_STD_STRING(name),
|
|
false);
|
|
|
|
if (! tryCatch.HasCaught()) {
|
|
TRI_LogV8Exception(isolate, &tryCatch);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
/// @brief executes all scripts
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool JSLoader::executeAllScripts (v8::Isolate* isolate,
|
|
v8::Handle<v8::Context>& context) {
|
|
v8::TryCatch tryCatch;
|
|
v8::HandleScope scope(isolate);
|
|
bool ok;
|
|
|
|
if (_directory.empty()) {
|
|
return true;
|
|
}
|
|
|
|
ok = TRI_ExecuteLocalJavaScriptDirectory(isolate, _directory.c_str());
|
|
|
|
return ok;
|
|
}
|
|
|
|
// -----------------------------------------------------------------------------
|
|
// --SECTION-- END-OF-FILE
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// Local Variables:
|
|
// mode: outline-minor
|
|
// outline-regexp: "/// @brief\\|/// {@inheritDoc}\\|/// @page\\|// --SECTION--\\|/// @\\}"
|
|
// End:
|