├── .eslintrc ├── .gitignore ├── client ├── app.js.es6 ├── app.scss ├── index.html └── vendor │ └── diff.js ├── gulpfile.js ├── package.json └── server.js /.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "parser": "babel-eslint", 3 | "env": { 4 | "browser": true 5 | }, 6 | "rules": { 7 | "strict": 0, 8 | "no-extra-strict": 0, 9 | "quotes": [2, "single", "avoid-escape"], 10 | "curly": 0, 11 | "no-underscore-dangle": 0, 12 | "no-use-before-define": 0, 13 | "no-unused-vars": [1, {"vars": "all", "args": "none"}], 14 | "no-multi-spaces": 0 15 | } 16 | } -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | client/dist -------------------------------------------------------------------------------- /client/app.js.es6: -------------------------------------------------------------------------------- 1 | (function () { 2 | 3 | let button = document.querySelector('button'); 4 | let first = document.querySelector('.js-first'); 5 | let second = document.querySelector('.js-second'); 6 | let headers = document.querySelector('.js-headers'); 7 | let body = document.querySelector('.js-body'); 8 | let inputs = document.querySelector('.js-inputs'); 9 | 10 | const animateIntro = () => { 11 | let elems = document.querySelectorAll('.is-animated'); 12 | [...elems].forEach((el) => el.classList.remove('is-animated')); 13 | first.classList.add('inputs-url--red'); 14 | second.classList.add('inputs-url--green'); 15 | }; 16 | 17 | const displayError = (errorMessage) => { 18 | inputs.dataset.errors = `Error – ${errorMessage}`; 19 | inputs.classList.add('is-error'); 20 | setTimeout(() => { 21 | inputs.classList.remove('is-error'); 22 | }, 5000); 23 | }; 24 | 25 | const prettyPrint = (response) => { 26 | try { 27 | return JSON.stringify(JSON.parse(response), null, '\t'); 28 | } 29 | catch (e) { return response; } 30 | } 31 | 32 | const successHandler = (xhr, cb) => { 33 | animateIntro(); 34 | cb(null, xhr.response); 35 | }; 36 | 37 | const errorHandler = (xhr, cb) => { 38 | console.error(xhr.statusText); 39 | let error = xhr.response ? xhr.response : xhr.statusText; 40 | cb(error); 41 | }; 42 | 43 | const request = (url, cb) => { 44 | let xhr = new XMLHttpRequest(); 45 | xhr.open('GET', url, true); 46 | xhr.onload = (evt) => { 47 | if (xhr.status === 200) successHandler(xhr, cb); 48 | else errorHandler(xhr, cb); 49 | }; 50 | xhr.onerror = (evt) => errorHandler(xhr, cb); 51 | xhr.send(); 52 | } 53 | 54 | const compare = (first, second, element) => { 55 | let diff = JsDiff.diffLines(first, second); 56 | let frag = document.createDocumentFragment(); 57 | 58 | diff.forEach(function (part){ 59 | let classes = ['diff-line']; 60 | if (part.added) classes.push('diff-line--green'); 61 | else if (part.removed) classes.push('diff-line--red'); 62 | 63 | let span = document.createElement('span'); 64 | classes.forEach((c) => span.classList.add(c)); 65 | 66 | span.appendChild(document.createTextNode(part.value)); 67 | frag.appendChild(span); 68 | }); 69 | 70 | element.appendChild(frag); 71 | } 72 | 73 | const validateInputs = () => { 74 | let valid1 = first.checkValidity(); 75 | let valid2 = second.checkValidity(); 76 | 77 | if (valid1 && valid2) return true; 78 | 79 | const msg = (()=> { 80 | if (!valid1 && !valid2) return 'Invalid URLs'; 81 | else if (!valid1) return 'URL 1 is invalid'; 82 | else return 'URL 2 is invalid'; 83 | })(); 84 | 85 | displayError(msg); 86 | return false; 87 | }; 88 | 89 | const start = () => { 90 | if (!validateInputs()) return; 91 | 92 | let firstURL = encodeURIComponent(first.value); 93 | let secondURL = encodeURIComponent(second.value); 94 | let url = `/proxy?url1=${firstURL}&url2=${secondURL}`; 95 | 96 | headers.innerHTML = ''; 97 | body.innerHTML = ''; 98 | 99 | request(url, function (err, res) { 100 | if (err) return displayError(err); 101 | 102 | let data = JSON.parse(res); 103 | compare(prettyPrint(data.first.headers), 104 | prettyPrint(data.second.headers), 105 | headers); 106 | compare(prettyPrint(data.first.body), 107 | prettyPrint(data.second.body), 108 | body); 109 | }); 110 | } 111 | 112 | button.addEventListener('click', start); 113 | first.value = 'https://httpbin.org/get'; 114 | second.value = 'https://httpbin.org/get?show_env=1'; 115 | 116 | })(); -------------------------------------------------------------------------------- /client/app.scss: -------------------------------------------------------------------------------- 1 | $lightblue: #F4F7FB; 2 | $brightblue: #5C7AFF; 3 | $mediumblue: #4054B2; 4 | $darkblue: #4D5B99; 5 | $darkgrey: #555; 6 | $black: #333; 7 | $darkgreen: #123F26; 8 | $darkred: #5D2121; 9 | $errorred: #FF0B00; 10 | $container-width: 840px; 11 | 12 | html, body { 13 | height: 100%; 14 | margin: 0; 15 | background-color: $lightblue; 16 | box-sizing: border-box; 17 | font-family: 'Roboto', 'Helvetica Neue', 'Arial', sans-serif; 18 | } 19 | 20 | *, *:before, *:after { 21 | box-sizing: inherit; 22 | } 23 | 24 | body { 25 | display: flex; 26 | height: 100%; 27 | flex-direction: column; 28 | } 29 | 30 | main { 31 | flex: 1 0 auto; 32 | } 33 | 34 | header, footer { 35 | flex: none; 36 | } 37 | 38 | // debug layout 39 | // * { 40 | // border: 1px solid red; 41 | // } 42 | 43 | header { 44 | padding: 0 1rem; 45 | } 46 | 47 | .title { 48 | display: block; 49 | height: 4rem; 50 | width: 100%; 51 | margin: 1rem 0 2rem; 52 | transition: transform 0.3s ease; 53 | } 54 | 55 | .title-text { 56 | font-size: 3rem; 57 | font-weight: 900; 58 | font-style: italic; 59 | transition: all 0.3s ease; 60 | } 61 | 62 | .inputs { 63 | display: flex; 64 | flex-direction: column; 65 | margin: 0 auto; 66 | padding: 0 1rem; 67 | transform: translateY(0); 68 | transition: transform 0.3s ease; 69 | will-change: transform; 70 | } 71 | 72 | .inputs::after { 73 | content: attr(data-errors); 74 | position: absolute; 75 | bottom: -1rem; 76 | left: 0; 77 | right: 0; 78 | color: $errorred; 79 | text-align: center; 80 | opacity: 0; 81 | transform: translate3d(0, 1rem, 0); 82 | transition: opacity .3s ease, transform .3s ease; 83 | will-change: opacity, transform; 84 | } 85 | 86 | .inputs.is-error::after { 87 | opacity: 1; 88 | transform: translate3d(0, 0, 0); 89 | } 90 | 91 | .inputs-button { 92 | height: 3rem; 93 | width: 100%; 94 | margin: 1rem auto; 95 | padding: 3px auto; 96 | vertical-align: 2px; 97 | border: none; 98 | border-radius: 30px; 99 | font-size: 0.875rem; 100 | color: white; 101 | background-color: $darkblue; 102 | cursor: pointer; 103 | outline: none; 104 | transition: all 0.2s; 105 | 106 | &:active { 107 | background-color: $brightblue; 108 | box-shadow: 0 0 0 4px rgba(92, 122, 255, 0.5); 109 | } 110 | } 111 | 112 | .inputs-label { 113 | display: block; 114 | color: $mediumblue; 115 | font-weight: 500; 116 | } 117 | 118 | .inputs-url { 119 | display: block; 120 | font-size: 1rem; 121 | margin: 1rem auto; 122 | padding: 0.5rem; 123 | border: 1px solid rgba(92, 122, 255, 0.5); 124 | border-radius: 4px; 125 | outline: none; 126 | width: 100%; 127 | color: $black; 128 | transition: all 0.2s; 129 | appearance: none; 130 | 131 | &:invalid, 132 | &:-moz-submit-invalid 133 | &:-moz-ui-invalid { 134 | box-shadow: none; 135 | } 136 | 137 | &:focus { 138 | box-shadow: 0 0 0 3px rgba(92, 122, 255, 0.5); 139 | } 140 | } 141 | 142 | .inputs-url--red { 143 | border: 1px solid transparent; 144 | box-shadow: 0 0 0 3px rgba(255, 59, 48, 0.3); 145 | } 146 | 147 | .inputs-url--green { 148 | border: 1px solid transparent; 149 | box-shadow: 0 0 0 3px rgba(90, 249, 178, 0.3); 150 | } 151 | 152 | .results { 153 | display: block; 154 | position: relative; 155 | opacity: 1; 156 | transform: translateX(0); 157 | transition: opacity 0.3s ease, transform 0.3s ease; 158 | will-change: transform, opacity; 159 | } 160 | 161 | .results::after { 162 | width: 100%; 163 | height: 34px; 164 | left: 0; 165 | z-index: 2; 166 | background: linear-gradient(rgba(255,255,255,0), rgba(255,255,255,1)); 167 | content: " "; 168 | position: absolute; 169 | pointer-events: none; 170 | bottom: 0; 171 | border-radius: 0 0 5px 5px; 172 | } 173 | 174 | .results-title { 175 | margin: 2rem auto 1rem; 176 | padding: 0 1rem; 177 | font-size: 1rem; 178 | font-weight: 500; 179 | color: $mediumblue; 180 | } 181 | 182 | .results-bg { 183 | width: 100%; 184 | padding: 1rem 0 0; 185 | background-color: white; 186 | box-shadow: 0 1px 2px rgba(51, 61, 71, 0.12); 187 | } 188 | 189 | .results-content { 190 | margin: 0 auto; 191 | padding-bottom: 1rem; 192 | background-color: transparent; 193 | max-height: 500px; 194 | max-width: 100%; 195 | overflow: scroll; 196 | } 197 | 198 | .diff-line { 199 | font-size: 0.875rem; 200 | color: $darkgrey; 201 | } 202 | 203 | .diff-line--red { 204 | color: $darkred; 205 | background-color: rgba(255, 59, 48, 0.15); 206 | } 207 | 208 | .diff-line--green { 209 | color: $darkgreen; 210 | background-color: rgba(90, 249, 178, 0.15); 211 | } 212 | 213 | footer { 214 | margin: 3rem 0 2rem; 215 | text-align: center; 216 | color: $darkblue; 217 | -webkit-font-smoothing: antialiased; 218 | 219 | a { 220 | color: $darkblue; 221 | text-decoration: none; 222 | border-bottom: 1px dotted $darkblue; 223 | } 224 | } 225 | 226 | /** 227 | * results animation 228 | */ 229 | 230 | .results.is-animated { 231 | position: absolute; 232 | opacity: 0; 233 | transform: translateY(20rem); 234 | } 235 | 236 | @media screen and (min-width: 32em) { 237 | .title { 238 | width: $container-width; 239 | margin: 2rem auto 3rem; 240 | padding: 0; 241 | } 242 | 243 | .title-text { 244 | font-size: 2rem; 245 | } 246 | 247 | .inputs { 248 | flex-direction: row; 249 | justify-content: space-between; 250 | width: $container-width; 251 | padding: 0; 252 | } 253 | 254 | .inputs::after { 255 | bottom: -2rem; 256 | } 257 | 258 | .inputs-label { 259 | display: inline-block; 260 | margin: 0.5rem 1rem 0 0; 261 | } 262 | 263 | .inputs-url { 264 | width: 21rem; 265 | } 266 | 267 | .inputs-button { 268 | display: inline-block; 269 | padding: 3px 6px; 270 | align-self: center; 271 | width: 112px; 272 | margin: 1.5rem 0 0 0; 273 | 274 | &:hover { 275 | background-color: $brightblue; 276 | box-shadow: 0 0 0 4px rgba(92, 122, 255, 0.5); 277 | } 278 | 279 | &:active { 280 | background-color: $darkblue; 281 | box-shadow: none; 282 | } 283 | } 284 | 285 | .results-title { 286 | padding: 0; 287 | } 288 | 289 | .results-title, .results-content { 290 | width: $container-width; 291 | } 292 | 293 | /** 294 | * title animation 295 | */ 296 | 297 | .title.is-animated { 298 | // dx = ($container-width - max-svg-width)/2 299 | transform: translateX(255px) translateY(10rem); 300 | 301 | & > .title-text { 302 | font-size: 4rem; 303 | } 304 | } 305 | 306 | /** 307 | * inputs animation 308 | */ 309 | 310 | .inputs.is-animated { 311 | transform: translateY(10rem); 312 | } 313 | 314 | footer { 315 | opacity: 0.6; 316 | transition: opacity 0.2s; 317 | 318 | &:hover { 319 | opacity: 1; 320 | } 321 | } 322 | } 323 | -------------------------------------------------------------------------------- /client/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Requestdiff 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | Requestdiff 27 | 28 |
29 | 30 |
31 |
32 | 36 | 40 | 41 | 42 |
43 | 44 |
45 |

HEADERS

46 |
47 |

48 |     
49 |
50 | 51 |
52 |

BODY

53 |
54 |

