//////////////////////////////////////////////////////////////////////////////// /// @brief V8 utility functions /// /// @file /// /// DISCLAIMER /// /// Copyright 2010-2011 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 triAGENS GmbH, Cologne, Germany /// /// @author Dr. Frank Celler /// @author Copyright 2011, triagens GmbH, Cologne, Germany //////////////////////////////////////////////////////////////////////////////// #include "v8-utils.h" #include #include #include #include #include #include #include #include #include #include "ShapedJson/shaped-json.h" #include #include "V8/v8-json.h" #include "V8/v8-globals.h" #if RL_READLINE_VERSION >= 0x0500 #define completion_matches rl_completion_matches #endif using namespace std; using namespace triagens::basics; // ----------------------------------------------------------------------------- // --SECTION-- forward declarations // ----------------------------------------------------------------------------- static bool FillShapeValueJson (TRI_shaper_t* shaper, TRI_shape_value_t* dst, v8::Handle json); static v8::Handle JsonShapeData (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size); // ----------------------------------------------------------------------------- // --SECTION-- class V8LineEditor // ----------------------------------------------------------------------------- // ----------------------------------------------------------------------------- // --SECTION-- private variables // ----------------------------------------------------------------------------- //////////////////////////////////////////////////////////////////////////////// /// @addtogroup V8Shell /// @{ //////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// /// @brief word break characters //////////////////////////////////////////////////////////////////////////////// static char WordBreakCharacters[] = { ' ', '\t', '\n', '"', '\\', '\'', '`', '@', '.', '>', '<', '=', ';', '|', '&', '{', '(', '\0' }; //////////////////////////////////////////////////////////////////////////////// /// @} //////////////////////////////////////////////////////////////////////////////// // ----------------------------------------------------------------------------- // --SECTION-- private functions // ----------------------------------------------------------------------------- //////////////////////////////////////////////////////////////////////////////// /// @addtogroup V8Shell /// @{ //////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// /// @brief completion generator //////////////////////////////////////////////////////////////////////////////// static char* CompletionGenerator (const char* text, int state) { return 0; } //////////////////////////////////////////////////////////////////////////////// /// @brief attempted completion //////////////////////////////////////////////////////////////////////////////// static char** AttemptedCompletion (const char* text, int start, int end) { char** result = completion_matches(text, CompletionGenerator); rl_attempted_completion_over = true; return result; } //////////////////////////////////////////////////////////////////////////////// /// @} //////////////////////////////////////////////////////////////////////////////// // ----------------------------------------------------------------------------- // --SECTION-- static public variables // ----------------------------------------------------------------------------- //////////////////////////////////////////////////////////////////////////////// /// @addtogroup V8Shell /// @{ //////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// /// @brief history file //////////////////////////////////////////////////////////////////////////////// const char* V8LineEditor::HISTORY_FILENAME = ".avocado"; //////////////////////////////////////////////////////////////////////////////// /// @} //////////////////////////////////////////////////////////////////////////////// // ----------------------------------------------------------------------------- // --SECTION-- public methods // ----------------------------------------------------------------------------- //////////////////////////////////////////////////////////////////////////////// /// @addtogroup V8Shell /// @{ //////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// /// @brief line editor open //////////////////////////////////////////////////////////////////////////////// bool V8LineEditor::open () { rl_initialize(); rl_attempted_completion_function = AttemptedCompletion; rl_completer_word_break_characters = WordBreakCharacters; rl_bind_key('\t', rl_complete); using_history(); stifle_history(MAX_HISTORY_ENTRIES); return read_history(HISTORY_FILENAME) == 0; } //////////////////////////////////////////////////////////////////////////////// /// @brief line editor prompt //////////////////////////////////////////////////////////////////////////////// char* V8LineEditor::prompt (char const* prompt) { char* result = readline(prompt); return result; } //////////////////////////////////////////////////////////////////////////////// /// @brief add to history //////////////////////////////////////////////////////////////////////////////// void V8LineEditor::addHistory (const char* str) { if (*str == '\0') { return; } history_set_pos(history_length-1); if (current_history()) { do { if (strcmp(current_history()->line, str) == 0) { remove_history(where_history()); break; } } while (previous_history()); } add_history(str); } //////////////////////////////////////////////////////////////////////////////// /// @} //////////////////////////////////////////////////////////////////////////////// // ----------------------------------------------------------------------------- // --SECTION-- CONVERSION FUNCTIONS // ----------------------------------------------------------------------------- // ----------------------------------------------------------------------------- // --SECTION-- private functions // ----------------------------------------------------------------------------- //////////////////////////////////////////////////////////////////////////////// /// @addtogroup V8Utils /// @{ //////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// /// @brief converts a null into TRI_shape_value_t //////////////////////////////////////////////////////////////////////////////// static bool FillShapeValueNull (TRI_shaper_t* shaper, TRI_shape_value_t* dst) { dst->_type = TRI_SHAPE_NULL; dst->_sid = shaper->_sidNull; dst->_fixedSized = true; dst->_size = 0; dst->_value = 0; return true; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a boolean into TRI_shape_value_t //////////////////////////////////////////////////////////////////////////////// static bool FillShapeValueBoolean (TRI_shaper_t* shaper, TRI_shape_value_t* dst, v8::Handle json) { TRI_shape_boolean_t* ptr; dst->_type = TRI_SHAPE_BOOLEAN; dst->_sid = shaper->_sidBoolean; dst->_fixedSized = true; dst->_size = sizeof(TRI_shape_boolean_t); dst->_value = (char*)(ptr = (TRI_shape_boolean_t*) TRI_Allocate(dst->_size)); *ptr = json->Value() ? 1 : 0; return true; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a boolean into TRI_shape_value_t //////////////////////////////////////////////////////////////////////////////// static bool FillShapeValueBoolean (TRI_shaper_t* shaper, TRI_shape_value_t* dst, v8::Handle json) { TRI_shape_boolean_t* ptr; dst->_type = TRI_SHAPE_BOOLEAN; dst->_sid = shaper->_sidBoolean; dst->_fixedSized = true; dst->_size = sizeof(TRI_shape_boolean_t); dst->_value = (char*)(ptr = (TRI_shape_boolean_t*) TRI_Allocate(dst->_size)); *ptr = json->BooleanValue() ? 1 : 0; return true; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a number into TRI_shape_value_t //////////////////////////////////////////////////////////////////////////////// static bool FillShapeValueNumber (TRI_shaper_t* shaper, TRI_shape_value_t* dst, v8::Handle json) { TRI_shape_number_t* ptr; dst->_type = TRI_SHAPE_NUMBER; dst->_sid = shaper->_sidNumber; dst->_fixedSized = true; dst->_size = sizeof(TRI_shape_number_t); dst->_value = (char*)(ptr = (TRI_shape_number_t*) TRI_Allocate(dst->_size)); *ptr = json->Value(); return true; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a number into TRI_shape_value_t //////////////////////////////////////////////////////////////////////////////// static bool FillShapeValueNumber (TRI_shaper_t* shaper, TRI_shape_value_t* dst, v8::Handle json) { TRI_shape_number_t* ptr; dst->_type = TRI_SHAPE_NUMBER; dst->_sid = shaper->_sidNumber; dst->_fixedSized = true; dst->_size = sizeof(TRI_shape_number_t); dst->_value = (char*)(ptr = (TRI_shape_number_t*) TRI_Allocate(dst->_size)); *ptr = json->NumberValue(); return true; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a string into TRI_shape_value_t //////////////////////////////////////////////////////////////////////////////// static bool FillShapeValueString (TRI_shaper_t* shaper, TRI_shape_value_t* dst, v8::Handle json) { char* ptr; v8::String::Utf8Value str(json); if (*str == 0) { dst->_type = TRI_SHAPE_SHORT_STRING; dst->_sid = shaper->_sidShortString; dst->_fixedSized = true; dst->_size = sizeof(TRI_shape_length_short_string_t) + TRI_SHAPE_SHORT_STRING_CUT; dst->_value = (ptr = (char*) TRI_Allocate(dst->_size)); * ((TRI_shape_length_short_string_t*) ptr) = 1; * (ptr + sizeof(TRI_shape_length_short_string_t)) = '\0'; } else if (str.length() < TRI_SHAPE_SHORT_STRING_CUT) { // includes '\0' size_t size = str.length() + 1; dst->_type = TRI_SHAPE_SHORT_STRING; dst->_sid = shaper->_sidShortString; dst->_fixedSized = true; dst->_size = sizeof(TRI_shape_length_short_string_t) + TRI_SHAPE_SHORT_STRING_CUT; dst->_value = (ptr = (char*) TRI_Allocate(dst->_size)); * ((TRI_shape_length_short_string_t*) ptr) = size; memcpy(ptr + sizeof(TRI_shape_length_short_string_t), *str, size); } else { size_t size = str.length() + 1; dst->_type = TRI_SHAPE_LONG_STRING; dst->_sid = shaper->_sidLongString; dst->_fixedSized = false; dst->_size = sizeof(TRI_shape_length_long_string_t) + size; dst->_value = (ptr = (char*) TRI_Allocate(dst->_size)); * ((TRI_shape_length_long_string_t*) ptr) = size; memcpy(ptr + sizeof(TRI_shape_length_long_string_t), *str, size); } return true; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a json list into TRI_shape_value_t //////////////////////////////////////////////////////////////////////////////// static bool FillShapeValueList (TRI_shaper_t* shaper, TRI_shape_value_t* dst, v8::Handle json) { size_t i; size_t n; size_t total; TRI_shape_value_t* values; TRI_shape_value_t* p; TRI_shape_value_t* e; bool hs; bool hl; TRI_shape_sid_t s; TRI_shape_sid_t l; TRI_shape_sid_t* sids; TRI_shape_size_t* offsets; TRI_shape_size_t offset; TRI_shape_t const* found; char* ptr; // check for special case "empty list" n = json->Length(); if (n == 0) { dst->_type = TRI_SHAPE_LIST; dst->_sid = shaper->_sidList; dst->_fixedSized = false; dst->_size = sizeof(TRI_shape_length_list_t); dst->_value = (ptr = (char*) TRI_Allocate(dst->_size)); * (TRI_shape_length_list_t*) ptr = 0; return true; } // convert into TRI_shape_value_t array p = (values = (TRI_shape_value_t*) TRI_Allocate(sizeof(TRI_shape_value_t) * n)); memset(values, 0, sizeof(TRI_shape_value_t) * n); total = 0; e = values + n; for (i = 0; i < n; ++i, ++p) { v8::Local el = json->Get(i); bool ok = FillShapeValueJson(shaper, p, el); if (! ok) { for (e = p, p = values; p < e; ++p) { if (p->_value != 0) { TRI_Free(p->_value); } } TRI_Free(values); return false; } total += p->_size; } // check if this list is homoegenous hs = true; hl = true; s = values[0]._sid; l = values[0]._size; p = values; for (; p < e; ++p) { if (p->_sid != s) { hs = false; break; } if (p->_size != l) { hl = false; } } // homogeneous sized if (hs && hl) { TRI_homogeneous_sized_list_shape_t* shape; shape = (TRI_homogeneous_sized_list_shape_t*) TRI_Allocate(sizeof(TRI_homogeneous_sized_list_shape_t)); shape->base._size = sizeof(TRI_homogeneous_sized_list_shape_t); shape->base._type = TRI_SHAPE_HOMOGENEOUS_SIZED_LIST; shape->base._dataSize = TRI_SHAPE_SIZE_VARIABLE; shape->_sidEntry = s; shape->_sizeEntry = l; found = shaper->findShape(shaper, &shape->base); if (found == 0) { for (p = values; p < e; ++p) { if (p->_value != 0) { TRI_Free(p->_value); } } TRI_Free(values); TRI_Free(shape); return false; } dst->_type = found->_type; dst->_sid = found->_sid; dst->_fixedSized = false; dst->_size = sizeof(TRI_shape_length_list_t) + total; dst->_value = (ptr = (char*) TRI_Allocate(dst->_size)); // copy sub-objects into data space * (TRI_shape_length_list_t*) ptr = n; ptr += sizeof(TRI_shape_length_list_t); for (p = values; p < e; ++p) { memcpy(ptr, p->_value, p->_size); ptr += p->_size; } } // homogeneous else if (hs) { TRI_homogeneous_list_shape_t* shape; shape = (TRI_homogeneous_list_shape_t*) TRI_Allocate(sizeof(TRI_homogeneous_list_shape_t)); shape->base._size = sizeof(TRI_homogeneous_list_shape_t); shape->base._type = TRI_SHAPE_HOMOGENEOUS_LIST; shape->base._dataSize = TRI_SHAPE_SIZE_VARIABLE; shape->_sidEntry = s; found = shaper->findShape(shaper, &shape->base); if (found == 0) { for (p = values; p < e; ++p) { if (p->_value != 0) { TRI_Free(p->_value); } } TRI_Free(values); TRI_Free(shape); return false; } dst->_type = found->_type; dst->_sid = found->_sid; offset = sizeof(TRI_shape_length_list_t) + (n + 1) * sizeof(TRI_shape_size_t); dst->_fixedSized = false; dst->_size = offset + total; dst->_value = (ptr = (char*) TRI_Allocate(dst->_size)); // copy sub-objects into data space * (TRI_shape_length_list_t*) ptr = n; ptr += sizeof(TRI_shape_length_list_t); offsets = (TRI_shape_size_t*) ptr; ptr += (n + 1) * sizeof(TRI_shape_size_t); for (p = values; p < e; ++p) { *offsets++ = offset; offset += p->_size; memcpy(ptr, p->_value, p->_size); ptr += p->_size; } *offsets = offset; } // in-homogeneous else { dst->_type = TRI_SHAPE_LIST; dst->_sid = shaper->_sidList; offset = sizeof(TRI_shape_length_list_t) + n * sizeof(TRI_shape_sid_t) + (n + 1) * sizeof(TRI_shape_size_t); dst->_fixedSized = false; dst->_size = offset + total; dst->_value = (ptr = (char*) TRI_Allocate(dst->_size)); // copy sub-objects into data space * (TRI_shape_length_list_t*) ptr = n; ptr += sizeof(TRI_shape_length_list_t); sids = (TRI_shape_sid_t*) ptr; ptr += n * sizeof(TRI_shape_sid_t); offsets = (TRI_shape_size_t*) ptr; ptr += (n + 1) * sizeof(TRI_shape_size_t); for (p = values; p < e; ++p) { *sids++ = p->_sid; *offsets++ = offset; offset += p->_size; memcpy(ptr, p->_value, p->_size); ptr += p->_size; } *offsets = offset; } // free TRI_shape_value_t array for (p = values; p < e; ++p) { if (p->_value != 0) { TRI_Free(p->_value); } } TRI_Free(values); return true; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a json array into TRI_shape_value_t //////////////////////////////////////////////////////////////////////////////// static bool FillShapeValueArray (TRI_shaper_t* shaper, TRI_shape_value_t* dst, v8::Handle json) { size_t n; size_t i; size_t total; size_t f; size_t v; TRI_shape_value_t* values; TRI_shape_value_t* p; TRI_shape_value_t* e; TRI_array_shape_t* a; TRI_shape_sid_t* sids; TRI_shape_aid_t* aids; TRI_shape_size_t* offsetsF; TRI_shape_size_t* offsetsV; TRI_shape_size_t offset; TRI_shape_t const* found; char* ptr; // number of attributes v8::Handle names = json->GetPropertyNames(); n = names->Length(); // convert into TRI_shape_value_t array p = (values = (TRI_shape_value_t*) TRI_Allocate(n * sizeof(TRI_shape_value_t))); memset(values, 0, n * sizeof(TRI_shape_value_t)); total = 0; f = 0; v = 0; for (i = 0; i < n; ++i, ++p) { v8::Handle key = names->Get(i); v8::Handle val = json->Get(key); // first find an identifier for the name v8::String::Utf8Value keyStr(key); if (*keyStr == 0) { --p; continue; } if (TRI_EqualString(*keyStr, "_id")) { --p; continue; } p->_aid = shaper->findAttributeName(shaper, *keyStr); // convert value bool ok; if (p->_aid == 0) { ok = false; } else { ok = FillShapeValueJson(shaper, p, val); } if (! ok) { for (e = p, p = values; p < e; ++p) { if (p->_value != 0) { TRI_Free(p->_value); } } TRI_Free(values); return false; } total += p->_size; // count fixed and variable sized values if (p->_fixedSized) { ++f; } else { ++v; } } n = f + v; // add variable offset table size total += (v + 1) * sizeof(TRI_shape_size_t); // now sort the shape entries TRI_SortShapeValues(values, n); #ifdef DEBUG_JSON_SHAPER printf("shape values\n------------\ntotal: %u, fixed: %u, variable: %u\n", (unsigned int) n, (unsigned int) f, (unsigned int) v); PrintShapeValues(values, n); printf("\n"); #endif // generate shape structure i = sizeof(TRI_array_shape_t) + n * sizeof(TRI_shape_sid_t) + n * sizeof(TRI_shape_aid_t) + (f + 1) * sizeof(TRI_shape_size_t); a = (TRI_array_shape_t*) (ptr = (char*) TRI_Allocate(i)); memset(ptr, 0, i); a->base._type = TRI_SHAPE_ARRAY; a->base._size = i; a->base._dataSize = (v == 0) ? total : TRI_SHAPE_SIZE_VARIABLE; a->_fixedEntries = f; a->_variableEntries = v; ptr += sizeof(TRI_array_shape_t); // array of shape identifiers sids = (TRI_shape_sid_t*) ptr; ptr += n * sizeof(TRI_shape_sid_t); // array of attribute identifiers aids = (TRI_shape_aid_t*) ptr; ptr += n * sizeof(TRI_shape_aid_t); // array of offsets for fixed part (within the shape) offset = (v + 1) * sizeof(TRI_shape_size_t); offsetsF = (TRI_shape_size_t*) ptr; // fill destination (except sid) dst->_type = TRI_SHAPE_ARRAY; dst->_fixedSized = true; dst->_size = total; dst->_value = (ptr = (char*) TRI_Allocate(dst->_size)); // array of offsets for variable part (within the value) offsetsV = (TRI_shape_size_t*) ptr; ptr += (v + 1) * sizeof(TRI_shape_size_t); // and fill in attributes e = values + n; for (p = values; p < e; ++p) { *aids++ = p->_aid; *sids++ = p->_sid; memcpy(ptr, p->_value, p->_size); ptr += p->_size; dst->_fixedSized &= p->_fixedSized; if (p->_fixedSized) { *offsetsF++ = offset; offset += p->_size; *offsetsF = offset; } else { *offsetsV++ = offset; offset += p->_size; *offsetsV = offset; } } // free TRI_shape_value_t array for (p = values; p < e; ++p) { if (p->_value != 0) { TRI_Free(p->_value); } } TRI_Free(values); // lookup this shape found = shaper->findShape(shaper, &a->base); if (found == 0) { TRI_Free(a); return false; } // and finally add the sid dst->_sid = found->_sid; return true; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a json object into TRI_shape_value_t //////////////////////////////////////////////////////////////////////////////// static bool FillShapeValueJson (TRI_shaper_t* shaper, TRI_shape_value_t* dst, v8::Handle json) { if (json->IsNull()) { return FillShapeValueNull(shaper, dst); } if (json->IsBoolean()) { return FillShapeValueBoolean(shaper, dst, json->ToBoolean()); } if (json->IsBooleanObject()) { v8::Handle bo = v8::Handle::Cast(json); return FillShapeValueBoolean(shaper, dst, bo); } if (json->IsNumber()) { return FillShapeValueNumber(shaper, dst, json->ToNumber()); } if (json->IsNumberObject()) { v8::Handle no = v8::Handle::Cast(json); return FillShapeValueNumber(shaper, dst, no); } if (json->IsString()) { return FillShapeValueString(shaper, dst, json->ToString()); } if (json->IsStringObject()) { v8::Handle so = v8::Handle::Cast(json); return FillShapeValueString(shaper, dst, so->StringValue()); } if (json->IsArray()) { v8::Handle array = v8::Handle::Cast(json); return FillShapeValueList(shaper, dst, array); } if (json->IsObject()) { return FillShapeValueArray(shaper, dst, json->ToObject()); } return false; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a data null blob into a json object //////////////////////////////////////////////////////////////////////////////// static v8::Handle JsonShapeDataNull (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size) { return v8::Null(); } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a data boolean blob into a json object //////////////////////////////////////////////////////////////////////////////// static v8::Handle JsonShapeDataBoolean (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size) { bool v; v = (* (TRI_shape_boolean_t const*) data) != 0; return v ? v8::True() : v8::False(); } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a data number blob into a json object //////////////////////////////////////////////////////////////////////////////// static v8::Handle JsonShapeDataNumber (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size) { TRI_shape_number_t v; v = * (TRI_shape_number_t const*) data; return v8::Number::New(v); } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a data short string blob into a json object //////////////////////////////////////////////////////////////////////////////// static v8::Handle JsonShapeDataShortString (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size) { TRI_shape_length_short_string_t l; l = * (TRI_shape_length_short_string_t const*) data; data += sizeof(TRI_shape_length_short_string_t); return v8::String::New(data, l - 1); } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a data long string blob into a json object //////////////////////////////////////////////////////////////////////////////// static v8::Handle JsonShapeDataLongString (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size) { TRI_shape_length_long_string_t l; l = * (TRI_shape_length_long_string_t const*) data; data += sizeof(TRI_shape_length_long_string_t); return v8::String::New(data, l - 1); } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a data array blob into a json object //////////////////////////////////////////////////////////////////////////////// static v8::Handle JsonShapeDataArray (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size) { TRI_array_shape_t const* s; TRI_shape_aid_t const* aids; TRI_shape_sid_t const* sids; TRI_shape_size_t const* offsetsF; TRI_shape_size_t const* offsetsV; TRI_shape_size_t f; TRI_shape_size_t i; TRI_shape_size_t n; TRI_shape_size_t v; char const* qtr; v8::Handle array; s = (TRI_array_shape_t const*) shape; f = s->_fixedEntries; v = s->_variableEntries; n = f + v; qtr = (char const*) shape; array = v8::Object::New(); qtr += sizeof(TRI_array_shape_t); sids = (TRI_shape_sid_t const*) qtr; qtr += n * sizeof(TRI_shape_sid_t); aids = (TRI_shape_aid_t const*) qtr; qtr += n * sizeof(TRI_shape_aid_t); offsetsF = (TRI_shape_size_t const*) qtr; for (i = 0; i < f; ++i, ++sids, ++aids, ++offsetsF) { TRI_shape_sid_t sid = *sids; TRI_shape_aid_t aid = *aids; TRI_shape_size_t offset; TRI_shape_t const* subshape; char const* name; v8::Handle element; offset = *offsetsF; subshape = shaper->lookupShapeId(shaper, sid); name = shaper->lookupAttributeId(shaper, aid); if (subshape == 0) { LOG_WARNING("cannot find shape #%u", (unsigned int) sid); continue; } if (name == 0) { LOG_WARNING("cannot find attribute #%u", (unsigned int) aid); continue; } element = JsonShapeData(shaper, subshape, data + offset, offsetsF[1] - offset); array->Set(v8::String::New(name), element); } offsetsV = (TRI_shape_size_t const*) data; for (i = 0; i < v; ++i, ++sids, ++aids, ++offsetsV) { TRI_shape_sid_t sid = *sids; TRI_shape_aid_t aid = *aids; TRI_shape_size_t offset; TRI_shape_t const* subshape; char const* name; v8::Handle element; offset = *offsetsV; subshape = shaper->lookupShapeId(shaper, sid); name = shaper->lookupAttributeId(shaper, aid); if (subshape == 0) { LOG_WARNING("cannot find shape #%u", (unsigned int) sid); continue; } if (name == 0) { LOG_WARNING("cannot find attribute #%u", (unsigned int) aid); continue; } element = JsonShapeData(shaper, subshape, data + offset, offsetsV[1] - offset); array->Set(v8::String::New(name), element); } return array; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a data list blob into a json object //////////////////////////////////////////////////////////////////////////////// static v8::Handle JsonShapeDataList (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size) { TRI_shape_length_list_t i; TRI_shape_length_list_t l; TRI_shape_sid_t const* sids; TRI_shape_size_t const* offsets; char const* ptr; v8::Handle list; list = v8::Array::New(); ptr = data; l = * (TRI_shape_length_list_t const*) ptr; ptr += sizeof(TRI_shape_length_list_t); sids = (TRI_shape_sid_t const*) ptr; ptr += l * sizeof(TRI_shape_sid_t); offsets = (TRI_shape_size_t const*) ptr; for (i = 0; i < l; ++i, ++sids, ++offsets) { TRI_shape_sid_t sid = *sids; TRI_shape_size_t offset; TRI_shape_t const* subshape; v8::Handle element; offset = *offsets; subshape = shaper->lookupShapeId(shaper, sid); if (subshape == 0) { LOG_WARNING("cannot find shape #%u", (unsigned int) sid); continue; } element = JsonShapeData(shaper, subshape, data + offset, offsets[1] - offset); list->Set(i, element); } return list; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a data homogeneous list blob into a json object //////////////////////////////////////////////////////////////////////////////// static v8::Handle JsonShapeDataHomogeneousList (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size) { TRI_homogeneous_list_shape_t const* s; TRI_shape_length_list_t i; TRI_shape_length_list_t l; TRI_shape_sid_t sid; TRI_shape_size_t const* offsets; char const* ptr; v8::Handle list; list = v8::Array::New(); s = (TRI_homogeneous_list_shape_t const*) shape; sid = s->_sidEntry; ptr = data; l = * (TRI_shape_length_list_t const*) ptr; ptr += sizeof(TRI_shape_length_list_t); offsets = (TRI_shape_size_t const*) ptr; for (i = 0; i < l; ++i, ++offsets) { TRI_shape_size_t offset; TRI_shape_t const* subshape; v8::Handle element; offset = *offsets; subshape = shaper->lookupShapeId(shaper, sid); if (subshape == 0) { LOG_WARNING("cannot find shape #%u", (unsigned int) sid); continue; } element = JsonShapeData(shaper, subshape, data + offset, offsets[1] - offset); list->Set(i, element); } return list; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a data homogeneous sized list blob into a json object //////////////////////////////////////////////////////////////////////////////// static v8::Handle JsonShapeDataHomogeneousSizedList (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size) { TRI_homogeneous_sized_list_shape_t const* s; TRI_shape_length_list_t i; TRI_shape_length_list_t l; TRI_shape_sid_t sid; TRI_shape_size_t length; TRI_shape_size_t offset; char const* ptr; v8::Handle list; list = v8::Array::New(); s = (TRI_homogeneous_sized_list_shape_t const*) shape; sid = s->_sidEntry; ptr = data; l = * (TRI_shape_length_list_t const*) ptr; length = s->_sizeEntry; offset = sizeof(TRI_shape_length_list_t); for (i = 0; i < l; ++i, offset += length) { TRI_shape_t const* subshape; v8::Handle element; subshape = shaper->lookupShapeId(shaper, sid); if (subshape == 0) { LOG_WARNING("cannot find shape #%u", (unsigned int) sid); continue; } element = JsonShapeData(shaper, subshape, data + offset, length); list->Set(i, element); } return list; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a data blob into a json object //////////////////////////////////////////////////////////////////////////////// static v8::Handle JsonShapeData (TRI_shaper_t* shaper, TRI_shape_t const* shape, char const* data, size_t size) { if (shape == 0) { return v8::Null(); } switch (shape->_type) { case TRI_SHAPE_NULL: return JsonShapeDataNull(shaper, shape, data, size); case TRI_SHAPE_BOOLEAN: return JsonShapeDataBoolean(shaper, shape, data, size); case TRI_SHAPE_NUMBER: return JsonShapeDataNumber(shaper, shape, data, size); case TRI_SHAPE_SHORT_STRING: return JsonShapeDataShortString(shaper, shape, data, size); case TRI_SHAPE_LONG_STRING: return JsonShapeDataLongString(shaper, shape, data, size); case TRI_SHAPE_ARRAY: return JsonShapeDataArray(shaper, shape, data, size); case TRI_SHAPE_LIST: return JsonShapeDataList(shaper, shape, data, size); case TRI_SHAPE_HOMOGENEOUS_LIST: return JsonShapeDataHomogeneousList(shaper, shape, data, size); case TRI_SHAPE_HOMOGENEOUS_SIZED_LIST: return JsonShapeDataHomogeneousSizedList(shaper, shape, data, size); } return v8::Null(); } //////////////////////////////////////////////////////////////////////////////// /// @} //////////////////////////////////////////////////////////////////////////////// // ----------------------------------------------------------------------------- // --SECTION-- public functions // ----------------------------------------------------------------------------- //////////////////////////////////////////////////////////////////////////////// /// @addtogroup V8Utils /// @{ //////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// /// @brief converts a TRI_json_t NULL into a V8 object //////////////////////////////////////////////////////////////////////////////// v8::Handle ObjectJsonNull (TRI_json_t const* json) { return v8::Null(); } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a TRI_json_t BOOLEAN into a V8 object //////////////////////////////////////////////////////////////////////////////// v8::Handle ObjectJsonBoolean (TRI_json_t const* json) { return json->_value._boolean ? v8::True() : v8::False(); } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a TRI_json_t NUMBER into a V8 object //////////////////////////////////////////////////////////////////////////////// v8::Handle ObjectJsonNumber (TRI_json_t const* json) { return v8::Number::New(json->_value._number); } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a TRI_json_t NUMBER into a V8 object //////////////////////////////////////////////////////////////////////////////// v8::Handle ObjectJsonString (TRI_json_t const* json) { return v8::String::New(json->_value._string.data); } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a TRI_json_t ARRAY into a V8 object //////////////////////////////////////////////////////////////////////////////// v8::Handle ObjectJsonArray (TRI_json_t const* json) { v8::Handle object = v8::Object::New(); size_t n = TRI_SizeVector(&json->_value._objects); for (size_t i = 0; i < n; i += 2) { TRI_json_t* key = (TRI_json_t*) TRI_AtVector(&json->_value._objects, i); if (key->_type != TRI_JSON_STRING) { continue; } TRI_json_t* j = (TRI_json_t*) TRI_AtVector(&json->_value._objects, i + 1); v8::Handle val = TRI_ObjectJson(j); object->Set(v8::String::New(key->_value._string.data), val); } return object; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a TRI_json_t LIST into a V8 object //////////////////////////////////////////////////////////////////////////////// v8::Handle ObjectJsonList (TRI_json_t const* json) { v8::Handle object = v8::Array::New(); size_t n = TRI_SizeVector(&json->_value._objects); for (size_t i = 0; i < n; ++i) { TRI_json_t* j = (TRI_json_t*) TRI_AtVector(&json->_value._objects, i); v8::Handle val = TRI_ObjectJson(j); object->Set(i, val); } return object; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a TRI_json_t into a V8 object //////////////////////////////////////////////////////////////////////////////// v8::Handle TRI_ObjectJson (TRI_json_t const* json) { v8::HandleScope scope; switch (json->_type) { case TRI_JSON_NULL: return scope.Close(ObjectJsonNull(json)); case TRI_JSON_BOOLEAN: return scope.Close(ObjectJsonBoolean(json)); case TRI_JSON_NUMBER: return scope.Close(ObjectJsonNumber(json)); case TRI_JSON_STRING: return scope.Close(ObjectJsonString(json)); case TRI_JSON_ARRAY: return scope.Close(ObjectJsonArray(json)); case TRI_JSON_LIST: return scope.Close(ObjectJsonList(json)); } return scope.Close(v8::Undefined()); } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a TRI_shaped_json_t into a V8 object //////////////////////////////////////////////////////////////////////////////// v8::Handle TRI_ObjectShapedJson (TRI_doc_collection_t* collection, TRI_voc_did_t did, TRI_shaper_t* shaper, TRI_shaped_json_t const* shaped) { TRI_shape_t const* shape; TRI_v8_global_t* v8g; v8g = (TRI_v8_global_t*) v8::Isolate::GetCurrent()->GetData(); shape = shaper->lookupShapeId(shaper, shaped->_sid); if (shape == 0) { LOG_WARNING("cannot find shape #%u", (unsigned int) shaped->_sid); return v8::Null(); } v8::Handle result = JsonShapeData(shaper, shape, shaped->_data.data, shaped->_data.length); if (result->IsObject()) { char* cidStr = TRI_StringUInt64(collection->base._cid); char* didStr = TRI_StringUInt64(did); string name = cidStr + string(":") + didStr; TRI_FreeString(didStr); TRI_FreeString(cidStr); result->ToObject()->Set(v8g->DidKey, v8::String::New(name.c_str())); } return result; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts a TRI_result_set_t into a V8 array //////////////////////////////////////////////////////////////////////////////// v8::Handle TRI_ArrayResultSet (TRI_result_set_t* rs) { v8::Handle array = v8::Array::New(); TRI_doc_collection_t* collection = rs->_containerElement->_container->_collection; TRI_shaper_t* shaper = collection->_shaper; size_t pos; for (pos = 0; rs->hasNext(rs); ++pos) { TRI_voc_did_t did; TRI_voc_rid_t rid; TRI_json_t const* augmented; TRI_shaped_json_t* element = rs->next(rs, &did, &rid, &augmented); v8::Handle object = TRI_ObjectShapedJson(collection, did, shaper, element); if (augmented != NULL) { TRI_AugmentObject(object, augmented); } array->Set(pos, object); } return array; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts an V8 object to a TRI_shaped_json_t //////////////////////////////////////////////////////////////////////////////// TRI_shaped_json_t* TRI_ShapedJsonV8Object (v8::Handle object, TRI_shaper_t* shaper) { TRI_shape_value_t dst; bool ok = FillShapeValueJson(shaper, &dst, object); if (! ok) { return 0; } TRI_shaped_json_t* shaped = (TRI_shaped_json_t*) TRI_Allocate(sizeof(TRI_shaped_json_t)); shaped->_sid = dst._sid; shaped->_data.length = dst._size; shaped->_data.data = dst._value; return shaped; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts an V8 object to a string //////////////////////////////////////////////////////////////////////////////// string TRI_ObjectToString (v8::Handle value) { v8::String::Utf8Value utf8Value(value); if (*utf8Value == 0) { return ""; } else { return string(*utf8Value, utf8Value.length()); } } //////////////////////////////////////////////////////////////////////////////// /// @brief converts an V8 object to a character //////////////////////////////////////////////////////////////////////////////// char TRI_ObjectToCharacter (v8::Handle value, bool& error) { error = false; if (! value->IsString() && ! value->IsStringObject()) { error = true; return '\0'; } v8::String::Utf8Value sep(value->ToString()); if (*sep == 0 || sep.length() != 1) { error = true; return '\0'; } return (*sep)[0]; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts an V8 object to a double //////////////////////////////////////////////////////////////////////////////// double TRI_ObjectToDouble (v8::Handle value) { if (value->IsNumber()) { return value->ToNumber()->Value(); } if (value->IsNumberObject()) { v8::Handle no = v8::Handle::Cast(value); return no->NumberValue(); } return 0.0; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts an V8 object to a double //////////////////////////////////////////////////////////////////////////////// double TRI_ObjectToDouble (v8::Handle value, bool& error) { error = false; if (value->IsNumber()) { return value->ToNumber()->Value(); } if (value->IsNumberObject()) { v8::Handle no = v8::Handle::Cast(value); return no->NumberValue(); } error = true; return 0.0; } //////////////////////////////////////////////////////////////////////////////// /// @brief converts an V8 object to a boolean //////////////////////////////////////////////////////////////////////////////// bool TRI_ObjectToBoolean (v8::Handle value) { if (value->IsBoolean()) { return value->ToBoolean()->Value(); } else if (value->IsBooleanObject()) { v8::Handle bo = v8::Handle::Cast(value); return bo->BooleanValue(); } return false; } //////////////////////////////////////////////////////////////////////////////// /// @} //////////////////////////////////////////////////////////////////////////////// // ----------------------------------------------------------------------------- // --SECTION-- GENERAL // ----------------------------------------------------------------------------- // ----------------------------------------------------------------------------- // --SECTION-- JS functions // ----------------------------------------------------------------------------- //////////////////////////////////////////////////////////////////////////////// /// @addtogroup V8Utils /// @{ //////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// /// @brief converts JSON string to object /// /// @FUN{fromJson(@FA{string})} /// /// Converts a string representation of a JSON object into a JavaScript /// object. /// /// @verbinclude fluent35 //////////////////////////////////////////////////////////////////////////////// static v8::Handle JS_FromJson (v8::Arguments const& argv) { v8::HandleScope scope; if (argv.Length() != 1) { return scope.Close(v8::ThrowException(v8::String::New("usage: fromJson()"))); } v8::String::Utf8Value str(argv[0]); if (*str == 0) { return scope.Close(v8::ThrowException(v8::String::New("cannot get UTF-8 representation of "))); } char* error; v8::Handle object = TRI_FromJsonString(*str, &error); if (object->IsUndefined()) { v8::Handle msg = v8::String::New(error); TRI_FreeString(error); return scope.Close(v8::ThrowException(msg)); } return scope.Close(object); } //////////////////////////////////////////////////////////////////////////////// /// @brief returns the current time /// /// @FUN{time()} /// /// Returns the current time in seconds. /// /// @verbinclude fluent36 //////////////////////////////////////////////////////////////////////////////// static v8::Handle JS_Time (v8::Arguments const& argv) { v8::HandleScope scope; return scope.Close(v8::Number::New(TRI_microtime())); } //////////////////////////////////////////////////////////////////////////////// /// @} //////////////////////////////////////////////////////////////////////////////// // ----------------------------------------------------------------------------- // --SECTION-- public functions // ----------------------------------------------------------------------------- //////////////////////////////////////////////////////////////////////////////// /// @addtogroup V8Utils /// @{ //////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////// /// @brief adds attributes to array //////////////////////////////////////////////////////////////////////////////// void TRI_AugmentObject (v8::Handle value, TRI_json_t const* json) { v8::HandleScope scope; if (! value->IsObject()) { return; } if (json->_type != TRI_JSON_ARRAY) { return; } v8::Handle object = value->ToObject(); size_t n = TRI_SizeVector(&json->_value._objects); for (size_t i = 0; i < n; i += 2) { TRI_json_t* key = (TRI_json_t*) TRI_AtVector(&json->_value._objects, i); if (key->_type != TRI_JSON_STRING) { continue; } TRI_json_t* j = (TRI_json_t*) TRI_AtVector(&json->_value._objects, i + 1); v8::Handle val = TRI_ObjectJson(j); object->Set(v8::String::New(key->_value._string.data), val); } } //////////////////////////////////////////////////////////////////////////////// /// @brief reports an exception //////////////////////////////////////////////////////////////////////////////// string TRI_ReportV8Exception (v8::TryCatch* tryCatch) { v8::HandleScope handle_scope; v8::String::Utf8Value exception(tryCatch->Exception()); const char* exceptionString = *exception; v8::Handle message = tryCatch->Message(); string result; // V8 didn't provide any extra information about this error; just print the exception. if (message.IsEmpty()) { if (exceptionString == 0) { LOG_ERROR("JavaScript exception"); result = "JavaScript exception"; } else { LOG_ERROR("JavaScript exception: %s", exceptionString); result = "JavaScript exception: " + string(exceptionString); } } else { v8::String::Utf8Value filename(message->GetScriptResourceName()); const char* filenameString = *filename; int linenum = message->GetLineNumber(); int start = message->GetStartColumn() + 1; if (filenameString == 0) { if (exceptionString == 0) { LOG_ERROR("JavaScript exception"); result = "JavaScript exception"; } else { LOG_ERROR("JavaScript exception: %s", exceptionString); result = "JavaScript exception: " + string(exceptionString); } } else { if (exceptionString == 0) { LOG_ERROR("JavaScript exception in file '%s' at %d,%d", filenameString, linenum, start); result = "JavaScript exception in file '" + string(filenameString) + "' at " + StringUtils::itoa(linenum) + "," + StringUtils::itoa(start); } else { LOG_ERROR("JavaScript exception in file '%s' at %d,%d: %s", filenameString, linenum, start, exceptionString); result = "JavaScript exception in file '" + string(filenameString) + "' at " + StringUtils::itoa(linenum) + "," + StringUtils::itoa(start) + ": " + exceptionString; } } v8::String::Utf8Value stacktrace(tryCatch->StackTrace()); if (*stacktrace && stacktrace.length() > 0) { LOG_DEBUG("stacktrace: %s", *stacktrace); } } return result; } //////////////////////////////////////////////////////////////////////////////// /// @brief prints an exception and stacktrace //////////////////////////////////////////////////////////////////////////////// void TRI_PrintV8Exception (v8::TryCatch* tryCatch) { v8::HandleScope handle_scope; v8::String::Utf8Value exception(tryCatch->Exception()); const char* exceptionString = *exception; v8::Handle message = tryCatch->Message(); // V8 didn't provide any extra information about this error; just print the exception. if (message.IsEmpty()) { if (exceptionString == 0) { printf("JavaScript exception\n"); } else { printf("JavaScript exception: %s\n", exceptionString); } } else { v8::String::Utf8Value filename(message->GetScriptResourceName()); const char* filenameString = *filename; int linenum = message->GetLineNumber(); int start = message->GetStartColumn() + 1; int end = message->GetEndColumn(); if (filenameString == 0) { if (exceptionString == 0) { printf("exception\n"); } else { printf("exception: %s\n", exceptionString); } } else { if (exceptionString == 0) { printf("exception in file '%s' at %d,%d\n", filenameString, linenum, start); } else { printf("exception in file '%s' at %d,%d: %s\n", filenameString, linenum, start, exceptionString); } } v8::String::Utf8Value sourceline(message->GetSourceLine()); if (*sourceline) { printf("%s\n", *sourceline); if (1 < start) { printf("%*s", start - 1, ""); } string e((size_t)(end - start + 1), '^'); printf("%s\n", e.c_str()); } v8::String::Utf8Value stacktrace(tryCatch->StackTrace()); if (*stacktrace && stacktrace.length() > 0) { printf("stacktrace:\n%s\n", *stacktrace); } } } //////////////////////////////////////////////////////////////////////////////// /// @brief reads a file into the current context //////////////////////////////////////////////////////////////////////////////// bool TRI_LoadJavaScriptFile (v8::Handle context, char const* filename) { v8::HandleScope handleScope; v8::TryCatch tryCatch; char* content = TRI_SlurpFile(filename); if (content == 0) { LOG_TRACE("cannot loaded java script file '%s': %s", filename, TRI_last_error()); return false; } v8::Handle name = v8::String::New(filename); v8::Handle source = v8::String::New(content); TRI_FreeString(content); v8::Handle script = v8::Script::Compile(source, name); // compilation failed, print errors that happened during compilation if (script.IsEmpty()) { TRI_ReportV8Exception(&tryCatch); return false; } // execute script v8::Handle result = script->Run(); if (result.IsEmpty()) { assert(tryCatch.HasCaught()); // print errors that happened during execution TRI_ReportV8Exception(&tryCatch); return false; } LOG_TRACE("loaded java script file: '%s'", filename); return true; } //////////////////////////////////////////////////////////////////////////////// /// @brief reads all files from a directory into the current context //////////////////////////////////////////////////////////////////////////////// bool TRI_LoadJavaScriptDirectory (v8::Handle context, char const* path) { TRI_vector_string_t files; bool result; regex_t re; size_t n; size_t i; LOG_TRACE("loading java script directory: '%s'", path); files = TRI_FilesDirectory(path); regcomp(&re, "^(.*)\\.js$", REG_ICASE | REG_EXTENDED); n = TRI_SizeVectorString(&files); result = true; for (i = 0; i < n; ++i) { bool ok; char const* filename; char* full; filename = TRI_AtVectorString(&files, i); if (! regexec(&re, filename, 0, 0, 0) == 0) { continue; } full = TRI_Concatenate2File(path, filename); ok = TRI_LoadJavaScriptFile(context, full); TRI_FreeString(full); result = result && ok; } TRI_DestroyVectorString(&files); return result; } //////////////////////////////////////////////////////////////////////////////// /// @brief executes a string within a V8 context //////////////////////////////////////////////////////////////////////////////// bool TRI_ExecuteStringVocBase (v8::Handle context, v8::Handle source, v8::Handle name, bool printResult, bool reportExceptions) { TRI_v8_global_t* v8g; v8::HandleScope handleScope; v8::TryCatch tryCatch; v8g = (TRI_v8_global_t*) v8::Isolate::GetCurrent()->GetData(); v8::Handle script = v8::Script::Compile(source, name); // compilation failed, print errors that happened during compilation if (script.IsEmpty()) { if (reportExceptions) { TRI_PrintV8Exception(&tryCatch); } return false; } // compilation succeeded, run the script else { v8::Handle result = script->Run(); if (result.IsEmpty()) { assert(tryCatch.HasCaught()); // print errors that happened during execution if (reportExceptions) { TRI_PrintV8Exception(&tryCatch); } return false; } else { assert(! tryCatch.HasCaught()); // if all went well and the result wasn't undefined then print the returned value if (printResult && ! result->IsUndefined()) { v8::Handle print = v8::Handle::Cast(context->Global()->Get(v8g->PrintFuncName)); v8::Handle args[] = { result }; print->Call(print, 1, args); } return true; } } } //////////////////////////////////////////////////////////////////////////////// /// @brief stores the V8 utils functions inside the global variable //////////////////////////////////////////////////////////////////////////////// void TRI_InitV8Utils (v8::Handle context) { v8::HandleScope scope; // check the isolate v8::Isolate* isolate = v8::Isolate::GetCurrent(); TRI_v8_global_t* v8g = (TRI_v8_global_t*) isolate->GetData(); if (v8g == 0) { v8g = new TRI_v8_global_t; isolate->SetData(v8g); } // ............................................................................. // global function names // ............................................................................. if (v8g->PrintFuncName.IsEmpty()) { v8g->PrintFuncName = v8::Persistent::New(v8::String::New("print")); } // ............................................................................. // create the global functions // ............................................................................. context->Global()->Set(v8::String::New("fromJson"), v8::FunctionTemplate::New(JS_FromJson)->GetFunction(), v8::ReadOnly); context->Global()->Set(v8::String::New("time"), v8::FunctionTemplate::New(JS_Time)->GetFunction(), v8::ReadOnly); // ............................................................................. // keys // ............................................................................. v8g->DidKey = v8::Persistent::New(v8::String::New("_id")); } //////////////////////////////////////////////////////////////////////////////// /// @} //////////////////////////////////////////////////////////////////////////////// // Local Variables: // mode: outline-minor // outline-regexp: "^\\(/// @brief\\|/// @addtogroup\\|// --SECTION--\\)" // End: