summaryrefslogtreecommitdiff
path: root/couchjs/js/util.js
blob: 9cc464c3ab250621d1ae89415d68fa816b998a9c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
// 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.

var resolveModule = function(names, parent, current, path) {
  if (names.length == 0) {
    if (typeof current != "string") {
      throw ["error","invalid_require_path",
        'Must require a JavaScript string, not: '+(typeof current)];
    }
    return [current, parent, path];
  }
  // we need to traverse the path
  var n = names.shift();
  if (n == '..') {
    if (!(parent && parent.parent)) {
      throw ["error", "invalid_require_path", 'Object has no parent '+JSON.stringify(current)];
    }
    path = path.slice(0, path.lastIndexOf('/'));
    return resolveModule(names, parent.parent.parent, parent.parent, path);
  } else if (n == '.') {
    if (!parent) {
      throw ["error", "invalid_require_path", 'Object has no parent '+JSON.stringify(current)];
    }
    return resolveModule(names, parent.parent, parent, path);
  }
  if (!current[n]) {
    throw ["error", "invalid_require_path", 'Object has no property "'+n+'". '+JSON.stringify(current)];
  }
  var p = current;
  current = current[n];
  current.parent = p;
  path = path ? path + '/' + n : n;
  return resolveModule(names, p, current, path);
};

var Couch = {
  // moving this away from global so we can move to json2.js later
  toJSON : function (val) {
    return JSON.stringify(val);
  },
  compileFunction : function(source, ddoc) {    
    if (!source) throw(["error","not_found","missing function"]);
    try {
      if (sandbox) {
        if (ddoc) {
          var require = function(name, parent) {
            if (!parent) {parent = {}};
            var resolved = resolveModule(name.split('/'), parent.actual, ddoc, parent.id);
            var s = "function (module, exports, require) { " + resolved[0] + " }";
            var module = {id:resolved[2], actual:resolved[1]};
            module.exports = {};
            try {
              var func = sandbox ? evalcx(s, sandbox) : eval(s);
              func.apply(sandbox, [module, module.exports, function(name) {return require(name, module)}]);
            } catch(e) { 
              throw ["error","compilation_error","Module require('"+name+"') raised error "+e.toSource()]; 
            }
            return module.exports;
          }
          sandbox.require = require;
        }
        var functionObject = evalcx(source, sandbox);
      } else {
        var functionObject = eval(source);
      }
    } catch (err) {
      throw(["error", "compilation_error", err.toSource() + " (" + source + ")"]);
    };
    if (typeof(functionObject) == "function") {
      return functionObject;
    } else {
      throw(["error","compilation_error",
        "Expression does not eval to a function. (" + source.toSource() + ")"]);
    };
  },
  recursivelySeal : function(obj) {
    // seal() is broken in current Spidermonkey
    seal(obj);
    for (var propname in obj) {
      if (typeof doc[propname] == "object") {
        recursivelySeal(doc[propname]);
      }
    }
  }
}

// prints the object as JSON, and rescues and logs any toJSON() related errors
function respond(obj) {
  try {
    print(Couch.toJSON(obj));
  } catch(e) {
    log("Error converting object to JSON: " + e.toString());
    log("error on obj: "+ obj.toSource());
  }
};

function log(message) {
  // idea: query_server_config option for log level
  if (typeof message != "string") {
    message = Couch.toJSON(message);
  }
  respond(["log", message]);
};