diff --git a/.circleci/config.yml b/.circleci/config.yml new file mode 100644 index 0000000..1d1176c --- /dev/null +++ b/.circleci/config.yml @@ -0,0 +1,37 @@ +version: 2 +jobs: + build: + docker: + - image: docker:dind + working_directory: /code + steps: + - checkout + - setup_remote_docker + - run: + name: Build application Docker image + command: | + docker build -t memserver . + - deploy: + name: Push application Docker image + command: | + echo $HUB_KEY | docker login -u $HUB_USERNAME --password-stdin + docker tag memserver "${HUB_USERNAME}/memserver:${CIRCLE_BRANCH}" + docker push "${HUB_USERNAME}/memserver:${CIRCLE_BRANCH}" + test: + docker: + - image: docker:dind + working_directory: ~/code + steps: + - setup_remote_docker + - run: docker pull ${HUB_USERNAME}/memserver:${CIRCLE_BRANCH} + - run: docker run -t -d --name="memserver" ${HUB_USERNAME}/memserver:${CIRCLE_BRANCH} /bin/bash + - run: docker exec -it memserver npm test + +workflows: + version: 2 + build_and_test: + jobs: + - build + - test: + requires: + - build diff --git a/.eslintrc.js b/.eslintrc.js index 60fc4d8..6455f17 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -1,32 +1,33 @@ module.exports = { - "env": { - "browser": true, - "commonjs": true, - "es6": true, - "node": true, - "mocha": true - }, - "extends": "eslint:recommended", - "parserOptions": { - "sourceType": "module" - }, - "rules": { - "indent": [ - "error", - 2 - ], - "linebreak-style": [ - "error", - "unix" - ], - "quotes": [ - "error", - "single" - ], - "semi": [ - "error", - "always" - ], - "no-console": "off" - } + root: true, + parserOptions: { + ecmaVersion: 2017, + sourceType: 'module' + }, + env: { + browser: true, + commonjs: true, + es6: true, + node: true, + }, + extends: ['eslint:recommended'], + rules: { + 'no-console': 'off', + indent: [ + 'error', + 2 + ], + 'linebreak-style': [ + 'error', + 'unix' + ], + 'quotes': [ + 'error', + 'single' + ], + 'semi': [ + 'error', + 'always' + ], + } }; diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..a84574f --- /dev/null +++ b/Dockerfile @@ -0,0 +1,17 @@ +FROM node:10 + +RUN apt-get update && \ + apt-get install -y lsof vim + +WORKDIR /code/ + +ADD package-lock.json /code/package-lock.json +ADD package.json /code/package.json + +RUN npm install + +ADD lib /code/lib +ADD test /code/test +ADD . /code/ + +ENTRYPOINT "/bin/bash" diff --git a/README.md b/README.md index de7d591..6b4cc0e 100644 --- a/README.md +++ b/README.md @@ -1,3 +1,7 @@ + + Build Status + + # What is MemServer? MemServer is an in-memory database/ORM and http mock server you can run in-browser and node environments. Extremely useful library for fast frontend tests, rapid prototyping, single-file SPA demo deployments. diff --git a/TODO b/TODO index f41a251..a8596ea 100644 --- a/TODO +++ b/TODO @@ -1,3 +1,5 @@ +- investigate silent Errors within MemServer.Server handler context +- make passthrough PASS - serializerExtraFields params - add initializer.js @@ -11,3 +13,31 @@ - non-json request/responses? - just kidding... or do I?!? - rollup -i ./lib/response.js -o response.js -f cjs - memserver glue [jsonEndpoint] + + + +export default Model({ + schema: { + username: { + type: String, + required: true, + default: 'xxx', + notNull: true, + unique: true // other min, max, + check(model, next) { + // custom checks if that returns + } + } + }, + relationships: { + emails: { // belongsTo, hasMany, hasMany through, hasOne, onReplace, maybe do FK contraint here + model: // + + } + } +}); + +// relationships declaration + +import { cast, validateLength, validateRequired, foreignKeyConstraint } from 'memserver/changeset'; +import { from } from 'memserver/query'; diff --git a/examples/photo-album/package.json b/examples/photo-album/package.json index c31fd3d..ff0a630 100644 --- a/examples/photo-album/package.json +++ b/examples/photo-album/package.json @@ -9,13 +9,9 @@ "author": "", "license": "ISC", "dependencies": { - "@std/esm": "^0.12.1", "core-object": "^3.1.5", "ember-inflector": "^2.0.1" }, - "@std/esm": { - "esm": "all" - }, "devDependencies": { "pryjs": "^1.0.3" } diff --git a/lib/model.js b/lib/model.js index 2878b3e..401728e 100644 --- a/lib/model.js +++ b/lib/model.js @@ -210,7 +210,9 @@ export default function(options) { } const objectRef = parentObject[`${underscore(relationshipName)}_id`] || - parentObject[`${underscore(relationshipName)}_uuid`]; + parentObject[`${underscore(relationshipName)}_uuid`] || + parentObject[`${underscore(targetRelationshipModel.modelName)}_id`] || + parentObject[`${underscore(targetRelationshipModel.modelName)}_uuid`]; if (objectRef && (typeof objectRef === 'number')) { return targetRelationshipModel.find(objectRef) ; diff --git a/model.js b/model.js index 77173ef..47b7a01 100644 --- a/model.js +++ b/model.js @@ -233,7 +233,9 @@ var model = function(options) { } const objectRef = parentObject[`${emberCliStringUtils.underscore(relationshipName)}_id`] || - parentObject[`${emberCliStringUtils.underscore(relationshipName)}_uuid`]; + parentObject[`${emberCliStringUtils.underscore(relationshipName)}_uuid`] || + parentObject[`${emberCliStringUtils.underscore(targetRelationshipModel.modelName)}_id`] || + parentObject[`${emberCliStringUtils.underscore(targetRelationshipModel.modelName)}_uuid`]; if (objectRef && (typeof objectRef === 'number')) { return targetRelationshipModel.find(objectRef) ; diff --git a/package-lock.json b/package-lock.json index 56e0050..3b634ab 100644 --- a/package-lock.json +++ b/package-lock.json @@ -4,1123 +4,1703 @@ "lockfileVersion": 1, "requires": true, "dependencies": { - "@sinonjs/formatio": { + "@ava/babel-plugin-throws-helper": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@ava/babel-plugin-throws-helper/-/babel-plugin-throws-helper-3.0.0.tgz", + "integrity": "sha512-mN9UolOs4WX09QkheU1ELkVy2WPnwonlO3XMdN8JF8fQqRVgVTR21xDbvEOUsbwz6Zwjq7ji9yzyjuXqDPalxg==", + "dev": true + }, + "@ava/babel-preset-stage-4": { "version": "2.0.0", - "resolved": "https://registry.npmjs.org/@sinonjs/formatio/-/formatio-2.0.0.tgz", - "integrity": "sha512-ls6CAMA6/5gG+O/IdsBcblvnd8qcO/l1TYoNeAzp3wcISOxlPXQEus0mLcdwazEkWjaBdaJ3TaxmNgCLWwvWzg==", + "resolved": "https://registry.npmjs.org/@ava/babel-preset-stage-4/-/babel-preset-stage-4-2.0.0.tgz", + "integrity": "sha512-OWqMYeTSZ16AfLx0Vn0Uj7tcu+uMRlbKmks+DVCFlln7vomVsOtst+Oz+HCussDSFGpE+30VtHAUHLy6pLDpHQ==", "dev": true, "requires": { - "samsam": "1.3.0" + "@babel/plugin-proposal-async-generator-functions": "^7.0.0", + "@babel/plugin-proposal-object-rest-spread": "^7.0.0", + "@babel/plugin-proposal-optional-catch-binding": "^7.0.0", + "@babel/plugin-transform-async-to-generator": "^7.0.0", + "@babel/plugin-transform-dotall-regex": "^7.0.0", + "@babel/plugin-transform-exponentiation-operator": "^7.0.0", + "@babel/plugin-transform-modules-commonjs": "^7.0.0" } }, - "@std/esm": { - "version": "0.12.5", - "resolved": "https://registry.npmjs.org/@std/esm/-/esm-0.12.5.tgz", - "integrity": "sha512-faJ//Sl37MVrLFM0t4+wpAgQL6LCLzP1B0PbDBb5WufMk8EJ6CY8FwwMmr8R/gPRXLEw8gQdL5PFX9DmCwGZcQ==" + "@ava/babel-preset-transform-test-files": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@ava/babel-preset-transform-test-files/-/babel-preset-transform-test-files-4.0.0.tgz", + "integrity": "sha512-V9hYHA/ZLb4I8imrrG8PT0mzgThjWWmahPV+mrQUZobVnsekBUDrf0JsfXVm4guS3binWxWn+MmQt+V81hTizA==", + "dev": true, + "requires": { + "@ava/babel-plugin-throws-helper": "^3.0.0", + "babel-plugin-espower": "^3.0.0" + } }, - "abab": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/abab/-/abab-1.0.4.tgz", - "integrity": "sha1-X6rZwsB/YN12dw9xzwJbYqY8/U4=" + "@ava/write-file-atomic": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@ava/write-file-atomic/-/write-file-atomic-2.2.0.tgz", + "integrity": "sha512-BTNB3nGbEfJT+69wuqXFr/bQH7Vr7ihx2xGOMNqPgDGhwspoZhiWumDDZNjBy7AScmqS5CELIOGtPVXESyrnDA==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.11", + "imurmurhash": "^0.1.4", + "slide": "^1.1.5" + } }, - "abstract-leveldown": { - "version": "0.12.4", - "resolved": "https://registry.npmjs.org/abstract-leveldown/-/abstract-leveldown-0.12.4.tgz", - "integrity": "sha1-KeGOYy5g5OIh1YECR4UqY9ey5BA=", + "@babel/code-frame": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.0.0.tgz", + "integrity": "sha512-OfC2uemaknXr87bdLUkWog7nYuliM9Ij5HUcajsVcMCpQrcLmtxRbVFTIqmcSkSeYRBFBRxs2FiUqFJDLdiebA==", + "dev": true, "requires": { - "xtend": "~3.0.0" + "@babel/highlight": "^7.0.0" + } + }, + "@babel/core": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.1.2.tgz", + "integrity": "sha512-IFeSSnjXdhDaoysIlev//UzHZbdEmm7D0EIH2qtse9xK7mXEZQpYjs2P00XlP1qYsYvid79p+Zgg6tz1mp6iVw==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "@babel/generator": "^7.1.2", + "@babel/helpers": "^7.1.2", + "@babel/parser": "^7.1.2", + "@babel/template": "^7.1.2", + "@babel/traverse": "^7.1.0", + "@babel/types": "^7.1.2", + "convert-source-map": "^1.1.0", + "debug": "^3.1.0", + "json5": "^0.5.0", + "lodash": "^4.17.10", + "resolve": "^1.3.2", + "semver": "^5.4.1", + "source-map": "^0.5.0" }, "dependencies": { - "xtend": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/xtend/-/xtend-3.0.0.tgz", - "integrity": "sha1-XM50B7r2Qsunvs2laBEcST9ZZlo=" + "debug": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", + "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", + "dev": true } } }, - "accepts": { - "version": "1.3.5", - "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.5.tgz", - "integrity": "sha1-63d99gEXI6OxTopywIBcjoZ0a9I=", + "@babel/generator": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.1.3.tgz", + "integrity": "sha512-ZoCZGcfIJFJuZBqxcY9OjC1KW2lWK64qrX1o4UYL3yshVhwKFYgzpWZ0vvtGMNJdTlvkw0W+HR1VnYN8q3QPFQ==", "dev": true, "requires": { - "mime-types": "~2.1.18", - "negotiator": "0.6.1" + "@babel/types": "^7.1.3", + "jsesc": "^2.5.1", + "lodash": "^4.17.10", + "source-map": "^0.5.0", + "trim-right": "^1.0.1" + }, + "dependencies": { + "jsesc": { + "version": "2.5.1", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.1.tgz", + "integrity": "sha1-5CGiqOINawgZ3yiQj3glJrlt0f4=", + "dev": true + } } }, - "acorn": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-5.7.1.tgz", - "integrity": "sha512-d+nbxBUGKg7Arpsvbnlq61mc12ek3EY8EQldM3GPAhWJ1UVxC6TDGbIvUMNU6obBX3i1+ptCIzV4vq0gFPEGVQ==" + "@babel/helper-annotate-as-pure": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.0.0.tgz", + "integrity": "sha512-3UYcJUj9kvSLbLbUIfQTqzcy5VX7GRZ/CCDrnOaZorFFM01aXp1+GJwuFGV4NDDoAS+mOUyHcO6UD/RfqOks3Q==", + "dev": true, + "requires": { + "@babel/types": "^7.0.0" + } }, - "ajv": { - "version": "5.5.2", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-5.5.2.tgz", - "integrity": "sha1-c7Xuyj+rZT49P5Qis0GtQiBdyWU=", + "@babel/helper-builder-binary-assignment-operator-visitor": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.1.0.tgz", + "integrity": "sha512-qNSR4jrmJ8M1VMM9tibvyRAHXQs2PmaksQF7c1CGJNipfe3D8p+wgNwgso/P2A2r2mdgBWAXljNWR0QRZAMW8w==", + "dev": true, "requires": { - "co": "^4.6.0", - "fast-deep-equal": "^1.0.0", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.3.0" + "@babel/helper-explode-assignable-expression": "^7.1.0", + "@babel/types": "^7.0.0" } }, - "ansi-regex": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", - "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" + "@babel/helper-explode-assignable-expression": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.1.0.tgz", + "integrity": "sha512-NRQpfHrJ1msCHtKjbzs9YcMmJZOg6mQMmGRB+hbamEdG5PNpaSm95275VD92DvJKuyl0s2sFiDmMZ+EnnvufqA==", + "dev": true, + "requires": { + "@babel/traverse": "^7.1.0", + "@babel/types": "^7.0.0" + } }, - "ansi-styles": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", - "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=" + "@babel/helper-function-name": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.1.0.tgz", + "integrity": "sha512-A95XEoCpb3TO+KZzJ4S/5uW5fNe26DjBGqf1o9ucyLyCmi1dXq/B3c8iaWTfBk3VvetUxl16e8tIrd5teOCfGw==", + "dev": true, + "requires": { + "@babel/helper-get-function-arity": "^7.0.0", + "@babel/template": "^7.1.0", + "@babel/types": "^7.0.0" + } }, - "arr-diff": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-2.0.0.tgz", - "integrity": "sha1-jzuCf5Vai9ZpaX5KQlasPOrjVs8=", + "@babel/helper-get-function-arity": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.0.0.tgz", + "integrity": "sha512-r2DbJeg4svYvt3HOS74U4eWKsUAMRH01Z1ds1zx8KNTPtpTL5JAsdFv8BNyOpVqdFhHkkRDIg5B4AsxmkjAlmQ==", + "dev": true, "requires": { - "arr-flatten": "^1.0.1" + "@babel/types": "^7.0.0" } }, - "arr-flatten": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", - "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==" + "@babel/helper-module-imports": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.0.0.tgz", + "integrity": "sha512-aP/hlLq01DWNEiDg4Jn23i+CXxW/owM4WpDLFUbpjxe4NS3BhLVZQ5i7E0ZrxuQ/vwekIeciyamgB1UIYxxM6A==", + "dev": true, + "requires": { + "@babel/types": "^7.0.0" + } }, - "array-equal": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/array-equal/-/array-equal-1.0.0.tgz", - "integrity": "sha1-jCpe8kcv2ep0KwTHenUJO6J1fJM=" + "@babel/helper-module-transforms": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.1.0.tgz", + "integrity": "sha512-0JZRd2yhawo79Rcm4w0LwSMILFmFXjugG3yqf+P/UsKsRS1mJCmMwwlHDlMg7Avr9LrvSpp4ZSULO9r8jpCzcw==", + "dev": true, + "requires": { + "@babel/helper-module-imports": "^7.0.0", + "@babel/helper-simple-access": "^7.1.0", + "@babel/helper-split-export-declaration": "^7.0.0", + "@babel/template": "^7.1.0", + "@babel/types": "^7.0.0", + "lodash": "^4.17.10" + } }, - "array-flatten": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", - "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=", + "@babel/helper-plugin-utils": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.0.0.tgz", + "integrity": "sha512-CYAOUCARwExnEixLdB6sDm2dIJ/YgEAKDM1MOeMeZu9Ld/bDgVo8aiWrXwcY7OBh+1Ea2uUcVRcxKk0GJvW7QA==", "dev": true }, - "array-unique": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.2.1.tgz", - "integrity": "sha1-odl8yvy8JiXMcPrc6zalDFiwGlM=" - }, - "asn1": { - "version": "0.2.3", - "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.3.tgz", - "integrity": "sha1-2sh4dxPJlmhJ/IGAd36+nB3fO4Y=" - }, - "asn1.js": { - "version": "4.10.1", - "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-4.10.1.tgz", - "integrity": "sha512-p32cOF5q0Zqs9uBiONKYLm6BClCoBCM5O9JfeUSlnQLBTxYdTK+pW+nXflm8UkKd2UYlEbYz5qEi0JuZR9ckSw==", + "@babel/helper-regex": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/helper-regex/-/helper-regex-7.0.0.tgz", + "integrity": "sha512-TR0/N0NDCcUIUEbqV6dCO+LptmmSQFQ7q70lfcEB4URsjD0E1HzicrwUH+ap6BAQ2jhCX9Q4UqZy4wilujWlkg==", + "dev": true, "requires": { - "bn.js": "^4.0.0", - "inherits": "^2.0.1", - "minimalistic-assert": "^1.0.0" + "lodash": "^4.17.10" } }, - "assert-plus": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", - "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" + "@babel/helper-remap-async-to-generator": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.1.0.tgz", + "integrity": "sha512-3fOK0L+Fdlg8S5al8u/hWE6vhufGSn0bN09xm2LXMy//REAF8kDCrYoOBKYmA8m5Nom+sV9LyLCwrFynA8/slg==", + "dev": true, + "requires": { + "@babel/helper-annotate-as-pure": "^7.0.0", + "@babel/helper-wrap-function": "^7.1.0", + "@babel/template": "^7.1.0", + "@babel/traverse": "^7.1.0", + "@babel/types": "^7.0.0" + } }, - "async-limiter": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.0.tgz", - "integrity": "sha512-jp/uFnooOiO+L211eZOoSyzpOITMXx1rBITauYykG3BRYPu8h0UcxsPNB04RR5vo4Tyz3+ay17tR6JVf9qzYWg==" + "@babel/helper-simple-access": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.1.0.tgz", + "integrity": "sha512-Vk+78hNjRbsiu49zAPALxTb+JUQCz1aolpd8osOF16BGnLtseD21nbHgLPGUwrXEurZgiCOUmvs3ExTu4F5x6w==", + "dev": true, + "requires": { + "@babel/template": "^7.1.0", + "@babel/types": "^7.0.0" + } }, - "asynckit": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", - "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" + "@babel/helper-split-export-declaration": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.0.0.tgz", + "integrity": "sha512-MXkOJqva62dfC0w85mEf/LucPPS/1+04nmmRMPEBUB++hiiThQ2zPtX/mEWQ3mtzCEjIJvPY8nuwxXtQeQwUag==", + "dev": true, + "requires": { + "@babel/types": "^7.0.0" + } }, - "aws-sign2": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", - "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=" + "@babel/helper-wrap-function": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.1.0.tgz", + "integrity": "sha512-R6HU3dete+rwsdAfrOzTlE9Mcpk4RjU3aX3gi9grtmugQY0u79X7eogUvfXA5sI81Mfq1cn6AgxihfN33STjJA==", + "dev": true, + "requires": { + "@babel/helper-function-name": "^7.1.0", + "@babel/template": "^7.1.0", + "@babel/traverse": "^7.1.0", + "@babel/types": "^7.0.0" + } }, - "aws4": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.7.0.tgz", - "integrity": "sha512-32NDda82rhwD9/JBCCkB+MRYDp0oSvlo2IL6rQWA10PQi7tDUM3eqMSltXmY+Oyl/7N3P3qNtAlv7X0d9bI28w==" + "@babel/helpers": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.1.2.tgz", + "integrity": "sha512-Myc3pUE8eswD73aWcartxB16K6CGmHDv9KxOmD2CeOs/FaEAQodr3VYGmlvOmog60vNQ2w8QbatuahepZwrHiA==", + "dev": true, + "requires": { + "@babel/template": "^7.1.2", + "@babel/traverse": "^7.1.0", + "@babel/types": "^7.1.2" + } }, - "babel-code-frame": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", - "integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=", + "@babel/highlight": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.0.0.tgz", + "integrity": "sha512-UFMC4ZeFC48Tpvj7C8UgLvtkaUuovQX+5xNWrsIoMG8o2z+XFKjKaN9iVmS84dPwVN00W4wPmqvYoZF3EGAsfw==", + "dev": true, "requires": { - "chalk": "^1.1.3", + "chalk": "^2.0.0", "esutils": "^2.0.2", - "js-tokens": "^3.0.2" + "js-tokens": "^4.0.0" }, "dependencies": { - "chalk": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", - "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", - "requires": { - "ansi-styles": "^2.2.1", - "escape-string-regexp": "^1.0.2", - "has-ansi": "^2.0.0", - "strip-ansi": "^3.0.0", - "supports-color": "^2.0.0" - } + "js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true } } }, - "babel-generator": { - "version": "6.26.1", - "resolved": "https://registry.npmjs.org/babel-generator/-/babel-generator-6.26.1.tgz", - "integrity": "sha512-HyfwY6ApZj7BYTcJURpM5tznulaBvyio7/0d4zFOeMPUmfxkCjHocCuoLa2SAGzBI8AREcH3eP3758F672DppA==", + "@babel/parser": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.1.3.tgz", + "integrity": "sha512-gqmspPZOMW3MIRb9HlrnbZHXI1/KHTOroBwN1NcLL6pWxzqzEKGvRTq0W/PxS45OtQGbaFikSQpkS5zbnsQm2w==", + "dev": true + }, + "@babel/plugin-proposal-async-generator-functions": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.1.0.tgz", + "integrity": "sha512-Fq803F3Jcxo20MXUSDdmZZXrPe6BWyGcWBPPNB/M7WaUYESKDeKMOGIxEzQOjGSmW/NWb6UaPZrtTB2ekhB/ew==", + "dev": true, "requires": { - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "detect-indent": "^4.0.0", - "jsesc": "^1.3.0", - "lodash": "^4.17.4", - "source-map": "^0.5.7", - "trim-right": "^1.0.1" - }, - "dependencies": { - "jsesc": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-1.3.0.tgz", - "integrity": "sha1-RsP+yMGJKxKwgz25vHYiF226s0s=" - } + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-remap-async-to-generator": "^7.1.0", + "@babel/plugin-syntax-async-generators": "^7.0.0" } }, - "babel-helper-builder-binary-assignment-operator-visitor": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-helper-builder-binary-assignment-operator-visitor/-/babel-helper-builder-binary-assignment-operator-visitor-6.24.1.tgz", - "integrity": "sha1-zORReto1b0IgvK6KAsKzRvmlZmQ=", + "@babel/plugin-proposal-object-rest-spread": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.0.0.tgz", + "integrity": "sha512-14fhfoPcNu7itSen7Py1iGN0gEm87hX/B+8nZPqkdmANyyYWYMY2pjA3r8WXbWVKMzfnSNS0xY8GVS0IjXi/iw==", + "dev": true, "requires": { - "babel-helper-explode-assignable-expression": "^6.24.1", - "babel-runtime": "^6.22.0", - "babel-types": "^6.24.1" + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/plugin-syntax-object-rest-spread": "^7.0.0" } }, - "babel-helper-call-delegate": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-helper-call-delegate/-/babel-helper-call-delegate-6.24.1.tgz", - "integrity": "sha1-7Oaqzdx25Bw0YfiL/Fdb0Nqi340=", + "@babel/plugin-proposal-optional-catch-binding": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.0.0.tgz", + "integrity": "sha512-JPqAvLG1s13B/AuoBjdBYvn38RqW6n1TzrQO839/sIpqLpbnXKacsAgpZHzLD83Sm8SDXMkkrAvEnJ25+0yIpw==", + "dev": true, "requires": { - "babel-helper-hoist-variables": "^6.24.1", - "babel-runtime": "^6.22.0", - "babel-traverse": "^6.24.1", - "babel-types": "^6.24.1" - }, - "dependencies": { - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", - "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" - } - }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "requires": { - "loose-envify": "^1.0.0" - } - } + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/plugin-syntax-optional-catch-binding": "^7.0.0" } }, - "babel-helper-define-map": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-helper-define-map/-/babel-helper-define-map-6.26.0.tgz", - "integrity": "sha1-pfVtq0GiX5fstJjH66ypgZ+Vvl8=", + "@babel/plugin-syntax-async-generators": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.0.0.tgz", + "integrity": "sha512-im7ged00ddGKAjcZgewXmp1vxSZQQywuQXe2B1A7kajjZmDeY/ekMPmWr9zJgveSaQH0k7BcGrojQhcK06l0zA==", + "dev": true, "requires": { - "babel-helper-function-name": "^6.24.1", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "lodash": "^4.17.4" + "@babel/helper-plugin-utils": "^7.0.0" } }, - "babel-helper-explode-assignable-expression": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-helper-explode-assignable-expression/-/babel-helper-explode-assignable-expression-6.24.1.tgz", - "integrity": "sha1-8luCz33BBDPFX3BZLVdGQArCLKo=", + "@babel/plugin-syntax-object-rest-spread": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.0.0.tgz", + "integrity": "sha512-5A0n4p6bIiVe5OvQPxBnesezsgFJdHhSs3uFSvaPdMqtsovajLZ+G2vZyvNe10EzJBWWo3AcHGKhAFUxqwp2dw==", + "dev": true, "requires": { - "babel-runtime": "^6.22.0", - "babel-traverse": "^6.24.1", - "babel-types": "^6.24.1" + "@babel/helper-plugin-utils": "^7.0.0" + } + }, + "@babel/plugin-syntax-optional-catch-binding": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.0.0.tgz", + "integrity": "sha512-Wc+HVvwjcq5qBg1w5RG9o9RVzmCaAg/Vp0erHCKpAYV8La6I94o4GQAmFYNmkzoMO6gzoOSulpKeSSz6mPEoZw==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.0.0" + } + }, + "@babel/plugin-transform-async-to-generator": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.1.0.tgz", + "integrity": "sha512-rNmcmoQ78IrvNCIt/R9U+cixUHeYAzgusTFgIAv+wQb9HJU4szhpDD6e5GCACmj/JP5KxuCwM96bX3L9v4ZN/g==", + "dev": true, + "requires": { + "@babel/helper-module-imports": "^7.0.0", + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-remap-async-to-generator": "^7.1.0" + } + }, + "@babel/plugin-transform-dotall-regex": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.0.0.tgz", + "integrity": "sha512-00THs8eJxOJUFVx1w8i1MBF4XH4PsAjKjQ1eqN/uCH3YKwP21GCKfrn6YZFZswbOk9+0cw1zGQPHVc1KBlSxig==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-regex": "^7.0.0", + "regexpu-core": "^4.1.3" }, "dependencies": { - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", + "regexpu-core": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-4.2.0.tgz", + "integrity": "sha512-Z835VSnJJ46CNBttalHD/dB+Sj2ezmY6Xp38npwU87peK6mqOzOpV8eYktdkLTEkzzD+JsTcxd84ozd8I14+rw==", + "dev": true, "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" + "regenerate": "^1.4.0", + "regenerate-unicode-properties": "^7.0.0", + "regjsgen": "^0.4.0", + "regjsparser": "^0.3.0", + "unicode-match-property-ecmascript": "^1.0.4", + "unicode-match-property-value-ecmascript": "^1.0.2" } }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "regjsgen": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/regjsgen/-/regjsgen-0.4.0.tgz", + "integrity": "sha512-X51Lte1gCYUdlwhF28+2YMO0U6WeN0GLpgpA7LK7mbdDnkQYiwvEpmpe0F/cv5L14EbxgrdayAG3JETBv0dbXA==", + "dev": true + }, + "regjsparser": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.3.0.tgz", + "integrity": "sha512-zza72oZBBHzt64G7DxdqrOo/30bhHkwMUoT0WqfGu98XLd7N+1tsy5MJ96Bk4MD0y74n629RhmrGW6XlnLLwCA==", + "dev": true, "requires": { - "loose-envify": "^1.0.0" + "jsesc": "~0.5.0" } } } }, - "babel-helper-function-name": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-helper-function-name/-/babel-helper-function-name-6.24.1.tgz", - "integrity": "sha1-00dbjAPtmCQqJbSDUasYOZ01gKk=", + "@babel/plugin-transform-exponentiation-operator": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.1.0.tgz", + "integrity": "sha512-uZt9kD1Pp/JubkukOGQml9tqAeI8NkE98oZnHZ2qHRElmeKCodbTZgOEUtujSCSLhHSBWbzNiFSDIMC4/RBTLQ==", + "dev": true, "requires": { - "babel-helper-get-function-arity": "^6.24.1", - "babel-runtime": "^6.22.0", - "babel-template": "^6.24.1", - "babel-traverse": "^6.24.1", - "babel-types": "^6.24.1" + "@babel/helper-builder-binary-assignment-operator-visitor": "^7.1.0", + "@babel/helper-plugin-utils": "^7.0.0" + } + }, + "@babel/plugin-transform-modules-commonjs": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.1.0.tgz", + "integrity": "sha512-wtNwtMjn1XGwM0AXPspQgvmE6msSJP15CX2RVfpTSTNPLhKhaOjaIfBaVfj4iUZ/VrFSodcFedwtPg/NxwQlPA==", + "dev": true, + "requires": { + "@babel/helper-module-transforms": "^7.1.0", + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/helper-simple-access": "^7.1.0" + } + }, + "@babel/template": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.1.2.tgz", + "integrity": "sha512-SY1MmplssORfFiLDcOETrW7fCLl+PavlwMh92rrGcikQaRq4iWPVH0MpwPpY3etVMx6RnDjXtr6VZYr/IbP/Ag==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "@babel/parser": "^7.1.2", + "@babel/types": "^7.1.2" + } + }, + "@babel/traverse": { + "version": "7.1.4", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.1.4.tgz", + "integrity": "sha512-my9mdrAIGdDiSVBuMjpn/oXYpva0/EZwWL3sm3Wcy/AVWO2eXnsoZruOT9jOGNRXU8KbCIu5zsKnXcAJ6PcV6Q==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "@babel/generator": "^7.1.3", + "@babel/helper-function-name": "^7.1.0", + "@babel/helper-split-export-declaration": "^7.0.0", + "@babel/parser": "^7.1.3", + "@babel/types": "^7.1.3", + "debug": "^3.1.0", + "globals": "^11.1.0", + "lodash": "^4.17.10" }, "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", + "debug": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", + "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", + "dev": true, "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" + "ms": "^2.1.1" } }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", - "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" - } + "globals": { + "version": "11.8.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.8.0.tgz", + "integrity": "sha512-io6LkyPVuzCHBSQV9fmOwxZkUk6nIaGmxheLDgmuFv89j0fm2aqDbIXKAGfzCMHqz3HLF2Zf8WSG6VqMh2qFmA==", + "dev": true }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "requires": { - "loose-envify": "^1.0.0" - } + "ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", + "dev": true } } }, - "babel-helper-get-function-arity": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-helper-get-function-arity/-/babel-helper-get-function-arity-6.24.1.tgz", - "integrity": "sha1-j3eCqpNAfEHTqlCQj4mwMbG2hT0=", + "@babel/types": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.1.3.tgz", + "integrity": "sha512-RpPOVfK+yatXyn8n4PB1NW6k9qjinrXrRR8ugBN8fD6hCy5RXI6PSbVqpOJBO9oSaY7Nom4ohj35feb0UR9hSA==", + "dev": true, "requires": { - "babel-runtime": "^6.22.0", - "babel-types": "^6.24.1" + "esutils": "^2.0.2", + "lodash": "^4.17.10", + "to-fast-properties": "^2.0.0" + }, + "dependencies": { + "to-fast-properties": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", + "integrity": "sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4=", + "dev": true + } } }, - "babel-helper-hoist-variables": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-helper-hoist-variables/-/babel-helper-hoist-variables-6.24.1.tgz", - "integrity": "sha1-HssnaJydJVE+rbyZFKc/VAi+enY=", + "@concordance/react": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@concordance/react/-/react-2.0.0.tgz", + "integrity": "sha512-huLSkUuM2/P+U0uy2WwlKuixMsTODD8p4JVQBI4VKeopkiN0C7M3N9XYVawb4M+4spN5RrO/eLhk7KoQX6nsfA==", + "dev": true, "requires": { - "babel-runtime": "^6.22.0", - "babel-types": "^6.24.1" + "arrify": "^1.0.1" } }, - "babel-helper-optimise-call-expression": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-helper-optimise-call-expression/-/babel-helper-optimise-call-expression-6.24.1.tgz", - "integrity": "sha1-96E0J7qfc/j0+pk8VKl4gtEkQlc=", + "@sinonjs/formatio": { + "version": "2.0.0", + "resolved": "http://registry.npmjs.org/@sinonjs/formatio/-/formatio-2.0.0.tgz", + "integrity": "sha512-ls6CAMA6/5gG+O/IdsBcblvnd8qcO/l1TYoNeAzp3wcISOxlPXQEus0mLcdwazEkWjaBdaJ3TaxmNgCLWwvWzg==", + "dev": true, "requires": { - "babel-runtime": "^6.22.0", - "babel-types": "^6.24.1" + "samsam": "1.3.0" } }, - "babel-helper-regex": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-helper-regex/-/babel-helper-regex-6.26.0.tgz", - "integrity": "sha1-MlxZ+QL4LyS3T6zu0DY5VPZJXnI=", + "@sinonjs/samsam": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@sinonjs/samsam/-/samsam-2.1.0.tgz", + "integrity": "sha512-5x2kFgJYupaF1ns/RmharQ90lQkd2ELS8A9X0ymkAAdemYHGtI2KiUHG8nX2WU0T1qgnOU5YMqnBM2V7NUanNw==", + "dev": true, "requires": { - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "lodash": "^4.17.4" + "array-from": "^2.1.1" } }, - "babel-helper-remap-async-to-generator": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-helper-remap-async-to-generator/-/babel-helper-remap-async-to-generator-6.24.1.tgz", - "integrity": "sha1-XsWBgnrXI/7N04HxySg5BnbkVRs=", + "abab": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.0.tgz", + "integrity": "sha512-sY5AXXVZv4Y1VACTtR11UJCPHHudgY5i26Qj5TypE6DKlIApbwb5uqhXcJ5UUGbvZNRh7EeIoW+LrJumBsKp7w==" + }, + "abstract-leveldown": { + "version": "0.12.4", + "resolved": "http://registry.npmjs.org/abstract-leveldown/-/abstract-leveldown-0.12.4.tgz", + "integrity": "sha1-KeGOYy5g5OIh1YECR4UqY9ey5BA=", "requires": { - "babel-helper-function-name": "^6.24.1", - "babel-runtime": "^6.22.0", - "babel-template": "^6.24.1", - "babel-traverse": "^6.24.1", - "babel-types": "^6.24.1" + "xtend": "~3.0.0" + }, + "dependencies": { + "xtend": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-3.0.0.tgz", + "integrity": "sha1-XM50B7r2Qsunvs2laBEcST9ZZlo=" + } + } + }, + "accepts": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.5.tgz", + "integrity": "sha1-63d99gEXI6OxTopywIBcjoZ0a9I=", + "dev": true, + "requires": { + "mime-types": "~2.1.18", + "negotiator": "0.6.1" + } + }, + "acorn": { + "version": "5.7.3", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-5.7.3.tgz", + "integrity": "sha512-T/zvzYRfbVojPWahDsE5evJdHb3oJoQfFbsrKM7w5Zcs++Tr257tia3BmMP8XYVjp1S9RZXQMh7gao96BlqZOw==" + }, + "acorn-globals": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-4.3.0.tgz", + "integrity": "sha512-hMtHj3s5RnuhvHPowpBYvJVj3rAar82JiDQHvGs1zO0l10ocX/xEdBShNHTJaboucJUsScghp74pH3s7EnHHQw==", + "requires": { + "acorn": "^6.0.1", + "acorn-walk": "^6.0.1" + }, + "dependencies": { + "acorn": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.0.2.tgz", + "integrity": "sha512-GXmKIvbrN3TV7aVqAzVFaMW8F8wzVX7voEBRO3bDA64+EX37YSayggRJP5Xig6HYHBkWKpFg9W5gg6orklubhg==" + } + } + }, + "acorn-walk": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-6.1.0.tgz", + "integrity": "sha512-ugTb7Lq7u4GfWSqqpwE0bGyoBZNMTok/zDBXxfEG0QM50jNlGhIWjRC1pPN7bvV1anhF+bs+/gNcRw+o55Evbg==" + }, + "ajv": { + "version": "5.5.2", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-5.5.2.tgz", + "integrity": "sha1-c7Xuyj+rZT49P5Qis0GtQiBdyWU=", + "requires": { + "co": "^4.6.0", + "fast-deep-equal": "^1.0.0", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.3.0" + } + }, + "ansi-align": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-2.0.0.tgz", + "integrity": "sha1-w2rsy6VjuJzrVW82kPCx2eNUf38=", + "dev": true, + "requires": { + "string-width": "^2.0.0" + } + }, + "ansi-escapes": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.1.0.tgz", + "integrity": "sha512-UgAb8H9D41AQnu/PbWlCofQVcnV4Gs2bBJi9eZPxfU/hgglFh3SMDMENRIqdr7H6XFnXdoknctFByVsCOotTVw==", + "dev": true + }, + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" + }, + "ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=" + }, + "anymatch": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", + "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", + "dev": true, + "requires": { + "micromatch": "^3.1.4", + "normalize-path": "^2.1.1" }, "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", + "arr-diff": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", + "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", + "dev": true + }, + "array-unique": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", + "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", + "dev": true + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } } }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", + "expand-brackets": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", + "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", + "dev": true, "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" + "debug": "^2.3.3", + "define-property": "^0.2.5", + "extend-shallow": "^2.0.1", + "posix-character-classes": "^0.1.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "is-accessor-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", + "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "is-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "is-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", + "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + } + }, + "kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "dev": true + } } }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "extglob": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", + "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", + "dev": true, "requires": { - "loose-envify": "^1.0.0" + "array-unique": "^0.3.2", + "define-property": "^1.0.0", + "expand-brackets": "^2.1.4", + "extend-shallow": "^2.0.1", + "fragment-cache": "^0.2.1", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } } - } - } - }, - "babel-helper-replace-supers": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-helper-replace-supers/-/babel-helper-replace-supers-6.24.1.tgz", - "integrity": "sha1-v22/5Dk40XNpohPKiov3S2qQqxo=", - "requires": { - "babel-helper-optimise-call-expression": "^6.24.1", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.22.0", - "babel-template": "^6.24.1", - "babel-traverse": "^6.24.1", - "babel-types": "^6.24.1" - }, - "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } } }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" + "kind-of": "^6.0.0" } }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, "requires": { - "loose-envify": "^1.0.0" + "kind-of": "^6.0.0" } - } - } - }, - "babel-helpers": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-helpers/-/babel-helpers-6.24.1.tgz", - "integrity": "sha1-NHHenK7DiOXIUOWX5Yom3fN2ArI=", - "requires": { - "babel-runtime": "^6.22.0", - "babel-template": "^6.24.1" - }, - "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" } }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } } }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "kind-of": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz", + "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==", + "dev": true + }, + "micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", + "dev": true, "requires": { - "loose-envify": "^1.0.0" + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" } } } }, - "babel-messages": { - "version": "6.23.0", - "resolved": "https://registry.npmjs.org/babel-messages/-/babel-messages-6.23.0.tgz", - "integrity": "sha1-8830cDhYA1sqKVHG7F7fbGLyYw4=", + "argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, "requires": { - "babel-runtime": "^6.22.0" + "sprintf-js": "~1.0.2" } }, - "babel-plugin-check-es2015-constants": { - "version": "6.22.0", - "resolved": "https://registry.npmjs.org/babel-plugin-check-es2015-constants/-/babel-plugin-check-es2015-constants-6.22.0.tgz", - "integrity": "sha1-NRV7EBQm/S/9PaP3XH0ekYNbv4o=", + "arr-diff": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-2.0.0.tgz", + "integrity": "sha1-jzuCf5Vai9ZpaX5KQlasPOrjVs8=", "requires": { - "babel-runtime": "^6.22.0" + "arr-flatten": "^1.0.1" } }, - "babel-plugin-syntax-async-functions": { - "version": "6.13.0", - "resolved": "https://registry.npmjs.org/babel-plugin-syntax-async-functions/-/babel-plugin-syntax-async-functions-6.13.0.tgz", - "integrity": "sha1-ytnK0RkbWtY0vzCuCHI5HgZHvpU=" + "arr-flatten": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", + "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==" }, - "babel-plugin-syntax-exponentiation-operator": { - "version": "6.13.0", - "resolved": "https://registry.npmjs.org/babel-plugin-syntax-exponentiation-operator/-/babel-plugin-syntax-exponentiation-operator-6.13.0.tgz", - "integrity": "sha1-nufoM3KQ2pUoggGmpX9BcDF4MN4=" + "arr-union": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", + "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=", + "dev": true }, - "babel-plugin-syntax-trailing-function-commas": { - "version": "6.22.0", - "resolved": "https://registry.npmjs.org/babel-plugin-syntax-trailing-function-commas/-/babel-plugin-syntax-trailing-function-commas-6.22.0.tgz", - "integrity": "sha1-ugNgk3+NBuQBgKQ/4NVhb/9TLPM=" + "array-differ": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/array-differ/-/array-differ-1.0.0.tgz", + "integrity": "sha1-7/UuN1gknTO+QCuLuOVkuytdQDE=", + "dev": true }, - "babel-plugin-transform-async-to-generator": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-async-to-generator/-/babel-plugin-transform-async-to-generator-6.24.1.tgz", - "integrity": "sha1-ZTbjeK/2yx1VF6wOQOs+n8jQh2E=", + "array-equal": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/array-equal/-/array-equal-1.0.0.tgz", + "integrity": "sha1-jCpe8kcv2ep0KwTHenUJO6J1fJM=" + }, + "array-find-index": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-find-index/-/array-find-index-1.0.2.tgz", + "integrity": "sha1-3wEKoSh+Fku9pvlyOwqWoexBh6E=", + "dev": true + }, + "array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=", + "dev": true + }, + "array-from": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/array-from/-/array-from-2.1.1.tgz", + "integrity": "sha1-z+nYwmYoudxa7MYqn12PHzUsEZU=", + "dev": true + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, "requires": { - "babel-helper-remap-async-to-generator": "^6.24.1", - "babel-plugin-syntax-async-functions": "^6.8.0", - "babel-runtime": "^6.22.0" + "array-uniq": "^1.0.1" + }, + "dependencies": { + "array-uniq": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", + "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=", + "dev": true + } } }, - "babel-plugin-transform-es2015-arrow-functions": { - "version": "6.22.0", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-arrow-functions/-/babel-plugin-transform-es2015-arrow-functions-6.22.0.tgz", - "integrity": "sha1-RSaSy3EdX3ncf4XkQM5BufJE0iE=", + "array-uniq": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-2.0.0.tgz", + "integrity": "sha512-O3QZEr+3wDj7otzF7PjNGs6CA3qmYMLvt5xGkjY/V0VxS+ovvqVo/5wKM/OVOAyuX4DTh9H31zE/yKtO66hTkg==", + "dev": true + }, + "array-unique": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.2.1.tgz", + "integrity": "sha1-odl8yvy8JiXMcPrc6zalDFiwGlM=" + }, + "arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", + "dev": true + }, + "asn1": { + "version": "0.2.4", + "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", + "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", "requires": { - "babel-runtime": "^6.22.0" + "safer-buffer": "~2.1.0" } }, - "babel-plugin-transform-es2015-block-scoped-functions": { - "version": "6.22.0", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-block-scoped-functions/-/babel-plugin-transform-es2015-block-scoped-functions-6.22.0.tgz", - "integrity": "sha1-u8UbSflk1wy42OC5ToICRs46YUE=", + "asn1.js": { + "version": "4.10.1", + "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-4.10.1.tgz", + "integrity": "sha512-p32cOF5q0Zqs9uBiONKYLm6BClCoBCM5O9JfeUSlnQLBTxYdTK+pW+nXflm8UkKd2UYlEbYz5qEi0JuZR9ckSw==", "requires": { - "babel-runtime": "^6.22.0" + "bn.js": "^4.0.0", + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0" } }, - "babel-plugin-transform-es2015-block-scoping": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-block-scoping/-/babel-plugin-transform-es2015-block-scoping-6.26.0.tgz", - "integrity": "sha1-1w9SmcEwjQXBL0Y4E7CgnnOxiV8=", + "assert-plus": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", + "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" + }, + "assign-symbols": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", + "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=", + "dev": true + }, + "async-each": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.1.tgz", + "integrity": "sha1-GdOGodntxufByF04iu28xW0zYC0=", + "dev": true + }, + "async-limiter": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.0.tgz", + "integrity": "sha512-jp/uFnooOiO+L211eZOoSyzpOITMXx1rBITauYykG3BRYPu8h0UcxsPNB04RR5vo4Tyz3+ay17tR6JVf9qzYWg==" + }, + "asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" + }, + "atob": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", + "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", + "dev": true + }, + "ava": { + "version": "1.0.0-rc.1", + "resolved": "https://registry.npmjs.org/ava/-/ava-1.0.0-rc.1.tgz", + "integrity": "sha512-tyHTMP7frwiewEvRUON4ikDneTg56981bnXzJe882sgL0BVxvUPH8jdiKKOtzZPZncGaTA/YXyDzYhjV71YRCg==", + "dev": true, "requires": { - "babel-runtime": "^6.26.0", - "babel-template": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "lodash": "^4.17.4" + "@ava/babel-preset-stage-4": "^2.0.0", + "@ava/babel-preset-transform-test-files": "^4.0.0", + "@ava/write-file-atomic": "^2.2.0", + "@babel/core": "^7.1.2", + "@babel/generator": "^7.1.3", + "@babel/plugin-syntax-async-generators": "^7.0.0", + "@babel/plugin-syntax-object-rest-spread": "^7.0.0", + "@babel/plugin-syntax-optional-catch-binding": "^7.0.0", + "@concordance/react": "^2.0.0", + "ansi-escapes": "^3.1.0", + "ansi-styles": "^3.2.1", + "arr-flatten": "^1.1.0", + "array-union": "^1.0.1", + "array-uniq": "^2.0.0", + "arrify": "^1.0.0", + "bluebird": "^3.5.2", + "chalk": "^2.4.1", + "chokidar": "^2.0.4", + "chunkd": "^1.0.0", + "ci-parallel-vars": "^1.0.0", + "clean-stack": "^1.1.1", + "clean-yaml-object": "^0.1.0", + "cli-cursor": "^2.1.0", + "cli-truncate": "^1.1.0", + "code-excerpt": "^2.1.1", + "common-path-prefix": "^1.0.0", + "concordance": "^4.0.0", + "convert-source-map": "^1.6.0", + "currently-unhandled": "^0.4.1", + "debug": "^4.1.0", + "del": "^3.0.0", + "dot-prop": "^4.2.0", + "emittery": "^0.4.1", + "empower-core": "^1.2.0", + "equal-length": "^1.0.0", + "escape-string-regexp": "^1.0.5", + "esm": "^3.0.84", + "figures": "^2.0.0", + "find-up": "^3.0.0", + "get-port": "^4.0.0", + "globby": "^7.1.1", + "ignore-by-default": "^1.0.0", + "import-local": "^2.0.0", + "indent-string": "^3.2.0", + "is-ci": "^1.2.1", + "is-error": "^2.2.1", + "is-observable": "^1.1.0", + "is-plain-object": "^2.0.4", + "is-promise": "^2.1.0", + "lodash.clone": "^4.5.0", + "lodash.clonedeep": "^4.5.0", + "lodash.clonedeepwith": "^4.5.0", + "lodash.debounce": "^4.0.3", + "lodash.difference": "^4.3.0", + "lodash.flatten": "^4.2.0", + "loud-rejection": "^1.2.0", + "make-dir": "^1.3.0", + "matcher": "^1.1.1", + "md5-hex": "^2.0.0", + "meow": "^5.0.0", + "ms": "^2.1.1", + "multimatch": "^2.1.0", + "observable-to-promise": "^0.5.0", + "ora": "^3.0.0", + "package-hash": "^2.0.0", + "pkg-conf": "^2.1.0", + "plur": "^3.0.1", + "pretty-ms": "^4.0.0", + "require-precompiled": "^0.1.0", + "resolve-cwd": "^2.0.0", + "slash": "^2.0.0", + "source-map-support": "^0.5.9", + "stack-utils": "^1.0.1", + "strip-ansi": "^5.0.0", + "strip-bom-buf": "^1.0.0", + "supertap": "^1.0.0", + "supports-color": "^5.5.0", + "trim-off-newlines": "^1.0.1", + "trim-right": "^1.0.1", + "unique-temp-dir": "^1.0.0", + "update-notifier": "^2.5.0" }, "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", - "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" - } + "ansi-regex": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.0.0.tgz", + "integrity": "sha512-iB5Dda8t/UqpPI/IjsejXu5jOGDrzn41wJyljwPH65VCIbk6+1BzFIMJGFwTNrYXT1CrD+B4l19U7awiQ8rk7w==", + "dev": true }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" + "color-convert": "^1.9.0" } }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "debug": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.0.tgz", + "integrity": "sha512-heNPJUJIqC+xB6ayLAMHaIrmN9HKa7aQO8MGqKpvCA+uJYVcvR6l5kgdrhRuwPFHU7P5/A1w0BjByPHwpfTDKg==", + "dev": true, "requires": { - "loose-envify": "^1.0.0" + "ms": "^2.1.1" } - } - } - }, - "babel-plugin-transform-es2015-classes": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-classes/-/babel-plugin-transform-es2015-classes-6.24.1.tgz", - "integrity": "sha1-WkxYpQyclGHlZLSyo7+ryXolhNs=", - "requires": { - "babel-helper-define-map": "^6.24.1", - "babel-helper-function-name": "^6.24.1", - "babel-helper-optimise-call-expression": "^6.24.1", - "babel-helper-replace-supers": "^6.24.1", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.22.0", - "babel-template": "^6.24.1", - "babel-traverse": "^6.24.1", - "babel-types": "^6.24.1" - }, - "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", + }, + "ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", + "dev": true + }, + "slash": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-2.0.0.tgz", + "integrity": "sha512-ZYKh3Wh2z1PpEXWr0MpSBZ0V6mZHAQfYevttO11c51CaWjGTaadiKZ+wVt1PbMlDV5qhMFslpZCemhwOK7C89A==", + "dev": true + }, + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true + }, + "source-map-support": { + "version": "0.5.9", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.9.tgz", + "integrity": "sha512-gR6Rw4MvUlYy83vP0vxoVNzM6t8MUXqNuRsuBmBHQDu1Fh6X015FrLdgoDKcNdkwGubozq0P4N0Q37UyFVr1EA==", + "dev": true, "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" } }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", + "strip-ansi": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.0.0.tgz", + "integrity": "sha512-Uu7gQyZI7J7gn5qLn1Np3G9vcYGTVqB+lFTytnDJv83dd8T22aGH451P3jueT2/QemInJDfxHB5Tde5OzgG1Ow==", + "dev": true, "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" + "ansi-regex": "^4.0.0" } }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, "requires": { - "loose-envify": "^1.0.0" + "has-flag": "^3.0.0" } } } }, - "babel-plugin-transform-es2015-computed-properties": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-computed-properties/-/babel-plugin-transform-es2015-computed-properties-6.24.1.tgz", - "integrity": "sha1-b+Ko0WiV1WNPTNmZttNICjCBWbM=", + "aws-sign2": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", + "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=" + }, + "aws4": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.8.0.tgz", + "integrity": "sha512-ReZxvNHIOv88FlT7rxcXIIC0fPt4KZqZbOlivyWtXLt8ESx84zd3kMC6iK5jVeS2qt+g7ftS7ye4fi06X5rtRQ==" + }, + "babel-code-frame": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", + "integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=", "requires": { - "babel-runtime": "^6.22.0", - "babel-template": "^6.24.1" + "chalk": "^1.1.3", + "esutils": "^2.0.2", + "js-tokens": "^3.0.2" }, "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", - "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" - } - }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", - "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" - } - }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "chalk": { + "version": "1.1.3", + "resolved": "http://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", "requires": { - "loose-envify": "^1.0.0" + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" } } } }, - "babel-plugin-transform-es2015-destructuring": { - "version": "6.23.0", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-destructuring/-/babel-plugin-transform-es2015-destructuring-6.23.0.tgz", - "integrity": "sha1-mXux8auWf2gtKwh2/jWNYOdlxW0=", - "requires": { - "babel-runtime": "^6.22.0" - } - }, - "babel-plugin-transform-es2015-duplicate-keys": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-duplicate-keys/-/babel-plugin-transform-es2015-duplicate-keys-6.24.1.tgz", - "integrity": "sha1-c+s9MQypaePvnskcU3QabxV2Qj4=", + "babel-core": { + "version": "6.26.3", + "resolved": "https://registry.npmjs.org/babel-core/-/babel-core-6.26.3.tgz", + "integrity": "sha512-6jyFLuDmeidKmUEb3NM+/yawG0M2bDZ9Z1qbZP59cyHLz8kYGKYwpJP0UwUKKUiTRNvxfLesJnTedqczP7cTDA==", "requires": { - "babel-runtime": "^6.22.0", - "babel-types": "^6.24.1" + "babel-code-frame": "^6.26.0", + "babel-generator": "^6.26.0", + "babel-helpers": "^6.24.1", + "babel-messages": "^6.23.0", + "babel-register": "^6.26.0", + "babel-runtime": "^6.26.0", + "babel-template": "^6.26.0", + "babel-traverse": "^6.26.0", + "babel-types": "^6.26.0", + "babylon": "^6.18.0", + "convert-source-map": "^1.5.1", + "debug": "^2.6.9", + "json5": "^0.5.1", + "lodash": "^4.17.4", + "minimatch": "^3.0.4", + "path-is-absolute": "^1.0.1", + "private": "^0.1.8", + "slash": "^1.0.0", + "source-map": "^0.5.7" } }, - "babel-plugin-transform-es2015-for-of": { - "version": "6.23.0", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-for-of/-/babel-plugin-transform-es2015-for-of-6.23.0.tgz", - "integrity": "sha1-9HyVsrYT3x0+zC/bdXNiPHUkhpE=", + "babel-generator": { + "version": "6.26.1", + "resolved": "https://registry.npmjs.org/babel-generator/-/babel-generator-6.26.1.tgz", + "integrity": "sha512-HyfwY6ApZj7BYTcJURpM5tznulaBvyio7/0d4zFOeMPUmfxkCjHocCuoLa2SAGzBI8AREcH3eP3758F672DppA==", "requires": { - "babel-runtime": "^6.22.0" + "babel-messages": "^6.23.0", + "babel-runtime": "^6.26.0", + "babel-types": "^6.26.0", + "detect-indent": "^4.0.0", + "jsesc": "^1.3.0", + "lodash": "^4.17.4", + "source-map": "^0.5.7", + "trim-right": "^1.0.1" + }, + "dependencies": { + "jsesc": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-1.3.0.tgz", + "integrity": "sha1-RsP+yMGJKxKwgz25vHYiF226s0s=" + } } }, - "babel-plugin-transform-es2015-function-name": { + "babel-helper-builder-binary-assignment-operator-visitor": { "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-function-name/-/babel-plugin-transform-es2015-function-name-6.24.1.tgz", - "integrity": "sha1-g0yJhTvDaxrw86TF26qU/Y6sqos=", + "resolved": "https://registry.npmjs.org/babel-helper-builder-binary-assignment-operator-visitor/-/babel-helper-builder-binary-assignment-operator-visitor-6.24.1.tgz", + "integrity": "sha1-zORReto1b0IgvK6KAsKzRvmlZmQ=", "requires": { - "babel-helper-function-name": "^6.24.1", + "babel-helper-explode-assignable-expression": "^6.24.1", "babel-runtime": "^6.22.0", "babel-types": "^6.24.1" } }, - "babel-plugin-transform-es2015-literals": { - "version": "6.22.0", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-literals/-/babel-plugin-transform-es2015-literals-6.22.0.tgz", - "integrity": "sha1-T1SgLWzWbPkVKAAZox0xklN3yi4=", - "requires": { - "babel-runtime": "^6.22.0" - } - }, - "babel-plugin-transform-es2015-modules-amd": { + "babel-helper-call-delegate": { "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-modules-amd/-/babel-plugin-transform-es2015-modules-amd-6.24.1.tgz", - "integrity": "sha1-Oz5UAXI5hC1tGcMBHEvS8AoA0VQ=", + "resolved": "https://registry.npmjs.org/babel-helper-call-delegate/-/babel-helper-call-delegate-6.24.1.tgz", + "integrity": "sha1-7Oaqzdx25Bw0YfiL/Fdb0Nqi340=", "requires": { - "babel-plugin-transform-es2015-modules-commonjs": "^6.24.1", + "babel-helper-hoist-variables": "^6.24.1", "babel-runtime": "^6.22.0", - "babel-template": "^6.24.1" - }, - "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", - "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" - } - }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", - "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" - } - }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "requires": { - "loose-envify": "^1.0.0" - } - } + "babel-traverse": "^6.24.1", + "babel-types": "^6.24.1" } }, - "babel-plugin-transform-es2015-modules-commonjs": { - "version": "6.26.2", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-modules-commonjs/-/babel-plugin-transform-es2015-modules-commonjs-6.26.2.tgz", - "integrity": "sha512-CV9ROOHEdrjcwhIaJNBGMBCodN+1cfkwtM1SbUHmvyy35KGT7fohbpOxkE2uLz1o6odKK2Ck/tz47z+VqQfi9Q==", + "babel-helper-define-map": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-helper-define-map/-/babel-helper-define-map-6.26.0.tgz", + "integrity": "sha1-pfVtq0GiX5fstJjH66ypgZ+Vvl8=", "requires": { - "babel-plugin-transform-strict-mode": "^6.24.1", + "babel-helper-function-name": "^6.24.1", "babel-runtime": "^6.26.0", - "babel-template": "^6.26.0", - "babel-types": "^6.26.0" - }, - "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", - "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" - } - }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", - "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" - } - }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "requires": { - "loose-envify": "^1.0.0" - } - } + "babel-types": "^6.26.0", + "lodash": "^4.17.4" } }, - "babel-plugin-transform-es2015-modules-systemjs": { + "babel-helper-explode-assignable-expression": { "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-modules-systemjs/-/babel-plugin-transform-es2015-modules-systemjs-6.24.1.tgz", - "integrity": "sha1-/4mhQrkRmpBhlfXxBuzzBdlAfSM=", + "resolved": "https://registry.npmjs.org/babel-helper-explode-assignable-expression/-/babel-helper-explode-assignable-expression-6.24.1.tgz", + "integrity": "sha1-8luCz33BBDPFX3BZLVdGQArCLKo=", "requires": { - "babel-helper-hoist-variables": "^6.24.1", "babel-runtime": "^6.22.0", - "babel-template": "^6.24.1" - }, - "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", - "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" - } - }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", - "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" - } - }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "requires": { - "loose-envify": "^1.0.0" - } - } + "babel-traverse": "^6.24.1", + "babel-types": "^6.24.1" } }, - "babel-plugin-transform-es2015-modules-umd": { + "babel-helper-function-name": { "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-modules-umd/-/babel-plugin-transform-es2015-modules-umd-6.24.1.tgz", - "integrity": "sha1-rJl+YoXNGO1hdq22B9YCNErThGg=", + "resolved": "https://registry.npmjs.org/babel-helper-function-name/-/babel-helper-function-name-6.24.1.tgz", + "integrity": "sha1-00dbjAPtmCQqJbSDUasYOZ01gKk=", "requires": { - "babel-plugin-transform-es2015-modules-amd": "^6.24.1", + "babel-helper-get-function-arity": "^6.24.1", "babel-runtime": "^6.22.0", - "babel-template": "^6.24.1" - }, - "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", - "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" - } - }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", - "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" - } - }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "requires": { - "loose-envify": "^1.0.0" - } - } + "babel-template": "^6.24.1", + "babel-traverse": "^6.24.1", + "babel-types": "^6.24.1" } }, - "babel-plugin-transform-es2015-object-super": { + "babel-helper-get-function-arity": { "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-object-super/-/babel-plugin-transform-es2015-object-super-6.24.1.tgz", - "integrity": "sha1-JM72muIcuDp/hgPa0CH1cusnj40=", + "resolved": "https://registry.npmjs.org/babel-helper-get-function-arity/-/babel-helper-get-function-arity-6.24.1.tgz", + "integrity": "sha1-j3eCqpNAfEHTqlCQj4mwMbG2hT0=", "requires": { - "babel-helper-replace-supers": "^6.24.1", - "babel-runtime": "^6.22.0" + "babel-runtime": "^6.22.0", + "babel-types": "^6.24.1" } }, - "babel-plugin-transform-es2015-parameters": { + "babel-helper-hoist-variables": { "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-parameters/-/babel-plugin-transform-es2015-parameters-6.24.1.tgz", - "integrity": "sha1-V6w1GrScrxSpfNE7CfZv3wpiXys=", + "resolved": "https://registry.npmjs.org/babel-helper-hoist-variables/-/babel-helper-hoist-variables-6.24.1.tgz", + "integrity": "sha1-HssnaJydJVE+rbyZFKc/VAi+enY=", "requires": { - "babel-helper-call-delegate": "^6.24.1", - "babel-helper-get-function-arity": "^6.24.1", "babel-runtime": "^6.22.0", - "babel-template": "^6.24.1", - "babel-traverse": "^6.24.1", "babel-types": "^6.24.1" - }, - "dependencies": { - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", - "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" - } - }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", - "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" - } - }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "requires": { - "loose-envify": "^1.0.0" - } - } } }, - "babel-plugin-transform-es2015-shorthand-properties": { + "babel-helper-optimise-call-expression": { "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-shorthand-properties/-/babel-plugin-transform-es2015-shorthand-properties-6.24.1.tgz", - "integrity": "sha1-JPh11nIch2YbvZmkYi5R8U3jiqA=", + "resolved": "https://registry.npmjs.org/babel-helper-optimise-call-expression/-/babel-helper-optimise-call-expression-6.24.1.tgz", + "integrity": "sha1-96E0J7qfc/j0+pk8VKl4gtEkQlc=", "requires": { "babel-runtime": "^6.22.0", "babel-types": "^6.24.1" } }, - "babel-plugin-transform-es2015-spread": { - "version": "6.22.0", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-spread/-/babel-plugin-transform-es2015-spread-6.22.0.tgz", - "integrity": "sha1-1taKmfia7cRTbIGlQujdnxdG+NE=", + "babel-helper-regex": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-helper-regex/-/babel-helper-regex-6.26.0.tgz", + "integrity": "sha1-MlxZ+QL4LyS3T6zu0DY5VPZJXnI=", "requires": { - "babel-runtime": "^6.22.0" + "babel-runtime": "^6.26.0", + "babel-types": "^6.26.0", + "lodash": "^4.17.4" } }, - "babel-plugin-transform-es2015-sticky-regex": { + "babel-helper-remap-async-to-generator": { "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-sticky-regex/-/babel-plugin-transform-es2015-sticky-regex-6.24.1.tgz", - "integrity": "sha1-AMHNsaynERLN8M9hJsLta0V8zbw=", + "resolved": "https://registry.npmjs.org/babel-helper-remap-async-to-generator/-/babel-helper-remap-async-to-generator-6.24.1.tgz", + "integrity": "sha1-XsWBgnrXI/7N04HxySg5BnbkVRs=", "requires": { - "babel-helper-regex": "^6.24.1", + "babel-helper-function-name": "^6.24.1", "babel-runtime": "^6.22.0", + "babel-template": "^6.24.1", + "babel-traverse": "^6.24.1", "babel-types": "^6.24.1" } }, - "babel-plugin-transform-es2015-template-literals": { - "version": "6.22.0", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-template-literals/-/babel-plugin-transform-es2015-template-literals-6.22.0.tgz", - "integrity": "sha1-qEs0UPfp+PH2g51taH2oS7EjbY0=", + "babel-helper-replace-supers": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-helper-replace-supers/-/babel-helper-replace-supers-6.24.1.tgz", + "integrity": "sha1-v22/5Dk40XNpohPKiov3S2qQqxo=", "requires": { - "babel-runtime": "^6.22.0" + "babel-helper-optimise-call-expression": "^6.24.1", + "babel-messages": "^6.23.0", + "babel-runtime": "^6.22.0", + "babel-template": "^6.24.1", + "babel-traverse": "^6.24.1", + "babel-types": "^6.24.1" } }, - "babel-plugin-transform-es2015-typeof-symbol": { - "version": "6.23.0", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-typeof-symbol/-/babel-plugin-transform-es2015-typeof-symbol-6.23.0.tgz", - "integrity": "sha1-3sCfHN3/lLUqxz1QXITfWdzOs3I=", + "babel-helpers": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-helpers/-/babel-helpers-6.24.1.tgz", + "integrity": "sha1-NHHenK7DiOXIUOWX5Yom3fN2ArI=", "requires": { - "babel-runtime": "^6.22.0" + "babel-runtime": "^6.22.0", + "babel-template": "^6.24.1" } }, - "babel-plugin-transform-es2015-unicode-regex": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-unicode-regex/-/babel-plugin-transform-es2015-unicode-regex-6.24.1.tgz", - "integrity": "sha1-04sS9C6nMj9yk4fxinxa4frrNek=", + "babel-messages": { + "version": "6.23.0", + "resolved": "https://registry.npmjs.org/babel-messages/-/babel-messages-6.23.0.tgz", + "integrity": "sha1-8830cDhYA1sqKVHG7F7fbGLyYw4=", "requires": { - "babel-helper-regex": "^6.24.1", - "babel-runtime": "^6.22.0", - "regexpu-core": "^2.0.0" - }, - "dependencies": { - "regexpu-core": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-2.0.0.tgz", - "integrity": "sha1-SdA4g3uNz4v6W5pCE5k45uoq4kA=", - "requires": { - "regenerate": "^1.2.1", - "regjsgen": "^0.2.0", - "regjsparser": "^0.1.4" - } - } + "babel-runtime": "^6.22.0" } }, - "babel-plugin-transform-exponentiation-operator": { - "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-exponentiation-operator/-/babel-plugin-transform-exponentiation-operator-6.24.1.tgz", - "integrity": "sha1-KrDJx/MJj6SJB3cruBP+QejeOg4=", + "babel-plugin-check-es2015-constants": { + "version": "6.22.0", + "resolved": "https://registry.npmjs.org/babel-plugin-check-es2015-constants/-/babel-plugin-check-es2015-constants-6.22.0.tgz", + "integrity": "sha1-NRV7EBQm/S/9PaP3XH0ekYNbv4o=", "requires": { - "babel-helper-builder-binary-assignment-operator-visitor": "^6.24.1", - "babel-plugin-syntax-exponentiation-operator": "^6.8.0", "babel-runtime": "^6.22.0" } }, - "babel-plugin-transform-regenerator": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-regenerator/-/babel-plugin-transform-regenerator-6.26.0.tgz", - "integrity": "sha1-4HA2lvveJ/Cj78rPi03KL3s6jy8=", + "babel-plugin-espower": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/babel-plugin-espower/-/babel-plugin-espower-3.0.0.tgz", + "integrity": "sha512-f2IUz5kQyrwXnShcv7tvGxf76QkrEl00ENYgd6R0VMrz4xqlwBLZXFs5vse2vehs1Z+T2sXTP3UWX2QxMorzzw==", + "dev": true, "requires": { - "regenerator-transform": "^0.10.0" + "@babel/generator": "^7.0.0", + "@babel/parser": "^7.0.0", + "call-matcher": "^1.0.0", + "core-js": "^2.0.0", + "espower-location-detector": "^1.0.0", + "espurify": "^1.6.0", + "estraverse": "^4.1.1" } }, - "babel-plugin-transform-strict-mode": { + "babel-plugin-syntax-async-functions": { + "version": "6.13.0", + "resolved": "http://registry.npmjs.org/babel-plugin-syntax-async-functions/-/babel-plugin-syntax-async-functions-6.13.0.tgz", + "integrity": "sha1-ytnK0RkbWtY0vzCuCHI5HgZHvpU=" + }, + "babel-plugin-syntax-exponentiation-operator": { + "version": "6.13.0", + "resolved": "http://registry.npmjs.org/babel-plugin-syntax-exponentiation-operator/-/babel-plugin-syntax-exponentiation-operator-6.13.0.tgz", + "integrity": "sha1-nufoM3KQ2pUoggGmpX9BcDF4MN4=" + }, + "babel-plugin-syntax-trailing-function-commas": { + "version": "6.22.0", + "resolved": "https://registry.npmjs.org/babel-plugin-syntax-trailing-function-commas/-/babel-plugin-syntax-trailing-function-commas-6.22.0.tgz", + "integrity": "sha1-ugNgk3+NBuQBgKQ/4NVhb/9TLPM=" + }, + "babel-plugin-transform-async-to-generator": { "version": "6.24.1", - "resolved": "https://registry.npmjs.org/babel-plugin-transform-strict-mode/-/babel-plugin-transform-strict-mode-6.24.1.tgz", - "integrity": "sha1-1fr3qleKZbvlkc9e2uBKDGcCB1g=", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-async-to-generator/-/babel-plugin-transform-async-to-generator-6.24.1.tgz", + "integrity": "sha1-ZTbjeK/2yx1VF6wOQOs+n8jQh2E=", "requires": { - "babel-runtime": "^6.22.0", - "babel-types": "^6.24.1" + "babel-helper-remap-async-to-generator": "^6.24.1", + "babel-plugin-syntax-async-functions": "^6.8.0", + "babel-runtime": "^6.22.0" } }, - "babel-preset-env": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/babel-preset-env/-/babel-preset-env-1.7.0.tgz", - "integrity": "sha512-9OR2afuKDneX2/q2EurSftUYM0xGu4O2D9adAhVfADDhrYDaxXV0rBbevVYoY9n6nyX1PmQW/0jtpJvUNr9CHg==", + "babel-plugin-transform-es2015-arrow-functions": { + "version": "6.22.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-arrow-functions/-/babel-plugin-transform-es2015-arrow-functions-6.22.0.tgz", + "integrity": "sha1-RSaSy3EdX3ncf4XkQM5BufJE0iE=", "requires": { - "babel-plugin-check-es2015-constants": "^6.22.0", - "babel-plugin-syntax-trailing-function-commas": "^6.22.0", + "babel-runtime": "^6.22.0" + } + }, + "babel-plugin-transform-es2015-block-scoped-functions": { + "version": "6.22.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-block-scoped-functions/-/babel-plugin-transform-es2015-block-scoped-functions-6.22.0.tgz", + "integrity": "sha1-u8UbSflk1wy42OC5ToICRs46YUE=", + "requires": { + "babel-runtime": "^6.22.0" + } + }, + "babel-plugin-transform-es2015-block-scoping": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-block-scoping/-/babel-plugin-transform-es2015-block-scoping-6.26.0.tgz", + "integrity": "sha1-1w9SmcEwjQXBL0Y4E7CgnnOxiV8=", + "requires": { + "babel-runtime": "^6.26.0", + "babel-template": "^6.26.0", + "babel-traverse": "^6.26.0", + "babel-types": "^6.26.0", + "lodash": "^4.17.4" + } + }, + "babel-plugin-transform-es2015-classes": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-classes/-/babel-plugin-transform-es2015-classes-6.24.1.tgz", + "integrity": "sha1-WkxYpQyclGHlZLSyo7+ryXolhNs=", + "requires": { + "babel-helper-define-map": "^6.24.1", + "babel-helper-function-name": "^6.24.1", + "babel-helper-optimise-call-expression": "^6.24.1", + "babel-helper-replace-supers": "^6.24.1", + "babel-messages": "^6.23.0", + "babel-runtime": "^6.22.0", + "babel-template": "^6.24.1", + "babel-traverse": "^6.24.1", + "babel-types": "^6.24.1" + } + }, + "babel-plugin-transform-es2015-computed-properties": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-computed-properties/-/babel-plugin-transform-es2015-computed-properties-6.24.1.tgz", + "integrity": "sha1-b+Ko0WiV1WNPTNmZttNICjCBWbM=", + "requires": { + "babel-runtime": "^6.22.0", + "babel-template": "^6.24.1" + } + }, + "babel-plugin-transform-es2015-destructuring": { + "version": "6.23.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-destructuring/-/babel-plugin-transform-es2015-destructuring-6.23.0.tgz", + "integrity": "sha1-mXux8auWf2gtKwh2/jWNYOdlxW0=", + "requires": { + "babel-runtime": "^6.22.0" + } + }, + "babel-plugin-transform-es2015-duplicate-keys": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-duplicate-keys/-/babel-plugin-transform-es2015-duplicate-keys-6.24.1.tgz", + "integrity": "sha1-c+s9MQypaePvnskcU3QabxV2Qj4=", + "requires": { + "babel-runtime": "^6.22.0", + "babel-types": "^6.24.1" + } + }, + "babel-plugin-transform-es2015-for-of": { + "version": "6.23.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-for-of/-/babel-plugin-transform-es2015-for-of-6.23.0.tgz", + "integrity": "sha1-9HyVsrYT3x0+zC/bdXNiPHUkhpE=", + "requires": { + "babel-runtime": "^6.22.0" + } + }, + "babel-plugin-transform-es2015-function-name": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-function-name/-/babel-plugin-transform-es2015-function-name-6.24.1.tgz", + "integrity": "sha1-g0yJhTvDaxrw86TF26qU/Y6sqos=", + "requires": { + "babel-helper-function-name": "^6.24.1", + "babel-runtime": "^6.22.0", + "babel-types": "^6.24.1" + } + }, + "babel-plugin-transform-es2015-literals": { + "version": "6.22.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-literals/-/babel-plugin-transform-es2015-literals-6.22.0.tgz", + "integrity": "sha1-T1SgLWzWbPkVKAAZox0xklN3yi4=", + "requires": { + "babel-runtime": "^6.22.0" + } + }, + "babel-plugin-transform-es2015-modules-amd": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-modules-amd/-/babel-plugin-transform-es2015-modules-amd-6.24.1.tgz", + "integrity": "sha1-Oz5UAXI5hC1tGcMBHEvS8AoA0VQ=", + "requires": { + "babel-plugin-transform-es2015-modules-commonjs": "^6.24.1", + "babel-runtime": "^6.22.0", + "babel-template": "^6.24.1" + } + }, + "babel-plugin-transform-es2015-modules-commonjs": { + "version": "6.26.2", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-modules-commonjs/-/babel-plugin-transform-es2015-modules-commonjs-6.26.2.tgz", + "integrity": "sha512-CV9ROOHEdrjcwhIaJNBGMBCodN+1cfkwtM1SbUHmvyy35KGT7fohbpOxkE2uLz1o6odKK2Ck/tz47z+VqQfi9Q==", + "requires": { + "babel-plugin-transform-strict-mode": "^6.24.1", + "babel-runtime": "^6.26.0", + "babel-template": "^6.26.0", + "babel-types": "^6.26.0" + } + }, + "babel-plugin-transform-es2015-modules-systemjs": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-modules-systemjs/-/babel-plugin-transform-es2015-modules-systemjs-6.24.1.tgz", + "integrity": "sha1-/4mhQrkRmpBhlfXxBuzzBdlAfSM=", + "requires": { + "babel-helper-hoist-variables": "^6.24.1", + "babel-runtime": "^6.22.0", + "babel-template": "^6.24.1" + } + }, + "babel-plugin-transform-es2015-modules-umd": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-modules-umd/-/babel-plugin-transform-es2015-modules-umd-6.24.1.tgz", + "integrity": "sha1-rJl+YoXNGO1hdq22B9YCNErThGg=", + "requires": { + "babel-plugin-transform-es2015-modules-amd": "^6.24.1", + "babel-runtime": "^6.22.0", + "babel-template": "^6.24.1" + } + }, + "babel-plugin-transform-es2015-object-super": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-object-super/-/babel-plugin-transform-es2015-object-super-6.24.1.tgz", + "integrity": "sha1-JM72muIcuDp/hgPa0CH1cusnj40=", + "requires": { + "babel-helper-replace-supers": "^6.24.1", + "babel-runtime": "^6.22.0" + } + }, + "babel-plugin-transform-es2015-parameters": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-parameters/-/babel-plugin-transform-es2015-parameters-6.24.1.tgz", + "integrity": "sha1-V6w1GrScrxSpfNE7CfZv3wpiXys=", + "requires": { + "babel-helper-call-delegate": "^6.24.1", + "babel-helper-get-function-arity": "^6.24.1", + "babel-runtime": "^6.22.0", + "babel-template": "^6.24.1", + "babel-traverse": "^6.24.1", + "babel-types": "^6.24.1" + } + }, + "babel-plugin-transform-es2015-shorthand-properties": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-shorthand-properties/-/babel-plugin-transform-es2015-shorthand-properties-6.24.1.tgz", + "integrity": "sha1-JPh11nIch2YbvZmkYi5R8U3jiqA=", + "requires": { + "babel-runtime": "^6.22.0", + "babel-types": "^6.24.1" + } + }, + "babel-plugin-transform-es2015-spread": { + "version": "6.22.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-spread/-/babel-plugin-transform-es2015-spread-6.22.0.tgz", + "integrity": "sha1-1taKmfia7cRTbIGlQujdnxdG+NE=", + "requires": { + "babel-runtime": "^6.22.0" + } + }, + "babel-plugin-transform-es2015-sticky-regex": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-sticky-regex/-/babel-plugin-transform-es2015-sticky-regex-6.24.1.tgz", + "integrity": "sha1-AMHNsaynERLN8M9hJsLta0V8zbw=", + "requires": { + "babel-helper-regex": "^6.24.1", + "babel-runtime": "^6.22.0", + "babel-types": "^6.24.1" + } + }, + "babel-plugin-transform-es2015-template-literals": { + "version": "6.22.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-template-literals/-/babel-plugin-transform-es2015-template-literals-6.22.0.tgz", + "integrity": "sha1-qEs0UPfp+PH2g51taH2oS7EjbY0=", + "requires": { + "babel-runtime": "^6.22.0" + } + }, + "babel-plugin-transform-es2015-typeof-symbol": { + "version": "6.23.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-typeof-symbol/-/babel-plugin-transform-es2015-typeof-symbol-6.23.0.tgz", + "integrity": "sha1-3sCfHN3/lLUqxz1QXITfWdzOs3I=", + "requires": { + "babel-runtime": "^6.22.0" + } + }, + "babel-plugin-transform-es2015-unicode-regex": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-es2015-unicode-regex/-/babel-plugin-transform-es2015-unicode-regex-6.24.1.tgz", + "integrity": "sha1-04sS9C6nMj9yk4fxinxa4frrNek=", + "requires": { + "babel-helper-regex": "^6.24.1", + "babel-runtime": "^6.22.0", + "regexpu-core": "^2.0.0" + } + }, + "babel-plugin-transform-exponentiation-operator": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-exponentiation-operator/-/babel-plugin-transform-exponentiation-operator-6.24.1.tgz", + "integrity": "sha1-KrDJx/MJj6SJB3cruBP+QejeOg4=", + "requires": { + "babel-helper-builder-binary-assignment-operator-visitor": "^6.24.1", + "babel-plugin-syntax-exponentiation-operator": "^6.8.0", + "babel-runtime": "^6.22.0" + } + }, + "babel-plugin-transform-regenerator": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-regenerator/-/babel-plugin-transform-regenerator-6.26.0.tgz", + "integrity": "sha1-4HA2lvveJ/Cj78rPi03KL3s6jy8=", + "requires": { + "regenerator-transform": "^0.10.0" + } + }, + "babel-plugin-transform-strict-mode": { + "version": "6.24.1", + "resolved": "https://registry.npmjs.org/babel-plugin-transform-strict-mode/-/babel-plugin-transform-strict-mode-6.24.1.tgz", + "integrity": "sha1-1fr3qleKZbvlkc9e2uBKDGcCB1g=", + "requires": { + "babel-runtime": "^6.22.0", + "babel-types": "^6.24.1" + } + }, + "babel-preset-env": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/babel-preset-env/-/babel-preset-env-1.7.0.tgz", + "integrity": "sha512-9OR2afuKDneX2/q2EurSftUYM0xGu4O2D9adAhVfADDhrYDaxXV0rBbevVYoY9n6nyX1PmQW/0jtpJvUNr9CHg==", + "requires": { + "babel-plugin-check-es2015-constants": "^6.22.0", + "babel-plugin-syntax-trailing-function-commas": "^6.22.0", "babel-plugin-transform-async-to-generator": "^6.22.0", "babel-plugin-transform-es2015-arrow-functions": "^6.22.0", "babel-plugin-transform-es2015-block-scoped-functions": "^6.22.0", @@ -1149,16 +1729,6 @@ "browserslist": "^3.2.6", "invariant": "^2.2.2", "semver": "^5.3.0" - }, - "dependencies": { - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "requires": { - "loose-envify": "^1.0.0" - } - } } }, "babel-register": { @@ -1173,70 +1743,6 @@ "lodash": "^4.17.4", "mkdirp": "^0.5.1", "source-map-support": "^0.4.15" - }, - "dependencies": { - "babel-core": { - "version": "6.26.3", - "resolved": "https://registry.npmjs.org/babel-core/-/babel-core-6.26.3.tgz", - "integrity": "sha512-6jyFLuDmeidKmUEb3NM+/yawG0M2bDZ9Z1qbZP59cyHLz8kYGKYwpJP0UwUKKUiTRNvxfLesJnTedqczP7cTDA==", - "requires": { - "babel-code-frame": "^6.26.0", - "babel-generator": "^6.26.0", - "babel-helpers": "^6.24.1", - "babel-messages": "^6.23.0", - "babel-register": "^6.26.0", - "babel-runtime": "^6.26.0", - "babel-template": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "convert-source-map": "^1.5.1", - "debug": "^2.6.9", - "json5": "^0.5.1", - "lodash": "^4.17.4", - "minimatch": "^3.0.4", - "path-is-absolute": "^1.0.1", - "private": "^0.1.8", - "slash": "^1.0.0", - "source-map": "^0.5.7" - } - }, - "babel-template": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", - "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", - "requires": { - "babel-runtime": "^6.26.0", - "babel-traverse": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "lodash": "^4.17.4" - } - }, - "babel-traverse": { - "version": "6.26.0", - "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", - "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", - "requires": { - "babel-code-frame": "^6.26.0", - "babel-messages": "^6.23.0", - "babel-runtime": "^6.26.0", - "babel-types": "^6.26.0", - "babylon": "^6.18.0", - "debug": "^2.6.8", - "globals": "^9.18.0", - "invariant": "^2.2.2", - "lodash": "^4.17.4" - } - }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "requires": { - "loose-envify": "^1.0.0" - } - } } }, "babel-runtime": { @@ -1248,6 +1754,34 @@ "regenerator-runtime": "^0.11.0" } }, + "babel-template": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-template/-/babel-template-6.26.0.tgz", + "integrity": "sha1-3gPi0WOWsGn0bdn/+FIfsaDjXgI=", + "requires": { + "babel-runtime": "^6.26.0", + "babel-traverse": "^6.26.0", + "babel-types": "^6.26.0", + "babylon": "^6.18.0", + "lodash": "^4.17.4" + } + }, + "babel-traverse": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-traverse/-/babel-traverse-6.26.0.tgz", + "integrity": "sha1-RqnL1+3MYsjlwGTi0tjQ9ANXZu4=", + "requires": { + "babel-code-frame": "^6.26.0", + "babel-messages": "^6.23.0", + "babel-runtime": "^6.26.0", + "babel-types": "^6.26.0", + "babylon": "^6.18.0", + "debug": "^2.6.8", + "globals": "^9.18.0", + "invariant": "^2.2.2", + "lodash": "^4.17.4" + } + }, "babel-types": { "version": "6.26.0", "resolved": "https://registry.npmjs.org/babel-types/-/babel-types-6.26.0.tgz", @@ -1269,15 +1803,87 @@ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" }, + "base": { + "version": "0.11.2", + "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz", + "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", + "dev": true, + "requires": { + "cache-base": "^1.0.1", + "class-utils": "^0.3.5", + "component-emitter": "^1.2.1", + "define-property": "^1.0.0", + "isobject": "^3.0.1", + "mixin-deep": "^1.2.0", + "pascalcase": "^0.1.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "kind-of": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz", + "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==", + "dev": true + } + } + }, "bcrypt-pbkdf": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", - "optional": true, "requires": { "tweetnacl": "^0.14.3" } }, + "binary-extensions": { + "version": "1.12.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.12.0.tgz", + "integrity": "sha512-DYWGk01lDcxeS/K9IHPGWfT8PsJmbXRtRd2Sx72Tnb8pcYZQFF1oSDb8hJtS1vhp212q1Rzi5dUf9+nq0o9UIg==", + "dev": true + }, "bindings": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.2.1.tgz", @@ -1286,7 +1892,7 @@ }, "bl": { "version": "0.8.2", - "resolved": "https://registry.npmjs.org/bl/-/bl-0.8.2.tgz", + "resolved": "http://registry.npmjs.org/bl/-/bl-0.8.2.tgz", "integrity": "sha1-yba8oI0bwuoA/Ir7Txpf0eHGbk4=", "requires": { "readable-stream": "~1.0.26" @@ -1299,7 +1905,7 @@ }, "readable-stream": { "version": "1.0.34", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz", + "resolved": "http://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz", "integrity": "sha1-Elgg40vIQtLyqq+v5MKRbuMsFXw=", "requires": { "core-util-is": "~1.0.0", @@ -1315,37 +1921,61 @@ } } }, + "bluebird": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.2.tgz", + "integrity": "sha512-dhHTWMI7kMx5whMQntl7Vr9C6BvV10lFXDAasnqnrMYhXVCzzk6IO9Fo2L75jXHT07WrOngL1WDXOp+yYS91Yg==", + "dev": true + }, "bn.js": { "version": "4.11.8", "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==" }, "body-parser": { - "version": "1.18.2", - "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.2.tgz", - "integrity": "sha1-h2eKGdhLR9hZuDGZvVm84iKxBFQ=", + "version": "1.18.3", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.3.tgz", + "integrity": "sha1-WykhmP/dVTs6DyDe0FkrlWlVyLQ=", "dev": true, "requires": { "bytes": "3.0.0", "content-type": "~1.0.4", "debug": "2.6.9", - "depd": "~1.1.1", - "http-errors": "~1.6.2", - "iconv-lite": "0.4.19", + "depd": "~1.1.2", + "http-errors": "~1.6.3", + "iconv-lite": "0.4.23", "on-finished": "~2.3.0", - "qs": "6.5.1", - "raw-body": "2.3.2", - "type-is": "~1.6.15" + "qs": "6.5.2", + "raw-body": "2.3.3", + "type-is": "~1.6.16" }, "dependencies": { - "qs": { - "version": "6.5.1", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz", - "integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A==", - "dev": true + "iconv-lite": { + "version": "0.4.23", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.23.tgz", + "integrity": "sha512-neyTUVFtahjf0mB3dZT77u+8O0QB89jFdnBkd5P1JgYPbPaia3gXXOVL2fq8VyU2gMMD7SaN7QukTB/pmXYvDA==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } } } }, + "boxen": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/boxen/-/boxen-1.3.0.tgz", + "integrity": "sha512-TNPjfTr432qx7yOjQyaXm3dSR0MH9vXp7eT1BFSl/C51g+EFnOR9hTg1IreahGBmDNCehscshe45f+C1TBZbLw==", + "dev": true, + "requires": { + "ansi-align": "^2.0.0", + "camelcase": "^4.0.0", + "chalk": "^2.0.1", + "cli-boxes": "^1.0.0", + "string-width": "^2.0.0", + "term-size": "^1.2.0", + "widest-line": "^2.0.0" + } + }, "brace-expansion": { "version": "1.1.11", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", @@ -1371,9 +2001,9 @@ "integrity": "sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=" }, "browser-process-hrtime": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/browser-process-hrtime/-/browser-process-hrtime-0.1.2.tgz", - "integrity": "sha1-Ql1opY00R/AqBKqJQYf86K+Le44=" + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/browser-process-hrtime/-/browser-process-hrtime-0.1.3.tgz", + "integrity": "sha512-bRFnI4NnjO6cnyLmOV/7PVoDEMJChlcfN0z4s1YMBY989/SvlfMI1lgCnkFUs53e9gQF+w7qu7XdllSTiSl8Aw==" }, "browser-stdout": { "version": "1.3.0", @@ -1383,7 +2013,7 @@ }, "browserify-aes": { "version": "1.2.0", - "resolved": "https://registry.npmjs.org/browserify-aes/-/browserify-aes-1.2.0.tgz", + "resolved": "http://registry.npmjs.org/browserify-aes/-/browserify-aes-1.2.0.tgz", "integrity": "sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==", "requires": { "buffer-xor": "^1.0.3", @@ -1392,20 +2022,6 @@ "evp_bytestokey": "^1.0.3", "inherits": "^2.0.1", "safe-buffer": "^5.0.1" - }, - "dependencies": { - "create-hash": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", - "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", - "requires": { - "cipher-base": "^1.0.1", - "inherits": "^2.0.1", - "md5.js": "^1.3.4", - "ripemd160": "^2.0.1", - "sha.js": "^2.4.0" - } - } } }, "browserify-cipher": { @@ -1437,81 +2053,11 @@ "level-filesystem": "^1.0.1", "level-js": "^2.1.3", "levelup": "^0.18.2" - }, - "dependencies": { - "isarray": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", - "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" - }, - "level-filesystem": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/level-filesystem/-/level-filesystem-1.2.0.tgz", - "integrity": "sha1-oArKmRnEpN+v3KaoEI0iWq3/Y7M=", - "requires": { - "concat-stream": "^1.4.4", - "errno": "^0.1.1", - "fwd-stream": "^1.0.4", - "level-blobs": "^0.1.7", - "level-peek": "^1.0.6", - "level-sublevel": "^5.2.0", - "octal": "^1.0.0", - "once": "^1.3.0", - "xtend": "^2.2.0" - } - }, - "levelup": { - "version": "0.18.6", - "resolved": "https://registry.npmjs.org/levelup/-/levelup-0.18.6.tgz", - "integrity": "sha1-5qAcsIlhbI7MApHCqb0/DETj5es=", - "requires": { - "bl": "~0.8.1", - "deferred-leveldown": "~0.2.0", - "errno": "~0.1.1", - "prr": "~0.0.0", - "readable-stream": "~1.0.26", - "semver": "~2.3.1", - "xtend": "~3.0.0" - }, - "dependencies": { - "xtend": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/xtend/-/xtend-3.0.0.tgz", - "integrity": "sha1-XM50B7r2Qsunvs2laBEcST9ZZlo=" - } - } - }, - "prr": { - "version": "0.0.0", - "resolved": "https://registry.npmjs.org/prr/-/prr-0.0.0.tgz", - "integrity": "sha1-GoS4WQgyVQFBGFPQCB7j+obikmo=" - }, - "readable-stream": { - "version": "1.0.34", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz", - "integrity": "sha1-Elgg40vIQtLyqq+v5MKRbuMsFXw=", - "requires": { - "core-util-is": "~1.0.0", - "inherits": "~2.0.1", - "isarray": "0.0.1", - "string_decoder": "~0.10.x" - } - }, - "semver": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-2.3.2.tgz", - "integrity": "sha1-uYSPJdbPNjMwc+ye+IVtQvEjPlI=" - }, - "string_decoder": { - "version": "0.10.31", - "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", - "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=" - } } }, "browserify-rsa": { "version": "4.0.1", - "resolved": "https://registry.npmjs.org/browserify-rsa/-/browserify-rsa-4.0.1.tgz", + "resolved": "http://registry.npmjs.org/browserify-rsa/-/browserify-rsa-4.0.1.tgz", "integrity": "sha1-IeCr+vbyApzy+vsTNWenAdQTVSQ=", "requires": { "bn.js": "^4.1.0", @@ -1530,33 +2076,6 @@ "elliptic": "^6.0.0", "inherits": "^2.0.1", "parse-asn1": "^5.0.0" - }, - "dependencies": { - "create-hash": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", - "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", - "requires": { - "cipher-base": "^1.0.1", - "inherits": "^2.0.1", - "md5.js": "^1.3.4", - "ripemd160": "^2.0.1", - "sha.js": "^2.4.0" - } - }, - "create-hmac": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz", - "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==", - "requires": { - "cipher-base": "^1.0.3", - "create-hash": "^1.1.0", - "inherits": "^2.0.1", - "ripemd160": "^2.0.0", - "safe-buffer": "^5.0.1", - "sha.js": "^2.4.8" - } - } } }, "browserslist": { @@ -1574,9 +2093,9 @@ "integrity": "sha1-8mNHuC33b9N+GLy1KIxJcM/VxAQ=" }, "buffer-from": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.0.tgz", - "integrity": "sha512-c5mRlguI/Pe2dSZmpER62rSCu0ryKmWddzRYsuXc50U2/g8jMOulc31VZMa4mYx31U5xsmSOpDCgH88Vl9cDGQ==" + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", + "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==" }, "buffer-xor": { "version": "1.0.3", @@ -1594,10 +2113,76 @@ "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=", "dev": true }, + "cache-base": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz", + "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==", + "dev": true, + "requires": { + "collection-visit": "^1.0.0", + "component-emitter": "^1.2.1", + "get-value": "^2.0.6", + "has-value": "^1.0.0", + "isobject": "^3.0.1", + "set-value": "^2.0.0", + "to-object-path": "^0.3.0", + "union-value": "^1.0.0", + "unset-value": "^1.0.0" + }, + "dependencies": { + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "call-matcher": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/call-matcher/-/call-matcher-1.1.0.tgz", + "integrity": "sha512-IoQLeNwwf9KTNbtSA7aEBb1yfDbdnzwjCetjkC8io5oGeOmK2CBNdg0xr+tadRYKO0p7uQyZzvon0kXlZbvGrw==", + "dev": true, + "requires": { + "core-js": "^2.0.0", + "deep-equal": "^1.0.0", + "espurify": "^1.6.0", + "estraverse": "^4.0.0" + } + }, + "call-signature": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/call-signature/-/call-signature-0.0.2.tgz", + "integrity": "sha1-qEq8glpV70yysCi9dOIFpluaSZY=", + "dev": true + }, + "camelcase": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-4.1.0.tgz", + "integrity": "sha1-1UVjW+HjPFQmScaRc+Xeas+uNN0=", + "dev": true + }, + "camelcase-keys": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/camelcase-keys/-/camelcase-keys-4.2.0.tgz", + "integrity": "sha1-oqpfsa9oh1glnDLBQUJteJI7m3c=", + "dev": true, + "requires": { + "camelcase": "^4.1.0", + "map-obj": "^2.0.0", + "quick-lru": "^1.0.0" + } + }, "caniuse-lite": { - "version": "1.0.30000865", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30000865.tgz", - "integrity": "sha512-vs79o1mOSKRGv/1pSkp4EXgl4ZviWeYReXw60XfacPU64uQWZwJT6vZNmxRF9O+6zu71sJwMxLK5JXxbzuVrLw==" + "version": "1.0.30000893", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30000893.tgz", + "integrity": "sha512-kOddHcTEef+NgN/fs0zmX2brHTNATVOWMEIhlZHCuwQRtXobjSw9pAECc44Op4bTBcavRjkLaPrGomknH7+Jvg==" + }, + "capture-stack-trace": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/capture-stack-trace/-/capture-stack-trace-1.0.1.tgz", + "integrity": "sha512-mYQLZnx5Qt1JgB1WEiMCf2647plpGeQ2NMR/5L0HNZzGQo4fuSPnK+wjfPnKZV0aiJDgzmWqqkV/g7JD+DW0qw==", + "dev": true }, "caseless": { "version": "0.12.0", @@ -1623,15 +2208,152 @@ } }, "supports-color": { - "version": "5.4.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.4.0.tgz", - "integrity": "sha512-zjaXglF5nnWpsq470jSv6P9DwPvgLkuapYmfDm3JWOm0vkNTVF2tI4UrN2r6jH1qM/uc/WtxYY1hYoA2dOKj5w==", + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", "requires": { "has-flag": "^3.0.0" } } } }, + "chokidar": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.0.4.tgz", + "integrity": "sha512-z9n7yt9rOvIJrMhvDtDictKrkFHeihkNl6uWMmZlmL6tJtX9Cs+87oK+teBx+JIgzvbX3yZHT3eF8vpbDxHJXQ==", + "dev": true, + "requires": { + "anymatch": "^2.0.0", + "async-each": "^1.0.0", + "braces": "^2.3.0", + "fsevents": "^1.2.2", + "glob-parent": "^3.1.0", + "inherits": "^2.0.1", + "is-binary-path": "^1.0.0", + "is-glob": "^4.0.0", + "lodash.debounce": "^4.0.8", + "normalize-path": "^2.1.1", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.0.0", + "upath": "^1.0.5" + }, + "dependencies": { + "array-unique": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", + "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", + "dev": true + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + } + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", + "dev": true + }, + "is-glob": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.0.tgz", + "integrity": "sha1-lSHHaEXMJhCoUgPd8ICpWML/q8A=", + "dev": true, + "requires": { + "is-extglob": "^2.1.1" + } + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "chunkd": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/chunkd/-/chunkd-1.0.0.tgz", + "integrity": "sha512-xx3Pb5VF9QaqCotolyZ1ywFBgyuJmu6+9dLiqBxgelEse9Xsr3yUlpoX3O4Oh11M00GT2kYMsRByTKIMJW2Lkg==", + "dev": true + }, + "ci-info": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-1.6.0.tgz", + "integrity": "sha512-vsGdkwSCDpWmP80ncATX7iea5DWQemg1UgCW5J8tqjU3lYw4FBYuj89J0CTVomA7BEfvSZd84GmHko+MxFQU2A==", + "dev": true + }, + "ci-parallel-vars": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/ci-parallel-vars/-/ci-parallel-vars-1.0.0.tgz", + "integrity": "sha512-u6dx20FBXm+apMi+5x7UVm6EH7BL1gc4XrcnQewjcB7HWRcor/V5qWc3RG2HwpgDJ26gIi2DSEu3B7sXynAw/g==", + "dev": true + }, "cipher-base": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/cipher-base/-/cipher-base-1.0.4.tgz", @@ -1641,6 +2363,78 @@ "safe-buffer": "^5.0.1" } }, + "class-utils": { + "version": "0.3.6", + "resolved": "https://registry.npmjs.org/class-utils/-/class-utils-0.3.6.tgz", + "integrity": "sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==", + "dev": true, + "requires": { + "arr-union": "^3.1.0", + "define-property": "^0.2.5", + "isobject": "^3.0.0", + "static-extend": "^0.1.1" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "clean-stack": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-1.3.0.tgz", + "integrity": "sha1-noIVAa6XmYbEax1m0tQy2y/UrjE=", + "dev": true + }, + "clean-yaml-object": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/clean-yaml-object/-/clean-yaml-object-0.1.0.tgz", + "integrity": "sha1-Y/sRDcLOGoTcIfbZM0h20BCui2g=", + "dev": true + }, + "cli-boxes": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-1.0.0.tgz", + "integrity": "sha1-T6kXw+WclKAEzWH47lCdplFocUM=", + "dev": true + }, + "cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", + "dev": true, + "requires": { + "restore-cursor": "^2.0.0" + } + }, + "cli-spinners": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-1.3.1.tgz", + "integrity": "sha512-1QL4544moEsDVH9T/l6Cemov/37iv1RtoKf7NJ04A60+4MREXNfx/QvavbH6QoGdsD4N4Mwy49cmaINR/o2mdg==", + "dev": true + }, + "cli-truncate": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/cli-truncate/-/cli-truncate-1.1.0.tgz", + "integrity": "sha512-bAtZo0u82gCfaAGfSNxUdTI9mNyza7D8w4CVCcaOsy7sgwDzvx6ekr6cuWJqY3UGzgnQ1+4wgENup5eIhgxEYA==", + "dev": true, + "requires": { + "slice-ansi": "^1.0.0", + "string-width": "^2.0.0" + } + }, "clone": { "version": "0.1.19", "resolved": "https://registry.npmjs.org/clone/-/clone-0.1.19.tgz", @@ -1651,29 +2445,48 @@ "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=" }, + "code-excerpt": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/code-excerpt/-/code-excerpt-2.1.1.tgz", + "integrity": "sha512-tJLhH3EpFm/1x7heIW0hemXJTUU5EWl2V0EIX558jp05Mt1U6DVryCgkp3l37cxqs+DNbNgxG43SkwJXpQ14Jw==", + "dev": true, + "requires": { + "convert-to-spaces": "^1.0.1" + } + }, "coffee-script": { "version": "1.12.7", "resolved": "https://registry.npmjs.org/coffee-script/-/coffee-script-1.12.7.tgz", "integrity": "sha512-fLeEhqwymYat/MpTPUjSKHVYYl0ec2mOyALEMLmzr5i1isuG+6jfI2j2d5oBO3VIzgUXgBVIcOT9uH1TFxBckw==", "dev": true }, - "color-convert": { - "version": "1.9.2", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.2.tgz", - "integrity": "sha512-3NUJZdhMhcdPn8vJ9v2UQJoH0qqoGUkYTgFEPZaPjEtwmmKUfNV46zZmgB2M5M4DCEQHMaCfWHCxiBflLm04Tg==", + "collection-visit": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz", + "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=", + "dev": true, "requires": { - "color-name": "1.1.1" + "map-visit": "^1.0.0", + "object-visit": "^1.0.0" + } + }, + "color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "requires": { + "color-name": "1.1.3" } }, "color-name": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.1.tgz", - "integrity": "sha1-SxQVMEz1ACjqgWQ2Q72C6gWANok=" + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" }, "combined-stream": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.6.tgz", - "integrity": "sha1-cj599ugBrFYTETp+RFqbactjKBg=", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.7.tgz", + "integrity": "sha512-brWl9y6vOB1xYPZcpZde3N9zDByXTosAeMDo4p1wzo6UMOX4vumB+TP1RZ76sfE6Md68Q0NJSrE/gbezd4Ul+w==", "requires": { "delayed-stream": "~1.0.0" } @@ -1684,6 +2497,18 @@ "integrity": "sha512-b0553uYA5YAEGgyYIGYROzKQ7X5RAqedkfjiZxwi0kL1g3bOaBNNZfYkzt/CL0umgD5wc9Jec2FbB98CjkMRvQ==", "dev": true }, + "common-path-prefix": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/common-path-prefix/-/common-path-prefix-1.0.0.tgz", + "integrity": "sha1-zVL28HEuC6q5fW+XModPIvR3UsA=", + "dev": true + }, + "component-emitter": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz", + "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=", + "dev": true + }, "concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", @@ -1700,6 +2525,39 @@ "typedarray": "^0.0.6" } }, + "concordance": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/concordance/-/concordance-4.0.0.tgz", + "integrity": "sha512-l0RFuB8RLfCS0Pt2Id39/oCPykE01pyxgAFypWTlaGRgvLkZrtczZ8atEHpTeEIW+zYWXTBuA9cCSeEOScxReQ==", + "dev": true, + "requires": { + "date-time": "^2.1.0", + "esutils": "^2.0.2", + "fast-diff": "^1.1.2", + "js-string-escape": "^1.0.1", + "lodash.clonedeep": "^4.5.0", + "lodash.flattendeep": "^4.4.0", + "lodash.islength": "^4.0.1", + "lodash.merge": "^4.6.1", + "md5-hex": "^2.0.0", + "semver": "^5.5.1", + "well-known-symbols": "^2.0.0" + } + }, + "configstore": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/configstore/-/configstore-3.1.2.tgz", + "integrity": "sha512-vtv5HtGjcYUgFrXc6Kx747B83MRRVS5R1VTEQoXvuP+kMI+if6uywV0nDGoiydJRy4yk7h9od5Og0kxx4zUXmw==", + "dev": true, + "requires": { + "dot-prop": "^4.1.0", + "graceful-fs": "^4.1.2", + "make-dir": "^1.0.0", + "unique-string": "^1.0.0", + "write-file-atomic": "^2.0.0", + "xdg-basedir": "^3.0.0" + } + }, "content-disposition": { "version": "0.5.2", "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", @@ -1713,9 +2571,18 @@ "dev": true }, "convert-source-map": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.5.1.tgz", - "integrity": "sha1-uCeAl7m8IpNl3lxiz1/K7YtVmeU=" + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.6.0.tgz", + "integrity": "sha512-eFu7XigvxdZ1ETfbgPBohgyQ/Z++C0eEhTor0qRwBw9unw+L0/6V8wkSuGgzdThkiS5lSpdptOQPD8Ak40a+7A==", + "requires": { + "safe-buffer": "~5.1.1" + } + }, + "convert-to-spaces": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/convert-to-spaces/-/convert-to-spaces-1.0.2.tgz", + "integrity": "sha1-fj5Iu+bZl7FBfdyihoIEtNPYVxU=", + "dev": true }, "cookie": { "version": "0.3.1", @@ -1729,6 +2596,12 @@ "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=", "dev": true }, + "copy-descriptor": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", + "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", + "dev": true + }, "core-js": { "version": "2.5.7", "resolved": "https://registry.npmjs.org/core-js/-/core-js-2.5.7.tgz", @@ -1758,6 +2631,51 @@ "elliptic": "^6.0.0" } }, + "create-error-class": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/create-error-class/-/create-error-class-3.0.2.tgz", + "integrity": "sha1-Br56vvlHo/FKMP1hBnHUAbyot7Y=", + "dev": true, + "requires": { + "capture-stack-trace": "^1.0.0" + } + }, + "create-hash": { + "version": "1.2.0", + "resolved": "http://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", + "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", + "requires": { + "cipher-base": "^1.0.1", + "inherits": "^2.0.1", + "md5.js": "^1.3.4", + "ripemd160": "^2.0.1", + "sha.js": "^2.4.0" + } + }, + "create-hmac": { + "version": "1.1.7", + "resolved": "http://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz", + "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==", + "requires": { + "cipher-base": "^1.0.3", + "create-hash": "^1.1.0", + "inherits": "^2.0.1", + "ripemd160": "^2.0.0", + "safe-buffer": "^5.0.1", + "sha.js": "^2.4.8" + } + }, + "cross-spawn": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz", + "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=", + "dev": true, + "requires": { + "lru-cache": "^4.0.1", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + } + }, "crypto-browserify": { "version": "3.12.0", "resolved": "https://registry.npmjs.org/crypto-browserify/-/crypto-browserify-3.12.0.tgz", @@ -1774,70 +2692,36 @@ "public-encrypt": "^4.0.0", "randombytes": "^2.0.0", "randomfill": "^1.0.3" - }, - "dependencies": { - "create-hash": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", - "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", - "requires": { - "cipher-base": "^1.0.1", - "inherits": "^2.0.1", - "md5.js": "^1.3.4", - "ripemd160": "^2.0.1", - "sha.js": "^2.4.0" - } - }, - "create-hmac": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz", - "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==", - "requires": { - "cipher-base": "^1.0.3", - "create-hash": "^1.1.0", - "inherits": "^2.0.1", - "ripemd160": "^2.0.0", - "safe-buffer": "^5.0.1", - "sha.js": "^2.4.8" - } - }, - "diffie-hellman": { - "version": "5.0.3", - "resolved": "https://registry.npmjs.org/diffie-hellman/-/diffie-hellman-5.0.3.tgz", - "integrity": "sha512-kqag/Nl+f3GwyK25fhUMYj81BUOrZ9IuJsjIcDE5icNM9FJHAVm3VcUDxdLPoQtTuUylWm6ZIknYJwwaPxsUzg==", - "requires": { - "bn.js": "^4.1.0", - "miller-rabin": "^4.0.0", - "randombytes": "^2.0.0" - } - }, - "pbkdf2": { - "version": "3.0.16", - "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.0.16.tgz", - "integrity": "sha512-y4CXP3thSxqf7c0qmOF+9UeOTrifiVTIM+u7NWlq+PRsHbr7r7dpCmvzrZxa96JJUNi0Y5w9VqG5ZNeCVMoDcA==", - "requires": { - "create-hash": "^1.1.2", - "create-hmac": "^1.1.4", - "ripemd160": "^2.0.1", - "safe-buffer": "^5.0.1", - "sha.js": "^2.4.8" - } - } } }, + "crypto-random-string": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-1.0.0.tgz", + "integrity": "sha1-ojD2T1aDEOFJgAmUB5DsmVRbyn4=", + "dev": true + }, "cssom": { "version": "0.3.4", "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.3.4.tgz", "integrity": "sha512-+7prCSORpXNeR4/fUP3rL+TzqtiFfhMvTd7uEqMdgPvLPt4+uzFUeufx5RHjGTACCargg/DiEt/moMQmvnfkog==" }, "cssstyle": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-0.3.1.tgz", - "integrity": "sha512-tNvaxM5blOnxanyxI6panOsnfiyLRj3HV4qjqqS45WPNS1usdYWRUQjqTEEELK73lpeP/1KoIGYUwrBn/VcECA==", + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-1.1.1.tgz", + "integrity": "sha512-364AI1l/M5TYcFH83JnOH/pSqgaNnKmYgKrm0didZMGKWjQB60dymwWy1rKUgL3J1ffdq9xVi2yGLHdSjjSNog==", "requires": { "cssom": "0.3.x" } }, + "currently-unhandled": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/currently-unhandled/-/currently-unhandled-0.4.1.tgz", + "integrity": "sha1-mI3zP+qxke95mmE2nddsF635V+o=", + "dev": true, + "requires": { + "array-find-index": "^1.0.1" + } + }, "dashdash": { "version": "1.14.1", "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", @@ -1847,13 +2731,34 @@ } }, "data-urls": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-1.0.0.tgz", - "integrity": "sha512-ai40PPQR0Fn1lD2PPie79CibnlMN2AYiDhwFX/rZHVsxbs5kNJSjegqXIprhouGXlRdEnfybva7kqRGnB6mypA==", + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-1.0.1.tgz", + "integrity": "sha512-0HdcMZzK6ubMUnsMmQmG0AcLQPvbvb47R0+7CCZQCYgcd8OUWG91CG7sM6GoXgjz+WLl4ArFzHtBMy/QqSF4eg==", + "requires": { + "abab": "^2.0.0", + "whatwg-mimetype": "^2.1.0", + "whatwg-url": "^7.0.0" + }, + "dependencies": { + "whatwg-url": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-7.0.0.tgz", + "integrity": "sha512-37GeVSIJ3kn1JgKyjiYNmSLP1yzbpb29jdmwBSgkD9h40/hyrR/OifpVUndji3tmwGgD8qpw7iQu3RSbCrBpsQ==", + "requires": { + "lodash.sortby": "^4.7.0", + "tr46": "^1.0.1", + "webidl-conversions": "^4.0.2" + } + } + } + }, + "date-time": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/date-time/-/date-time-2.1.0.tgz", + "integrity": "sha512-/9+C44X7lot0IeiyfgJmETtRMhBidBYM2QFFIkGa0U1k+hSyY87Nw7PY3eDqpvCBm7I3WCSfPeZskW/YYq6m4g==", + "dev": true, "requires": { - "abab": "^1.0.4", - "whatwg-mimetype": "^2.0.0", - "whatwg-url": "^6.4.0" + "time-zone": "^1.0.0" } }, "deasync": { @@ -1874,11 +2779,70 @@ "ms": "2.0.0" } }, + "decamelize": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", + "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", + "dev": true + }, + "decamelize-keys": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/decamelize-keys/-/decamelize-keys-1.1.0.tgz", + "integrity": "sha1-0XGoeTMlKAfrPLYdwcFEXQeN8tk=", + "dev": true, + "requires": { + "decamelize": "^1.1.0", + "map-obj": "^1.0.0" + }, + "dependencies": { + "map-obj": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/map-obj/-/map-obj-1.0.1.tgz", + "integrity": "sha1-2TPOuSBdgr3PSIb2dCvcK03qFG0=", + "dev": true + } + } + }, + "decode-uri-component": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", + "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", + "dev": true + }, + "deep-equal": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-1.0.1.tgz", + "integrity": "sha1-9dJgKStmDghO/0zbyfCK0yR0SLU=", + "dev": true + }, + "deep-extend": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", + "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", + "dev": true + }, "deep-is": { "version": "0.1.3", "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=" }, + "defaults": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.3.tgz", + "integrity": "sha1-xlYFHpgX2f8I7YgUd/P+QBnz730=", + "dev": true, + "requires": { + "clone": "^1.0.2" + }, + "dependencies": { + "clone": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz", + "integrity": "sha1-2jCcwmPfFZlMaIypAheco8fNfH4=", + "dev": true + } + } + }, "deferred-leveldown": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/deferred-leveldown/-/deferred-leveldown-0.2.0.tgz", @@ -1887,6 +2851,96 @@ "abstract-leveldown": "~0.12.1" } }, + "define-property": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-2.0.2.tgz", + "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", + "dev": true, + "requires": { + "is-descriptor": "^1.0.2", + "isobject": "^3.0.1" + }, + "dependencies": { + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "kind-of": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz", + "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==", + "dev": true + } + } + }, + "del": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/del/-/del-3.0.0.tgz", + "integrity": "sha1-U+z2mf/LyzljdpGrE7rxYIGXZuU=", + "dev": true, + "requires": { + "globby": "^6.1.0", + "is-path-cwd": "^1.0.0", + "is-path-in-cwd": "^1.0.0", + "p-map": "^1.1.1", + "pify": "^3.0.0", + "rimraf": "^2.2.8" + }, + "dependencies": { + "globby": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-6.1.0.tgz", + "integrity": "sha1-9abXDoOV4hyFj7BInWTfAkJNUGw=", + "dev": true, + "requires": { + "array-union": "^1.0.1", + "glob": "^7.0.3", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + }, + "dependencies": { + "pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", + "dev": true + } + } + } + } + }, "delayed-stream": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", @@ -1927,6 +2981,26 @@ "integrity": "sha512-MKPHZDMB0o6yHyDryUOScqZibp914ksXwAMYMTHj6KO8UeKsRYNJD3oNCKjTqZon+V488P7N/HzXF8t7ZR95ww==", "dev": true }, + "diffie-hellman": { + "version": "5.0.3", + "resolved": "http://registry.npmjs.org/diffie-hellman/-/diffie-hellman-5.0.3.tgz", + "integrity": "sha512-kqag/Nl+f3GwyK25fhUMYj81BUOrZ9IuJsjIcDE5icNM9FJHAVm3VcUDxdLPoQtTuUylWm6ZIknYJwwaPxsUzg==", + "requires": { + "bn.js": "^4.1.0", + "miller-rabin": "^4.0.0", + "randombytes": "^2.0.0" + } + }, + "dir-glob": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.0.0.tgz", + "integrity": "sha512-37qirFDz8cA5fimp9feo43fSuRo2gHwaIn6dXL8Ber1dGwUosDrGZeCCXq57WnIqE4aQ+u3eQZzsk1yOzhdwag==", + "dev": true, + "requires": { + "arrify": "^1.0.1", + "path-type": "^3.0.0" + } + }, "domexception": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/domexception/-/domexception-1.0.1.tgz", @@ -1935,13 +3009,28 @@ "webidl-conversions": "^4.0.2" } }, + "dot-prop": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-4.2.0.tgz", + "integrity": "sha512-tUMXrxlExSW6U2EXiiKGSBVdYgtV8qlHL+C10TsW4PURY/ic+eaysnSkwB4kA/mBlCyy/IKDJ+Lc3wbWeaXtuQ==", + "dev": true, + "requires": { + "is-obj": "^1.0.0" + } + }, + "duplexer3": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", + "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", + "dev": true + }, "ecc-jsbn": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.1.tgz", - "integrity": "sha1-D8c6ntXw1Tw4GTOYUj735UN3dQU=", - "optional": true, + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", + "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", "requires": { - "jsbn": "~0.1.0" + "jsbn": "~0.1.0", + "safer-buffer": "^2.1.0" } }, "ee-first": { @@ -1951,14 +3040,14 @@ "dev": true }, "electron-to-chromium": { - "version": "1.3.52", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.52.tgz", - "integrity": "sha1-0tnxJwuko7lnuDHEDvcftNmrXOA=" + "version": "1.3.80", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.80.tgz", + "integrity": "sha512-WClidEWEUNx7OfwXehB0qaxCuetjbKjev2SmXWgybWPLKAThBiMTF/2Pd8GSUDtoGOavxVzdkKwfFAPRSWlkLw==" }, "elliptic": { - "version": "6.4.0", - "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.4.0.tgz", - "integrity": "sha1-ysmvh2LIWDYYcAPI3+GT5eLq5d8=", + "version": "6.4.1", + "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.4.1.tgz", + "integrity": "sha512-BsXLz5sqX8OHcsh7CqBMztyXARmGQ3LWPtGjJi6DiJHq5C/qvi9P3OqgswKSDftbu8+IoI/QDTAm2fFnQ9SZSQ==", "requires": { "bn.js": "^4.4.0", "brorand": "^1.0.1", @@ -1974,12 +3063,34 @@ "resolved": "https://registry.npmjs.org/ember-cli-string-utils/-/ember-cli-string-utils-1.1.0.tgz", "integrity": "sha1-ObZ3/CgF9VFzc1N2/O8njqpEUqE=" }, + "emittery": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.4.1.tgz", + "integrity": "sha512-r4eRSeStEGf6M5SKdrQhhLK5bOwOBxQhIE3YSTnZE3GpKiLfnnhE+tPtrJE79+eDJgm39BM6LSoI8SCx4HbwlQ==", + "dev": true + }, + "empower-core": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/empower-core/-/empower-core-1.2.0.tgz", + "integrity": "sha512-g6+K6Geyc1o6FdXs9HwrXleCFan7d66G5xSCfSF7x1mJDCes6t0om9lFQG3zOrzh3Bkb/45N0cZ5Gqsf7YrzGQ==", + "dev": true, + "requires": { + "call-signature": "0.0.2", + "core-js": "^2.0.0" + } + }, "encodeurl": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=", "dev": true }, + "equal-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/equal-length/-/equal-length-1.0.1.tgz", + "integrity": "sha1-IcoRLUirJLTh5//A5TOdMf38J0w=", + "dev": true + }, "errno": { "version": "0.1.7", "resolved": "https://registry.npmjs.org/errno/-/errno-0.1.7.tgz", @@ -1988,6 +3099,21 @@ "prr": "~1.0.1" } }, + "error-ex": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", + "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "dev": true, + "requires": { + "is-arrayish": "^0.2.1" + } + }, + "es6-error": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/es6-error/-/es6-error-4.1.1.tgz", + "integrity": "sha512-Um/+FxMr9CISWh0bi5Zv0iOD+4cFh5qLeks1qhAopKVAJw3drgKbKySikp7wGhDL0HPeaja0P5ULZrxLkniUVg==", + "dev": true + }, "escape-html": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", @@ -1999,16 +3125,65 @@ "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=" }, + "escodegen": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.11.0.tgz", + "integrity": "sha512-IeMV45ReixHS53K/OmfKAIztN/igDHzTJUhZM3k1jMhIZWjk45SMwAtBsEXiJp3vSPmTcu6CXn7mDvFHRN66fw==", + "requires": { + "esprima": "^3.1.3", + "estraverse": "^4.2.0", + "esutils": "^2.0.2", + "optionator": "^0.8.1", + "source-map": "~0.6.1" + }, + "dependencies": { + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "optional": true + } + } + }, "esm": { - "version": "3.0.26", - "resolved": "https://registry.npmjs.org/esm/-/esm-3.0.26.tgz", - "integrity": "sha512-AgtiHp4RZLtjnCGahgglLyRtha9SippxmO6JI8MCwJPpslzJqOANmNBI33u7v6znU4abVhhY0qb4v0OTBi2ykA==" + "version": "3.0.84", + "resolved": "https://registry.npmjs.org/esm/-/esm-3.0.84.tgz", + "integrity": "sha512-SzSGoZc17S7P+12R9cg21Bdb7eybX25RnIeRZ80xZs+VZ3kdQKzqTp2k4hZJjR7p9l0186TTXSgrxzlMDBktlw==" + }, + "espower-location-detector": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/espower-location-detector/-/espower-location-detector-1.0.0.tgz", + "integrity": "sha1-oXt+zFnTDheeK+9z+0E3cEyzMbU=", + "dev": true, + "requires": { + "is-url": "^1.2.1", + "path-is-absolute": "^1.0.0", + "source-map": "^0.5.0", + "xtend": "^4.0.0" + }, + "dependencies": { + "xtend": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.1.tgz", + "integrity": "sha1-pcbVMr5lbiPbgg77lDofBJmNY68=", + "dev": true + } + } }, "esprima": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/esprima/-/esprima-3.1.3.tgz", "integrity": "sha1-/cpRzuYTOJXjyI1TXOSdv/YqRjM=" }, + "espurify": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/espurify/-/espurify-1.8.1.tgz", + "integrity": "sha512-ZDko6eY/o+D/gHCWyHTU85mKDgYcS4FJj7S+YD6WIInm7GQ6AnOjmcL4+buFV/JOztVLELi/7MmuGU5NHta0Mg==", + "dev": true, + "requires": { + "core-js": "^2.0.0" + } + }, "estraverse": { "version": "4.2.0", "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.2.0.tgz", @@ -2039,6 +3214,21 @@ "safe-buffer": "^5.1.1" } }, + "execa": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-0.7.0.tgz", + "integrity": "sha1-lEvs00zEHuMqY6n68nrVpl/Fl3c=", + "dev": true, + "requires": { + "cross-spawn": "^5.0.1", + "get-stream": "^3.0.0", + "is-stream": "^1.1.0", + "npm-run-path": "^2.0.0", + "p-finally": "^1.0.0", + "signal-exit": "^3.0.0", + "strip-eof": "^1.0.0" + } + }, "expand-brackets": { "version": "0.1.5", "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-0.1.5.tgz", @@ -2056,14 +3246,14 @@ } }, "express": { - "version": "4.16.3", - "resolved": "https://registry.npmjs.org/express/-/express-4.16.3.tgz", - "integrity": "sha1-avilAjUNsyRuzEvs9rWjTSL37VM=", + "version": "4.16.4", + "resolved": "https://registry.npmjs.org/express/-/express-4.16.4.tgz", + "integrity": "sha512-j12Uuyb4FMrd/qQAm6uCHAkPtO8FDTRJZBDd5D2KOL2eLaz1yUNdUB/NOIyq0iU4q4cFarsUCrnFDPBcnksuOg==", "dev": true, "requires": { "accepts": "~1.3.5", "array-flatten": "1.1.1", - "body-parser": "1.18.2", + "body-parser": "1.18.3", "content-disposition": "0.5.2", "content-type": "~1.0.4", "cookie": "0.3.1", @@ -2080,10 +3270,10 @@ "on-finished": "~2.3.0", "parseurl": "~1.3.2", "path-to-regexp": "0.1.7", - "proxy-addr": "~2.0.3", - "qs": "6.5.1", + "proxy-addr": "~2.0.4", + "qs": "6.5.2", "range-parser": "~1.2.0", - "safe-buffer": "5.1.1", + "safe-buffer": "5.1.2", "send": "0.16.2", "serve-static": "1.13.2", "setprototypeof": "1.1.0", @@ -2091,170 +3281,729 @@ "type-is": "~1.6.16", "utils-merge": "1.0.1", "vary": "~1.1.2" + } + }, + "extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==" + }, + "extend-shallow": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", + "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", + "dev": true, + "requires": { + "assign-symbols": "^1.0.0", + "is-extendable": "^1.0.1" }, "dependencies": { - "qs": { - "version": "6.5.1", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz", - "integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A==", - "dev": true - }, - "safe-buffer": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz", - "integrity": "sha512-kKvNJn6Mm93gAczWVJg7wH+wGYWNrDHdWvpUmHyEsgCtIwwo3bqPtV4tR5tuPaUhTOo/kvhVwd8XwwOllGYkbg==", - "dev": true + "is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, + "requires": { + "is-plain-object": "^2.0.4" + } + } + } + }, + "extglob": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/extglob/-/extglob-0.3.2.tgz", + "integrity": "sha1-Lhj/PS9JqydlzskCPwEdqo2DSaE=", + "requires": { + "is-extglob": "^1.0.0" + } + }, + "extsprintf": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", + "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=" + }, + "fake-xml-http-request": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/fake-xml-http-request/-/fake-xml-http-request-1.6.0.tgz", + "integrity": "sha512-99XPwwSg89BfzPuv4XCpZxn3EbauMCgAQCxq9MzrvS6DFD73OON6AnUTicL4A0HZtYMBwCZBWVnRqGjZDgQkTg==" + }, + "faker": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/faker/-/faker-4.1.0.tgz", + "integrity": "sha1-HkW7vsxndLPBlfrSg1EJxtdIzD8=", + "dev": true + }, + "fast-deep-equal": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz", + "integrity": "sha1-wFNHeBfIa1HaqFPIHgWbcz0CNhQ=" + }, + "fast-diff": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.2.0.tgz", + "integrity": "sha512-xJuoT5+L99XlZ8twedaRf6Ax2TgQVxvgZOYoPKqZufmJib0tL2tegPBOZb1pVNgIhlqDlA0eO0c3wBvQcmzx4w==", + "dev": true + }, + "fast-json-stable-stringify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", + "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=" + }, + "fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=" + }, + "figures": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + }, + "filename-regex": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/filename-regex/-/filename-regex-2.0.1.tgz", + "integrity": "sha1-wcS5vuPglyXdsQa3XB4wH+LxiyY=" + }, + "fill-range": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-2.2.4.tgz", + "integrity": "sha512-cnrcCbj01+j2gTG921VZPnHbjmdAf8oQV/iGeV2kZxGSyfYjjTyY79ErsK1WJWMpw6DaApEX72binqJE+/d+5Q==", + "requires": { + "is-number": "^2.1.0", + "isobject": "^2.0.0", + "randomatic": "^3.0.0", + "repeat-element": "^1.1.2", + "repeat-string": "^1.5.2" + } + }, + "finalhandler": { + "version": "1.1.1", + "resolved": "http://registry.npmjs.org/finalhandler/-/finalhandler-1.1.1.tgz", + "integrity": "sha512-Y1GUDo39ez4aHAw7MysnUD5JzYX+WaIj8I57kO3aEPT1fFRL4sr7mjei97FgnwhAyyzRYmQZaTHb2+9uZ1dPtg==", + "dev": true, + "requires": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.2", + "statuses": "~1.4.0", + "unpipe": "~1.0.0" + } + }, + "find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "requires": { + "locate-path": "^3.0.0" + } + }, + "for-in": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", + "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=" + }, + "for-own": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/for-own/-/for-own-0.1.5.tgz", + "integrity": "sha1-UmXGgaTylNq78XyVCbZ2OqhFEM4=", + "requires": { + "for-in": "^1.0.1" + } + }, + "foreach": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/foreach/-/foreach-2.0.5.tgz", + "integrity": "sha1-C+4AUBiusmDQo6865ljdATbsG5k=" + }, + "forever-agent": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", + "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=" + }, + "form-data": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", + "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.6", + "mime-types": "^2.1.12" + } + }, + "forwarded": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", + "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=", + "dev": true + }, + "fragment-cache": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", + "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=", + "dev": true, + "requires": { + "map-cache": "^0.2.2" + } + }, + "fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=", + "dev": true + }, + "fs-extra": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.0.tgz", + "integrity": "sha512-EglNDLRpmaTWiD/qraZn6HREAEAHJcJOmxNEYwq6xeMKnVMAy3GUcFB+wXt2C6k4CNvB/mP1y/U3dzvKKj5OtQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", + "dev": true + }, + "fsevents": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.4.tgz", + "integrity": "sha512-z8H8/diyk76B7q5wg+Ud0+CqzcAF3mBBI/bA5ne5zrRUUIvNkJY//D3BqyH571KuAC4Nr7Rw7CjWX4r0y9DvNg==", + "dev": true, + "optional": true, + "requires": { + "nan": "^2.9.2", + "node-pre-gyp": "^0.10.0" + }, + "dependencies": { + "abbrev": { + "version": "1.1.1", + "bundled": true, + "dev": true, + "optional": true + }, + "ansi-regex": { + "version": "2.1.1", + "bundled": true, + "dev": true + }, + "aproba": { + "version": "1.2.0", + "bundled": true, + "dev": true, + "optional": true + }, + "are-we-there-yet": { + "version": "1.1.4", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "delegates": "^1.0.0", + "readable-stream": "^2.0.6" + } + }, + "balanced-match": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "brace-expansion": { + "version": "1.1.11", + "bundled": true, + "dev": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "chownr": { + "version": "1.0.1", + "bundled": true, + "dev": true, + "optional": true + }, + "code-point-at": { + "version": "1.1.0", + "bundled": true, + "dev": true + }, + "concat-map": { + "version": "0.0.1", + "bundled": true, + "dev": true + }, + "console-control-strings": { + "version": "1.1.0", + "bundled": true, + "dev": true + }, + "core-util-is": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "optional": true + }, + "debug": { + "version": "2.6.9", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "ms": "2.0.0" + } + }, + "deep-extend": { + "version": "0.5.1", + "bundled": true, + "dev": true, + "optional": true + }, + "delegates": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "optional": true + }, + "detect-libc": { + "version": "1.0.3", + "bundled": true, + "dev": true, + "optional": true + }, + "fs-minipass": { + "version": "1.2.5", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "minipass": "^2.2.1" + } + }, + "fs.realpath": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "optional": true + }, + "gauge": { + "version": "2.7.4", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "aproba": "^1.0.3", + "console-control-strings": "^1.0.0", + "has-unicode": "^2.0.0", + "object-assign": "^4.1.0", + "signal-exit": "^3.0.0", + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wide-align": "^1.1.0" + } + }, + "glob": { + "version": "7.1.2", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "has-unicode": { + "version": "2.0.1", + "bundled": true, + "dev": true, + "optional": true + }, + "iconv-lite": { + "version": "0.4.21", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "safer-buffer": "^2.1.0" + } + }, + "ignore-walk": { + "version": "3.0.1", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "minimatch": "^3.0.4" + } + }, + "inflight": { + "version": "1.0.6", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.3", + "bundled": true, + "dev": true + }, + "ini": { + "version": "1.3.5", + "bundled": true, + "dev": true, + "optional": true + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "isarray": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "optional": true + }, + "minimatch": { + "version": "3.0.4", + "bundled": true, + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "minimist": { + "version": "0.0.8", + "bundled": true, + "dev": true + }, + "minipass": { + "version": "2.2.4", + "bundled": true, + "dev": true, + "requires": { + "safe-buffer": "^5.1.1", + "yallist": "^3.0.0" + } + }, + "minizlib": { + "version": "1.1.0", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "minipass": "^2.2.1" + } + }, + "mkdirp": { + "version": "0.5.1", + "bundled": true, + "dev": true, + "requires": { + "minimist": "0.0.8" + } + }, + "ms": { + "version": "2.0.0", + "bundled": true, + "dev": true, + "optional": true + }, + "needle": { + "version": "2.2.0", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "debug": "^2.1.2", + "iconv-lite": "^0.4.4", + "sax": "^1.2.4" + } + }, + "node-pre-gyp": { + "version": "0.10.0", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "detect-libc": "^1.0.2", + "mkdirp": "^0.5.1", + "needle": "^2.2.0", + "nopt": "^4.0.1", + "npm-packlist": "^1.1.6", + "npmlog": "^4.0.2", + "rc": "^1.1.7", + "rimraf": "^2.6.1", + "semver": "^5.3.0", + "tar": "^4" + } + }, + "nopt": { + "version": "4.0.1", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "abbrev": "1", + "osenv": "^0.1.4" + } + }, + "npm-bundled": { + "version": "1.0.3", + "bundled": true, + "dev": true, + "optional": true + }, + "npm-packlist": { + "version": "1.1.10", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "ignore-walk": "^3.0.1", + "npm-bundled": "^1.0.1" + } + }, + "npmlog": { + "version": "4.1.2", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "are-we-there-yet": "~1.1.2", + "console-control-strings": "~1.1.0", + "gauge": "~2.7.3", + "set-blocking": "~2.0.0" + } + }, + "number-is-nan": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "object-assign": { + "version": "4.1.1", + "bundled": true, + "dev": true, + "optional": true + }, + "once": { + "version": "1.4.0", + "bundled": true, + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "os-homedir": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "optional": true + }, + "os-tmpdir": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "optional": true + }, + "osenv": { + "version": "0.1.5", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "os-homedir": "^1.0.0", + "os-tmpdir": "^1.0.0" + } + }, + "path-is-absolute": { + "version": "1.0.1", + "bundled": true, + "dev": true, + "optional": true + }, + "process-nextick-args": { + "version": "2.0.0", + "bundled": true, + "dev": true, + "optional": true + }, + "rc": { + "version": "1.2.7", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "deep-extend": "^0.5.1", + "ini": "~1.3.0", + "minimist": "^1.2.0", + "strip-json-comments": "~2.0.1" + }, + "dependencies": { + "minimist": { + "version": "1.2.0", + "bundled": true, + "dev": true, + "optional": true + } + } + }, + "readable-stream": { + "version": "2.3.6", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "rimraf": { + "version": "2.6.2", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "glob": "^7.0.5" + } + }, + "safe-buffer": { + "version": "5.1.1", + "bundled": true, + "dev": true + }, + "safer-buffer": { + "version": "2.1.2", + "bundled": true, + "dev": true, + "optional": true + }, + "sax": { + "version": "1.2.4", + "bundled": true, + "dev": true, + "optional": true + }, + "semver": { + "version": "5.5.0", + "bundled": true, + "dev": true, + "optional": true + }, + "set-blocking": { + "version": "2.0.0", + "bundled": true, + "dev": true, + "optional": true + }, + "signal-exit": { + "version": "3.0.2", + "bundled": true, + "dev": true, + "optional": true + }, + "string-width": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + }, + "string_decoder": { + "version": "1.1.1", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "safe-buffer": "~5.1.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "bundled": true, + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "strip-json-comments": { + "version": "2.0.1", + "bundled": true, + "dev": true, + "optional": true + }, + "tar": { + "version": "4.4.1", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "chownr": "^1.0.1", + "fs-minipass": "^1.2.5", + "minipass": "^2.2.4", + "minizlib": "^1.1.0", + "mkdirp": "^0.5.0", + "safe-buffer": "^5.1.1", + "yallist": "^3.0.2" + } }, - "send": { - "version": "0.16.2", - "resolved": "https://registry.npmjs.org/send/-/send-0.16.2.tgz", - "integrity": "sha512-E64YFPUssFHEFBvpbbjr44NCLtI1AohxQ8ZSiJjQLskAdKuriYEP6VyGEsRDH8ScozGpkaX1BGvhanqCwkcEZw==", + "util-deprecate": { + "version": "1.0.2", + "bundled": true, "dev": true, + "optional": true + }, + "wide-align": { + "version": "1.1.2", + "bundled": true, + "dev": true, + "optional": true, "requires": { - "debug": "2.6.9", - "depd": "~1.1.2", - "destroy": "~1.0.4", - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "etag": "~1.8.1", - "fresh": "0.5.2", - "http-errors": "~1.6.2", - "mime": "1.4.1", - "ms": "2.0.0", - "on-finished": "~2.3.0", - "range-parser": "~1.2.0", - "statuses": "~1.4.0" + "string-width": "^1.0.2" } + }, + "wrappy": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "yallist": { + "version": "3.0.2", + "bundled": true, + "dev": true } } }, - "extend": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.1.tgz", - "integrity": "sha1-p1Xqe8Gt/MWjHOfnYtuq3F5jZEQ=" - }, - "extglob": { - "version": "0.3.2", - "resolved": "https://registry.npmjs.org/extglob/-/extglob-0.3.2.tgz", - "integrity": "sha1-Lhj/PS9JqydlzskCPwEdqo2DSaE=", - "requires": { - "is-extglob": "^1.0.0" - } - }, - "extsprintf": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", - "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=" - }, - "fake-xml-http-request": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/fake-xml-http-request/-/fake-xml-http-request-1.6.0.tgz", - "integrity": "sha512-99XPwwSg89BfzPuv4XCpZxn3EbauMCgAQCxq9MzrvS6DFD73OON6AnUTicL4A0HZtYMBwCZBWVnRqGjZDgQkTg==" - }, - "faker": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/faker/-/faker-4.1.0.tgz", - "integrity": "sha1-HkW7vsxndLPBlfrSg1EJxtdIzD8=", - "dev": true - }, - "fast-deep-equal": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz", - "integrity": "sha1-wFNHeBfIa1HaqFPIHgWbcz0CNhQ=" - }, - "fast-json-stable-stringify": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", - "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=" - }, - "fast-levenshtein": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", - "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=" - }, - "filename-regex": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/filename-regex/-/filename-regex-2.0.1.tgz", - "integrity": "sha1-wcS5vuPglyXdsQa3XB4wH+LxiyY=" - }, - "fill-range": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-2.2.4.tgz", - "integrity": "sha512-cnrcCbj01+j2gTG921VZPnHbjmdAf8oQV/iGeV2kZxGSyfYjjTyY79ErsK1WJWMpw6DaApEX72binqJE+/d+5Q==", - "requires": { - "is-number": "^2.1.0", - "isobject": "^2.0.0", - "randomatic": "^3.0.0", - "repeat-element": "^1.1.2", - "repeat-string": "^1.5.2" - } - }, - "finalhandler": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.1.tgz", - "integrity": "sha512-Y1GUDo39ez4aHAw7MysnUD5JzYX+WaIj8I57kO3aEPT1fFRL4sr7mjei97FgnwhAyyzRYmQZaTHb2+9uZ1dPtg==", - "dev": true, - "requires": { - "debug": "2.6.9", - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "on-finished": "~2.3.0", - "parseurl": "~1.3.2", - "statuses": "~1.4.0", - "unpipe": "~1.0.0" - } - }, - "for-in": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", - "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=" - }, - "for-own": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/for-own/-/for-own-0.1.5.tgz", - "integrity": "sha1-UmXGgaTylNq78XyVCbZ2OqhFEM4=", - "requires": { - "for-in": "^1.0.1" - } - }, - "foreach": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/foreach/-/foreach-2.0.5.tgz", - "integrity": "sha1-C+4AUBiusmDQo6865ljdATbsG5k=" - }, - "forever-agent": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", - "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=" - }, - "form-data": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.2.tgz", - "integrity": "sha1-SXBJi+YEwgwAXU9cI67NIda0kJk=", - "requires": { - "asynckit": "^0.4.0", - "combined-stream": "1.0.6", - "mime-types": "^2.1.12" - } - }, - "forwarded": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", - "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=", - "dev": true - }, - "fresh": { - "version": "0.5.2", - "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", - "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=", - "dev": true - }, - "fs.realpath": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", - "dev": true - }, "fwd-stream": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/fwd-stream/-/fwd-stream-1.0.4.tgz", @@ -2270,7 +4019,7 @@ }, "readable-stream": { "version": "1.0.34", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz", + "resolved": "http://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz", "integrity": "sha1-Elgg40vIQtLyqq+v5MKRbuMsFXw=", "requires": { "core-util-is": "~1.0.0", @@ -2286,6 +4035,24 @@ } } }, + "get-port": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/get-port/-/get-port-4.0.0.tgz", + "integrity": "sha512-Yy3yNI2oShgbaWg4cmPhWjkZfktEvpKI09aDX4PZzNtlU9obuYrX7x2mumQsrNxlF+Ls7OtMQW/u+X4s896bOQ==", + "dev": true + }, + "get-stream": { + "version": "3.0.0", + "resolved": "http://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz", + "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=", + "dev": true + }, + "get-value": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", + "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=", + "dev": true + }, "getpass": { "version": "0.1.7", "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", @@ -2295,9 +4062,9 @@ } }, "glob": { - "version": "7.1.2", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz", - "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==", + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", + "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", "dev": true, "requires": { "fs.realpath": "^1.0.0", @@ -2325,11 +4092,59 @@ "is-glob": "^2.0.0" } }, + "global-dirs": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-0.1.1.tgz", + "integrity": "sha1-sxnA3UYH81PzvpzKTHL8FIxJ9EU=", + "dev": true, + "requires": { + "ini": "^1.3.4" + } + }, "globals": { "version": "9.18.0", "resolved": "https://registry.npmjs.org/globals/-/globals-9.18.0.tgz", "integrity": "sha512-S0nG3CLEQiY/ILxqtztTWH/3iRRdyBLw6KMDxnKMchrtbj2OFmehVh0WUCfW3DUrIgx/qFrJPICrq4Z4sTR9UQ==" }, + "globby": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/globby/-/globby-7.1.1.tgz", + "integrity": "sha1-+yzP+UAfhgCUXfral0QMypcrhoA=", + "dev": true, + "requires": { + "array-union": "^1.0.1", + "dir-glob": "^2.0.0", + "glob": "^7.1.2", + "ignore": "^3.3.5", + "pify": "^3.0.0", + "slash": "^1.0.0" + } + }, + "got": { + "version": "6.7.1", + "resolved": "http://registry.npmjs.org/got/-/got-6.7.1.tgz", + "integrity": "sha1-JAzQV4WpoY5WHcG0S0HHY+8ejbA=", + "dev": true, + "requires": { + "create-error-class": "^3.0.0", + "duplexer3": "^0.1.4", + "get-stream": "^3.0.0", + "is-redirect": "^1.0.0", + "is-retry-allowed": "^1.0.0", + "is-stream": "^1.0.0", + "lowercase-keys": "^1.0.0", + "safe-buffer": "^5.0.1", + "timed-out": "^4.0.0", + "unzip-response": "^2.0.1", + "url-parse-lax": "^1.0.0" + } + }, + "graceful-fs": { + "version": "4.1.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz", + "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=", + "dev": true + }, "growl": { "version": "1.10.3", "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.3.tgz", @@ -2342,11 +4157,11 @@ "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=" }, "har-validator": { - "version": "5.0.3", - "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.0.3.tgz", - "integrity": "sha1-ukAsJmGU8VlW7xXg/PJCmT9qff0=", + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.0.tgz", + "integrity": "sha512-+qnmNjI4OfH2ipQ9VQOw23bBd/ibtfbVdK2fYbY4acTDqKTW/YDp9McimZdDbG8iV9fZizUqQMD5xvriB146TA==", "requires": { - "ajv": "^5.1.0", + "ajv": "^5.3.0", "har-schema": "^2.0.0" } }, @@ -2363,6 +4178,66 @@ "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=" }, + "has-value": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz", + "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=", + "dev": true, + "requires": { + "get-value": "^2.0.6", + "has-values": "^1.0.0", + "isobject": "^3.0.0" + }, + "dependencies": { + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "has-values": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz", + "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=", + "dev": true, + "requires": { + "is-number": "^3.0.0", + "kind-of": "^4.0.0" + }, + "dependencies": { + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "kind-of": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", + "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, "hash-base": { "version": "3.0.4", "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.0.4.tgz", @@ -2406,6 +4281,12 @@ "os-tmpdir": "^1.0.1" } }, + "hosted-git-info": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.7.1.tgz", + "integrity": "sha512-7T/BxH19zbcCTa8XkMlbK5lTo1WtgkFi3GvdWEyNuc4Vex7/9Dqbnpsf4JMydcfj9HCg4zUWFTL3Za6lapg5/w==", + "dev": true + }, "html-encoding-sniffer": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-1.0.2.tgz", @@ -2416,7 +4297,7 @@ }, "http-errors": { "version": "1.6.3", - "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", + "resolved": "http://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", "dev": true, "requires": { @@ -2426,21 +4307,74 @@ "statuses": ">= 1.4.0 < 2" } }, + "http-signature": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", + "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", + "requires": { + "assert-plus": "^1.0.0", + "jsprim": "^1.2.2", + "sshpk": "^1.7.0" + } + }, "i": { "version": "0.3.6", "resolved": "https://registry.npmjs.org/i/-/i-0.3.6.tgz", "integrity": "sha1-2WyScyB28HJxG2sQ/X1PZa2O4j0=" }, "iconv-lite": { - "version": "0.4.19", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.19.tgz", - "integrity": "sha512-oTZqweIP51xaGPI4uPa56/Pri/480R+mo7SeU+YETByQNhDG55ycFyNLIgta9vXhILrxXDmF7ZGhqZIcuN0gJQ==" + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } }, "idb-wrapper": { "version": "1.7.2", "resolved": "https://registry.npmjs.org/idb-wrapper/-/idb-wrapper-1.7.2.tgz", "integrity": "sha512-zfNREywMuf0NzDo9mVsL0yegjsirJxHpKHvWcyRozIqQy89g0a3U+oBPOCN4cc0oCiOuYgZHimzaW/R46G1Mpg==" }, + "ignore": { + "version": "3.3.10", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz", + "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", + "dev": true + }, + "ignore-by-default": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", + "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", + "dev": true + }, + "import-lazy": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", + "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", + "dev": true + }, + "import-local": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-2.0.0.tgz", + "integrity": "sha512-b6s04m3O+s3CGSbqDIyP4R6aAwAeYlVq9+WUWep6iHa8ETRf9yei1U48C5MmfJmV9AiLYYBKPMq/W+/WRpQmCQ==", + "dev": true, + "requires": { + "pkg-dir": "^3.0.0", + "resolve-cwd": "^2.0.0" + } + }, + "imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", + "dev": true + }, + "indent-string": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-3.2.0.tgz", + "integrity": "sha1-Sl/W0nzDMvN+VBmlBNu4NxBckok=", + "dev": true + }, "indexof": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/indexof/-/indexof-0.0.1.tgz", @@ -2461,22 +4395,120 @@ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" }, + "ini": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", + "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==", + "dev": true + }, + "invariant": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", + "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "requires": { + "loose-envify": "^1.0.0" + } + }, "ipaddr.js": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.6.0.tgz", - "integrity": "sha1-4/o1e3c9phnybpXwSdBVxyeW+Gs=", + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.8.0.tgz", + "integrity": "sha1-6qM9bd16zo9/b+DJygRA5wZzix4=", + "dev": true + }, + "irregular-plurals": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/irregular-plurals/-/irregular-plurals-2.0.0.tgz", + "integrity": "sha512-Y75zBYLkh0lJ9qxeHlMjQ7bSbyiSqNW/UOPWDmzC7cXskL1hekSITh1Oc6JV0XCWWZ9DE8VYSB71xocLk3gmGw==", "dev": true }, "is": { "version": "0.2.7", - "resolved": "https://registry.npmjs.org/is/-/is-0.2.7.tgz", + "resolved": "http://registry.npmjs.org/is/-/is-0.2.7.tgz", "integrity": "sha1-OzSixI81mXLzUEKEkZOucmS2NWI=" }, + "is-accessor-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", + "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", + "dev": true + }, + "is-binary-path": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", + "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", + "dev": true, + "requires": { + "binary-extensions": "^1.0.0" + } + }, "is-buffer": { "version": "1.1.6", "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==" }, + "is-builtin-module": { + "version": "1.0.0", + "resolved": "http://registry.npmjs.org/is-builtin-module/-/is-builtin-module-1.0.0.tgz", + "integrity": "sha1-VAVy0096wxGfj3bDDLwbHgN6/74=", + "dev": true, + "requires": { + "builtin-modules": "^1.0.0" + }, + "dependencies": { + "builtin-modules": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", + "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", + "dev": true + } + } + }, + "is-ci": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-1.2.1.tgz", + "integrity": "sha512-s6tfsaQaQi3JNciBH6shVqEDvhGut0SUXr31ag8Pd8BBbVVlcGfWhpPmEOoM6RJ5TFhbypvf5yyRw/VXW1IiWg==", + "dev": true, + "requires": { + "ci-info": "^1.5.0" + } + }, + "is-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "is-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", + "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + }, + "dependencies": { + "kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "dev": true + } + } + }, "is-dotfile": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/is-dotfile/-/is-dotfile-1.0.3.tgz", @@ -2490,6 +4522,12 @@ "is-primitive": "^2.0.0" } }, + "is-error": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/is-error/-/is-error-2.2.1.tgz", + "integrity": "sha1-aEqW2EB2V3yY9M20DG0mpRI78Zw=", + "dev": true + }, "is-extendable": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", @@ -2508,6 +4546,12 @@ "number-is-nan": "^1.0.0" } }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, "is-glob": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-2.0.1.tgz", @@ -2516,11 +4560,27 @@ "is-extglob": "^1.0.0" } }, + "is-installed-globally": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.1.0.tgz", + "integrity": "sha1-Df2Y9akRFxbdU13aZJL2e/PSWoA=", + "dev": true, + "requires": { + "global-dirs": "^0.1.0", + "is-path-inside": "^1.0.0" + } + }, "is-module": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-module/-/is-module-1.0.0.tgz", "integrity": "sha1-Mlj7afeMFNW4FdZkM2tM/7ZEFZE=" }, + "is-npm": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-1.0.0.tgz", + "integrity": "sha1-8vtjpl5JBbQGyGBydloaTceTufQ=", + "dev": true + }, "is-number": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-2.1.0.tgz", @@ -2529,11 +4589,73 @@ "kind-of": "^3.0.2" } }, + "is-obj": { + "version": "1.0.1", + "resolved": "http://registry.npmjs.org/is-obj/-/is-obj-1.0.1.tgz", + "integrity": "sha1-PkcprB9f3gJc19g6iW2rn09n2w8=", + "dev": true + }, "is-object": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/is-object/-/is-object-0.1.2.tgz", "integrity": "sha1-AO+8CIFsM8/ErIJR0TLhDcZQmNc=" }, + "is-observable": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-observable/-/is-observable-1.1.0.tgz", + "integrity": "sha512-NqCa4Sa2d+u7BWc6CukaObG3Fh+CU9bvixbpcXYhy2VvYS7vVGIdAgnIS5Ks3A/cqk4rebLJ9s8zBstT2aKnIA==", + "dev": true, + "requires": { + "symbol-observable": "^1.1.0" + } + }, + "is-path-cwd": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-1.0.0.tgz", + "integrity": "sha1-0iXsIxMuie3Tj9p2dHLmLmXxEG0=", + "dev": true + }, + "is-path-in-cwd": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-1.0.1.tgz", + "integrity": "sha512-FjV1RTW48E7CWM7eE/J2NJvAEEVektecDBVBE5Hh3nM1Jd0kvhHtX68Pr3xsDf857xt3Y4AkwVULK1Vku62aaQ==", + "dev": true, + "requires": { + "is-path-inside": "^1.0.0" + } + }, + "is-path-inside": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.1.tgz", + "integrity": "sha1-jvW33lBDej/cprToZe96pVy0gDY=", + "dev": true, + "requires": { + "path-is-inside": "^1.0.1" + } + }, + "is-plain-obj": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-1.1.0.tgz", + "integrity": "sha1-caUMhCnfync8kqOQpKA7OfzVHT4=", + "dev": true + }, + "is-plain-object": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", + "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", + "dev": true, + "requires": { + "isobject": "^3.0.1" + }, + "dependencies": { + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, "is-posix-bracket": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/is-posix-bracket/-/is-posix-bracket-0.1.1.tgz", @@ -2544,11 +4666,53 @@ "resolved": "https://registry.npmjs.org/is-primitive/-/is-primitive-2.0.0.tgz", "integrity": "sha1-IHurkWOEmcB7Kt8kCkGochADRXU=" }, + "is-promise": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz", + "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=", + "dev": true + }, + "is-redirect": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-redirect/-/is-redirect-1.0.0.tgz", + "integrity": "sha1-HQPd7VO9jbDzDCbk+V02/HyH3CQ=", + "dev": true + }, + "is-retry-allowed": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-retry-allowed/-/is-retry-allowed-1.1.0.tgz", + "integrity": "sha1-EaBgVotnM5REAz0BJaYaINVk+zQ=", + "dev": true + }, + "is-stream": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", + "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", + "dev": true + }, "is-typedarray": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=" }, + "is-url": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/is-url/-/is-url-1.2.4.tgz", + "integrity": "sha512-ITvGim8FhRiYe4IQ5uHSkj7pVaPDrCTkNd3yq3cV7iZAcJdHTUMPMEHcqSOy9xZ9qFenQCvi+2wjH9a1nXqHww==", + "dev": true + }, + "is-utf8": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz", + "integrity": "sha1-Sw2hRCEE0bM2NA6AeX6GXPOffXI=", + "dev": true + }, + "is-windows": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", + "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", + "dev": true + }, "isarray": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", @@ -2559,6 +4723,12 @@ "resolved": "https://registry.npmjs.org/isbuffer/-/isbuffer-0.0.0.tgz", "integrity": "sha1-OMFG2d9Si4v5sHAcPUPPEt8/w5s=" }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", + "dev": true + }, "isobject": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", @@ -2578,128 +4748,71 @@ "integrity": "sha512-Ubldcmxp5np52/ENotGxlLe6aGMvmF4R8S6tZjsP6Knsaxd/xp3Zrh50cG93lR6nPXyUFwzN3ZSOQI0wRJNdGg==", "dev": true }, + "js-string-escape": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/js-string-escape/-/js-string-escape-1.0.1.tgz", + "integrity": "sha1-4mJbrbwNZ8dTPp7cEGjFh65BN+8=", + "dev": true + }, "js-tokens": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls=" }, + "js-yaml": { + "version": "3.12.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.12.0.tgz", + "integrity": "sha512-PIt2cnwmPfL4hKNwqeiuz4bKfnzHTBv6HyVgjahA6mPLwPDzjDWrplJBMjHUFxku/N3FlmrbyPclad+I+4mJ3A==", + "dev": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "dependencies": { + "esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true + } + } + }, "jsbn": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", - "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", - "optional": true + "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=" }, "jsdom": { - "version": "11.11.0", - "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-11.11.0.tgz", - "integrity": "sha512-ou1VyfjwsSuWkudGxb03FotDajxAto6USAlmMZjE2lc0jCznt7sBWkhfRBRaWwbnmDqdMSTKTLT5d9sBFkkM7A==", + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-11.12.0.tgz", + "integrity": "sha512-y8Px43oyiBM13Zc1z780FrfNLJCXTL40EWlty/LXUtcjykRBNgLlCjWXpfSPBl2iv+N7koQN+dvqszHZgT/Fjw==", "requires": { - "abab": "^1.0.4", - "acorn": "^5.3.0", + "abab": "^2.0.0", + "acorn": "^5.5.3", "acorn-globals": "^4.1.0", "array-equal": "^1.0.0", "cssom": ">= 0.3.2 < 0.4.0", - "cssstyle": ">= 0.3.1 < 0.4.0", + "cssstyle": "^1.0.0", "data-urls": "^1.0.0", - "domexception": "^1.0.0", - "escodegen": "^1.9.0", + "domexception": "^1.0.1", + "escodegen": "^1.9.1", "html-encoding-sniffer": "^1.0.2", - "left-pad": "^1.2.0", - "nwsapi": "^2.0.0", + "left-pad": "^1.3.0", + "nwsapi": "^2.0.7", "parse5": "4.0.0", "pn": "^1.1.0", - "request": "^2.83.0", + "request": "^2.87.0", "request-promise-native": "^1.0.5", "sax": "^1.2.4", "symbol-tree": "^3.2.2", - "tough-cookie": "^2.3.3", + "tough-cookie": "^2.3.4", "w3c-hr-time": "^1.0.1", "webidl-conversions": "^4.0.2", "whatwg-encoding": "^1.0.3", "whatwg-mimetype": "^2.1.0", "whatwg-url": "^6.4.1", - "ws": "^4.0.0", + "ws": "^5.2.0", "xml-name-validator": "^3.0.0" - }, - "dependencies": { - "acorn-globals": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-4.1.0.tgz", - "integrity": "sha512-KjZwU26uG3u6eZcfGbTULzFcsoz6pegNKtHPksZPOUsiKo5bUmiBPa38FuHZ/Eun+XYh/JCCkS9AS3Lu4McQOQ==", - "requires": { - "acorn": "^5.0.0" - } - }, - "escodegen": { - "version": "1.10.0", - "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.10.0.tgz", - "integrity": "sha512-fjUOf8johsv23WuIKdNQU4P9t9jhQ4Qzx6pC2uW890OloK3Zs1ZAoCNpg/2larNF501jLl3UNy0kIRcF6VI22g==", - "requires": { - "esprima": "^3.1.3", - "estraverse": "^4.2.0", - "esutils": "^2.0.2", - "optionator": "^0.8.1", - "source-map": "~0.6.1" - } - }, - "http-signature": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", - "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", - "requires": { - "assert-plus": "^1.0.0", - "jsprim": "^1.2.2", - "sshpk": "^1.7.0" - } - }, - "punycode": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", - "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=" - }, - "request": { - "version": "2.87.0", - "resolved": "https://registry.npmjs.org/request/-/request-2.87.0.tgz", - "integrity": "sha512-fcogkm7Az5bsS6Sl0sibkbhcKsnyon/jV1kF3ajGmF0c8HrttdKTPRT9hieOaQHA5HEq6r8OyWOo/o781C1tNw==", - "requires": { - "aws-sign2": "~0.7.0", - "aws4": "^1.6.0", - "caseless": "~0.12.0", - "combined-stream": "~1.0.5", - "extend": "~3.0.1", - "forever-agent": "~0.6.1", - "form-data": "~2.3.1", - "har-validator": "~5.0.3", - "http-signature": "~1.2.0", - "is-typedarray": "~1.0.0", - "isstream": "~0.1.2", - "json-stringify-safe": "~5.0.1", - "mime-types": "~2.1.17", - "oauth-sign": "~0.8.2", - "performance-now": "^2.1.0", - "qs": "~6.5.1", - "safe-buffer": "^5.1.1", - "tough-cookie": "~2.3.3", - "tunnel-agent": "^0.6.0", - "uuid": "^3.1.0" - }, - "dependencies": { - "tough-cookie": { - "version": "2.3.4", - "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.3.4.tgz", - "integrity": "sha512-TZ6TTfI5NtZnuyy/Kecv+CnoROnyXn2DN97LontgQpCwsX2XyLYCC0ENhYkehSOwAp8rTQKc/NUIF7BkQ5rKLA==", - "requires": { - "punycode": "^1.4.1" - } - } - } - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "optional": true - } } }, "jsesc": { @@ -2707,6 +4820,12 @@ "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-0.5.0.tgz", "integrity": "sha1-597mbjXW/Bb3EP6R1c9p9w8IkR0=" }, + "json-parse-better-errors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", + "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", + "dev": true + }, "json-schema": { "version": "0.2.3", "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", @@ -2724,9 +4843,18 @@ }, "json5": { "version": "0.5.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-0.5.1.tgz", + "resolved": "http://registry.npmjs.org/json5/-/json5-0.5.1.tgz", "integrity": "sha1-Hq3nrMASA0rYTiOWdn6tn6VJWCE=" }, + "jsonfile": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", + "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.6" + } + }, "jsprim": { "version": "1.4.1", "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", @@ -2739,9 +4867,9 @@ } }, "just-extend": { - "version": "1.1.27", - "resolved": "https://registry.npmjs.org/just-extend/-/just-extend-1.1.27.tgz", - "integrity": "sha512-mJVp13Ix6gFo3SBAy9U/kL+oeZqzlYYYLQBwXVBlVzIsZwBqGREnOro24oC/8s8aox+rJhtZ2DiQof++IrkA+g==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/just-extend/-/just-extend-3.0.0.tgz", + "integrity": "sha512-Fu3T6pKBuxjWT/p4DkqGHFRsysc8OauWr4ZRTY9dIx07Y9O0RkoR5jcv28aeD1vuAwhm3nLkDurwLXoALp4DpQ==", "dev": true }, "kind-of": { @@ -2752,6 +4880,15 @@ "is-buffer": "^1.1.5" } }, + "latest-version": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-3.1.0.tgz", + "integrity": "sha1-ogU4P+oyKzO1rjsYq+4NwvNW7hU=", + "dev": true, + "requires": { + "package-json": "^4.0.0" + } + }, "left-pad": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/left-pad/-/left-pad-1.3.0.tgz", @@ -2774,7 +4911,7 @@ }, "readable-stream": { "version": "1.1.14", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", + "resolved": "http://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", "requires": { "core-util-is": "~1.0.0", @@ -2790,14 +4927,30 @@ } } }, + "level-filesystem": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/level-filesystem/-/level-filesystem-1.2.0.tgz", + "integrity": "sha1-oArKmRnEpN+v3KaoEI0iWq3/Y7M=", + "requires": { + "concat-stream": "^1.4.4", + "errno": "^0.1.1", + "fwd-stream": "^1.0.4", + "level-blobs": "^0.1.7", + "level-peek": "^1.0.6", + "level-sublevel": "^5.2.0", + "octal": "^1.0.0", + "once": "^1.3.0", + "xtend": "^2.2.0" + } + }, "level-fix-range": { "version": "1.0.2", - "resolved": "https://registry.npmjs.org/level-fix-range/-/level-fix-range-1.0.2.tgz", + "resolved": "http://registry.npmjs.org/level-fix-range/-/level-fix-range-1.0.2.tgz", "integrity": "sha1-vxW5Fa422EcMgh6IPd95zRZCCCg=" }, "level-hooks": { "version": "4.5.0", - "resolved": "https://registry.npmjs.org/level-hooks/-/level-hooks-4.5.0.tgz", + "resolved": "http://registry.npmjs.org/level-hooks/-/level-hooks-4.5.0.tgz", "integrity": "sha1-G5rmGSKTDzMF0aYfxNg8gQLA3ZM=", "requires": { "string-range": "~1.2" @@ -2833,7 +4986,7 @@ }, "level-peek": { "version": "1.0.6", - "resolved": "https://registry.npmjs.org/level-peek/-/level-peek-1.0.6.tgz", + "resolved": "http://registry.npmjs.org/level-peek/-/level-peek-1.0.6.tgz", "integrity": "sha1-vsUccqgu5GTTNkNMfIdsP8vM538=", "requires": { "level-fix-range": "~1.0.2" @@ -2841,7 +4994,7 @@ }, "level-sublevel": { "version": "5.2.3", - "resolved": "https://registry.npmjs.org/level-sublevel/-/level-sublevel-5.2.3.tgz", + "resolved": "http://registry.npmjs.org/level-sublevel/-/level-sublevel-5.2.3.tgz", "integrity": "sha1-dEwSxy0ucr543eO5tc2E1iGRQTo=", "requires": { "level-fix-range": "2.0", @@ -2852,7 +5005,7 @@ "dependencies": { "level-fix-range": { "version": "2.0.0", - "resolved": "https://registry.npmjs.org/level-fix-range/-/level-fix-range-2.0.0.tgz", + "resolved": "http://registry.npmjs.org/level-fix-range/-/level-fix-range-2.0.0.tgz", "integrity": "sha1-xBfWIVlEIVGhnZojZ4aPFyTC1Ug=", "requires": { "clone": "~0.1.9" @@ -2869,6 +5022,58 @@ } } }, + "levelup": { + "version": "0.18.6", + "resolved": "https://registry.npmjs.org/levelup/-/levelup-0.18.6.tgz", + "integrity": "sha1-5qAcsIlhbI7MApHCqb0/DETj5es=", + "requires": { + "bl": "~0.8.1", + "deferred-leveldown": "~0.2.0", + "errno": "~0.1.1", + "prr": "~0.0.0", + "readable-stream": "~1.0.26", + "semver": "~2.3.1", + "xtend": "~3.0.0" + }, + "dependencies": { + "isarray": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", + "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=" + }, + "prr": { + "version": "0.0.0", + "resolved": "https://registry.npmjs.org/prr/-/prr-0.0.0.tgz", + "integrity": "sha1-GoS4WQgyVQFBGFPQCB7j+obikmo=" + }, + "readable-stream": { + "version": "1.0.34", + "resolved": "http://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz", + "integrity": "sha1-Elgg40vIQtLyqq+v5MKRbuMsFXw=", + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.1", + "isarray": "0.0.1", + "string_decoder": "~0.10.x" + } + }, + "semver": { + "version": "2.3.2", + "resolved": "http://registry.npmjs.org/semver/-/semver-2.3.2.tgz", + "integrity": "sha1-uYSPJdbPNjMwc+ye+IVtQvEjPlI=" + }, + "string_decoder": { + "version": "0.10.31", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", + "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=" + }, + "xtend": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-3.0.0.tgz", + "integrity": "sha1-XM50B7r2Qsunvs2laBEcST9ZZlo=" + } + } + }, "levn": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", @@ -2878,10 +5083,74 @@ "type-check": "~0.3.2" } }, + "load-json-file": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-4.0.0.tgz", + "integrity": "sha1-L19Fq5HjMhYjT9U62rZo607AmTs=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "parse-json": "^4.0.0", + "pify": "^3.0.0", + "strip-bom": "^3.0.0" + } + }, + "locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "requires": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + } + }, "lodash": { - "version": "4.17.10", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.10.tgz", - "integrity": "sha512-UejweD1pDoXu+AD825lWwp4ZGtSwgnpZxb3JDViD7StjQz+Nb/6l093lx4OQ0foGWNRoc19mWy7BzL+UAK2iVg==" + "version": "4.17.11", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.11.tgz", + "integrity": "sha512-cQKh8igo5QUhZ7lg38DYWAxMvjSAKG0A8wGSVimP07SIUEK2UO+arSRKbRZWtelMtN5V0Hkwh5ryOto/SshYIg==" + }, + "lodash.clone": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.clone/-/lodash.clone-4.5.0.tgz", + "integrity": "sha1-GVhwRQ9aExkkeN9Lw9I9LeoZB7Y=", + "dev": true + }, + "lodash.clonedeep": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz", + "integrity": "sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8=", + "dev": true + }, + "lodash.clonedeepwith": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.clonedeepwith/-/lodash.clonedeepwith-4.5.0.tgz", + "integrity": "sha1-buMFc6A6GmDWcKYu8zwQzxr9vdQ=", + "dev": true + }, + "lodash.debounce": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz", + "integrity": "sha1-gteb/zCmfEAF/9XiUVMArZyk168=", + "dev": true + }, + "lodash.difference": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.difference/-/lodash.difference-4.5.0.tgz", + "integrity": "sha1-nMtOUF1Ia5FlE0V3KIWi3yf9AXw=", + "dev": true + }, + "lodash.flatten": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz", + "integrity": "sha1-8xwiIlqWMtK7+OSt2+8kCqdlph8=", + "dev": true + }, + "lodash.flattendeep": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/lodash.flattendeep/-/lodash.flattendeep-4.4.0.tgz", + "integrity": "sha1-+wMJF/hqMTTlvJvsDWngAT3f7bI=", + "dev": true }, "lodash.get": { "version": "4.4.2", @@ -2889,15 +5158,36 @@ "integrity": "sha1-LRd/ZS+jHpObRDjVNBSZ36OCXpk=", "dev": true }, + "lodash.islength": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.islength/-/lodash.islength-4.0.1.tgz", + "integrity": "sha1-Tpho1FJXXXUK/9NYyXlUPcIO1Xc=", + "dev": true + }, + "lodash.merge": { + "version": "4.6.1", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.1.tgz", + "integrity": "sha512-AOYza4+Hf5z1/0Hztxpm2/xiPZgi/cjMqdnKTUWTBSKchJlxXXuUSxCCl8rJlf4g6yww/j6mA8nC8Hw/EZWxKQ==", + "dev": true + }, "lodash.sortby": { "version": "4.7.0", "resolved": "https://registry.npmjs.org/lodash.sortby/-/lodash.sortby-4.7.0.tgz", "integrity": "sha1-7dFMgk4sycHgsKG0K7UhBRakJDg=" }, + "log-symbols": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", + "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==", + "dev": true, + "requires": { + "chalk": "^2.0.1" + } + }, "lolex": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/lolex/-/lolex-2.7.1.tgz", - "integrity": "sha512-Oo2Si3RMKV3+lV5MsSWplDQFoTClz/24S0MMHYcgGWWmFXr6TMlqcqk/l1GtH+d5wLBwNRiqGnwDRMirtFalJw==", + "version": "2.7.5", + "resolved": "https://registry.npmjs.org/lolex/-/lolex-2.7.5.tgz", + "integrity": "sha512-l9x0+1offnKKIzYVjyXU2SiwhXDLekRzKyhnbyldPHvC7BvLPVpdNUNR2KeMAiCN2D/kLNttZgQD5WjSxuBx3Q==", "dev": true }, "loose-envify": { @@ -2908,6 +5198,32 @@ "js-tokens": "^3.0.0 || ^4.0.0" } }, + "loud-rejection": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/loud-rejection/-/loud-rejection-1.6.0.tgz", + "integrity": "sha1-W0b4AUft7leIcPCG0Eghz5mOVR8=", + "dev": true, + "requires": { + "currently-unhandled": "^0.4.1", + "signal-exit": "^3.0.0" + } + }, + "lowercase-keys": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", + "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", + "dev": true + }, + "lru-cache": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.3.tgz", + "integrity": "sha512-fFEhvcgzuIoJVUF8fYr5KR0YqxD238zgObTps31YdADwPPAp82a4M8TrckkWyx7ekNlf9aBcVn81cFwwXngrJA==", + "dev": true, + "requires": { + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" + } + }, "ltgt": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/ltgt/-/ltgt-2.2.1.tgz", @@ -2915,24 +5231,79 @@ }, "magic-string": { "version": "0.22.5", - "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.22.5.tgz", + "resolved": "http://registry.npmjs.org/magic-string/-/magic-string-0.22.5.tgz", "integrity": "sha512-oreip9rJZkzvA8Qzk9HFs8fZGF/u7H/gtrE8EN6RjKJ9kh2HlC+yQ2QezifqTZfGyiuAV0dRv5a+y/8gBb1m9w==", "requires": { "vlq": "^0.2.2" } }, + "make-dir": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-1.3.0.tgz", + "integrity": "sha512-2w31R7SJtieJJnQtGc7RVL2StM2vGYVfqUOvUDxH6bC6aJTxPxTF0GnIgCyu7tjockiUWAYQRbxa7vKn34s5sQ==", + "dev": true, + "requires": { + "pify": "^3.0.0" + } + }, + "map-cache": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", + "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=", + "dev": true + }, + "map-obj": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/map-obj/-/map-obj-2.0.0.tgz", + "integrity": "sha1-plzSkIepJZi4eRJXpSPgISIqwfk=", + "dev": true + }, + "map-visit": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz", + "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=", + "dev": true, + "requires": { + "object-visit": "^1.0.0" + } + }, + "matcher": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/matcher/-/matcher-1.1.1.tgz", + "integrity": "sha512-+BmqxWIubKTRKNWx/ahnCkk3mG8m7OturVlqq6HiojGJTd5hVYbgZm6WzcYPCoB+KBT4Vd6R7WSRG2OADNaCjg==", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.4" + } + }, "math-random": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/math-random/-/math-random-1.0.1.tgz", "integrity": "sha1-izqsWIuKZuSXXjzepn97sylgH6w=" }, + "md5-hex": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/md5-hex/-/md5-hex-2.0.0.tgz", + "integrity": "sha1-0FiOnxx0lUSS7NJKwKxs6ZfZLjM=", + "dev": true, + "requires": { + "md5-o-matic": "^0.1.1" + } + }, + "md5-o-matic": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/md5-o-matic/-/md5-o-matic-0.1.1.tgz", + "integrity": "sha1-givM1l4RfFFPqxdrJZRdVBAKA8M=", + "dev": true + }, "md5.js": { - "version": "1.3.4", - "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.4.tgz", - "integrity": "sha1-6b296UogpawYsENA/Fdk1bCdkB0=", + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz", + "integrity": "sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==", "requires": { "hash-base": "^3.0.0", - "inherits": "^2.0.1" + "inherits": "^2.0.1", + "safe-buffer": "^5.1.2" } }, "media-typer": { @@ -2941,6 +5312,23 @@ "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=", "dev": true }, + "meow": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/meow/-/meow-5.0.0.tgz", + "integrity": "sha512-CbTqYU17ABaLefO8vCU153ZZlprKYWDljcndKKDCFcYQITzWCXZAVk4QMFZPgvzrnUQ3uItnIE/LoUOwrT15Ig==", + "dev": true, + "requires": { + "camelcase-keys": "^4.0.0", + "decamelize-keys": "^1.0.0", + "loud-rejection": "^1.0.0", + "minimist-options": "^3.0.1", + "normalize-package-data": "^2.3.4", + "read-pkg-up": "^3.0.0", + "redent": "^2.0.0", + "trim-newlines": "^2.0.0", + "yargs-parser": "^10.0.0" + } + }, "merge-descriptors": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", @@ -2989,18 +5377,24 @@ "dev": true }, "mime-db": { - "version": "1.33.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.33.0.tgz", - "integrity": "sha512-BHJ/EKruNIqJf/QahvxwQZXKygOQ256myeN/Ew+THcAa5q+PjyTTMMeNQC4DZw5AwfvelsUrA6B67NKMqXDbzQ==" + "version": "1.37.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.37.0.tgz", + "integrity": "sha512-R3C4db6bgQhlIhPU48fUtdVmKnflq+hRdad7IyKhtFj06VPNVdk2RhiYL3UjQIlso8L+YxAtFkobT0VK+S/ybg==" }, "mime-types": { - "version": "2.1.18", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.18.tgz", - "integrity": "sha512-lc/aahn+t4/SWV/qcmumYjymLsWfN3ELhpmVuUFjgsORruuZPVSwAQryq+HHGvO/SI2KVX26bx+En+zhM8g8hQ==", + "version": "2.1.21", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.21.tgz", + "integrity": "sha512-3iL6DbwpyLzjR3xHSFNFeb9Nz/M8WDkX33t1GFQnFOllWk8pOrh/LSrB5OXlnlW5P9LH73X6loW/eogc+F5lJg==", "requires": { - "mime-db": "~1.33.0" + "mime-db": "~1.37.0" } }, + "mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true + }, "minimalistic-assert": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", @@ -3021,12 +5415,43 @@ }, "minimist": { "version": "0.0.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", + "resolved": "http://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" }, + "minimist-options": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/minimist-options/-/minimist-options-3.0.2.tgz", + "integrity": "sha512-FyBrT/d0d4+uiZRbqznPXqw3IpZZG3gl3wKWiX784FycUKVwBt0uLBFkQrtE4tZOrgo78nZp2jnKz3L65T5LdQ==", + "dev": true, + "requires": { + "arrify": "^1.0.1", + "is-plain-obj": "^1.1.0" + } + }, + "mixin-deep": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.1.tgz", + "integrity": "sha512-8ZItLHeEgaqEvd5lYBXfm4EZSFCX29Jb9K+lAHhDKzReKBQKj3R+7NOF6tjqYi9t4oI8VUfaWITJQm86wnXGNQ==", + "dev": true, + "requires": { + "for-in": "^1.0.2", + "is-extendable": "^1.0.1" + }, + "dependencies": { + "is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, + "requires": { + "is-plain-object": "^2.0.4" + } + } + } + }, "mkdirp": { "version": "0.5.1", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", + "resolved": "http://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", "requires": { "minimist": "0.0.8" @@ -3059,6 +5484,20 @@ "ms": "2.0.0" } }, + "glob": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz", + "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, "has-flag": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-2.0.0.tgz", @@ -3087,12 +5526,63 @@ "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" }, + "multimatch": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/multimatch/-/multimatch-2.1.0.tgz", + "integrity": "sha1-nHkGoi+0wCkZ4vX3UWG0zb1LKis=", + "dev": true, + "requires": { + "array-differ": "^1.0.0", + "array-union": "^1.0.1", + "arrify": "^1.0.0", + "minimatch": "^3.0.0" + } + }, "nan": { - "version": "2.10.0", - "resolved": "https://registry.npmjs.org/nan/-/nan-2.10.0.tgz", - "integrity": "sha512-bAdJv7fBLhWC+/Bls0Oza+mvTaNQtP+1RyhhhvD95pgUJz6XM5IzgmxOkItJ9tkoCiplvAnXI1tNmmUD/eScyA==", + "version": "2.11.1", + "resolved": "https://registry.npmjs.org/nan/-/nan-2.11.1.tgz", + "integrity": "sha512-iji6k87OSXa0CcrLl9z+ZiYSuR2o+c0bGuNmXdrhTQTakxytAFsC56SArGYoiHlJlFoHSnvmhpceZJaXkVuOtA==", "dev": true }, + "nanomatch": { + "version": "1.2.13", + "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", + "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", + "dev": true, + "requires": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "fragment-cache": "^0.2.1", + "is-windows": "^1.0.2", + "kind-of": "^6.0.2", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "arr-diff": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", + "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", + "dev": true + }, + "array-unique": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", + "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", + "dev": true + }, + "kind-of": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz", + "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==", + "dev": true + } + } + }, "negotiator": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.1.tgz", @@ -3100,18 +5590,27 @@ "dev": true }, "nise": { - "version": "1.4.2", - "resolved": "https://registry.npmjs.org/nise/-/nise-1.4.2.tgz", - "integrity": "sha512-BxH/DxoQYYdhKgVAfqVy4pzXRZELHOIewzoesxpjYvpU+7YOalQhGNPf7wAx8pLrTNPrHRDlLOkAl8UI0ZpXjw==", + "version": "1.4.6", + "resolved": "https://registry.npmjs.org/nise/-/nise-1.4.6.tgz", + "integrity": "sha512-1GedetLKzmqmgwabuMSqPsT7oumdR77SBpDfNNJhADRIeA3LN/2RVqR4fFqwvzhAqcTef6PPCzQwITE/YQ8S8A==", "dev": true, "requires": { - "@sinonjs/formatio": "^2.0.0", - "just-extend": "^1.1.27", + "@sinonjs/formatio": "3.0.0", + "just-extend": "^3.0.0", "lolex": "^2.3.2", "path-to-regexp": "^1.7.0", "text-encoding": "^0.6.4" }, "dependencies": { + "@sinonjs/formatio": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@sinonjs/formatio/-/formatio-3.0.0.tgz", + "integrity": "sha512-vdjoYLDptCgvtJs57ULshak3iJe4NW3sJ3g36xVDGff5AE8P30S6A093EIEPjdi2noGhfuNOEkbxt3J3awFW1w==", + "dev": true, + "requires": { + "@sinonjs/samsam": "2.1.0" + } + }, "isarray": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", @@ -3129,6 +5628,18 @@ } } }, + "normalize-package-data": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.4.0.tgz", + "integrity": "sha512-9jjUFbTPfEy3R/ad/2oNbKtW9Hgovl5O1FvFWKkKblNXoN/Oou6+9+KKohPK13Yc3/TyunyWhJp6gvRNR/PPAw==", + "dev": true, + "requires": { + "hosted-git-info": "^2.1.4", + "is-builtin-module": "^1.0.0", + "semver": "2 || 3 || 4 || 5", + "validate-npm-package-license": "^3.0.1" + } + }, "normalize-path": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", @@ -3137,20 +5648,29 @@ "remove-trailing-separator": "^1.0.1" } }, + "npm-run-path": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz", + "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=", + "dev": true, + "requires": { + "path-key": "^2.0.0" + } + }, "number-is-nan": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=" }, "nwsapi": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.0.4.tgz", - "integrity": "sha512-Zt6HRR6RcJkuj5/N9zeE7FN6YitRW//hK2wTOwX274IBphbY3Zf5+yn5mZ9v/SzAOTMjQNxZf9KkmPLWn0cV4g==" + "version": "2.0.9", + "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.0.9.tgz", + "integrity": "sha512-nlWFSCTYQcHk/6A9FFnfhKc14c3aFhfdNBXgo8Qgi9QTBu/qg3Ww+Uiz9wMzXd1T8GFxPc2QIHB6Qtf2XFryFQ==" }, "oauth-sign": { - "version": "0.8.2", - "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.8.2.tgz", - "integrity": "sha1-Rqarfwrq2N6unsBWV4C31O/rnUM=" + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", + "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==" }, "object-assign": { "version": "4.1.1", @@ -3158,6 +5678,28 @@ "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", "dev": true }, + "object-copy": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", + "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=", + "dev": true, + "requires": { + "copy-descriptor": "^0.1.0", + "define-property": "^0.2.5", + "kind-of": "^3.0.3" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + } + } + }, "object-keys": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-0.2.0.tgz", @@ -3168,6 +5710,23 @@ "is": "~0.2.6" } }, + "object-visit": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz", + "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=", + "dev": true, + "requires": { + "isobject": "^3.0.0" + }, + "dependencies": { + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, "object.omit": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/object.omit/-/object.omit-2.0.1.tgz", @@ -3177,6 +5736,52 @@ "is-extendable": "^0.1.1" } }, + "object.pick": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", + "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=", + "dev": true, + "requires": { + "isobject": "^3.0.1" + }, + "dependencies": { + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "observable-to-promise": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/observable-to-promise/-/observable-to-promise-0.5.0.tgz", + "integrity": "sha1-yCjw8NxH6fhq+KSXfF1VB2znqR8=", + "dev": true, + "requires": { + "is-observable": "^0.2.0", + "symbol-observable": "^1.0.4" + }, + "dependencies": { + "is-observable": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/is-observable/-/is-observable-0.2.0.tgz", + "integrity": "sha1-s2ExHYPG5dcmyr9eJQsCNxBvWuI=", + "dev": true, + "requires": { + "symbol-observable": "^0.2.2" + }, + "dependencies": { + "symbol-observable": { + "version": "0.2.4", + "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-0.2.4.tgz", + "integrity": "sha1-lag9smGG1q9+ehjb2XYKL4bQj0A=", + "dev": true + } + } + } + } + }, "octal": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/octal/-/octal-1.0.0.tgz", @@ -3199,6 +5804,15 @@ "wrappy": "1" } }, + "onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", + "dev": true, + "requires": { + "mimic-fn": "^1.0.0" + } + }, "optionator": { "version": "0.8.2", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.2.tgz", @@ -3212,6 +5826,37 @@ "wordwrap": "~1.0.0" } }, + "ora": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ora/-/ora-3.0.0.tgz", + "integrity": "sha512-LBS97LFe2RV6GJmXBi6OKcETKyklHNMV0xw7BtsVn2MlsgsydyZetSCbCANr+PFLmDyv4KV88nn0eCKza665Mg==", + "dev": true, + "requires": { + "chalk": "^2.3.1", + "cli-cursor": "^2.1.0", + "cli-spinners": "^1.1.0", + "log-symbols": "^2.2.0", + "strip-ansi": "^4.0.0", + "wcwidth": "^1.0.1" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + }, "os-homedir": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", @@ -3222,9 +5867,69 @@ "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=" }, + "p-finally": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", + "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", + "dev": true + }, + "p-limit": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.0.0.tgz", + "integrity": "sha512-fl5s52lI5ahKCernzzIyAP0QAZbGIovtVHGwpcu1Jr/EpzLVDI2myISHwGqK7m8uQFugVWSrbxH7XnhGtvEc+A==", + "dev": true, + "requires": { + "p-try": "^2.0.0" + } + }, + "p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dev": true, + "requires": { + "p-limit": "^2.0.0" + } + }, + "p-map": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/p-map/-/p-map-1.2.0.tgz", + "integrity": "sha512-r6zKACMNhjPJMTl8KcFH4li//gkrXWfbD6feV8l6doRHlzljFWGJ2AP6iKaCJXyZmAUMOPtvbW7EXkbWO/pLEA==", + "dev": true + }, + "p-try": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.0.0.tgz", + "integrity": "sha512-hMp0onDKIajHfIkdRk3P4CdCmErkYAxxDtP3Wx/4nZ3aGlau2VKh3mZpcuFkH27WQkL/3WBCPOktzA9ZOAnMQQ==", + "dev": true + }, + "package-hash": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/package-hash/-/package-hash-2.0.0.tgz", + "integrity": "sha1-eK4ybIngWk2BO2hgGXevBcANKg0=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.11", + "lodash.flattendeep": "^4.4.0", + "md5-hex": "^2.0.0", + "release-zalgo": "^1.0.0" + } + }, + "package-json": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/package-json/-/package-json-4.0.1.tgz", + "integrity": "sha1-iGmgQBJTZhxMTKPabCEh7VVfXu0=", + "dev": true, + "requires": { + "got": "^6.7.1", + "registry-auth-token": "^3.0.1", + "registry-url": "^3.0.3", + "semver": "^5.1.0" + } + }, "parse-asn1": { "version": "5.1.1", - "resolved": "https://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.1.tgz", + "resolved": "http://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.1.tgz", "integrity": "sha512-KPx7flKXg775zZpnp9SxJlz00gTd4BmJ2yJufSc44gMCRrRQ7NSzAcSJQfifuOLgW6bEi+ftrALtsgALeB2Adw==", "requires": { "asn1.js": "^4.0.0", @@ -3232,100 +5937,234 @@ "create-hash": "^1.1.0", "evp_bytestokey": "^1.0.0", "pbkdf2": "^3.0.3" + } + }, + "parse-glob": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/parse-glob/-/parse-glob-3.0.4.tgz", + "integrity": "sha1-ssN2z7EfNVE7rdFz7wu246OIORw=", + "requires": { + "glob-base": "^0.3.0", + "is-dotfile": "^1.0.0", + "is-extglob": "^1.0.0", + "is-glob": "^2.0.0" + } + }, + "parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "dev": true, + "requires": { + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" + } + }, + "parse-ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/parse-ms/-/parse-ms-2.0.0.tgz", + "integrity": "sha512-AddiXFSLLCqj+tCRJ9MrUtHZB4DWojO3tk0NVZ+g5MaMQHF2+p2ktqxuoXyPFLljz/aUK0Nfhd/uGWnhXVXEyA==", + "dev": true + }, + "parse5": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-4.0.0.tgz", + "integrity": "sha512-VrZ7eOd3T1Fk4XWNXMgiGBK/z0MG48BWG2uQNU4I72fkQuKUTZpl+u9k+CxEG0twMVzSmXEEz12z5Fnw1jIQFA==" + }, + "parseurl": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.2.tgz", + "integrity": "sha1-/CidTtiZMRlGDBViUyYs3I3mW/M=", + "dev": true + }, + "pascalcase": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", + "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", + "dev": true + }, + "path-dirname": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", + "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", + "dev": true + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" + }, + "path-is-inside": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", + "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=", + "dev": true + }, + "path-key": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", + "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", + "dev": true + }, + "path-parse": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", + "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==" + }, + "path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=", + "dev": true + }, + "path-type": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "dev": true, + "requires": { + "pify": "^3.0.0" + } + }, + "pbkdf2": { + "version": "3.0.17", + "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.0.17.tgz", + "integrity": "sha512-U/il5MsrZp7mGg3mSQfn742na2T+1/vHDCG5/iTI3X9MKUuYUZVLQhyRsg06mCgDBTd57TxzgZt7P+fYfjRLtA==", + "requires": { + "create-hash": "^1.1.2", + "create-hmac": "^1.1.4", + "ripemd160": "^2.0.1", + "safe-buffer": "^5.0.1", + "sha.js": "^2.4.8" + } + }, + "performance-now": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", + "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=" + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "pinkie": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", + "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=", + "dev": true + }, + "pinkie-promise": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", + "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=", + "dev": true, + "requires": { + "pinkie": "^2.0.0" + } + }, + "pkg-conf": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/pkg-conf/-/pkg-conf-2.1.0.tgz", + "integrity": "sha1-ISZRTKbyq/69FoWW3xi6V4Z/AFg=", + "dev": true, + "requires": { + "find-up": "^2.0.0", + "load-json-file": "^4.0.0" }, "dependencies": { - "create-hash": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", - "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", + "find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "dev": true, + "requires": { + "locate-path": "^2.0.0" + } + }, + "locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", + "dev": true, "requires": { - "cipher-base": "^1.0.1", - "inherits": "^2.0.1", - "md5.js": "^1.3.4", - "ripemd160": "^2.0.1", - "sha.js": "^2.4.0" + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" } }, - "create-hmac": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz", - "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==", + "p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, "requires": { - "cipher-base": "^1.0.3", - "create-hash": "^1.1.0", - "inherits": "^2.0.1", - "ripemd160": "^2.0.0", - "safe-buffer": "^5.0.1", - "sha.js": "^2.4.8" + "p-try": "^1.0.0" } }, - "pbkdf2": { - "version": "3.0.16", - "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.0.16.tgz", - "integrity": "sha512-y4CXP3thSxqf7c0qmOF+9UeOTrifiVTIM+u7NWlq+PRsHbr7r7dpCmvzrZxa96JJUNi0Y5w9VqG5ZNeCVMoDcA==", + "p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", + "dev": true, "requires": { - "create-hash": "^1.1.2", - "create-hmac": "^1.1.4", - "ripemd160": "^2.0.1", - "safe-buffer": "^5.0.1", - "sha.js": "^2.4.8" + "p-limit": "^1.1.0" } + }, + "p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", + "dev": true } } }, - "parse-glob": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/parse-glob/-/parse-glob-3.0.4.tgz", - "integrity": "sha1-ssN2z7EfNVE7rdFz7wu246OIORw=", + "pkg-dir": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", + "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "dev": true, "requires": { - "glob-base": "^0.3.0", - "is-dotfile": "^1.0.0", - "is-extglob": "^1.0.0", - "is-glob": "^2.0.0" + "find-up": "^3.0.0" } }, - "parse5": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/parse5/-/parse5-4.0.0.tgz", - "integrity": "sha512-VrZ7eOd3T1Fk4XWNXMgiGBK/z0MG48BWG2uQNU4I72fkQuKUTZpl+u9k+CxEG0twMVzSmXEEz12z5Fnw1jIQFA==" - }, - "parseurl": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.2.tgz", - "integrity": "sha1-/CidTtiZMRlGDBViUyYs3I3mW/M=", - "dev": true - }, - "path-is-absolute": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", - "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" - }, - "path-parse": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.5.tgz", - "integrity": "sha1-PBrfhx6pzWyUMbbqK9dKD/BVxME=" - }, - "path-to-regexp": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", - "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=", - "dev": true - }, - "performance-now": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", - "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=" + "plur": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/plur/-/plur-3.0.1.tgz", + "integrity": "sha512-lJl0ojUynAM1BZn58Pas2WT/TXeC1+bS+UqShl0x9+49AtOn7DixRXVzaC8qrDOIxNDmepKnLuMTH7NQmkX0PA==", + "dev": true, + "requires": { + "irregular-plurals": "^2.0.0" + } }, "pn": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/pn/-/pn-1.1.0.tgz", "integrity": "sha512-2qHaIQr2VLRFoxe2nASzsV6ef4yOOH+Fi9FBOVH6cqeSgUnoyySPZkxzLuzd+RYOQTRpROA0ztTMqxROKSb/nA==" }, + "posix-character-classes": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz", + "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=", + "dev": true + }, "prelude-ls": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=" }, + "prepend-http": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz", + "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=", + "dev": true + }, "preserve": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/preserve/-/preserve-0.2.0.tgz", @@ -3340,6 +6179,15 @@ "route-recognizer": "^0.3.3" } }, + "pretty-ms": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/pretty-ms/-/pretty-ms-4.0.0.tgz", + "integrity": "sha512-qG66ahoLCwpLXD09ZPHSCbUWYTqdosB7SMP4OffgTgL2PBKXMuUsrk5Bwg8q4qPkjTXsKBMr+YK3Ltd/6F9s/Q==", + "dev": true, + "requires": { + "parse-ms": "^2.0.0" + } + }, "private": { "version": "0.1.8", "resolved": "https://registry.npmjs.org/private/-/private-0.1.8.tgz", @@ -3356,13 +6204,13 @@ "integrity": "sha512-MtEC1TqN0EU5nephaJ4rAtThHtC86dNN9qCuEhtshvpVBkAW5ZO7BASN9REnF9eoXGcRub+pFuKEpOHE+HbEMw==" }, "proxy-addr": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.3.tgz", - "integrity": "sha512-jQTChiCJteusULxjBp8+jftSQE5Obdl3k4cnmLA6WXtK6XFuWRnvVL7aCiBqaLPM8c4ph0S4tKna8XvmIwEnXQ==", + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.4.tgz", + "integrity": "sha512-5erio2h9jp5CHGwcybmxmVqHmnCBZeewlfJ0pex+UW7Qny7OOZXTtH56TGNyBizkgiOwhJtMKrVzDTeKcySZwA==", "dev": true, "requires": { "forwarded": "~0.1.2", - "ipaddr.js": "1.6.0" + "ipaddr.js": "1.8.0" } }, "prr": { @@ -3385,7 +6233,7 @@ "dependencies": { "ansi-regex": { "version": "0.2.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-0.2.1.tgz", + "resolved": "http://registry.npmjs.org/ansi-regex/-/ansi-regex-0.2.1.tgz", "integrity": "sha1-DY6UaWej2BQ/k+JOKYUl/BsiNfk=", "dev": true }, @@ -3397,7 +6245,7 @@ }, "chalk": { "version": "0.5.1", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-0.5.1.tgz", + "resolved": "http://registry.npmjs.org/chalk/-/chalk-0.5.1.tgz", "integrity": "sha1-Zjs6ZItotV0EaQ1JFnqoN4WPIXQ=", "dev": true, "requires": { @@ -3419,7 +6267,7 @@ }, "strip-ansi": { "version": "0.3.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-0.3.0.tgz", + "resolved": "http://registry.npmjs.org/strip-ansi/-/strip-ansi-0.3.0.tgz", "integrity": "sha1-JfSOoiynkYfzF0pNuHWTR7sSYiA=", "dev": true, "requires": { @@ -3434,35 +6282,28 @@ } } }, + "pseudomap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", + "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=", + "dev": true + }, "psl": { - "version": "1.1.28", - "resolved": "https://registry.npmjs.org/psl/-/psl-1.1.28.tgz", - "integrity": "sha512-+AqO1Ae+N/4r7Rvchrdm432afjT9hqJRyBN3DQv9At0tPz4hIFSGKbq64fN9dVoCow4oggIIax5/iONx0r9hZw==" + "version": "1.1.29", + "resolved": "https://registry.npmjs.org/psl/-/psl-1.1.29.tgz", + "integrity": "sha512-AeUmQ0oLN02flVHXWh9sSJF7mcdFq0ppid/JkErufc3hGIV/AMa8Fo9VgDo/cT2jFdOWoFvHp90qqBH54W+gjQ==" }, "public-encrypt": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/public-encrypt/-/public-encrypt-4.0.2.tgz", - "integrity": "sha512-4kJ5Esocg8X3h8YgJsKAuoesBgB7mqH3eowiDzMUPKiRDDE7E/BqqZD1hnTByIaAFiwAw246YEltSq7tdrOH0Q==", + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/public-encrypt/-/public-encrypt-4.0.3.tgz", + "integrity": "sha512-zVpa8oKZSz5bTMTFClc1fQOnyyEzpl5ozpi1B5YcvBrdohMjH2rfsBtyXcuNuwjsDIXmBYlF2N5FlJYhR29t8Q==", "requires": { "bn.js": "^4.1.0", "browserify-rsa": "^4.0.0", "create-hash": "^1.1.0", "parse-asn1": "^5.0.0", - "randombytes": "^2.0.1" - }, - "dependencies": { - "create-hash": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", - "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", - "requires": { - "cipher-base": "^1.0.1", - "inherits": "^2.0.1", - "md5.js": "^1.3.4", - "ripemd160": "^2.0.1", - "sha.js": "^2.4.0" - } - } + "randombytes": "^2.0.1", + "safe-buffer": "^5.1.2" } }, "punycode": { @@ -3482,7 +6323,7 @@ "dependencies": { "bl": { "version": "0.4.2", - "resolved": "https://registry.npmjs.org/bl/-/bl-0.4.2.tgz", + "resolved": "http://registry.npmjs.org/bl/-/bl-0.4.2.tgz", "integrity": "sha1-XbMdcvA4xU5orcOVeBJf47Ct3JY=", "dev": true, "requires": { @@ -3497,7 +6338,7 @@ }, "readable-stream": { "version": "1.0.34", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz", + "resolved": "http://registry.npmjs.org/readable-stream/-/readable-stream-1.0.34.tgz", "integrity": "sha1-Elgg40vIQtLyqq+v5MKRbuMsFXw=", "dev": true, "requires": { @@ -3520,10 +6361,16 @@ "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==" }, + "quick-lru": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/quick-lru/-/quick-lru-1.1.0.tgz", + "integrity": "sha1-Q2CxfGETatOAeDl/8RQW4Ybc+7g=", + "dev": true + }, "randomatic": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/randomatic/-/randomatic-3.0.0.tgz", - "integrity": "sha512-VdxFOIEY3mNO5PtSRkkle/hPJDHvQhK21oa73K4yAc9qmp6N429gAyF1gZMOTMeS0/AYzaV/2Trcef+NaIonSA==", + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/randomatic/-/randomatic-3.1.0.tgz", + "integrity": "sha512-KnGPVE0lo2WoXxIZ7cPR8YBpiol4gsSuOwDSg410oHh80ZMp5EiypNqL2K4Z77vJn6lB5rap7IkAmcUlalcnBQ==", "requires": { "is-number": "^4.0.0", "kind-of": "^6.0.0", @@ -3538,83 +6385,451 @@ "kind-of": { "version": "6.0.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz", - "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==" - } - } - }, - "randombytes": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.0.6.tgz", - "integrity": "sha512-CIQ5OFxf4Jou6uOKe9t1AOgqpeU5fd70A8NPdHSGeYXqXsPe6peOwI0cUl88RWZ6sP1vPMV3avd/R6cZ5/sP1A==", - "requires": { - "safe-buffer": "^5.1.0" - } - }, - "randomfill": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/randomfill/-/randomfill-1.0.4.tgz", - "integrity": "sha512-87lcbR8+MhcWcUiQ+9e+Rwx8MyR2P7qnt15ynUlbm3TU/fjbgz4GsvfSUDTemtCCtVCqb4ZcEFlyPNTh9bBTLw==", - "requires": { - "randombytes": "^2.0.5", - "safe-buffer": "^5.1.0" - } - }, - "range-parser": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz", - "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4=", - "dev": true - }, - "raw-body": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.2.tgz", - "integrity": "sha1-vNYMd9Prk83gBQKVw/N5OJvIj4k=", - "dev": true, - "requires": { - "bytes": "3.0.0", - "http-errors": "1.6.2", - "iconv-lite": "0.4.19", - "unpipe": "1.0.0" - }, - "dependencies": { - "depd": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.1.tgz", - "integrity": "sha1-V4O04cRZ8G+lyif5kfPQbnoxA1k=", + "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==" + } + } + }, + "randombytes": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.0.6.tgz", + "integrity": "sha512-CIQ5OFxf4Jou6uOKe9t1AOgqpeU5fd70A8NPdHSGeYXqXsPe6peOwI0cUl88RWZ6sP1vPMV3avd/R6cZ5/sP1A==", + "requires": { + "safe-buffer": "^5.1.0" + } + }, + "randomfill": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/randomfill/-/randomfill-1.0.4.tgz", + "integrity": "sha512-87lcbR8+MhcWcUiQ+9e+Rwx8MyR2P7qnt15ynUlbm3TU/fjbgz4GsvfSUDTemtCCtVCqb4ZcEFlyPNTh9bBTLw==", + "requires": { + "randombytes": "^2.0.5", + "safe-buffer": "^5.1.0" + } + }, + "range-parser": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz", + "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4=", + "dev": true + }, + "raw-body": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.3.tgz", + "integrity": "sha512-9esiElv1BrZoI3rCDuOuKCBRbuApGGaDPQfjSflGxdy4oyzqghxu6klEkkVIvBje+FF0BX9coEv8KqW6X/7njw==", + "dev": true, + "requires": { + "bytes": "3.0.0", + "http-errors": "1.6.3", + "iconv-lite": "0.4.23", + "unpipe": "1.0.0" + }, + "dependencies": { + "iconv-lite": { + "version": "0.4.23", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.23.tgz", + "integrity": "sha512-neyTUVFtahjf0mB3dZT77u+8O0QB89jFdnBkd5P1JgYPbPaia3gXXOVL2fq8VyU2gMMD7SaN7QukTB/pmXYvDA==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + } + } + }, + "rc": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", + "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", + "dev": true, + "requires": { + "deep-extend": "^0.6.0", + "ini": "~1.3.0", + "minimist": "^1.2.0", + "strip-json-comments": "~2.0.1" + }, + "dependencies": { + "minimist": { + "version": "1.2.0", + "resolved": "http://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", + "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=", + "dev": true + } + } + }, + "read-pkg": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-3.0.0.tgz", + "integrity": "sha1-nLxoaXj+5l0WwA4rGcI3/Pbjg4k=", + "dev": true, + "requires": { + "load-json-file": "^4.0.0", + "normalize-package-data": "^2.3.2", + "path-type": "^3.0.0" + } + }, + "read-pkg-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-3.0.0.tgz", + "integrity": "sha1-PtSWaF26D4/hGNBpHcUfSh/5bwc=", + "dev": true, + "requires": { + "find-up": "^2.0.0", + "read-pkg": "^3.0.0" + }, + "dependencies": { + "find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "dev": true, + "requires": { + "locate-path": "^2.0.0" + } + }, + "locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", + "dev": true, + "requires": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + } + }, + "p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, + "requires": { + "p-try": "^1.0.0" + } + }, + "p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", + "dev": true, + "requires": { + "p-limit": "^1.1.0" + } + }, + "p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", + "dev": true + } + } + }, + "readable-stream": { + "version": "2.3.6", + "resolved": "http://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", + "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "readdirp": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", + "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" + }, + "dependencies": { + "arr-diff": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", + "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", + "dev": true + }, + "array-unique": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", + "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", + "dev": true + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "expand-brackets": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", + "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", + "dev": true, + "requires": { + "debug": "^2.3.3", + "define-property": "^0.2.5", + "extend-shallow": "^2.0.1", + "posix-character-classes": "^0.1.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "is-accessor-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", + "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "is-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "is-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", + "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + } + }, + "kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "dev": true + } + } + }, + "extglob": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", + "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", + "dev": true, + "requires": { + "array-unique": "^0.3.2", + "define-property": "^1.0.0", + "expand-brackets": "^2.1.4", + "extend-shallow": "^2.0.1", + "fragment-cache": "^0.2.1", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "kind-of": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz", + "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==", "dev": true }, - "http-errors": { - "version": "1.6.2", - "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.2.tgz", - "integrity": "sha1-CgAsyFcHGSp+eUbO7cERVfYOxzY=", + "micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", "dev": true, "requires": { - "depd": "1.1.1", - "inherits": "2.0.3", - "setprototypeof": "1.0.3", - "statuses": ">= 1.3.1 < 2" + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" } - }, - "setprototypeof": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.0.3.tgz", - "integrity": "sha1-ZlZ+NwQ+608E2RvWWMDL77VbjgQ=", - "dev": true } } }, - "readable-stream": { - "version": "2.3.6", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", - "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", + "redent": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/redent/-/redent-2.0.0.tgz", + "integrity": "sha1-wbIAe0LVfrE4kHmzyDM2OdXhzKo=", + "dev": true, "requires": { - "core-util-is": "~1.0.0", - "inherits": "~2.0.3", - "isarray": "~1.0.0", - "process-nextick-args": "~2.0.0", - "safe-buffer": "~5.1.1", - "string_decoder": "~1.1.1", - "util-deprecate": "~1.0.1" + "indent-string": "^3.0.0", + "strip-indent": "^2.0.0" } }, "regenerate": { @@ -3622,6 +6837,15 @@ "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.0.tgz", "integrity": "sha512-1G6jJVDWrt0rK99kBjvEtziZNCICAuvIPkSiUFIQxVP06RCVpq3dmDo2oi6ABpYaDYaTRr67BEhL8r1wgEZZKg==" }, + "regenerate-unicode-properties": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/regenerate-unicode-properties/-/regenerate-unicode-properties-7.0.0.tgz", + "integrity": "sha512-s5NGghCE4itSlUS+0WUj88G6cfMVMmH8boTPNvABf8od+2dhT9WDlWu8n01raQAJZMOK8Ch6jSexaRO7swd6aw==", + "dev": true, + "requires": { + "regenerate": "^1.4.0" + } + }, "regenerator-runtime": { "version": "0.11.1", "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.11.1.tgz", @@ -3645,6 +6869,45 @@ "is-equal-shallow": "^0.1.3" } }, + "regex-not": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.2.tgz", + "integrity": "sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==", + "dev": true, + "requires": { + "extend-shallow": "^3.0.2", + "safe-regex": "^1.1.0" + } + }, + "regexpu-core": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-2.0.0.tgz", + "integrity": "sha1-SdA4g3uNz4v6W5pCE5k45uoq4kA=", + "requires": { + "regenerate": "^1.2.1", + "regjsgen": "^0.2.0", + "regjsparser": "^0.1.4" + } + }, + "registry-auth-token": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-3.3.2.tgz", + "integrity": "sha512-JL39c60XlzCVgNrO+qq68FoNb56w/m7JYvGR2jT5iR1xBrUA3Mfx5Twk5rqTThPmQKMWydGmq8oFtDlxfrmxnQ==", + "dev": true, + "requires": { + "rc": "^1.1.6", + "safe-buffer": "^5.0.1" + } + }, + "registry-url": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-3.1.0.tgz", + "integrity": "sha1-PU74cPc93h138M+aOBQyRE4XSUI=", + "dev": true, + "requires": { + "rc": "^1.0.1" + } + }, "regjsgen": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/regjsgen/-/regjsgen-0.2.0.tgz", @@ -3658,15 +6921,24 @@ "jsesc": "~0.5.0" } }, + "release-zalgo": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/release-zalgo/-/release-zalgo-1.0.0.tgz", + "integrity": "sha1-CXALflB0Mpc5Mw5TXFqQ+2eFFzA=", + "dev": true, + "requires": { + "es6-error": "^4.0.1" + } + }, "remove-trailing-separator": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=" }, "repeat-element": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.2.tgz", - "integrity": "sha1-7wiaF40Ug7quTZPrmLT55OEdmQo=" + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.3.tgz", + "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==" }, "repeat-string": { "version": "1.6.1", @@ -3681,6 +6953,33 @@ "is-finite": "^1.0.0" } }, + "request": { + "version": "2.88.0", + "resolved": "https://registry.npmjs.org/request/-/request-2.88.0.tgz", + "integrity": "sha512-NAqBSrijGLZdM0WZNsInLJpkJokL72XYjUpnB0iwsRgxh7dB6COrHnTBNwN0E+lHDAJzu7kLAkDeY08z2/A0hg==", + "requires": { + "aws-sign2": "~0.7.0", + "aws4": "^1.8.0", + "caseless": "~0.12.0", + "combined-stream": "~1.0.6", + "extend": "~3.0.2", + "forever-agent": "~0.6.1", + "form-data": "~2.3.2", + "har-validator": "~5.1.0", + "http-signature": "~1.2.0", + "is-typedarray": "~1.0.0", + "isstream": "~0.1.2", + "json-stringify-safe": "~5.0.1", + "mime-types": "~2.1.19", + "oauth-sign": "~0.9.0", + "performance-now": "^2.1.0", + "qs": "~6.5.2", + "safe-buffer": "^5.1.2", + "tough-cookie": "~2.4.3", + "tunnel-agent": "^0.6.0", + "uuid": "^3.3.2" + } + }, "request-promise-core": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/request-promise-core/-/request-promise-core-1.1.1.tgz", @@ -3699,6 +6998,12 @@ "tough-cookie": ">=2.3.3" } }, + "require-precompiled": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/require-precompiled/-/require-precompiled-0.1.0.tgz", + "integrity": "sha1-WhtS63Dr7UPrmC6XTIWrWVceVvo=", + "dev": true + }, "resolve": { "version": "1.8.1", "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.8.1.tgz", @@ -3707,6 +7012,43 @@ "path-parse": "^1.0.5" } }, + "resolve-cwd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-2.0.0.tgz", + "integrity": "sha1-AKn3OHVW4nA46uIyyqNypqWbZlo=", + "dev": true, + "requires": { + "resolve-from": "^3.0.0" + } + }, + "resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", + "dev": true + }, + "resolve-url": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", + "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=", + "dev": true + }, + "restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", + "dev": true, + "requires": { + "onetime": "^2.0.0", + "signal-exit": "^3.0.2" + } + }, + "ret": { + "version": "0.1.15", + "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", + "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==", + "dev": true + }, "rimraf": { "version": "2.6.2", "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.2.tgz", @@ -3732,7 +7074,7 @@ }, "rollup-plugin-commonjs": { "version": "8.4.1", - "resolved": "https://registry.npmjs.org/rollup-plugin-commonjs/-/rollup-plugin-commonjs-8.4.1.tgz", + "resolved": "http://registry.npmjs.org/rollup-plugin-commonjs/-/rollup-plugin-commonjs-8.4.1.tgz", "integrity": "sha512-mg+WuD+jlwoo8bJtW3Mvx7Tz6TsIdMsdhuvCnDMoyjh0oxsVgsjB/N0X984RJCWwc5IIiqNVJhXeeITcc73++A==", "requires": { "acorn": "^5.2.1", @@ -3754,22 +7096,22 @@ } }, "rollup-plugin-node-globals": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/rollup-plugin-node-globals/-/rollup-plugin-node-globals-1.2.1.tgz", - "integrity": "sha512-PZgkJkVLWZRdwx33GaAeD92UjmvCM7kM9i/8wgoe9hN901RrjVs8eVjg5DzQ+2kGZSiqyx0aiIunLnOOCWshWQ==", + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/rollup-plugin-node-globals/-/rollup-plugin-node-globals-1.4.0.tgz", + "integrity": "sha512-xRkB+W/m1KLIzPUmG0ofvR+CPNcvuCuNdjVBVS7ALKSxr3EDhnzNceGkGi1m8MToSli13AzKFYH4ie9w3I5L3g==", "requires": { - "acorn": "^5.5.0", + "acorn": "^5.7.3", "buffer-es6": "^4.9.3", - "estree-walker": "^0.5.1", - "magic-string": "^0.22.4", + "estree-walker": "^0.5.2", + "magic-string": "^0.22.5", "process-es6": "^0.11.6", - "rollup-pluginutils": "^2.0.1" + "rollup-pluginutils": "^2.3.1" } }, "rollup-plugin-node-resolve": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/rollup-plugin-node-resolve/-/rollup-plugin-node-resolve-3.3.0.tgz", - "integrity": "sha512-9zHGr3oUJq6G+X0oRMYlzid9fXicBdiydhwGChdyeNRGPcN/majtegApRKHLR5drboUvEWU+QeUmGTyEZQs3WA==", + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/rollup-plugin-node-resolve/-/rollup-plugin-node-resolve-3.4.0.tgz", + "integrity": "sha512-PJcd85dxfSBWih84ozRtBkB731OjXk0KnzN0oGp7WOWcarAFkVa71cV5hTJg2qpVsV2U8EUwrzHP3tvy9vS3qg==", "requires": { "builtin-modules": "^2.0.0", "is-module": "^1.0.0", @@ -3782,24 +7124,33 @@ "integrity": "sha512-HCTBpV8MwP5lNzZrHD2moVxHIToHU1EkzkKGVj6Z0DcgUfxrxrZmeQirQeLz2yhnkJqRjwiVywK9CS8jDYakrw==" }, "rollup-pluginutils": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/rollup-pluginutils/-/rollup-pluginutils-2.3.0.tgz", - "integrity": "sha512-xB6hsRsjdJdIYWEyYUJy/3ki5g69wrf0luHPGNK3ZSocV6HLNfio59l3dZ3TL4xUwEKgROhFi9jOCt6c5gfUWw==", + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/rollup-pluginutils/-/rollup-pluginutils-2.3.3.tgz", + "integrity": "sha512-2XZwja7b6P5q4RZ5FhyX1+f46xi1Z3qBKigLRZ6VTZjwbN0K1IFGMlwm06Uu0Emcre2Z63l77nq/pzn+KxIEoA==", "requires": { "estree-walker": "^0.5.2", "micromatch": "^2.3.11" } }, "route-recognizer": { - "version": "0.3.3", - "resolved": "https://registry.npmjs.org/route-recognizer/-/route-recognizer-0.3.3.tgz", - "integrity": "sha1-HTZeJ/ppleCRZ199yUCowANTvSk=" + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/route-recognizer/-/route-recognizer-0.3.4.tgz", + "integrity": "sha512-2+MhsfPhvauN1O8KaXpXAOfR/fwe8dnUXVM+xw7yt40lJRfPVQxV6yryZm0cgRvAj5fMF/mdRZbL2ptwbs5i2g==" }, "safe-buffer": { "version": "5.1.2", "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" }, + "safe-regex": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", + "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", + "dev": true, + "requires": { + "ret": "~0.1.10" + } + }, "safer-buffer": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", @@ -3817,9 +7168,45 @@ "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==" }, "semver": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.5.0.tgz", - "integrity": "sha512-4SJ3dm0WAwWy/NVeioZh5AntkdJoWKxHxcmyP622fOkgHa4z3R0TdBJICINyaSDE6uNwVc8gZr+ZinwZAH4xIA==" + "version": "5.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.6.0.tgz", + "integrity": "sha512-RS9R6R35NYgQn++fkDWaOmqGoj4Ek9gGs+DPxNUZKuwE183xjJroKvyo1IzVFeXvUrvmALy6FWD5xrdJT25gMg==" + }, + "semver-diff": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-2.1.0.tgz", + "integrity": "sha1-S7uEN8jTfksM8aaP1ybsbWRdbTY=", + "dev": true, + "requires": { + "semver": "^5.0.3" + } + }, + "send": { + "version": "0.16.2", + "resolved": "https://registry.npmjs.org/send/-/send-0.16.2.tgz", + "integrity": "sha512-E64YFPUssFHEFBvpbbjr44NCLtI1AohxQ8ZSiJjQLskAdKuriYEP6VyGEsRDH8ScozGpkaX1BGvhanqCwkcEZw==", + "dev": true, + "requires": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.6.2", + "mime": "1.4.1", + "ms": "2.0.0", + "on-finished": "~2.3.0", + "range-parser": "~1.2.0", + "statuses": "~1.4.0" + } + }, + "serialize-error": { + "version": "2.1.0", + "resolved": "http://registry.npmjs.org/serialize-error/-/serialize-error-2.1.0.tgz", + "integrity": "sha1-ULZ51WNc34Rme9yOWa9OW4HV9go=", + "dev": true }, "serve-static": { "version": "1.13.2", @@ -3831,27 +7218,27 @@ "escape-html": "~1.0.3", "parseurl": "~1.3.2", "send": "0.16.2" + } + }, + "set-value": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.0.tgz", + "integrity": "sha512-hw0yxk9GT/Hr5yJEYnHNKYXkIA8mVJgd9ditYZCe16ZczcaELYYcfvaXesNACk2O8O0nTiPQcQhGUQj8JLzeeg==", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-extendable": "^0.1.1", + "is-plain-object": "^2.0.3", + "split-string": "^3.0.1" }, "dependencies": { - "send": { - "version": "0.16.2", - "resolved": "https://registry.npmjs.org/send/-/send-0.16.2.tgz", - "integrity": "sha512-E64YFPUssFHEFBvpbbjr44NCLtI1AohxQ8ZSiJjQLskAdKuriYEP6VyGEsRDH8ScozGpkaX1BGvhanqCwkcEZw==", + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", "dev": true, "requires": { - "debug": "2.6.9", - "depd": "~1.1.2", - "destroy": "~1.0.4", - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "etag": "~1.8.1", - "fresh": "0.5.2", - "http-errors": "~1.6.2", - "mime": "1.4.1", - "ms": "2.0.0", - "on-finished": "~2.3.0", - "range-parser": "~1.2.0", - "statuses": "~1.4.0" + "is-extendable": "^0.1.0" } } } @@ -3864,16 +7251,37 @@ }, "sha.js": { "version": "2.4.11", - "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz", + "resolved": "http://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz", "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==", "requires": { "inherits": "^2.0.1", "safe-buffer": "^5.0.1" } }, + "shebang-command": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", + "dev": true, + "requires": { + "shebang-regex": "^1.0.0" + } + }, + "shebang-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", + "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", + "dev": true + }, + "signal-exit": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", + "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", + "dev": true + }, "sinon": { "version": "4.5.0", - "resolved": "https://registry.npmjs.org/sinon/-/sinon-4.5.0.tgz", + "resolved": "http://registry.npmjs.org/sinon/-/sinon-4.5.0.tgz", "integrity": "sha512-trdx+mB0VBBgoYucy6a9L7/jfQOmvGeaKZT4OOJ+lPAtI8623xyGr8wLiE4eojzBS8G9yXbhx42GHUOVLr4X2w==", "dev": true, "requires": { @@ -3887,9 +7295,9 @@ }, "dependencies": { "supports-color": { - "version": "5.4.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.4.0.tgz", - "integrity": "sha512-zjaXglF5nnWpsq470jSv6P9DwPvgLkuapYmfDm3JWOm0vkNTVF2tI4UrN2r6jH1qM/uc/WtxYY1hYoA2dOKj5w==", + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", "dev": true, "requires": { "has-flag": "^3.0.0" @@ -3897,28 +7305,217 @@ } } }, - "slash": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-1.0.0.tgz", - "integrity": "sha1-xB8vbDn8FtHNF61LXYlhFK5HDVU=" - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=" + "slash": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-1.0.0.tgz", + "integrity": "sha1-xB8vbDn8FtHNF61LXYlhFK5HDVU=" + }, + "slice-ansi": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-1.0.0.tgz", + "integrity": "sha512-POqxBK6Lb3q6s047D/XsDVNPnF9Dl8JSaqe9h9lURl0OdNqy/ujDrOiIHtsqXMGbWWTIomRzAMaTyawAU//Reg==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0" + } + }, + "slide": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/slide/-/slide-1.1.6.tgz", + "integrity": "sha1-VusCfWW00tzmyy4tMsTUr8nh1wc=", + "dev": true + }, + "snapdragon": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz", + "integrity": "sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==", + "dev": true, + "requires": { + "base": "^0.11.1", + "debug": "^2.2.0", + "define-property": "^0.2.5", + "extend-shallow": "^2.0.1", + "map-cache": "^0.2.2", + "source-map": "^0.5.6", + "source-map-resolve": "^0.5.0", + "use": "^3.1.0" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "snapdragon-node": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz", + "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==", + "dev": true, + "requires": { + "define-property": "^1.0.0", + "isobject": "^3.0.0", + "snapdragon-util": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "kind-of": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz", + "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==", + "dev": true + } + } + }, + "snapdragon-util": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/snapdragon-util/-/snapdragon-util-3.0.1.tgz", + "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==", + "dev": true, + "requires": { + "kind-of": "^3.2.0" + } + }, + "source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=" + }, + "source-map-resolve": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.2.tgz", + "integrity": "sha512-MjqsvNwyz1s0k81Goz/9vRBe9SZdB09Bdw+/zYyO+3CuPk6fouTaxscHkgtE8jKvf01kVfl8riHzERQ/kefaSA==", + "dev": true, + "requires": { + "atob": "^2.1.1", + "decode-uri-component": "^0.2.0", + "resolve-url": "^0.2.1", + "source-map-url": "^0.4.0", + "urix": "^0.1.0" + } + }, + "source-map-support": { + "version": "0.4.18", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.4.18.tgz", + "integrity": "sha512-try0/JqxPLF9nOjvSta7tVondkP5dwgyLDjVoyMDlmjugT2lRZ1OfsrYTkCd2hkDnJTKRbO/Rl3orm8vlsUzbA==", + "requires": { + "source-map": "^0.5.6" + } + }, + "source-map-url": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.0.tgz", + "integrity": "sha1-PpNdfd1zYxuXZZlW1VEo6HtQhKM=", + "dev": true + }, + "spdx-correct": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.0.2.tgz", + "integrity": "sha512-q9hedtzyXHr5S0A1vEPoK/7l8NpfkFYTq6iCY+Pno2ZbdZR6WexZFtqeVGkGxW3TEJMN914Z55EnAGMmenlIQQ==", + "dev": true, + "requires": { + "spdx-expression-parse": "^3.0.0", + "spdx-license-ids": "^3.0.0" + } + }, + "spdx-exceptions": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.2.0.tgz", + "integrity": "sha512-2XQACfElKi9SlVb1CYadKDXvoajPgBVPn/gOQLrTvHdElaVhr7ZEbqJaRnJLVNeaI4cMEAgVCeBMKF6MWRDCRA==", + "dev": true + }, + "spdx-expression-parse": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.0.tgz", + "integrity": "sha512-Yg6D3XpRD4kkOmTpdgbUiEJFKghJH03fiC1OPll5h/0sO6neh2jqRDVHOQ4o/LMea0tgCkbMgea5ip/e+MkWyg==", + "dev": true, + "requires": { + "spdx-exceptions": "^2.1.0", + "spdx-license-ids": "^3.0.0" + } + }, + "spdx-license-ids": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.1.tgz", + "integrity": "sha512-TfOfPcYGBB5sDuPn3deByxPhmfegAhpDYKSOXZQN81Oyrrif8ZCodOLzK3AesELnCx03kikhyDwh0pfvvQvF8w==", + "dev": true }, - "source-map-support": { - "version": "0.4.18", - "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.4.18.tgz", - "integrity": "sha512-try0/JqxPLF9nOjvSta7tVondkP5dwgyLDjVoyMDlmjugT2lRZ1OfsrYTkCd2hkDnJTKRbO/Rl3orm8vlsUzbA==", + "split-string": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz", + "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", + "dev": true, "requires": { - "source-map": "^0.5.6" + "extend-shallow": "^3.0.0" } }, + "sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", + "dev": true + }, "sshpk": { - "version": "1.14.2", - "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.14.2.tgz", - "integrity": "sha1-xvxhZIo9nE52T9P8306hBeSSupg=", + "version": "1.15.1", + "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.15.1.tgz", + "integrity": "sha512-mSdgNUaidk+dRU5MhYtN9zebdzF2iG0cNPWy8HG+W8y+fT1JnSkh0fzzpjOa0L7P8i1Rscz38t0h4gPcKz43xA==", "requires": { "asn1": "~0.2.3", "assert-plus": "^1.0.0", @@ -3931,6 +7528,33 @@ "tweetnacl": "~0.14.0" } }, + "stack-utils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-1.0.1.tgz", + "integrity": "sha1-1PM6tU6OOHeLDKXP07OvsS22hiA=", + "dev": true + }, + "static-extend": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz", + "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=", + "dev": true, + "requires": { + "define-property": "^0.2.5", + "object-copy": "^0.1.0" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + } + } + }, "statuses": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.4.0.tgz", @@ -3944,9 +7568,36 @@ }, "string-range": { "version": "1.2.2", - "resolved": "https://registry.npmjs.org/string-range/-/string-range-1.2.2.tgz", + "resolved": "http://registry.npmjs.org/string-range/-/string-range-1.2.2.tgz", "integrity": "sha1-qJPtNH5yKZvIO++78qaSqNI51d0=" }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + }, "string_decoder": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", @@ -3957,25 +7608,103 @@ }, "strip-ansi": { "version": "3.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "resolved": "http://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", "requires": { "ansi-regex": "^2.0.0" } }, + "strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", + "dev": true + }, + "strip-bom-buf": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/strip-bom-buf/-/strip-bom-buf-1.0.0.tgz", + "integrity": "sha1-HLRar1dTD0yvhsf3UXnSyaUd1XI=", + "dev": true, + "requires": { + "is-utf8": "^0.2.1" + } + }, + "strip-eof": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz", + "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=", + "dev": true + }, + "strip-indent": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-indent/-/strip-indent-2.0.0.tgz", + "integrity": "sha1-XvjbKV0B5u1sv3qrlpmNeCJSe2g=", + "dev": true + }, + "strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", + "dev": true + }, + "supertap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supertap/-/supertap-1.0.0.tgz", + "integrity": "sha512-HZJ3geIMPgVwKk2VsmO5YHqnnJYl6bV5A9JW2uzqV43WmpgliNEYbuvukfor7URpaqpxuw3CfZ3ONdVbZjCgIA==", + "dev": true, + "requires": { + "arrify": "^1.0.1", + "indent-string": "^3.2.0", + "js-yaml": "^3.10.0", + "serialize-error": "^2.1.0", + "strip-ansi": "^4.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + }, "supports-color": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=" }, + "symbol-observable": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.2.0.tgz", + "integrity": "sha512-e900nM8RRtGhlV36KGEU9k65K3mPb1WV70OdjfxlG2EAuM1noi/E/BaW/uMhL7bPEssK8QV57vN3esixjUvcXQ==", + "dev": true + }, "symbol-tree": { "version": "3.2.2", "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.2.tgz", "integrity": "sha1-rifbOPZgp64uHDt9G8KQgZuFGeY=" }, + "term-size": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/term-size/-/term-size-1.2.0.tgz", + "integrity": "sha1-RYuDiH8oj8Vtb/+/rSYuJmOO+mk=", + "dev": true, + "requires": { + "execa": "^0.7.0" + } + }, "text-encoding": { "version": "0.6.4", - "resolved": "https://registry.npmjs.org/text-encoding/-/text-encoding-0.6.4.tgz", + "resolved": "http://registry.npmjs.org/text-encoding/-/text-encoding-0.6.4.tgz", "integrity": "sha1-45mpgiV6J22uQou5KEXLcb3CbRk=", "dev": true }, @@ -4003,7 +7732,7 @@ }, "readable-stream": { "version": "1.1.14", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", + "resolved": "http://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", "dev": true, "requires": { @@ -4030,11 +7759,65 @@ } } }, + "time-zone": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/time-zone/-/time-zone-1.0.0.tgz", + "integrity": "sha1-mcW/VZWJZq9tBtg73zgA3IL67F0=", + "dev": true + }, + "timed-out": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/timed-out/-/timed-out-4.0.1.tgz", + "integrity": "sha1-8y6srFoXW+ol1/q1Zas+2HQe9W8=", + "dev": true + }, "to-fast-properties": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-1.0.3.tgz", "integrity": "sha1-uDVx+k2MJbguIxsG46MFXeTKGkc=" }, + "to-object-path": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", + "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "to-regex": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/to-regex/-/to-regex-3.0.2.tgz", + "integrity": "sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==", + "dev": true, + "requires": { + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "regex-not": "^1.0.2", + "safe-regex": "^1.1.0" + } + }, + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", + "dev": true, + "requires": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + }, + "dependencies": { + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + } + } + }, "tough-cookie": { "version": "2.4.3", "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.4.3.tgz", @@ -4059,6 +7842,18 @@ "punycode": "^2.1.0" } }, + "trim-newlines": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/trim-newlines/-/trim-newlines-2.0.0.tgz", + "integrity": "sha1-tAPQuRvlDDMd/EuC7s6yLD3hbSA=", + "dev": true + }, + "trim-off-newlines": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/trim-off-newlines/-/trim-off-newlines-1.0.1.tgz", + "integrity": "sha1-n5up2e+odkw4dpi8v+sshI8RrbM=", + "dev": true + }, "trim-right": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/trim-right/-/trim-right-1.0.1.tgz", @@ -4075,8 +7870,7 @@ "tweetnacl": { "version": "0.14.5", "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", - "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", - "optional": true + "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=" }, "type-check": { "version": "0.3.2", @@ -4112,18 +7906,210 @@ "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-1.0.4.tgz", "integrity": "sha1-m7i6DoQfs/TPH+fCRenz+opf6Zw=" }, + "uid2": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/uid2/-/uid2-0.0.3.tgz", + "integrity": "sha1-SDEm4Rd03y9xuLY53NeZw3YWK4I=", + "dev": true + }, "underscore": { "version": "1.9.1", "resolved": "https://registry.npmjs.org/underscore/-/underscore-1.9.1.tgz", "integrity": "sha512-5/4etnCkd9c8gwgowi5/om/mYO5ajCaOgdzj/oW+0eQV9WxKBDZw5+ycmKmeaTXjInS/W0BzpGLo2xR2aBwZdg==", "dev": true }, + "unicode-canonical-property-names-ecmascript": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-1.0.4.tgz", + "integrity": "sha512-jDrNnXWHd4oHiTZnx/ZG7gtUTVp+gCcTTKr8L0HjlwphROEW3+Him+IpvC+xcJEFegapiMZyZe02CyuOnRmbnQ==", + "dev": true + }, + "unicode-match-property-ecmascript": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-1.0.4.tgz", + "integrity": "sha512-L4Qoh15vTfntsn4P1zqnHulG0LdXgjSO035fEpdtp6YxXhMT51Q6vgM5lYdG/5X3MjS+k/Y9Xw4SFCY9IkR0rg==", + "dev": true, + "requires": { + "unicode-canonical-property-names-ecmascript": "^1.0.4", + "unicode-property-aliases-ecmascript": "^1.0.4" + } + }, + "unicode-match-property-value-ecmascript": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-1.0.2.tgz", + "integrity": "sha512-Rx7yODZC1L/T8XKo/2kNzVAQaRE88AaMvI1EF/Xnj3GW2wzN6fop9DDWuFAKUVFH7vozkz26DzP0qyWLKLIVPQ==", + "dev": true + }, + "unicode-property-aliases-ecmascript": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-1.0.4.tgz", + "integrity": "sha512-2WSLa6OdYd2ng8oqiGIWnJqyFArvhn+5vgx5GTxMbUYjCYKUcuKS62YLFF0R/BDGlB1yzXjQOLtPAfHsgirEpg==", + "dev": true + }, + "union-value": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.0.tgz", + "integrity": "sha1-XHHDTLW61dzr4+oM0IIHulqhrqQ=", + "dev": true, + "requires": { + "arr-union": "^3.1.0", + "get-value": "^2.0.6", + "is-extendable": "^0.1.1", + "set-value": "^0.4.3" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "set-value": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/set-value/-/set-value-0.4.3.tgz", + "integrity": "sha1-fbCPnT0i3H945Trzw79GZuzfzPE=", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-extendable": "^0.1.1", + "is-plain-object": "^2.0.1", + "to-object-path": "^0.3.0" + } + } + } + }, + "unique-string": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-1.0.0.tgz", + "integrity": "sha1-nhBXzKhRq7kzmPizOuGHuZyuwRo=", + "dev": true, + "requires": { + "crypto-random-string": "^1.0.0" + } + }, + "unique-temp-dir": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unique-temp-dir/-/unique-temp-dir-1.0.0.tgz", + "integrity": "sha1-bc6VsmgcoAPuv7MEpBX5y6vMU4U=", + "dev": true, + "requires": { + "mkdirp": "^0.5.1", + "os-tmpdir": "^1.0.1", + "uid2": "0.0.3" + } + }, + "universalify": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", + "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", + "dev": true + }, "unpipe": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=", "dev": true }, + "unset-value": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", + "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=", + "dev": true, + "requires": { + "has-value": "^0.3.1", + "isobject": "^3.0.0" + }, + "dependencies": { + "has-value": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", + "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", + "dev": true, + "requires": { + "get-value": "^2.0.3", + "has-values": "^0.1.4", + "isobject": "^2.0.0" + }, + "dependencies": { + "isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", + "dev": true, + "requires": { + "isarray": "1.0.0" + } + } + } + }, + "has-values": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", + "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=", + "dev": true + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "unzip-response": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/unzip-response/-/unzip-response-2.0.1.tgz", + "integrity": "sha1-0vD3N9FrBhXnKmk17QQhRXLVb5c=", + "dev": true + }, + "upath": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/upath/-/upath-1.1.0.tgz", + "integrity": "sha512-bzpH/oBhoS/QI/YtbkqCg6VEiPYjSZtrHQM6/QnJS6OL9pKUFLqb3aFh4Scvwm45+7iAgiMkLhSbaZxUqmrprw==", + "dev": true + }, + "update-notifier": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-2.5.0.tgz", + "integrity": "sha512-gwMdhgJHGuj/+wHJJs9e6PcCszpxR1b236igrOkUofGhqJuG+amlIKwApH1IW1WWl7ovZxsX49lMBWLxSdm5Dw==", + "dev": true, + "requires": { + "boxen": "^1.2.1", + "chalk": "^2.0.1", + "configstore": "^3.0.0", + "import-lazy": "^2.1.0", + "is-ci": "^1.0.10", + "is-installed-globally": "^0.1.0", + "is-npm": "^1.0.0", + "latest-version": "^3.0.0", + "semver-diff": "^2.0.0", + "xdg-basedir": "^3.0.0" + } + }, + "urix": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", + "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", + "dev": true + }, + "url-parse-lax": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-1.0.0.tgz", + "integrity": "sha1-evjzA2Rem9eaJy56FKxovAYJ2nM=", + "dev": true, + "requires": { + "prepend-http": "^1.0.1" + } + }, + "use": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz", + "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", + "dev": true + }, "util-deprecate": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", @@ -4140,6 +8126,16 @@ "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.3.2.tgz", "integrity": "sha512-yXJmeNaw3DnnKAOKJE51sL/ZaYfWJRl1pK9dr19YFCu0ObS231AB1/LbqTKRAQ5kw8A90rA6fr4riOUpTZvQZA==" }, + "validate-npm-package-license": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", + "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", + "dev": true, + "requires": { + "spdx-correct": "^3.0.0", + "spdx-expression-parse": "^3.0.0" + } + }, "vary": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", @@ -4169,23 +8165,38 @@ "browser-process-hrtime": "^0.1.2" } }, + "wcwidth": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/wcwidth/-/wcwidth-1.0.1.tgz", + "integrity": "sha1-8LDc+RW8X/FSivrbLA4XtTLaL+g=", + "dev": true, + "requires": { + "defaults": "^1.0.3" + } + }, "webidl-conversions": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-4.0.2.tgz", "integrity": "sha512-YQ+BmxuTgd6UXZW3+ICGfyqRyHXVlD5GtQr5+qjiNW7bF0cqrzX500HVXPBOvgXb5YnzDd+h0zqyv61KUD7+Sg==" }, + "well-known-symbols": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/well-known-symbols/-/well-known-symbols-2.0.0.tgz", + "integrity": "sha512-ZMjC3ho+KXo0BfJb7JgtQ5IBuvnShdlACNkKkdsqBmYw3bPAaJfPeYUo6tLUaT5tG/Gkh7xkpBhKRQ9e7pyg9Q==", + "dev": true + }, "whatwg-encoding": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-1.0.3.tgz", - "integrity": "sha512-jLBwwKUhi8WtBfsMQlL4bUUcT8sMkAtQinscJAe/M4KHCkHuUJAF6vuB0tueNIw4c8ziO6AkRmgY+jL3a0iiPw==", + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-1.0.5.tgz", + "integrity": "sha512-b5lim54JOPN9HtzvK9HFXvBma/rnfFeqsic0hSpjtDbVxR3dJKLc+KB4V6GgiGOvl7CY/KNh8rxSo9DKQrnUEw==", "requires": { - "iconv-lite": "0.4.19" + "iconv-lite": "0.4.24" } }, "whatwg-mimetype": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-2.1.0.tgz", - "integrity": "sha512-FKxhYLytBQiUKjkYteN71fAUA3g6KpNXoho1isLiLSB3N1G4F35Q5vUxWfKFhBwi5IWF27VE6WxhrnnC+m0Mew==" + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-2.2.0.tgz", + "integrity": "sha512-5YSO1nMd5D1hY3WzAQV3PzZL83W3YeyR1yW9PcH26Weh1t+Vzh9B6XkDh7aXm83HBZ4nSMvkjvN2H2ySWIvBgw==" }, "whatwg-url": { "version": "6.5.0", @@ -4197,6 +8208,24 @@ "webidl-conversions": "^4.0.2" } }, + "which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + }, + "widest-line": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-2.0.1.tgz", + "integrity": "sha512-Ba5m9/Fa4Xt9eb2ELXt77JxVDV8w7qQrH0zS/TWSJdLyAwQjWoOzpzj5lwVftDz6n/EOu3tNACS84v509qwnJA==", + "dev": true, + "requires": { + "string-width": "^2.1.1" + } + }, "wordwrap": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", @@ -4207,15 +8236,31 @@ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" }, + "write-file-atomic": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-2.3.0.tgz", + "integrity": "sha512-xuPeK4OdjWqtfi59ylvVL0Yn35SF3zgcAcv7rBPFHVaEapaDr4GdGgm3j7ckTwH9wHL7fGmgfAnb0+THrHb8tA==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.11", + "imurmurhash": "^0.1.4", + "signal-exit": "^3.0.2" + } + }, "ws": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ws/-/ws-4.1.0.tgz", - "integrity": "sha512-ZGh/8kF9rrRNffkLFV4AzhvooEclrOH0xaugmqGsIfFgOE/pIz4fMc4Ef+5HSQqTEug2S9JZIWDR47duDSLfaA==", + "version": "5.2.2", + "resolved": "https://registry.npmjs.org/ws/-/ws-5.2.2.tgz", + "integrity": "sha512-jaHFD6PFv6UgoIVda6qZllptQsMlDEJkTQcybzzXDYM1XO9Y8em691FGMPmM46WGyLU4z9KMgQN+qrux/nhlHA==", "requires": { - "async-limiter": "~1.0.0", - "safe-buffer": "~5.1.0" + "async-limiter": "~1.0.0" } }, + "xdg-basedir": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-3.0.0.tgz", + "integrity": "sha1-SWsswQnsqNus/i3HK2A8F8WHCtQ=", + "dev": true + }, "xml-name-validator": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz", @@ -4225,6 +8270,21 @@ "version": "2.2.0", "resolved": "https://registry.npmjs.org/xtend/-/xtend-2.2.0.tgz", "integrity": "sha1-7vax8ZjByN6vrYsXZaBNrUoBxak=" + }, + "yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=", + "dev": true + }, + "yargs-parser": { + "version": "10.1.0", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-10.1.0.tgz", + "integrity": "sha512-VCIyR1wJoEBZUqk5PA+oOBF6ypbwh5aNB3I50guxAL/quggdfs4TtNHQrSazFA3fYZ+tEqfs0zIGlv0c/rgjbQ==", + "dev": true, + "requires": { + "camelcase": "^4.1.0" + } } } } diff --git a/package.json b/package.json index 00a7213..e9feda7 100644 --- a/package.json +++ b/package.json @@ -13,22 +13,20 @@ "memserver": "cli.js" }, "scripts": { - "test": "sh run-tests.sh", + "test": "node_modules/.bin/ava --verbose", "bin": "node cli.js", "build:browser": "rollup -c --environment BUILD:development", "build:model": "rollup -i ./lib/model.js -o model.js -f cjs", - "build:model-iife": "rollup -i ./lib/model.js -o model.js -f cjs", "build:response": "rollup -i ./lib/response.js -o response.js -f cjs", "build:memserver-cjs": "rollup -i ./lib/mem-server.js -o ./lib/mem-server-cjs.js -f cjs", "prepublish": "npm run build:model && npm run build:response && npm run build:memserver-cjs" }, "dependencies": { - "@std/esm": "^0.12.5", "babel-preset-env": "^1.6.1", "babel-register": "^6.26.0", "chalk": "^2.3.0", "ember-cli-string-utils": "^1.1.0", - "esm": "^3.0.26", + "esm": "^3.0.84", "fake-xml-http-request": "^1.6.0", "i": "^0.3.6", "jsdom": "^11.3.0", @@ -43,18 +41,23 @@ "route-recognizer": "^0.3.3" }, "devDependencies": { + "ava": "1.0.0-rc.1", "cors": "^2.8.4", "express": "^4.16.2", "faker": "^4.1.0", + "fs-extra": "^7.0.0", "jquery": "^3.2.1", "mkdirp": "^0.5.1", - "mocha": "^4.0.1", "moment": "^2.19.1", "pryjs": "^1.0.3", "rimraf": "^2.6.2", "sinon": "^4.0.1" }, - "@std/esm": { - "esm": "all" + "ava": { + "require": [ + "esm" + ], + "concurrency": 1, + "failFast": true } } diff --git a/rollup.config.js b/rollup.config.js index ef591a3..35298ef 100644 --- a/rollup.config.js +++ b/rollup.config.js @@ -11,10 +11,11 @@ import Inflector from 'i'; import { classify, dasherize } from 'ember-cli-string-utils'; import virtual from 'rollup-plugin-virtual'; +const CWD = process.cwd(); const { pluralize } = Inflector(); -const modelFileNames = fs.readdirSync(`${process.cwd()}/memserver/models`); -const targetMemServerPath = `${process.cwd()}/memserver/server.js`; +const modelFileNames = fs.readdirSync(`${CWD}/memserver/models`); +const targetMemServerPath = `${CWD}/memserver/server.js`; if (!fs.existsSync(targetMemServerPath)) { throw new Error('memserver/server.js doesnt exist! Please create a memserver/server.js to use MemServer'); @@ -26,7 +27,7 @@ const targetInput = IS_DEVELOPMENT ? 'lib/browser.js' : `${require.resolve('mems export default { input: targetInput, output: { - file: IS_DEVELOPMENT ? 'dist/memserver.dist.js' : `${process.cwd()}/dist/memserver.dist.js`, + file: IS_DEVELOPMENT ? 'dist/memserver.dist.js' : `${CWD}/dist/memserver.dist.js`, format: 'iife' }, name: 'MEMSERVER', @@ -37,7 +38,7 @@ export default { '_memserver_fixtures': generateInMemoryFixturesImport(modelFileNames), '_memserver_initializer': generateInitializerImport(), '_memserver': ` - import server from '${process.cwd()}/memserver/server'; + import server from '${CWD}/memserver/server'; export default server;` }), resolve({ jsnext: true }), @@ -59,7 +60,7 @@ function generateInMemoryFixturesImport(modelFileNames) { const imports = modelFileNames.reduce((codeText, modelFileName) => { const modelName = classify(modelFileName.slice(0, -3)); - return codeText + `import ${modelName} from '${process.cwd()}/memserver/fixtures/${dasherize(pluralize(modelName))}';`; + return codeText + `import ${modelName} from '${CWD}/memserver/fixtures/${dasherize(pluralize(modelName))}';`; }, ''); return imports.concat(createObjectFromModels(modelFileNames)); @@ -69,7 +70,7 @@ function generateInMemoryModelsImport(modelFileNames) { const imports = modelFileNames.reduce((codeText, modelFileName) => { const modelName = classify(modelFileName.slice(0, -3)); - return codeText + `import ${modelName} from '${process.cwd()}/memserver/models/${modelFileName}';`; + return codeText + `import ${modelName} from '${CWD}/memserver/models/${modelFileName}';`; }, ''); return imports.concat(createObjectFromModels(modelFileNames)); @@ -88,10 +89,10 @@ function createObjectFromModels(modelFileNames) { } function generateInitializerImport() { - const initializerExists = fs.existsSync(`${process.cwd()}/memserver/initializer`); + const initializerExists = fs.existsSync(`${CWD}/memserver/initializer`); if (initializerExists) { - return `import server from '${process.cwd()}/memserver/server'; + return `import server from '${CWD}/memserver/server'; export default server;`; } diff --git a/test/cli/generator-fixtures.js b/test/cli/generator-fixtures.js new file mode 100644 index 0000000..dd710db --- /dev/null +++ b/test/cli/generator-fixtures.js @@ -0,0 +1,496 @@ +import test from 'ava'; +import fs from 'fs-extra'; +import child_process from 'child_process'; + +const CWD = process.cwd(); +const shell = child_process.exec; + +test.beforeEach(async () => { + await fs.remove(`${CWD}/memserver`); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.afterEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + await fs.remove(`${CWD}/memserver`); +}); + +test.serial.cb('$ memserver g fixtures | without memserver directory raises error', (t) => { + t.plan(3); + + fs.exists(`${CWD}/memserver`).then((folderExistence) => { + t.true(!folderExistence); + + shell(`node ${CWD}/cli.js generate fixtures`, (error, stdout) => { + t.is(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); + + shell(`node ${CWD}/cli.js g fixtures`, (error, stdout) => { + t.is(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); + + t.end(); + }); + }); + }); +}); + +test.serial.cb('$ memserver g fixtures [modelName] | without memserver directory raises error', (t) => { + t.plan(3); + + fs.exists(`${CWD}/memserver`).then((folderExistence) => { + t.true(!folderExistence); + + shell(`node ${process.cwd()}/cli.js generate fixtures user`, (error, stdout) => { + t.is(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); + + shell(`node ${process.cwd()}/cli.js g fixtures`, (error, stdout) => { + t.is(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); + + t.end(); + }); + }); + }); +}); + +test.serial.cb('$ memserver g fixtures | works for the entire state', (t) => { + t.plan(9); + + generateMemServerState().then(() => { + shell(`node ${CWD}/cli.js g fixtures`, async (error, stdout) => { + t.true(stdout.includes('[MemServer] data written to /fixtures/users.js')); + t.true(stdout.includes('[MemServer] data written to /fixtures/likes.js')); + t.true(stdout.includes('[MemServer] data written to /fixtures/photos.js')); + t.true(stdout.includes('[MemServer] data written to /fixtures/photo-comments.js')); + + const fixturesPath = `${CWD}/memserver/fixtures`; + + t.deepEqual(await fs.readdir(fixturesPath), [ + 'likes.js', 'photo-comments.js', 'photos.js', 'users.js' + ]); + t.deepEqual(require(`${fixturesPath}/users.js`).default, [ + { + password: '123456', + authentication_token: '12or12rnfasdfzlemfp1m3epfm134', + id: 1, + first_name: 'Izel', + last_name: 'Nakri' + } + ]); + t.deepEqual(require(`${fixturesPath}/photos.js`).default, [ + { id: 1, + name: 'Me skiing', + href: 'ski-trip.jpeg', + is_public: false }, + { id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true }, + { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false }, + { id: 4, + name: 'Travel photo', + href: 'travel-photo.jpeg', + is_public: false } + ]); + t.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ + { uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 }, + { uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 }, + { uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 }, + { uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 }, + { uuid: '374c7f4a-85d6-429a-bf2a-0719525faaaa', + content: 'This is badass', + photo_id: 1, + user_id: 1 } + ]); + t.deepEqual(require(`${fixturesPath}/likes.js`).default, []); + + t.end(); + }); + }); +}); + +test.serial.cb('$ memserver generate fixtures [modelName] works', (t) => { + t.plan(12); + + generateMemServerState().then(() => { + shell(`node ${CWD}/cli.js generate fixtures users`, async (error, stdout) => { + t.true(stdout.includes('[MemServer] data written to /fixtures/users.js')); + + const fixturesPath = `${CWD}/memserver/fixtures`; + + t.deepEqual(await fs.readdir(fixturesPath), [ + 'photo-comments.js', 'photos.js', 'users.js' + ]); + t.deepEqual(require(`${fixturesPath}/users.js`).default, [ + { + password: '123456', + authentication_token: '12or12rnfasdfzlemfp1m3epfm134', + id: 1, + first_name: 'Izel', + last_name: 'Nakri' + } + ]); + t.deepEqual(require(`${fixturesPath}/photos.js`).default, [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ]); + t.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ]); + t.true(!await fs.exists(`${fixturesPath}/likes.js`)); + + shell(`node ${process.cwd()}/cli.js generate fixtures photos`, async (error, stdout) => { + t.true(stdout.includes('[MemServer] data written to /fixtures/photos.js')); + + t.deepEqual(await fs.readdir(fixturesPath), [ + 'photo-comments.js', 'photos.js', 'users.js' + ]); + t.deepEqual(require(`${fixturesPath}/users.js`).default, [ + { + password: '123456', + authentication_token: '12or12rnfasdfzlemfp1m3epfm134', + id: 1, + first_name: 'Izel', + last_name: 'Nakri' + } + ]); + + const photosFixtureData = (await fs.readFile(`${fixturesPath}/photos.js`)).toString(); + + t.deepEqual(eval(photosFixtureData.slice(15, photosFixtureData.length - 1)), [ // TODO: babel cache: false doesnt work + { id: 1, + name: 'Me skiing', + href: 'ski-trip.jpeg', + is_public: false }, + { id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true }, + { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false }, + { id: 4, + name: 'Travel photo', + href: 'travel-photo.jpeg', + is_public: false } + ]); + t.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ]); + t.true(!await fs.exists(`${fixturesPath}/likes.js`)); + + t.end(); + }); + }); + }); +}); + +test.serial.cb('$ memserver g fixtures [modelName] works', (t) => { + t.plan(12); + + generateMemServerState().then(() => { + shell(`node ${CWD}/cli.js g fixtures users`, async (error, stdout) => { + t.true(stdout.includes('[MemServer] data written to /fixtures/users.js')); + + const fixturesPath = `${CWD}/memserver/fixtures`; + + t.deepEqual(await fs.readdir(fixturesPath), [ + 'photo-comments.js', 'photos.js', 'users.js' + ]); + t.deepEqual(require(`${fixturesPath}/users.js`).default, [ + { + password: '123456', + authentication_token: '12or12rnfasdfzlemfp1m3epfm134', + id: 1, + first_name: 'Izel', + last_name: 'Nakri' + } + ]); + t.deepEqual(require(`${fixturesPath}/photos.js`).default, [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ]); + t.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ]); + t.true(!await fs.exists(`${fixturesPath}/likes.js`)); + + shell(`node ${CWD}/cli.js g fixtures photos`, async (error, stdout) => { + t.true(stdout.includes('[MemServer] data written to /fixtures/photos.js')); + + t.deepEqual(await fs.readdir(fixturesPath), [ + 'photo-comments.js', 'photos.js', 'users.js' + ]); + t.deepEqual(require(`${fixturesPath}/users.js`).default, [ + { + password: '123456', + authentication_token: '12or12rnfasdfzlemfp1m3epfm134', + id: 1, + first_name: 'Izel', + last_name: 'Nakri' + } + ]); + + const photosFixtureData = (await fs.readFile(`${fixturesPath}/photos.js`)).toString(); + + t.deepEqual(eval(photosFixtureData.slice(15, photosFixtureData.length - 1)), [ // TODO: babel cache: false doesnt work + { id: 1, + name: 'Me skiing', + href: 'ski-trip.jpeg', + is_public: false }, + { id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true }, + { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false }, + { id: 4, + name: 'Travel photo', + href: 'travel-photo.jpeg', + is_public: false } + ]); + t.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ]); + t.true(!await fs.exists(`${fixturesPath}/likes.js`)); + + t.end(); + }); + }); + }); +}); + +// test('$ memserver generate fixtures [modelName] with wrong modelName raises', (t) => { +// }); + +// test('$ memserver g fixtures [modelName] with wrong modelName raises', (t) => { +// }); + +function generateMemServerState() { + return new Promise(async (resolve) => { + const MODEL_FILE_CONTENT = `import Model from '${CWD}/lib/model'; + export default Model({});`; + + await fs.mkdir(`${CWD}/memserver`); + await Promise.all([ + fs.mkdir(`${CWD}/memserver/models`), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/user.js`, `import Model from '${CWD}/lib/model'; + export default Model({ + defaultAttributes: { + password: '123456', + authentication_token() { + return '12or12rnfasdfzlemfp1m3epfm134'; + } + } + });`), + fs.writeFile(`${CWD}/memserver/models/photo.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, `import Model from '${CWD}/lib/model'; + export default Model({ + createForUser(user, photo, photoOptions) { + return this.insert(Object.assign({}, photoOptions, { + photo_id: photo.id, user_id: user.id + })); + } + });`), + fs.writeFile(`${CWD}/memserver/models/likes.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}'), + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`), + fs.writeFile(`${CWD}/memserver/initializer.js`, `export default function({ User, Photo, PhotoComment }) { + const user = User.insert({ first_name: 'Izel', last_name: 'Nakri' }); + const photo = Photo.find(1); + + Photo.insert({ name: 'Travel photo', href: 'travel-photo.jpeg', is_public: false }); + PhotoComment.createForUser(user, photo, { + uuid: '374c7f4a-85d6-429a-bf2a-0719525faaaa', content: 'This is badass' + }); + + Photo.update({ id: 1, name: 'Me skiing' }); + }`) + ]); + + resolve(); + }); +} diff --git a/test/cli/generator-model.js b/test/cli/generator-model.js new file mode 100644 index 0000000..394e912 --- /dev/null +++ b/test/cli/generator-model.js @@ -0,0 +1,131 @@ +import test from 'ava'; +import fs from 'fs-extra'; +import child_process from 'child_process'; + +const CWD = process.cwd(); +const shell = child_process.exec; + +test.beforeEach(async () => { + await fs.remove(`${CWD}/memserver`); +}); + +test.serial.cb('$ memserver g | and $ memserver generate | without memserver directory raises', (t) => { + t.plan(2); + + shell(`node ${CWD}/cli.js generate`, (error, stdout) => { + t.is(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); + + shell(`node ${CWD}/cli.js g`, (error, stdout) => { + t.is(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); + + t.end(); + }); + }); +}); + +test.serial.cb('$ memserver g model | and $ memserver generate model | without memserver directory raises', (t) => { + t.plan(2); + + shell(`node ${CWD}/cli.js generate model`, (error, stdout) => { + t.is(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); + + shell(`node ${CWD}/cli.js g model`, (error, stdout) => { + t.is(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); + + t.end(); + }); + }); +}); + +test.serial.cb('$ memserver g model | and $ memserver generate model | without model raises', (t) => { + t.plan(1); + + initializeMemServer().then(() => { + shell(`node ${CWD}/cli.js generate model`, (error, stdout) => { + t.is(stdout, '[MemServer CLI] Please put a modelName to the memserver generate. Example: $ memserver generate model user\n'); + + t.end(); + }); + }); +}); + +const EXPECTED_OUTPUT = '[MemServer CLI] /memserver/models/user.js created\n' + + '[MemServer CLI] /memserver/fixtures/users.js created\n'; + +test.serial.cb('$ memserver g [modelName] | works', (t) => { + t.plan(5); + + initializeMemServer().then(async () => { + const [userModelExists, userFixturesExists] = await Promise.all([ + fs.exists(`${CWD}/memserver/models/user.js`), + fs.exists(`${CWD}/memserver/fixtures/users.js`) + ]); + + t.true(!userModelExists); + t.true(!userFixturesExists); + + shell(`node ${CWD}/cli.js g model user`, async (error, stdout) => { + t.is(stdout, EXPECTED_OUTPUT); + + const [userModelBuffer, userFixturesBuffer] = await Promise.all([ + fs.readFile(`${CWD}/memserver/models/user.js`), + fs.readFile(`${CWD}/memserver/fixtures/users.js`) + ]); + + t.is(userModelBuffer.toString(), 'import Model from \'memserver/model\';\n\nexport default Model({\n\n});'); + t.is(userFixturesBuffer.toString(), 'export default [\n];'); + + t.end(); + }); + }); +}); + +test.serial.cb('$ memserver generate [modelName] | works', (t) => { + t.plan(5); + + initializeMemServer().then(async () => { + const [userModelExists, userFixturesExists] = await Promise.all([ + fs.exists(`${CWD}/memserver/models/user.js`), + fs.exists(`${CWD}/memserver/fixtures/users.js`) + ]); + + t.true(!userModelExists); + t.true(!userFixturesExists); + + shell(`node ${process.cwd()}/cli.js generate model user`, async (error, stdout) => { + t.is(stdout, EXPECTED_OUTPUT); + + const [userModelBuffer, userFixturesBuffer] = await Promise.all([ + fs.readFile(`${CWD}/memserver/models/user.js`), + fs.readFile(`${CWD}/memserver/fixtures/users.js`) + ]); + + t.is(userModelBuffer.toString(), 'import Model from \'memserver/model\';\n\nexport default Model({\n\n});'); + t.is(userFixturesBuffer.toString(), 'export default [\n];'); + + t.end(); + }); + }); +}); + +function initializeMemServer() { + return new Promise(async (resolve) => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } + + const memServerDirectory = `${CWD}/memserver`; + + await Promise.all([ + fs.mkdir(memServerDirectory), + fs.writeFile(`${memServerDirectory}/server.js`, `export default function(Models) { + }`), + fs.writeFile(`${memServerDirectory}/initializer.js`, `export default function(Models) { + }`), + fs.mkdir(`${memServerDirectory}/fixtures`), + fs.mkdir(`${memServerDirectory}/models`) + ]); + + resolve(); + }); +} diff --git a/test/cli/index.js b/test/cli/index.js new file mode 100644 index 0000000..32ecb09 --- /dev/null +++ b/test/cli/index.js @@ -0,0 +1,117 @@ +import test from 'ava'; +import fs from 'fs-extra'; +import child_process from 'child_process'; + +const CWD = process.cwd(); +const shell = child_process.exec; + +test.afterEach.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial.cb('$ memserver | and $ memserver helper | and $ memserver h | without arguments shows help screen', (t) => { + t.plan(3); + + const expectedOutput = `[MemServer CLI] Usage: memserver + +memserver init | new # Sets up the initial memserver folder structure +memserver generate model [ModelName] # Generates the initial files for a MemServer Model [alias: "memserver g model"] +memserver generate fixtures # Outputs your initial MemServer state as pure javascript fixture files +memserver generate fixtures [ModelName] # Outputs your initial MemServer state for certain model as pure javascript fixture +memserver console # Starts a MemServer console in node.js [alias: "memserver c"] +memserver serve | server [outputFile] # Builds an ES5 javascript bundle with all your memserver code continuosly on watch [alias: "memserver s"] +memserver build | rollup [outputFile] # Builds an ES5 javascript bundle with all your memserver code +memserver version | v # Displays memserver version`; + + shell(`node ${CWD}/cli.js`, (error, stdout) => { + t.true(stdout.includes(expectedOutput)); + + shell(`node ${CWD}/cli.js help`, (error, stdout) => { + t.true(stdout.includes(expectedOutput)); + + shell(`node ${CWD}/cli.js help`, (error, stdout) => { + t.true(stdout.includes(expectedOutput)); + + t.end(); + }); + }); + }); +}); + +test.serial.cb('$ memserver init | sets up the initial folder structure', (t) => { + t.plan(6); + + t.true(!fs.existsSync(`${CWD}/memserver`)); + + const expectedOutput = '[MemServer CLI] /memserver/server.js created\n' + + '[MemServer CLI] /memserver/initializer.js created\n' + + '[MemServer CLI] /memserver/fixtures folder created\n' + + '[MemServer CLI] /memserver/models folder created\n'; + + shell(`node ${process.cwd()}/cli.js init`, async (error, stdout) => { + t.is(stdout, expectedOutput); + + const [ + serverBuffer, initializerBuffer, fixturesFolderExistence, modelsFolderExists + ] = await Promise.all([ + fs.readFile(`${CWD}/memserver/server.js`), + fs.readFile(`${CWD}/memserver/initializer.js`), + fs.exists(`${CWD}/memserver/fixtures`), + fs.exists(`${CWD}/memserver/models`) + ]); + + t.is(serverBuffer.toString(), 'export default function(Models) {\n}'); + t.is(initializerBuffer.toString(), 'export default function(Models) {\n}'); + t.true(fixturesFolderExistence); + t.true(modelsFolderExists); + + t.end(); + }); +}); + +test.serial.cb('$ memserver new | sets up the initial folder structure', (t) => { + t.plan(6); + + t.true(!fs.existsSync(`${CWD}/memserver`)); + + const expectedOutput = '[MemServer CLI] /memserver/server.js created\n' + + '[MemServer CLI] /memserver/initializer.js created\n' + + '[MemServer CLI] /memserver/fixtures folder created\n' + + '[MemServer CLI] /memserver/models folder created\n'; + + shell(`node ${CWD}/cli.js new`, async (error, stdout) => { + t.is(stdout, expectedOutput); + + const [ + serverBuffer, initializerBuffer, fixturesFolderExistence, modelsFolderExists + ] = await Promise.all([ + fs.readFile(`${CWD}/memserver/server.js`), + fs.readFile(`${CWD}/memserver/initializer.js`), + fs.exists(`${CWD}/memserver/fixtures`), + fs.exists(`${CWD}/memserver/models`) + ]); + + t.is(serverBuffer.toString(), 'export default function(Models) {\n}'); + t.is(initializerBuffer.toString(), 'export default function(Models) {\n}'); + t.true(fixturesFolderExistence); + t.true(modelsFolderExists); + + t.end(); + }); +}); + +test.serial.cb('$ memserver version | and $ memserver v', (t) => { + t.plan(2); + + shell(`node ${CWD}/cli.js v`, (error, stdout) => { + t.is(stdout, `[MemServer CLI] ${require(CWD + '/package.json').version}\n`); + + shell(`node ${CWD}/cli.js version`, (error, stdout) => { + t.is(stdout, `[MemServer CLI] ${require(CWD + '/package.json').version}\n`); + + t.end(); + }); + }); +}); diff --git a/test/cli/mem-server.cli.generator.fixtures.js b/test/cli/mem-server.cli.generator.fixtures.js deleted file mode 100644 index 2c43734..0000000 --- a/test/cli/mem-server.cli.generator.fixtures.js +++ /dev/null @@ -1,549 +0,0 @@ -require('babel-register')({ - presets: ['env'], - cache: false -}); - -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); -const shell = require('child_process').exec; - -describe('[MemServer CLI] $ memserver fixtures tests', function() { - beforeEach(function(done) { - rimraf.sync(`${process.cwd()}/memserver`); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - done(); - }); - - afterEach(function(done) { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - done(); - }); - - it('memserver g fixtures without memserver directory raises error', function(done) { - assert.ok(!fs.existsSync(`${process.cwd()}/memserver`)); - - shell(`node ${process.cwd()}/cli.js generate fixtures`, (error, stdout) => { - assert.equal(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); - - shell(`node ${process.cwd()}/cli.js g fixtures`, (error, stdout) => { - assert.equal(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); - - done(); - }); - }); - }); - - it('memserver g fixtures [modelName] without memserver directory raises error', function(done) { - assert.ok(!fs.existsSync(`${process.cwd()}/memserver`)); - - shell(`node ${process.cwd()}/cli.js generate fixtures user`, (error, stdout) => { - assert.equal(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); - - shell(`node ${process.cwd()}/cli.js g fixtures`, (error, stdout) => { - assert.equal(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); - - done(); - }); - }); - }); - - it('memserver g fixtures works for the entire state', function(done) { - this.timeout(5000); - - generateMemServerState(); - - shell(`node ${process.cwd()}/cli.js g fixtures`, (error, stdout) => { - assert.ok(stdout.includes('[MemServer] data written to /fixtures/users.js')); - assert.ok(stdout.includes('[MemServer] data written to /fixtures/likes.js')); - assert.ok(stdout.includes('[MemServer] data written to /fixtures/photos.js')); - assert.ok(stdout.includes('[MemServer] data written to /fixtures/photo-comments.js')); - - const fixturesPath = `${process.cwd()}/memserver/fixtures`; - - assert.deepEqual(fs.readdirSync(fixturesPath), [ - 'likes.js', 'photo-comments.js', 'photos.js', 'users.js' - ]); - assert.deepEqual(require(`${fixturesPath}/users.js`).default, [ - { - password: '123456', - authentication_token: '12or12rnfasdfzlemfp1m3epfm134', - id: 1, - first_name: 'Izel', - last_name: 'Nakri' - } - ]); - assert.deepEqual(require(`${fixturesPath}/photos.js`).default, [ - { id: 1, - name: 'Me skiing', - href: 'ski-trip.jpeg', - is_public: false }, - { id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true }, - { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false }, - { id: 4, - name: 'Travel photo', - href: 'travel-photo.jpeg', - is_public: false } - ]); - assert.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ - { uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 }, - { uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 }, - { uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 }, - { uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 }, - { uuid: '374c7f4a-85d6-429a-bf2a-0719525faaaa', - content: 'This is badass', - photo_id: 1, - user_id: 1 } - ]); - assert.deepEqual(require(`${fixturesPath}/likes.js`).default, []); - - done(); - }); - }); - - it('memserver generate fixtures works for the entire state', function(done) { - this.timeout(5000); - - generateMemServerState(); - - shell(`node ${process.cwd()}/cli.js generate fixtures`, (error, stdout) => { - assert.ok(stdout.includes('[MemServer] data written to /fixtures/users.js')); - assert.ok(stdout.includes('[MemServer] data written to /fixtures/likes.js')); - assert.ok(stdout.includes('[MemServer] data written to /fixtures/photos.js')); - assert.ok(stdout.includes('[MemServer] data written to /fixtures/photo-comments.js')); - - const fixturesPath = `${process.cwd()}/memserver/fixtures`; - - assert.deepEqual(fs.readdirSync(fixturesPath), [ - 'likes.js', 'photo-comments.js', 'photos.js', 'users.js' - ]); - assert.deepEqual(require(`${fixturesPath}/users.js`).default, [ - { - password: '123456', - authentication_token: '12or12rnfasdfzlemfp1m3epfm134', - id: 1, - first_name: 'Izel', - last_name: 'Nakri' - } - ]); - assert.deepEqual(require(`${fixturesPath}/photos.js`).default, [ - { id: 1, - name: 'Me skiing', - href: 'ski-trip.jpeg', - is_public: false }, - { id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true }, - { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false }, - { id: 4, - name: 'Travel photo', - href: 'travel-photo.jpeg', - is_public: false } - ]); - assert.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ - { uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 }, - { uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 }, - { uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 }, - { uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 }, - { uuid: '374c7f4a-85d6-429a-bf2a-0719525faaaa', - content: 'This is badass', - photo_id: 1, - user_id: 1 } - ]); - assert.deepEqual(require(`${fixturesPath}/likes.js`).default, []); - - done(); - }); - }); - - it('memserver generate fixtures [modelName] works', function(done) { - this.timeout(5000); - - generateMemServerState(); - - shell(`node ${process.cwd()}/cli.js generate fixtures users`, (error, stdout) => { - assert.ok(stdout.includes('[MemServer] data written to /fixtures/users.js')); - - const fixturesPath = `${process.cwd()}/memserver/fixtures`; - - assert.deepEqual(fs.readdirSync(fixturesPath), [ - 'photo-comments.js', 'photos.js', 'users.js' - ]); - assert.deepEqual(require(`${fixturesPath}/users.js`).default, [ - { - password: '123456', - authentication_token: '12or12rnfasdfzlemfp1m3epfm134', - id: 1, - first_name: 'Izel', - last_name: 'Nakri' - } - ]); - assert.deepEqual(require(`${fixturesPath}/photos.js`).default, [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ]); - assert.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ]); - assert.ok(!fs.existsSync(`${fixturesPath}/likes.js`)); - - shell(`node ${process.cwd()}/cli.js generate fixtures photos`, (error, stdout) => { - assert.ok(stdout.includes('[MemServer] data written to /fixtures/photos.js')); - - assert.deepEqual(fs.readdirSync(fixturesPath), [ - 'photo-comments.js', 'photos.js', 'users.js' - ]); - assert.deepEqual(require(`${fixturesPath}/users.js`).default, [ - { - password: '123456', - authentication_token: '12or12rnfasdfzlemfp1m3epfm134', - id: 1, - first_name: 'Izel', - last_name: 'Nakri' - } - ]); - const photosFixtureData = fs.readFileSync(`${fixturesPath}/photos.js`).toString(); - assert.deepEqual(eval(photosFixtureData.slice(15, photosFixtureData.length - 1)), [ // TODO: babel cache: false doesnt work - { id: 1, - name: 'Me skiing', - href: 'ski-trip.jpeg', - is_public: false }, - { id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true }, - { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false }, - { id: 4, - name: 'Travel photo', - href: 'travel-photo.jpeg', - is_public: false } - ]); - assert.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ]); - assert.ok(!fs.existsSync(`${fixturesPath}/likes.js`)); - - done(); - }); - }); - }); - - it('memserver g fixtures [modelName] works', function(done) { - this.timeout(5000); - - generateMemServerState(); - - shell(`node ${process.cwd()}/cli.js g fixtures users`, (error, stdout) => { - assert.ok(stdout.includes('[MemServer] data written to /fixtures/users.js')); - - const fixturesPath = `${process.cwd()}/memserver/fixtures`; - - assert.deepEqual(fs.readdirSync(fixturesPath), [ - 'photo-comments.js', 'photos.js', 'users.js' - ]); - assert.deepEqual(require(`${fixturesPath}/users.js`).default, [ - { - password: '123456', - authentication_token: '12or12rnfasdfzlemfp1m3epfm134', - id: 1, - first_name: 'Izel', - last_name: 'Nakri' - } - ]); - assert.deepEqual(require(`${fixturesPath}/photos.js`).default, [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ]); - assert.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ]); - assert.ok(!fs.existsSync(`${fixturesPath}/likes.js`)); - - shell(`node ${process.cwd()}/cli.js g fixtures photos`, (error, stdout) => { - assert.ok(stdout.includes('[MemServer] data written to /fixtures/photos.js')); - - assert.deepEqual(fs.readdirSync(fixturesPath), [ - 'photo-comments.js', 'photos.js', 'users.js' - ]); - assert.deepEqual(require(`${fixturesPath}/users.js`).default, [ - { - password: '123456', - authentication_token: '12or12rnfasdfzlemfp1m3epfm134', - id: 1, - first_name: 'Izel', - last_name: 'Nakri' - } - ]); - const photosFixtureData = fs.readFileSync(`${fixturesPath}/photos.js`).toString(); - assert.deepEqual(eval(photosFixtureData.slice(15, photosFixtureData.length - 1)), [ // TODO: babel cache: false doesnt work - { id: 1, - name: 'Me skiing', - href: 'ski-trip.jpeg', - is_public: false }, - { id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true }, - { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false }, - { id: 4, - name: 'Travel photo', - href: 'travel-photo.jpeg', - is_public: false } - ]); - assert.deepEqual(require(`${fixturesPath}/photo-comments.js`).default, [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ]); - assert.ok(!fs.existsSync(`${fixturesPath}/likes.js`)); - - done(); - }); - }); - }); - - // it('memserver generate fixtures [modelName] with wrong modelName raises', function() { - // }); - - // it('memserver g fixtures [modelName] with wrong modelName raises', function() { - // }); -}); - -function generateMemServerState() { - const modelFileContent = `import Model from '${process.cwd()}/lib/model'; - export default Model({});`; - - fs.mkdirSync('./memserver'); - fs.mkdirSync('./memserver/models'); - fs.mkdirSync('./memserver/fixtures'); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, `import Model from '${process.cwd()}/lib/model'; - export default Model({ - defaultAttributes: { - password: '123456', - authentication_token() { - return '12or12rnfasdfzlemfp1m3epfm134'; - } - } - });`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, `import Model from '${process.cwd()}/lib/model'; - export default Model({ - createForUser(user, photo, photoOptions) { - return this.insert(Object.assign({}, photoOptions, { - photo_id: photo.id, user_id: user.id - })); - } - });`); - fs.writeFileSync(`${process.cwd()}/memserver/models/likes.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/initializer.js`, `export default function({ User, Photo, PhotoComment }) { - const user = User.insert({ first_name: 'Izel', last_name: 'Nakri' }); - const photo = Photo.find(1); - - Photo.insert({ name: 'Travel photo', href: 'travel-photo.jpeg', is_public: false }); - PhotoComment.createForUser(user, photo, { - uuid: '374c7f4a-85d6-429a-bf2a-0719525faaaa', content: 'This is badass' - }); - - Photo.update({ id: 1, name: 'Me skiing' }); - }`); -} diff --git a/test/cli/mem-server.cli.generator.model.js b/test/cli/mem-server.cli.generator.model.js deleted file mode 100644 index ba8df60..0000000 --- a/test/cli/mem-server.cli.generator.model.js +++ /dev/null @@ -1,95 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); -const shell = require('child_process').exec; - -describe('[MemServer CLI] $ memserver g model [modelName] tests', function() { - beforeEach(function(done) { - rimraf.sync(`${process.cwd()}/memserver`); - - done(); - }); - - it('memserver g and memserver generate without model and memserver directory raises', function(done) { - shell(`node ${process.cwd()}/cli.js generate`, (error, stdout) => { - assert.equal(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); - - shell(`node ${process.cwd()}/cli.js g`, (error, stdout) => { - assert.equal(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); - - done(); - }); - }); - }); - - it('memserver g and memserver generate with model without memserver directory raises', function(done) { - shell(`node ${process.cwd()}/cli.js generate model`, (error, stdout) => { - assert.equal(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); - - shell(`node ${process.cwd()}/cli.js g model`, (error, stdout) => { - assert.equal(stdout, '[MemServer CLI] cannot find /memserver folder. Did you run $ memserver init ?\n'); - - done(); - }); - }); - }); - - it('memserver g and memserver generate without model raises', function(done) { - initializeMemServer(); - - shell(`node ${process.cwd()}/cli.js generate model`, (error, stdout) => { - assert.equal(stdout, '[MemServer CLI] Please put a modelName to the memserver generate. Example: $ memserver generate model user\n'); - - done(); - }); - }); - - const expectedOutput = '[MemServer CLI] /memserver/models/user.js created\n' + - '[MemServer CLI] /memserver/fixtures/users.js created\n'; - - it('memserver g [modelName] works', function(done) { - initializeMemServer(); - - assert.ok(!fs.existsSync(`${process.cwd()}/memserver/models/user.js`)); - assert.ok(!fs.existsSync(`${process.cwd()}/memserver/fixtures/users.js`)); - - shell(`node ${process.cwd()}/cli.js g model user`, (error, stdout) => { - assert.equal(stdout, expectedOutput); - assert.equal(fs.readFileSync(`${process.cwd()}/memserver/models/user.js`).toString(), 'import Model from \'memserver/model\';\n\nexport default Model({\n\n});'); - assert.equal(fs.readFileSync(`${process.cwd()}/memserver/fixtures/users.js`).toString(), 'export default [\n];'); - - done(); - }); - }); - - it('memserver generate [modelName] works', function(done) { - initializeMemServer(); - - assert.ok(!fs.existsSync(`${process.cwd()}/memserver/models/user.js`)); - assert.ok(!fs.existsSync(`${process.cwd()}/memserver/fixtures/users.js`)); - - shell(`node ${process.cwd()}/cli.js generate model user`, (error, stdout) => { - assert.equal(stdout, expectedOutput); - assert.equal(fs.readFileSync(`${process.cwd()}/memserver/models/user.js`).toString(), 'import Model from \'memserver/model\';\n\nexport default Model({\n\n});'); - assert.equal(fs.readFileSync(`${process.cwd()}/memserver/fixtures/users.js`).toString(), 'export default [\n];'); - - done(); - }); - }); -}); - -function initializeMemServer() { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - const memServerDirectory = `${process.cwd()}/memserver`; - - fs.mkdirSync(memServerDirectory); - fs.writeFileSync(`${memServerDirectory}/server.js`, `export default function(Models) { -}`); - fs.writeFileSync(`${memServerDirectory}/initializer.js`, `export default function(Models) { - }`); - fs.mkdirSync(`${memServerDirectory}/fixtures`); - fs.mkdirSync(`${memServerDirectory}/models`); -} diff --git a/test/cli/mem-server.cli.js b/test/cli/mem-server.cli.js deleted file mode 100644 index 27f8357..0000000 --- a/test/cli/mem-server.cli.js +++ /dev/null @@ -1,93 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); -const shell = require('child_process').exec; - -describe('[MemServer CLI] general commands', function() { - afterEach(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - done(); - }); - - it('$ memserver and $ memserver help and $ memserver h without arguments shows help screen', function(done) { - const expectedOutput = `[MemServer CLI] Usage: memserver - -memserver init | new # Sets up the initial memserver folder structure -memserver generate model [ModelName] # Generates the initial files for a MemServer Model [alias: "memserver g model"] -memserver generate fixtures # Outputs your initial MemServer state as pure javascript fixture files -memserver generate fixtures [ModelName] # Outputs your initial MemServer state for certain model as pure javascript fixture -memserver console # Starts a MemServer console in node.js [alias: "memserver c"] -memserver serve | server [outputFile] # Builds an ES5 javascript bundle with all your memserver code continuosly on watch [alias: "memserver s"] -memserver build | rollup [outputFile] # Builds an ES5 javascript bundle with all your memserver code -memserver version | v # Displays memserver version`; - - shell(`node ${process.cwd()}/cli.js`, (error, stdout) => { - assert.ok(stdout.includes(expectedOutput)); - - shell(`node ${process.cwd()}/cli.js help`, (error, stdout) => { - assert.ok(stdout.includes(expectedOutput)); - - shell(`node ${process.cwd()}/cli.js help`, (error, stdout) => { - assert.ok(stdout.includes(expectedOutput)); - - done(); - }); - }); - }); - }); - - it('memserver init sets up the initial folder structure', function(done) { - assert.ok(!fs.existsSync(`${process.cwd()}/memserver`)); - - const expectedOutput = '[MemServer CLI] /memserver/server.js created\n' + - '[MemServer CLI] /memserver/initializer.js created\n' + - '[MemServer CLI] /memserver/fixtures folder created\n' + - '[MemServer CLI] /memserver/models folder created\n'; - - shell(`node ${process.cwd()}/cli.js init`, (error, stdout) => { - assert.equal(stdout, expectedOutput); - - assert.equal(fs.readFileSync(`${process.cwd()}/memserver/server.js`).toString(), 'export default function(Models) {\n}'); - assert.equal(fs.readFileSync(`${process.cwd()}/memserver/initializer.js`).toString(), 'export default function(Models) {\n}'); - assert.ok(fs.existsSync(`${process.cwd()}/memserver/fixtures`)); - assert.ok(fs.existsSync(`${process.cwd()}/memserver/models`)); - - done(); - }); - }); - - it('memserver new sets up the initial folder structure', function(done) { - assert.ok(!fs.existsSync(`${process.cwd()}/memserver`)); - - const expectedOutput = '[MemServer CLI] /memserver/server.js created\n' + - '[MemServer CLI] /memserver/initializer.js created\n' + - '[MemServer CLI] /memserver/fixtures folder created\n' + - '[MemServer CLI] /memserver/models folder created\n'; - - shell(`node ${process.cwd()}/cli.js new`, (error, stdout) => { - assert.equal(stdout, expectedOutput); - - assert.equal(fs.readFileSync(`${process.cwd()}/memserver/server.js`).toString(), 'export default function(Models) {\n}'); - assert.equal(fs.readFileSync(`${process.cwd()}/memserver/initializer.js`).toString(), 'export default function(Models) {\n}'); - assert.ok(fs.existsSync(`${process.cwd()}/memserver/fixtures`)); - assert.ok(fs.existsSync(`${process.cwd()}/memserver/models`)); - - done(); - }); - }); - - it('memserver version and memserver v works', function(done) { - shell(`node ${process.cwd()}/cli.js v`, (error, stdout) => { - assert.equal(stdout, `[MemServer CLI] ${require(process.cwd() + '/package.json').version}\n`); - - shell(`node ${process.cwd()}/cli.js version`, (error, stdout) => { - assert.equal(stdout, `[MemServer CLI] ${require(process.cwd() + '/package.json').version}\n`); - - done(); - }); - }); - }); -}); diff --git a/test/mem-server.fixtures.js b/test/mem-server.fixtures.js deleted file mode 100644 index 2047845..0000000 --- a/test/mem-server.fixtures.js +++ /dev/null @@ -1,311 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -describe('MemServer fixture constraint feature', function() { - beforeEach(function() { - const modelFileContent = `import Model from '${process.cwd()}/lib/model'; - export default Model({});`; - - fs.mkdirSync('./memserver'); - fs.mkdirSync('./memserver/models'); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - afterEach(function(done) { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - done(); - }); - - it('should throw error if any of the fixtures missing id or uuid', function() { - this.timeout(5000); - - if (!fs.existsSync('./memserver/fixtures')) { - fs.mkdirSync('./memserver/fixtures'); - } - - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - assert.deepEqual(MemServer.Server, {}); - assert.deepEqual(MemServer.DB, {}); - - assert.deepEqual(Photo.findAll(), []); - assert.deepEqual(PhotoComment.findAll(), []); - - assert.throws(() => MemServer.start(), (err) => { - return (err instanceof Error) && - /\[MemServer\] DATABASE ERROR: At least one of your PhotoComment fixtures missing a primary key\. Please make sure all your PhotoComment fixtures have either id or uuid primaryKey/.test(err); - }); - - assert.deepEqual(MemServer.Server, {}); - assert.deepEqual(MemServer.DB, {}); - assert.deepEqual(Photo.findAll(), []); - assert.deepEqual(PhotoComment.findAll(), []); - }); - - it('should throw error if any of the id fixtures have an incorrect type', function() { - fs.mkdirSync('./memserver/fixtures'); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: '2', - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - assert.deepEqual(MemServer.Server, {}); - assert.deepEqual(MemServer.DB, {}); - - assert.deepEqual(Photo.findAll(), []); - assert.deepEqual(PhotoComment.findAll(), []); - - assert.throws(() => MemServer.start(), (err) => { - return (err instanceof Error) && - /\[MemServer\] Photo model primaryKey type is 'id'\. Instead you've tried to enter id: 2 with string type/.test(err); - }); - - assert.deepEqual(MemServer.Server, {}); - assert.deepEqual(MemServer.DB, {}); - assert.deepEqual(Photo.findAll(), []); - assert.deepEqual(PhotoComment.findAll(), []); - }); - - it('should throw error if any of the uuid fixtures have an incorrect type', function() { - this.timeout(5000); - - fs.mkdirSync('./memserver/fixtures'); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: 12, - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - assert.deepEqual(MemServer.Server, {}); - assert.deepEqual(MemServer.DB, {}); - - assert.deepEqual(Photo.findAll(), []); - assert.deepEqual(PhotoComment.findAll(), []); - - assert.throws(() => MemServer.start(), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment model primaryKey type is 'uuid'. Instead you've tried to enter uuid: 12 with number type/.test(err); - }); - - assert.deepEqual(MemServer.Server, {}); - assert.deepEqual(MemServer.DB, {}); - assert.deepEqual(Photo.findAll(), []); - assert.deepEqual(PhotoComment.findAll(), []); - }); - - it('should throw error if there are duplicate id fixtures', function() { - fs.mkdirSync('./memserver/fixtures'); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 2, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - const MemServer = require('../lib/index.js'); - - assert.throws(() => MemServer.start(), (err) => { - return (err instanceof Error) && - /\[MemServer\] DATABASE ERROR: Duplication in Photo fixtures with id: 2/.test(err); - }); - }); - - it('should throw error if there are duplicate uuid fixtures', function() { - fs.mkdirSync('./memserver/fixtures'); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - - const MemServer = require('../lib/index.js'); - - assert.throws(() => MemServer.start(), (err) => { - return (err instanceof Error) && - /\[MemServer\] DATABASE ERROR: Duplication in PhotoComment fixtures with uuid: 499ec646-493f-4eea-b92e-e383d94182f4/.test(err); - }); - }); -}); diff --git a/test/mem-server.js b/test/mem-server.js deleted file mode 100644 index 48a819f..0000000 --- a/test/mem-server.js +++ /dev/null @@ -1,96 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -describe('MemServer', function() { - describe('requiring MemServer', function() { - afterEach(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - done(); - }); - - it('should throw error if /memserver folder doesnt exist', function() { - this.timeout(5000); - - assert.throws(() => require('../lib/index.js'), (err) => { - return (err instanceof Error) && - /\/memserver folder doesn't exist for this directory!/.test(err); - }); - }); - - it('should throw error if /memserver/models folder doesnt exist', function() { - this.timeout(5000); - - fs.mkdirSync(`./memserver`); - - assert.throws(() => require('../lib/index.js'), (err) => { - return (err instanceof Error) && - /\/memserver\/models folder doesn't exist for this directory!/.test(err); - }); - }); - - it('should throw error if /memserver/server.js doesnt exist', function() { - this.timeout(5000); - - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - - assert.throws(() => require('../lib/index.js'), (err) => { - return (err instanceof Error) && - /\/memserver\/server.js doesn't exist for this directory!/.test(err); - }); - }); - - it('exports a MemServer with right functions and empty DB when there is no model', function() { - this.timeout(5000); - - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - - const MemServer = require('../lib/index.js'); - - assert.deepEqual(MemServer.DB, {}); - assert.deepEqual(MemServer.Server, {}); - assert.deepEqual(Object.keys(MemServer), ['DB', 'Server', 'Models', 'start', 'shutdown']); - assert.deepEqual(MemServer.Models, {}); - }); - - it('exports a MemServer with right functions and empty DB and models', function() { - this.timeout(5000); - - const modelFileContent = `import Model from '${process.cwd()}/lib/model'; - - export default Model({});`; - - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - const MemServer = require('../lib/index.js'); - const models = Object.keys(MemServer.Models); - - assert.deepEqual(MemServer.DB, {}); - assert.deepEqual(MemServer.Server, {}); - assert.deepEqual(Object.keys(MemServer), ['DB', 'Server', 'Models', 'start', 'shutdown']); - assert.deepEqual(models, ['PhotoComment', 'Photo', 'User']); - models.forEach((modelName) => { - const model = MemServer.Models[modelName]; - - assert.equal(model.modelName, modelName); - assert.deepEqual(Object.keys(MemServer.Models[modelName]), [ - 'modelName', 'primaryKey', 'defaultAttributes', 'attributes', 'count', 'find', 'findBy', - 'findAll', 'insert', 'update', 'delete', 'embed', 'embedReferences', 'serializer', - 'serialize', 'getRelationship' - ]); - }); - }); - }); -}); diff --git a/test/mem-server.model.delete.js b/test/mem-server.model.delete.js deleted file mode 100644 index b7a02b5..0000000 --- a/test/mem-server.model.delete.js +++ /dev/null @@ -1,176 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -describe('MemServer.Model Delete Interface', function() { - before(function() { - const modelFileContent = `import Model from '${process.cwd()}/lib/model'; - export default Model({});`; - - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - }); - - beforeEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - done(); - }); - - describe('$Model.delete() interface', function() { - it('can delete existing items', function() { - this.timeout(5000); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start() - - const deletedPhoto = Photo.delete({ id: 2 }); - const deletedComment = PhotoComment.delete({ uuid: '499ec646-493f-4eea-b92e-e383d94182f4' }); - - PhotoComment.delete({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29' }); - - assert.deepEqual(deletedPhoto, { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }); - assert.deepEqual(deletedComment, { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }); - assert.deepEqual(Photo.findAll(), [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ]); - assert.deepEqual(PhotoComment.findAll(), [ - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - } - ]); - }); - - it('throws when the $Model.delete(model) doesnt exist in the database', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - assert.throws(() => Photo.delete({ id: 1 }), (err) => { - return (err instanceof Error) && - /\[MemServer\] Photo has no records in the database to delete\. Photo\.delete\(\{ id: 1 \}\) failed/.test(err); - }); - assert.throws(() => PhotoComment.delete({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5111' }), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment has no records in the database to delete\. PhotoComment\.delete\(\{ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5111' \}\) failed/.test(err); - }); - - MemServer.start() - - Photo.delete({ id: 1 }); - assert.throws(() => Photo.delete({ id: 1 }), (err) => { - return (err instanceof Error) && - /\[MemServer\] Could not find Photo with id 1 to delete\. Photo\.delete\(\{ id: 1 \}\) failed/.test(err); - }); - assert.throws(() => PhotoComment.delete({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5111' }), (err) => { - return (err instanceof Error) && - /\[MemServer\] Could not find PhotoComment with uuid 374c7f4a-85d6-429a-bf2a-0719525f5111 to delete\. PhotoComment\.delete\(\{ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5111' \}\) failed/.test(err); - }); - }); - - it('throws when $Model.delete() called without a parameter', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start() - - assert.throws(() => Photo.delete(), (err) => { - return (err instanceof Error) && - /\[MemServer\] Photo\.delete\(model\) model object parameter required to delete a model/.test(err); - }); - assert.throws(() => PhotoComment.delete(), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment\.delete\(model\) model object parameter required to delete a model/.test(err); - }); - }); - - // NOTE: Photo.delete(primaryKey) feature ? - }); -}); diff --git a/test/mem-server.model.insert.js b/test/mem-server.model.insert.js deleted file mode 100644 index 208a09e..0000000 --- a/test/mem-server.model.insert.js +++ /dev/null @@ -1,399 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); -const moment = require('moment'); - -const modelFileContent = `import Model from '${process.cwd()}/lib/model'; - export default Model({});`; - -describe('MemServer.Model Insert Interface', function() { - before(function(done) { - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - done(); - }); - - beforeEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - is_public: true, - name() { - return 'Some default name'; - } - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - inserted_at() { - return '2017-10-25T20:54:04.447Z'; - }, - is_important: true - } - }); - `); - }); - - afterEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - done(); - }); - - describe('$Model.insert() factory interface', function() { - it('$Model.insert() will insert an empty model and auto-generate primaryKeys', function() { - this.timeout(5000); - - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, modelFileContent); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - assert.deepEqual(Photo.findAll().map((photo) => photo.id), [1, 2, 3]); - - Photo.insert(); - - assert.deepEqual(Photo.findAll().map((photo) => photo.id), [1, 2, 3, 4]) - - Photo.insert(); - - assert.equal(Photo.count(), 5); - assert.deepEqual(Photo.findAll(), [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - }, - { - id: 4, - href: undefined, - is_public: undefined, - name: undefined - }, - { - id: 5, - href: undefined, - is_public: undefined, - name: undefined - } - ]); - - const initialCommentUUIDs = PhotoComment.findAll().map((photoComment) => photoComment.uuid); - - assert.deepEqual(initialCommentUUIDs, [ - '499ec646-493f-4eea-b92e-e383d94182f4', '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - 'd351963d-e725-4092-a37c-1ca1823b57d3', '374c7f4a-85d6-429a-bf2a-0719525f5f29' - ]); - - PhotoComment.insert(); - - const allPhotoComments = PhotoComment.findAll(); - const lastPhotoComment = allPhotoComments[allPhotoComments.length - 1]; - - assert.equal(PhotoComment.count(), 5); - assert.ok(!initialCommentUUIDs[lastPhotoComment.uuid], 'inserted comment has a unique uuid'); - }); - - it('$Model.insert() will insert a model with defaultAttributes and auto-generated primaryKey', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - Photo.insert(); - Photo.insert(); - - assert.equal(Photo.count(), 5); - assert.deepEqual(Photo.findAll(), [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - }, - { - id: 4, - href: undefined, - is_public: true, - name: 'Some default name' - }, - { - id: 5, - href: undefined, - is_public: true, - name: 'Some default name' - } - ]); - - const initialCommentUUIDs = PhotoComment.findAll().map((comment) => comment.uuid); - - PhotoComment.insert(); - PhotoComment.insert(); - - assert.equal(PhotoComment.count(), 6); - - const allComments = PhotoComment.findAll(); - const lastInsertedComments = allComments.slice(4, allComments.length); - - lastInsertedComments.forEach((comment) => { - assert.ok(!initialCommentUUIDs.includes(comment.uuid), 'inserted comment uuid is unique'); - assert.equal(comment.is_important, true); - assert.equal(comment.inserted_at, '2017-10-25T20:54:04.447Z'); - }); - }); - - it('$Model.insert(options) will insert a model with overriden attributes', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - Photo.insert({ id: 99, href: '/izel.html', is_public: false }); - Photo.insert({ name: 'Baby photo', href: '/baby.jpg' }); - - assert.equal(Photo.count(), 5); - assert.deepEqual(Photo.findAll(), [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - }, - { - id: 99, - href: '/izel.html', - is_public: false, - name: 'Some default name' - }, - { - id: 4, - href: '/baby.jpg', - is_public: true, - name: 'Baby photo' - } - ]); - - const initialCommentUUIDs = PhotoComment.findAll().map((comment) => comment.uuid); - - const commentOne = PhotoComment.insert({ - inserted_at: '2015-10-25T20:54:04.447Z', photo_id: 1 - }); - const commentTwo = PhotoComment.insert({ - uuid: '6401f27c-49aa-4da7-9835-08f6f669e29f', is_important: false - }); - - assert.equal(PhotoComment.count(), 6); - - const allComments = PhotoComment.findAll(); - const lastInsertedComments = allComments.slice(4, allComments.length); - - assert.ok(allComments.includes(commentOne), 'first comment insert is in the database'); - assert.ok(allComments.includes(commentTwo), 'second comment insert is in the database'); - - assert.equal(commentOne.inserted_at, '2015-10-25T20:54:04.447Z'); - assert.equal(commentOne.photo_id, 1); - assert.equal(commentOne.is_important, true); - assert.equal(commentTwo.uuid, '6401f27c-49aa-4da7-9835-08f6f669e29f'); - assert.equal(commentTwo.inserted_at, '2017-10-25T20:54:04.447Z'); - assert.equal(commentTwo.photo_id, undefined); - assert.equal(commentTwo.is_important, false); - - lastInsertedComments.forEach((comment) => { - assert.ok(!initialCommentUUIDs.includes(comment.uuid), 'inserted comment uuid is unique'); - }); - }); - - it('$Model.insert(options) will throw if overriden primaryKey already exists', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - assert.throws(() => Photo.insert({ id: 1 }), (err) => { - return (err instanceof Error) && - /\[MemServer\] Photo id 1 already exists in the database! Photo.insert\(\{ id: 1 \}\) fails/.test(err); - }); - assert.throws(() => PhotoComment.insert({ uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3' }), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment uuid d351963d-e725-4092-a37c-1ca1823b57d3 already exists in the database! PhotoComment.insert\(\{ uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3' \}\) fails/.test(err); - }); - }); - - it('$Model.insert(options) will throw if overriden primaryKey is wrong type', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - assert.throws(() => Photo.insert({ id: '99' }), (err) => { - return (err instanceof Error) && - /\[MemServer\] Photo model primaryKey type is 'id'. Instead you've tried to enter id: 99 with string type/.test(err); - }); - assert.throws(() => PhotoComment.insert({ uuid: 1 }), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment model primaryKey type is 'uuid'. Instead you've tried to enter uuid: 1 with number type/.test(err); - }); - }); - }); - - it('can add new values to $Model.attributes when new attributes are discovered', function() { - this.timeout(5000); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - Photo.insert({ published_at: moment('2017-10-10').toJSON(), description: 'Some description' }); - Photo.insert({ location: 'Istanbul', is_public: false }); - PhotoComment.insert({ updated_at: moment('2017-01-10').toJSON(), like_count: 22 }); - PhotoComment.insert({ reply_id: 1 }); - - assert.deepEqual(Photo.attributes, [ - 'is_public', 'name', 'id', 'href', 'published_at', 'description', 'location' - ]); - assert.deepEqual(PhotoComment.attributes, [ - 'inserted_at', 'is_important', 'uuid', 'content', 'photo_id', 'user_id', 'updated_at', - 'like_count', 'reply_id' - ]); - assert.deepEqual(Photo.findAll(), [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - }, - { - id: 4, - href: undefined, - is_public: true, - published_at: '2017-10-09T22:00:00.000Z', - description: 'Some description', - name: 'Some default name' - }, - { - id: 5, - href: undefined, - is_public: false, - location: 'Istanbul', - published_at: undefined, - name: 'Some default name', - description: undefined - } - ]); - }); -}); diff --git a/test/mem-server.model.js b/test/mem-server.model.js deleted file mode 100644 index 5052a0d..0000000 --- a/test/mem-server.model.js +++ /dev/null @@ -1,246 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -describe('MemServer.Model Interface', function() { - before(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import faker from 'faker'; - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - is_public: true, - name() { - return faker.name.title(); - } - }, - publicPhotos() { - return this.findAll({ is_public: true }); - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, ` - import moment from 'moment'; - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - inserted_at() { - return moment().toJSON(); - }, - is_important: true - }, - forPhoto(photo) { - return this.findAll({ photo_id: photo.id }); - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({}); - `); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - }); - - beforeEach(function(done) { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - done(); - }); - - afterEach(function(done) { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - done(); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - done(); - }); - - describe('$Model interface', function() { - it('sets modelNames correctly', function() { - this.timeout(5000); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - assert.equal(Photo.modelName, 'Photo'); - assert.equal(PhotoComment.modelName, 'PhotoComment'); - - MemServer.start(); - - assert.equal(Photo.modelName, 'Photo'); - assert.equal(PhotoComment.modelName, 'PhotoComment'); - }); - - it('sets primaryKeys correctly', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment, User } = MemServer.Models; - - assert.equal(Photo.primaryKey, null); - assert.equal(PhotoComment.primaryKey, null); - assert.equal(User.primaryKey, null); - - MemServer.start(); - - assert.equal(Photo.primaryKey, 'id'); - assert.equal(PhotoComment.primaryKey, 'uuid'); - assert.equal(User.primaryKey, null); - }); - - it('reads the defaultAttributes correctly', function() { - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import faker from 'faker'; - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - is_public: true, - name() { - return faker.name.title(); - } - }, - publicPhotos() { - return this.findAll({ is_public: true }); - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, ` - import moment from 'moment'; - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - inserted_at() { - return moment().toJSON(); - }, - is_important: true - }, - forPhoto(photo) { - return this.findAll({ photo_id: photo.id }); - } - }); - `); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment, User } = MemServer.Models; - const initialPhotoDefaultAttributes = Photo.defaultAttributes; - const initialPhotoCommentDefaultAttributes = PhotoComment.defaultAttributes; - - assert.deepEqual(Object.keys(initialPhotoDefaultAttributes), ['is_public', 'name']); - assert.equal(initialPhotoDefaultAttributes.is_public, true); - assert.ok(initialPhotoDefaultAttributes.name.toString().includes('name.title();')); - - assert.deepEqual(Object.keys(initialPhotoCommentDefaultAttributes), ['inserted_at', 'is_important']); - assert.ok(initialPhotoCommentDefaultAttributes.inserted_at.toString().includes('.toJSON();')); - assert.equal(initialPhotoCommentDefaultAttributes.is_important, true); - assert.deepEqual(User.defaultAttributes, {}); - - MemServer.start(); - - assert.equal(Photo.defaultAttributes, initialPhotoDefaultAttributes); - assert.deepEqual(PhotoComment.defaultAttributes, initialPhotoCommentDefaultAttributes); - assert.deepEqual(User.defaultAttributes, {}); - }); - - it('sets attributes correctly', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment, User } = MemServer.Models; - - assert.deepEqual(Photo.attributes, ['is_public', 'name']); - assert.deepEqual(PhotoComment.attributes, ['inserted_at', 'is_important']); - assert.deepEqual(User.attributes, []); - - MemServer.start(); - - assert.deepEqual(Photo.attributes, ['is_public', 'name', 'id', 'href']); - assert.deepEqual(PhotoComment.attributes, ['inserted_at', 'is_important', 'uuid', 'content', 'photo_id', 'user_id']); - assert.deepEqual(User.attributes, []); - }); - - it('counts the models correctly with $Model.count()', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment, User } = MemServer.Models; - - assert.equal(Photo.count(), 0); - assert.equal(PhotoComment.count(), 0); - - MemServer.start(); - - assert.equal(Photo.count(), 3); - assert.equal(PhotoComment.count(), 4); - }); - - it('can have custom queries for a Model', function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - const photo = Photo.find(1); - - assert.deepEqual(PhotoComment.forPhoto(photo), PhotoComment.findAll({ photo_id: photo.id })); - assert.deepEqual(Photo.publicPhotos(), Photo.findAll({ is_public: true })); - }); - }); -}); diff --git a/test/mem-server.model.query.js b/test/mem-server.model.query.js deleted file mode 100644 index 479b96a..0000000 --- a/test/mem-server.model.query.js +++ /dev/null @@ -1,270 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -process.setMaxListeners(0); - -describe('MemServer.Model Query Interface', function() { - before(function() { - const modelFileContent = `import Model from '${process.cwd()}/lib/model'; - export default Model({});`; - - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - done(); - }); - - // NOTE: maybe needs to be inside the inner describe blocks - beforeEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - it('find() throws without a number id or ids', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start() - - const array = [null, undefined, '', '1', true, {}]; - array.forEach((param) => { - assert.throws(() => Photo.find(param), (err) => { - return (err instanceof Error) && - /\[MemServer\] Photo.find\(id\) cannot be called without a valid id/.test(err); - }); - assert.throws(() => PhotoComment.find(param), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment.find\(id\) cannot be called without a valid id/.test(err); - }); - }); - }); - - it('find(id) works for different models', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - assert.deepEqual(Photo.find(1), { - id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false - }); - assert.deepEqual(Photo.find(3), { - id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false - }); - }); - - it('find(ids) works for multiple ids', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - assert.deepEqual(Photo.find([1, 3]), [ - { id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false }, - { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false } - ]); - assert.deepEqual(Photo.find([2, 3]), [ - { id: 2, name: 'Family photo', href: 'family-photo.jpeg', is_public: true }, - { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false } - ]); - }); - - it('findBy() throws without params', function() { - const MemServer = require('../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - - assert.throws(() => Photo.findBy(), (err) => { - return (err instanceof Error) && - /\[MemServer\] Photo.findBy\(id\) cannot be called without a parameter/.test(err); - }); - }); - - it('findBy(options) returns a single model for the options', function() { - const firstPhoto = { id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false }; - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - assert.deepEqual(Photo.findBy({ is_public: false }), firstPhoto); - assert.deepEqual(Photo.findBy(firstPhoto), firstPhoto); - assert.deepEqual(Photo.findBy({ name: 'Family photo', href: 'family-photo.jpeg' }), { - id: 2, name: 'Family photo', href: 'family-photo.jpeg', is_public: true - }); - assert.deepEqual(PhotoComment.findBy({ uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3' }), { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }); - }); - - it('findAll() without parameters returns all the models in the database', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - assert.deepEqual(Photo.findAll(), [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ]); - assert.deepEqual(PhotoComment.findAll(), [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ]); - }); - - it('findAll(options) returns right models in the database', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - assert.deepEqual(Photo.findAll({ is_public: false }), [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ]); - assert.deepEqual(PhotoComment.findAll({ photo_id: 1, user_id: 1 }), [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - } - ]); - assert.deepEqual(PhotoComment.findAll({ user_id: 1 }), [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ]); - }); -}); diff --git a/test/mem-server.model.relationship-id.js b/test/mem-server.model.relationship-id.js deleted file mode 100644 index e0aef00..0000000 --- a/test/mem-server.model.relationship-id.js +++ /dev/null @@ -1,384 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -describe('MemServer.Model ID Relationships Interface', function() { - before(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import Model from '${process.cwd()}/lib/model'; - import PhotoComment from '${process.cwd()}/memserver/models/photo-comment.js'; - - export default Model({ - embedReferences: { - comments: PhotoComment - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, ` - import Model from '${process.cwd()}/lib/model'; - import User from '${process.cwd()}/memserver/models/user.js'; - - export default Model({ - embedReferences: { - author: User - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({}); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/email.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({}); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/activity.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/activities.js`, `export default [ - { - id: 1, - user_id: 1, - photo_id: 1 - }, - { - id: 2, - user_id: 1, - photo_id: null - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/users.js`, `export default [ - { - id: 1, - authentication_token: '1RQFPDXxNBvhGwZAEOj8ztGFItejDusXJw_F1FAg5-GknxhqrcfH9h4p9NGCiCVG', - password_digest: 'tL4rJzy3GrjSQ7K0ZMNqKsgMthsikbWfIEPTi/HJXD3lme7q6HT57RpuCKJOcAC9DFb3lXtEONmkB3fO0q3zWA==', - primary_email_id: 1 - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/emails.js`, `export default [ - { - id: 1, - address: 'contact@izelnakri.com', - is_public: false, - confirmed_at: '2018-02-25T23:00:00.000Z', - confirmation_token: '951d3321-9e66-4099-a4a5-cc1e4795d4ss', - confirmation_token_sent_at: '2018-02-25T22:16:01.133Z', - person_id: 1 - } - ];`); - }); - - beforeEach(function(done) { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - done(); - }); - - afterEach(function(done) { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - done(); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - done(); - }); - - describe('$Model.getRelationship method for id relationships', function() { - it('getRelationship() works for hasOne/belongsTo id relationships both sides', function() { - const MemServer = require('../lib/index.js'); - const { Photo, Activity } = MemServer.Models; - - MemServer.start(); - - const activity = Photo.getRelationship(Photo.find(1), 'activity'); - - assert.deepEqual(activity, { id: 1, user_id: 1, photo_id: 1 }); - assert.deepEqual(Photo.getRelationship(Photo.find(2), 'activity'), null); - assert.deepEqual(Activity.getRelationship(activity, 'photo'), Photo.find(1)); - assert.deepEqual(Activity.getRelationship(Activity.find(2), 'photo'), null); - }); - - it('getRelationship() works for hasMany/belongsTo id relationships both sides', function() { - const photoCommentCode = fs.readFileSync(`${process.cwd()}/memserver/models/photo-comment.js`); - const commentFixtures = fs.readFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`); - - fs.writeFileSync(`${process.cwd()}/memserver/models/comment.js`, photoCommentCode); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/comments.js`, commentFixtures); - - const MemServer = require('../lib/index.js'); - const { Photo, Comment } = MemServer.Models; - - MemServer.start(); - - const firstPhotoComments = Photo.getRelationship(Photo.find(1), 'comments'); - const secondPhotoComments = Photo.getRelationship(Photo.find(2), 'comments'); - const thirdPhotoComments = Photo.getRelationship(Photo.find(3), 'comments'); - - assert.deepEqual(firstPhotoComments, [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', content: 'What a nice photo!', photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', content: 'I agree', photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', content: 'I was kidding', photo_id: 1, - user_id: 1 - } - ]); - assert.deepEqual(secondPhotoComments, [ - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Interesting indeed', photo_id: 2, - user_id: 1 - } - ]); - assert.deepEqual(thirdPhotoComments, []); - assert.throws(() => Comment.getRelationship(firstPhotoComments, 'photo'), (err) => { - return (err instanceof Error) && - /\[MemServer\] Comment\.getRelationship expects model input to be an object not an array/.test(err); - }); - assert.deepEqual(Comment.getRelationship(firstPhotoComments[0], 'photo'), { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }); - assert.deepEqual(Comment.getRelationship(secondPhotoComments[0], 'photo'), { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }); - - fs.unlinkSync(`${process.cwd()}/memserver/models/comment.js`); - fs.unlinkSync(`${process.cwd()}/memserver/fixtures/comments.js`); - }); - - it('getRelationship() works for custom named hasOne/belongsTo id relationships both side', function() { - this.timeout(5000); - - const MemServer = require('../lib/index.js'); - const { Photo, Activity, User, Email } = MemServer.Models; - - MemServer.start(); - - const activity = Photo.getRelationship(Photo.find(1), 'userActivity', Activity); - - assert.deepEqual(activity, { id: 1, user_id: 1, photo_id: 1 }); - assert.deepEqual(User.getRelationship(User.find(1), 'primaryEmail', Email), { - id: 1, - address: 'contact@izelnakri.com', - is_public: false, - confirmed_at: '2018-02-25T23:00:00.000Z', - confirmation_token: '951d3321-9e66-4099-a4a5-cc1e4795d4ss', - confirmation_token_sent_at: '2018-02-25T22:16:01.133Z', - person_id: 1 - }); - assert.deepEqual(Photo.getRelationship(Photo.find(2), 'userActivity', Activity), null); - assert.deepEqual(Activity.getRelationship(activity, 'photo', Photo), Photo.find(1)); - assert.deepEqual(Activity.getRelationship(Activity.find(2), 'userPhoto', Photo), null); - assert.deepEqual(Activity.getRelationship(activity, 'photo', Photo), Photo.find(1)); - }); - - it('getRelationship() works for custom named hasMany/belongsTo id relationships both side', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - const firstPhotoComments = Photo.getRelationship(Photo.find(1), 'comments', PhotoComment); - const secondPhotoComments = Photo.getRelationship(Photo.find(2), 'comments', PhotoComment); - const thirdPhotoComments = Photo.getRelationship(Photo.find(3), 'comments', PhotoComment); - - assert.deepEqual(firstPhotoComments, [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', content: 'What a nice photo!', photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', content: 'I agree', photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', content: 'I was kidding', photo_id: 1, - user_id: 1 - } - ]); - assert.deepEqual(secondPhotoComments, [ - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Interesting indeed', photo_id: 2, - user_id: 1 - } - ]); - assert.deepEqual(thirdPhotoComments, []); - assert.throws(() => PhotoComment.getRelationship(firstPhotoComments, 'photo'), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment\.getRelationship expects model input to be an object not an array/.test(err); - }); - assert.deepEqual(PhotoComment.getRelationship(firstPhotoComments[0], 'photo'), { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }); - assert.deepEqual(PhotoComment.getRelationship(secondPhotoComments[0], 'photo'), { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }); - }); - - it('throws an error when id relationship reference is invalid', function() { - const MemServer = require('../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - - assert.throws(() => Photo.getRelationship(Photo.find(1), 'comments'), (err) => { - return (err instanceof Error) && - /\[MemServer\] comments relationship could not be found on Photo model\. Please put the comments Model object as the third parameter to Photo\.getRelationship function/.test(err); - }); - assert.throws(() => Photo.getRelationship(Photo.find(2), 'userActivity'), (err) => { - return (err instanceof Error) && - /\[MemServer\] userActivity relationship could not be found on Photo model\. Please put the userActivity Model object as the third parameter to Photo\.getRelationship function/.test(err); - }); - }); - }); - - describe('$Model relationship embedding', function() { - it('can register relationship embeds before runtime', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment, User } = MemServer.Models; - - MemServer.start(); - - assert.deepEqual(Photo.embedReferences, { comments: PhotoComment }); - assert.deepEqual(PhotoComment.embedReferences, { author: User }); - }); - - it('can register relationships embeds during runtime', function() { - const MemServer = require('../lib/index.js'); - const { Activity, Photo, PhotoComment, User } = MemServer.Models; - - MemServer.start(); - - Photo.embed({ userActivity: Activity }); - User.embed({ activities: Activity }); - - assert.deepEqual(Photo.embedReferences, { comments: PhotoComment, userActivity: Activity }); - assert.deepEqual(User.embedReferences, { activities: Activity }); - }); - - it('throws error when runtime $Model.embed() doesnt receive an object parameter', function() { - const MemServer = require('../lib/index.js'); - const { Activity, User } = MemServer.Models; - - MemServer.start(); - - assert.throws(() => User.embed(), (err) => { - return (err instanceof Error) && - /\[MemServer\] User\.embed\(relationshipObject\) requires an object as a parameter: { relationshipKey: \$RelationshipModel }/.test(err); - }); - assert.throws(() => User.embed(Activity), (err) => { - return (err instanceof Error) && - /\[MemServer\] User\.embed\(relationshipObject\) requires an object as a parameter: { relationshipKey: \$RelationshipModel }/.test(err); - }); - }); - - it('throws error when runtime $Model.embed(relationship) called with a Model that doesnt exist', function() { - const MemServer = require('../lib/index.js'); - const { Activity, User } = MemServer.Models; - - MemServer.start(); - - assert.throws(() => User.embed({ activities: undefined }), (err) => { - return (err instanceof Error) && - /\[MemServer\] User\.embed\(\) fails: activities Model reference is not a valid\. Please put a valid \$ModelName to User\.embed\(\)/.test(err); - }); - }); - }); -}); diff --git a/test/mem-server.model.relationship-uuid.js b/test/mem-server.model.relationship-uuid.js deleted file mode 100644 index 93f3d7e..0000000 --- a/test/mem-server.model.relationship-uuid.js +++ /dev/null @@ -1,341 +0,0 @@ -// TODO: change ids to uuid -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -describe('MemServer.Model UUID Relationships Interface', function() { - before(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import Model from '${process.cwd()}/lib/model'; - import PhotoComment from '${process.cwd()}/memserver/models/photo-comment.js'; - - export default Model({ - embedReferences: { - comments: PhotoComment - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, ` - import Model from '${process.cwd()}/lib/model'; - import User from '${process.cwd()}/memserver/models/user.js'; - - export default Model({ - embedReferences: { - author: User - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({}); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/email.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({}); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/activity.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - uuid: '6f0c74bb-13e0-4609-b34d-568cd3cee6bc', - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', - user_id: 1 - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/activities.js`, `export default [ - { - id: 1, - user_id: 1, - photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' - }, - { - id: 2, - user_id: 1, - photo_uuid: null - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/users.js`, `export default [ - { - id: 1, - authentication_token: '1RQFPDXxNBvhGwZAEOj8ztGFItejDusXJw_F1FAg5-GknxhqrcfH9h4p9NGCiCVG', - password_digest: 'tL4rJzy3GrjSQ7K0ZMNqKsgMthsikbWfIEPTi/HJXD3lme7q6HT57RpuCKJOcAC9DFb3lXtEONmkB3fO0q3zWA==', - primary_email_uuid: '951d3321-9e66-4099-a4a5-cc1e4795d4zz' - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/emails.js`, `export default [ - { - uuid: '951d3321-9e66-4099-a4a5-cc1e4795d4zz', - address: 'contact@izelnakri.com', - is_public: false, - confirmed_at: '2018-02-25T23:00:00.000Z', - confirmation_token: '951d3321-9e66-4099-a4a5-cc1e4795d4ss', - confirmation_token_sent_at: '2018-02-25T22:16:01.133Z', - person_id: 1 - } - ];`); - }); - - beforeEach(function(done) { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - done(); - }); - - afterEach(function(done) { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - done(); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - done(); - }); - - describe('$Model.getRelationship method for uuid relationships', function() { - it('getRelationship() works for hasOne/belongsTo uuid relationships both sides', function() { - const MemServer = require('../lib/index.js'); - const { Photo, Activity } = MemServer.Models; - - MemServer.start(); - - const activity = Photo.getRelationship(Photo.findBy({ - uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' - }), 'activity'); - - assert.deepEqual(activity, { - id: 1, user_id: 1, photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' - }); - assert.deepEqual(Photo.getRelationship(Photo.findBy({ - uuid: '2ae860da-ee55-4fd2-affb-da62e263980b' - }), 'activity'), null); - assert.deepEqual(Activity.getRelationship(activity, 'photo'), Photo.findBy({ - uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' - })); - assert.deepEqual(Activity.getRelationship(Activity.find(2), 'photo'), null); - - }); - - it('getRelationship() works for hasMany/belongsTo uuid relationships both sides', function() { - const photoCommentCode = fs.readFileSync(`${process.cwd()}/memserver/models/photo-comment.js`); - const commentFixtures = fs.readFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`); - - fs.writeFileSync(`${process.cwd()}/memserver/models/comment.js`, photoCommentCode); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/comments.js`, commentFixtures); - - const MemServer = require('../lib/index.js'); - const { Photo, Comment } = MemServer.Models; - - MemServer.start(); - - const firstPhotoComments = Photo.getRelationship(Photo.findBy({ - uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' - }), 'comments'); - const secondPhotoComments = Photo.getRelationship(Photo.findBy({ - uuid: '2ae860da-ee55-4fd2-affb-da62e263980b' - }), 'comments'); - const thirdPhotoComments = Photo.getRelationship(Photo.findBy({ - uuid: '6f0c74bb-13e0-4609-b34d-568cd3cee6bc' - }), 'comments'); - - assert.deepEqual(firstPhotoComments, [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', content: 'What a nice photo!', - photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', content: 'I agree', - photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', content: 'I was kidding', - photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 1 - } - ]); - assert.deepEqual(secondPhotoComments, [ - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Interesting indeed', - photo_uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', user_id: 1 - } - ]); - assert.deepEqual(thirdPhotoComments, []); - assert.throws(() => Comment.getRelationship(firstPhotoComments, 'photo'), (err) => { - return (err instanceof Error) && - /\[MemServer\] Comment\.getRelationship expects model input to be an object not an array/.test(err); - }); - assert.deepEqual(Comment.getRelationship(firstPhotoComments[0], 'photo'), { - uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }); - assert.deepEqual(Comment.getRelationship(secondPhotoComments[0], 'photo'), { - uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }); - - fs.unlinkSync(`${process.cwd()}/memserver/models/comment.js`); - fs.unlinkSync(`${process.cwd()}/memserver/fixtures/comments.js`); - }); - - it('getRelationship() works for custom named hasOne/belongsTo uuid relationships both side', function() { - const MemServer = require('../lib/index.js'); - const { Photo, Activity, User, Email } = MemServer.Models; - - MemServer.start(); - - const activity = Photo.getRelationship(Photo.findBy({ - uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' - }), 'userActivity', Activity); - - assert.deepEqual(activity, { - id: 1, user_id: 1, photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' - }); - assert.deepEqual(User.getRelationship(User.find(1), 'primaryEmail', Email), { - uuid: '951d3321-9e66-4099-a4a5-cc1e4795d4zz', - address: 'contact@izelnakri.com', - is_public: false, - confirmed_at: '2018-02-25T23:00:00.000Z', - confirmation_token: '951d3321-9e66-4099-a4a5-cc1e4795d4ss', - confirmation_token_sent_at: '2018-02-25T22:16:01.133Z', - person_id: 1 - }); - assert.deepEqual(Photo.getRelationship(Photo.findBy({ - uuid: '2ae860da-ee55-4fd2-affb-da62e263980b' - }), 'userActivity', Activity), null); - assert.deepEqual(Activity.getRelationship(activity, 'photo', Photo), Photo.findBy({ - uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' - })); - assert.deepEqual(Activity.getRelationship(Activity.find(2), 'photo', Photo), null); - }); - - it('getRelationship() works for custom named hasMany/belongsTo uuid relationships both side', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - const firstPhotoComments = Photo.getRelationship(Photo.findBy({ - uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' - }), 'comments', PhotoComment); - const secondPhotoComments = Photo.getRelationship(Photo.findBy({ - uuid: '2ae860da-ee55-4fd2-affb-da62e263980b' - }), 'comments', PhotoComment); - const thirdPhotoComments = Photo.getRelationship(Photo.findBy({ - uuid: '6f0c74bb-13e0-4609-b34d-568cd3cee6bc' - }), 'comments', PhotoComment); - - assert.deepEqual(firstPhotoComments, [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', content: 'What a nice photo!', - photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', content: 'I agree', - photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', content: 'I was kidding', - photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 1 - } - ]); - assert.deepEqual(secondPhotoComments, [ - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Interesting indeed', - photo_uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', user_id: 1 - } - ]); - assert.deepEqual(thirdPhotoComments, []); - assert.throws(() => PhotoComment.getRelationship(firstPhotoComments, 'photo'), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment\.getRelationship expects model input to be an object not an array/.test(err); - }); - assert.deepEqual(PhotoComment.getRelationship(firstPhotoComments[0], 'photo'), { - uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }); - assert.deepEqual(PhotoComment.getRelationship(secondPhotoComments[0], 'photo'), { - uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }); - }); - - it('throws an error when uuid relationship reference is invalid', function() { - const MemServer = require('../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - - assert.throws(() => Photo.getRelationship(Photo.findBy({ - uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' - }), 'comments'), (err) => { - return (err instanceof Error) && - /\[MemServer\] comments relationship could not be found on Photo model\. Please put the comments Model object as the third parameter to Photo\.getRelationship function/.test(err); - }); - assert.throws(() => Photo.getRelationship(Photo.findBy({ - uuid: '2ae860da-ee55-4fd2-affb-da62e263980b' - }), 'userActivity'), (err) => { - return (err instanceof Error) && - /\[MemServer\] userActivity relationship could not be found on Photo model\. Please put the userActivity Model object as the third parameter to Photo\.getRelationship function/.test(err); - }); - }); - }); -}); diff --git a/test/mem-server.model.serialize.js b/test/mem-server.model.serialize.js deleted file mode 100644 index fe07e5b..0000000 --- a/test/mem-server.model.serialize.js +++ /dev/null @@ -1,243 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -const modelFileContent = `import Model from '${process.cwd()}/lib/model'; - export default Model({});`; - -describe('MemServer.Model Serialize Interface', function() { - before(function(done) { - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - done(); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - done(); - }); - - beforeEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - it('can serialize a model', function() { - this.timeout(5000); - - const MemServer = require('../lib/index.js');; - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - const photo = Photo.find(1); - const photoComment = PhotoComment.findBy({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29' }); - - assert.deepEqual(Photo.serialize(photo), { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }); - assert.deepEqual(PhotoComment.serialize(photoComment), { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - }); - }); - - it('can serialize models', function() { - const MemServer = require('../lib/index.js');; - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - const photos = Photo.findAll({ is_public: false }); - const photoComments = PhotoComment.findAll({ photo_id: 1 }); - - assert.deepEqual(Photo.serializer(photos), [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ]); - assert.deepEqual(PhotoComment.serializer(photoComments), [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - } - ]); - }); - - it('can serialize empty record and record arrays', function() { - const MemServer = require('../lib/index.js');; - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start(); - - const notFoundPhoto = Photo.find(99); - const notFoundPhotos = Photo.findAll({ name: 'Wubba lubba dub' }); - const notFoundComment = PhotoComment.findBy({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5111' }); - const notFoundComments = Photo.findAll({ content: 'Aint easy' }); - - assert.equal(Photo.serializer(notFoundPhoto), undefined); - assert.deepEqual(Photo.serializer({}), { - id: null, href: null, is_public: null, name: null - }); - assert.deepEqual(Photo.serializer(notFoundPhotos), []); - assert.equal(PhotoComment.serializer(notFoundComment), undefined); - assert.deepEqual(PhotoComment.serializer({}), { - uuid: null, content: null, photo_id: null, user_id: null - }); - assert.deepEqual(PhotoComment.serializer(notFoundComments), []); - }); - - it('can serialize embeded records recursively', function() { - const MemServer = require('../lib/index.js');; - const { Photo, PhotoComment, User } = MemServer.Models; - - MemServer.start(); - - User.insert({ id: 1, first_name: 'Izel', last_name: 'Nakri' }); - User.insert({ id: 2, first_name: 'Benjamin', last_name: 'Graham' }); - Photo.embed({ comments: PhotoComment }); - PhotoComment.embed({ author: User }); - - const firstComment = PhotoComment.findBy({ uuid: '499ec646-493f-4eea-b92e-e383d94182f4' }); - const firstPhoto = Photo.find(1); - const targetSerializedUser = User.find(1); - - assert.deepEqual(targetSerializedUser, { id: 1, first_name: 'Izel', last_name: 'Nakri' }); - assert.deepEqual(User.serializer(targetSerializedUser), targetSerializedUser); - assert.deepEqual(PhotoComment.serializer(firstComment), Object.assign({}, firstComment, { - author: targetSerializedUser - })); - assert.deepEqual(Photo.serializer(firstPhoto), Object.assign({}, firstPhoto, { - comments: PhotoComment.findAll({ photo_id: 1 }).map((comment) => { - return Object.assign({}, comment, { author: User.find(comment.user_id) }); - }) - })); - - const targetUsers = User.findAll(); - const photoComments = PhotoComment.findAll(); - const targetPhotos = [Photo.find(1), Photo.find(2)]; - - assert.deepEqual(User.findAll(), [ - { id: 1, first_name: 'Izel', last_name: 'Nakri' }, - { id: 2, first_name: 'Benjamin', last_name: 'Graham' } - ]); - assert.deepEqual(User.serializer(targetUsers), targetUsers); - assert.deepEqual(PhotoComment.serializer(photoComments), photoComments.map((comment) => { - return Object.assign({}, comment, { author: User.find(comment.user_id) }); - })); - assert.deepEqual(Photo.serializer(targetPhotos), targetPhotos.map((photo) => { - return Object.assign({}, photo, { - comments: PhotoComment.findAll({ photo_id: photo.id }).map((comment) => { - return Object.assign({}, comment, { author: User.find(comment.user_id) }); - }) - }); - })) - }); - - it('allows for custom serializer declarations', function() { - const MemServer = require('../lib/index.js');; - const { User } = MemServer.Models; - - MemServer.start(); - - const user = User.insert({ id: 1, first_name: 'Izel', last_name: 'Nakri' }); - const secondUser = User.insert({ id: 2, first_name: 'Benjamin', last_name: 'Graham' }); - - User.authenticationSerializer = function(user) { - let serializedResponse = this.serializer(user); - - if (Array.isArray(serializedResponse)) { - serializedResponse.forEach((serializedModel) => delete serializedModel.last_name); - } else { - delete serializedResponse.last_name; - } - - return serializedResponse; - } - - assert.deepEqual(User.authenticationSerializer(user), { id: 1, first_name: 'Izel' }); - assert.deepEqual(User.authenticationSerializer([user, secondUser]), [ - { id: 1, first_name: 'Izel' }, { id: 2, first_name: 'Benjamin' } - ]) - }); -}); diff --git a/test/mem-server.model.update.js b/test/mem-server.model.update.js deleted file mode 100644 index 1c9978b..0000000 --- a/test/mem-server.model.update.js +++ /dev/null @@ -1,154 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -describe('MemServer.Model Update Interface', function() { - before(function() { - const modelFileContent = `import Model from '${process.cwd()}/lib/model'; - export default Model({});`; - - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - is_public: true, - name() { - return 'Some default name'; - } - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - inserted_at() { - return '2017-10-25T20:54:04.447Z'; - }, - is_important: true - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - }); - - beforeEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - done(); - }); - - describe('$Model.update() interface', function() { - it('can update models', function() { - this.timeout(5000); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start() - - Photo.update({ id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false }); - Photo.update({ id: 2, href: 'family-photo-2.jpeg', is_public: false }); - PhotoComment.update({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Cool' }); - - assert.deepEqual(Photo.find(1), { - id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false - }); - assert.deepEqual(Photo.find(2), { - id: 2, name: 'Family photo', href: 'family-photo-2.jpeg', is_public: false - }); - assert.deepEqual(PhotoComment.findBy({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29' }), { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Cool', photo_id: 2, user_id: 1 - }); - }); - - it('throws error when updating a nonexistent model', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start() - - assert.throws(() => Photo.update({ id: 99, href: 'family-photo-2.jpeg' }), (err) => { - return (err instanceof Error) && - /\[MemServer\] Photo\.update\(record\) failed because Photo with id: 99 does not exist/.test(err); - }); - assert.throws(() => PhotoComment.update({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5666', content: 'Nice' }), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment\.update\(record\) failed because PhotoComment with uuid: 374c7f4a-85d6-429a-bf2a-0719525f5666 does not exist/.test(err); - }); - }); - - it('throws error when a model get updated with an unknown $Model.attribute', function() { - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - MemServer.start() - - assert.throws(() => Photo.update({ id: 1, name: 'ME', is_verified: false }), (err) => { - return (err instanceof Error) && - /\[MemServer\] Photo\.update id: 1 fails, Photo model does not have is_verified attribute to update/.test(err); - }); - assert.throws(() => PhotoComment.update({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', location: 'Amsterdam' }), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment\.update uuid: 374c7f4a-85d6-429a-bf2a-0719525f5f29 fails, PhotoComment model does not have location attribute to update/.test(err); - }); - }); - }); -}); diff --git a/test/mem-server.response.js b/test/mem-server.response.js deleted file mode 100644 index dab82ca..0000000 --- a/test/mem-server.response.js +++ /dev/null @@ -1,136 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -const modelFileContent = `import Model from '${process.cwd()}/lib/model'; - export default Model({});`; - -describe('MemServer.Response Interface', function() { - before(function(done) { - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - export default function(Models) { - const { Photo } = Models; - - this.get('/photos', () => { - const photos = Photo.findAll(); - - return { photos: Photo.serializer(photos) }; - }); - }`); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/users.js`, `export default [ - { - id: 1, - first_name: 'Izel', - last_name: 'Nakri' - } - ]`); - - done(); - }); - - beforeEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - done(); - }); - - it('can be used outside the server file', function(done) { - this.timeout(5000); - - const MemServer = require('../lib/index.js'); - const Response = require('../lib/response.js').default; - - MemServer.start(); - window.$ = require('jquery'); - - const { Server } = MemServer; - const { Photo, User } = MemServer.Models; - - Server.get('/users/:id', (request) => { - const user = User.find(Number(request.params.id)); - - if (user) { - return Response(200, { user: User.serializer(user) }); - } - }); - - window.$.getJSON('/users/1', (data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { user: { id: 1, first_name: 'Izel', last_name: 'Nakri' } }); - - done(); - }); - }); - - it('can be used inside the server file', function(done) { - this.timeout(5000); - - const MemServer = require('../lib/index.js'); - const Response = require('../lib/response.js').default; - - MemServer.start(); - window.$ = require('jquery'); - - const { Server } = MemServer; - const { Photo, User } = MemServer.Models; - - window.$.getJSON('/photos', (data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photos: Photo.findAll() }); - - done(); - }); - }); - - it('can overwrite an existing server route', function(done) { - this.timeout(5000); - - const MemServer = require('../lib/index.js'); - const Response = require('../lib/response.js').default; - - MemServer.start(); - window.$ = require('jquery'); - - const { Server } = MemServer; - const { Photo, User } = MemServer.Models; - - Server.get('/photos', () => Response(500, { error: 'Internal Server Error'} )); - - window.$.getJSON('/photos').fail((jqXHR) => { - assert.equal(jqXHR.status, 500); - assert.deepEqual(jqXHR.responseJSON, { error: 'Internal Server Error' }); - - done(); - }); - }); -}); diff --git a/test/mem-server.start.js b/test/mem-server.start.js deleted file mode 100644 index 7305bdd..0000000 --- a/test/mem-server.start.js +++ /dev/null @@ -1,285 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); -const sinon = require('sinon'); - -describe('MemServer start/stop functionality', function() { - beforeEach(function() { - this.timeout(5000); - - const modelFileContent = `import Model from '${process.cwd()}/lib/model'; - export default Model({});`; - - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, modelFileContent); - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, 'export default function(Models) {}'); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - afterEach(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - done(); - }); - - it('can be started without fixtures', function() { - this.timeout(5000); - - const MemServer = require('../lib/index.js'); - - assert.deepEqual(MemServer.Server, {}); - - MemServer.start(); - - assert.deepEqual(Object.keys(MemServer.Server), [ - 'hosts', 'handlers', 'handledRequests', 'passthroughRequests', 'unhandledRequests', - 'requestReferences', 'forcePassthrough', 'disableUnhandled', '_nativeXMLHttpRequest', - 'running', 'handledRequest', 'passthroughRequest', 'unhandledRequest', 'passthrough' - ]); - }); - - it('can be started with fixtures', function() { - this.timeout(5000); - - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - assert.deepEqual(MemServer.Server, {}); - assert.deepEqual(MemServer.DB, {}); - - assert.deepEqual(Photo.findAll(), []); - assert.deepEqual(PhotoComment.findAll(), []); - - MemServer.start(); - - assert.deepEqual(Object.keys(MemServer.Server), [ - 'hosts', 'handlers', 'handledRequests', 'passthroughRequests', 'unhandledRequests', - 'requestReferences', 'forcePassthrough', 'disableUnhandled', '_nativeXMLHttpRequest', - 'running', 'handledRequest', 'passthroughRequest', 'unhandledRequest', 'passthrough' - ]); - assert.deepEqual(MemServer.DB, { - Photo: Photo.findAll(), PhotoComment: PhotoComment.findAll(), User: [] - }); - assert.equal(Photo.primaryKey, 'id'); - assert.deepEqual(Photo.findAll().length, 3); - assert.deepEqual(Photo.find(1), { - id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false - }); - - assert.equal(PhotoComment.primaryKey, 'uuid'); - assert.deepEqual(PhotoComment.findAll().length, 4); - assert.deepEqual(PhotoComment.findBy({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29' }), { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Interesting indeed', - photo_id: 2, user_id: 1 - }); - }); - - it('can be shut down', () => { - this.timeout(5000); - - const MemServer = require('../lib/index.js'); - - assert.deepEqual(MemServer.Server, {}); - - MemServer.start(); - - assert.deepEqual(Object.keys(MemServer.Server), [ - 'hosts', 'handlers', 'handledRequests', 'passthroughRequests', 'unhandledRequests', - 'requestReferences', 'forcePassthrough', 'disableUnhandled', '_nativeXMLHttpRequest', - 'running', 'handledRequest', 'passthroughRequest', 'unhandledRequest', 'passthrough' - ]); - - MemServer.Server.shutdown = sinon.spy(); - MemServer.shutdown(); - - assert.ok(MemServer.Server.shutdown.calledOnce, 'MemServer.shutdown() calls shutdown on Pretender instance'); - }); - - it('can be shut down and started again with restarted INITIAL STATE, NO MUTATION', () => { - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ];`); - - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - - const MemServer = require('../lib/index.js'); - const { Photo, PhotoComment } = MemServer.Models; - - assert.deepEqual(MemServer.Server, {}); - - MemServer.start(); - - assert.deepEqual(Object.keys(MemServer.Server), [ - 'hosts', 'handlers', 'handledRequests', 'passthroughRequests', 'unhandledRequests', - 'requestReferences', 'forcePassthrough', 'disableUnhandled', '_nativeXMLHttpRequest', - 'running', 'handledRequest', 'passthroughRequest', 'unhandledRequest', 'passthrough' - ]); - - const initialPhotos = Photo.findAll(); - const initialPhotoComments = PhotoComment.findAll(); - - assert.equal(initialPhotos.length, 3); - assert.equal(initialPhotoComments.length, 4); - - Photo.insert({ id: 55, name: 'Great another photo' }); - PhotoComment.insert({ uuid: '21f5ef47-332a-4d68-91b8-b7e13aa2f941', content: 'interesting photo indeed' }); - Photo.insert({ id: 56, name: 'one more' }); - Photo.insert({ id: 57, name: 'another' }); - - assert.equal(Photo.findAll().length, 6); - assert.equal(PhotoComment.findAll().length, 5); - - Photo.update({ id: 1, name: 'Mutated photo name' }); - Photo.update({ id: 2, name: 'Another mutated photo name' }); - PhotoComment.update({ - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', content: 'Mutated photo comment' - }); - - MemServer.shutdown(); - MemServer.start(); - - assert.deepEqual(Photo.findAll(), [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false - } - ]); - assert.deepEqual(PhotoComment.findAll(), [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ]); - }); -}); diff --git a/test/mem-server.utils.js b/test/mem-server.utils.js deleted file mode 100644 index f7b759d..0000000 --- a/test/mem-server.utils.js +++ /dev/null @@ -1,31 +0,0 @@ -require('babel-register')({ - presets: ['env'] -}); - -const assert = require('assert'); -const { generateUUID, primaryKeyTypeSafetyCheck } = require('../lib/utils.js'); - -describe('MemServer Utils Unit tests', function() { - it('exports generateUUID correctly', function() { - const UUIDs = Array.from({ length: 10 }).map(() => generateUUID()); - - assert.equal(UUIDs.length, 10); - UUIDs.forEach((currentUUID) => { - assert.equal(currentUUID.length, 36); - assert.equal(UUIDs.filter((uuid) => uuid === currentUUID).length, 1); - }) - }) - - it('exports primaryKeyTypeSafetyCheck correctly', function() { - assert.throws(() => primaryKeyTypeSafetyCheck('id', '22', 'Photo'), (err) => { - return (err instanceof Error) && - /\[MemServer\] Photo model primaryKey type is 'id'. Instead you've tried to enter id: 22 with string type/.test(err); - }); - assert.doesNotThrow(() => primaryKeyTypeSafetyCheck('id', 22, 'Photo'), Error); - assert.throws(() => primaryKeyTypeSafetyCheck('uuid', 22, 'PhotoComment'), (err) => { - return (err instanceof Error) && - /\[MemServer\] PhotoComment model primaryKey type is 'uuid'. Instead you've tried to enter uuid: 22 with number type/.test(err); - }); - assert.doesNotThrow(() => primaryKeyTypeSafetyCheck('uuid', '166a435d-ad3d-4662-9f6f-04373280a38b', 'PhotoComment'), Error); - }); -}); diff --git a/test/memserver-fixtures.js b/test/memserver-fixtures.js new file mode 100644 index 0000000..b194930 --- /dev/null +++ b/test/memserver-fixtures.js @@ -0,0 +1,311 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); +const MODEL_FILE_CONTENT = `import Model from '${process.cwd()}/lib/model'; + export default Model({});`; + +test.beforeEach(async (t) => { + await fs.mkdirp(`${CWD}/memserver`); + await fs.mkdirp(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/user.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}') + ]); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.afterEach.always(async () => { // NOTE: maybe remove require cache if needed + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('MemServer fixtures should throw error if any of the fixtures missing id or uuid', async (t) => { + t.plan(10); + + if (!await fs.exists(`${CWD}/memserver/fixtures`)) { + await fs.mkdir(`${CWD}/memserver/fixtures`); + } + + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); + + const MemServer = require('../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + t.deepEqual(MemServer.Server, {}); + t.deepEqual(MemServer.DB, {}); + t.deepEqual(Photo.findAll(), []); + t.deepEqual(PhotoComment.findAll(), []); + + const error = t.throws(() => MemServer.start(), Error); + + t.true(/\[MemServer\] DATABASE ERROR: At least one of your PhotoComment fixtures missing a primary key\. Please make sure all your PhotoComment fixtures have either id or uuid primaryKey/.test(error.message)) + + t.deepEqual(MemServer.Server, {}); + t.deepEqual(MemServer.DB, {}); + t.deepEqual(Photo.findAll(), []); + t.deepEqual(PhotoComment.findAll(), []); +}); + +test.serial('MemServer fixtures should throw error if any of the id fixtures have an incorrect type', async (t) => { + t.plan(10); + + await fs.mkdirp(`${CWD}/memserver/fixtures`); + await Promise.all([ + fs.writeFile(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: '2', + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); + + const MemServer = require('../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + t.deepEqual(MemServer.Server, {}); + t.deepEqual(MemServer.DB, {}); + + t.deepEqual(Photo.findAll(), []); + t.deepEqual(PhotoComment.findAll(), []); + + const error = t.throws(() => MemServer.start(), Error); + + t.true(/\[MemServer\] Photo model primaryKey type is 'id'\. Instead you've tried to enter id: 2 with string type/.test(error.message)); + t.deepEqual(MemServer.Server, {}); + t.deepEqual(MemServer.DB, {}); + t.deepEqual(Photo.findAll(), []); + t.deepEqual(PhotoComment.findAll(), []); +}); + +test('MemServer fixtures should throw error if any of the uuid fixtures have an incorrect type', async (t) => { + t.plan(10); + + await fs.mkdirp('./memserver/fixtures'); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: 12, + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); + + const MemServer = require('../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + t.deepEqual(MemServer.Server, {}); + t.deepEqual(MemServer.DB, {}); + t.deepEqual(Photo.findAll(), []); + t.deepEqual(PhotoComment.findAll(), []); + + const error = t.throws(() => MemServer.start(), Error); + + t.true(/\[MemServer\] PhotoComment model primaryKey type is 'uuid'. Instead you've tried to enter uuid: 12 with number type/.test(error.message)); + + t.deepEqual(MemServer.Server, {}); + t.deepEqual(MemServer.DB, {}); + t.deepEqual(Photo.findAll(), []); + t.deepEqual(PhotoComment.findAll(), []); +}); + +test.serial('MemServer fixtures should throw error if there are duplicate id fixtures', async (t) => { + t.plan(2); + + await fs.mkdirp(`${CWD}/memserver/fixtures`); + await fs.writeFile(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 2, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + const MemServer = require('../lib/index.js'); + + const error = t.throws(() => MemServer.start(), Error); + + t.true(/\[MemServer\] DATABASE ERROR: Duplication in Photo fixtures with id: 2/.test(error.message)); +}); + +test.serial('MemServer fixtures should throw error if there are duplicate uuid fixtures', async (t) => { + t.plan(2); + + await fs.mkdirp('./memserver/fixtures'); + await fs.writeFile(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`); + + const MemServer = require('../lib/index.js'); + + const error = t.throws(() => MemServer.start(), Error); + + t.true(/\[MemServer\] DATABASE ERROR: Duplication in PhotoComment fixtures with uuid: 499ec646-493f-4eea-b92e-e383d94182f4/.test(error.message)); +}); diff --git a/test/memserver-index.js b/test/memserver-index.js new file mode 100644 index 0000000..711a17c --- /dev/null +++ b/test/memserver-index.js @@ -0,0 +1,86 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); + +test.afterEach.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('MemServer require() should throw error if /memserver folder doesnt exist', (t) => { + const error = t.throws(() => require('../lib/index.js'), Error); + + t.true(/\/memserver folder doesn't exist for this directory!/.test(error.message)); +}); + +test.serial('MemServer require() should throw error if /memserver/models folder doesnt exist', async (t) => { + await fs.mkdir(`${CWD}/memserver`); + + const error = t.throws(() => require('../lib/index.js'), Error); + + t.true(/\/memserver\/models folder doesn't exist for this directory!/.test(error.message)); +}); + +test.serial('MemServer require() should throw error if /memserver/server.js doesnt exist', async (t) => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + + const error = t.throws(() => require('../lib/index.js'), Error); + + t.true(/\/memserver\/server.js doesn't exist for this directory!/.test(error.message)); +}); + +test.serial('MemServer require() exports a MemServer with right functions and empty DB when there is no model', async (t) => { + t.plan(4); + + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}'); + + const MemServer = require('../lib/index.js'); + + t.deepEqual(MemServer.DB, {}); + t.deepEqual(MemServer.Server, {}); + t.deepEqual(Object.keys(MemServer), ['DB', 'Server', 'Models', 'start', 'shutdown']); + t.deepEqual(MemServer.Models, {}); +}); + +test.serial('MemServer require() exports a MemServer with right functions and empty DB and models', async (t) => { + t.plan(10); + + const modelFileContent = `import Model from '${CWD}/lib/model'; + + export default Model({});`; + + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, modelFileContent), + fs.writeFile(`${CWD}/memserver/models/user.js`, modelFileContent), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, modelFileContent), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}') + ]); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + const MemServer = require('../lib/index.js'); + const models = Object.keys(MemServer.Models); + + t.deepEqual(MemServer.DB, {}); + t.deepEqual(MemServer.Server, {}); + t.deepEqual(Object.keys(MemServer), ['DB', 'Server', 'Models', 'start', 'shutdown']); + t.deepEqual(models, ['PhotoComment', 'Photo', 'User']); + + models.forEach((modelName) => { + const model = MemServer.Models[modelName]; + + t.is(model.modelName, modelName); + t.deepEqual(Object.keys(MemServer.Models[modelName]), [ + 'modelName', 'primaryKey', 'defaultAttributes', 'attributes', 'count', 'find', 'findBy', + 'findAll', 'insert', 'update', 'delete', 'embed', 'embedReferences', 'serializer', + 'serialize', 'getRelationship' + ]); + }); +}); diff --git a/test/memserver-response.js b/test/memserver-response.js new file mode 100644 index 0000000..e013b65 --- /dev/null +++ b/test/memserver-response.js @@ -0,0 +1,134 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); +const MODEL_FILE_CONTENT = `import Model from '${process.cwd()}/lib/model'; + export default Model({});`; + +test.before(async () => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/user.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/server.js`, ` + const Response = require('../lib/response').default; + + export default function(Models) { + const { Photo } = Models; + + this.get('/photos', () => { + const photos = Photo.findAll(); + + return Response(202, { photos: Photo.serializer(photos) }); + }); + }`), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/users.js`, `export default [ + { + id: 1, + first_name: 'Izel', + last_name: 'Nakri' + } + ]`) + ]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.serial.cb('MemServer.Response can be used outside the server file', (t) => { + t.plan(2); + + const MemServer = require('../lib/index.js'); + const Response = require('../lib/response.js').default; + + MemServer.start(); + window.$ = require('jquery'); + + const { Server } = MemServer; + const { User } = MemServer.Models; + + Server.get('/users/:id', (request) => { + const user = User.find(Number(request.params.id)); + + if (user) { + return Response(200, { user: User.serializer(user) }); + } + }); + + window.$.getJSON('/users/1', (data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { user: { id: 1, first_name: 'Izel', last_name: 'Nakri' } }); + + t.end(); + }); +}); + +test.serial.cb('MemServer.Response can be used inside the server file', (t) => { + t.plan(2); + + const MemServer = require('../lib/index.js'); + + MemServer.start(); + window.$ = require('jquery'); + + const { Photo } = MemServer.Models; + + window.$.getJSON('/photos', (data, textStatus, jqXHR) => { + t.is(jqXHR.status, 202); + t.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); + + t.end(); + }); +}); + +test.serial.cb('MemServer.Response can be used when overwriting an existing server route', (t) => { + t.plan(2); + + const MemServer = require('../lib/index.js'); + const Response = require('../lib/response.js').default; + + MemServer.start(); + window.$ = require('jquery'); + + const { Server } = MemServer; + + Server.get('/photos', () => Response(500, { error: 'Internal Server Error'} )); + + window.$.getJSON('/photos').fail((jqXHR) => { + t.is(jqXHR.status, 500); + t.deepEqual(jqXHR.responseJSON, { error: 'Internal Server Error' }); + + t.end(); + }); +}); diff --git a/test/memserver-start.js b/test/memserver-start.js new file mode 100644 index 0000000..d26825f --- /dev/null +++ b/test/memserver-start.js @@ -0,0 +1,284 @@ +import test from 'ava'; +import fs from 'fs-extra'; +import sinon from 'sinon'; + +const CWD = process.cwd(); +const MODEL_FILE_CONTENT = `import Model from '${CWD}/lib/model'; + export default Model({});`; + +test.beforeEach(async () => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/user.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}') + ]); +}); + +test.afterEach.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('MemServer can be started without fixtures', (t) => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + const MemServer = require('../lib/index.js'); + + t.deepEqual(MemServer.Server, {}); + + MemServer.start(); + + t.deepEqual(Object.keys(MemServer.Server), [ + 'hosts', 'handlers', 'handledRequests', 'passthroughRequests', 'unhandledRequests', + 'requestReferences', 'forcePassthrough', 'disableUnhandled', '_nativeXMLHttpRequest', + 'running', 'ctx', 'handledRequest', 'passthroughRequest', 'unhandledRequest', 'passthrough' + ]); +}); + +test.serial('MemServer can be started with fixtures', async (t) => { + t.plan(12); + + await fs.mkdir(`${CWD}/memserver/fixtures`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); + + const MemServer = require('../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + t.deepEqual(MemServer.Server, {}); + t.deepEqual(MemServer.DB, {}); + + t.deepEqual(Photo.findAll(), []); + t.deepEqual(PhotoComment.findAll(), []); + + MemServer.start(); + + t.deepEqual(Object.keys(MemServer.Server), [ + 'hosts', 'handlers', 'handledRequests', 'passthroughRequests', 'unhandledRequests', + 'requestReferences', 'forcePassthrough', 'disableUnhandled', '_nativeXMLHttpRequest', + 'running', 'ctx', 'handledRequest', 'passthroughRequest', 'unhandledRequest', 'passthrough' + ]); + t.deepEqual(MemServer.DB, { + Photo: Photo.findAll(), PhotoComment: PhotoComment.findAll(), User: [] + }); + t.is(Photo.primaryKey, 'id'); + t.deepEqual(Photo.findAll().length, 3); + t.deepEqual(Photo.find(1), { + id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false + }); + + t.is(PhotoComment.primaryKey, 'uuid'); + t.deepEqual(PhotoComment.findAll().length, 4); + t.deepEqual(PhotoComment.findBy({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29' }), { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Interesting indeed', + photo_id: 2, user_id: 1 + }); +}); + +test('MemServer can be shut down', (t) => { + const MemServer = require('../lib/index.js'); + + t.deepEqual(MemServer.Server, {}); + + MemServer.start(); + + t.deepEqual(Object.keys(MemServer.Server), [ + 'hosts', 'handlers', 'handledRequests', 'passthroughRequests', 'unhandledRequests', + 'requestReferences', 'forcePassthrough', 'disableUnhandled', '_nativeXMLHttpRequest', + 'running', 'ctx', 'handledRequest', 'passthroughRequest', 'unhandledRequest', 'passthrough' + ]); + + MemServer.Server.shutdown = sinon.spy(); + MemServer.shutdown(); + + t.true(MemServer.Server.shutdown.calledOnce, 'MemServer.shutdown() calls shutdown on Pretender instance'); +}); + +test.serial('MemServer can be shut down and started again with restarted INITIAL STATE, NO MUTATION', async (t) => { + t.plan(8); + + await fs.mkdir(`${CWD}/memserver/fixtures`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); + + const MemServer = require('../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + t.deepEqual(MemServer.Server, {}); + + MemServer.start(); + + t.deepEqual(Object.keys(MemServer.Server), [ + 'hosts', 'handlers', 'handledRequests', 'passthroughRequests', 'unhandledRequests', + 'requestReferences', 'forcePassthrough', 'disableUnhandled', '_nativeXMLHttpRequest', + 'running', 'ctx', 'handledRequest', 'passthroughRequest', 'unhandledRequest', 'passthrough' + ]); + + const initialPhotos = Photo.findAll(); + const initialPhotoComments = PhotoComment.findAll(); + + t.is(initialPhotos.length, 3); + t.is(initialPhotoComments.length, 4); + + Photo.insert({ id: 55, name: 'Great another photo' }); + PhotoComment.insert({ uuid: '21f5ef47-332a-4d68-91b8-b7e13aa2f941', content: 'interesting photo indeed' }); + Photo.insert({ id: 56, name: 'one more' }); + Photo.insert({ id: 57, name: 'another' }); + + t.is(Photo.findAll().length, 6); + t.is(PhotoComment.findAll().length, 5); + + Photo.update({ id: 1, name: 'Mutated photo name' }); + Photo.update({ id: 2, name: 'Another mutated photo name' }); + PhotoComment.update({ + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', content: 'Mutated photo comment' + }); + + MemServer.shutdown(); + MemServer.start(); + + t.deepEqual(Photo.findAll(), [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ]); + t.deepEqual(PhotoComment.findAll(), [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ]); +}); diff --git a/test/memserver-utils.js b/test/memserver-utils.js new file mode 100644 index 0000000..7c4beb9 --- /dev/null +++ b/test/memserver-utils.js @@ -0,0 +1,29 @@ +import test from 'ava'; +import { generateUUID, primaryKeyTypeSafetyCheck } from '../lib/utils'; + +test('memserver/lib/utils exports generateUUID correctly', (t) => { + t.plan(21); + + const UUIDs = Array.from({ length: 10 }).map(() => generateUUID()); + + t.is(UUIDs.length, 10); + + UUIDs.forEach((currentUUID) => { + t.is(currentUUID.length, 36); + t.is(UUIDs.filter((uuid) => uuid === currentUUID).length, 1); + }); +}); + +test('memserver/lib/utils exports primaryKeyTypeSafetyCheck correctly', (t) => { + t.plan(6); + + const error = t.throws(() => primaryKeyTypeSafetyCheck('id', '22', 'Photo'), Error); + + t.true(/\[MemServer\] Photo model primaryKey type is 'id'. Instead you've tried to enter id: 22 with string type/.test(error.message)); + t.notThrows(() => primaryKeyTypeSafetyCheck('id', 22, 'Photo')); + + const secondError = t.throws(() => primaryKeyTypeSafetyCheck('uuid', 22, 'PhotoComment'), Error); + + t.true(/\[MemServer\] PhotoComment model primaryKey type is 'uuid'. Instead you've tried to enter uuid: 22 with number type/.test(secondError.message)); + t.notThrows(() => primaryKeyTypeSafetyCheck('uuid', '166a435d-ad3d-4662-9f6f-04373280a38b', 'PhotoComment')); +}); diff --git a/test/model/delete.js b/test/model/delete.js new file mode 100644 index 0000000..8f426f5 --- /dev/null +++ b/test/model/delete.js @@ -0,0 +1,180 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); + +test.before(async () => { + const MODEL_FILE_CONTENT = `import Model from '${CWD}/lib/model'; + export default Model({});`; + + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/user.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/photo.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}'), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('$Model.delete() can delete existing items', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const deletedPhoto = Photo.delete({ id: 2 }); + const deletedComment = PhotoComment.delete({ uuid: '499ec646-493f-4eea-b92e-e383d94182f4' }); + + PhotoComment.delete({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29' }); + + t.deepEqual(deletedPhoto, { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }); + t.deepEqual(deletedComment, { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }); + t.deepEqual(Photo.findAll(), [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ]); + t.deepEqual(PhotoComment.findAll(), [ + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + } + ]); +}); + +test.serial('$Model.delete(model) throws when the model primaryKey doesnt exist in the database', (t) => { + t.plan(8); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + const error = t.throws(() => Photo.delete({ id: 1 }), Error); + + t.true(/\[MemServer\] Photo has no records in the database to delete\. Photo\.delete\(\{ id: 1 \}\) failed/.test(error.message)); + + const secondError = t.throws(() => { + return PhotoComment.delete({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5111' }); + }, Error); + + t.true(/\[MemServer\] PhotoComment has no records in the database to delete\. PhotoComment\.delete\(\{ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5111' \}\) failed/.test(secondError.message)); + + MemServer.start(); + + Photo.delete({ id: 1 }); + + const thirdError = t.throws(() => Photo.delete({ id: 1 }), Error); + + t.true(/\[MemServer\] Could not find Photo with id 1 to delete\. Photo\.delete\(\{ id: 1 \}\) failed/.test(thirdError.message)); + + const fourthError = t.throws(() => PhotoComment.delete({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5111' }), Error); + + t.true(/\[MemServer\] Could not find PhotoComment with uuid 374c7f4a-85d6-429a-bf2a-0719525f5111 to delete\. PhotoComment\.delete\(\{ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5111' \}\) failed/.test(fourthError.message)); +}); + +test('$Model.delete() throws when called without a parameter', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const error = t.throws(() => Photo.delete(), Error); + + t.true(/\[MemServer\] Photo\.delete\(model\) model object parameter required to delete a model/.test(error.message)); + + const secondError = t.throws(() => PhotoComment.delete(), Error); + + t.true(/\[MemServer\] PhotoComment\.delete\(model\) model object parameter required to delete a model/.test(secondError.message)); +}); + +// NOTE: $Model.delete(primaryKey) feature ? diff --git a/test/model/index.js b/test/model/index.js new file mode 100644 index 0000000..b260617 --- /dev/null +++ b/test/model/index.js @@ -0,0 +1,255 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); + +test.before(async () => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import faker from 'faker'; + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + is_public: true, + name() { + return faker.name.title(); + } + }, + publicPhotos() { + return this.findAll({ is_public: true }); + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, ` + import moment from 'moment'; + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + inserted_at() { + return moment().toJSON(); + }, + is_important: true + }, + forPhoto(photo) { + return this.findAll({ photo_id: photo.id }); + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/user.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({}); + `), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}'), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.afterEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('$Model.modelName gets set correctly', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + t.is(Photo.modelName, 'Photo'); + t.is(PhotoComment.modelName, 'PhotoComment'); + + MemServer.start(); + + t.is(Photo.modelName, 'Photo'); + t.is(PhotoComment.modelName, 'PhotoComment'); +}); + +test.serial('$Model.primaryKey gets set correctly', (t) => { + t.plan(6); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment, User } = MemServer.Models; + + t.is(Photo.primaryKey, null); + t.is(PhotoComment.primaryKey, null); + t.is(User.primaryKey, null); + + MemServer.start(); + + t.is(Photo.primaryKey, 'id'); + t.is(PhotoComment.primaryKey, 'uuid'); + t.is(User.primaryKey, null); +}); + +test.serial('$Model.defaultAttributes gets set correctly', async (t) => { + t.plan(10); + + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import faker from 'faker'; + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + is_public: true, + name() { + return faker.name.title(); + } + }, + publicPhotos() { + return this.findAll({ is_public: true }); + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, ` + import moment from 'moment'; + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + inserted_at() { + return moment().toJSON(); + }, + is_important: true + }, + forPhoto(photo) { + return this.findAll({ photo_id: photo.id }); + } + }); + `) + ]); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment, User } = MemServer.Models; + const initialPhotoDefaultAttributes = Photo.defaultAttributes; + const initialPhotoCommentDefaultAttributes = PhotoComment.defaultAttributes; + + t.deepEqual(Object.keys(initialPhotoDefaultAttributes), ['is_public', 'name']); + t.is(initialPhotoDefaultAttributes.is_public, true); + t.true(initialPhotoDefaultAttributes.name.toString().includes('name.title();')); + + t.deepEqual(Object.keys(initialPhotoCommentDefaultAttributes), ['inserted_at', 'is_important']); + t.true(initialPhotoCommentDefaultAttributes.inserted_at.toString().includes('.toJSON();')); + t.is(initialPhotoCommentDefaultAttributes.is_important, true); + t.deepEqual(User.defaultAttributes, {}); + + MemServer.start(); + + t.is(Photo.defaultAttributes, initialPhotoDefaultAttributes); + t.deepEqual(PhotoComment.defaultAttributes, initialPhotoCommentDefaultAttributes); + t.deepEqual(User.defaultAttributes, {}); +}); + +test.serial('$Model.attributes gets set correctly', (t) => { + t.plan(6); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment, User } = MemServer.Models; + + t.deepEqual(Photo.attributes, ['is_public', 'name']); + t.deepEqual(PhotoComment.attributes, ['inserted_at', 'is_important']); + t.deepEqual(User.attributes, []); + + MemServer.start(); + + t.deepEqual(Photo.attributes, ['is_public', 'name', 'id', 'href']); + t.deepEqual(PhotoComment.attributes, ['inserted_at', 'is_important', 'uuid', 'content', 'photo_id', 'user_id']); + t.deepEqual(User.attributes, []); +}); + +test.serial('$Model.count counts the models correctly', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + t.is(Photo.count(), 0); + t.is(PhotoComment.count(), 0); + + MemServer.start(); + + t.is(Photo.count(), 3); + t.is(PhotoComment.count(), 4); +}); + +test.serial('$Model can have custom methods/queries for the model', (t) => { + t.plan(2); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const photo = Photo.find(1); + + t.deepEqual(PhotoComment.forPhoto(photo), PhotoComment.findAll({ photo_id: photo.id })); + t.deepEqual(Photo.publicPhotos(), Photo.findAll({ is_public: true })); +}); diff --git a/test/model/insert.js b/test/model/insert.js new file mode 100644 index 0000000..c9b7c17 --- /dev/null +++ b/test/model/insert.js @@ -0,0 +1,339 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); +const MODEL_FILE_CONTENT = `import Model from '${process.cwd()}/lib/model'; + export default Model({});`; + +test.before(async () => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/user.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}'), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); +}); + +test.beforeEach(async () => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + is_public: true, + name() { + return 'Some default name'; + } + } + })`), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + inserted_at() { + return '2017-10-25T20:54:04.447Z'; + }, + is_important: true + } + });`) + ]); +}); + +test.afterEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('$Model.insert() will insert an empty model and auto-generate primaryKeys', async (t) => { + t.plan(7); + + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, MODEL_FILE_CONTENT) + ]); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + t.deepEqual(Photo.findAll().map((photo) => photo.id), [1, 2, 3]); + + Photo.insert(); + + t.deepEqual(Photo.findAll().map((photo) => photo.id), [1, 2, 3, 4]); + + Photo.insert(); + + t.is(Photo.count(), 5); + t.deepEqual(Photo.findAll(), [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + }, + { + id: 4, + href: undefined, + is_public: undefined, + name: undefined + }, + { + id: 5, + href: undefined, + is_public: undefined, + name: undefined + } + ]); + + const initialCommentUUIDs = PhotoComment.findAll().map((photoComment) => photoComment.uuid); + + t.deepEqual(initialCommentUUIDs, [ + '499ec646-493f-4eea-b92e-e383d94182f4', '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + 'd351963d-e725-4092-a37c-1ca1823b57d3', '374c7f4a-85d6-429a-bf2a-0719525f5f29' + ]); + + PhotoComment.insert(); + + const allPhotoComments = PhotoComment.findAll(); + const lastPhotoComment = allPhotoComments[allPhotoComments.length - 1]; + + t.is(PhotoComment.count(), 5); + t.true(!initialCommentUUIDs[lastPhotoComment.uuid], 'inserted comment has a unique uuid'); +}); + +test.serial('$Model.insert(attributes) will insert a model with overriden attributes', (t) => { + t.plan(14); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + Photo.insert({ id: 99, href: '/izel.html', is_public: false }); + Photo.insert({ name: 'Baby photo', href: '/baby.jpg' }); + + t.is(Photo.count(), 5); + t.deepEqual(Photo.findAll(), [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + }, + { + id: 99, + href: '/izel.html', + is_public: false, + name: 'Some default name' + }, + { + id: 4, + href: '/baby.jpg', + is_public: true, + name: 'Baby photo' + } + ]); + + const initialCommentUUIDs = PhotoComment.findAll().map((comment) => comment.uuid); + const commentOne = PhotoComment.insert({ + inserted_at: '2015-10-25T20:54:04.447Z', photo_id: 1 + }); + const commentTwo = PhotoComment.insert({ + uuid: '6401f27c-49aa-4da7-9835-08f6f669e29f', is_important: false + }); + + t.is(PhotoComment.count(), 6); + + const allComments = PhotoComment.findAll(); + const lastInsertedComments = allComments.slice(4, allComments.length); + + t.true(allComments.includes(commentOne), 'first comment insert is in the database'); + t.true(allComments.includes(commentTwo), 'second comment insert is in the database'); + + t.is(commentOne.inserted_at, '2015-10-25T20:54:04.447Z'); + t.is(commentOne.photo_id, 1); + t.is(commentOne.is_important, true); + t.is(commentTwo.uuid, '6401f27c-49aa-4da7-9835-08f6f669e29f'); + t.is(commentTwo.inserted_at, '2017-10-25T20:54:04.447Z'); + t.is(commentTwo.photo_id, undefined); + t.is(commentTwo.is_important, false); + + lastInsertedComments.forEach((comment) => { + t.true(!initialCommentUUIDs.includes(comment.uuid), 'inserted comment uuid is unique'); + }); +}); + +test('$Model.insert(attributes) will throw if overriden primaryKey already exists', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const error = t.throws(() => Photo.insert({ id: 1 }), Error); + + t.true(/\[MemServer\] Photo id 1 already exists in the database! Photo.insert\(\{ id: 1 \}\) fails/.test(error.message)); + + const secondError = t.throws(() => PhotoComment.insert({ uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3' }), Error); + + t.true(/\[MemServer\] PhotoComment uuid d351963d-e725-4092-a37c-1ca1823b57d3 already exists in the database! PhotoComment.insert\(\{ uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3' \}\) fails/.test(secondError.message)); +}); + +test('$Model.insert(attributes) will throw if overriden primaryKey is wrong type', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const error = t.throws(() => Photo.insert({ id: '99' }), Error); + + t.true(/\[MemServer\] Photo model primaryKey type is 'id'. Instead you've tried to enter id: 99 with string type/.test(error.message)); + + const secondError = t.throws(() => PhotoComment.insert({ uuid: 1 }), Error); + + t.true(/\[MemServer\] PhotoComment model primaryKey type is 'uuid'. Instead you've tried to enter uuid: 1 with number type/.test(secondError.message)); +}); + +test('$Model.insert(attributes) can add new values to $Model.attributes when new attributes are discovered', (t) => { + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + Photo.insert({ published_at: new Date('2017-10-10').toJSON(), description: 'Some description' }); + Photo.insert({ location: 'Istanbul', is_public: false }); + PhotoComment.insert({ updated_at: new Date('2017-01-10').toJSON(), like_count: 22 }); + PhotoComment.insert({ reply_id: 1 }); + + t.deepEqual(Photo.attributes, [ + 'is_public', 'name', 'id', 'href', 'published_at', 'description', 'location' + ]); + t.deepEqual(PhotoComment.attributes, [ + 'inserted_at', 'is_important', 'uuid', 'content', 'photo_id', 'user_id', 'updated_at', + 'like_count', 'reply_id' + ]); + t.deepEqual(Photo.findAll(), [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + }, + { + id: 4, + href: undefined, + is_public: true, + published_at: '2017-10-10T00:00:00.000Z', + description: 'Some description', + name: 'Some default name' + }, + { + id: 5, + href: undefined, + is_public: false, + location: 'Istanbul', + published_at: undefined, + name: 'Some default name', + description: undefined + } + ]); +}); diff --git a/test/model/query.js b/test/model/query.js new file mode 100644 index 0000000..cbe0437 --- /dev/null +++ b/test/model/query.js @@ -0,0 +1,280 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); +const MODEL_FILE_CONTENT = `import Model from '${CWD}/lib/model'; + export default Model({});`; + +test.before(async () => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/user.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}'), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.serial('$Model.find() throws without a number id or ids', (t) => { + t.plan(24); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const array = [null, undefined, '', '1', true, {}]; + + array.forEach((param) => { + const error = t.throws(() => Photo.find(param), Error); + + t.true(/\[MemServer\] Photo.find\(id\) cannot be called without a valid id/.test(error.message)); + + const secondError = t.throws(() => PhotoComment.find(param), Error); + + t.true(/\[MemServer\] PhotoComment.find\(id\) cannot be called without a valid id/.test(secondError.message)); + }); +}); + +test.serial('$Model.find(id) works for different models', (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + + t.deepEqual(Photo.find(1), { + id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false + }); + t.deepEqual(Photo.find(3), { + id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false + }); +}); + +test.serial('$Model.find(ids) works for multiple ids', (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + + t.deepEqual(Photo.find([1, 3]), [ + { id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false }, + { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false } + ]); + t.deepEqual(Photo.find([2, 3]), [ + { id: 2, name: 'Family photo', href: 'family-photo.jpeg', is_public: true }, + { id: 3, name: 'Selfie', href: 'selfie.jpeg', is_public: false } + ]); +}); + +test.serial('$Model.findBy() throws without params', (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + + const error = t.throws(() => Photo.findBy(), Error); + + t.true(/\[MemServer\] Photo.findBy\(id\) cannot be called without a parameter/.test(error.message)); +}); + +test.serial('$Model.findBy(attributes) returns a single model for the options', (t) => { + t.plan(4); + + const firstPhoto = { id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false }; + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + t.deepEqual(Photo.findBy({ is_public: false }), firstPhoto); + t.deepEqual(Photo.findBy(firstPhoto), firstPhoto); + t.deepEqual(Photo.findBy({ name: 'Family photo', href: 'family-photo.jpeg' }), { + id: 2, name: 'Family photo', href: 'family-photo.jpeg', is_public: true + }); + t.deepEqual(PhotoComment.findBy({ uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3' }), { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }); +}); + +test.serial('$Model.findAll() without parameters returns all the models in the database', (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + t.deepEqual(Photo.findAll(), [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ]); + t.deepEqual(PhotoComment.findAll(), [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ]); +}); + +test('$Model.findAll(attributes) returns right models in the database', (t) => { + t.plan(3); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + t.deepEqual(Photo.findAll({ is_public: false }), [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ]); + t.deepEqual(PhotoComment.findAll({ photo_id: 1, user_id: 1 }), [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + } + ]); + t.deepEqual(PhotoComment.findAll({ user_id: 1 }), [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ]); +}); diff --git a/test/model/relationship-id.js b/test/model/relationship-id.js new file mode 100644 index 0000000..d2d710a --- /dev/null +++ b/test/model/relationship-id.js @@ -0,0 +1,395 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); + +test.before(async () => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import Model from '${CWD}/lib/model'; + import PhotoComment from '${CWD}/memserver/models/photo-comment.js'; + + export default Model({ + embedReferences: { + comments: PhotoComment + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, ` + import Model from '${CWD}/lib/model'; + import User from '${CWD}/memserver/models/user.js'; + + export default Model({ + embedReferences: { + author: User + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/user.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({}); + `), + fs.writeFile(`${CWD}/memserver/models/email.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({}); + `), + fs.writeFile(`${CWD}/memserver/models/activity.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + }); + `), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}'), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/activities.js`, `export default [ + { + id: 1, + user_id: 1, + photo_id: 1 + }, + { + id: 2, + user_id: 1, + photo_id: null + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/users.js`, `export default [ + { + id: 1, + authentication_token: '1RQFPDXxNBvhGwZAEOj8ztGFItejDusXJw_F1FAg5-GknxhqrcfH9h4p9NGCiCVG', + password_digest: 'tL4rJzy3GrjSQ7K0ZMNqKsgMthsikbWfIEPTi/HJXD3lme7q6HT57RpuCKJOcAC9DFb3lXtEONmkB3fO0q3zWA==', + primary_email_id: 1 + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/emails.js`, `export default [ + { + id: 1, + address: 'contact@izelnakri.com', + is_public: false, + confirmed_at: '2018-02-25T23:00:00.000Z', + confirmation_token: '951d3321-9e66-4099-a4a5-cc1e4795d4ss', + confirmation_token_sent_at: '2018-02-25T22:16:01.133Z', + person_id: 1 + } + ];`) + ]); +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.afterEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('$Model.getRelationship() works for hasOne/belongsTo id relationships both sides on id relationships', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo, Activity } = MemServer.Models; + + MemServer.start(); + + const activity = Photo.getRelationship(Photo.find(1), 'activity'); + + t.deepEqual(activity, { id: 1, user_id: 1, photo_id: 1 }); + t.is(Photo.getRelationship(Photo.find(2), 'activity'), undefined); + t.deepEqual(Activity.getRelationship(activity, 'photo'), Photo.find(1)); + t.is(Activity.getRelationship(Activity.find(2), 'photo'), undefined); +}); + +test.serial('$Model.getRelationship() works for hasMany/belongsTo id relationships both sides on id relationships', async (t) => { + t.plan(7); + + const photoCommentCode = await fs.readFile(`${CWD}/memserver/models/photo-comment.js`); + const commentFixtures = await fs.readFile(`${CWD}/memserver/fixtures/photo-comments.js`); + + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/comment.js`, photoCommentCode), + fs.writeFile(`${CWD}/memserver/fixtures/comments.js`, commentFixtures) + ]); + + const MemServer = require('../../lib/index.js'); + const { Photo, Comment } = MemServer.Models; + + MemServer.start(); + + const firstPhotoComments = Photo.getRelationship(Photo.find(1), 'comments'); + const secondPhotoComments = Photo.getRelationship(Photo.find(2), 'comments'); + const thirdPhotoComments = Photo.getRelationship(Photo.find(3), 'comments'); + + t.deepEqual(firstPhotoComments, [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', content: 'What a nice photo!', photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', content: 'I agree', photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', content: 'I was kidding', photo_id: 1, + user_id: 1 + } + ]); + t.deepEqual(secondPhotoComments, [ + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Interesting indeed', photo_id: 2, + user_id: 1 + } + ]); + t.deepEqual(thirdPhotoComments, []); + + const error = t.throws(() => Comment.getRelationship(firstPhotoComments, 'photo'), Error); + + t.true(/\[MemServer\] Comment\.getRelationship expects model input to be an object not an array/.test(error.message)); + t.deepEqual(Comment.getRelationship(firstPhotoComments[0], 'photo'), { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }); + t.deepEqual(Comment.getRelationship(secondPhotoComments[0], 'photo'), { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }); + + await Promise.all([ + fs.remove(`${CWD}/memserver/models/comment.js`), + fs.remove(`${CWD}/memserver/fixtures/comments.js`) + ]); +}); + +test.serial('$Model.getRelationship() works for custom named hasOne/belongsTo id relationships both side on id relationships', (t) => { + t.plan(6); + + const MemServer = require('../../lib/index.js'); + const { Photo, Activity, User, Email } = MemServer.Models; + + MemServer.start(); + + const activity = Photo.getRelationship(Photo.find(1), 'userActivity', Activity); + + t.deepEqual(activity, { id: 1, user_id: 1, photo_id: 1 }); + t.deepEqual(User.getRelationship(User.find(1), 'primaryEmail', Email), { + id: 1, + address: 'contact@izelnakri.com', + is_public: false, + confirmed_at: '2018-02-25T23:00:00.000Z', + confirmation_token: '951d3321-9e66-4099-a4a5-cc1e4795d4ss', + confirmation_token_sent_at: '2018-02-25T22:16:01.133Z', + person_id: 1 + }); + t.is(Photo.getRelationship(Photo.find(2), 'userActivity', Activity), undefined); + t.deepEqual(Activity.getRelationship(activity, 'photo', Photo), Photo.find(1)); + t.is(Activity.getRelationship(Activity.find(2), 'userPhoto', Photo), undefined); + t.deepEqual(Activity.getRelationship(activity, 'photo', Photo), Photo.find(1)); +}); + +test.serial('$Model.getRelationship() works for custom named hasMany/belongsTo id relationships both side on id relationships', (t) => { + t.plan(7); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const firstPhotoComments = Photo.getRelationship(Photo.find(1), 'comments', PhotoComment); + const secondPhotoComments = Photo.getRelationship(Photo.find(2), 'comments', PhotoComment); + const thirdPhotoComments = Photo.getRelationship(Photo.find(3), 'comments', PhotoComment); + + t.deepEqual(firstPhotoComments, [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', content: 'What a nice photo!', photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', content: 'I agree', photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', content: 'I was kidding', photo_id: 1, + user_id: 1 + } + ]); + t.deepEqual(secondPhotoComments, [ + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Interesting indeed', photo_id: 2, + user_id: 1 + } + ]); + t.deepEqual(thirdPhotoComments, []); + + const error = t.throws(() => PhotoComment.getRelationship(firstPhotoComments, 'photo'), Error); + + t.true(/\[MemServer\] PhotoComment\.getRelationship expects model input to be an object not an array/.test(error.message)); + t.deepEqual(PhotoComment.getRelationship(firstPhotoComments[0], 'photo'), { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }); + t.deepEqual(PhotoComment.getRelationship(secondPhotoComments[0], 'photo'), { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }); +}); + +test.serial('$Model.getRelationship() throws an error when id relationship reference is invalid', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + + const error = t.throws(() => Photo.getRelationship(Photo.find(1), 'comments'), Error); + + t.true(/\[MemServer\] comments relationship could not be found on Photo model\. Please put the comments Model object as the third parameter to Photo\.getRelationship function/.test(error.message)); + + const secondError = t.throws(() => Photo.getRelationship(Photo.find(2), 'userActivity'), Error); + + t.true(/\[MemServer\] userActivity relationship could not be found on Photo model\. Please put the userActivity Model object as the third parameter to Photo\.getRelationship function/.test(secondError.message)); +}); + +test.serial('$Model.embedReferences can be set before runtime', (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment, User } = MemServer.Models; + + MemServer.start(); + + t.deepEqual(Photo.embedReferences, { comments: PhotoComment }); + t.deepEqual(PhotoComment.embedReferences, { author: User }); +}); + +test.serial('$Model.embed({ embedName: ModelName }) sets an embedReference during runtime', (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + const { Activity, Photo, PhotoComment, User } = MemServer.Models; + + MemServer.start(); + + Photo.embed({ userActivity: Activity }); + User.embed({ activities: Activity }); + + t.deepEqual(Photo.embedReferences, { comments: PhotoComment, userActivity: Activity }); + t.deepEqual(User.embedReferences, { activities: Activity }); +}); + +test.serial('$Model.embed() throws error at runtime doesnt receive an object as parameter', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Activity, User } = MemServer.Models; + + MemServer.start(); + + const error = t.throws(() => User.embed(), Error); + + t.true(/\[MemServer\] User\.embed\(relationshipObject\) requires an object as a parameter: { relationshipKey: \$RelationshipModel }/.test(error.message)); + + const secondError = t.throws(() => User.embed(Activity), Error); + + t.true(/\[MemServer\] User\.embed\(relationshipObject\) requires an object as a parameter: { relationshipKey: \$RelationshipModel }/.test(secondError.message)); +}); + +test.serial('$Model.embed() throws error when runtime $Model.embed(relationship) called with a Model that doesnt exist', (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + const { User } = MemServer.Models; + + MemServer.start(); + + const error = t.throws(() => User.embed({ activities: undefined }), Error); + + t.true(/\[MemServer\] User\.embed\(\) fails: activities Model reference is not a valid\. Please put a valid \$ModelName to User\.embed\(\)/.test(error.message)); +}); diff --git a/test/model/relationship-uuid.js b/test/model/relationship-uuid.js new file mode 100644 index 0000000..7660f8b --- /dev/null +++ b/test/model/relationship-uuid.js @@ -0,0 +1,349 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); + +test.before(async () => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import Model from '${CWD}/lib/model'; + import PhotoComment from '${CWD}/memserver/models/photo-comment.js'; + + export default Model({ + embedReferences: { + comments: PhotoComment + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, ` + import Model from '${CWD}/lib/model'; + import User from '${CWD}/memserver/models/user.js'; + + export default Model({ + embedReferences: { + author: User + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/user.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({}); + `), + fs.writeFile(`${CWD}/memserver/models/email.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({}); + `), + fs.writeFile(`${CWD}/memserver/models/activity.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + }); + `), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}'), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + uuid: '6f0c74bb-13e0-4609-b34d-568cd3cee6bc', + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', + user_id: 1 + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/activities.js`, `export default [ + { + id: 1, + user_id: 1, + photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' + }, + { + id: 2, + user_id: 1, + photo_uuid: null + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/users.js`, `export default [ + { + id: 1, + authentication_token: '1RQFPDXxNBvhGwZAEOj8ztGFItejDusXJw_F1FAg5-GknxhqrcfH9h4p9NGCiCVG', + password_digest: 'tL4rJzy3GrjSQ7K0ZMNqKsgMthsikbWfIEPTi/HJXD3lme7q6HT57RpuCKJOcAC9DFb3lXtEONmkB3fO0q3zWA==', + primary_email_uuid: '951d3321-9e66-4099-a4a5-cc1e4795d4zz' + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/emails.js`, `export default [ + { + uuid: '951d3321-9e66-4099-a4a5-cc1e4795d4zz', + address: 'contact@izelnakri.com', + is_public: false, + confirmed_at: '2018-02-25T23:00:00.000Z', + confirmation_token: '951d3321-9e66-4099-a4a5-cc1e4795d4ss', + confirmation_token_sent_at: '2018-02-25T22:16:01.133Z', + person_id: 1 + } + ];`) + ]); +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.afterEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('$Model.getRelationship() works for hasOne/belongsTo uuid relationships both sides on uuid relationship', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo, Activity } = MemServer.Models; + + MemServer.start(); + + const activity = Photo.getRelationship(Photo.findBy({ + uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' + }), 'activity'); + + t.deepEqual(activity, { + id: 1, user_id: 1, photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' + }); + t.is(Photo.getRelationship(Photo.findBy({ + uuid: '2ae860da-ee55-4fd2-affb-da62e263980b' + }), 'activity'), undefined); + t.deepEqual(Activity.getRelationship(activity, 'photo'), Photo.findBy({ + uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' + })); + t.is(Activity.getRelationship(Activity.find(2), 'photo'), undefined); +}); + +test.serial('$Model.getRelationship() works for hasMany/belongsTo uuid relationship both sides on uuid', async (t) => { + t.plan(7); + + const photoCommentCode = await fs.readFile(`${CWD}/memserver/models/photo-comment.js`); + const commentFixtures = await fs.readFile(`${CWD}/memserver/fixtures/photo-comments.js`); + + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/comment.js`, photoCommentCode), + fs.writeFile(`${CWD}/memserver/fixtures/comments.js`, commentFixtures) + ]); + + const MemServer = require('../../lib/index.js'); + const { Photo, Comment } = MemServer.Models; + + MemServer.start(); + + const firstPhotoComments = Photo.getRelationship(Photo.findBy({ + uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' + }), 'comments'); + const secondPhotoComments = Photo.getRelationship(Photo.findBy({ + uuid: '2ae860da-ee55-4fd2-affb-da62e263980b' + }), 'comments'); + const thirdPhotoComments = Photo.getRelationship(Photo.findBy({ + uuid: '6f0c74bb-13e0-4609-b34d-568cd3cee6bc' + }), 'comments'); + + t.deepEqual(firstPhotoComments, [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', content: 'What a nice photo!', + photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', content: 'I agree', + photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', content: 'I was kidding', + photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 1 + } + ]); + t.deepEqual(secondPhotoComments, [ + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Interesting indeed', + photo_uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', user_id: 1 + } + ]); + t.deepEqual(thirdPhotoComments, []); + + const error = t.throws(() => Comment.getRelationship(firstPhotoComments, 'photo'), Error); + + t.true(/\[MemServer\] Comment\.getRelationship expects model input to be an object not an array/.test(error.message)); + t.deepEqual(Comment.getRelationship(firstPhotoComments[0], 'photo'), { + uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }); + t.deepEqual(Comment.getRelationship(secondPhotoComments[0], 'photo'), { + uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }); + + await Promise.all([ + fs.remove(`${CWD}/memserver/models/comment.js`), + fs.remove(`${CWD}/memserver/fixtures/comments.js`) + ]); +}); + +test.serial('$Model.getRelationship() works for custom named hasOne/belongsTo uuid relationships both side on uuid relationship', (t) => { + t.plan(5); + + const MemServer = require('../../lib/index.js'); + const { Photo, Activity, User, Email } = MemServer.Models; + + MemServer.start(); + + const activity = Photo.getRelationship(Photo.findBy({ + uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' + }), 'userActivity', Activity); + + t.deepEqual(activity, { + id: 1, user_id: 1, photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' + }); + t.deepEqual(User.getRelationship(User.find(1), 'primaryEmail', Email), { + uuid: '951d3321-9e66-4099-a4a5-cc1e4795d4zz', + address: 'contact@izelnakri.com', + is_public: false, + confirmed_at: '2018-02-25T23:00:00.000Z', + confirmation_token: '951d3321-9e66-4099-a4a5-cc1e4795d4ss', + confirmation_token_sent_at: '2018-02-25T22:16:01.133Z', + person_id: 1 + }); + t.is(Photo.getRelationship(Photo.findBy({ + uuid: '2ae860da-ee55-4fd2-affb-da62e263980b' + }), 'userActivity', Activity), undefined); + t.deepEqual(Activity.getRelationship(activity, 'photo', Photo), Photo.findBy({ + uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' + })); + t.is(Activity.getRelationship(Activity.find(2), 'photo', Photo), undefined); +}); + +test.serial('$Model.getRelationship() works for custom named hasMany/belongsTo uuid relationships both side on uuid relationship', (t) => { + t.plan(7); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const firstPhotoComments = Photo.getRelationship(Photo.findBy({ + uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' + }), 'comments', PhotoComment); + const secondPhotoComments = Photo.getRelationship(Photo.findBy({ + uuid: '2ae860da-ee55-4fd2-affb-da62e263980b' + }), 'comments', PhotoComment); + const thirdPhotoComments = Photo.getRelationship(Photo.findBy({ + uuid: '6f0c74bb-13e0-4609-b34d-568cd3cee6bc' + }), 'comments', PhotoComment); + + t.deepEqual(firstPhotoComments, [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', content: 'What a nice photo!', + photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', content: 'I agree', + photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', content: 'I was kidding', + photo_uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', user_id: 1 + } + ]); + t.deepEqual(secondPhotoComments, [ + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Interesting indeed', + photo_uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', user_id: 1 + } + ]); + t.deepEqual(thirdPhotoComments, []); + + const error = t.throws(() => PhotoComment.getRelationship(firstPhotoComments, 'photo'), Error); + + t.true(/\[MemServer\] PhotoComment\.getRelationship expects model input to be an object not an array/.test(error.message)); + t.deepEqual(PhotoComment.getRelationship(firstPhotoComments[0], 'photo'), { + uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a', + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }); + t.deepEqual(PhotoComment.getRelationship(secondPhotoComments[0], 'photo'), { + uuid: '2ae860da-ee55-4fd2-affb-da62e263980b', + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }); +}); + +test.serial('$Model.getRelationship() throws an error when uuid relationship reference is invalid', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + + const error = t.throws(() => Photo.getRelationship(Photo.findBy({ + uuid: '65075a0c-3f4c-47af-9995-d4a01747ff7a' + }), 'comments'), Error); + + t.true(/\[MemServer\] comments relationship could not be found on Photo model\. Please put the comments Model object as the third parameter to Photo\.getRelationship function/.test(error.message)); + + const secondError = t.throws(() => Photo.getRelationship(Photo.findBy({ + uuid: '2ae860da-ee55-4fd2-affb-da62e263980b' + }), 'userActivity'), Error); + + t.true(/\[MemServer\] userActivity relationship could not be found on Photo model\. Please put the userActivity Model object as the third parameter to Photo\.getRelationship function/.test(secondError.message)); +}); diff --git a/test/model/serialize.js b/test/model/serialize.js new file mode 100644 index 0000000..c7e0744 --- /dev/null +++ b/test/model/serialize.js @@ -0,0 +1,249 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); +const MODEL_FILE_CONTENT = `import Model from '${CWD}/lib/model'; + export default Model({});`; + +test.before(async () => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/user.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}'), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('$Model.serialize(model) serializes a model', (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const photo = Photo.find(1); + const photoComment = PhotoComment.findBy({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29' }); + + t.deepEqual(Photo.serialize(photo), { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }); + t.deepEqual(PhotoComment.serialize(photoComment), { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + }); +}); + +test.serial('$Model.serialize(models) can serialize models', (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const photos = Photo.findAll({ is_public: false }); + const photoComments = PhotoComment.findAll({ photo_id: 1 }); + + t.deepEqual(Photo.serializer(photos), [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ]); + t.deepEqual(PhotoComment.serializer(photoComments), [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + } + ]); +}); + +test.serial('$Model.serialize() can serialize empty record and record arrays', (t) => { + t.plan(6); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const notFoundPhoto = Photo.find(99); + const notFoundPhotos = Photo.findAll({ name: 'Wubba lubba dub' }); + const notFoundComment = PhotoComment.findBy({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5111' }); + const notFoundComments = Photo.findAll({ content: 'Aint easy' }); + + t.is(Photo.serializer(notFoundPhoto), undefined); + t.deepEqual(Photo.serializer({}), { + id: null, href: null, is_public: null, name: null + }); + t.deepEqual(Photo.serializer(notFoundPhotos), []); + t.is(PhotoComment.serializer(notFoundComment), undefined); + t.deepEqual(PhotoComment.serializer({}), { + uuid: null, content: null, photo_id: null, user_id: null + }); + t.deepEqual(PhotoComment.serializer(notFoundComments), []); +}); + +test.serial('$Model.serialize(model) can serialize embeded records recursively', (t) => { + t.plan(8); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment, User } = MemServer.Models; + + MemServer.start(); + + User.insert({ id: 1, first_name: 'Izel', last_name: 'Nakri' }); + User.insert({ id: 2, first_name: 'Benjamin', last_name: 'Graham' }); + Photo.embed({ comments: PhotoComment }); // this works + PhotoComment.embed({ author: User }); // this doesnt work!?!? + + const firstComment = PhotoComment.findBy({ uuid: '499ec646-493f-4eea-b92e-e383d94182f4' }); + const firstPhoto = Photo.find(1); + const targetSerializedUser = User.find(1); + + t.deepEqual(targetSerializedUser, { id: 1, first_name: 'Izel', last_name: 'Nakri' }); + t.deepEqual(User.serializer(targetSerializedUser), targetSerializedUser); + t.deepEqual(PhotoComment.serializer(firstComment), Object.assign({}, firstComment, { + author: targetSerializedUser + })); + t.deepEqual(Photo.serializer(firstPhoto), Object.assign({}, firstPhoto, { + comments: PhotoComment.findAll({ photo_id: 1 }).map((comment) => { + return Object.assign({}, comment, { author: User.find(comment.user_id) }); + }) + })); + + const targetUsers = User.findAll(); + const photoComments = PhotoComment.findAll(); + const targetPhotos = [Photo.find(1), Photo.find(2)]; + + t.deepEqual(User.findAll(), [ + { id: 1, first_name: 'Izel', last_name: 'Nakri' }, + { id: 2, first_name: 'Benjamin', last_name: 'Graham' } + ]); + t.deepEqual(User.serializer(targetUsers), targetUsers); + t.deepEqual(PhotoComment.serializer(photoComments), photoComments.map((comment) => { + return Object.assign({}, comment, { author: User.find(comment.user_id) }); + })); + t.deepEqual(Photo.serializer(targetPhotos), targetPhotos.map((photo) => { + return Object.assign({}, photo, { + comments: PhotoComment.findAll({ photo_id: photo.id }).map((comment) => { + return Object.assign({}, comment, { author: User.find(comment.user_id) }); + }) + }); + })); +}); + +test.serial('$Model allows for custom serializer declarations', (t) => { + const MemServer = require('../../lib/index.js'); + const { User } = MemServer.Models; + + MemServer.start(); + + const user = User.insert({ id: 1, first_name: 'Izel', last_name: 'Nakri' }); + const secondUser = User.insert({ id: 2, first_name: 'Benjamin', last_name: 'Graham' }); + + User.authenticationSerializer = function(user) { + let serializedResponse = this.serializer(user); + + if (Array.isArray(serializedResponse)) { + serializedResponse.forEach((serializedModel) => delete serializedModel.last_name); + } else { + delete serializedResponse.last_name; + } + + return serializedResponse; + }; + + t.deepEqual(User.authenticationSerializer(user), { id: 1, first_name: 'Izel' }); + t.deepEqual(User.authenticationSerializer([user, secondUser]), [ + { id: 1, first_name: 'Izel' }, { id: 2, first_name: 'Benjamin' } + ]); +}); diff --git a/test/model/update.js b/test/model/update.js new file mode 100644 index 0000000..e6410f7 --- /dev/null +++ b/test/model/update.js @@ -0,0 +1,155 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); +const MODEL_FILE_CONTENT = `import Model from '${CWD}/lib/model'; + export default Model({});`; + +test.before(async () => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/user.js`, MODEL_FILE_CONTENT), + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + is_public: true, + name() { + return 'Some default name'; + } + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + inserted_at() { + return '2017-10-25T20:54:04.447Z'; + }, + is_important: true + } + }); + `), + fs.writeFile(`${CWD}/memserver/server.js`, 'export default function(Models) {}'), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('$Model.update(attributes) can update models', (t) => { + t.plan(3); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + Photo.update({ id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false }); + Photo.update({ id: 2, href: 'family-photo-2.jpeg', is_public: false }); + PhotoComment.update({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Cool' }); + + t.deepEqual(Photo.find(1), { + id: 1, name: 'Ski trip', href: 'ski-trip.jpeg', is_public: false + }); + t.deepEqual(Photo.find(2), { + id: 2, name: 'Family photo', href: 'family-photo-2.jpeg', is_public: false + }); + t.deepEqual(PhotoComment.findBy({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29' }), { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', content: 'Cool', photo_id: 2, user_id: 1 + }); +}); + +test.serial('$Model.update(attributes) throws an exception when updating a nonexistent model', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const error = t.throws(() => Photo.update({ id: 99, href: 'family-photo-2.jpeg' }), Error); + + t.true(/\[MemServer\] Photo\.update\(record\) failed because Photo with id: 99 does not exist/.test(error.message)); + + const secondError = t.throws(() => PhotoComment.update({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5666', content: 'Nice' }), Error); + + t.true(/\[MemServer\] PhotoComment\.update\(record\) failed because PhotoComment with uuid: 374c7f4a-85d6-429a-bf2a-0719525f5666 does not exist/.test(secondError.message)); +}); + +test.serial('$Model.update(attributes) throws an exception when a model gets updated with an unknown $Model.attribute', (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo, PhotoComment } = MemServer.Models; + + MemServer.start(); + + const error = t.throws(() => Photo.update({ id: 1, name: 'ME', is_verified: false }), Error); + + t.true(/\[MemServer\] Photo\.update id: 1 fails, Photo model does not have is_verified attribute to update/.test(error.message)); + + const secondError = t.throws(() => PhotoComment.update({ uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', location: 'Amsterdam' }), Error); + + t.true(/\[MemServer\] PhotoComment\.update uuid: 374c7f4a-85d6-429a-bf2a-0719525f5f29 fails, PhotoComment model does not have location attribute to update/.test(secondError.message)); +}); diff --git a/test/server/config.js b/test/server/config.js new file mode 100644 index 0000000..2abdbfb --- /dev/null +++ b/test/server/config.js @@ -0,0 +1,286 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); + +test.before(async () => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + is_public: true, + name() { + return 'Some default name'; + } + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + inserted_at() { + return '2017-10-25T20:54:04.447Z'; + }, + is_important: true + } + }); + `), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false, + user_id: 1 + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true, + user_id: 1 + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false, + user_id: 1 + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`) + ]); +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('namespace configuration option could be passed in during MemServer.start()', async (t) => { + t.plan(2); + + await fs.writeFile(`${CWD}/memserver/server.js`, ` + import Response from '../lib/response'; + + export default function({ Photo }) { + this.get('/photos', () => { + const photos = Photo.findAll(); + + if (!photos || photos.length === 0) { + return Response(404, { error: 'Not found' }); + } + + return { photos: Photo.serializer(photos) }; + }); + } + `); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start({ namespace: 'api/v1' }); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: '/api/v1/photos', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); + }); +}); + +test.serial('server this.namespace() configuration can overwrite existing namespace config', async (t) => { + t.plan(2); + + await fs.writeFile(`${CWD}/memserver/server.js`, ` + import Response from '../lib/response'; + + export default function({ Photo }) { + this.namespace = 'api/'; + + this.get('/photos', () => { + const photos = Photo.findAll(); + + if (!photos || photos.length === 0) { + return Response(404, { error: 'Not found' }); + } + + return { photos: Photo.serializer(photos) }; + }); + } + `); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start({ namespace: 'api/v1' }); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: '/api/photos', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); + }); +}); + +test.serial('urlPrefix configuration option could be passed in during MemServer.start()', async (t) => { + t.plan(2); + + await fs.writeFile(`${CWD}/memserver/server.js`, ` + import Response from '../lib/response'; + + export default function({ Photo }) { + this.namespace = 'api/'; + this.get('/photos', () => { + const photos = Photo.findAll(); + + if (!photos || photos.length === 0) { + return Response(404, { error: 'Not found' }); + } + + return { photos: Photo.serializer(photos) }; + }); + } + `); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start({ urlPrefix: 'http://twitter.com' }); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: 'http://twitter.com/api/photos', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); + }); +}); + +test.serial('server this.urlPrefix() configuration can overwrite existing urlPrefix config', async (t) => { + t.plan(2); + + await fs.writeFile(`${CWD}/memserver/server.js`, ` + import Response from '../lib/response'; + + export default function({ Photo }) { + this.urlPrefix = 'http://facebook.com/'; + this.namespace = 'api'; + + this.get('/photos', () => { + const photos = Photo.findAll(); + + if (!photos || photos.length === 0) { + return Response(404, { error: 'Not found' }); + } + + return { photos: Photo.serializer(photos) }; + }); + } + `); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start({ urlPrefix: 'http://twitter.com' }); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: 'http://facebook.com/api/photos', + headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); + }); +}); + +test.serial('timing configuration option could be passed in during MemServer.start()', async (t) => { + t.plan(3); + + await fs.writeFile(`${CWD}/memserver/server.js`, ` + import Response from '../lib/response'; + + export default function({ Photo }) { + this.get('/photos', () => { + const photos = Photo.findAll(); + + if (!photos || photos.length === 0) { + return Response(404, { error: 'Not found' }); + } + + return { photos: Photo.serializer(photos) }; + }); + } + `); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start({ timing: 3000 }); + window.$ = require('jquery'); + + let ThreeSecondsPassed = false; + + setTimeout(() => { ThreeSecondsPassed = true; }, 2200); + + await window.$.ajax({ + type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.true(ThreeSecondsPassed); + t.is(jqXHR.status, 200); + t.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); + }); +}); + +// test.serial('server this.get(url, timing) configuration can overwrite existing timing config', async () => { +// +// }); diff --git a/test/server/handler-defaults.js b/test/server/handler-defaults.js new file mode 100644 index 0000000..2aab07c --- /dev/null +++ b/test/server/handler-defaults.js @@ -0,0 +1,314 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const CWD = process.cwd(); + +test.before(async () => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + is_public: true, + name() { + return 'Some default name'; + } + } + }); + `), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false, + user_id: 1 + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true, + user_id: 1 + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false, + user_id: 1 + } + ];`); +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('POST /resources work with shortcut', async (t) => { + t.plan(5); + + await writeServerFileWithNoHandlers(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + t.is(Photo.count(), 3); + + await window.$.ajax({ + type: 'POST', url: '/photos', headers: { 'Content-Type': 'application/json' }, + data: JSON.stringify({ photo: { name: 'Izel Nakri' }}) + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 201); + t.deepEqual(data, { photo: Photo.serializer(Photo.find(4)) }); + t.is(Photo.count(), 4); + t.deepEqual(Photo.find(4), { + id: 4, name: 'Izel Nakri', is_public: true, href: undefined, user_id: undefined + }); + }); +}); + +test.serial('GET /resources works with shortcut', async (t) => { + t.plan(4); + + await writeServerFileWithNoHandlers(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + t.is(Photo.count(), 3); + + await window.$.ajax({ + type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); + t.is(Photo.count(), 3); + }); +}); + +test.serial('GET /resources/:id works with shortcut', async (t) => { + t.plan(2); + + await writeServerFileWithNoHandlers(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: '/photos/1', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photo: Photo.serializer(Photo.find(1)) }); + }); +}); + +test.serial('PUT /resources/:id works with shortcut', async (t) => { + t.plan(4); + + await writeServerFileWithNoHandlers(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + t.is(Photo.find(1).name, 'Ski trip'); + + await window.$.ajax({ + type: 'PUT', url: '/photos/1', headers: { 'Content-Type': 'application/json' }, + data: JSON.stringify({ photo: { id: 1, name: 'New custom title'} }) + }).then((data, textStatus, jqXHR) => { + const photo = Photo.find(1); + + t.is(jqXHR.status, 200); + t.deepEqual(data, { photo: Photo.serializer(photo) }); + t.is(photo.name, 'New custom title'); + }); +}); + +test.serial('DELETE /resources/:id works with shortcut', async (t) => { + t.plan(5); + + await writeServerFileWithNoHandlers(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + t.is(Photo.count(), 3); + + await window.$.ajax({ + type: 'DELETE', url: '/photos/1', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 204); + t.deepEqual(data, undefined); + t.is(Photo.count(), 2); + t.is(Photo.find(1), undefined); + }); +}); + +test.serial('throws an helpful error message when shortcuts model is not found', async (t) => { + t.plan(2); + + await fs.writeFile(`${CWD}/memserver/server.js`, ` + export default function(Models) { + this.post('/photos'); + this.get('/photos'); + this.get('/photos/:id'); + this.put('/photos/:id'); + this.delete('/photos/:id'); + + this.get('/houses'); + } + `); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); + + const MemServer = require('../../lib/index.js'); + const error = t.throws(() => MemServer.start(), Error); + + t.true(/\[MemServer\] GET \/houses route handler cannot be generated automatically: House is not a valid MemServer\.Model, please check that your route name matches the model reference or create a custom handler function/.test(error.message)); +}); + +test.serial('POST /resources works correctly with undefined handler response', async (t) => { + t.plan(4); + + await writeServerFileWithEmptyHandlers(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + t.is(Photo.count(), 3); + + await window.$.ajax({ + type: 'POST', url: '/photos', headers: { 'Content-Type': 'application/json' } + }).catch((jqXHR) => { + t.is(jqXHR.status, 500); + t.deepEqual(jqXHR.responseJSON, { error: '[MemServer] POST /photos route handler did not return anything to respond to the request!' }); + t.is(Photo.count(), 3); + }); +}); + +test.serial('GET /resources works correctly with undefined handler response', async (t) => { + t.plan(2); + + await writeServerFileWithEmptyHandlers(); + + const MemServer = require('../../lib/index.js'); + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } + }).catch((jqXHR) => { + t.is(jqXHR.status, 500); + t.deepEqual(jqXHR.responseJSON, { error: '[MemServer] GET /photos route handler did not return anything to respond to the request!' }); + }); +}); + +test.serial('GET /resources/:id works correctly with undefined handler response', async (t) => { + t.plan(2); + + await writeServerFileWithEmptyHandlers(); + + const MemServer = require('../../lib/index.js'); + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: '/photos/1', headers: { 'Content-Type': 'application/json' } + }).catch((jqXHR) => { + t.is(jqXHR.status, 500); + t.deepEqual(jqXHR.responseJSON, { error: '[MemServer] GET /photos/1 route handler did not return anything to respond to the request!' }); + }); +}); + +test.serial('PUT /resources/:id works correctly with undefined handler response', async (t) => { + t.plan(2); + + await writeServerFileWithEmptyHandlers(); + + const MemServer = require('../../lib/index.js'); + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'PUT', url: '/photos/1', headers: { 'Content-Type': 'application/json' }, + data: JSON.stringify({ photo: { id: 1, name: 'New Name' }}) + }).catch((jqXHR) => { + t.is(jqXHR.status, 500); + t.deepEqual(jqXHR.responseJSON, { error: '[MemServer] PUT /photos/1 route handler did not return anything to respond to the request!' }); + }); +}); + +test.serial('DELETE /resources/:id works correctly with undefined handler response', async (t) => { + t.plan(2); + + await writeServerFileWithNoHandlers(); + + const MemServer = require('../../lib/index.js'); + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'DELETE', url: '/photos/1', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 204); + t.deepEqual(jqXHR.responseJSON, {}); + }); +}); + +async function writeServerFileWithNoHandlers() { + await fs.writeFile(`${CWD}/memserver/server.js`, ` + export default function(Models) { + this.post('/photos'); + this.get('/photos'); + this.get('/photos/:id'); + this.put('/photos/:id'); + this.delete('/photos/:id'); + } + `); +} + +async function writeServerFileWithEmptyHandlers() { + await fs.writeFile(`${CWD}/memserver/server.js`, ` + export default function(Models) { + this.post('/photos', () => {}); + this.get('/photos', () => {}); + this.get('/photos/:id', () => {}); + this.put('/photos/:id', () => {}); + this.delete('/photos/:id', () => {}); + } + `); +} diff --git a/test/server/index.js b/test/server/index.js new file mode 100644 index 0000000..d9f33f1 --- /dev/null +++ b/test/server/index.js @@ -0,0 +1,328 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const AUTHENTICATION_TOKEN = 'ec25fc7b-6ee2-4bda-b57c-6c9867b30ff4'; +const AJAX_AUTHORIZATION_HEADERS = { + 'Content-Type': 'application/json', 'Authorization': `Token ${AUTHENTICATION_TOKEN}` +}; +const CWD = process.cwd(); + +test.before(async () => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/user.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + findFromHeaderToken(headers) { + const authorizationHeader = headers.Authorization; + const token = authorizationHeader ? authorizationHeader.slice(6) : false; + + return this.findBy({ authentication_token: token }) || false; + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + is_public: true, + name() { + return 'Some default name'; + } + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + inserted_at() { + return '2017-10-25T20:54:04.447Z'; + }, + is_important: true + } + }); + `), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/users.js`, `export default [ + { + id: 1, + email: 'contact@izelnakri.com', + username: 'izelnakri', + authentication_token: '${AUTHENTICATION_TOKEN}' + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false, + user_id: 1 + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true, + user_id: 1 + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false, + user_id: 1 + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`), + fs.writeFile(`${CWD}/memserver/server.js`, ` + import Response from '../lib/response'; + + export default function({ User, Photo }) { + this.post('/photos', ({ headers }) => { + const user = User.findFromHeaderToken(headers); + + if (!user) { + return Response(401, { error: 'Unauthorized' }); + } + + const photo = Photo.insert({ user_id: user.id }); + + return { photo: Photo.serializer(photo) }; + }); + + this.get('/photos', ({ headers }) => { + const user = User.findFromHeaderToken(headers); + + if (!user) { + return Response(404, { error: 'Not found' }); + } + + const photos = Photo.findAll({ user_id: user.id }); + + return { photos: Photo.serializer(photos) }; + }); + + this.get('/photos/:id', ({ headers, params }) => { + const user = User.findFromHeaderToken(headers); + + if (!user) { + return Response(401, { error: 'Unauthorized' }); + } + + const photo = Photo.findBy({ id: params.id, user_id: user.id }); + + return photo ? { photo: Photo.serializer(photo) } : Response(404, { error: 'Not found'}); + }); + + this.put('/photos/:id', ({ headers, params }) => { + const user = User.findFromHeaderToken(headers); + + if (!user) { + return Response(401, { error: 'Unauthorized' }); + } + + if (Photo.findBy({ id: params.id, user_id: user.id })) { + return { photo: Photo.update(params.photo) }; + } + }); + + this.delete('/photos/:id', ({ headers, params }) => { + const user = User.findFromHeaderToken(headers); + + if (user && Photo.findBy({ id: params.id, user_id: user.id })) { + return Photo.delete({ id: params.id }); + } + }); + + this.get('http://izelnakri.com', () => { + return Response(200, { result: 'external urls work!!' }) + }) + } + `) + ]); +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('POST /resources work with custom headers and responses', async (t) => { + t.plan(5); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + t.is(Photo.count(), 3); + + await window.$.ajax({ + type: 'POST', url: '/photos', headers: { 'Content-Type': 'application/json' } + }).catch((jqXHR) => { + t.is(jqXHR.status, 401); + t.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); + }); + await window.$.ajax({ + type: 'POST', url: '/photos', headers: AJAX_AUTHORIZATION_HEADERS + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 201); + t.deepEqual(data, { + photo: { is_public: true, name: 'Some default name', id: 4, user_id: 1, href: null } + }); + }); +}); + +test.serial('GET /resources works with custom headers and responses', async (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } + }).catch((jqXHR) => { + t.is(jqXHR.status, 404); + t.deepEqual(jqXHR.responseJSON, { error: 'Not found' }); + }); + await window.$.ajax({ + type: 'GET', url: '/photos', headers: AJAX_AUTHORIZATION_HEADERS + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photos: Photo.serializer(Photo.findAll())}); + }); +}); + +test.serial('GET /resources/:id works with custom headers and responses', async (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: '/photos/1', headers: { 'Content-Type': 'application/json' } + }).catch((jqXHR) => { + t.is(jqXHR.status, 401); + t.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); + }); + await window.$.ajax({ + type: 'GET', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photo: Photo.serializer(Photo.find(1))}); + }); +}); + +test.serial('PUT /resources/:id works with custom headers and responses', async (t) => { + t.plan(5); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'PUT', url: '/photos/1', headers: { 'Content-Type': 'application/json' }, + data: JSON.stringify({ photo: { id: 1, name: 'Photo after edit' }}) + }).catch((jqXHR) => { + t.is(jqXHR.status, 401); + t.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); + }); + await window.$.ajax({ + type: 'PUT', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS, + data: JSON.stringify({ photo: { id: 1, name: 'Photo after edit' } }) + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photo: Photo.serializer(Photo.find(1))}); + t.is(Photo.find(1).name, 'Photo after edit'); + }); +}); + +test.serial('DELETE /resources/:id works with custom headers and responses', async (t) => { + t.plan(4); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + t.truthy(Photo.find(1), 'User id: 1 exists'); + + await window.$.ajax({ + type: 'DELETE', url: '/photos/1', headers: { 'Content-Type': 'application/json' } + }).catch((jqXHR) => { + t.is(jqXHR.status, 401); + t.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); + t.true(Photo.find(1), 'User id: 1 exists'); + }); + await window.$.ajax({ + type: 'DELETE', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 204); + t.deepEqual(data, undefined); + t.true(!Photo.find(1), 'User id: 1 gets deleted'); + }); +}); + +test.serial('MemServer.Server works for external links', async (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: 'http://izelnakri.com', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(jqXHR.responseJSON, { result: 'external urls work!!' }); + }); +}); diff --git a/test/server/mem-server.server.config.js b/test/server/mem-server.server.config.js deleted file mode 100644 index 28cd90c..0000000 --- a/test/server/mem-server.server.config.js +++ /dev/null @@ -1,279 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -describe('MemServer.Server shortcut functionality', function() { - before(function() { - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - is_public: true, - name() { - return 'Some default name'; - } - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - inserted_at() { - return '2017-10-25T20:54:04.447Z'; - }, - is_important: true - } - }); - `); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false, - user_id: 1 - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true, - user_id: 1 - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false, - user_id: 1 - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - }); - - beforeEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - done(); - }); - - it('namespace configuration option could be passed in during MemServer.start()', async function() { - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - import Response from '../lib/response'; - - export default function({ Photo }) { - this.get('/photos', () => { - const photos = Photo.findAll(); - - if (!photos || photos.length === 0) { - return Response(404, { error: 'Not found' }); - } - - return { photos: Photo.serializer(photos) }; - }); - } - `); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start({ namespace: 'api/v1' }); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/api/v1/photos', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); - }); - }); - - it('server this.namespace() configuration can overwrite existing namespace config', async function() { - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - import Response from '../lib/response'; - - export default function({ Photo }) { - this.namespace = 'api/'; - - this.get('/photos', () => { - const photos = Photo.findAll(); - - if (!photos || photos.length === 0) { - return Response(404, { error: 'Not found' }); - } - - return { photos: Photo.serializer(photos) }; - }); - } - `); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start({ namespace: 'api/v1' }); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/api/photos', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); - }); - }); - - it('urlPrefix configuration option could be passed in during MemServer.start()', async function() { - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - import Response from '../lib/response'; - - export default function({ Photo }) { - this.namespace = 'api/'; - this.get('/photos', () => { - const photos = Photo.findAll(); - - if (!photos || photos.length === 0) { - return Response(404, { error: 'Not found' }); - } - - return { photos: Photo.serializer(photos) }; - }); - } - `); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start({ urlPrefix: 'http://twitter.com' }); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: 'http://twitter.com/api/photos', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); - }); - }); - - it('server this.urlPrefix() configuration can overwrite existing urlPrefix config', async function() { - this.timeout(5000); - - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - import Response from '../lib/response'; - - export default function({ Photo }) { - this.urlPrefix = 'http://facebook.com/'; - this.namespace = 'api'; - - this.get('/photos', () => { - const photos = Photo.findAll(); - - if (!photos || photos.length === 0) { - return Response(404, { error: 'Not found' }); - } - - return { photos: Photo.serializer(photos) }; - }); - } - `); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start({ urlPrefix: 'http://twitter.com' }); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: 'http://facebook.com/api/photos', - headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); - }); - }); - - // it('timing configuration option could be passed in during MemServer.start()', async function() { - // this.timeout(5000); - // - // fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - // import Response from '../lib/response'; - // - // export default function({ Photo }) { - // this.get('/photos', () => { - // const photos = Photo.findAll(); - // - // if (!photos || photos.length === 0) { - // return Response(404, { error: 'Not found' }); - // } - // - // return { photos: Photo.serializer(photos) }; - // }); - // } - // `); - // - // const MemServer = require('../../lib/index.js'); - // const { Photo } = MemServer.Models; - // - // MemServer.start({ timing: 3000 }); - // - // let timer; - // var ThreeSecondsPassed = false; - // setTimeout(() => { ThreeSecondsPassed = true; }, 1900); - // await window.$.ajax({ - // type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } - // }).then((data, textStatus, jqXHR) => { - // assert.equal(ThreeSecondsPassed, true); - // assert.equal(jqXHR.status, 200); - // assert.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); - // }); - // }); - - // it('server this.get(url, timing) configuration can overwrite existing timing config', function() { - // const MemServer = require('../../lib/index.js'); - // - // MemServer.start({ timing: 2000 }); - // }); -}); diff --git a/test/server/mem-server.server.defaults.js b/test/server/mem-server.server.defaults.js deleted file mode 100644 index 56c085d..0000000 --- a/test/server/mem-server.server.defaults.js +++ /dev/null @@ -1,289 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -process.setMaxListeners(0); - -describe('MemServer.Server shortcut functionality', function() { - before(function() { - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - is_public: true, - name() { - return 'Some default name'; - } - } - }); - `); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false, - user_id: 1 - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true, - user_id: 1 - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false, - user_id: 1 - } - ];`); - }); - - beforeEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - done(); - }); - - describe('route shortcuts work', function() { - before(function() { - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - export default function(Models) { - this.post('/photos'); - this.get('/photos'); - this.get('/photos/:id'); - this.put('/photos/:id'); - this.delete('/photos/:id'); - } - `); - }); - - it('POST /resources work with shortcut', async function() { - this.timeout(5000); - - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - assert.equal(Photo.count(), 3); - - await window.$.ajax({ - type: 'POST', url: '/photos', headers: { 'Content-Type': 'application/json' }, - data: JSON.stringify({ photo: { name: 'Izel Nakri' }}) - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 201); - assert.deepEqual(data, { photo: Photo.serializer(Photo.find(4)) }); - assert.equal(Photo.count(), 4); - assert.deepEqual(Photo.find(4), { - id: 4, name: 'Izel Nakri', is_public: true, href: null, user_id: null - }) - }); - }); - - it('GET /resources works with shortcut', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - assert.equal(Photo.count(), 3); - - await window.$.ajax({ - type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photos: Photo.serializer(Photo.findAll()) }); - assert.equal(Photo.count(), 3); - }); - }); - - it('GET /resources/:id works with shortcut', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/photos/1', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photo: Photo.serializer(Photo.find(1)) }); - }); - }); - - it('PUT /resources/:id works with shortcut', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - assert.equal(Photo.find(1).name, 'Ski trip') - - await window.$.ajax({ - type: 'PUT', url: '/photos/1', headers: { 'Content-Type': 'application/json' }, - data: JSON.stringify({ photo: { id: 1, name: 'New custom title'} }) - }, (data, textStatus, jqXHR) => { - const photo = Photo.find(1); - - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photo: Photo.serializer(photo) }); - assert.equal(photo.name, 'New custom title'); - }); - }); - - it('DELETE /resources/:id works with shortcut', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - assert.equal(Photo.count(), 3); - - await window.$.ajax({ - type: 'DELETE', url: '/photos/1', headers: { 'Content-Type': 'application/json' } - }, (data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 204); - assert.deepEqual(data, {}); - assert.equal(Photo.count(), 2); - assert.equal(PHoto.find(1), undefined); - }); - }); - }); - - it('throws an helpful error message when shortcuts model is not found', async function() { - this.timeout(5000); - - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - export default function(Models) { - this.post('/photos'); - this.get('/photos'); - this.get('/photos/:id'); - this.put('/photos/:id'); - this.delete('/photos/:id'); - - this.get('/houses'); - } - `); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - const MemServer = require('../../lib/index.js'); - - assert.throws(() => MemServer.start(), (err) => { - return (err instanceof Error) && - /\[MemServer\] GET \/houses route handler cannot be generated automatically\: House is not a valid MemServer\.Model, please check that your route name matches the model reference or create a custom handler function/.test(err); - }); - }); - - describe('Server route handlers default responses', function() { - before(function() { - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - export default function(Models) { - this.post('/photos', () => {}); - this.get('/photos', () => {}); - this.get('/photos/:id', () => {}); - this.put('/photos/:id', () => {}); - this.delete('/photos/:id', () => {}); - } - `); - }); - - it('POST /resources works correctly with undefined handler response', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - assert.equal(Photo.count(), 3); - - await window.$.ajax({ - type: 'POST', url: '/photos', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 500); - assert.deepEqual(jqXHR.responseJSON, { error: '[MemServer] POST /photos route handler did not return anything to respond to the request!' }); - assert.equal(Photo.count(), 3); - }); - }); - - it('GET /resources works correctly with undefined handler response', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 500); - assert.deepEqual(jqXHR.responseJSON, { error: '[MemServer] GET /photos route handler did not return anything to respond to the request!' }); - }); - }); - - it('GET /resources/:id works correctly with undefined handler response', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/photos/1', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 500); - assert.deepEqual(jqXHR.responseJSON, { error: '[MemServer] GET /photos/1 route handler did not return anything to respond to the request!' }); - }); - }); - - it('PUT /resources/:id works correctly with undefined handler response', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'PUT', url: '/photos/1', headers: { 'Content-Type': 'application/json' }, - data: JSON.stringify({ photo: { id: 1, name: 'New Name' }}) - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 500); - assert.deepEqual(jqXHR.responseJSON, { error: '[MemServer] PUT /photos/1 route handler did not return anything to respond to the request!' }); - }); - }); - - it('DELETE /resources/:id works correctly with undefined handler response', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'DELETE', url: '/photos/1', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 204); - assert.deepEqual(jqXHR.responseJSON, {}); - }); - }); - }); -}); diff --git a/test/server/mem-server.server.js b/test/server/mem-server.server.js deleted file mode 100644 index 1d3c0ae..0000000 --- a/test/server/mem-server.server.js +++ /dev/null @@ -1,340 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -const AUTHENTICATION_TOKEN = 'ec25fc7b-6ee2-4bda-b57c-6c9867b30ff4'; -const AJAX_AUTHORIZATION_HEADERS = { - 'Content-Type': 'application/json', 'Authorization': `Token ${AUTHENTICATION_TOKEN}` -}; - -process.setMaxListeners(0); - -describe('MemServer.Server general functionality', function() { - before(function() { - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - findFromHeaderToken(headers) { - const authorizationHeader = headers.Authorization; - const token = authorizationHeader ? authorizationHeader.slice(6) : false; - - return this.findBy({ authentication_token: token }) || false; - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - is_public: true, - name() { - return 'Some default name'; - } - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - inserted_at() { - return '2017-10-25T20:54:04.447Z'; - }, - is_important: true - } - }); - `); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/users.js`, `export default [ - { - id: 1, - email: 'contact@izelnakri.com', - username: 'izelnakri', - authentication_token: '${AUTHENTICATION_TOKEN}' - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false, - user_id: 1 - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true, - user_id: 1 - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false, - user_id: 1 - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - }); - - beforeEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - done(); - }); - - describe('server can process custom headers and responses', function() { - before(function() { - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - import Response from '../lib/response'; - - export default function({ User, Photo }) { - this.post('/photos', ({ headers }) => { - const user = User.findFromHeaderToken(headers); - - if (!user) { - return Response(401, { error: 'Unauthorized' }); - } - - const photo = Photo.insert({ user_id: user.id }); - - return { photo: Photo.serializer(photo) }; - }); - - this.get('/photos', ({ headers }) => { - const user = User.findFromHeaderToken(headers); - - if (!user) { - return Response(404, { error: 'Not found' }); - } - - const photos = Photo.findAll({ user_id: user.id }); - - return { photos: Photo.serializer(photos) }; - }); - - this.get('/photos/:id', ({ headers, params }) => { - const user = User.findFromHeaderToken(headers); - - if (!user) { - return Response(401, { error: 'Unauthorized' }); - } - - const photo = Photo.findBy({ id: params.id, user_id: user.id }); - - return photo ? { photo: Photo.serializer(photo) } : Response(404, { error: 'Not found'}); - }); - - this.put('/photos/:id', ({ headers, params }) => { - const user = User.findFromHeaderToken(headers); - - if (!user) { - return Response(401, { error: 'Unauthorized' }); - } - - if (Photo.findBy({ id: params.id, user_id: user.id })) { - return { photo: Photo.update(params.photo) }; - } - }); - - this.delete('/photos/:id', ({ headers, params }) => { - const user = User.findFromHeaderToken(headers); - - if (user && Photo.findBy({ id: params.id, user_id: user.id })) { - return Photo.delete({ id: params.id }); - } - }); - } - `); - }); - - it('POST /resources work with custom headers and responses', async function() { - this.timeout(5000); - - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - assert.equal(Photo.count(), 3); - - await window.$.ajax({ - type: 'POST', url: '/photos', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 401); - assert.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); - }); - - await window.$.ajax({ - type: 'POST', url: '/photos', headers: AJAX_AUTHORIZATION_HEADERS - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 201); - assert.deepEqual(data, { - photo: { is_public: true, name: 'Some default name', id: 4, user_id: 1, href: null } - }); - }); - }); - - it('GET /resources works with custom headers and responses', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 404); - assert.deepEqual(jqXHR.responseJSON, { error: 'Not found' }); - }); - - await window.$.ajax({ - type: 'GET', url: '/photos', headers: AJAX_AUTHORIZATION_HEADERS - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photos: Photo.serializer(Photo.findAll())}); - }); - }); - - it('GET /resources/:id works with custom headers and responses', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/photos/1', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 401); - assert.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); - }); - - await window.$.ajax({ - type: 'GET', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photo: Photo.serializer(Photo.find(1))}); - }); - }); - - it('PUT /resources/:id works with custom headers and responses', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'PUT', url: '/photos/1', headers: { 'Content-Type': 'application/json' }, - data: JSON.stringify({ photo: { id: 1, name: 'Photo after edit' }}) - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 401); - assert.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); - }); - - await window.$.ajax({ - type: 'PUT', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS, - data: JSON.stringify({ photo: { id: 1, name: 'Photo after edit' } }) - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photo: Photo.serializer(Photo.find(1))}); - assert.equal(Photo.find(1).name, 'Photo after edit'); - }); - }); - - it('DELETE /resources/:id works with custom headers and responses', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - assert.ok(Photo.find(1), 'User id: 1 exists'); - - await window.$.ajax({ - type: 'DELETE', url: '/photos/1', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 401); - assert.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); - assert.ok(Photo.find(1), 'User id: 1 exists'); - }); - - await window.$.ajax({ - type: 'DELETE', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 204); - assert.deepEqual(data, undefined); - assert.ok(!Photo.find(1), 'User id: 1 gets deleted'); - }); - }); - }); - - describe('some edge cases', function() { - before(function() { - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - import Response from '../lib/response'; - - export default function() { - this.get('http://izelnakri.com', () => { - return Response(200, { result: 'external urls work!!' }) - }) - } - `); - }); - - it('works for external links', async function() { - const MemServer = require('../../lib/index.js'); - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: 'http://izelnakri.com', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(jqXHR.responseJSON, { result: 'external urls work!!' }); - }); - }); - }); -}); diff --git a/test/server/mem-server.server.params.js b/test/server/mem-server.server.params.js deleted file mode 100644 index 185426c..0000000 --- a/test/server/mem-server.server.params.js +++ /dev/null @@ -1,517 +0,0 @@ -// TODO: if status is 0 then THROW ASSERTION!! -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); - -const AUTHENTICATION_TOKEN = 'ec25fc7b-6ee2-4bda-b57c-6c9867b30ff4'; -const AJAX_AUTHORIZATION_HEADERS = { - 'Content-Type': 'application/json', 'Authorization': `Token ${AUTHENTICATION_TOKEN}` -}; - -process.setMaxListeners(0); - -describe('MemServer.Server Parameters and Query Parameters', function() { - before(function() { - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/user.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - findFromHeaderToken(headers) { - const authorizationHeader = headers.Authorization; - const token = authorizationHeader ? authorizationHeader.slice(6) : false; - - return this.findBy({ authentication_token: token }) || false; - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - is_public: true, - name() { - return 'Some default name'; - } - } - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo-comment.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - defaultAttributes: { - inserted_at() { - return '2017-10-25T20:54:04.447Z'; - }, - is_important: true - } - }); - `); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/users.js`, `export default [ - { - id: 1, - email: 'contact@izelnakri.com', - username: 'izelnakri', - authentication_token: '${AUTHENTICATION_TOKEN}' - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false, - user_id: 1 - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true, - user_id: 1 - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false, - user_id: 1 - } - ];`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photo-comments.js`, `export default [ - { - uuid: '499ec646-493f-4eea-b92e-e383d94182f4', - content: 'What a nice photo!', - photo_id: 1, - user_id: 1 - }, - { - uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', - content: 'I agree', - photo_id: 1, - user_id: 2 - }, - { - uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', - content: 'I was kidding', - photo_id: 1, - user_id: 1 - }, - { - uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', - content: 'Interesting indeed', - photo_id: 2, - user_id: 1 - } - ];`); - }); - - beforeEach(function() { - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - done(); - }); - - describe('server can process custom queryParams and responses', function() { - before(function(){ - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - import Response from '../lib/response'; - - export default function({ User, Photo }) { - this.post('/photos', ({ headers, params, queryParams }) => { - const user = User.findFromHeaderToken(headers); - - if (!user || !queryParams.is_admin) { - return Response(401, { error: 'Unauthorized' }); - } - - const photo = Photo.insert(Object.assign({}, params.photo, { user_id: user.id })); - - return { photo: Photo.serializer(photo) }; - }); - - this.get('/photos', ({ headers, queryParams }) => { - const user = User.findFromHeaderToken(headers); - - if (!user) { - return Response(404, { error: 'Not found' }); - } - - const photos = Photo.findAll(Object.assign({}, { user_id: user.id }, queryParams)); - - if (!photos || photos.length === 0) { - return Response(404, { error: 'Not found' }); - } - - return { photos: Photo.serializer(photos) }; - }); - - this.get('/photos/:id', ({ headers, params, queryParams }) => { - const user = User.findFromHeaderToken(headers); - - if (!user) { - return Response(401, { error: 'Unauthorized' }); - } else if (queryParams.nonce === 123123123) { - const photo = Photo.findBy({ id: params.id, user_id: user.id }); - - return photo ? { photo: Photo.serializer(photo) } : Response(404, { error: 'Not found' }); - } - - return Response(404, { error: 'Not found' }); - }); - - this.put('/photos/:id', ({ headers, params, queryParams }) => { - const user = User.findFromHeaderToken(headers); - const validRequest = user && queryParams.nonce === 123123123 && - Photo.findBy({ id: params.id, user_id: user.id }); - - if (validRequest) { - return { photo: Photo.serializer(Photo.update(params.photo)) }; - } - - return Response(500, { error: 'Unexpected error occured' }); - }); - - this.delete('/photos/:id', ({ headers, params, queryParams }) => { - const user = User.findFromHeaderToken(headers); - - if (!(queryParams.nonce === 123123123)) { - return Response(500, { error: 'Invalid nonce to delete a photo' }); - } else if (!user && !Photo.findBy({ id: params.id, user_id: user.id })) { - return Response(404, { error: 'Not found' }); - } - - Photo.delete({ id: params.id }); // NOTE: what to do with this response - }); - } - `); - }); - - it('POST /resources work with custom headers, queryParams and responses', async function() { - this.timeout(10000); - - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - assert.equal(Photo.count(), 3); - - await window.$.ajax({ - type: 'POST', url: '/photos', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 401); - assert.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); - }); - - await window.$.ajax({ - type: 'POST', url: '/photos', headers: AJAX_AUTHORIZATION_HEADERS - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 401); - assert.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); - }); - - await window.$.ajax({ - type: 'POST', url: '/photos?is_admin=true', headers: AJAX_AUTHORIZATION_HEADERS - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 201); - assert.deepEqual(data, { photo: Photo.serializer(Photo.find(4)) }); - assert.equal(Photo.count(), 4); - }); - }); - - it('GET /resources works with custom headers, queryParams and responses', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 404); - assert.deepEqual(jqXHR.responseJSON, { error: 'Not found' }); - }); - - await window.$.ajax({ - type: 'GET', url: '/photos?is_public=false', headers: AJAX_AUTHORIZATION_HEADERS - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photos: Photo.serializer(Photo.findAll({ is_public: false })) }); - }); - - await window.$.ajax({ - type: 'GET', url: '/photos?href=family-photo.jpeg', headers: AJAX_AUTHORIZATION_HEADERS - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photos: Photo.serializer(Photo.findAll({ href: 'family-photo.jpeg' })) }); - }); - }); - - it('GET /resources/:id works with custom headers, queryParams and responses', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 404); - assert.deepEqual(jqXHR.responseJSON, { error: 'Not found' }); - }); - - await window.$.ajax({ - type: 'GET', url: '/photos/1?nonce=123123123', headers: AJAX_AUTHORIZATION_HEADERS - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photo: Photo.serializer(Photo.find(1)) }); - }); - }); - - it('PUT /resources/:id works with custom headers, queryParams and responses', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'PUT', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS, - data: JSON.stringify({ photo: { id: 1, name: 'Life' } }) - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 500); - assert.deepEqual(jqXHR.responseJSON, { error: 'Unexpected error occured' }); - }); - - await window.$.ajax({ - type: 'PUT', url: '/photos/1?nonce=123123123', headers: AJAX_AUTHORIZATION_HEADERS, - data: JSON.stringify({ photo: { id: 1, name: 'Life' } }) - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photo: Photo.serializer(Photo.find(1)) }); - }); - }); - - it('DELETE /resources/:id works with custom headers, queryParams and responses', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'DELETE', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 500); - assert.deepEqual(jqXHR.responseJSON, { error: 'Invalid nonce to delete a photo' }); - }); - - await window.$.ajax({ - type: 'DELETE', url: '/photos/1?nonce=123123123', headers: AJAX_AUTHORIZATION_HEADERS - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 204); - }); - }); - }); - - describe('some edge cases', function() { - before(function() { - fs.writeFileSync(`${process.cwd()}/memserver/models/ethereum-account.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - }); - `); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/ethereum-accounts.js`, `export default [ - { - id: 1, - address: '0x7be8315acfef37816c9ad4dc5e82195f2a52934c5d0c74883f9978675e26d600' - } - ];`); - - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - import Response from '../lib/response'; - - export default function({ EthereumAccount, Photo, PhotoComment }) { - this.get('/ethereum-accounts', ({ queryParams }) => { - const ethereumAccounts = EthereumAccount.findAll({ address: queryParams.address }); - - return { ethereum_accounts: EthereumAccount.serializer(ethereumAccounts) }; - }); - - this.get('/ethereum-accounts/:address', ({ params }) => { - const ethereumAccount = EthereumAccount.findBy({ address: params.address }); - - return { ethereum_account: EthereumAccount.serializer(ethereumAccount) }; - }); - - this.get('/photos', ({ queryParams }) => { - const photos = Photo.find(queryParams.ids || []); - - if (!photos || photos.length === 0) { - return Response(404, { error: 'Not found' }); - } - - return { photos: Photo.serializer(photos) }; - }); - - this.get('/photo-comments/:uuid', ({ params }) => { - const photoComment = PhotoComment.findBy({ uuid: params.uuid }); - - return { photo_comment: PhotoComment.serializer(photoComment) }; - }); - - this.get('/photo-comments', ({ queryParams }) => { - const photoComments = PhotoComment.findAll({ uuid: queryParams.uuid }); - - return { photo_comments: PhotoComment.serializer(photoComments) }; - }); - } - `); - }); - - it('works for coalasceFindRequests routes', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } - }).catch((jqXHR) => { - assert.equal(jqXHR.status, 404); - assert.deepEqual(jqXHR.responseJSON, { error: 'Not found' }); - }); - - await window.$.ajax({ - type: 'GET', url: '/photos?ids[]=1&ids[]=2', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(jqXHR.responseJSON, { photos: Photo.serializer(Photo.find([1, 2])) }); - }); - - await window.$.ajax({ - type: 'GET', url: '/photos?ids[]=2&ids[]=3', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(jqXHR.responseJSON, { photos: Photo.serializer(Photo.find([2, 3])) }); - }); - }); - - it('converts empty strings to null in request and formats query params', async function() { - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - MemServer.Server.post('/photos', ({ params, queryParams }) => { - assert.deepEqual(params, { name: null, title: 'Cool' }); - assert.deepEqual(queryParams, { is_important: true, filter: 32 }); - - return { photo: Photo.serializer(Photo.insert(params)) }; - }); - - await window.$.ajax({ - type: 'POST', url: '/photos?is_important=true&filter=32', data: { name: '', title: 'Cool' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 201); - assert.equal(Photo.count(), 4); - }); - }); - - it('casts uuids correctly as params', async function() { - const MemServer = require('../../lib/index.js'); - const { PhotoComment } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - const targetComment = PhotoComment.findBy({ uuid: '499ec646-493f-4eea-b92e-e383d94182f4' }); - - await window.$.ajax({ - type: 'GET', url: '/photo-comments/499ec646-493f-4eea-b92e-e383d94182f4' - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photo_comment: PhotoComment.serializer(targetComment) }); - }); - }); - - it('casts uuids correctly as queryParams', async function() { - const MemServer = require('../../lib/index.js'); - const { PhotoComment } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - const targetComments = PhotoComment.findAll({ uuid: '499ec646-493f-4eea-b92e-e383d94182f4' }); - - await window.$.ajax({ - type: 'GET', url: '/photo-comments?uuid=499ec646-493f-4eea-b92e-e383d94182f4' - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { photo_comments: PhotoComment.serializer(targetComments) }); - }); - }); - - it('casts ethereum addresses correctly as string request.params', async function() { - this.timeout(10000); - - const MemServer = require('../../lib/index.js'); - const { EthereumAccount } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - const targetAccount = EthereumAccount.findBy({ - address: '0x7be8315acfef37816c9ad4dc5e82195f2a52934c5d0c74883f9978675e26d600' - }); - - await window.$.ajax({ - type: 'GET', - url: '/ethereum-accounts/0x7be8315acfef37816c9ad4dc5e82195f2a52934c5d0c74883f9978675e26d600' - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { ethereum_account: EthereumAccount.serializer(targetAccount) }); - }); - }); - - it('casts ethereum addresses correctly as string request.queryParams', async function() { - this.timeout(10000); - - const MemServer = require('../../lib/index.js'); - const { EthereumAccount } = MemServer.Models; - - MemServer.start(); - window.$ = require('jquery'); - - const targetAccounts = EthereumAccount.findAll({ - address: '0x7be8315acfef37816c9ad4dc5e82195f2a52934c5d0c74883f9978675e26d600' - }); - - await window.$.ajax({ - type: 'GET', - url: '/ethereum-accounts?address=0x7be8315acfef37816c9ad4dc5e82195f2a52934c5d0c74883f9978675e26d600' - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(data, { ethereum_accounts: EthereumAccount.serializer(targetAccounts) }); - }); - - }); - }); -}); diff --git a/test/server/mem-server.server.passthrough.js b/test/server/mem-server.server.passthrough.js deleted file mode 100644 index d1a4005..0000000 --- a/test/server/mem-server.server.passthrough.js +++ /dev/null @@ -1,198 +0,0 @@ -const assert = require('assert'); -const fs = require('fs'); -const rimraf = require('rimraf'); -const sinon = require('sinon'); -const express = require('express'); -const cors = require('cors'); - -let actualServer; - -describe('MemServer.Server general functionality', function() { - before(function() { - fs.mkdirSync(`./memserver`); - fs.mkdirSync(`./memserver/models`); - fs.writeFileSync(`${process.cwd()}/memserver/models/photo.js`, ` - import Model from '${process.cwd()}/lib/model'; - - export default Model({ - }); - `); - fs.mkdirSync(`./memserver/fixtures`); - fs.writeFileSync(`${process.cwd()}/memserver/fixtures/photos.js`, `export default [ - { - id: 1, - name: 'Ski trip', - href: 'ski-trip.jpeg', - is_public: false, - user_id: 1 - }, - { - id: 2, - name: 'Family photo', - href: 'family-photo.jpeg', - is_public: true, - user_id: 1 - }, - { - id: 3, - name: 'Selfie', - href: 'selfie.jpeg', - is_public: false, - user_id: 1 - } - ];`); - - let app = express(); - - app.use(cors()); - - app.get('/films', (req, res) => { - res.json({ film: 'responsed correctly' }); - }); - - app.get('/movies/too-big-to-fail', (req, res) => { - res.json({ movie: 'is too-big-to-fail' }); - }); - - actualServer = app.listen(4000, () => console.log('Web server started on port 4000')); - }); - - beforeEach(function() { - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - import Response from '../lib/response'; - - export default function({ Photo }) { - this.get('/photos', () => { - const photos = Photo.findAll(); - - if (!photos || photos.length === 0) { - return Response(404, { error: 'Not found' }); - } - - return { photos: Photo.serializer(photos) }; - }); - - this.passthrough('/films'); - this.passthrough('http://localhost:4000/films'); - this.passthrough('http://localhost:4000/movies/*'); - } - `); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - }); - - after(function(done) { - if (fs.existsSync(`${process.cwd()}/memserver`)) { - rimraf.sync(`${process.cwd()}/memserver`); - } - - actualServer.close(); - done(); - }); - - it('throws an error when MemServer tried to intercept an undeclared route', function() { - const MemServer = require('../../lib/index.js'); - - MemServer.start(); - MemServer.Server.unhandledRequest = sinon.spy(); - window.$ = require('jquery'); - - window.$.ajax({ - type: 'GET', url: '/izelnakri', headers: { 'Content-Type': 'application/json' } - }); - - assert.ok(MemServer.Server.unhandledRequest.calledOnce, 'MemServer.Server.unhandledRequest called once'); - }); - - it('this.passthrough(url) shortcut works', async function() { - const MemServer = require('../../lib/index.js'); - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: 'http://localhost:4000/films', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(jqXHR.responseJSON, { film: 'responsed correctly' }); - }); - }); - - it('this.passthrough(url) shortcut works with wild cards', async function() { - const MemServer = require('../../lib/index.js'); - - MemServer.start(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: 'http://localhost:4000/movies/too-big-to-fail', - headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(jqXHR.responseJSON, { movie: 'is too-big-to-fail' }); - }); - }); - - describe('global passthrough feature', function() { - beforeEach(function(done) { - fs.writeFileSync(`${process.cwd()}/memserver/server.js`, ` - import Response from '../lib/response'; - - export default function({ Photo }) { - this.get('/photos', () => { - const photos = Photo.findAll(); - - if (!photos || photos.length === 0) { - return Response(404, { error: 'Not found' }); - } - - return { photos: Photo.serializer(photos) }; - }); - - this.passthrough(); - } - `); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - done(); - }); - - it('can create global passthrough via this.passthrough()', async function() { - this.timeout(10000); - - Object.keys(require.cache).forEach((key) => delete require.cache[key]); - - const MemServer = require('../../lib/index.js'); - const { Photo } = MemServer.Models; - - MemServer.start(); - MemServer.Server.unhandledRequest = sinon.spy(); - window.$ = require('jquery'); - - await window.$.ajax({ - type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(jqXHR.responseJSON, { photos: Photo.serializer(Photo.findAll()) }); - }); - await window.$.ajax({ - type: 'GET', url: 'http://localhost:4000/films', headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(jqXHR.responseJSON, { film: 'responsed correctly' }); - }); - await window.$.ajax({ - type: 'GET', url: 'http://localhost:4000/movies/too-big-to-fail', - headers: { 'Content-Type': 'application/json' } - }).then((data, textStatus, jqXHR) => { - assert.equal(jqXHR.status, 200); - assert.deepEqual(jqXHR.responseJSON, { movie: 'is too-big-to-fail' }); - }); - }); - }); - - // TODO: test this.passthrough('/something') when there is this.namespace; - - // NOTE: passthrough order? investigate -}); diff --git a/test/server/params.js b/test/server/params.js new file mode 100644 index 0000000..aba5008 --- /dev/null +++ b/test/server/params.js @@ -0,0 +1,548 @@ +import test from 'ava'; +import fs from 'fs-extra'; + +const AUTHENTICATION_TOKEN = 'ec25fc7b-6ee2-4bda-b57c-6c9867b30ff4'; +const AJAX_AUTHORIZATION_HEADERS = { + 'Content-Type': 'application/json', 'Authorization': `Token ${AUTHENTICATION_TOKEN}` +}; +const CWD = process.cwd(); + +test.before(async () => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/user.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + findFromHeaderToken(headers) { + const authorizationHeader = headers.Authorization; + const token = authorizationHeader ? authorizationHeader.slice(6) : false; + + return this.findBy({ authentication_token: token }) || false; + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + is_public: true, + name() { + return 'Some default name'; + } + } + }); + `), + fs.writeFile(`${CWD}/memserver/models/photo-comment.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + defaultAttributes: { + inserted_at() { + return '2017-10-25T20:54:04.447Z'; + }, + is_important: true + } + }); + `), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/fixtures/users.js`, `export default [ + { + id: 1, + email: 'contact@izelnakri.com', + username: 'izelnakri', + authentication_token: '${AUTHENTICATION_TOKEN}' + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false, + user_id: 1 + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true, + user_id: 1 + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false, + user_id: 1 + } + ];`), + fs.writeFile(`${CWD}/memserver/fixtures/photo-comments.js`, `export default [ + { + uuid: '499ec646-493f-4eea-b92e-e383d94182f4', + content: 'What a nice photo!', + photo_id: 1, + user_id: 1 + }, + { + uuid: '77653ad3-47e4-4ec2-b49f-57ea36a627e7', + content: 'I agree', + photo_id: 1, + user_id: 2 + }, + { + uuid: 'd351963d-e725-4092-a37c-1ca1823b57d3', + content: 'I was kidding', + photo_id: 1, + user_id: 1 + }, + { + uuid: '374c7f4a-85d6-429a-bf2a-0719525f5f29', + content: 'Interesting indeed', + photo_id: 2, + user_id: 1 + } + ];`), + + ]); +}); + +test.beforeEach(() => { + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } +}); + +test.serial('[MemServer.Server] POST /resources work with custom headers, queryParams and responses', async (t) => { + t.plan(8); + + await writeSimpleRESTServerFile(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + t.is(Photo.count(), 3); + + await window.$.ajax({ + type: 'POST', url: '/photos', headers: { 'Content-Type': 'application/json' } + }).catch((jqXHR) => { + t.is(jqXHR.status, 401); + t.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); + }); + + await window.$.ajax({ + type: 'POST', url: '/photos', headers: AJAX_AUTHORIZATION_HEADERS + }).catch((jqXHR) => { + t.is(jqXHR.status, 401); + t.deepEqual(jqXHR.responseJSON, { error: 'Unauthorized' }); + }); + + await window.$.ajax({ + type: 'POST', url: '/photos?is_admin=true', headers: AJAX_AUTHORIZATION_HEADERS + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 201); + t.deepEqual(data, { photo: Photo.serializer(Photo.find(4)) }); + t.is(Photo.count(), 4); + }); +}); + +test.serial('[MemServer.Server] GET /resources works with custom headers, queryParams and responses', async (t) => { + t.plan(6); + + await writeSimpleRESTServerFile(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } + }).catch((jqXHR) => { + t.is(jqXHR.status, 404); + t.deepEqual(jqXHR.responseJSON, { error: 'Not found' }); + }); + + await window.$.ajax({ + type: 'GET', url: '/photos?is_public=false', headers: AJAX_AUTHORIZATION_HEADERS + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photos: Photo.serializer(Photo.findAll({ is_public: false })) }); + }); + + await window.$.ajax({ + type: 'GET', url: '/photos?href=family-photo.jpeg', headers: AJAX_AUTHORIZATION_HEADERS + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photos: Photo.serializer(Photo.findAll({ href: 'family-photo.jpeg' })) }); + }); +}); + +test.serial('[MemServer.Server] GET /resources/:id works with custom headers, queryParams and responses', async (t) => { + t.plan(4); + + await writeSimpleRESTServerFile(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS + }).catch((jqXHR) => { + t.is(jqXHR.status, 404); + t.deepEqual(jqXHR.responseJSON, { error: 'Not found' }); + }); + + await window.$.ajax({ + type: 'GET', url: '/photos/1?nonce=123123123', headers: AJAX_AUTHORIZATION_HEADERS + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photo: Photo.serializer(Photo.find(1)) }); + }); +}); + +test.serial('[MemServer.Server] PUT /resources/:id works with custom headers, queryParams and responses', async (t) => { + t.plan(4); + + await writeSimpleRESTServerFile(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'PUT', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS, + data: JSON.stringify({ photo: { id: 1, name: 'Life' } }) + }).catch((jqXHR) => { + t.is(jqXHR.status, 500); + t.deepEqual(jqXHR.responseJSON, { error: 'Unexpected error occured' }); + }); + + await window.$.ajax({ + type: 'PUT', url: '/photos/1?nonce=123123123', headers: AJAX_AUTHORIZATION_HEADERS, + data: JSON.stringify({ photo: { id: 1, name: 'Life' } }) + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photo: Photo.serializer(Photo.find(1)) }); + }); +}); + +test.serial('[MemServer.Server] DELETE /resources/:id works with custom headers, queryParams and responses', async (t) => { + t.plan(3); + + await writeSimpleRESTServerFile(); + + const MemServer = require('../../lib/index.js'); + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'DELETE', url: '/photos/1', headers: AJAX_AUTHORIZATION_HEADERS + }).catch((jqXHR) => { + t.is(jqXHR.status, 500); + t.deepEqual(jqXHR.responseJSON, { error: 'Invalid nonce to delete a photo' }); + }); + + await window.$.ajax({ + type: 'DELETE', url: '/photos/1?nonce=123123123', headers: AJAX_AUTHORIZATION_HEADERS + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 204); + }); +}); + +test.serial('[MemServer.Server Edge cases] Server works for coalasceFindRequests routes', async (t) => { + t.plan(6); + + await writeEdgeCasesServerFile(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } + }).catch((jqXHR) => { + t.is(jqXHR.status, 404); + t.deepEqual(jqXHR.responseJSON, { error: 'Not found' }); + }); + + await window.$.ajax({ + type: 'GET', url: '/photos?ids[]=1&ids[]=2', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(jqXHR.responseJSON, { photos: Photo.serializer(Photo.find([1, 2])) }); + }); + + await window.$.ajax({ + type: 'GET', url: '/photos?ids[]=2&ids[]=3', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(jqXHR.responseJSON, { photos: Photo.serializer(Photo.find([2, 3])) }); + }); +}); + +test.serial('[MemServer.Server Edge cases] Server converts empty strings to null during a request and formats query params', async (t) => { + t.plan(4); + + await writeEdgeCasesServerFile(); + + const MemServer = require('../../lib/index.js'); + const { Photo } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + MemServer.Server.post('/photos', ({ params, queryParams }) => { + t.deepEqual(params, { name: null, title: 'Cool' }); + t.deepEqual(queryParams, { is_important: true, filter: 32 }); + + return { photo: Photo.serializer(Photo.insert(params)) }; + }); + + await window.$.ajax({ + type: 'POST', url: '/photos?is_important=true&filter=32', data: { name: '', title: 'Cool' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 201); + t.is(Photo.count(), 4); + }); +}); + +test.serial('[MemServer.Server Edge cases] Server casts uuids correctly as params', async (t) => { + t.plan(2); + + await writeEdgeCasesServerFile(); + + const MemServer = require('../../lib/index.js'); + const { PhotoComment } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + const targetComment = PhotoComment.findBy({ uuid: '499ec646-493f-4eea-b92e-e383d94182f4' }); + + await window.$.ajax({ + type: 'GET', url: '/photo-comments/499ec646-493f-4eea-b92e-e383d94182f4' + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photo_comment: PhotoComment.serializer(targetComment) }); + }); +}); + +test.serial('[MemServer.Server Edge cases] Server casts uuids correct as queryParams', async (t) => { + t.plan(2); + + await writeEdgeCasesServerFile(); + + const MemServer = require('../../lib/index.js'); + const { PhotoComment } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + const targetComments = PhotoComment.findAll({ uuid: '499ec646-493f-4eea-b92e-e383d94182f4' }); + + await window.$.ajax({ + type: 'GET', url: '/photo-comments?uuid=499ec646-493f-4eea-b92e-e383d94182f4' + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { photo_comments: PhotoComment.serializer(targetComments) }); + }); +}); + +test.serial('[MemServer.Server Edga cases] Server casts ethereum adresses correctly as string request.params', async (t) => { + t.plan(2); + + await writeEdgeCasesServerFile(); + + const MemServer = require('../../lib/index.js'); + const { EthereumAccount } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + const targetAccount = EthereumAccount.findBy({ + address: '0x7be8315acfef37816c9ad4dc5e82195f2a52934c5d0c74883f9978675e26d600' + }); + + await window.$.ajax({ + type: 'GET', + url: '/ethereum-accounts/0x7be8315acfef37816c9ad4dc5e82195f2a52934c5d0c74883f9978675e26d600' + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { ethereum_account: EthereumAccount.serializer(targetAccount) }); + }); +}); + +test.serial('[MemServer.Server Edge cases] Server casts ethereum addresses correctly as string in request.queryParams', async (t) => { + t.plan(2); + + await writeEdgeCasesServerFile(); + + const MemServer = require('../../lib/index.js'); + const { EthereumAccount } = MemServer.Models; + + MemServer.start(); + window.$ = require('jquery'); + + const targetAccounts = EthereumAccount.findAll({ + address: '0x7be8315acfef37816c9ad4dc5e82195f2a52934c5d0c74883f9978675e26d600' + }); + + await window.$.ajax({ + type: 'GET', + url: '/ethereum-accounts?address=0x7be8315acfef37816c9ad4dc5e82195f2a52934c5d0c74883f9978675e26d600' + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(data, { ethereum_accounts: EthereumAccount.serializer(targetAccounts) }); + }); +}); + +async function writeSimpleRESTServerFile() { + await fs.writeFile(`${CWD}/memserver/server.js`, ` + import Response from '../lib/response'; + + export default function({ User, Photo }) { + this.post('/photos', ({ headers, params, queryParams }) => { + const user = User.findFromHeaderToken(headers); + + if (!user || !queryParams.is_admin) { + return Response(401, { error: 'Unauthorized' }); + } + + const photo = Photo.insert(Object.assign({}, params.photo, { user_id: user.id })); + + return { photo: Photo.serializer(photo) }; + }); + + this.get('/photos', ({ headers, queryParams }) => { + const user = User.findFromHeaderToken(headers); + + if (!user) { + return Response(404, { error: 'Not found' }); + } + + const photos = Photo.findAll(Object.assign({}, { user_id: user.id }, queryParams)); + + if (!photos || photos.length === 0) { + return Response(404, { error: 'Not found' }); + } + + return { photos: Photo.serializer(photos) }; + }); + + this.get('/photos/:id', ({ headers, params, queryParams }) => { + const user = User.findFromHeaderToken(headers); + + if (!user) { + return Response(401, { error: 'Unauthorized' }); + } else if (queryParams.nonce === 123123123) { + const photo = Photo.findBy({ id: params.id, user_id: user.id }); + + return photo ? { photo: Photo.serializer(photo) } : Response(404, { error: 'Not found' }); + } + + return Response(404, { error: 'Not found' }); + }); + + this.put('/photos/:id', ({ headers, params, queryParams }) => { + const user = User.findFromHeaderToken(headers); + const validRequest = user && queryParams.nonce === 123123123 && + Photo.findBy({ id: params.id, user_id: user.id }); + + if (validRequest) { + return { photo: Photo.serializer(Photo.update(params.photo)) }; + } + + return Response(500, { error: 'Unexpected error occured' }); + }); + + this.delete('/photos/:id', ({ headers, params, queryParams }) => { + const user = User.findFromHeaderToken(headers); + + if (!(queryParams.nonce === 123123123)) { + return Response(500, { error: 'Invalid nonce to delete a photo' }); + } else if (!user && !Photo.findBy({ id: params.id, user_id: user.id })) { + return Response(404, { error: 'Not found' }); + } + + Photo.delete({ id: params.id }); // NOTE: what to do with this response + }); + } + `); +} + +async function writeEdgeCasesServerFile() { + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/ethereum-account.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + }); + `), + fs.writeFile(`${CWD}/memserver/fixtures/ethereum-accounts.js`, `export default [ + { + id: 1, + address: '0x7be8315acfef37816c9ad4dc5e82195f2a52934c5d0c74883f9978675e26d600' + } + ];`), + fs.writeFile(`${CWD}/memserver/server.js`, ` + import Response from '../lib/response'; + + export default function({ EthereumAccount, Photo, PhotoComment }) { + this.get('/ethereum-accounts', ({ queryParams }) => { + const ethereumAccounts = EthereumAccount.findAll({ address: queryParams.address }); + + return { ethereum_accounts: EthereumAccount.serializer(ethereumAccounts) }; + }); + + this.get('/ethereum-accounts/:address', ({ params }) => { + const ethereumAccount = EthereumAccount.findBy({ address: params.address }); + + return { ethereum_account: EthereumAccount.serializer(ethereumAccount) }; + }); + + this.get('/photos', ({ queryParams }) => { + const photos = Photo.find(queryParams.ids || []); + + if (!photos || photos.length === 0) { + return Response(404, { error: 'Not found' }); + } + + return { photos: Photo.serializer(photos) }; + }); + + this.get('/photo-comments/:uuid', ({ params }) => { + const photoComment = PhotoComment.findBy({ uuid: params.uuid }); + + return { photo_comment: PhotoComment.serializer(photoComment) }; + }); + + this.get('/photo-comments', ({ queryParams }) => { + const photoComments = PhotoComment.findAll({ uuid: queryParams.uuid }); + + return { photo_comments: PhotoComment.serializer(photoComments) }; + }); + } + `) + ]); +} diff --git a/test/server/passthrough.js b/test/server/passthrough.js new file mode 100644 index 0000000..2a7e570 --- /dev/null +++ b/test/server/passthrough.js @@ -0,0 +1,207 @@ +import test from 'ava'; +import cors from 'cors'; +import express from 'express'; +import fs from 'fs-extra'; +import sinon from 'sinon'; + +let actualServer; + +const AUTHENTICATION_TOKEN = 'ec25fc7b-6ee2-4bda-b57c-6c9867b30ff4'; +const AJAX_AUTHORIZATION_HEADERS = { + 'Content-Type': 'application/json', 'Authorization': `Token ${AUTHENTICATION_TOKEN}` +}; +const CWD = process.cwd(); + +test.before(async () => { + await fs.mkdir(`${CWD}/memserver`); + await fs.mkdir(`${CWD}/memserver/models`); + await Promise.all([ + fs.writeFile(`${CWD}/memserver/models/photo.js`, ` + import Model from '${CWD}/lib/model'; + + export default Model({ + });`), + fs.mkdir(`${CWD}/memserver/fixtures`) + ]); + await fs.writeFile(`${CWD}/memserver/fixtures/photos.js`, `export default [ + { + id: 1, + name: 'Ski trip', + href: 'ski-trip.jpeg', + is_public: false, + user_id: 1 + }, + { + id: 2, + name: 'Family photo', + href: 'family-photo.jpeg', + is_public: true, + user_id: 1 + }, + { + id: 3, + name: 'Selfie', + href: 'selfie.jpeg', + is_public: false, + user_id: 1 + } + ];`); + + const app = express(); + + app.use(cors()); + + app.get('/films', (req, res) => { + res.json({ film: 'responsed correctly' }); + }); + + app.get('/movies/too-big-to-fail', (req, res) => { + res.json({ movie: 'is too-big-to-fail' }); + }); + + actualServer = app.listen(4000, () => console.log('Web server started on port 4000')); +}); + +test.beforeEach(async () => { + await fs.writeFile(`${CWD}/memserver/server.js`, ` + import Response from '../lib/response'; + + export default function({ Photo }) { + this.get('/photos', () => { + const photos = Photo.findAll(); + + if (!photos || photos.length === 0) { + return Response(404, { error: 'Not found' }); + } + + return { photos: Photo.serializer(photos) }; + }); + + this.passthrough('/films'); + this.passthrough('http://localhost:4000/films'); + this.passthrough('http://localhost:4000/movies/*'); + } + `); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}); + +test.after.always(async () => { + if (await fs.exists(`${CWD}/memserver`)) { + await fs.remove(`${CWD}/memserver`); + } + + actualServer.close(); +}); + +test.serial('[MemServer.Server] throws an error when MemServer tried to intercept an undeclared route', (t) => { + t.plan(1); + + const MemServer = require('../../lib/index.js'); + + MemServer.start(); + MemServer.Server.unhandledRequest = sinon.spy(); + window.$ = require('jquery'); + + window.$.ajax({ + type: 'GET', url: '/izelnakri', headers: { 'Content-Type': 'application/json' } + }); + + t.true(MemServer.Server.unhandledRequest.calledOnce, 'MemServer.Server.unhandledRequest called once'); +}); + +test.serial('[MemServer.Server] this.passthrough(url) shortcut works', async (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: 'http://localhost:4000/films', headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(jqXHR.responseJSON, { film: 'responsed correctly' }); + }); +}); + + +test.serial('[MemServer.Server] this.passthrough(url) shortcut works with wild cards', async (t) => { + t.plan(2); + + const MemServer = require('../../lib/index.js'); + + MemServer.start(); + window.$ = require('jquery'); + + await window.$.ajax({ + type: 'GET', url: 'http://localhost:4000/movies/too-big-to-fail', + headers: { 'Content-Type': 'application/json' } + }).then((data, textStatus, jqXHR) => { + t.is(jqXHR.status, 200); + t.deepEqual(jqXHR.responseJSON, { movie: 'is too-big-to-fail' }); + }); +}); + +// TODO: TEST BELOW ISNT WORKING: has beforeEach!! afterwards +// test.serial('[MemServer.Server] can create global passthrough via this.passthrough()', async (t) => { +// t.plan(6); +// +// Object.keys(require.cache).forEach((key) => delete require.cache[key]); +// +// await writeGlobalPassthroughServerFile(); +// +// const MemServer = require('../../lib/index.js'); +// const { Photo } = MemServer.Models; +// +// MemServer.start(); +// MemServer.Server.unhandledRequest = sinon.spy(); +// window.$ = require('jquery'); +// +// await window.$.ajax({ +// type: 'GET', url: '/photos', headers: { 'Content-Type': 'application/json' } +// }).then((data, textStatus, jqXHR) => { +// t.is(jqXHR.status, 200); +// t.deepEqual(jqXHR.responseJSON, { photos: Photo.serializer(Photo.findAll()) }); +// }); +// await window.$.ajax({ +// type: 'GET', url: 'http://localhost:4000/films', headers: { 'Content-Type': 'application/json' } +// }).then((data, textStatus, jqXHR) => { +// t.equal(jqXHR.status, 200); +// t.deepEqual(jqXHR.responseJSON, { film: 'responsed correctly' }); +// }); +// await window.$.ajax({ +// type: 'GET', url: 'http://localhost:4000/movies/too-big-to-fail', +// headers: { 'Content-Type': 'application/json' } +// }).then((data, textStatus, jqXHR) => { +// t.is(jqXHR.status, 200); +// t.deepEqual(jqXHR.responseJSON, { movie: 'is too-big-to-fail' }); +// }); +// }); + +// TODO: test this.passthrough('/something') when there is this.namespace; + +// NOTE: passthrough order? investigate + +async function writeGlobalPassthroughServerFile() { + await fs.writeFile(`${CWD}/memserver/server.js`, ` + import Response from '../lib/response'; + + export default function({ Photo }) { + this.get('/photos', () => { + const photos = Photo.findAll(); + + if (!photos || photos.length === 0) { + return Response(404, { error: 'Not found' }); + } + + return { photos: Photo.serializer(photos) }; + }); + + this.passthrough(); + } + `); + + Object.keys(require.cache).forEach((key) => delete require.cache[key]); +}