/issues/node_modules/pipette/node_modules/typ/lib/typ.js

http://story-steward.googlecode.com/ · JavaScript · 488 lines · 325 code · 93 blank · 70 comment · 56 complexity · da82edc3daccb45844747a49ce911b48 MD5 · raw file

  1. // Copyright 2012 The Obvious Corporation.
  2. /*
  3. * Typ: Useful type functions.
  4. */
  5. /*
  6. * Modules used
  7. */
  8. "use strict";
  9. var assert = require("assert");
  10. var util = require("util");
  11. /*
  12. * Variable definitions
  13. */
  14. /** extended type name */
  15. var ARRAY = "array";
  16. /** type name */
  17. var BOOLEAN = "boolean";
  18. /** extended type name */
  19. var BUFFER = "buffer";
  20. /** extended type name */
  21. var DATE = "date";
  22. /** extended type name */
  23. var ERROR = "error";
  24. /** type name */
  25. var FUNCTION = "function";
  26. /** extended type name */
  27. var INT = "int";
  28. /** extended type name */
  29. var MAP = "map";
  30. /** extended type name */
  31. var NULL = "null";
  32. /** extended type name */
  33. var NULLISH = "null or undefined";
  34. /** type name */
  35. var NUMBER = "number";
  36. /** type name */
  37. var OBJECT = "object";
  38. /** extended type name */
  39. var REGEXP = "regexp";
  40. /** type name */
  41. var STRING = "string";
  42. /** extended type name */
  43. var UINT = "uint";
  44. /** type name */
  45. var UNDEFINED = "undefined";
  46. /** the prototype of plain (map) objects */
  47. var OBJECT_PROTOTYPE = Object.getPrototypeOf({});
  48. /** the prototype of normal arrays */
  49. var ARRAY_PROTOTYPE = Object.getPrototypeOf([]);
  50. /** the prototype of normal functions */
  51. var FUNCTION_PROTOTYPE = Object.getPrototypeOf(Object);
  52. /** the prototype of date objects */
  53. var DATE_PROTOTYPE = Date.prototype;
  54. /** the base prototype of error objects */
  55. var ERROR_PROTOTYPE = Error.prototype;
  56. /** the prototype of regexp objects */
  57. var REGEXP_PROTOTYPE = RegExp.prototype;
  58. /** the function object `Object.hasOwnProperty` */
  59. var HAS_OWN_PROPERTY_FUNC = Object.hasOwnProperty;
  60. /** the base `Object.toString()` method */
  61. var objectToString = Object.prototype.toString;
  62. /*
  63. * Helper functions
  64. */
  65. /**
  66. * Call the base `Object.toString()` method. This is used as part of
  67. * type determination to help avoid cases where (through ignorance or malice)
  68. * user-defined objects try to abuse the core underlying classes.
  69. *
  70. * The tactic here (that is, how this function is used) was learned
  71. * from the Node `util` module.
  72. */
  73. function baseToString(value) {
  74. return objectToString.call(value);
  75. }
  76. function extendedTypeOf(value) {
  77. var type = typeof value;
  78. switch (type) {
  79. case BOOLEAN:
  80. case FUNCTION:
  81. case STRING:
  82. case UNDEFINED: {
  83. return type;
  84. }
  85. case NUMBER: {
  86. if (isInt(value)) {
  87. return (value > 0) ? UINT : INT;
  88. } else {
  89. return NUMBER;
  90. }
  91. }
  92. case OBJECT: {
  93. if (value === null) {
  94. return NULL;
  95. } else if (isArray(value)) {
  96. return ARRAY;
  97. } else if (isBuffer(value)) {
  98. return BUFFER;
  99. } else if (isDate(value)) {
  100. return DATE;
  101. } else if (isError(value)) {
  102. return ERROR;
  103. } else if (isRegExp(value)) {
  104. return REGEXP;
  105. } else if (isMap(value)) {
  106. return MAP;
  107. } else {
  108. return OBJECT;
  109. }
  110. }
  111. }
  112. }
  113. /**
  114. * Somewhat more helpful failure message, or just whatever the client
  115. * specified.
  116. */
  117. function failType(value, expectedTypeName, message) {
  118. var gotType = extendedTypeOf(value);
  119. if (isUndefined(message)) {
  120. message = helpfulDetails();
  121. } else {
  122. message = message.replace(/%[%s]/g, function(escape) {
  123. return (escape === '%%') ? '%' : helpfulDetails();
  124. });
  125. }
  126. assert.fail(gotType, expectedTypeName, message, "!==");
  127. function helpfulDetails() {
  128. var details = "Expected " + expectedTypeName + "; got " + gotType;
  129. switch (gotType) {
  130. case BOOLEAN:
  131. case DATE:
  132. case INT:
  133. case NUMBER:
  134. case REGEXP:
  135. case UINT: {
  136. details += " (" + value + ")";
  137. break;
  138. }
  139. case ERROR: {
  140. if (value.message) {
  141. details += " (" + value.message + ")";
  142. }
  143. break;
  144. }
  145. case FUNCTION: {
  146. if (value.name) {
  147. details += " (" + value.name + ")";
  148. }
  149. break;
  150. }
  151. }
  152. return details + ".";
  153. }
  154. }
  155. /*
  156. * Exported bindings
  157. */
  158. /**
  159. * Return whether or not the given object has the default object
  160. * prototype.
  161. */
  162. function hasDefaultPrototype(obj) {
  163. return Object.getPrototypeOf(obj) === OBJECT_PROTOTYPE;
  164. }
  165. /**
  166. * Safe version of `obj.hasOwnProperty()`.
  167. */
  168. function hasOwnProperty(obj, name) {
  169. return HAS_OWN_PROPERTY_FUNC.call(obj, name);
  170. }
  171. // For symmetry.
  172. var isArray = Array.isArray;
  173. function isBoolean(x) {
  174. return (x === true) || (x === false);
  175. }
  176. // For symmetry.
  177. var isBuffer = Buffer.isBuffer;
  178. function isDate(x) {
  179. return isObject(x) &&
  180. (Object.getPrototypeOf(x) === DATE_PROTOTYPE) &&
  181. (baseToString(x) === '[object Date]');
  182. }
  183. function isDefined(x) {
  184. // `(void 0)` is a particularly safe way to say `undefined`.
  185. return x !== (void 0);
  186. }
  187. function isUndefined(x) {
  188. return x === (void 0);
  189. }
  190. function isError(x) {
  191. return isObject(x) &&
  192. (baseToString(x) === '[object Error]') &&
  193. hasErrorProto(x);
  194. function hasErrorProto(obj) {
  195. while (obj && (obj !== OBJECT_PROTOTYPE)) {
  196. if (obj === ERROR_PROTOTYPE) {
  197. return true;
  198. }
  199. obj = Object.getPrototypeOf(obj);
  200. }
  201. return false;
  202. }
  203. }
  204. function isRegExp(x) {
  205. return isObject(x) &&
  206. (Object.getPrototypeOf(x) === REGEXP_PROTOTYPE) &&
  207. (baseToString(x) === '[object RegExp]');
  208. }
  209. function isString(x) {
  210. return (typeof x) === STRING;
  211. }
  212. /**
  213. * A "map" is an object whose prototype is the default object prototype
  214. * and which furthermore defines no enumerable dynamic properties.
  215. */
  216. function isMap(x) {
  217. if (!isObject(x)) {
  218. return false;
  219. }
  220. if (Object.getPrototypeOf(x) !== OBJECT_PROTOTYPE) {
  221. return false;
  222. }
  223. var keys = Object.keys(x);
  224. for (var i = 0; i < keys.length; i++) {
  225. var props = Object.getOwnPropertyDescriptor(x, keys[i]);
  226. if (props.get || props.set) {
  227. return false;
  228. }
  229. }
  230. return true;
  231. }
  232. function isNull(x) {
  233. return (x === null);
  234. }
  235. function isNullish(x) {
  236. return (x === null) || isUndefined(x);
  237. }
  238. function isNumber(x) {
  239. return (typeof x) === NUMBER;
  240. }
  241. function isObject(x) {
  242. var type = typeof x;
  243. return ((type === OBJECT) || (type === FUNCTION)) && (x !== null);
  244. }
  245. function isInt(x) {
  246. if (!isNumber(x) || (x !== Math.floor(x)) || !isFinite(x)) {
  247. return false;
  248. }
  249. if (x !== 0) {
  250. return true;
  251. }
  252. // Zero is special. We don't count "-0" as an int, but you can't
  253. // just test that with === because === doesn't distinguish positive
  254. // and negative zeroes. However, we can use it as a divisor to see
  255. // its effect.
  256. return (1/x) === Infinity;
  257. }
  258. function isUInt(x) {
  259. return isInt(x) && (x >= 0);
  260. }
  261. function isFunction(x) {
  262. return (typeof x) === FUNCTION;
  263. }
  264. function assertArray(x, message) {
  265. if (!isArray(x)) {
  266. failType(x, ARRAY, message);
  267. }
  268. }
  269. function assertBoolean(x, message) {
  270. if (!isBoolean(x)) {
  271. failType(x, BOOLEAN, message);
  272. }
  273. }
  274. function assertBuffer(x, message) {
  275. if (!isBuffer(x)) {
  276. failType(x, BUFFER, message);
  277. }
  278. }
  279. function assertDate(x, message) {
  280. if (!isDate(x)) {
  281. failType(x, DATE, message);
  282. }
  283. }
  284. function assertDefined(x, message) {
  285. if (!isDefined(x)) {
  286. failType(x, "anything but undefined", message);
  287. }
  288. }
  289. function assertUndefined(x, message) {
  290. if (!isUndefined(x)) {
  291. failType(x, UNDEFINED, message);
  292. }
  293. }
  294. function assertError(x, message) {
  295. if (!isError(x)) {
  296. failType(x, ERROR, message);
  297. }
  298. }
  299. function assertInt(x, message) {
  300. if (!isInt(x)) {
  301. failType(x, INT, message);
  302. }
  303. }
  304. function assertNull(x, message) {
  305. if (!isNull(x)) {
  306. failType(x, NULL, message);
  307. }
  308. }
  309. function assertNullish(x, message) {
  310. if (!isNullish(x)) {
  311. failType(x, NULLISH, message);
  312. }
  313. }
  314. function assertNumber(x, message) {
  315. if (!isNumber(x)) {
  316. failType(x, NUMBER, message);
  317. }
  318. }
  319. function assertMap(x, message) {
  320. if (!isMap(x)) {
  321. failType(x, MAP, message);
  322. }
  323. }
  324. function assertObject(x, message) {
  325. if (!isObject(x)) {
  326. failType(x, OBJECT, message);
  327. }
  328. }
  329. function assertRegExp(x, message) {
  330. if (!isRegExp(x)) {
  331. failType(x, REGEXP, message);
  332. }
  333. }
  334. function assertString(x, message) {
  335. if (!isString(x)) {
  336. failType(x, STRING, message);
  337. }
  338. }
  339. function assertUInt(x, message) {
  340. if (!isUInt(x)) {
  341. failType(x, UINT, message);
  342. }
  343. }
  344. function assertFunction(x, message) {
  345. if (!isFunction(x)) {
  346. failType(x, FUNCTION, message);
  347. }
  348. }
  349. module.exports = {
  350. BOOLEAN: BOOLEAN,
  351. FUNCTION: FUNCTION,
  352. NUMBER: NUMBER,
  353. OBJECT: OBJECT,
  354. STRING: STRING,
  355. UNDEFINED: UNDEFINED,
  356. ARRAY_PROTOTYPE: ARRAY_PROTOTYPE,
  357. FUNCTION_PROTOTYPE: FUNCTION_PROTOTYPE,
  358. OBJECT_PROTOTYPE: OBJECT_PROTOTYPE,
  359. assertArray: assertArray,
  360. assertBoolean: assertBoolean,
  361. assertBuffer: assertBuffer,
  362. assertDate: assertDate,
  363. assertDefined: assertDefined,
  364. assertError: assertError,
  365. assertFunction: assertFunction,
  366. assertInt: assertInt,
  367. assertMap: assertMap,
  368. assertNull: assertNull,
  369. assertNullish: assertNullish,
  370. assertNumber: assertNumber,
  371. assertObject: assertObject,
  372. assertRegExp: assertRegExp,
  373. assertString: assertString,
  374. assertUInt: assertUInt,
  375. assertUndefined: assertUndefined,
  376. hasDefaultPrototype: hasDefaultPrototype,
  377. hasOwnProperty: hasOwnProperty,
  378. isArray: isArray,
  379. isBoolean: isBoolean,
  380. isBuffer: isBuffer,
  381. isDate: isDate,
  382. isDefined: isDefined,
  383. isError: isError,
  384. isInt: isInt,
  385. isFunction: isFunction,
  386. isMap: isMap,
  387. isNull: isNull,
  388. isNullish: isNullish,
  389. isNumber: isNumber,
  390. isObject: isObject,
  391. isRegExp: isRegExp,
  392. isString: isString,
  393. isUInt: isUInt,
  394. isUndefined: isUndefined
  395. };