• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1// Copyright 2014 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5// Routines used to validate and normalize arguments.
6// TODO(benwells): unit test this file.
7
8var JSONSchemaValidator = require('json_schema').JSONSchemaValidator;
9
10var schemaValidator = new JSONSchemaValidator();
11
12// Validate arguments.
13function validate(args, parameterSchemas) {
14  if (args.length > parameterSchemas.length)
15    throw new Error("Too many arguments.");
16  for (var i = 0; i < parameterSchemas.length; i++) {
17    if (i in args && args[i] !== null && args[i] !== undefined) {
18      schemaValidator.resetErrors();
19      schemaValidator.validate(args[i], parameterSchemas[i]);
20      if (schemaValidator.errors.length == 0)
21        continue;
22      var message = "Invalid value for argument " + (i + 1) + ". ";
23      for (var i = 0, err;
24          err = schemaValidator.errors[i]; i++) {
25        if (err.path) {
26          message += "Property '" + err.path + "': ";
27        }
28        message += err.message;
29        message = message.substring(0, message.length - 1);
30        message += ", ";
31      }
32      message = message.substring(0, message.length - 2);
33      message += ".";
34      throw new Error(message);
35    } else if (!parameterSchemas[i].optional) {
36      throw new Error("Parameter " + (i + 1) + " (" +
37          parameterSchemas[i].name + ") is required.");
38    }
39  }
40}
41
42// Generate all possible signatures for a given API function.
43function getSignatures(parameterSchemas) {
44  if (parameterSchemas.length === 0)
45    return [[]];
46  var signatures = [];
47  var remaining = getSignatures($Array.slice(parameterSchemas, 1));
48  for (var i = 0; i < remaining.length; i++)
49    $Array.push(signatures, $Array.concat([parameterSchemas[0]], remaining[i]))
50  if (parameterSchemas[0].optional)
51    return $Array.concat(signatures, remaining);
52  return signatures;
53};
54
55// Return true if arguments match a given signature's schema.
56function argumentsMatchSignature(args, candidateSignature) {
57  if (args.length != candidateSignature.length)
58    return false;
59  for (var i = 0; i < candidateSignature.length; i++) {
60    var argType =  JSONSchemaValidator.getType(args[i]);
61    if (!schemaValidator.isValidSchemaType(argType,
62        candidateSignature[i]))
63      return false;
64  }
65  return true;
66};
67
68// Finds the function signature for the given arguments.
69function resolveSignature(args, definedSignature) {
70  var candidateSignatures = getSignatures(definedSignature);
71  for (var i = 0; i < candidateSignatures.length; i++) {
72    if (argumentsMatchSignature(args, candidateSignatures[i]))
73      return candidateSignatures[i];
74  }
75  return null;
76};
77
78// Returns a string representing the defined signature of the API function.
79// Example return value for chrome.windows.getCurrent:
80// "windows.getCurrent(optional object populate, function callback)"
81function getParameterSignatureString(name, definedSignature) {
82  var getSchemaTypeString = function(schema) {
83    var schemaTypes = schemaValidator.getAllTypesForSchema(schema);
84    var typeName = schemaTypes.join(" or ") + " " + schema.name;
85    if (schema.optional)
86      return "optional " + typeName;
87    return typeName;
88  };
89  var typeNames = definedSignature.map(getSchemaTypeString);
90  return name + "(" + typeNames.join(", ") + ")";
91};
92
93// Returns a string representing a call to an API function.
94// Example return value for call: chrome.windows.get(1, callback) is:
95// "windows.get(int, function)"
96function getArgumentSignatureString(name, args) {
97  var typeNames = args.map(JSONSchemaValidator.getType);
98  return name + "(" + typeNames.join(", ") + ")";
99};
100
101// Finds the correct signature for the given arguments, then validates the
102// arguments against that signature. Returns a 'normalized' arguments list
103// where nulls are inserted where optional parameters were omitted.
104// |args| is expected to be an array.
105function normalizeArgumentsAndValidate(args, funDef) {
106  if (funDef.allowAmbiguousOptionalArguments) {
107    validate(args, funDef.definition.parameters);
108    return args;
109  }
110  var definedSignature = funDef.definition.parameters;
111  var resolvedSignature = resolveSignature(args, definedSignature);
112  if (!resolvedSignature)
113    throw new Error("Invocation of form " +
114        getArgumentSignatureString(funDef.name, args) +
115        " doesn't match definition " +
116        getParameterSignatureString(funDef.name, definedSignature));
117  validate(args, resolvedSignature);
118  var normalizedArgs = [];
119  var ai = 0;
120  for (var si = 0; si < definedSignature.length; si++) {
121    if (definedSignature[si] === resolvedSignature[ai])
122      $Array.push(normalizedArgs, args[ai++]);
123    else
124      $Array.push(normalizedArgs, null);
125  }
126  return normalizedArgs;
127};
128
129// Validates that a given schema for an API function is not ambiguous.
130function isFunctionSignatureAmbiguous(functionDef) {
131  if (functionDef.allowAmbiguousOptionalArguments)
132    return false;
133  var signaturesAmbiguous = function(signature1, signature2) {
134    if (signature1.length != signature2.length)
135      return false;
136    for (var i = 0; i < signature1.length; i++) {
137      if (!schemaValidator.checkSchemaOverlap(
138          signature1[i], signature2[i]))
139        return false;
140    }
141    return true;
142  };
143  var candidateSignatures = getSignatures(functionDef.parameters);
144  for (var i = 0; i < candidateSignatures.length; i++) {
145    for (var j = i + 1; j < candidateSignatures.length; j++) {
146      if (signaturesAmbiguous(candidateSignatures[i], candidateSignatures[j]))
147        return true;
148    }
149  }
150  return false;
151};
152
153exports.isFunctionSignatureAmbiguous = isFunctionSignatureAmbiguous;
154exports.normalizeArgumentsAndValidate = normalizeArgumentsAndValidate;
155exports.schemaValidator = schemaValidator;
156exports.validate = validate;
157