├── .editorconfig ├── .eslintrc.js ├── .gitattributes ├── .gitignore ├── README.md ├── assert-audio.js ├── golden-master.jpg ├── index.html ├── mocha-audio-snapshots.js ├── mocha-audio.js ├── package-lock.json ├── package.json ├── sadtrombone-test.js ├── sadtrombone.js ├── snapshots ├── Sad Trombone should fail so that we can see the player.wav ├── Sad Trombone should render sad trombone with a faster tempo.wav └── Sad Trombone should render sad trombone.wav └── test.html /.editorconfig: -------------------------------------------------------------------------------- 1 | # editorconfig.org 2 | 3 | root = true 4 | 5 | [*] 6 | charset = utf-8 7 | indent_size = 2 8 | indent_style = space 9 | insert_final_newline = true 10 | trim_trailing_whitespace = true 11 | 12 | [*.md] 13 | trim_trailing_whitespace = false 14 | -------------------------------------------------------------------------------- /.eslintrc.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | "env": { 3 | "browser": true, 4 | "es6": true, 5 | "mocha": true 6 | }, 7 | "extends": "eslint:recommended", 8 | "parserOptions": { 9 | "ecmaVersion": 2017, 10 | "sourceType": "module" 11 | }, 12 | "rules": { 13 | "no-console": 0, 14 | "indent": [ 15 | "error", 16 | 2 17 | ], 18 | "linebreak-style": [ 19 | "error", 20 | "unix" 21 | ], 22 | "quotes": [ 23 | "error", 24 | "single" 25 | ], 26 | "semi": [ 27 | "error", 28 | "never" 29 | ] 30 | } 31 | }; 32 | -------------------------------------------------------------------------------- /.gitattributes: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/salomvary/audio-snapshot-testing/6fffcd55edd5702e770084bc75aed9d8ac0f80f9/.gitattributes -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules 3 | npm-debug.log 4 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Audio Snapshot Testing 2 | 3 |  4 | 5 | [Image source: Wikipedia](https://commons.wikimedia.org/wiki/File:The_Sounds_of_Earth_-_GPN-2000-001976.jpg) 6 | 7 | ## What is snapshot testing? 8 | 9 | Snapshot tests compare the **output of a previous known to be good** version 10 | of a system with the **output of the current version** to protect against 11 | unintended changes. 12 | 13 | Also known as *golden master testing* or *characterization testing*. 14 | 15 | ## What can be snapshot tested? 16 | 17 | Anything :) 18 | 19 | - DOM tree rendered by a React component 20 | - Image generated on an HTML `
47 | 48 | 55 | 56 | 63 | 64 | 65 | -------------------------------------------------------------------------------- /mocha-audio-snapshots.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Audio snapshot loader hook for Moacha 3 | * 4 | * Usage: 5 | * beforeEach(fetchAudioSnapshots(new AudioContext())) 6 | * 7 | * The audio snapshot is fetched as WAV file under a URL generated from the test 8 | * name and exposed as an AudioBuffer at this.snapshot. 9 | * 10 | * @param {AudioContext} audioCtx 11 | * @returns {function} the beforeEach hook 12 | */ 13 | export function fetchAudioSnapshots (audioCtx) { 14 | return async function fetchAudioSnapshots () { 15 | // Load audio snapshot from an url derived from the test title 16 | const url = 'snapshots/' + this.currentTest.fullTitle() + '.wav' 17 | const arrayBuffer = await fetch(url, {credentials: 'same-origin'}) 18 | .then(response => { 19 | if (response.ok) { 20 | return response.arrayBuffer() 21 | } else { 22 | console.warn( 23 | `Audio snapshot was not found for "${this.currentTest.fullTitle()}". ` + 24 | `Make sure there is a WAV file at "${url}"`) 25 | } 26 | }) 27 | 28 | // Decode the snapshot if available or replace with an empty buffer 29 | const audioBuffer = arrayBuffer 30 | ? await decodeAudioData(arrayBuffer) 31 | : audioCtx.createBuffer(2, 2, 44100) 32 | 33 | this.snapshot = audioBuffer 34 | } 35 | 36 | // Safari does not have the promise based API for decodeAudioDataxw 37 | function decodeAudioData (arrayBuffer) { 38 | return new Promise (function (resolve, reject) { 39 | audioCtx.decodeAudioData(arrayBuffer, resolve, reject) 40 | }) 41 | } 42 | } 43 | -------------------------------------------------------------------------------- /mocha-audio.js: -------------------------------------------------------------------------------- 1 | /* global audioBufferToWav saveAs */ 2 | 3 | /** 4 | * Mocha HTML reporter extension to support audio snapshot teting 5 | * 6 | * Usage: 7 | * const runner = mocha.run() 8 | * mochaAudio(runner) 9 | * 10 | * @param {Runner} runner Mocha Runner instance 11 | */ 12 | export default function mochaAudio (runner) { 13 | const audioCtx = new (window.AudioContext || window.webkitAudioContext)() 14 | 15 | runner.on('fail', function onFail (test) { 16 | if (!(test.err.actual instanceof AudioBuffer && test.err.expected instanceof AudioBuffer)) { 17 | return 18 | } 19 | 20 | // The Mocha UI extension was inspired by 21 | // https://toucantoco.com/en/tech-blog/tech/visual-tdd 22 | 23 | try { 24 | // Add a wrapper element after the last test 25 | // (they are appended sequentially by Mocha) 26 | const testsReportsElements = document.getElementsByClassName('test') 27 | const testReport = testsReportsElements[testsReportsElements.length - 1] 28 | const wrapper = document.createElement('li') 29 | wrapper.className = 'mocha-audio-snapshots test' 30 | testReport.parentElement.appendChild(wrapper) 31 | 32 | // Append a player for the actual output 33 | addPlayer(wrapper, { 34 | title: 'Actual:', 35 | buffer: test.err.actual, 36 | fileName: test.fullTitle() + '.wav' 37 | }) 38 | 39 | // Append a player for the expected output 40 | addPlayer(wrapper, { 41 | title: 'Expected:', 42 | buffer: test.err.expected, 43 | fileName: test.fullTitle() + '.wav' 44 | }) 45 | } catch (e) { 46 | // Mocha swallows exceptions 47 | console.error('Error in audio snapshots reporter', e) 48 | } 49 | }) 50 | 51 | function addPlayer (parent, {title, buffer, fileName}) { 52 | const label = document.createElement('span') 53 | label.textContent = title 54 | parent.appendChild(label) 55 | 56 | const playButton = document.createElement('button') 57 | playButton.innerHTML = '▶' 58 | label.appendChild(playButton) 59 | 60 | const downloadButton = document.createElement('button') 61 | downloadButton.innerHTML = '⬇︎' 62 | label.appendChild(downloadButton) 63 | 64 | let source 65 | 66 | function onEnded () { 67 | playButton.innerHTML = '▶' 68 | source = null 69 | } 70 | 71 | playButton.onclick = function onPlayClick () { 72 | if (source) { 73 | source.stop() 74 | source = null 75 | playButton.innerHTML = '▶' 76 | } else { 77 | source = audioCtx.createBufferSource() 78 | source.buffer = buffer 79 | source.connect(audioCtx.destination) 80 | source.start() 81 | source.onended = onEnded 82 | playButton.innerHTML = '◾' 83 | } 84 | } 85 | 86 | downloadButton.onclick = function onDownloadClick () { 87 | exportWav(buffer, fileName) 88 | } 89 | } 90 | 91 | function exportWav (buffer, fileName) { 92 | const data = audioBufferToWav(buffer) 93 | const blob = new Blob( 94 | [new DataView(data)], { 95 | type: 'audio/wav' 96 | }) 97 | saveAs(blob, fileName) 98 | } 99 | } 100 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "audio-snapshot-testing", 3 | "version": "1.0.0", 4 | "lockfileVersion": 1, 5 | "requires": true, 6 | "dependencies": { 7 | "@babel/code-frame": { 8 | "version": "7.0.0", 9 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.0.0.tgz", 10 | "integrity": "sha512-OfC2uemaknXr87bdLUkWog7nYuliM9Ij5HUcajsVcMCpQrcLmtxRbVFTIqmcSkSeYRBFBRxs2FiUqFJDLdiebA==", 11 | "dev": true, 12 | "requires": { 13 | "@babel/highlight": "^7.0.0" 14 | } 15 | }, 16 | "@babel/highlight": { 17 | "version": "7.0.0", 18 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.0.0.tgz", 19 | "integrity": "sha512-UFMC4ZeFC48Tpvj7C8UgLvtkaUuovQX+5xNWrsIoMG8o2z+XFKjKaN9iVmS84dPwVN00W4wPmqvYoZF3EGAsfw==", 20 | "dev": true, 21 | "requires": { 22 | "chalk": "^2.0.0", 23 | "esutils": "^2.0.2", 24 | "js-tokens": "^4.0.0" 25 | } 26 | }, 27 | "acorn": { 28 | "version": "6.0.4", 29 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.0.4.tgz", 30 | "integrity": "sha512-VY4i5EKSKkofY2I+6QLTbTTN/UvEQPCo6eiwzzSaSWfpaDhOmStMCMod6wmuPciNq+XS0faCglFu2lHZpdHUtg==", 31 | "dev": true 32 | }, 33 | "acorn-jsx": { 34 | "version": "5.0.1", 35 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.0.1.tgz", 36 | "integrity": "sha512-HJ7CfNHrfJLlNTzIEUTj43LNWGkqpRLxm3YjAlcD0ACydk9XynzYsCBHxut+iqt+1aBXkx9UP/w/ZqMr13XIzg==", 37 | "dev": true 38 | }, 39 | "ajv": { 40 | "version": "6.6.1", 41 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.6.1.tgz", 42 | "integrity": "sha512-ZoJjft5B+EJBjUyu9C9Hc0OZyPZSSlOF+plzouTrg6UlA8f+e/n8NIgBFG/9tppJtpPWfthHakK7juJdNDODww==", 43 | "dev": true, 44 | "requires": { 45 | "fast-deep-equal": "^2.0.1", 46 | "fast-json-stable-stringify": "^2.0.0", 47 | "json-schema-traverse": "^0.4.1", 48 | "uri-js": "^4.2.2" 49 | } 50 | }, 51 | "ansi-escapes": { 52 | "version": "3.1.0", 53 | "resolved": "http://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.1.0.tgz", 54 | "integrity": "sha512-UgAb8H9D41AQnu/PbWlCofQVcnV4Gs2bBJi9eZPxfU/hgglFh3SMDMENRIqdr7H6XFnXdoknctFByVsCOotTVw==", 55 | "dev": true 56 | }, 57 | "ansi-regex": { 58 | "version": "3.0.0", 59 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", 60 | "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", 61 | "dev": true 62 | }, 63 | "ansi-styles": { 64 | "version": "3.2.1", 65 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 66 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 67 | "dev": true, 68 | "requires": { 69 | "color-convert": "^1.9.0" 70 | } 71 | }, 72 | "argparse": { 73 | "version": "1.0.10", 74 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", 75 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", 76 | "dev": true, 77 | "requires": { 78 | "sprintf-js": "~1.0.2" 79 | } 80 | }, 81 | "astral-regex": { 82 | "version": "1.0.0", 83 | "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", 84 | "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", 85 | "dev": true 86 | }, 87 | "balanced-match": { 88 | "version": "1.0.0", 89 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", 90 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", 91 | "dev": true 92 | }, 93 | "brace-expansion": { 94 | "version": "1.1.11", 95 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 96 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 97 | "dev": true, 98 | "requires": { 99 | "balanced-match": "^1.0.0", 100 | "concat-map": "0.0.1" 101 | } 102 | }, 103 | "caller-path": { 104 | "version": "0.1.0", 105 | "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-0.1.0.tgz", 106 | "integrity": "sha1-lAhe9jWB7NPaqSREqP6U6CV3dR8=", 107 | "dev": true, 108 | "requires": { 109 | "callsites": "^0.2.0" 110 | } 111 | }, 112 | "callsites": { 113 | "version": "0.2.0", 114 | "resolved": "http://registry.npmjs.org/callsites/-/callsites-0.2.0.tgz", 115 | "integrity": "sha1-r6uWJikQp/M8GaV3WCXGnzTjUMo=", 116 | "dev": true 117 | }, 118 | "chalk": { 119 | "version": "2.4.1", 120 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.1.tgz", 121 | "integrity": "sha512-ObN6h1v2fTJSmUXoS3nMQ92LbDK9be4TV+6G+omQlGJFdcUX5heKi1LZ1YnRMIgwTLEj3E24bT6tYni50rlCfQ==", 122 | "dev": true, 123 | "requires": { 124 | "ansi-styles": "^3.2.1", 125 | "escape-string-regexp": "^1.0.5", 126 | "supports-color": "^5.3.0" 127 | } 128 | }, 129 | "chardet": { 130 | "version": "0.7.0", 131 | "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", 132 | "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", 133 | "dev": true 134 | }, 135 | "circular-json": { 136 | "version": "0.3.3", 137 | "resolved": "https://registry.npmjs.org/circular-json/-/circular-json-0.3.3.tgz", 138 | "integrity": "sha512-UZK3NBx2Mca+b5LsG7bY183pHWt5Y1xts4P3Pz7ENTwGVnJOUWbRb3ocjvX7hx9tq/yTAdclXm9sZ38gNuem4A==", 139 | "dev": true 140 | }, 141 | "cli-cursor": { 142 | "version": "2.1.0", 143 | "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", 144 | "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", 145 | "dev": true, 146 | "requires": { 147 | "restore-cursor": "^2.0.0" 148 | } 149 | }, 150 | "cli-width": { 151 | "version": "2.2.0", 152 | "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.0.tgz", 153 | "integrity": "sha1-/xnt6Kml5XkyQUewwR8PvLq+1jk=", 154 | "dev": true 155 | }, 156 | "color-convert": { 157 | "version": "1.9.3", 158 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 159 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 160 | "dev": true, 161 | "requires": { 162 | "color-name": "1.1.3" 163 | } 164 | }, 165 | "color-name": { 166 | "version": "1.1.3", 167 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 168 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", 169 | "dev": true 170 | }, 171 | "concat-map": { 172 | "version": "0.0.1", 173 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 174 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", 175 | "dev": true 176 | }, 177 | "cross-spawn": { 178 | "version": "6.0.5", 179 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", 180 | "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", 181 | "dev": true, 182 | "requires": { 183 | "nice-try": "^1.0.4", 184 | "path-key": "^2.0.1", 185 | "semver": "^5.5.0", 186 | "shebang-command": "^1.2.0", 187 | "which": "^1.2.9" 188 | } 189 | }, 190 | "debug": { 191 | "version": "4.1.0", 192 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.0.tgz", 193 | "integrity": "sha512-heNPJUJIqC+xB6ayLAMHaIrmN9HKa7aQO8MGqKpvCA+uJYVcvR6l5kgdrhRuwPFHU7P5/A1w0BjByPHwpfTDKg==", 194 | "dev": true, 195 | "requires": { 196 | "ms": "^2.1.1" 197 | } 198 | }, 199 | "deep-is": { 200 | "version": "0.1.3", 201 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", 202 | "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", 203 | "dev": true 204 | }, 205 | "doctrine": { 206 | "version": "2.1.0", 207 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", 208 | "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", 209 | "dev": true, 210 | "requires": { 211 | "esutils": "^2.0.2" 212 | } 213 | }, 214 | "escape-string-regexp": { 215 | "version": "1.0.5", 216 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 217 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", 218 | "dev": true 219 | }, 220 | "eslint": { 221 | "version": "5.9.0", 222 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-5.9.0.tgz", 223 | "integrity": "sha512-g4KWpPdqN0nth+goDNICNXGfJF7nNnepthp46CAlJoJtC5K/cLu3NgCM3AHu1CkJ5Hzt9V0Y0PBAO6Ay/gGb+w==", 224 | "dev": true, 225 | "requires": { 226 | "@babel/code-frame": "^7.0.0", 227 | "ajv": "^6.5.3", 228 | "chalk": "^2.1.0", 229 | "cross-spawn": "^6.0.5", 230 | "debug": "^4.0.1", 231 | "doctrine": "^2.1.0", 232 | "eslint-scope": "^4.0.0", 233 | "eslint-utils": "^1.3.1", 234 | "eslint-visitor-keys": "^1.0.0", 235 | "espree": "^4.0.0", 236 | "esquery": "^1.0.1", 237 | "esutils": "^2.0.2", 238 | "file-entry-cache": "^2.0.0", 239 | "functional-red-black-tree": "^1.0.1", 240 | "glob": "^7.1.2", 241 | "globals": "^11.7.0", 242 | "ignore": "^4.0.6", 243 | "imurmurhash": "^0.1.4", 244 | "inquirer": "^6.1.0", 245 | "is-resolvable": "^1.1.0", 246 | "js-yaml": "^3.12.0", 247 | "json-stable-stringify-without-jsonify": "^1.0.1", 248 | "levn": "^0.3.0", 249 | "lodash": "^4.17.5", 250 | "minimatch": "^3.0.4", 251 | "mkdirp": "^0.5.1", 252 | "natural-compare": "^1.4.0", 253 | "optionator": "^0.8.2", 254 | "path-is-inside": "^1.0.2", 255 | "pluralize": "^7.0.0", 256 | "progress": "^2.0.0", 257 | "regexpp": "^2.0.1", 258 | "require-uncached": "^1.0.3", 259 | "semver": "^5.5.1", 260 | "strip-ansi": "^4.0.0", 261 | "strip-json-comments": "^2.0.1", 262 | "table": "^5.0.2", 263 | "text-table": "^0.2.0" 264 | } 265 | }, 266 | "eslint-scope": { 267 | "version": "4.0.0", 268 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-4.0.0.tgz", 269 | "integrity": "sha512-1G6UTDi7Jc1ELFwnR58HV4fK9OQK4S6N985f166xqXxpjU6plxFISJa2Ba9KCQuFa8RCnj/lSFJbHo7UFDBnUA==", 270 | "dev": true, 271 | "requires": { 272 | "esrecurse": "^4.1.0", 273 | "estraverse": "^4.1.1" 274 | } 275 | }, 276 | "eslint-utils": { 277 | "version": "1.3.1", 278 | "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.3.1.tgz", 279 | "integrity": "sha512-Z7YjnIldX+2XMcjr7ZkgEsOj/bREONV60qYeB/bjMAqqqZ4zxKyWX+BOUkdmRmA9riiIPVvo5x86m5elviOk0Q==", 280 | "dev": true 281 | }, 282 | "eslint-visitor-keys": { 283 | "version": "1.0.0", 284 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz", 285 | "integrity": "sha512-qzm/XxIbxm/FHyH341ZrbnMUpe+5Bocte9xkmFMzPMjRaZMcXww+MpBptFvtU+79L362nqiLhekCxCxDPaUMBQ==", 286 | "dev": true 287 | }, 288 | "espree": { 289 | "version": "4.1.0", 290 | "resolved": "https://registry.npmjs.org/espree/-/espree-4.1.0.tgz", 291 | "integrity": "sha512-I5BycZW6FCVIub93TeVY1s7vjhP9CY6cXCznIRfiig7nRviKZYdRnj/sHEWC6A7WE9RDWOFq9+7OsWSYz8qv2w==", 292 | "dev": true, 293 | "requires": { 294 | "acorn": "^6.0.2", 295 | "acorn-jsx": "^5.0.0", 296 | "eslint-visitor-keys": "^1.0.0" 297 | } 298 | }, 299 | "esprima": { 300 | "version": "4.0.1", 301 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", 302 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", 303 | "dev": true 304 | }, 305 | "esquery": { 306 | "version": "1.0.1", 307 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.0.1.tgz", 308 | "integrity": "sha512-SmiyZ5zIWH9VM+SRUReLS5Q8a7GxtRdxEBVZpm98rJM7Sb+A9DVCndXfkeFUd3byderg+EbDkfnevfCwynWaNA==", 309 | "dev": true, 310 | "requires": { 311 | "estraverse": "^4.0.0" 312 | } 313 | }, 314 | "esrecurse": { 315 | "version": "4.2.1", 316 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.2.1.tgz", 317 | "integrity": "sha512-64RBB++fIOAXPw3P9cy89qfMlvZEXZkqqJkjqqXIvzP5ezRZjW+lPWjw35UX/3EhUPFYbg5ER4JYgDw4007/DQ==", 318 | "dev": true, 319 | "requires": { 320 | "estraverse": "^4.1.0" 321 | } 322 | }, 323 | "estraverse": { 324 | "version": "4.2.0", 325 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.2.0.tgz", 326 | "integrity": "sha1-De4/7TH81GlhjOc0IJn8GvoL2xM=", 327 | "dev": true 328 | }, 329 | "esutils": { 330 | "version": "2.0.2", 331 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz", 332 | "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=", 333 | "dev": true 334 | }, 335 | "external-editor": { 336 | "version": "3.0.3", 337 | "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.0.3.tgz", 338 | "integrity": "sha512-bn71H9+qWoOQKyZDo25mOMVpSmXROAsTJVVVYzrrtol3d4y+AsKjf4Iwl2Q+IuT0kFSQ1qo166UuIwqYq7mGnA==", 339 | "dev": true, 340 | "requires": { 341 | "chardet": "^0.7.0", 342 | "iconv-lite": "^0.4.24", 343 | "tmp": "^0.0.33" 344 | } 345 | }, 346 | "fast-deep-equal": { 347 | "version": "2.0.1", 348 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz", 349 | "integrity": "sha1-ewUhjd+WZ79/Nwv3/bLLFf3Qqkk=", 350 | "dev": true 351 | }, 352 | "fast-json-stable-stringify": { 353 | "version": "2.0.0", 354 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", 355 | "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=", 356 | "dev": true 357 | }, 358 | "fast-levenshtein": { 359 | "version": "2.0.6", 360 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", 361 | "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", 362 | "dev": true 363 | }, 364 | "figures": { 365 | "version": "2.0.0", 366 | "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", 367 | "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", 368 | "dev": true, 369 | "requires": { 370 | "escape-string-regexp": "^1.0.5" 371 | } 372 | }, 373 | "file-entry-cache": { 374 | "version": "2.0.0", 375 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-2.0.0.tgz", 376 | "integrity": "sha1-w5KZDD5oR4PYOLjISkXYoEhFg2E=", 377 | "dev": true, 378 | "requires": { 379 | "flat-cache": "^1.2.1", 380 | "object-assign": "^4.0.1" 381 | } 382 | }, 383 | "flat-cache": { 384 | "version": "1.3.4", 385 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-1.3.4.tgz", 386 | "integrity": "sha512-VwyB3Lkgacfik2vhqR4uv2rvebqmDvFu4jlN/C1RzWoJEo8I7z4Q404oiqYCkq41mni8EzQnm95emU9seckwtg==", 387 | "dev": true, 388 | "requires": { 389 | "circular-json": "^0.3.1", 390 | "graceful-fs": "^4.1.2", 391 | "rimraf": "~2.6.2", 392 | "write": "^0.2.1" 393 | } 394 | }, 395 | "fs.realpath": { 396 | "version": "1.0.0", 397 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", 398 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", 399 | "dev": true 400 | }, 401 | "functional-red-black-tree": { 402 | "version": "1.0.1", 403 | "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", 404 | "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", 405 | "dev": true 406 | }, 407 | "glob": { 408 | "version": "7.1.3", 409 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", 410 | "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", 411 | "dev": true, 412 | "requires": { 413 | "fs.realpath": "^1.0.0", 414 | "inflight": "^1.0.4", 415 | "inherits": "2", 416 | "minimatch": "^3.0.4", 417 | "once": "^1.3.0", 418 | "path-is-absolute": "^1.0.0" 419 | } 420 | }, 421 | "globals": { 422 | "version": "11.9.0", 423 | "resolved": "https://registry.npmjs.org/globals/-/globals-11.9.0.tgz", 424 | "integrity": "sha512-5cJVtyXWH8PiJPVLZzzoIizXx944O4OmRro5MWKx5fT4MgcN7OfaMutPeaTdJCCURwbWdhhcCWcKIffPnmTzBg==", 425 | "dev": true 426 | }, 427 | "graceful-fs": { 428 | "version": "4.1.15", 429 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.15.tgz", 430 | "integrity": "sha512-6uHUhOPEBgQ24HM+r6b/QwWfZq+yiFcipKFrOFiBEnWdy5sdzYoi+pJeQaPI5qOLRFqWmAXUPQNsielzdLoecA==", 431 | "dev": true 432 | }, 433 | "has-flag": { 434 | "version": "3.0.0", 435 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 436 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", 437 | "dev": true 438 | }, 439 | "iconv-lite": { 440 | "version": "0.4.24", 441 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", 442 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", 443 | "dev": true, 444 | "requires": { 445 | "safer-buffer": ">= 2.1.2 < 3" 446 | } 447 | }, 448 | "ignore": { 449 | "version": "4.0.6", 450 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", 451 | "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", 452 | "dev": true 453 | }, 454 | "imurmurhash": { 455 | "version": "0.1.4", 456 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", 457 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", 458 | "dev": true 459 | }, 460 | "inflight": { 461 | "version": "1.0.6", 462 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", 463 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", 464 | "dev": true, 465 | "requires": { 466 | "once": "^1.3.0", 467 | "wrappy": "1" 468 | } 469 | }, 470 | "inherits": { 471 | "version": "2.0.3", 472 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", 473 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", 474 | "dev": true 475 | }, 476 | "inquirer": { 477 | "version": "6.2.1", 478 | "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-6.2.1.tgz", 479 | "integrity": "sha512-088kl3DRT2dLU5riVMKKr1DlImd6X7smDhpXUCkJDCKvTEJeRiXh0G132HG9u5a+6Ylw9plFRY7RuTnwohYSpg==", 480 | "dev": true, 481 | "requires": { 482 | "ansi-escapes": "^3.0.0", 483 | "chalk": "^2.0.0", 484 | "cli-cursor": "^2.1.0", 485 | "cli-width": "^2.0.0", 486 | "external-editor": "^3.0.0", 487 | "figures": "^2.0.0", 488 | "lodash": "^4.17.10", 489 | "mute-stream": "0.0.7", 490 | "run-async": "^2.2.0", 491 | "rxjs": "^6.1.0", 492 | "string-width": "^2.1.0", 493 | "strip-ansi": "^5.0.0", 494 | "through": "^2.3.6" 495 | }, 496 | "dependencies": { 497 | "ansi-regex": { 498 | "version": "4.0.0", 499 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.0.0.tgz", 500 | "integrity": "sha512-iB5Dda8t/UqpPI/IjsejXu5jOGDrzn41wJyljwPH65VCIbk6+1BzFIMJGFwTNrYXT1CrD+B4l19U7awiQ8rk7w==", 501 | "dev": true 502 | }, 503 | "strip-ansi": { 504 | "version": "5.0.0", 505 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.0.0.tgz", 506 | "integrity": "sha512-Uu7gQyZI7J7gn5qLn1Np3G9vcYGTVqB+lFTytnDJv83dd8T22aGH451P3jueT2/QemInJDfxHB5Tde5OzgG1Ow==", 507 | "dev": true, 508 | "requires": { 509 | "ansi-regex": "^4.0.0" 510 | } 511 | } 512 | } 513 | }, 514 | "is-fullwidth-code-point": { 515 | "version": "2.0.0", 516 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", 517 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", 518 | "dev": true 519 | }, 520 | "is-promise": { 521 | "version": "2.1.0", 522 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz", 523 | "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=", 524 | "dev": true 525 | }, 526 | "is-resolvable": { 527 | "version": "1.1.0", 528 | "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.1.0.tgz", 529 | "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==", 530 | "dev": true 531 | }, 532 | "isexe": { 533 | "version": "2.0.0", 534 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 535 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", 536 | "dev": true 537 | }, 538 | "js-tokens": { 539 | "version": "4.0.0", 540 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 541 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 542 | "dev": true 543 | }, 544 | "js-yaml": { 545 | "version": "3.12.0", 546 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.12.0.tgz", 547 | "integrity": "sha512-PIt2cnwmPfL4hKNwqeiuz4bKfnzHTBv6HyVgjahA6mPLwPDzjDWrplJBMjHUFxku/N3FlmrbyPclad+I+4mJ3A==", 548 | "dev": true, 549 | "requires": { 550 | "argparse": "^1.0.7", 551 | "esprima": "^4.0.0" 552 | } 553 | }, 554 | "json-schema-traverse": { 555 | "version": "0.4.1", 556 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", 557 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", 558 | "dev": true 559 | }, 560 | "json-stable-stringify-without-jsonify": { 561 | "version": "1.0.1", 562 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", 563 | "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", 564 | "dev": true 565 | }, 566 | "levn": { 567 | "version": "0.3.0", 568 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", 569 | "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", 570 | "dev": true, 571 | "requires": { 572 | "prelude-ls": "~1.1.2", 573 | "type-check": "~0.3.2" 574 | } 575 | }, 576 | "lodash": { 577 | "version": "4.17.11", 578 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.11.tgz", 579 | "integrity": "sha512-cQKh8igo5QUhZ7lg38DYWAxMvjSAKG0A8wGSVimP07SIUEK2UO+arSRKbRZWtelMtN5V0Hkwh5ryOto/SshYIg==", 580 | "dev": true 581 | }, 582 | "mimic-fn": { 583 | "version": "1.2.0", 584 | "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", 585 | "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", 586 | "dev": true 587 | }, 588 | "minimatch": { 589 | "version": "3.0.4", 590 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", 591 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", 592 | "dev": true, 593 | "requires": { 594 | "brace-expansion": "^1.1.7" 595 | } 596 | }, 597 | "minimist": { 598 | "version": "0.0.8", 599 | "resolved": "http://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", 600 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", 601 | "dev": true 602 | }, 603 | "mkdirp": { 604 | "version": "0.5.1", 605 | "resolved": "http://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", 606 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", 607 | "dev": true, 608 | "requires": { 609 | "minimist": "0.0.8" 610 | } 611 | }, 612 | "ms": { 613 | "version": "2.1.1", 614 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", 615 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", 616 | "dev": true 617 | }, 618 | "mute-stream": { 619 | "version": "0.0.7", 620 | "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", 621 | "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=", 622 | "dev": true 623 | }, 624 | "natural-compare": { 625 | "version": "1.4.0", 626 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", 627 | "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", 628 | "dev": true 629 | }, 630 | "nice-try": { 631 | "version": "1.0.5", 632 | "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", 633 | "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", 634 | "dev": true 635 | }, 636 | "object-assign": { 637 | "version": "4.1.1", 638 | "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", 639 | "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", 640 | "dev": true 641 | }, 642 | "once": { 643 | "version": "1.4.0", 644 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", 645 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", 646 | "dev": true, 647 | "requires": { 648 | "wrappy": "1" 649 | } 650 | }, 651 | "onetime": { 652 | "version": "2.0.1", 653 | "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", 654 | "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", 655 | "dev": true, 656 | "requires": { 657 | "mimic-fn": "^1.0.0" 658 | } 659 | }, 660 | "optionator": { 661 | "version": "0.8.2", 662 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.2.tgz", 663 | "integrity": "sha1-NkxeQJ0/TWMB1sC0wFu6UBgK62Q=", 664 | "dev": true, 665 | "requires": { 666 | "deep-is": "~0.1.3", 667 | "fast-levenshtein": "~2.0.4", 668 | "levn": "~0.3.0", 669 | "prelude-ls": "~1.1.2", 670 | "type-check": "~0.3.2", 671 | "wordwrap": "~1.0.0" 672 | } 673 | }, 674 | "os-tmpdir": { 675 | "version": "1.0.2", 676 | "resolved": "http://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", 677 | "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", 678 | "dev": true 679 | }, 680 | "path-is-absolute": { 681 | "version": "1.0.1", 682 | "resolved": "http://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", 683 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", 684 | "dev": true 685 | }, 686 | "path-is-inside": { 687 | "version": "1.0.2", 688 | "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", 689 | "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=", 690 | "dev": true 691 | }, 692 | "path-key": { 693 | "version": "2.0.1", 694 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", 695 | "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", 696 | "dev": true 697 | }, 698 | "pluralize": { 699 | "version": "7.0.0", 700 | "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-7.0.0.tgz", 701 | "integrity": "sha512-ARhBOdzS3e41FbkW/XWrTEtukqqLoK5+Z/4UeDaLuSW+39JPeFgs4gCGqsrJHVZX0fUrx//4OF0K1CUGwlIFow==", 702 | "dev": true 703 | }, 704 | "prelude-ls": { 705 | "version": "1.1.2", 706 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", 707 | "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", 708 | "dev": true 709 | }, 710 | "progress": { 711 | "version": "2.0.2", 712 | "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.2.tgz", 713 | "integrity": "sha512-/OLz5F9beZUWwSHZDreXgap1XShX6W+DCHQCqwCF7uZ88s6uTlD2cR3JBE77SegCmNtb1Idst+NfmwcdU6KVhw==", 714 | "dev": true 715 | }, 716 | "punycode": { 717 | "version": "2.1.1", 718 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", 719 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", 720 | "dev": true 721 | }, 722 | "regexpp": { 723 | "version": "2.0.1", 724 | "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz", 725 | "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==", 726 | "dev": true 727 | }, 728 | "require-uncached": { 729 | "version": "1.0.3", 730 | "resolved": "http://registry.npmjs.org/require-uncached/-/require-uncached-1.0.3.tgz", 731 | "integrity": "sha1-Tg1W1slmL9MeQwEcS5WqSZVUIdM=", 732 | "dev": true, 733 | "requires": { 734 | "caller-path": "^0.1.0", 735 | "resolve-from": "^1.0.0" 736 | } 737 | }, 738 | "resolve-from": { 739 | "version": "1.0.1", 740 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-1.0.1.tgz", 741 | "integrity": "sha1-Jsv+k10a7uq7Kbw/5a6wHpPUQiY=", 742 | "dev": true 743 | }, 744 | "restore-cursor": { 745 | "version": "2.0.0", 746 | "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", 747 | "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", 748 | "dev": true, 749 | "requires": { 750 | "onetime": "^2.0.0", 751 | "signal-exit": "^3.0.2" 752 | } 753 | }, 754 | "rimraf": { 755 | "version": "2.6.2", 756 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.2.tgz", 757 | "integrity": "sha512-lreewLK/BlghmxtfH36YYVg1i8IAce4TI7oao75I1g245+6BctqTVQiBP3YUJ9C6DQOXJmkYR9X9fCLtCOJc5w==", 758 | "dev": true, 759 | "requires": { 760 | "glob": "^7.0.5" 761 | } 762 | }, 763 | "run-async": { 764 | "version": "2.3.0", 765 | "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.3.0.tgz", 766 | "integrity": "sha1-A3GrSuC91yDUFm19/aZP96RFpsA=", 767 | "dev": true, 768 | "requires": { 769 | "is-promise": "^2.1.0" 770 | } 771 | }, 772 | "rxjs": { 773 | "version": "6.3.3", 774 | "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.3.3.tgz", 775 | "integrity": "sha512-JTWmoY9tWCs7zvIk/CvRjhjGaOd+OVBM987mxFo+OW66cGpdKjZcpmc74ES1sB//7Kl/PAe8+wEakuhG4pcgOw==", 776 | "dev": true, 777 | "requires": { 778 | "tslib": "^1.9.0" 779 | } 780 | }, 781 | "safer-buffer": { 782 | "version": "2.1.2", 783 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", 784 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", 785 | "dev": true 786 | }, 787 | "semver": { 788 | "version": "5.6.0", 789 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.6.0.tgz", 790 | "integrity": "sha512-RS9R6R35NYgQn++fkDWaOmqGoj4Ek9gGs+DPxNUZKuwE183xjJroKvyo1IzVFeXvUrvmALy6FWD5xrdJT25gMg==", 791 | "dev": true 792 | }, 793 | "shebang-command": { 794 | "version": "1.2.0", 795 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", 796 | "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", 797 | "dev": true, 798 | "requires": { 799 | "shebang-regex": "^1.0.0" 800 | } 801 | }, 802 | "shebang-regex": { 803 | "version": "1.0.0", 804 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", 805 | "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", 806 | "dev": true 807 | }, 808 | "signal-exit": { 809 | "version": "3.0.2", 810 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", 811 | "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", 812 | "dev": true 813 | }, 814 | "slice-ansi": { 815 | "version": "2.0.0", 816 | "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-2.0.0.tgz", 817 | "integrity": "sha512-4j2WTWjp3GsZ+AOagyzVbzp4vWGtZ0hEZ/gDY/uTvm6MTxUfTUIsnMIFb1bn8o0RuXiqUw15H1bue8f22Vw2oQ==", 818 | "dev": true, 819 | "requires": { 820 | "ansi-styles": "^3.2.0", 821 | "astral-regex": "^1.0.0", 822 | "is-fullwidth-code-point": "^2.0.0" 823 | } 824 | }, 825 | "sprintf-js": { 826 | "version": "1.0.3", 827 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", 828 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", 829 | "dev": true 830 | }, 831 | "string-width": { 832 | "version": "2.1.1", 833 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", 834 | "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", 835 | "dev": true, 836 | "requires": { 837 | "is-fullwidth-code-point": "^2.0.0", 838 | "strip-ansi": "^4.0.0" 839 | } 840 | }, 841 | "strip-ansi": { 842 | "version": "4.0.0", 843 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", 844 | "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", 845 | "dev": true, 846 | "requires": { 847 | "ansi-regex": "^3.0.0" 848 | } 849 | }, 850 | "strip-json-comments": { 851 | "version": "2.0.1", 852 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", 853 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", 854 | "dev": true 855 | }, 856 | "supports-color": { 857 | "version": "5.5.0", 858 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 859 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 860 | "dev": true, 861 | "requires": { 862 | "has-flag": "^3.0.0" 863 | } 864 | }, 865 | "table": { 866 | "version": "5.1.1", 867 | "resolved": "https://registry.npmjs.org/table/-/table-5.1.1.tgz", 868 | "integrity": "sha512-NUjapYb/qd4PeFW03HnAuOJ7OMcBkJlqeClWxeNlQ0lXGSb52oZXGzkO0/I0ARegQ2eUT1g2VDJH0eUxDRcHmw==", 869 | "dev": true, 870 | "requires": { 871 | "ajv": "^6.6.1", 872 | "lodash": "^4.17.11", 873 | "slice-ansi": "2.0.0", 874 | "string-width": "^2.1.1" 875 | } 876 | }, 877 | "text-table": { 878 | "version": "0.2.0", 879 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", 880 | "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", 881 | "dev": true 882 | }, 883 | "through": { 884 | "version": "2.3.8", 885 | "resolved": "http://registry.npmjs.org/through/-/through-2.3.8.tgz", 886 | "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", 887 | "dev": true 888 | }, 889 | "tmp": { 890 | "version": "0.0.33", 891 | "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", 892 | "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", 893 | "dev": true, 894 | "requires": { 895 | "os-tmpdir": "~1.0.2" 896 | } 897 | }, 898 | "tslib": { 899 | "version": "1.9.3", 900 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.9.3.tgz", 901 | "integrity": "sha512-4krF8scpejhaOgqzBEcGM7yDIEfi0/8+8zDRZhNZZ2kjmHJ4hv3zCbQWxoJGz1iw5U0Jl0nma13xzHXcncMavQ==", 902 | "dev": true 903 | }, 904 | "type-check": { 905 | "version": "0.3.2", 906 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", 907 | "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", 908 | "dev": true, 909 | "requires": { 910 | "prelude-ls": "~1.1.2" 911 | } 912 | }, 913 | "uri-js": { 914 | "version": "4.2.2", 915 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz", 916 | "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==", 917 | "dev": true, 918 | "requires": { 919 | "punycode": "^2.1.0" 920 | } 921 | }, 922 | "which": { 923 | "version": "1.3.1", 924 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", 925 | "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", 926 | "dev": true, 927 | "requires": { 928 | "isexe": "^2.0.0" 929 | } 930 | }, 931 | "wordwrap": { 932 | "version": "1.0.0", 933 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", 934 | "integrity": "sha1-J1hIEIkUVqQXHI0CJkQa3pDLyus=", 935 | "dev": true 936 | }, 937 | "wrappy": { 938 | "version": "1.0.2", 939 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", 940 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", 941 | "dev": true 942 | }, 943 | "write": { 944 | "version": "0.2.1", 945 | "resolved": "https://registry.npmjs.org/write/-/write-0.2.1.tgz", 946 | "integrity": "sha1-X8A4KOJkzqP+kUVUdvejxWbLB1c=", 947 | "dev": true, 948 | "requires": { 949 | "mkdirp": "^0.5.1" 950 | } 951 | } 952 | } 953 | } 954 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "audio-snapshot-testing", 3 | "version": "1.0.0", 4 | "description": "", 5 | "main": "index.js", 6 | "scripts": { 7 | "test": "echo \"Error: no test specified\" && exit 1" 8 | }, 9 | "keywords": [], 10 | "author": "", 11 | "license": "MIT", 12 | "devDependencies": { 13 | "eslint": "^5.9.0" 14 | } 15 | } 16 | -------------------------------------------------------------------------------- /sadtrombone-test.js: -------------------------------------------------------------------------------- 1 | import sadtrombone from './sadtrombone.js' 2 | import {fetchAudioSnapshots} from './mocha-audio-snapshots.js' 3 | 4 | const OfflineCtx = window.OfflineAudioContext || window.webkitOfflineAudioContext 5 | const audioCtx = new (window.AudioContext || window.webkitAudioContext)() 6 | 7 | describe('Sad Trombone', function () { 8 | beforeEach(fetchAudioSnapshots(audioCtx)) 9 | 10 | it('should render sad trombone', async function () { 11 | const rendered = await render(1.8, (offlineCtx) => { 12 | sadtrombone(offlineCtx) 13 | }) 14 | 15 | rendered.should.be.eqlAudio(this.snapshot) 16 | }) 17 | 18 | it('should render sad trombone with a faster tempo', async function () { 19 | const rendered = await render(0.6, (offlineCtx) => { 20 | const tempo = 0.15 21 | sadtrombone(offlineCtx, tempo) 22 | }) 23 | 24 | rendered.should.be.eqlAudio(this.snapshot) 25 | }) 26 | 27 | it('should fail so that we can see the player', async function () { 28 | const rendered = await render(1.8, (offlineCtx) => { 29 | sadtrombone(offlineCtx) 30 | }) 31 | 32 | rendered.should.be.eqlAudio(this.snapshot) 33 | }) 34 | }) 35 | 36 | /** 37 | * Render audio output to an "offline buffer" 38 | * 39 | * @param {number} length the length of the buffer to capture in seconds 40 | * @param {function} fn 41 | * @returns {AudioBuffer} the rendered buffer 42 | */ 43 | async function render (length, fn) { 44 | const offlineCtx = new OfflineCtx(2, length * 44100, 44100) 45 | fn(offlineCtx) 46 | offlineCtx.startRendering() 47 | // Safari does not have the promise based API for startRendering 48 | return new Promise(function (resolve) { 49 | offlineCtx.oncomplete = (e) => { 50 | resolve(e.renderedBuffer) 51 | } 52 | }) 53 | } 54 | -------------------------------------------------------------------------------- /sadtrombone.js: -------------------------------------------------------------------------------- 1 | /** 2 | * A simple synth that plays a short but sad tune :( 3 | * 4 | * @param {AudioContext} audioContext 5 | */ 6 | export default function sadtrombone (audioContext, tempo = 0.3) { 7 | const base = 300 8 | const step = 6/5 9 | 10 | const osc = audioContext.createOscillator() 11 | osc.type = 'sawtooth' 12 | osc.connect(audioContext.destination) 13 | osc.start() 14 | 15 | playNote(0, 0) 16 | playNote(1, 1) 17 | playNote(2, 2) 18 | playNote(3, 3) 19 | stop(6) 20 | 21 | function playNote (note, start) { 22 | const frequency = base / Math.pow(step, note) 23 | const time = audioContext.currentTime + start * tempo 24 | osc.frequency.setValueAtTime(frequency, time) 25 | } 26 | 27 | function stop (time) { 28 | osc.stop(audioContext.currentTime + time * tempo) 29 | } 30 | } 31 | 32 | -------------------------------------------------------------------------------- /snapshots/Sad Trombone should fail so that we can see the player.wav: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/salomvary/audio-snapshot-testing/6fffcd55edd5702e770084bc75aed9d8ac0f80f9/snapshots/Sad Trombone should fail so that we can see the player.wav -------------------------------------------------------------------------------- /snapshots/Sad Trombone should render sad trombone with a faster tempo.wav: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/salomvary/audio-snapshot-testing/6fffcd55edd5702e770084bc75aed9d8ac0f80f9/snapshots/Sad Trombone should render sad trombone with a faster tempo.wav -------------------------------------------------------------------------------- /snapshots/Sad Trombone should render sad trombone.wav: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/salomvary/audio-snapshot-testing/6fffcd55edd5702e770084bc75aed9d8ac0f80f9/snapshots/Sad Trombone should render sad trombone.wav -------------------------------------------------------------------------------- /test.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 |