55 |     
56 |
57 |
58 | 59 | 62 | 63 | 64 | 72 | 73 | -------------------------------------------------------------------------------- /client/vendor/diff.js: -------------------------------------------------------------------------------- 1 | (function webpackUniversalModuleDefinition(root, factory) { 2 | if(typeof exports === 'object' && typeof module === 'object') 3 | module.exports = factory(); 4 | else if(typeof define === 'function' && define.amd) 5 | define([], factory); 6 | else if(typeof exports === 'object') 7 | exports["JsDiff"] = factory(); 8 | else 9 | root["JsDiff"] = factory(); 10 | })(this, function() { 11 | return /******/ (function(modules) { // webpackBootstrap 12 | /******/ // The module cache 13 | /******/ var installedModules = {}; 14 | 15 | /******/ // The require function 16 | /******/ function __webpack_require__(moduleId) { 17 | 18 | /******/ // Check if module is in cache 19 | /******/ if(installedModules[moduleId]) 20 | /******/ return installedModules[moduleId].exports; 21 | 22 | /******/ // Create a new module (and put it into the cache) 23 | /******/ var module = installedModules[moduleId] = { 24 | /******/ exports: {}, 25 | /******/ id: moduleId, 26 | /******/ loaded: false 27 | /******/ }; 28 | 29 | /******/ // Execute the module function 30 | /******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__); 31 | 32 | /******/ // Flag the module as loaded 33 | /******/ module.loaded = true; 34 | 35 | /******/ // Return the exports of the module 36 | /******/ return module.exports; 37 | /******/ } 38 | 39 | 40 | /******/ // expose the modules object (__webpack_modules__) 41 | /******/ __webpack_require__.m = modules; 42 | 43 | /******/ // expose the module cache 44 | /******/ __webpack_require__.c = installedModules; 45 | 46 | /******/ // __webpack_public_path__ 47 | /******/ __webpack_require__.p = ""; 48 | 49 | /******/ // Load entry module and return exports 50 | /******/ return __webpack_require__(0); 51 | /******/ }) 52 | /************************************************************************/ 53 | /******/ ([ 54 | /* 0 */ 55 | /***/ function(module, exports, __webpack_require__) { 56 | 57 | /* See LICENSE file for terms of use */ 58 | 59 | /* 60 | * Text diff implementation. 61 | * 62 | * This library supports the following APIS: 63 | * JsDiff.diffChars: Character by character diff 64 | * JsDiff.diffWords: Word (as defined by \b regex) diff which ignores whitespace 65 | * JsDiff.diffLines: Line based diff 66 | * 67 | * JsDiff.diffCss: Diff targeted at CSS content 68 | * 69 | * These methods are based on the implementation proposed in 70 | * "An O(ND) Difference Algorithm and its Variations" (Myers, 1986). 71 | * http://citeseerx.ist.psu.edu/viewdoc/summary?doi=10.1.1.4.6927 72 | */ 73 | 'use strict'; 74 | 75 | exports.__esModule = true; 76 | // istanbul ignore next 77 | 78 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { 'default': obj }; } 79 | 80 | var _diffBase = __webpack_require__(1); 81 | 82 | var _diffBase2 = _interopRequireDefault(_diffBase); 83 | 84 | var _diffCharacter = __webpack_require__(2); 85 | 86 | var _diffWord = __webpack_require__(3); 87 | 88 | var _diffLine = __webpack_require__(5); 89 | 90 | var _diffSentence = __webpack_require__(6); 91 | 92 | var _diffCss = __webpack_require__(7); 93 | 94 | var _diffJson = __webpack_require__(8); 95 | 96 | var _patchApply = __webpack_require__(9); 97 | 98 | var _patchParse = __webpack_require__(10); 99 | 100 | var _patchCreate = __webpack_require__(11); 101 | 102 | var _convertDmp = __webpack_require__(12); 103 | 104 | var _convertXml = __webpack_require__(13); 105 | 106 | exports.Diff = _diffBase2['default']; 107 | exports.diffChars = _diffCharacter.diffChars; 108 | exports.diffWords = _diffWord.diffWords; 109 | exports.diffWordsWithSpace = _diffWord.diffWordsWithSpace; 110 | exports.diffLines = _diffLine.diffLines; 111 | exports.diffTrimmedLines = _diffLine.diffTrimmedLines; 112 | exports.diffSentences = _diffSentence.diffSentences; 113 | exports.diffCss = _diffCss.diffCss; 114 | exports.diffJson = _diffJson.diffJson; 115 | exports.structuredPatch = _patchCreate.structuredPatch; 116 | exports.createTwoFilesPatch = _patchCreate.createTwoFilesPatch; 117 | exports.createPatch = _patchCreate.createPatch; 118 | exports.applyPatch = _patchApply.applyPatch; 119 | exports.applyPatches = _patchApply.applyPatches; 120 | exports.parsePatch = _patchParse.parsePatch; 121 | exports.convertChangesToDMP = _convertDmp.convertChangesToDMP; 122 | exports.convertChangesToXML = _convertXml.convertChangesToXML; 123 | exports.canonicalize = _diffJson.canonicalize; 124 | //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozt3QkFnQmlCLGFBQWE7Ozs7NkJBQ04sa0JBQWtCOzt3QkFDRSxhQUFhOzt3QkFDZixhQUFhOzs0QkFDM0IsaUJBQWlCOzt1QkFFdkIsWUFBWTs7d0JBQ0csYUFBYTs7MEJBRVgsZUFBZTs7MEJBQzdCLGVBQWU7OzJCQUN3QixnQkFBZ0I7OzBCQUU5QyxlQUFlOzswQkFDZixlQUFlOztRQUcvQyxJQUFJO1FBRUosU0FBUztRQUNULFNBQVM7UUFDVCxrQkFBa0I7UUFDbEIsU0FBUztRQUNULGdCQUFnQjtRQUNoQixhQUFhO1FBRWIsT0FBTztRQUNQLFFBQVE7UUFFUixlQUFlO1FBQ2YsbUJBQW1CO1FBQ25CLFdBQVc7UUFDWCxVQUFVO1FBQ1YsWUFBWTtRQUNaLFVBQVU7UUFDVixtQkFBbUI7UUFDbkIsbUJBQW1CO1FBQ25CLFlBQVkiLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBTZWUgTElDRU5TRSBmaWxlIGZvciB0ZXJtcyBvZiB1c2UgKi9cblxuLypcbiAqIFRleHQgZGlmZiBpbXBsZW1lbnRhdGlvbi5cbiAqXG4gKiBUaGlzIGxpYnJhcnkgc3VwcG9ydHMgdGhlIGZvbGxvd2luZyBBUElTOlxuICogSnNEaWZmLmRpZmZDaGFyczogQ2hhcmFjdGVyIGJ5IGNoYXJhY3RlciBkaWZmXG4gKiBKc0RpZmYuZGlmZldvcmRzOiBXb3JkIChhcyBkZWZpbmVkIGJ5IFxcYiByZWdleCkgZGlmZiB3aGljaCBpZ25vcmVzIHdoaXRlc3BhY2VcbiAqIEpzRGlmZi5kaWZmTGluZXM6IExpbmUgYmFzZWQgZGlmZlxuICpcbiAqIEpzRGlmZi5kaWZmQ3NzOiBEaWZmIHRhcmdldGVkIGF0IENTUyBjb250ZW50XG4gKlxuICogVGhlc2UgbWV0aG9kcyBhcmUgYmFzZWQgb24gdGhlIGltcGxlbWVudGF0aW9uIHByb3Bvc2VkIGluXG4gKiBcIkFuIE8oTkQpIERpZmZlcmVuY2UgQWxnb3JpdGhtIGFuZCBpdHMgVmFyaWF0aW9uc1wiIChNeWVycywgMTk4NikuXG4gKiBodHRwOi8vY2l0ZXNlZXJ4LmlzdC5wc3UuZWR1L3ZpZXdkb2Mvc3VtbWFyeT9kb2k9MTAuMS4xLjQuNjkyN1xuICovXG5pbXBvcnQgRGlmZiBmcm9tICcuL2RpZmYvYmFzZSc7XG5pbXBvcnQge2RpZmZDaGFyc30gZnJvbSAnLi9kaWZmL2NoYXJhY3Rlcic7XG5pbXBvcnQge2RpZmZXb3JkcywgZGlmZldvcmRzV2l0aFNwYWNlfSBmcm9tICcuL2RpZmYvd29yZCc7XG5pbXBvcnQge2RpZmZMaW5lcywgZGlmZlRyaW1tZWRMaW5lc30gZnJvbSAnLi9kaWZmL2xpbmUnO1xuaW1wb3J0IHtkaWZmU2VudGVuY2VzfSBmcm9tICcuL2RpZmYvc2VudGVuY2UnO1xuXG5pbXBvcnQge2RpZmZDc3N9IGZyb20gJy4vZGlmZi9jc3MnO1xuaW1wb3J0IHtkaWZmSnNvbiwgY2Fub25pY2FsaXplfSBmcm9tICcuL2RpZmYvanNvbic7XG5cbmltcG9ydCB7YXBwbHlQYXRjaCwgYXBwbHlQYXRjaGVzfSBmcm9tICcuL3BhdGNoL2FwcGx5JztcbmltcG9ydCB7cGFyc2VQYXRjaH0gZnJvbSAnLi9wYXRjaC9wYXJzZSc7XG5pbXBvcnQge3N0cnVjdHVyZWRQYXRjaCwgY3JlYXRlVHdvRmlsZXNQYXRjaCwgY3JlYXRlUGF0Y2h9IGZyb20gJy4vcGF0Y2gvY3JlYXRlJztcblxuaW1wb3J0IHtjb252ZXJ0Q2hhbmdlc1RvRE1QfSBmcm9tICcuL2NvbnZlcnQvZG1wJztcbmltcG9ydCB7Y29udmVydENoYW5nZXNUb1hNTH0gZnJvbSAnLi9jb252ZXJ0L3htbCc7XG5cbmV4cG9ydCB7XG4gIERpZmYsXG5cbiAgZGlmZkNoYXJzLFxuICBkaWZmV29yZHMsXG4gIGRpZmZXb3Jkc1dpdGhTcGFjZSxcbiAgZGlmZkxpbmVzLFxuICBkaWZmVHJpbW1lZExpbmVzLFxuICBkaWZmU2VudGVuY2VzLFxuXG4gIGRpZmZDc3MsXG4gIGRpZmZKc29uLFxuXG4gIHN0cnVjdHVyZWRQYXRjaCxcbiAgY3JlYXRlVHdvRmlsZXNQYXRjaCxcbiAgY3JlYXRlUGF0Y2gsXG4gIGFwcGx5UGF0Y2gsXG4gIGFwcGx5UGF0Y2hlcyxcbiAgcGFyc2VQYXRjaCxcbiAgY29udmVydENoYW5nZXNUb0RNUCxcbiAgY29udmVydENoYW5nZXNUb1hNTCxcbiAgY2Fub25pY2FsaXplXG59O1xuIl19 125 | 126 | 127 | /***/ }, 128 | /* 1 */ 129 | /***/ function(module, exports) { 130 | 131 | 'use strict'; 132 | 133 | exports.__esModule = true; 134 | exports['default'] = Diff; 135 | 136 | function Diff() {} 137 | 138 | Diff.prototype = { 139 | diff: function diff(oldString, newString) { 140 | var options = arguments.length <= 2 || arguments[2] === undefined ? {} : arguments[2]; 141 | 142 | var callback = options.callback; 143 | if (typeof options === 'function') { 144 | callback = options; 145 | options = {}; 146 | } 147 | this.options = options; 148 | 149 | var self = this; 150 | 151 | function done(value) { 152 | if (callback) { 153 | setTimeout(function () { 154 | callback(undefined, value); 155 | }, 0); 156 | return true; 157 | } else { 158 | return value; 159 | } 160 | } 161 | 162 | // Allow subclasses to massage the input prior to running 163 | oldString = this.castInput(oldString); 164 | newString = this.castInput(newString); 165 | 166 | oldString = this.removeEmpty(this.tokenize(oldString)); 167 | newString = this.removeEmpty(this.tokenize(newString)); 168 | 169 | var newLen = newString.length, 170 | oldLen = oldString.length; 171 | var editLength = 1; 172 | var maxEditLength = newLen + oldLen; 173 | var bestPath = [{ newPos: -1, components: [] }]; 174 | 175 | // Seed editLength = 0, i.e. the content starts with the same values 176 | var oldPos = this.extractCommon(bestPath[0], newString, oldString, 0); 177 | if (bestPath[0].newPos + 1 >= newLen && oldPos + 1 >= oldLen) { 178 | // Identity per the equality and tokenizer 179 | return done([{ value: newString.join(''), count: newString.length }]); 180 | } 181 | 182 | // Main worker method. checks all permutations of a given edit length for acceptance. 183 | function execEditLength() { 184 | for (var diagonalPath = -1 * editLength; diagonalPath <= editLength; diagonalPath += 2) { 185 | var basePath = undefined; 186 | var addPath = bestPath[diagonalPath - 1], 187 | removePath = bestPath[diagonalPath + 1], 188 | _oldPos = (removePath ? removePath.newPos : 0) - diagonalPath; 189 | if (addPath) { 190 | // No one else is going to attempt to use this value, clear it 191 | bestPath[diagonalPath - 1] = undefined; 192 | } 193 | 194 | var canAdd = addPath && addPath.newPos + 1 < newLen, 195 | canRemove = removePath && 0 <= _oldPos && _oldPos < oldLen; 196 | if (!canAdd && !canRemove) { 197 | // If this path is a terminal then prune 198 | bestPath[diagonalPath] = undefined; 199 | continue; 200 | } 201 | 202 | // Select the diagonal that we want to branch from. We select the prior 203 | // path whose position in the new string is the farthest from the origin 204 | // and does not pass the bounds of the diff graph 205 | if (!canAdd || canRemove && addPath.newPos < removePath.newPos) { 206 | basePath = clonePath(removePath); 207 | self.pushComponent(basePath.components, undefined, true); 208 | } else { 209 | basePath = addPath; // No need to clone, we've pulled it from the list 210 | basePath.newPos++; 211 | self.pushComponent(basePath.components, true, undefined); 212 | } 213 | 214 | _oldPos = self.extractCommon(basePath, newString, oldString, diagonalPath); 215 | 216 | // If we have hit the end of both strings, then we are done 217 | if (basePath.newPos + 1 >= newLen && _oldPos + 1 >= oldLen) { 218 | return done(buildValues(self, basePath.components, newString, oldString, self.useLongestToken)); 219 | } else { 220 | // Otherwise track this path as a potential candidate and continue. 221 | bestPath[diagonalPath] = basePath; 222 | } 223 | } 224 | 225 | editLength++; 226 | } 227 | 228 | // Performs the length of edit iteration. Is a bit fugly as this has to support the 229 | // sync and async mode which is never fun. Loops over execEditLength until a value 230 | // is produced. 231 | if (callback) { 232 | (function exec() { 233 | setTimeout(function () { 234 | // This should not happen, but we want to be safe. 235 | /* istanbul ignore next */ 236 | if (editLength > maxEditLength) { 237 | return callback(); 238 | } 239 | 240 | if (!execEditLength()) { 241 | exec(); 242 | } 243 | }, 0); 244 | })(); 245 | } else { 246 | while (editLength <= maxEditLength) { 247 | var ret = execEditLength(); 248 | if (ret) { 249 | return ret; 250 | } 251 | } 252 | } 253 | }, 254 | 255 | pushComponent: function pushComponent(components, added, removed) { 256 | var last = components[components.length - 1]; 257 | if (last && last.added === added && last.removed === removed) { 258 | // We need to clone here as the component clone operation is just 259 | // as shallow array clone 260 | components[components.length - 1] = { count: last.count + 1, added: added, removed: removed }; 261 | } else { 262 | components.push({ count: 1, added: added, removed: removed }); 263 | } 264 | }, 265 | extractCommon: function extractCommon(basePath, newString, oldString, diagonalPath) { 266 | var newLen = newString.length, 267 | oldLen = oldString.length, 268 | newPos = basePath.newPos, 269 | oldPos = newPos - diagonalPath, 270 | commonCount = 0; 271 | while (newPos + 1 < newLen && oldPos + 1 < oldLen && this.equals(newString[newPos + 1], oldString[oldPos + 1])) { 272 | newPos++; 273 | oldPos++; 274 | commonCount++; 275 | } 276 | 277 | if (commonCount) { 278 | basePath.components.push({ count: commonCount }); 279 | } 280 | 281 | basePath.newPos = newPos; 282 | return oldPos; 283 | }, 284 | 285 | equals: function equals(left, right) { 286 | return left === right; 287 | }, 288 | removeEmpty: function removeEmpty(array) { 289 | var ret = []; 290 | for (var i = 0; i < array.length; i++) { 291 | if (array[i]) { 292 | ret.push(array[i]); 293 | } 294 | } 295 | return ret; 296 | }, 297 | castInput: function castInput(value) { 298 | return value; 299 | }, 300 | tokenize: function tokenize(value) { 301 | return value.split(''); 302 | } 303 | }; 304 | 305 | function buildValues(diff, components, newString, oldString, useLongestToken) { 306 | var componentPos = 0, 307 | componentLen = components.length, 308 | newPos = 0, 309 | oldPos = 0; 310 | 311 | for (; componentPos < componentLen; componentPos++) { 312 | var component = components[componentPos]; 313 | if (!component.removed) { 314 | if (!component.added && useLongestToken) { 315 | var value = newString.slice(newPos, newPos + component.count); 316 | value = value.map(function (value, i) { 317 | var oldValue = oldString[oldPos + i]; 318 | return oldValue.length > value.length ? oldValue : value; 319 | }); 320 | 321 | component.value = value.join(''); 322 | } else { 323 | component.value = newString.slice(newPos, newPos + component.count).join(''); 324 | } 325 | newPos += component.count; 326 | 327 | // Common case 328 | if (!component.added) { 329 | oldPos += component.count; 330 | } 331 | } else { 332 | component.value = oldString.slice(oldPos, oldPos + component.count).join(''); 333 | oldPos += component.count; 334 | 335 | // Reverse add and remove so removes are output first to match common convention 336 | // The diffing algorithm is tied to add then remove output and this is the simplest 337 | // route to get the desired output with minimal overhead. 338 | if (componentPos && components[componentPos - 1].added) { 339 | var tmp = components[componentPos - 1]; 340 | components[componentPos - 1] = components[componentPos]; 341 | components[componentPos] = tmp; 342 | } 343 | } 344 | } 345 | 346 | // Special case handle for when one terminal is ignored. For this case we merge the 347 | // terminal into the prior string and drop the change. 348 | var lastComponent = components[componentLen - 1]; 349 | if ((lastComponent.added || lastComponent.removed) && diff.equals('', lastComponent.value)) { 350 | components[componentLen - 2].value += lastComponent.value; 351 | components.pop(); 352 | } 353 | 354 | return components; 355 | } 356 | 357 | function clonePath(path) { 358 | return { newPos: path.newPos, components: path.components.slice(0) }; 359 | } 360 | module.exports = exports['default']; 361 | //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/diff/base.js"],"names":[],"mappings":"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","file":"base.js","sourcesContent":["export default function Diff() {}\n\nDiff.prototype = {\n  diff(oldString, newString, options = {}) {\n    let callback = options.callback;\n    if (typeof options === 'function') {\n      callback = options;\n      options = {};\n    }\n    this.options = options;\n\n    let self = this;\n\n    function done(value) {\n      if (callback) {\n        setTimeout(function() { callback(undefined, value); }, 0);\n        return true;\n      } else {\n        return value;\n      }\n    }\n\n    // Allow subclasses to massage the input prior to running\n    oldString = this.castInput(oldString);\n    newString = this.castInput(newString);\n\n    oldString = this.removeEmpty(this.tokenize(oldString));\n    newString = this.removeEmpty(this.tokenize(newString));\n\n    let newLen = newString.length, oldLen = oldString.length;\n    let editLength = 1;\n    let maxEditLength = newLen + oldLen;\n    let bestPath = [{ newPos: -1, components: [] }];\n\n    // Seed editLength = 0, i.e. the content starts with the same values\n    let oldPos = this.extractCommon(bestPath[0], newString, oldString, 0);\n    if (bestPath[0].newPos + 1 >= newLen && oldPos + 1 >= oldLen) {\n      // Identity per the equality and tokenizer\n      return done([{value: newString.join(''), count: newString.length}]);\n    }\n\n    // Main worker method. checks all permutations of a given edit length for acceptance.\n    function execEditLength() {\n      for (let diagonalPath = -1 * editLength; diagonalPath <= editLength; diagonalPath += 2) {\n        let basePath;\n        let addPath = bestPath[diagonalPath - 1],\n            removePath = bestPath[diagonalPath + 1],\n            oldPos = (removePath ? removePath.newPos : 0) - diagonalPath;\n        if (addPath) {\n          // No one else is going to attempt to use this value, clear it\n          bestPath[diagonalPath - 1] = undefined;\n        }\n\n        let canAdd = addPath && addPath.newPos + 1 < newLen,\n            canRemove = removePath && 0 <= oldPos && oldPos < oldLen;\n        if (!canAdd && !canRemove) {\n          // If this path is a terminal then prune\n          bestPath[diagonalPath] = undefined;\n          continue;\n        }\n\n        // Select the diagonal that we want to branch from. We select the prior\n        // path whose position in the new string is the farthest from the origin\n        // and does not pass the bounds of the diff graph\n        if (!canAdd || (canRemove && addPath.newPos < removePath.newPos)) {\n          basePath = clonePath(removePath);\n          self.pushComponent(basePath.components, undefined, true);\n        } else {\n          basePath = addPath;   // No need to clone, we've pulled it from the list\n          basePath.newPos++;\n          self.pushComponent(basePath.components, true, undefined);\n        }\n\n        oldPos = self.extractCommon(basePath, newString, oldString, diagonalPath);\n\n        // If we have hit the end of both strings, then we are done\n        if (basePath.newPos + 1 >= newLen && oldPos + 1 >= oldLen) {\n          return done(buildValues(self, basePath.components, newString, oldString, self.useLongestToken));\n        } else {\n          // Otherwise track this path as a potential candidate and continue.\n          bestPath[diagonalPath] = basePath;\n        }\n      }\n\n      editLength++;\n    }\n\n    // Performs the length of edit iteration. Is a bit fugly as this has to support the\n    // sync and async mode which is never fun. Loops over execEditLength until a value\n    // is produced.\n    if (callback) {\n      (function exec() {\n        setTimeout(function() {\n          // This should not happen, but we want to be safe.\n          /* istanbul ignore next */\n          if (editLength > maxEditLength) {\n            return callback();\n          }\n\n          if (!execEditLength()) {\n            exec();\n          }\n        }, 0);\n      }());\n    } else {\n      while (editLength <= maxEditLength) {\n        let ret = execEditLength();\n        if (ret) {\n          return ret;\n        }\n      }\n    }\n  },\n\n  pushComponent(components, added, removed) {\n    let last = components[components.length - 1];\n    if (last && last.added === added && last.removed === removed) {\n      // We need to clone here as the component clone operation is just\n      // as shallow array clone\n      components[components.length - 1] = {count: last.count + 1, added: added, removed: removed };\n    } else {\n      components.push({count: 1, added: added, removed: removed });\n    }\n  },\n  extractCommon(basePath, newString, oldString, diagonalPath) {\n    let newLen = newString.length,\n        oldLen = oldString.length,\n        newPos = basePath.newPos,\n        oldPos = newPos - diagonalPath,\n\n        commonCount = 0;\n    while (newPos + 1 < newLen && oldPos + 1 < oldLen && this.equals(newString[newPos + 1], oldString[oldPos + 1])) {\n      newPos++;\n      oldPos++;\n      commonCount++;\n    }\n\n    if (commonCount) {\n      basePath.components.push({count: commonCount});\n    }\n\n    basePath.newPos = newPos;\n    return oldPos;\n  },\n\n  equals(left, right) {\n    return left === right;\n  },\n  removeEmpty(array) {\n    let ret = [];\n    for (let i = 0; i < array.length; i++) {\n      if (array[i]) {\n        ret.push(array[i]);\n      }\n    }\n    return ret;\n  },\n  castInput(value) {\n    return value;\n  },\n  tokenize(value) {\n    return value.split('');\n  }\n};\n\nfunction buildValues(diff, components, newString, oldString, useLongestToken) {\n  let componentPos = 0,\n      componentLen = components.length,\n      newPos = 0,\n      oldPos = 0;\n\n  for (; componentPos < componentLen; componentPos++) {\n    let component = components[componentPos];\n    if (!component.removed) {\n      if (!component.added && useLongestToken) {\n        let value = newString.slice(newPos, newPos + component.count);\n        value = value.map(function(value, i) {\n          let oldValue = oldString[oldPos + i];\n          return oldValue.length > value.length ? oldValue : value;\n        });\n\n        component.value = value.join('');\n      } else {\n        component.value = newString.slice(newPos, newPos + component.count).join('');\n      }\n      newPos += component.count;\n\n      // Common case\n      if (!component.added) {\n        oldPos += component.count;\n      }\n    } else {\n      component.value = oldString.slice(oldPos, oldPos + component.count).join('');\n      oldPos += component.count;\n\n      // Reverse add and remove so removes are output first to match common convention\n      // The diffing algorithm is tied to add then remove output and this is the simplest\n      // route to get the desired output with minimal overhead.\n      if (componentPos && components[componentPos - 1].added) {\n        let tmp = components[componentPos - 1];\n        components[componentPos - 1] = components[componentPos];\n        components[componentPos] = tmp;\n      }\n    }\n  }\n\n  // Special case handle for when one terminal is ignored. For this case we merge the\n  // terminal into the prior string and drop the change.\n  let lastComponent = components[componentLen - 1];\n  if ((lastComponent.added || lastComponent.removed) && diff.equals('', lastComponent.value)) {\n    components[componentLen - 2].value += lastComponent.value;\n    components.pop();\n  }\n\n  return components;\n}\n\nfunction clonePath(path) {\n  return { newPos: path.newPos, components: path.components.slice(0) };\n}\n"]} 362 | 363 | 364 | /***/ }, 365 | /* 2 */ 366 | /***/ function(module, exports, __webpack_require__) { 367 | 368 | 'use strict'; 369 | 370 | exports.__esModule = true; 371 | exports.diffChars = diffChars; 372 | // istanbul ignore next 373 | 374 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { 'default': obj }; } 375 | 376 | var _base = __webpack_require__(1); 377 | 378 | var _base2 = _interopRequireDefault(_base); 379 | 380 | var characterDiff = new _base2['default'](); 381 | exports.characterDiff = characterDiff; 382 | 383 | function diffChars(oldStr, newStr, callback) { 384 | return characterDiff.diff(oldStr, newStr, callback); 385 | } 386 | //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kaWZmL2NoYXJhY3Rlci5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7OztvQkFBaUIsUUFBUTs7OztBQUVsQixJQUFNLGFBQWEsR0FBRyx1QkFBVSxDQUFDOzs7QUFDakMsU0FBUyxTQUFTLENBQUMsTUFBTSxFQUFFLE1BQU0sRUFBRSxRQUFRLEVBQUU7QUFBRSxTQUFPLGFBQWEsQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLE1BQU0sRUFBRSxRQUFRLENBQUMsQ0FBQztDQUFFIiwiZmlsZSI6ImNoYXJhY3Rlci5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBEaWZmIGZyb20gJy4vYmFzZSc7XG5cbmV4cG9ydCBjb25zdCBjaGFyYWN0ZXJEaWZmID0gbmV3IERpZmYoKTtcbmV4cG9ydCBmdW5jdGlvbiBkaWZmQ2hhcnMob2xkU3RyLCBuZXdTdHIsIGNhbGxiYWNrKSB7IHJldHVybiBjaGFyYWN0ZXJEaWZmLmRpZmYob2xkU3RyLCBuZXdTdHIsIGNhbGxiYWNrKTsgfVxuIl19 387 | 388 | 389 | /***/ }, 390 | /* 3 */ 391 | /***/ function(module, exports, __webpack_require__) { 392 | 393 | 'use strict'; 394 | 395 | exports.__esModule = true; 396 | exports.diffWords = diffWords; 397 | exports.diffWordsWithSpace = diffWordsWithSpace; 398 | // istanbul ignore next 399 | 400 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { 'default': obj }; } 401 | 402 | var _base = __webpack_require__(1); 403 | 404 | var _base2 = _interopRequireDefault(_base); 405 | 406 | var _utilParams = __webpack_require__(4); 407 | 408 | // Based on https://en.wikipedia.org/wiki/Latin_script_in_Unicode 409 | // 410 | // Ranges and exceptions: 411 | // Latin-1 Supplement, 0080–00FF 412 | // - U+00D7 × Multiplication sign 413 | // - U+00F7 ÷ Division sign 414 | // Latin Extended-A, 0100–017F 415 | // Latin Extended-B, 0180–024F 416 | // IPA Extensions, 0250–02AF 417 | // Spacing Modifier Letters, 02B0–02FF 418 | // - U+02C7 ˇ ˇ Caron 419 | // - U+02D8 ˘ ˘ Breve 420 | // - U+02D9 ˙ ˙ Dot Above 421 | // - U+02DA ˚ ˚ Ring Above 422 | // - U+02DB ˛ ˛ Ogonek 423 | // - U+02DC ˜ ˜ Small Tilde 424 | // - U+02DD ˝ ˝ Double Acute Accent 425 | // Latin Extended Additional, 1E00–1EFF 426 | var extendedWordChars = /^[A-Za-z\xC0-\u02C6\u02C8-\u02D7\u02DE-\u02FF\u1E00-\u1EFF]+$/; 427 | 428 | var reWhitespace = /\S/; 429 | 430 | var wordDiff = new _base2['default'](); 431 | exports.wordDiff = wordDiff; 432 | wordDiff.equals = function (left, right) { 433 | return left === right || this.options.ignoreWhitespace && !reWhitespace.test(left) && !reWhitespace.test(right); 434 | }; 435 | wordDiff.tokenize = function (value) { 436 | var tokens = value.split(/(\s+|\b)/); 437 | 438 | // Join the boundary splits that we do not consider to be boundaries. This is primarily the extended Latin character set. 439 | for (var i = 0; i < tokens.length - 1; i++) { 440 | // If we have an empty string in the next field and we have only word chars before and after, merge 441 | if (!tokens[i + 1] && tokens[i + 2] && extendedWordChars.test(tokens[i]) && extendedWordChars.test(tokens[i + 2])) { 442 | tokens[i] += tokens[i + 2]; 443 | tokens.splice(i + 1, 2); 444 | i--; 445 | } 446 | } 447 | 448 | return tokens; 449 | }; 450 | 451 | function diffWords(oldStr, newStr, callback) { 452 | var options = _utilParams.generateOptions(callback, { ignoreWhitespace: true }); 453 | return wordDiff.diff(oldStr, newStr, options); 454 | } 455 | 456 | function diffWordsWithSpace(oldStr, newStr, callback) { 457 | return wordDiff.diff(oldStr, newStr, callback); 458 | } 459 | //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kaWZmL3dvcmQuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7O29CQUFpQixRQUFROzs7OzBCQUNLLGdCQUFnQjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFvQjlDLElBQU0saUJBQWlCLEdBQUcsK0RBQXFHLENBQUM7O0FBRWhJLElBQU0sWUFBWSxHQUFHLElBQUksQ0FBQzs7QUFFbkIsSUFBTSxRQUFRLEdBQUcsdUJBQVUsQ0FBQzs7QUFDbkMsUUFBUSxDQUFDLE1BQU0sR0FBRyxVQUFTLElBQUksRUFBRSxLQUFLLEVBQUU7QUFDdEMsU0FBTyxJQUFJLEtBQUssS0FBSyxJQUFLLElBQUksQ0FBQyxPQUFPLENBQUMsZ0JBQWdCLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQUFBQyxDQUFDO0NBQ25ILENBQUM7QUFDRixRQUFRLENBQUMsUUFBUSxHQUFHLFVBQVMsS0FBSyxFQUFFO0FBQ2xDLE1BQUksTUFBTSxHQUFHLEtBQUssQ0FBQyxLQUFLLENBQUMsVUFBVSxDQUFDLENBQUM7OztBQUdyQyxPQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsTUFBTSxDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUU7O0FBRTFDLFFBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxJQUFJLE1BQU0sQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLElBQzFCLGlCQUFpQixDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFDakMsaUJBQWlCLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRTtBQUM5QyxZQUFNLENBQUMsQ0FBQyxDQUFDLElBQUksTUFBTSxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztBQUMzQixZQUFNLENBQUMsTUFBTSxDQUFDLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7QUFDeEIsT0FBQyxFQUFFLENBQUM7S0FDTDtHQUNGOztBQUVELFNBQU8sTUFBTSxDQUFDO0NBQ2YsQ0FBQzs7QUFFSyxTQUFTLFNBQVMsQ0FBQyxNQUFNLEVBQUUsTUFBTSxFQUFFLFFBQVEsRUFBRTtBQUNsRCxNQUFJLE9BQU8sR0FBRyw0QkFBZ0IsUUFBUSxFQUFFLEVBQUMsZ0JBQWdCLEVBQUUsSUFBSSxFQUFDLENBQUMsQ0FBQztBQUNsRSxTQUFPLFFBQVEsQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLE1BQU0sRUFBRSxPQUFPLENBQUMsQ0FBQztDQUMvQzs7QUFDTSxTQUFTLGtCQUFrQixDQUFDLE1BQU0sRUFBRSxNQUFNLEVBQUUsUUFBUSxFQUFFO0FBQzNELFNBQU8sUUFBUSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsTUFBTSxFQUFFLFFBQVEsQ0FBQyxDQUFDO0NBQ2hEIiwiZmlsZSI6IndvcmQuanMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgRGlmZiBmcm9tICcuL2Jhc2UnO1xuaW1wb3J0IHtnZW5lcmF0ZU9wdGlvbnN9IGZyb20gJy4uL3V0aWwvcGFyYW1zJztcblxuLy8gQmFzZWQgb24gaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvTGF0aW5fc2NyaXB0X2luX1VuaWNvZGVcbi8vXG4vLyBSYW5nZXMgYW5kIGV4Y2VwdGlvbnM6XG4vLyBMYXRpbi0xIFN1cHBsZW1lbnQsIDAwODDigJMwMEZGXG4vLyAgLSBVKzAwRDcgIMOXIE11bHRpcGxpY2F0aW9uIHNpZ25cbi8vICAtIFUrMDBGNyAgw7cgRGl2aXNpb24gc2lnblxuLy8gTGF0aW4gRXh0ZW5kZWQtQSwgMDEwMOKAkzAxN0Zcbi8vIExhdGluIEV4dGVuZGVkLUIsIDAxODDigJMwMjRGXG4vLyBJUEEgRXh0ZW5zaW9ucywgMDI1MOKAkzAyQUZcbi8vIFNwYWNpbmcgTW9kaWZpZXIgTGV0dGVycywgMDJCMOKAkzAyRkZcbi8vICAtIFUrMDJDNyAgy4cgJiM3MTE7ICBDYXJvblxuLy8gIC0gVSswMkQ4ICDLmCAmIzcyODsgIEJyZXZlXG4vLyAgLSBVKzAyRDkgIMuZICYjNzI5OyAgRG90IEFib3ZlXG4vLyAgLSBVKzAyREEgIMuaICYjNzMwOyAgUmluZyBBYm92ZVxuLy8gIC0gVSswMkRCICDLmyAmIzczMTsgIE9nb25la1xuLy8gIC0gVSswMkRDICDLnCAmIzczMjsgIFNtYWxsIFRpbGRlXG4vLyAgLSBVKzAyREQgIMudICYjNzMzOyAgRG91YmxlIEFjdXRlIEFjY2VudFxuLy8gTGF0aW4gRXh0ZW5kZWQgQWRkaXRpb25hbCwgMUUwMOKAkzFFRkZcbmNvbnN0IGV4dGVuZGVkV29yZENoYXJzID0gL15bYS16QS1aXFx1e0MwfS1cXHV7RkZ9XFx1e0Q4fS1cXHV7RjZ9XFx1e0Y4fS1cXHV7MkM2fVxcdXsyQzh9LVxcdXsyRDd9XFx1ezJERX0tXFx1ezJGRn1cXHV7MUUwMH0tXFx1ezFFRkZ9XSskL3U7XG5cbmNvbnN0IHJlV2hpdGVzcGFjZSA9IC9cXFMvO1xuXG5leHBvcnQgY29uc3Qgd29yZERpZmYgPSBuZXcgRGlmZigpO1xud29yZERpZmYuZXF1YWxzID0gZnVuY3Rpb24obGVmdCwgcmlnaHQpIHtcbiAgcmV0dXJuIGxlZnQgPT09IHJpZ2h0IHx8ICh0aGlzLm9wdGlvbnMuaWdub3JlV2hpdGVzcGFjZSAmJiAhcmVXaGl0ZXNwYWNlLnRlc3QobGVmdCkgJiYgIXJlV2hpdGVzcGFjZS50ZXN0KHJpZ2h0KSk7XG59O1xud29yZERpZmYudG9rZW5pemUgPSBmdW5jdGlvbih2YWx1ZSkge1xuICBsZXQgdG9rZW5zID0gdmFsdWUuc3BsaXQoLyhcXHMrfFxcYikvKTtcblxuICAvLyBKb2luIHRoZSBib3VuZGFyeSBzcGxpdHMgdGhhdCB3ZSBkbyBub3QgY29uc2lkZXIgdG8gYmUgYm91bmRhcmllcy4gVGhpcyBpcyBwcmltYXJpbHkgdGhlIGV4dGVuZGVkIExhdGluIGNoYXJhY3RlciBzZXQuXG4gIGZvciAobGV0IGkgPSAwOyBpIDwgdG9rZW5zLmxlbmd0aCAtIDE7IGkrKykge1xuICAgIC8vIElmIHdlIGhhdmUgYW4gZW1wdHkgc3RyaW5nIGluIHRoZSBuZXh0IGZpZWxkIGFuZCB3ZSBoYXZlIG9ubHkgd29yZCBjaGFycyBiZWZvcmUgYW5kIGFmdGVyLCBtZXJnZVxuICAgIGlmICghdG9rZW5zW2kgKyAxXSAmJiB0b2tlbnNbaSArIDJdXG4gICAgICAgICAgJiYgZXh0ZW5kZWRXb3JkQ2hhcnMudGVzdCh0b2tlbnNbaV0pXG4gICAgICAgICAgJiYgZXh0ZW5kZWRXb3JkQ2hhcnMudGVzdCh0b2tlbnNbaSArIDJdKSkge1xuICAgICAgdG9rZW5zW2ldICs9IHRva2Vuc1tpICsgMl07XG4gICAgICB0b2tlbnMuc3BsaWNlKGkgKyAxLCAyKTtcbiAgICAgIGktLTtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gdG9rZW5zO1xufTtcblxuZXhwb3J0IGZ1bmN0aW9uIGRpZmZXb3JkcyhvbGRTdHIsIG5ld1N0ciwgY2FsbGJhY2spIHtcbiAgbGV0IG9wdGlvbnMgPSBnZW5lcmF0ZU9wdGlvbnMoY2FsbGJhY2ssIHtpZ25vcmVXaGl0ZXNwYWNlOiB0cnVlfSk7XG4gIHJldHVybiB3b3JkRGlmZi5kaWZmKG9sZFN0ciwgbmV3U3RyLCBvcHRpb25zKTtcbn1cbmV4cG9ydCBmdW5jdGlvbiBkaWZmV29yZHNXaXRoU3BhY2Uob2xkU3RyLCBuZXdTdHIsIGNhbGxiYWNrKSB7XG4gIHJldHVybiB3b3JkRGlmZi5kaWZmKG9sZFN0ciwgbmV3U3RyLCBjYWxsYmFjayk7XG59XG4iXX0= 460 | 461 | 462 | /***/ }, 463 | /* 4 */ 464 | /***/ function(module, exports) { 465 | 466 | 'use strict'; 467 | 468 | exports.__esModule = true; 469 | exports.generateOptions = generateOptions; 470 | 471 | function generateOptions(options, defaults) { 472 | if (typeof options === 'function') { 473 | defaults.callback = options; 474 | } else if (options) { 475 | for (var _name in options) { 476 | /* istanbul ignore else */ 477 | if (options.hasOwnProperty(_name)) { 478 | defaults[_name] = options[_name]; 479 | } 480 | } 481 | } 482 | return defaults; 483 | } 484 | //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy91dGlsL3BhcmFtcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFPLFNBQVMsZUFBZSxDQUFDLE9BQU8sRUFBRSxRQUFRLEVBQUU7QUFDakQsTUFBSSxPQUFPLE9BQU8sS0FBSyxVQUFVLEVBQUU7QUFDakMsWUFBUSxDQUFDLFFBQVEsR0FBRyxPQUFPLENBQUM7R0FDN0IsTUFBTSxJQUFJLE9BQU8sRUFBRTtBQUNsQixTQUFLLElBQUksS0FBSSxJQUFJLE9BQU8sRUFBRTs7QUFFeEIsVUFBSSxPQUFPLENBQUMsY0FBYyxDQUFDLEtBQUksQ0FBQyxFQUFFO0FBQ2hDLGdCQUFRLENBQUMsS0FBSSxDQUFDLEdBQUcsT0FBTyxDQUFDLEtBQUksQ0FBQyxDQUFDO09BQ2hDO0tBQ0Y7R0FDRjtBQUNELFNBQU8sUUFBUSxDQUFDO0NBQ2pCIiwiZmlsZSI6InBhcmFtcy5qcyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBmdW5jdGlvbiBnZW5lcmF0ZU9wdGlvbnMob3B0aW9ucywgZGVmYXVsdHMpIHtcbiAgaWYgKHR5cGVvZiBvcHRpb25zID09PSAnZnVuY3Rpb24nKSB7XG4gICAgZGVmYXVsdHMuY2FsbGJhY2sgPSBvcHRpb25zO1xuICB9IGVsc2UgaWYgKG9wdGlvbnMpIHtcbiAgICBmb3IgKGxldCBuYW1lIGluIG9wdGlvbnMpIHtcbiAgICAgIC8qIGlzdGFuYnVsIGlnbm9yZSBlbHNlICovXG4gICAgICBpZiAob3B0aW9ucy5oYXNPd25Qcm9wZXJ0eShuYW1lKSkge1xuICAgICAgICBkZWZhdWx0c1tuYW1lXSA9IG9wdGlvbnNbbmFtZV07XG4gICAgICB9XG4gICAgfVxuICB9XG4gIHJldHVybiBkZWZhdWx0cztcbn1cbiJdfQ== 485 | 486 | 487 | /***/ }, 488 | /* 5 */ 489 | /***/ function(module, exports, __webpack_require__) { 490 | 491 | 'use strict'; 492 | 493 | exports.__esModule = true; 494 | exports.diffLines = diffLines; 495 | exports.diffTrimmedLines = diffTrimmedLines; 496 | // istanbul ignore next 497 | 498 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { 'default': obj }; } 499 | 500 | var _base = __webpack_require__(1); 501 | 502 | var _base2 = _interopRequireDefault(_base); 503 | 504 | var _utilParams = __webpack_require__(4); 505 | 506 | var lineDiff = new _base2['default'](); 507 | exports.lineDiff = lineDiff; 508 | lineDiff.tokenize = function (value) { 509 | var retLines = [], 510 | linesAndNewlines = value.split(/(\n|\r\n)/); 511 | 512 | // Ignore the final empty token that occurs if the string ends with a new line 513 | if (!linesAndNewlines[linesAndNewlines.length - 1]) { 514 | linesAndNewlines.pop(); 515 | } 516 | 517 | // Merge the content and line separators into single tokens 518 | for (var i = 0; i < linesAndNewlines.length; i++) { 519 | var line = linesAndNewlines[i]; 520 | 521 | if (i % 2 && !this.options.newlineIsToken) { 522 | retLines[retLines.length - 1] += line; 523 | } else { 524 | if (this.options.ignoreWhitespace) { 525 | line = line.trim(); 526 | } 527 | retLines.push(line); 528 | } 529 | } 530 | 531 | return retLines; 532 | }; 533 | 534 | function diffLines(oldStr, newStr, callback) { 535 | return lineDiff.diff(oldStr, newStr, callback); 536 | } 537 | 538 | function diffTrimmedLines(oldStr, newStr, callback) { 539 | var options = _utilParams.generateOptions(callback, { ignoreWhitespace: true }); 540 | return lineDiff.diff(oldStr, newStr, options); 541 | } 542 | //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kaWZmL2xpbmUuanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7O29CQUFpQixRQUFROzs7OzBCQUNLLGdCQUFnQjs7QUFFdkMsSUFBTSxRQUFRLEdBQUcsdUJBQVUsQ0FBQzs7QUFDbkMsUUFBUSxDQUFDLFFBQVEsR0FBRyxVQUFTLEtBQUssRUFBRTtBQUNsQyxNQUFJLFFBQVEsR0FBRyxFQUFFO01BQ2IsZ0JBQWdCLEdBQUcsS0FBSyxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsQ0FBQzs7O0FBR2hELE1BQUksQ0FBQyxnQkFBZ0IsQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLEVBQUU7QUFDbEQsb0JBQWdCLENBQUMsR0FBRyxFQUFFLENBQUM7R0FDeEI7OztBQUdELE9BQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxnQkFBZ0IsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxFQUFFLEVBQUU7QUFDaEQsUUFBSSxJQUFJLEdBQUcsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLENBQUM7O0FBRS9CLFFBQUksQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsY0FBYyxFQUFFO0FBQ3pDLGNBQVEsQ0FBQyxRQUFRLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQyxJQUFJLElBQUksQ0FBQztLQUN2QyxNQUFNO0FBQ0wsVUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLGdCQUFnQixFQUFFO0FBQ2pDLFlBQUksR0FBRyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7T0FDcEI7QUFDRCxjQUFRLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO0tBQ3JCO0dBQ0Y7O0FBRUQsU0FBTyxRQUFRLENBQUM7Q0FDakIsQ0FBQzs7QUFFSyxTQUFTLFNBQVMsQ0FBQyxNQUFNLEVBQUUsTUFBTSxFQUFFLFFBQVEsRUFBRTtBQUFFLFNBQU8sUUFBUSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsTUFBTSxFQUFFLFFBQVEsQ0FBQyxDQUFDO0NBQUU7O0FBQ2hHLFNBQVMsZ0JBQWdCLENBQUMsTUFBTSxFQUFFLE1BQU0sRUFBRSxRQUFRLEVBQUU7QUFDekQsTUFBSSxPQUFPLEdBQUcsNEJBQWdCLFFBQVEsRUFBRSxFQUFDLGdCQUFnQixFQUFFLElBQUksRUFBQyxDQUFDLENBQUM7QUFDbEUsU0FBTyxRQUFRLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxNQUFNLEVBQUUsT0FBTyxDQUFDLENBQUM7Q0FDL0MiLCJmaWxlIjoibGluZS5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBEaWZmIGZyb20gJy4vYmFzZSc7XG5pbXBvcnQge2dlbmVyYXRlT3B0aW9uc30gZnJvbSAnLi4vdXRpbC9wYXJhbXMnO1xuXG5leHBvcnQgY29uc3QgbGluZURpZmYgPSBuZXcgRGlmZigpO1xubGluZURpZmYudG9rZW5pemUgPSBmdW5jdGlvbih2YWx1ZSkge1xuICBsZXQgcmV0TGluZXMgPSBbXSxcbiAgICAgIGxpbmVzQW5kTmV3bGluZXMgPSB2YWx1ZS5zcGxpdCgvKFxcbnxcXHJcXG4pLyk7XG5cbiAgLy8gSWdub3JlIHRoZSBmaW5hbCBlbXB0eSB0b2tlbiB0aGF0IG9jY3VycyBpZiB0aGUgc3RyaW5nIGVuZHMgd2l0aCBhIG5ldyBsaW5lXG4gIGlmICghbGluZXNBbmROZXdsaW5lc1tsaW5lc0FuZE5ld2xpbmVzLmxlbmd0aCAtIDFdKSB7XG4gICAgbGluZXNBbmROZXdsaW5lcy5wb3AoKTtcbiAgfVxuXG4gIC8vIE1lcmdlIHRoZSBjb250ZW50IGFuZCBsaW5lIHNlcGFyYXRvcnMgaW50byBzaW5nbGUgdG9rZW5zXG4gIGZvciAobGV0IGkgPSAwOyBpIDwgbGluZXNBbmROZXdsaW5lcy5sZW5ndGg7IGkrKykge1xuICAgIGxldCBsaW5lID0gbGluZXNBbmROZXdsaW5lc1tpXTtcblxuICAgIGlmIChpICUgMiAmJiAhdGhpcy5vcHRpb25zLm5ld2xpbmVJc1Rva2VuKSB7XG4gICAgICByZXRMaW5lc1tyZXRMaW5lcy5sZW5ndGggLSAxXSArPSBsaW5lO1xuICAgIH0gZWxzZSB7XG4gICAgICBpZiAodGhpcy5vcHRpb25zLmlnbm9yZVdoaXRlc3BhY2UpIHtcbiAgICAgICAgbGluZSA9IGxpbmUudHJpbSgpO1xuICAgICAgfVxuICAgICAgcmV0TGluZXMucHVzaChsaW5lKTtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gcmV0TGluZXM7XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gZGlmZkxpbmVzKG9sZFN0ciwgbmV3U3RyLCBjYWxsYmFjaykgeyByZXR1cm4gbGluZURpZmYuZGlmZihvbGRTdHIsIG5ld1N0ciwgY2FsbGJhY2spOyB9XG5leHBvcnQgZnVuY3Rpb24gZGlmZlRyaW1tZWRMaW5lcyhvbGRTdHIsIG5ld1N0ciwgY2FsbGJhY2spIHtcbiAgbGV0IG9wdGlvbnMgPSBnZW5lcmF0ZU9wdGlvbnMoY2FsbGJhY2ssIHtpZ25vcmVXaGl0ZXNwYWNlOiB0cnVlfSk7XG4gIHJldHVybiBsaW5lRGlmZi5kaWZmKG9sZFN0ciwgbmV3U3RyLCBvcHRpb25zKTtcbn1cbiJdfQ== 543 | 544 | 545 | /***/ }, 546 | /* 6 */ 547 | /***/ function(module, exports, __webpack_require__) { 548 | 549 | 'use strict'; 550 | 551 | exports.__esModule = true; 552 | exports.diffSentences = diffSentences; 553 | // istanbul ignore next 554 | 555 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { 'default': obj }; } 556 | 557 | var _base = __webpack_require__(1); 558 | 559 | var _base2 = _interopRequireDefault(_base); 560 | 561 | var sentenceDiff = new _base2['default'](); 562 | exports.sentenceDiff = sentenceDiff; 563 | sentenceDiff.tokenize = function (value) { 564 | return value.split(/(\S.+?[.!?])(?=\s+|$)/); 565 | }; 566 | 567 | function diffSentences(oldStr, newStr, callback) { 568 | return sentenceDiff.diff(oldStr, newStr, callback); 569 | } 570 | //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kaWZmL3NlbnRlbmNlLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7O29CQUFpQixRQUFROzs7O0FBR2xCLElBQU0sWUFBWSxHQUFHLHVCQUFVLENBQUM7O0FBQ3ZDLFlBQVksQ0FBQyxRQUFRLEdBQUcsVUFBUyxLQUFLLEVBQUU7QUFDdEMsU0FBTyxLQUFLLENBQUMsS0FBSyxDQUFDLHVCQUF1QixDQUFDLENBQUM7Q0FDN0MsQ0FBQzs7QUFFSyxTQUFTLGFBQWEsQ0FBQyxNQUFNLEVBQUUsTUFBTSxFQUFFLFFBQVEsRUFBRTtBQUFFLFNBQU8sWUFBWSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsTUFBTSxFQUFFLFFBQVEsQ0FBQyxDQUFDO0NBQUUiLCJmaWxlIjoic2VudGVuY2UuanMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgRGlmZiBmcm9tICcuL2Jhc2UnO1xuXG5cbmV4cG9ydCBjb25zdCBzZW50ZW5jZURpZmYgPSBuZXcgRGlmZigpO1xuc2VudGVuY2VEaWZmLnRva2VuaXplID0gZnVuY3Rpb24odmFsdWUpIHtcbiAgcmV0dXJuIHZhbHVlLnNwbGl0KC8oXFxTLis/Wy4hP10pKD89XFxzK3wkKS8pO1xufTtcblxuZXhwb3J0IGZ1bmN0aW9uIGRpZmZTZW50ZW5jZXMob2xkU3RyLCBuZXdTdHIsIGNhbGxiYWNrKSB7IHJldHVybiBzZW50ZW5jZURpZmYuZGlmZihvbGRTdHIsIG5ld1N0ciwgY2FsbGJhY2spOyB9XG4iXX0= 571 | 572 | 573 | /***/ }, 574 | /* 7 */ 575 | /***/ function(module, exports, __webpack_require__) { 576 | 577 | 'use strict'; 578 | 579 | exports.__esModule = true; 580 | exports.diffCss = diffCss; 581 | // istanbul ignore next 582 | 583 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { 'default': obj }; } 584 | 585 | var _base = __webpack_require__(1); 586 | 587 | var _base2 = _interopRequireDefault(_base); 588 | 589 | var cssDiff = new _base2['default'](); 590 | exports.cssDiff = cssDiff; 591 | cssDiff.tokenize = function (value) { 592 | return value.split(/([{}:;,]|\s+)/); 593 | }; 594 | 595 | function diffCss(oldStr, newStr, callback) { 596 | return cssDiff.diff(oldStr, newStr, callback); 597 | } 598 | //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kaWZmL2Nzcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7OztvQkFBaUIsUUFBUTs7OztBQUVsQixJQUFNLE9BQU8sR0FBRyx1QkFBVSxDQUFDOztBQUNsQyxPQUFPLENBQUMsUUFBUSxHQUFHLFVBQVMsS0FBSyxFQUFFO0FBQ2pDLFNBQU8sS0FBSyxDQUFDLEtBQUssQ0FBQyxlQUFlLENBQUMsQ0FBQztDQUNyQyxDQUFDOztBQUVLLFNBQVMsT0FBTyxDQUFDLE1BQU0sRUFBRSxNQUFNLEVBQUUsUUFBUSxFQUFFO0FBQUUsU0FBTyxPQUFPLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxNQUFNLEVBQUUsUUFBUSxDQUFDLENBQUM7Q0FBRSIsImZpbGUiOiJjc3MuanMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgRGlmZiBmcm9tICcuL2Jhc2UnO1xuXG5leHBvcnQgY29uc3QgY3NzRGlmZiA9IG5ldyBEaWZmKCk7XG5jc3NEaWZmLnRva2VuaXplID0gZnVuY3Rpb24odmFsdWUpIHtcbiAgcmV0dXJuIHZhbHVlLnNwbGl0KC8oW3t9OjssXXxcXHMrKS8pO1xufTtcblxuZXhwb3J0IGZ1bmN0aW9uIGRpZmZDc3Mob2xkU3RyLCBuZXdTdHIsIGNhbGxiYWNrKSB7IHJldHVybiBjc3NEaWZmLmRpZmYob2xkU3RyLCBuZXdTdHIsIGNhbGxiYWNrKTsgfVxuIl19 599 | 600 | 601 | /***/ }, 602 | /* 8 */ 603 | /***/ function(module, exports, __webpack_require__) { 604 | 605 | 'use strict'; 606 | 607 | exports.__esModule = true; 608 | exports.diffJson = diffJson; 609 | exports.canonicalize = canonicalize; 610 | // istanbul ignore next 611 | 612 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { 'default': obj }; } 613 | 614 | var _base = __webpack_require__(1); 615 | 616 | var _base2 = _interopRequireDefault(_base); 617 | 618 | var _line = __webpack_require__(5); 619 | 620 | var objectPrototypeToString = Object.prototype.toString; 621 | 622 | var jsonDiff = new _base2['default'](); 623 | exports.jsonDiff = jsonDiff; 624 | // Discriminate between two lines of pretty-printed, serialized JSON where one of them has a 625 | // dangling comma and the other doesn't. Turns out including the dangling comma yields the nicest output: 626 | jsonDiff.useLongestToken = true; 627 | 628 | jsonDiff.tokenize = _line.lineDiff.tokenize; 629 | jsonDiff.castInput = function (value) { 630 | return typeof value === 'string' ? value : JSON.stringify(canonicalize(value), undefined, ' '); 631 | }; 632 | jsonDiff.equals = function (left, right) { 633 | return _base2['default'].prototype.equals(left.replace(/,([\r\n])/g, '$1'), right.replace(/,([\r\n])/g, '$1')); 634 | }; 635 | 636 | function diffJson(oldObj, newObj, callback) { 637 | return jsonDiff.diff(oldObj, newObj, callback); 638 | } 639 | 640 | // This function handles the presence of circular references by bailing out when encountering an 641 | // object that is already on the "stack" of items being processed. 642 | 643 | function canonicalize(obj, stack, replacementStack) { 644 | stack = stack || []; 645 | replacementStack = replacementStack || []; 646 | 647 | var i = undefined; 648 | 649 | for (i = 0; i < stack.length; i += 1) { 650 | if (stack[i] === obj) { 651 | return replacementStack[i]; 652 | } 653 | } 654 | 655 | var canonicalizedObj = undefined; 656 | 657 | if ('[object Array]' === objectPrototypeToString.call(obj)) { 658 | stack.push(obj); 659 | canonicalizedObj = new Array(obj.length); 660 | replacementStack.push(canonicalizedObj); 661 | for (i = 0; i < obj.length; i += 1) { 662 | canonicalizedObj[i] = canonicalize(obj[i], stack, replacementStack); 663 | } 664 | stack.pop(); 665 | replacementStack.pop(); 666 | } else if (typeof obj === 'object' && obj !== null) { 667 | stack.push(obj); 668 | canonicalizedObj = {}; 669 | replacementStack.push(canonicalizedObj); 670 | var sortedKeys = [], 671 | key = undefined; 672 | for (key in obj) { 673 | /* istanbul ignore else */ 674 | if (obj.hasOwnProperty(key)) { 675 | sortedKeys.push(key); 676 | } 677 | } 678 | sortedKeys.sort(); 679 | for (i = 0; i < sortedKeys.length; i += 1) { 680 | key = sortedKeys[i]; 681 | canonicalizedObj[key] = canonicalize(obj[key], stack, replacementStack); 682 | } 683 | stack.pop(); 684 | replacementStack.pop(); 685 | } else { 686 | canonicalizedObj = obj; 687 | } 688 | return canonicalizedObj; 689 | } 690 | //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kaWZmL2pzb24uanMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7O29CQUFpQixRQUFROzs7O29CQUNGLFFBQVE7O0FBRS9CLElBQU0sdUJBQXVCLEdBQUcsTUFBTSxDQUFDLFNBQVMsQ0FBQyxRQUFRLENBQUM7O0FBR25ELElBQU0sUUFBUSxHQUFHLHVCQUFVLENBQUM7Ozs7QUFHbkMsUUFBUSxDQUFDLGVBQWUsR0FBRyxJQUFJLENBQUM7O0FBRWhDLFFBQVEsQ0FBQyxRQUFRLEdBQUcsZUFBUyxRQUFRLENBQUM7QUFDdEMsUUFBUSxDQUFDLFNBQVMsR0FBRyxVQUFTLEtBQUssRUFBRTtBQUNuQyxTQUFPLE9BQU8sS0FBSyxLQUFLLFFBQVEsR0FBRyxLQUFLLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxZQUFZLENBQUMsS0FBSyxDQUFDLEVBQUUsU0FBUyxFQUFFLElBQUksQ0FBQyxDQUFDO0NBQ2pHLENBQUM7QUFDRixRQUFRLENBQUMsTUFBTSxHQUFHLFVBQVMsSUFBSSxFQUFFLEtBQUssRUFBRTtBQUN0QyxTQUFPLGtCQUFLLFNBQVMsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxZQUFZLEVBQUUsSUFBSSxDQUFDLEVBQUUsS0FBSyxDQUFDLE9BQU8sQ0FBQyxZQUFZLEVBQUUsSUFBSSxDQUFDLENBQUMsQ0FBQztDQUNuRyxDQUFDOztBQUVLLFNBQVMsUUFBUSxDQUFDLE1BQU0sRUFBRSxNQUFNLEVBQUUsUUFBUSxFQUFFO0FBQUUsU0FBTyxRQUFRLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxNQUFNLEVBQUUsUUFBUSxDQUFDLENBQUM7Q0FBRTs7Ozs7QUFLL0YsU0FBUyxZQUFZLENBQUMsR0FBRyxFQUFFLEtBQUssRUFBRSxnQkFBZ0IsRUFBRTtBQUN6RCxPQUFLLEdBQUcsS0FBSyxJQUFJLEVBQUUsQ0FBQztBQUNwQixrQkFBZ0IsR0FBRyxnQkFBZ0IsSUFBSSxFQUFFLENBQUM7O0FBRTFDLE1BQUksQ0FBQyxZQUFBLENBQUM7O0FBRU4sT0FBSyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxLQUFLLENBQUMsTUFBTSxFQUFFLENBQUMsSUFBSSxDQUFDLEVBQUU7QUFDcEMsUUFBSSxLQUFLLENBQUMsQ0FBQyxDQUFDLEtBQUssR0FBRyxFQUFFO0FBQ3BCLGFBQU8sZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLENBQUM7S0FDNUI7R0FDRjs7QUFFRCxNQUFJLGdCQUFnQixZQUFBLENBQUM7O0FBRXJCLE1BQUksZ0JBQWdCLEtBQUssdUJBQXVCLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFO0FBQzFELFNBQUssQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7QUFDaEIsb0JBQWdCLEdBQUcsSUFBSSxLQUFLLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxDQUFDO0FBQ3pDLG9CQUFnQixDQUFDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO0FBQ3hDLFNBQUssQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsR0FBRyxDQUFDLE1BQU0sRUFBRSxDQUFDLElBQUksQ0FBQyxFQUFFO0FBQ2xDLHNCQUFnQixDQUFDLENBQUMsQ0FBQyxHQUFHLFlBQVksQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsS0FBSyxFQUFFLGdCQUFnQixDQUFDLENBQUM7S0FDckU7QUFDRCxTQUFLLENBQUMsR0FBRyxFQUFFLENBQUM7QUFDWixvQkFBZ0IsQ0FBQyxHQUFHLEVBQUUsQ0FBQztHQUN4QixNQUFNLElBQUksT0FBTyxHQUFHLEtBQUssUUFBUSxJQUFJLEdBQUcsS0FBSyxJQUFJLEVBQUU7QUFDbEQsU0FBSyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztBQUNoQixvQkFBZ0IsR0FBRyxFQUFFLENBQUM7QUFDdEIsb0JBQWdCLENBQUMsSUFBSSxDQUFDLGdCQUFnQixDQUFDLENBQUM7QUFDeEMsUUFBSSxVQUFVLEdBQUcsRUFBRTtRQUNmLEdBQUcsWUFBQSxDQUFDO0FBQ1IsU0FBSyxHQUFHLElBQUksR0FBRyxFQUFFOztBQUVmLFVBQUksR0FBRyxDQUFDLGNBQWMsQ0FBQyxHQUFHLENBQUMsRUFBRTtBQUMzQixrQkFBVSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztPQUN0QjtLQUNGO0FBQ0QsY0FBVSxDQUFDLElBQUksRUFBRSxDQUFDO0FBQ2xCLFNBQUssQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsVUFBVSxDQUFDLE1BQU0sRUFBRSxDQUFDLElBQUksQ0FBQyxFQUFFO0FBQ3pDLFNBQUcsR0FBRyxVQUFVLENBQUMsQ0FBQyxDQUFDLENBQUM7QUFDcEIsc0JBQWdCLENBQUMsR0FBRyxDQUFDLEdBQUcsWUFBWSxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsRUFBRSxLQUFLLEVBQUUsZ0JBQWdCLENBQUMsQ0FBQztLQUN6RTtBQUNELFNBQUssQ0FBQyxHQUFHLEVBQUUsQ0FBQztBQUNaLG9CQUFnQixDQUFDLEdBQUcsRUFBRSxDQUFDO0dBQ3hCLE1BQU07QUFDTCxvQkFBZ0IsR0FBRyxHQUFHLENBQUM7R0FDeEI7QUFDRCxTQUFPLGdCQUFnQixDQUFDO0NBQ3pCIiwiZmlsZSI6Impzb24uanMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgRGlmZiBmcm9tICcuL2Jhc2UnO1xuaW1wb3J0IHtsaW5lRGlmZn0gZnJvbSAnLi9saW5lJztcblxuY29uc3Qgb2JqZWN0UHJvdG90eXBlVG9TdHJpbmcgPSBPYmplY3QucHJvdG90eXBlLnRvU3RyaW5nO1xuXG5cbmV4cG9ydCBjb25zdCBqc29uRGlmZiA9IG5ldyBEaWZmKCk7XG4vLyBEaXNjcmltaW5hdGUgYmV0d2VlbiB0d28gbGluZXMgb2YgcHJldHR5LXByaW50ZWQsIHNlcmlhbGl6ZWQgSlNPTiB3aGVyZSBvbmUgb2YgdGhlbSBoYXMgYVxuLy8gZGFuZ2xpbmcgY29tbWEgYW5kIHRoZSBvdGhlciBkb2Vzbid0LiBUdXJucyBvdXQgaW5jbHVkaW5nIHRoZSBkYW5nbGluZyBjb21tYSB5aWVsZHMgdGhlIG5pY2VzdCBvdXRwdXQ6XG5qc29uRGlmZi51c2VMb25nZXN0VG9rZW4gPSB0cnVlO1xuXG5qc29uRGlmZi50b2tlbml6ZSA9IGxpbmVEaWZmLnRva2VuaXplO1xuanNvbkRpZmYuY2FzdElucHV0ID0gZnVuY3Rpb24odmFsdWUpIHtcbiAgcmV0dXJuIHR5cGVvZiB2YWx1ZSA9PT0gJ3N0cmluZycgPyB2YWx1ZSA6IEpTT04uc3RyaW5naWZ5KGNhbm9uaWNhbGl6ZSh2YWx1ZSksIHVuZGVmaW5lZCwgJyAgJyk7XG59O1xuanNvbkRpZmYuZXF1YWxzID0gZnVuY3Rpb24obGVmdCwgcmlnaHQpIHtcbiAgcmV0dXJuIERpZmYucHJvdG90eXBlLmVxdWFscyhsZWZ0LnJlcGxhY2UoLywoW1xcclxcbl0pL2csICckMScpLCByaWdodC5yZXBsYWNlKC8sKFtcXHJcXG5dKS9nLCAnJDEnKSk7XG59O1xuXG5leHBvcnQgZnVuY3Rpb24gZGlmZkpzb24ob2xkT2JqLCBuZXdPYmosIGNhbGxiYWNrKSB7IHJldHVybiBqc29uRGlmZi5kaWZmKG9sZE9iaiwgbmV3T2JqLCBjYWxsYmFjayk7IH1cblxuXG4vLyBUaGlzIGZ1bmN0aW9uIGhhbmRsZXMgdGhlIHByZXNlbmNlIG9mIGNpcmN1bGFyIHJlZmVyZW5jZXMgYnkgYmFpbGluZyBvdXQgd2hlbiBlbmNvdW50ZXJpbmcgYW5cbi8vIG9iamVjdCB0aGF0IGlzIGFscmVhZHkgb24gdGhlIFwic3RhY2tcIiBvZiBpdGVtcyBiZWluZyBwcm9jZXNzZWQuXG5leHBvcnQgZnVuY3Rpb24gY2Fub25pY2FsaXplKG9iaiwgc3RhY2ssIHJlcGxhY2VtZW50U3RhY2spIHtcbiAgc3RhY2sgPSBzdGFjayB8fCBbXTtcbiAgcmVwbGFjZW1lbnRTdGFjayA9IHJlcGxhY2VtZW50U3RhY2sgfHwgW107XG5cbiAgbGV0IGk7XG5cbiAgZm9yIChpID0gMDsgaSA8IHN0YWNrLmxlbmd0aDsgaSArPSAxKSB7XG4gICAgaWYgKHN0YWNrW2ldID09PSBvYmopIHtcbiAgICAgIHJldHVybiByZXBsYWNlbWVudFN0YWNrW2ldO1xuICAgIH1cbiAgfVxuXG4gIGxldCBjYW5vbmljYWxpemVkT2JqO1xuXG4gIGlmICgnW29iamVjdCBBcnJheV0nID09PSBvYmplY3RQcm90b3R5cGVUb1N0cmluZy5jYWxsKG9iaikpIHtcbiAgICBzdGFjay5wdXNoKG9iaik7XG4gICAgY2Fub25pY2FsaXplZE9iaiA9IG5ldyBBcnJheShvYmoubGVuZ3RoKTtcbiAgICByZXBsYWNlbWVudFN0YWNrLnB1c2goY2Fub25pY2FsaXplZE9iaik7XG4gICAgZm9yIChpID0gMDsgaSA8IG9iai5sZW5ndGg7IGkgKz0gMSkge1xuICAgICAgY2Fub25pY2FsaXplZE9ialtpXSA9IGNhbm9uaWNhbGl6ZShvYmpbaV0sIHN0YWNrLCByZXBsYWNlbWVudFN0YWNrKTtcbiAgICB9XG4gICAgc3RhY2sucG9wKCk7XG4gICAgcmVwbGFjZW1lbnRTdGFjay5wb3AoKTtcbiAgfSBlbHNlIGlmICh0eXBlb2Ygb2JqID09PSAnb2JqZWN0JyAmJiBvYmogIT09IG51bGwpIHtcbiAgICBzdGFjay5wdXNoKG9iaik7XG4gICAgY2Fub25pY2FsaXplZE9iaiA9IHt9O1xuICAgIHJlcGxhY2VtZW50U3RhY2sucHVzaChjYW5vbmljYWxpemVkT2JqKTtcbiAgICBsZXQgc29ydGVkS2V5cyA9IFtdLFxuICAgICAgICBrZXk7XG4gICAgZm9yIChrZXkgaW4gb2JqKSB7XG4gICAgICAvKiBpc3RhbmJ1bCBpZ25vcmUgZWxzZSAqL1xuICAgICAgaWYgKG9iai5oYXNPd25Qcm9wZXJ0eShrZXkpKSB7XG4gICAgICAgIHNvcnRlZEtleXMucHVzaChrZXkpO1xuICAgICAgfVxuICAgIH1cbiAgICBzb3J0ZWRLZXlzLnNvcnQoKTtcbiAgICBmb3IgKGkgPSAwOyBpIDwgc29ydGVkS2V5cy5sZW5ndGg7IGkgKz0gMSkge1xuICAgICAga2V5ID0gc29ydGVkS2V5c1tpXTtcbiAgICAgIGNhbm9uaWNhbGl6ZWRPYmpba2V5XSA9IGNhbm9uaWNhbGl6ZShvYmpba2V5XSwgc3RhY2ssIHJlcGxhY2VtZW50U3RhY2spO1xuICAgIH1cbiAgICBzdGFjay5wb3AoKTtcbiAgICByZXBsYWNlbWVudFN0YWNrLnBvcCgpO1xuICB9IGVsc2Uge1xuICAgIGNhbm9uaWNhbGl6ZWRPYmogPSBvYmo7XG4gIH1cbiAgcmV0dXJuIGNhbm9uaWNhbGl6ZWRPYmo7XG59XG4iXX0= 691 | 692 | 693 | /***/ }, 694 | /* 9 */ 695 | /***/ function(module, exports, __webpack_require__) { 696 | 697 | 'use strict'; 698 | 699 | exports.__esModule = true; 700 | exports.applyPatch = applyPatch; 701 | exports.applyPatches = applyPatches; 702 | 703 | var _parse = __webpack_require__(10); 704 | 705 | function applyPatch(source, uniDiff) { 706 | var options = arguments.length <= 2 || arguments[2] === undefined ? {} : arguments[2]; 707 | 708 | if (typeof uniDiff === 'string') { 709 | uniDiff = _parse.parsePatch(uniDiff); 710 | } 711 | 712 | if (Array.isArray(uniDiff)) { 713 | if (uniDiff.length > 1) { 714 | throw new Error('applyPatch only works with a single input.'); 715 | } 716 | 717 | uniDiff = uniDiff[0]; 718 | } 719 | 720 | // Apply the diff to the input 721 | var lines = source.split('\n'), 722 | hunks = uniDiff.hunks, 723 | compareLine = options.compareLine || function (lineNumber, line, operation, patchContent) { 724 | return line === patchContent; 725 | }, 726 | errorCount = 0, 727 | fuzzFactor = options.fuzzFactor || 0, 728 | removeEOFNL = undefined, 729 | addEOFNL = undefined; 730 | 731 | for (var i = 0; i < hunks.length; i++) { 732 | var hunk = hunks[i], 733 | toPos = hunk.newStart - 1; 734 | 735 | // Sanity check the input string. Bail if we don't match. 736 | for (var j = 0; j < hunk.lines.length; j++) { 737 | var line = hunk.lines[j], 738 | operation = line[0], 739 | content = line.substr(1); 740 | if (operation === ' ' || operation === '-') { 741 | // Context sanity check 742 | if (!compareLine(toPos + 1, lines[toPos], operation, content)) { 743 | errorCount++; 744 | 745 | if (errorCount > fuzzFactor) { 746 | return false; 747 | } 748 | } 749 | } 750 | 751 | if (operation === ' ') { 752 | toPos++; 753 | } else if (operation === '-') { 754 | lines.splice(toPos, 1); 755 | /* istanbul ignore else */ 756 | } else if (operation === '+') { 757 | lines.splice(toPos, 0, content); 758 | toPos++; 759 | } else if (operation === '\\') { 760 | var previousOperation = hunk.lines[j - 1] ? hunk.lines[j - 1][0] : null; 761 | if (previousOperation === '+') { 762 | removeEOFNL = true; 763 | } else if (previousOperation === '-') { 764 | addEOFNL = true; 765 | } 766 | } 767 | } 768 | } 769 | 770 | // Handle EOFNL insertion/removal 771 | if (removeEOFNL) { 772 | while (!lines[lines.length - 1]) { 773 | lines.pop(); 774 | } 775 | } else if (addEOFNL) { 776 | lines.push(''); 777 | } 778 | return lines.join('\n'); 779 | } 780 | 781 | // Wrapper that supports multiple file patches via callbacks. 782 | 783 | function applyPatches(uniDiff, options) { 784 | if (typeof uniDiff === 'string') { 785 | uniDiff = _parse.parsePatch(uniDiff); 786 | } 787 | 788 | var currentIndex = 0; 789 | function processIndex() { 790 | var index = uniDiff[currentIndex++]; 791 | if (!index) { 792 | options.complete(); 793 | } 794 | 795 | options.loadFile(index, function (err, data) { 796 | if (err) { 797 | return options.complete(err); 798 | } 799 | 800 | var updatedContent = applyPatch(data, index, options); 801 | options.patched(index, updatedContent); 802 | 803 | setTimeout(processIndex, 0); 804 | }); 805 | } 806 | processIndex(); 807 | } 808 | //# sourceMappingURL=data:application/json;base64,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dkMsZ0JBQVUsQ0FBQyxZQUFZLEVBQUUsQ0FBQyxDQUFDLENBQUM7S0FDN0IsQ0FBQyxDQUFDO0dBQ0o7QUFDRCxjQUFZLEVBQUUsQ0FBQztDQUNoQiIsImZpbGUiOiJhcHBseS5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7cGFyc2VQYXRjaH0gZnJvbSAnLi9wYXJzZSc7XG5cbmV4cG9ydCBmdW5jdGlvbiBhcHBseVBhdGNoKHNvdXJjZSwgdW5pRGlmZiwgb3B0aW9ucyA9IHt9KSB7XG4gIGlmICh0eXBlb2YgdW5pRGlmZiA9PT0gJ3N0cmluZycpIHtcbiAgICB1bmlEaWZmID0gcGFyc2VQYXRjaCh1bmlEaWZmKTtcbiAgfVxuXG4gIGlmIChBcnJheS5pc0FycmF5KHVuaURpZmYpKSB7XG4gICAgaWYgKHVuaURpZmYubGVuZ3RoID4gMSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCdhcHBseVBhdGNoIG9ubHkgd29ya3Mgd2l0aCBhIHNpbmdsZSBpbnB1dC4nKTtcbiAgICB9XG5cbiAgICB1bmlEaWZmID0gdW5pRGlmZlswXTtcbiAgfVxuXG4gIC8vIEFwcGx5IHRoZSBkaWZmIHRvIHRoZSBpbnB1dFxuICBsZXQgbGluZXMgPSBzb3VyY2Uuc3BsaXQoJ1xcbicpLFxuICAgICAgaHVua3MgPSB1bmlEaWZmLmh1bmtzLFxuXG4gICAgICBjb21wYXJlTGluZSA9IG9wdGlvbnMuY29tcGFyZUxpbmUgfHwgKChsaW5lTnVtYmVyLCBsaW5lLCBvcGVyYXRpb24sIHBhdGNoQ29udGVudCkgPT4gbGluZSA9PT0gcGF0Y2hDb250ZW50KSxcbiAgICAgIGVycm9yQ291bnQgPSAwLFxuICAgICAgZnV6ekZhY3RvciA9IG9wdGlvbnMuZnV6ekZhY3RvciB8fCAwLFxuXG4gICAgICByZW1vdmVFT0ZOTCxcbiAgICAgIGFkZEVPRk5MO1xuXG4gIGZvciAobGV0IGkgPSAwOyBpIDwgaHVua3MubGVuZ3RoOyBpKyspIHtcbiAgICBsZXQgaHVuayA9IGh1bmtzW2ldLFxuICAgICAgICB0b1BvcyA9IGh1bmsubmV3U3RhcnQgLSAxO1xuXG4gICAgLy8gU2FuaXR5IGNoZWNrIHRoZSBpbnB1dCBzdHJpbmcuIEJhaWwgaWYgd2UgZG9uJ3QgbWF0Y2guXG4gICAgZm9yIChsZXQgaiA9IDA7IGogPCBodW5rLmxpbmVzLmxlbmd0aDsgaisrKSB7XG4gICAgICBsZXQgbGluZSA9IGh1bmsubGluZXNbal0sXG4gICAgICAgICAgb3BlcmF0aW9uID0gbGluZVswXSxcbiAgICAgICAgICBjb250ZW50ID0gbGluZS5zdWJzdHIoMSk7XG4gICAgICBpZiAob3BlcmF0aW9uID09PSAnICcgfHwgb3BlcmF0aW9uID09PSAnLScpIHtcbiAgICAgICAgLy8gQ29udGV4dCBzYW5pdHkgY2hlY2tcbiAgICAgICAgaWYgKCFjb21wYXJlTGluZSh0b1BvcyArIDEsIGxpbmVzW3RvUG9zXSwgb3BlcmF0aW9uLCBjb250ZW50KSkge1xuICAgICAgICAgIGVycm9yQ291bnQrKztcblxuICAgICAgICAgIGlmIChlcnJvckNvdW50ID4gZnV6ekZhY3Rvcikge1xuICAgICAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBpZiAob3BlcmF0aW9uID09PSAnICcpIHtcbiAgICAgICAgdG9Qb3MrKztcbiAgICAgIH0gZWxzZSBpZiAob3BlcmF0aW9uID09PSAnLScpIHtcbiAgICAgICAgbGluZXMuc3BsaWNlKHRvUG9zLCAxKTtcbiAgICAgIC8qIGlzdGFuYnVsIGlnbm9yZSBlbHNlICovXG4gICAgICB9IGVsc2UgaWYgKG9wZXJhdGlvbiA9PT0gJysnKSB7XG4gICAgICAgIGxpbmVzLnNwbGljZSh0b1BvcywgMCwgY29udGVudCk7XG4gICAgICAgIHRvUG9zKys7XG4gICAgICB9IGVsc2UgaWYgKG9wZXJhdGlvbiA9PT0gJ1xcXFwnKSB7XG4gICAgICAgIGxldCBwcmV2aW91c09wZXJhdGlvbiA9IGh1bmsubGluZXNbaiAtIDFdID8gaHVuay5saW5lc1tqIC0gMV1bMF0gOiBudWxsO1xuICAgICAgICBpZiAocHJldmlvdXNPcGVyYXRpb24gPT09ICcrJykge1xuICAgICAgICAgIHJlbW92ZUVPRk5MID0gdHJ1ZTtcbiAgICAgICAgfSBlbHNlIGlmIChwcmV2aW91c09wZXJhdGlvbiA9PT0gJy0nKSB7XG4gICAgICAgICAgYWRkRU9GTkwgPSB0cnVlO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgLy8gSGFuZGxlIEVPRk5MIGluc2VydGlvbi9yZW1vdmFsXG4gIGlmIChyZW1vdmVFT0ZOTCkge1xuICAgIHdoaWxlICghbGluZXNbbGluZXMubGVuZ3RoIC0gMV0pIHtcbiAgICAgIGxpbmVzLnBvcCgpO1xuICAgIH1cbiAgfSBlbHNlIGlmIChhZGRFT0ZOTCkge1xuICAgIGxpbmVzLnB1c2goJycpO1xuICB9XG4gIHJldHVybiBsaW5lcy5qb2luKCdcXG4nKTtcbn1cblxuLy8gV3JhcHBlciB0aGF0IHN1cHBvcnRzIG11bHRpcGxlIGZpbGUgcGF0Y2hlcyB2aWEgY2FsbGJhY2tzLlxuZXhwb3J0IGZ1bmN0aW9uIGFwcGx5UGF0Y2hlcyh1bmlEaWZmLCBvcHRpb25zKSB7XG4gIGlmICh0eXBlb2YgdW5pRGlmZiA9PT0gJ3N0cmluZycpIHtcbiAgICB1bmlEaWZmID0gcGFyc2VQYXRjaCh1bmlEaWZmKTtcbiAgfVxuXG4gIGxldCBjdXJyZW50SW5kZXggPSAwO1xuICBmdW5jdGlvbiBwcm9jZXNzSW5kZXgoKSB7XG4gICAgbGV0IGluZGV4ID0gdW5pRGlmZltjdXJyZW50SW5kZXgrK107XG4gICAgaWYgKCFpbmRleCkge1xuICAgICAgb3B0aW9ucy5jb21wbGV0ZSgpO1xuICAgIH1cblxuICAgIG9wdGlvbnMubG9hZEZpbGUoaW5kZXgsIGZ1bmN0aW9uKGVyciwgZGF0YSkge1xuICAgICAgaWYgKGVycikge1xuICAgICAgICByZXR1cm4gb3B0aW9ucy5jb21wbGV0ZShlcnIpO1xuICAgICAgfVxuXG4gICAgICBsZXQgdXBkYXRlZENvbnRlbnQgPSBhcHBseVBhdGNoKGRhdGEsIGluZGV4LCBvcHRpb25zKTtcbiAgICAgIG9wdGlvbnMucGF0Y2hlZChpbmRleCwgdXBkYXRlZENvbnRlbnQpO1xuXG4gICAgICBzZXRUaW1lb3V0KHByb2Nlc3NJbmRleCwgMCk7XG4gICAgfSk7XG4gIH1cbiAgcHJvY2Vzc0luZGV4KCk7XG59XG4iXX0= 809 | 810 | 811 | /***/ }, 812 | /* 10 */ 813 | /***/ function(module, exports) { 814 | 815 | 'use strict'; 816 | 817 | exports.__esModule = true; 818 | exports.parsePatch = parsePatch; 819 | 820 | function parsePatch(uniDiff) { 821 | var options = arguments.length <= 1 || arguments[1] === undefined ? {} : arguments[1]; 822 | 823 | var diffstr = uniDiff.split('\n'), 824 | list = [], 825 | i = 0; 826 | 827 | function parseIndex() { 828 | var index = {}; 829 | list.push(index); 830 | 831 | // Ignore any leading junk 832 | while (i < diffstr.length) { 833 | if (/^Index:/.test(diffstr[i]) || /^@@/.test(diffstr[i])) { 834 | break; 835 | } 836 | i++; 837 | } 838 | 839 | var header = /^Index: (.*)/.exec(diffstr[i]); 840 | if (header) { 841 | index.index = header[1]; 842 | i++; 843 | 844 | if (/^===/.test(diffstr[i])) { 845 | i++; 846 | } 847 | 848 | parseFileHeader(index); 849 | parseFileHeader(index); 850 | } else { 851 | // Ignore erant header components that might occur at the start of the file 852 | parseFileHeader({}); 853 | parseFileHeader({}); 854 | } 855 | 856 | index.hunks = []; 857 | 858 | while (i < diffstr.length) { 859 | if (/^Index:/.test(diffstr[i])) { 860 | break; 861 | } else if (/^@@/.test(diffstr[i])) { 862 | index.hunks.push(parseHunk()); 863 | } else if (diffstr[i] && options.strict) { 864 | // Ignore unexpected content unless in strict mode 865 | throw new Error('Unknown line ' + (i + 1) + ' ' + JSON.stringify(diffstr[i])); 866 | } else { 867 | i++; 868 | } 869 | } 870 | } 871 | 872 | // Parses the --- and +++ headers, if none are found, no lines 873 | // are consumed. 874 | function parseFileHeader(index) { 875 | var fileHeader = /^(\-\-\-|\+\+\+)\s(\S+)\s?(.*)/.exec(diffstr[i]); 876 | if (fileHeader) { 877 | var keyPrefix = fileHeader[1] === '---' ? 'old' : 'new'; 878 | index[keyPrefix + 'FileName'] = fileHeader[2]; 879 | index[keyPrefix + 'Header'] = fileHeader[3]; 880 | 881 | i++; 882 | } 883 | } 884 | 885 | // Parses a hunk 886 | // This assumes that we are at the start of a hunk. 887 | function parseHunk() { 888 | var chunkHeaderIndex = i, 889 | chunkHeaderLine = diffstr[i++], 890 | chunkHeader = chunkHeaderLine.split(/@@ -(\d+)(?:,(\d+))? \+(\d+)(?:,(\d+))? @@/); 891 | 892 | var hunk = { 893 | oldStart: +chunkHeader[1], 894 | oldLines: +chunkHeader[2] || 1, 895 | newStart: +chunkHeader[3], 896 | newLines: +chunkHeader[4] || 1, 897 | lines: [] 898 | }; 899 | 900 | var addCount = 0, 901 | removeCount = 0; 902 | for (; i < diffstr.length; i++) { 903 | var operation = diffstr[i][0]; 904 | 905 | if (operation === '+' || operation === '-' || operation === ' ' || operation === '\\') { 906 | hunk.lines.push(diffstr[i]); 907 | 908 | if (operation === '+') { 909 | addCount++; 910 | } else if (operation === '-') { 911 | removeCount++; 912 | } else if (operation === ' ') { 913 | addCount++; 914 | removeCount++; 915 | } 916 | } else { 917 | break; 918 | } 919 | } 920 | 921 | // Handle the empty block count case 922 | if (!addCount && hunk.newLines === 1) { 923 | hunk.newLines = 0; 924 | } 925 | if (!removeCount && hunk.oldLines === 1) { 926 | hunk.oldLines = 0; 927 | } 928 | 929 | // Perform optional sanity checking 930 | if (options.strict) { 931 | if (addCount !== hunk.newLines) { 932 | throw new Error('Added line count did not match for hunk at line ' + (chunkHeaderIndex + 1)); 933 | } 934 | if (removeCount !== hunk.oldLines) { 935 | throw new Error('Removed line count did not match for hunk at line ' + (chunkHeaderIndex + 1)); 936 | } 937 | } 938 | 939 | return hunk; 940 | } 941 | 942 | while (i < diffstr.length) { 943 | parseIndex(); 944 | } 945 | 946 | return list; 947 | } 948 | //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/patch/parse.js"],"names":[],"mappings":"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","file":"parse.js","sourcesContent":["export function parsePatch(uniDiff, options = {}) {\n  let diffstr = uniDiff.split('\\n'),\n      list = [],\n      i = 0;\n\n  function parseIndex() {\n    let index = {};\n    list.push(index);\n\n    // Ignore any leading junk\n    while (i < diffstr.length) {\n      if ((/^Index:/.test(diffstr[i])) || (/^@@/.test(diffstr[i]))) {\n        break;\n      }\n      i++;\n    }\n\n    let header = (/^Index: (.*)/.exec(diffstr[i]));\n    if (header) {\n      index.index = header[1];\n      i++;\n\n      if (/^===/.test(diffstr[i])) {\n        i++;\n      }\n\n      parseFileHeader(index);\n      parseFileHeader(index);\n    } else {\n      // Ignore erant header components that might occur at the start of the file\n      parseFileHeader({});\n      parseFileHeader({});\n    }\n\n    index.hunks = [];\n\n    while (i < diffstr.length) {\n      if (/^Index:/.test(diffstr[i])) {\n        break;\n      } else if (/^@@/.test(diffstr[i])) {\n        index.hunks.push(parseHunk());\n      } else if (diffstr[i] && options.strict) {\n        // Ignore unexpected content unless in strict mode\n        throw new Error('Unknown line ' + (i + 1) + ' ' + JSON.stringify(diffstr[i]));\n      } else {\n        i++;\n      }\n    }\n  }\n\n  // Parses the --- and +++ headers, if none are found, no lines\n  // are consumed.\n  function parseFileHeader(index) {\n    let fileHeader = (/^(\\-\\-\\-|\\+\\+\\+)\\s(\\S+)\\s?(.*)/.exec(diffstr[i]));\n    if (fileHeader) {\n      let keyPrefix = fileHeader[1] === '---' ? 'old' : 'new';\n      index[keyPrefix + 'FileName'] = fileHeader[2];\n      index[keyPrefix + 'Header'] = fileHeader[3];\n\n      i++;\n    }\n  }\n\n  // Parses a hunk\n  // This assumes that we are at the start of a hunk.\n  function parseHunk() {\n    let chunkHeaderIndex = i,\n        chunkHeaderLine = diffstr[i++],\n        chunkHeader = chunkHeaderLine.split(/@@ -(\\d+)(?:,(\\d+))? \\+(\\d+)(?:,(\\d+))? @@/);\n\n    let hunk = {\n      oldStart: +chunkHeader[1],\n      oldLines: +chunkHeader[2] || 1,\n      newStart: +chunkHeader[3],\n      newLines: +chunkHeader[4] || 1,\n      lines: []\n    };\n\n    let addCount = 0,\n        removeCount = 0;\n    for (; i < diffstr.length; i++) {\n      let operation = diffstr[i][0];\n\n      if (operation === '+' || operation === '-' || operation === ' ' || operation === '\\\\') {\n        hunk.lines.push(diffstr[i]);\n\n        if (operation === '+') {\n          addCount++;\n        } else if (operation === '-') {\n          removeCount++;\n        } else if (operation === ' ') {\n          addCount++;\n          removeCount++;\n        }\n      } else {\n        break;\n      }\n    }\n\n    // Handle the empty block count case\n    if (!addCount && hunk.newLines === 1) {\n      hunk.newLines = 0;\n    }\n    if (!removeCount && hunk.oldLines === 1) {\n      hunk.oldLines = 0;\n    }\n\n    // Perform optional sanity checking\n    if (options.strict) {\n      if (addCount !== hunk.newLines) {\n        throw new Error('Added line count did not match for hunk at line ' + (chunkHeaderIndex + 1));\n      }\n      if (removeCount !== hunk.oldLines) {\n        throw new Error('Removed line count did not match for hunk at line ' + (chunkHeaderIndex + 1));\n      }\n    }\n\n    return hunk;\n  }\n\n  while (i < diffstr.length) {\n    parseIndex();\n  }\n\n  return list;\n}\n"]} 949 | 950 | 951 | /***/ }, 952 | /* 11 */ 953 | /***/ function(module, exports, __webpack_require__) { 954 | 955 | 'use strict'; 956 | 957 | exports.__esModule = true; 958 | exports.structuredPatch = structuredPatch; 959 | exports.createTwoFilesPatch = createTwoFilesPatch; 960 | exports.createPatch = createPatch; 961 | // istanbul ignore next 962 | 963 | function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) arr2[i] = arr[i]; return arr2; } else { return Array.from(arr); } } 964 | 965 | var _diffLine = __webpack_require__(5); 966 | 967 | function structuredPatch(oldFileName, newFileName, oldStr, newStr, oldHeader, newHeader, options) { 968 | if (!options) { 969 | options = { context: 4 }; 970 | } 971 | 972 | var diff = _diffLine.diffLines(oldStr, newStr); 973 | diff.push({ value: '', lines: [] }); // Append an empty value to make cleanup easier 974 | 975 | function contextLines(lines) { 976 | return lines.map(function (entry) { 977 | return ' ' + entry; 978 | }); 979 | } 980 | 981 | var hunks = []; 982 | var oldRangeStart = 0, 983 | newRangeStart = 0, 984 | curRange = [], 985 | oldLine = 1, 986 | newLine = 1; 987 | 988 | var _loop = function (i) { 989 | var current = diff[i], 990 | lines = current.lines || current.value.replace(/\n$/, '').split('\n'); 991 | current.lines = lines; 992 | 993 | if (current.added || current.removed) { 994 | // istanbul ignore next 995 | 996 | var _curRange; 997 | 998 | // If we have previous context, start with that 999 | if (!oldRangeStart) { 1000 | var prev = diff[i - 1]; 1001 | oldRangeStart = oldLine; 1002 | newRangeStart = newLine; 1003 | 1004 | if (prev) { 1005 | curRange = options.context > 0 ? contextLines(prev.lines.slice(-options.context)) : []; 1006 | oldRangeStart -= curRange.length; 1007 | newRangeStart -= curRange.length; 1008 | } 1009 | } 1010 | 1011 | // Output our changes 1012 | (_curRange = curRange).push.apply(_curRange, _toConsumableArray(lines.map(function (entry) { 1013 | return (current.added ? '+' : '-') + entry; 1014 | }))); 1015 | 1016 | // Track the updated file position 1017 | if (current.added) { 1018 | newLine += lines.length; 1019 | } else { 1020 | oldLine += lines.length; 1021 | } 1022 | } else { 1023 | // Identical context lines. Track line changes 1024 | if (oldRangeStart) { 1025 | // Close out any changes that have been output (or join overlapping) 1026 | if (lines.length <= options.context * 2 && i < diff.length - 2) { 1027 | // istanbul ignore next 1028 | 1029 | var _curRange2; 1030 | 1031 | // Overlapping 1032 | (_curRange2 = curRange).push.apply(_curRange2, _toConsumableArray(contextLines(lines))); 1033 | } else { 1034 | // istanbul ignore next 1035 | 1036 | var _curRange3; 1037 | 1038 | // end the range and output 1039 | var contextSize = Math.min(lines.length, options.context); 1040 | (_curRange3 = curRange).push.apply(_curRange3, _toConsumableArray(contextLines(lines.slice(0, contextSize)))); 1041 | 1042 | var hunk = { 1043 | oldStart: oldRangeStart, 1044 | oldLines: oldLine - oldRangeStart + contextSize, 1045 | newStart: newRangeStart, 1046 | newLines: newLine - newRangeStart + contextSize, 1047 | lines: curRange 1048 | }; 1049 | if (i >= diff.length - 2 && lines.length <= options.context) { 1050 | // EOF is inside this hunk 1051 | var oldEOFNewline = /\n$/.test(oldStr); 1052 | var newEOFNewline = /\n$/.test(newStr); 1053 | if (lines.length == 0 && !oldEOFNewline) { 1054 | // special case: old has no eol and no trailing context; no-nl can end up before adds 1055 | curRange.splice(hunk.oldLines, 0, '\\ No newline at end of file'); 1056 | } else if (!oldEOFNewline || !newEOFNewline) { 1057 | curRange.push('\\ No newline at end of file'); 1058 | } 1059 | } 1060 | hunks.push(hunk); 1061 | 1062 | oldRangeStart = 0; 1063 | newRangeStart = 0; 1064 | curRange = []; 1065 | } 1066 | } 1067 | oldLine += lines.length; 1068 | newLine += lines.length; 1069 | } 1070 | }; 1071 | 1072 | for (var i = 0; i < diff.length; i++) { 1073 | _loop(i); 1074 | } 1075 | 1076 | return { 1077 | oldFileName: oldFileName, newFileName: newFileName, 1078 | oldHeader: oldHeader, newHeader: newHeader, 1079 | hunks: hunks 1080 | }; 1081 | } 1082 | 1083 | function createTwoFilesPatch(oldFileName, newFileName, oldStr, newStr, oldHeader, newHeader, options) { 1084 | var diff = structuredPatch(oldFileName, newFileName, oldStr, newStr, oldHeader, newHeader, options); 1085 | 1086 | var ret = []; 1087 | if (oldFileName == newFileName) { 1088 | ret.push('Index: ' + oldFileName); 1089 | } 1090 | ret.push('==================================================================='); 1091 | ret.push('--- ' + diff.oldFileName + (typeof diff.oldHeader === 'undefined' ? '' : '\t' + diff.oldHeader)); 1092 | ret.push('+++ ' + diff.newFileName + (typeof diff.newHeader === 'undefined' ? '' : '\t' + diff.newHeader)); 1093 | 1094 | for (var i = 0; i < diff.hunks.length; i++) { 1095 | var hunk = diff.hunks[i]; 1096 | ret.push('@@ -' + hunk.oldStart + ',' + hunk.oldLines + ' +' + hunk.newStart + ',' + hunk.newLines + ' @@'); 1097 | ret.push.apply(ret, hunk.lines); 1098 | } 1099 | 1100 | return ret.join('\n') + '\n'; 1101 | } 1102 | 1103 | function createPatch(fileName, oldStr, newStr, oldHeader, newHeader, options) { 1104 | return createTwoFilesPatch(fileName, fileName, oldStr, newStr, oldHeader, newHeader, options); 1105 | } 1106 | //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/patch/create.js"],"names":[],"mappings":";;;;;;;;;;wBAAwB,cAAc;;AAE/B,SAAS,eAAe,CAAC,WAAW,EAAE,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,SAAS,EAAE,OAAO,EAAE;AACvG,MAAI,CAAC,OAAO,EAAE;AACZ,WAAO,GAAG,EAAE,OAAO,EAAE,CAAC,EAAE,CAAC;GAC1B;;AAED,MAAM,IAAI,GAAG,oBAAU,MAAM,EAAE,MAAM,CAAC,CAAC;AACvC,MAAI,CAAC,IAAI,CAAC,EAAC,KAAK,EAAE,EAAE,EAAE,KAAK,EAAE,EAAE,EAAC,CAAC,CAAC;;AAElC,WAAS,YAAY,CAAC,KAAK,EAAE;AAC3B,WAAO,KAAK,CAAC,GAAG,CAAC,UAAS,KAAK,EAAE;AAAE,aAAO,GAAG,GAAG,KAAK,CAAC;KAAE,CAAC,CAAC;GAC3D;;AAED,MAAI,KAAK,GAAG,EAAE,CAAC;AACf,MAAI,aAAa,GAAG,CAAC;MAAE,aAAa,GAAG,CAAC;MAAE,QAAQ,GAAG,EAAE;MACnD,OAAO,GAAG,CAAC;MAAE,OAAO,GAAG,CAAC,CAAC;;wBACpB,CAAC;AACR,QAAM,OAAO,GAAG,IAAI,CAAC,CAAC,CAAC;QACjB,KAAK,GAAG,OAAO,CAAC,KAAK,IAAI,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;AAC5E,WAAO,CAAC,KAAK,GAAG,KAAK,CAAC;;AAEtB,QAAI,OAAO,CAAC,KAAK,IAAI,OAAO,CAAC,OAAO,EAAE;;;;;;AAEpC,UAAI,CAAC,aAAa,EAAE;AAClB,YAAM,IAAI,GAAG,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;AACzB,qBAAa,GAAG,OAAO,CAAC;AACxB,qBAAa,GAAG,OAAO,CAAC;;AAExB,YAAI,IAAI,EAAE;AACR,kBAAQ,GAAG,OAAO,CAAC,OAAO,GAAG,CAAC,GAAG,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,CAAC;AACvF,uBAAa,IAAI,QAAQ,CAAC,MAAM,CAAC;AACjC,uBAAa,IAAI,QAAQ,CAAC,MAAM,CAAC;SAClC;OACF;;;AAGD,mBAAA,QAAQ,EAAC,IAAI,MAAA,+BAAK,KAAK,CAAC,GAAG,CAAC,UAAS,KAAK,EAAE;AAC1C,eAAO,CAAC,OAAO,CAAC,KAAK,GAAG,GAAG,GAAG,GAAG,CAAA,GAAI,KAAK,CAAC;OAC5C,CAAC,EAAC,CAAC;;;AAGJ,UAAI,OAAO,CAAC,KAAK,EAAE;AACjB,eAAO,IAAI,KAAK,CAAC,MAAM,CAAC;OACzB,MAAM;AACL,eAAO,IAAI,KAAK,CAAC,MAAM,CAAC;OACzB;KACF,MAAM;;AAEL,UAAI,aAAa,EAAE;;AAEjB,YAAI,KAAK,CAAC,MAAM,IAAI,OAAO,CAAC,OAAO,GAAG,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,EAAE;;;;;;AAE9D,wBAAA,QAAQ,EAAC,IAAI,MAAA,gCAAK,YAAY,CAAC,KAAK,CAAC,EAAC,CAAC;SACxC,MAAM;;;;;;AAEL,cAAI,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,EAAE,OAAO,CAAC,OAAO,CAAC,CAAC;AAC1D,wBAAA,QAAQ,EAAC,IAAI,MAAA,gCAAK,YAAY,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC,EAAC,CAAC;;AAE7D,cAAI,IAAI,GAAG;AACT,oBAAQ,EAAE,aAAa;AACvB,oBAAQ,EAAG,OAAO,GAAG,aAAa,GAAG,WAAW,AAAC;AACjD,oBAAQ,EAAE,aAAa;AACvB,oBAAQ,EAAG,OAAO,GAAG,aAAa,GAAG,WAAW,AAAC;AACjD,iBAAK,EAAE,QAAQ;WAChB,CAAC;AACF,cAAI,CAAC,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,IAAI,KAAK,CAAC,MAAM,IAAI,OAAO,CAAC,OAAO,EAAE;;AAE3D,gBAAI,aAAa,GAAI,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,AAAC,CAAC;AACzC,gBAAI,aAAa,GAAI,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,AAAC,CAAC;AACzC,gBAAI,KAAK,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE;;AAEvC,sBAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,EAAE,8BAA8B,CAAC,CAAC;aACnE,MAAM,IAAI,CAAC,aAAa,IAAI,CAAC,aAAa,EAAE;AAC3C,sBAAQ,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;aAC/C;WACF;AACD,eAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;;AAEjB,uBAAa,GAAG,CAAC,CAAC;AAClB,uBAAa,GAAG,CAAC,CAAC;AAClB,kBAAQ,GAAG,EAAE,CAAC;SACf;OACF;AACD,aAAO,IAAI,KAAK,CAAC,MAAM,CAAC;AACxB,aAAO,IAAI,KAAK,CAAC,MAAM,CAAC;KACzB;;;AArEH,OAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;UAA7B,CAAC;GAsET;;AAED,SAAO;AACL,eAAW,EAAE,WAAW,EAAE,WAAW,EAAE,WAAW;AAClD,aAAS,EAAE,SAAS,EAAE,SAAS,EAAE,SAAS;AAC1C,SAAK,EAAE,KAAK;GACb,CAAC;CACH;;AAEM,SAAS,mBAAmB,CAAC,WAAW,EAAE,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,SAAS,EAAE,OAAO,EAAE;AAC3G,MAAM,IAAI,GAAG,eAAe,CAAC,WAAW,EAAE,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,SAAS,EAAE,OAAO,CAAC,CAAC;;AAEtG,MAAM,GAAG,GAAG,EAAE,CAAC;AACf,MAAI,WAAW,IAAI,WAAW,EAAE;AAC9B,OAAG,CAAC,IAAI,CAAC,SAAS,GAAG,WAAW,CAAC,CAAC;GACnC;AACD,KAAG,CAAC,IAAI,CAAC,qEAAqE,CAAC,CAAC;AAChF,KAAG,CAAC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW,IAAI,OAAO,IAAI,CAAC,SAAS,KAAK,WAAW,GAAG,EAAE,GAAG,IAAI,GAAG,IAAI,CAAC,SAAS,CAAA,AAAC,CAAC,CAAC;AAC3G,KAAG,CAAC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW,IAAI,OAAO,IAAI,CAAC,SAAS,KAAK,WAAW,GAAG,EAAE,GAAG,IAAI,GAAG,IAAI,CAAC,SAAS,CAAA,AAAC,CAAC,CAAC;;AAE3G,OAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;AAC1C,QAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;AAC3B,OAAG,CAAC,IAAI,CACN,MAAM,GAAG,IAAI,CAAC,QAAQ,GAAG,GAAG,GAAG,IAAI,CAAC,QAAQ,GAC1C,IAAI,GAAG,IAAI,CAAC,QAAQ,GAAG,GAAG,GAAG,IAAI,CAAC,QAAQ,GAC1C,KAAK,CACR,CAAC;AACF,OAAG,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;GACjC;;AAED,SAAO,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC;CAC9B;;AAEM,SAAS,WAAW,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,SAAS,EAAE,OAAO,EAAE;AACnF,SAAO,mBAAmB,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,SAAS,EAAE,OAAO,CAAC,CAAC;CAC/F","file":"create.js","sourcesContent":["import {diffLines} from '../diff/line';\n\nexport function structuredPatch(oldFileName, newFileName, oldStr, newStr, oldHeader, newHeader, options) {\n  if (!options) {\n    options = { context: 4 };\n  }\n\n  const diff = diffLines(oldStr, newStr);\n  diff.push({value: '', lines: []});   // Append an empty value to make cleanup easier\n\n  function contextLines(lines) {\n    return lines.map(function(entry) { return ' ' + entry; });\n  }\n\n  let hunks = [];\n  let oldRangeStart = 0, newRangeStart = 0, curRange = [],\n      oldLine = 1, newLine = 1;\n  for (let i = 0; i < diff.length; i++) {\n    const current = diff[i],\n          lines = current.lines || current.value.replace(/\\n$/, '').split('\\n');\n    current.lines = lines;\n\n    if (current.added || current.removed) {\n      // If we have previous context, start with that\n      if (!oldRangeStart) {\n        const prev = diff[i - 1];\n        oldRangeStart = oldLine;\n        newRangeStart = newLine;\n\n        if (prev) {\n          curRange = options.context > 0 ? contextLines(prev.lines.slice(-options.context)) : [];\n          oldRangeStart -= curRange.length;\n          newRangeStart -= curRange.length;\n        }\n      }\n\n      // Output our changes\n      curRange.push(... lines.map(function(entry) {\n        return (current.added ? '+' : '-') + entry;\n      }));\n\n      // Track the updated file position\n      if (current.added) {\n        newLine += lines.length;\n      } else {\n        oldLine += lines.length;\n      }\n    } else {\n      // Identical context lines. Track line changes\n      if (oldRangeStart) {\n        // Close out any changes that have been output (or join overlapping)\n        if (lines.length <= options.context * 2 && i < diff.length - 2) {\n          // Overlapping\n          curRange.push(... contextLines(lines));\n        } else {\n          // end the range and output\n          let contextSize = Math.min(lines.length, options.context);\n          curRange.push(... contextLines(lines.slice(0, contextSize)));\n\n          let hunk = {\n            oldStart: oldRangeStart,\n            oldLines: (oldLine - oldRangeStart + contextSize),\n            newStart: newRangeStart,\n            newLines: (newLine - newRangeStart + contextSize),\n            lines: curRange\n          };\n          if (i >= diff.length - 2 && lines.length <= options.context) {\n            // EOF is inside this hunk\n            let oldEOFNewline = (/\\n$/.test(oldStr));\n            let newEOFNewline = (/\\n$/.test(newStr));\n            if (lines.length == 0 && !oldEOFNewline) {\n              // special case: old has no eol and no trailing context; no-nl can end up before adds\n              curRange.splice(hunk.oldLines, 0, '\\\\ No newline at end of file');\n            } else if (!oldEOFNewline || !newEOFNewline) {\n              curRange.push('\\\\ No newline at end of file');\n            }\n          }\n          hunks.push(hunk);\n\n          oldRangeStart = 0;\n          newRangeStart = 0;\n          curRange = [];\n        }\n      }\n      oldLine += lines.length;\n      newLine += lines.length;\n    }\n  }\n\n  return {\n    oldFileName: oldFileName, newFileName: newFileName,\n    oldHeader: oldHeader, newHeader: newHeader,\n    hunks: hunks\n  };\n}\n\nexport function createTwoFilesPatch(oldFileName, newFileName, oldStr, newStr, oldHeader, newHeader, options) {\n  const diff = structuredPatch(oldFileName, newFileName, oldStr, newStr, oldHeader, newHeader, options);\n\n  const ret = [];\n  if (oldFileName == newFileName) {\n    ret.push('Index: ' + oldFileName);\n  }\n  ret.push('===================================================================');\n  ret.push('--- ' + diff.oldFileName + (typeof diff.oldHeader === 'undefined' ? '' : '\\t' + diff.oldHeader));\n  ret.push('+++ ' + diff.newFileName + (typeof diff.newHeader === 'undefined' ? '' : '\\t' + diff.newHeader));\n\n  for (let i = 0; i < diff.hunks.length; i++) {\n    const hunk = diff.hunks[i];\n    ret.push(\n      '@@ -' + hunk.oldStart + ',' + hunk.oldLines\n      + ' +' + hunk.newStart + ',' + hunk.newLines\n      + ' @@'\n    );\n    ret.push.apply(ret, hunk.lines);\n  }\n\n  return ret.join('\\n') + '\\n';\n}\n\nexport function createPatch(fileName, oldStr, newStr, oldHeader, newHeader, options) {\n  return createTwoFilesPatch(fileName, fileName, oldStr, newStr, oldHeader, newHeader, options);\n}\n"]} 1107 | 1108 | 1109 | /***/ }, 1110 | /* 12 */ 1111 | /***/ function(module, exports) { 1112 | 1113 | // See: http://code.google.com/p/google-diff-match-patch/wiki/API 1114 | "use strict"; 1115 | 1116 | exports.__esModule = true; 1117 | exports.convertChangesToDMP = convertChangesToDMP; 1118 | 1119 | function convertChangesToDMP(changes) { 1120 | var ret = [], 1121 | change = undefined, 1122 | operation = undefined; 1123 | for (var i = 0; i < changes.length; i++) { 1124 | change = changes[i]; 1125 | if (change.added) { 1126 | operation = 1; 1127 | } else if (change.removed) { 1128 | operation = -1; 1129 | } else { 1130 | operation = 0; 1131 | } 1132 | 1133 | ret.push([operation, change.value]); 1134 | } 1135 | return ret; 1136 | } 1137 | //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb252ZXJ0L2RtcC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFDTyxTQUFTLG1CQUFtQixDQUFDLE9BQU8sRUFBRTtBQUMzQyxNQUFJLEdBQUcsR0FBRyxFQUFFO01BQ1IsTUFBTSxZQUFBO01BQ04sU0FBUyxZQUFBLENBQUM7QUFDZCxPQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsT0FBTyxDQUFDLE1BQU0sRUFBRSxDQUFDLEVBQUUsRUFBRTtBQUN2QyxVQUFNLEdBQUcsT0FBTyxDQUFDLENBQUMsQ0FBQyxDQUFDO0FBQ3BCLFFBQUksTUFBTSxDQUFDLEtBQUssRUFBRTtBQUNoQixlQUFTLEdBQUcsQ0FBQyxDQUFDO0tBQ2YsTUFBTSxJQUFJLE1BQU0sQ0FBQyxPQUFPLEVBQUU7QUFDekIsZUFBUyxHQUFHLENBQUMsQ0FBQyxDQUFDO0tBQ2hCLE1BQU07QUFDTCxlQUFTLEdBQUcsQ0FBQyxDQUFDO0tBQ2Y7O0FBRUQsT0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDLFNBQVMsRUFBRSxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztHQUNyQztBQUNELFNBQU8sR0FBRyxDQUFDO0NBQ1oiLCJmaWxlIjoiZG1wLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLy8gU2VlOiBodHRwOi8vY29kZS5nb29nbGUuY29tL3AvZ29vZ2xlLWRpZmYtbWF0Y2gtcGF0Y2gvd2lraS9BUElcbmV4cG9ydCBmdW5jdGlvbiBjb252ZXJ0Q2hhbmdlc1RvRE1QKGNoYW5nZXMpIHtcbiAgbGV0IHJldCA9IFtdLFxuICAgICAgY2hhbmdlLFxuICAgICAgb3BlcmF0aW9uO1xuICBmb3IgKGxldCBpID0gMDsgaSA8IGNoYW5nZXMubGVuZ3RoOyBpKyspIHtcbiAgICBjaGFuZ2UgPSBjaGFuZ2VzW2ldO1xuICAgIGlmIChjaGFuZ2UuYWRkZWQpIHtcbiAgICAgIG9wZXJhdGlvbiA9IDE7XG4gICAgfSBlbHNlIGlmIChjaGFuZ2UucmVtb3ZlZCkge1xuICAgICAgb3BlcmF0aW9uID0gLTE7XG4gICAgfSBlbHNlIHtcbiAgICAgIG9wZXJhdGlvbiA9IDA7XG4gICAgfVxuXG4gICAgcmV0LnB1c2goW29wZXJhdGlvbiwgY2hhbmdlLnZhbHVlXSk7XG4gIH1cbiAgcmV0dXJuIHJldDtcbn1cbiJdfQ== 1138 | 1139 | 1140 | /***/ }, 1141 | /* 13 */ 1142 | /***/ function(module, exports) { 1143 | 1144 | 'use strict'; 1145 | 1146 | exports.__esModule = true; 1147 | exports.convertChangesToXML = convertChangesToXML; 1148 | 1149 | function convertChangesToXML(changes) { 1150 | var ret = []; 1151 | for (var i = 0; i < changes.length; i++) { 1152 | var change = changes[i]; 1153 | if (change.added) { 1154 | ret.push(''); 1155 | } else if (change.removed) { 1156 | ret.push(''); 1157 | } 1158 | 1159 | ret.push(escapeHTML(change.value)); 1160 | 1161 | if (change.added) { 1162 | ret.push(''); 1163 | } else if (change.removed) { 1164 | ret.push(''); 1165 | } 1166 | } 1167 | return ret.join(''); 1168 | } 1169 | 1170 | function escapeHTML(s) { 1171 | var n = s; 1172 | n = n.replace(/&/g, '&'); 1173 | n = n.replace(//g, '>'); 1175 | n = n.replace(/"/g, '"'); 1176 | 1177 | return n; 1178 | } 1179 | //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb252ZXJ0L3htbC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFPLFNBQVMsbUJBQW1CLENBQUMsT0FBTyxFQUFFO0FBQzNDLE1BQUksR0FBRyxHQUFHLEVBQUUsQ0FBQztBQUNiLE9BQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxPQUFPLENBQUMsTUFBTSxFQUFFLENBQUMsRUFBRSxFQUFFO0FBQ3ZDLFFBQUksTUFBTSxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUN4QixRQUFJLE1BQU0sQ0FBQyxLQUFLLEVBQUU7QUFDaEIsU0FBRyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQztLQUNuQixNQUFNLElBQUksTUFBTSxDQUFDLE9BQU8sRUFBRTtBQUN6QixTQUFHLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO0tBQ25COztBQUVELE9BQUcsQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDOztBQUVuQyxRQUFJLE1BQU0sQ0FBQyxLQUFLLEVBQUU7QUFDaEIsU0FBRyxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsQ0FBQztLQUNwQixNQUFNLElBQUksTUFBTSxDQUFDLE9BQU8sRUFBRTtBQUN6QixTQUFHLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0tBQ3BCO0dBQ0Y7QUFDRCxTQUFPLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUM7Q0FDckI7O0FBRUQsU0FBUyxVQUFVLENBQUMsQ0FBQyxFQUFFO0FBQ3JCLE1BQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztBQUNWLEdBQUMsR0FBRyxDQUFDLENBQUMsT0FBTyxDQUFDLElBQUksRUFBRSxPQUFPLENBQUMsQ0FBQztBQUM3QixHQUFDLEdBQUcsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxJQUFJLEVBQUUsTUFBTSxDQUFDLENBQUM7QUFDNUIsR0FBQyxHQUFHLENBQUMsQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUFFLE1BQU0sQ0FBQyxDQUFDO0FBQzVCLEdBQUMsR0FBRyxDQUFDLENBQUMsT0FBTyxDQUFDLElBQUksRUFBRSxRQUFRLENBQUMsQ0FBQzs7QUFFOUIsU0FBTyxDQUFDLENBQUM7Q0FDViIsImZpbGUiOiJ4bWwuanMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgZnVuY3Rpb24gY29udmVydENoYW5nZXNUb1hNTChjaGFuZ2VzKSB7XG4gIGxldCByZXQgPSBbXTtcbiAgZm9yIChsZXQgaSA9IDA7IGkgPCBjaGFuZ2VzLmxlbmd0aDsgaSsrKSB7XG4gICAgbGV0IGNoYW5nZSA9IGNoYW5nZXNbaV07XG4gICAgaWYgKGNoYW5nZS5hZGRlZCkge1xuICAgICAgcmV0LnB1c2goJzxpbnM+Jyk7XG4gICAgfSBlbHNlIGlmIChjaGFuZ2UucmVtb3ZlZCkge1xuICAgICAgcmV0LnB1c2goJzxkZWw+Jyk7XG4gICAgfVxuXG4gICAgcmV0LnB1c2goZXNjYXBlSFRNTChjaGFuZ2UudmFsdWUpKTtcblxuICAgIGlmIChjaGFuZ2UuYWRkZWQpIHtcbiAgICAgIHJldC5wdXNoKCc8L2lucz4nKTtcbiAgICB9IGVsc2UgaWYgKGNoYW5nZS5yZW1vdmVkKSB7XG4gICAgICByZXQucHVzaCgnPC9kZWw+Jyk7XG4gICAgfVxuICB9XG4gIHJldHVybiByZXQuam9pbignJyk7XG59XG5cbmZ1bmN0aW9uIGVzY2FwZUhUTUwocykge1xuICBsZXQgbiA9IHM7XG4gIG4gPSBuLnJlcGxhY2UoLyYvZywgJyZhbXA7Jyk7XG4gIG4gPSBuLnJlcGxhY2UoLzwvZywgJyZsdDsnKTtcbiAgbiA9IG4ucmVwbGFjZSgvPi9nLCAnJmd0OycpO1xuICBuID0gbi5yZXBsYWNlKC9cIi9nLCAnJnF1b3Q7Jyk7XG5cbiAgcmV0dXJuIG47XG59XG4iXX0= 1180 | 1181 | 1182 | /***/ } 1183 | /******/ ]) 1184 | }); 1185 | ; -------------------------------------------------------------------------------- /gulpfile.js: -------------------------------------------------------------------------------- 1 | var gulp = require('gulp'); 2 | var sass = require('gulp-sass'); 3 | var autoprefixer = require('gulp-autoprefixer'); 4 | var babel = require('gulp-babel'); 5 | var browserSync = require('browser-sync'); 6 | var nodemon = require('gulp-nodemon'); 7 | var reload = browserSync.reload; 8 | var eslint = require('gulp-eslint'); 9 | var concat = require('gulp-concat'); 10 | var uglify = require('gulp-uglify'); 11 | var minifyCss = require('gulp-minify-css'); 12 | var rename = require('gulp-rename'); 13 | 14 | gulp.task('js', function() { 15 | return gulp.src(['client/vendor/*.js', 'client/*.js.es6']) 16 | .pipe(concat('client/dist/app.js')) 17 | .pipe(babel({ blacklist: ['useStrict'] })) 18 | .pipe(gulp.dest('.')); 19 | }); 20 | 21 | gulp.task('scss', function () { 22 | return gulp.src('client/*.scss') 23 | .pipe(sass()) 24 | .pipe(autoprefixer()) 25 | .pipe(gulp.dest('client/dist/')); 26 | }); 27 | 28 | gulp.task('js-dist', ['js'] ,function () { 29 | return gulp.src('client/dist/app.js') 30 | .pipe(uglify()) 31 | .pipe(rename({ extname: '.min.js' })) 32 | .pipe(gulp.dest('client/dist/')); 33 | }); 34 | 35 | gulp.task('css-dist', ['scss'] ,function () { 36 | return gulp.src('client/dist/app.css') 37 | .pipe(minifyCss()) 38 | .pipe(rename({ extname: '.min.css' })) 39 | .pipe(gulp.dest('client/dist/')); 40 | }); 41 | 42 | gulp.task('lint', function () { 43 | return gulp.src('client/*.js.es6') 44 | .pipe(eslint()) 45 | .pipe(eslint.format()); 46 | }); 47 | 48 | gulp.task('server', function (cb) { 49 | var started = false; 50 | return nodemon({ 51 | script: 'server.js' 52 | }).on('start', function () { 53 | if (!started) { 54 | cb(); 55 | started = true; 56 | } 57 | }).on('restart', function () { 58 | setTimeout(function () { 59 | reload({ stream: false }); 60 | }, 1000); 61 | }); 62 | }); 63 | 64 | gulp.task('dev', ['server'], function() { 65 | browserSync({ 66 | proxy: 'http://localhost:8000', 67 | files: ['client/*.*'], 68 | browser: 'google chrome', 69 | port: 3000 70 | }); 71 | gulp.watch('client/*.scss', ['scss']); 72 | gulp.watch('client/*.js.es6', ['js']); 73 | }); 74 | 75 | 76 | gulp.task('dist', ['js-dist', 'css-dist']); -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "Requestdiff", 3 | "version": "1.0.0", 4 | "description": "Visually diff two HTTP requests", 5 | "main": "server.js", 6 | "dependencies": { 7 | "diff": "^2.1.3", 8 | "request": "^2.64.0", 9 | "serve-static": "^1.10.0" 10 | }, 11 | "devDependencies": { 12 | "babel-eslint": "^4.1.3", 13 | "browser-sync": "^2.9.11", 14 | "eslint": "^1.6.0", 15 | "gulp": "^3.9.0", 16 | "gulp-autoprefixer": "^3.0.2", 17 | "gulp-babel": "^5.2.1", 18 | "gulp-concat": "^2.6.0", 19 | "gulp-eslint": "^1.0.0", 20 | "gulp-minify-css": "^1.2.1", 21 | "gulp-nodemon": "^2.0.4", 22 | "gulp-rename": "^1.2.2", 23 | "gulp-sass": "^2.0.4", 24 | "gulp-uglify": "^1.4.2" 25 | }, 26 | "scripts": { 27 | "test": "echo \"Error: no test specified\" && exit 1" 28 | }, 29 | "author": "Victor Delgado", 30 | "license": "MIT" 31 | } 32 | -------------------------------------------------------------------------------- /server.js: -------------------------------------------------------------------------------- 1 | 'use strict'; 2 | 3 | const http = require('http'); 4 | const url = require('url'); 5 | const query = require('querystring'); 6 | const fs = require('fs'); 7 | const statics = require('serve-static'); 8 | const request = require('request'); 9 | 10 | const req = request.defaults({ gzip: true }); 11 | const serve = statics('client'); 12 | 13 | /** 14 | * Extract important info from API response 15 | * and normalize field formats 16 | * @param {Object} res1 17 | * @param {Object} res2 18 | * - res.body - String 19 | * - res.headers - Object 20 | * - res.statusCode - Number 21 | * @return {Object} 22 | */ 23 | 24 | const formatRes = (res1, res2) => { 25 | return { 26 | first: { 27 | body: res1.body, 28 | headers: JSON.stringify(res1.headers), 29 | status: res1.statusCode 30 | }, 31 | second: { 32 | body: res2.body, 33 | headers: JSON.stringify(res2.headers), 34 | status: res2.statusCode 35 | } 36 | }; 37 | }; 38 | 39 | /** 40 | * Request two URLs and return bundle response 41 | */ 42 | 43 | const proxy = (q, cb) => { 44 | console.log(q); 45 | req(q.url1, (err1, res1) => { 46 | if (err1) return cb(err1); 47 | 48 | req(q.url2, (err2, res2) => { 49 | if (err2) return cb(err2); 50 | const msg = formatRes(res1, res2); 51 | console.log(msg); 52 | cb(null, msg); 53 | }); 54 | }); 55 | }; 56 | 57 | /** 58 | * HTTP request handler 59 | */ 60 | 61 | const handler = (req, res) => { 62 | 63 | const parsedUrl = url.parse(req.url); 64 | 65 | if (parsedUrl.pathname === '/proxy') { 66 | const params = query.parse(parsedUrl.query); 67 | proxy(params, (err, msg) => { 68 | if (err) { 69 | console.error(err); 70 | res.writeHead(400, 'Content-Type: application/json'); 71 | return res.end(err.message); 72 | } 73 | res.writeHead(200, 'Content-Type: application/json'); 74 | res.end(JSON.stringify(msg)); 75 | }) 76 | } 77 | else { 78 | console.log(parsedUrl.path); 79 | serve(req, res, () => res.end()); 80 | } 81 | }; 82 | 83 | /** 84 | * Start server 85 | */ 86 | 87 | const server = http.createServer(handler); 88 | server.listen(parseInt(process.env.PORT) || 8000, () => { 89 | console.log(' server started'); 90 | }); 91 | --------------------------------------------------------------------------------