From 158e2df11f8811c5aa3b82f3df50b1af97985f11 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:09:52 -0700 Subject: [PATCH 01/14] Removing structure and previous files. --- projects/web-scraper/index.js | 114 -- projects/web-scraper/package-lock.json | 399 ---- projects/web-scraper/package.json | 24 - projects/web-scraper/siteMap.xml | 2512 ------------------------ 4 files changed, 3049 deletions(-) delete mode 100644 projects/web-scraper/index.js delete mode 100644 projects/web-scraper/package-lock.json delete mode 100644 projects/web-scraper/package.json delete mode 100644 projects/web-scraper/siteMap.xml diff --git a/projects/web-scraper/index.js b/projects/web-scraper/index.js deleted file mode 100644 index 360ce27..0000000 --- a/projects/web-scraper/index.js +++ /dev/null @@ -1,114 +0,0 @@ -const puppeteer = require('puppeteer'); // v 1.1.0 -const { URL } = require('url'); -const fse = require('fs-extra'); // v 5.0.0 -const path = require('path'); -const xml = require('xml2js'); -const parser = new xml.Parser(); -let z = 1; - -// Puppeteer worker. -async function start(urlToFetch, siteName) { - /* 1 */ - const browser = await puppeteer.launch(); - const page = await browser.newPage(); - //await page.setRequestInterception(true); - /* 2 */ - page.on('response', async (response) => { - - try { - const status = response.status; - - page.on( "console", function (log) { - console.log(log.text()); - }); - const url = new URL(response.url()); - - // This intercept is a method of blocking potential URL redirects - // console.log(interceptedRequest.url(), url, status); - // if (interceptedRequest.url().endsWith('/api')) { - // interceptedRequest.respond({ - // status: 422, - // body: "FAKE" - // }) - // } else interceptedRequest.continue(); - - let filePath = path.resolve(`./output/` + siteName + '/' + `${url.pathname}`); - if (path.extname(url.pathname).trim() === '') { - console.log(url.pathname, `${filePath}/index.html`) - filePath = `${filePath}/index.html`; - } - - // Save Data - if ((status >= 300) && (status <= 399)) { - console.log('Redirect from', response.url(), 'to', response.headers()['location']); - await new Promise(function(resolve, reject) { - resolve(null); - }); - } else { - await fse.outputFile(filePath, await response.buffer()); - await fse.outputFile(filePath + z, await response.buffer()); - z++; - } - - } catch (err) { - console.log('Redirect from', response.url(), 'to', response.headers()['location'], err, response.status); - } - }); - - /* 3 */ - await page.goto(urlToFetch, { - waitUntil: 'load' - }); - - /* 4 */ - setTimeout(async () => { - await browser.close(); - }, 60000 * 4); -} - -// Headless chrome requires a lot or resources. -function createQueue(tasks, maxNumOfWorkers = 4) { - var numOfWorkers = 0; - var taskIndex = 0; - - return new Promise( done => { - const handleResult = index => result => { - tasks[index] = result; - numOfWorkers--; - getNextTask(); - }; - const getNextTask = () => { - if (numOfWorkers < maxNumOfWorkers && taskIndex < tasks.length) { - tasks[taskIndex]().then(handleResult(taskIndex)).catch(handleResult(taskIndex)); - taskIndex++; - numOfWorkers++; - getNextTask(); - } else if (numOfWorkers === 0 && taskIndex === tasks.length) { - done(tasks); - } - }; - getNextTask(); - }); -} - -let items = []; - -// Read sites generated by web indexer. -let xmlInjest = (siteName, path) => { - fse.readFile(__dirname + path, (err, data) => { - parser.parseString(data, (err, result) => { - result.urlset.url.forEach((item, index) => { - items.push(async () => { - //console.log(item.loc[0]); - start(item.loc[0]); - }); - }); - console.log(JSON.stringify(items[0])); - // Read X items, Y at a time. - createQueue(items.slice(0,2), 1); - }); - }); -}; - -//start('https://www.lamayor.org/', 'www'); -xmlInjest('www-lamayor', '/siteMap.xml'); \ No newline at end of file diff --git a/projects/web-scraper/package-lock.json b/projects/web-scraper/package-lock.json deleted file mode 100644 index 2703ced..0000000 --- a/projects/web-scraper/package-lock.json +++ /dev/null @@ -1,399 +0,0 @@ -{ - "name": "web-scraper", - "version": "0.0.1", - "lockfileVersion": 1, - "requires": true, - "dependencies": { - "@types/mime-types": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/@types/mime-types/-/mime-types-2.1.0.tgz", - "integrity": "sha1-nKUs2jY/aZxpRmwqbM2q2RPqenM=" - }, - "agent-base": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-5.1.1.tgz", - "integrity": "sha512-TMeqbNl2fMW0nMjTEPOwe3J/PRFP4vqeoNuQMG0HlMrtm5QxKqdvAkZ1pRBQ/ulIyDD5Yq0nJ7YbdD8ey0TO3g==" - }, - "async-limiter": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz", - "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==" - }, - "balanced-match": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", - "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" - }, - "brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "requires": { - "balanced-match": "1.0.0", - "concat-map": "0.0.1" - } - }, - "buffer-from": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", - "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==" - }, - "concat-map": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" - }, - "concat-stream": { - "version": "1.6.2", - "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", - "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", - "requires": { - "buffer-from": "1.1.1", - "inherits": "2.0.3", - "readable-stream": "2.3.7", - "typedarray": "0.0.6" - } - }, - "core-util-is": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", - "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" - }, - "debug": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", - "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", - "requires": { - "ms": "2.1.2" - } - }, - "extract-zip": { - "version": "1.6.7", - "resolved": "https://registry.npmjs.org/extract-zip/-/extract-zip-1.6.7.tgz", - "integrity": "sha1-qEC0uK9kAyZMjbV/Txp0Mz74H+k=", - "requires": { - "concat-stream": "1.6.2", - "debug": "2.6.9", - "mkdirp": "0.5.1", - "yauzl": "2.4.1" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" - } - } - }, - "fd-slicer": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/fd-slicer/-/fd-slicer-1.0.1.tgz", - "integrity": "sha1-i1vL2ewyfFBBv5qwI/1nUPEXfmU=", - "requires": { - "pend": "1.2.0" - } - }, - "fs-extra": { - "version": "8.1.0", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", - "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", - "requires": { - "graceful-fs": "4.2.3", - "jsonfile": "4.0.0", - "universalify": "0.1.2" - } - }, - "fs.realpath": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" - }, - "glob": { - "version": "7.1.6", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", - "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", - "requires": { - "fs.realpath": "1.0.0", - "inflight": "1.0.6", - "inherits": "2.0.3", - "minimatch": "3.0.4", - "once": "1.4.0", - "path-is-absolute": "1.0.1" - } - }, - "graceful-fs": { - "version": "4.2.3", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.3.tgz", - "integrity": "sha512-a30VEBm4PEdx1dRB7MFK7BejejvCvBronbLjht+sHuGYj8PHs7M/5Z+rt5lw551vZ7yfTCj4Vuyy3mSJytDWRQ==" - }, - "https-proxy-agent": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-4.0.0.tgz", - "integrity": "sha512-zoDhWrkR3of1l9QAL8/scJZyLu8j/gBkcwcaQOZh7Gyh/+uJQzGVETdgT30akuwkpL8HTRfssqI3BZuV18teDg==", - "requires": { - "agent-base": "5.1.1", - "debug": "4.1.1" - } - }, - "inflight": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", - "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", - "requires": { - "once": "1.4.0", - "wrappy": "1.0.2" - } - }, - "inherits": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" - }, - "isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" - }, - "jsonfile": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", - "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", - "requires": { - "graceful-fs": "4.2.3" - } - }, - "mime": { - "version": "2.4.4", - "resolved": "https://registry.npmjs.org/mime/-/mime-2.4.4.tgz", - "integrity": "sha512-LRxmNwziLPT828z+4YkNzloCFC2YM4wrB99k+AV5ZbEyfGNWfG8SO1FUXLmLDBSo89NrJZ4DIWeLjy1CHGhMGA==" - }, - "mime-db": { - "version": "1.43.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.43.0.tgz", - "integrity": "sha512-+5dsGEEovYbT8UY9yD7eE4XTc4UwJ1jBYlgaQQF38ENsKR3wj/8q8RFZrF9WIZpB2V1ArTVFUva8sAul1NzRzQ==" - }, - "mime-types": { - "version": "2.1.26", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.26.tgz", - "integrity": "sha512-01paPWYgLrkqAyrlDorC1uDwl2p3qZT7yl806vW7DvDoxwXi46jsjFbg+WdwotBIk6/MbEhO/dh5aZ5sNj/dWQ==", - "requires": { - "mime-db": "1.43.0" - } - }, - "minimatch": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", - "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", - "requires": { - "brace-expansion": "1.1.11" - } - }, - "minimist": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", - "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" - }, - "mkdirp": { - "version": "0.5.1", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", - "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", - "requires": { - "minimist": "0.0.8" - } - }, - "ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" - }, - "once": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", - "requires": { - "wrappy": "1.0.2" - } - }, - "path": { - "version": "0.12.7", - "resolved": "https://registry.npmjs.org/path/-/path-0.12.7.tgz", - "integrity": "sha1-1NwqUGxM4hl+tIHr/NWzbAFAsQ8=", - "requires": { - "process": "0.11.10", - "util": "0.10.4" - } - }, - "path-is-absolute": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", - "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" - }, - "pend": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/pend/-/pend-1.2.0.tgz", - "integrity": "sha1-elfrVQpng/kRUzH89GY9XI4AelA=" - }, - "process": { - "version": "0.11.10", - "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", - "integrity": "sha1-czIwDoQBYb2j5podHZGn1LwW8YI=" - }, - "process-nextick-args": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", - "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" - }, - "progress": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", - "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==" - }, - "proxy-from-env": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.0.0.tgz", - "integrity": "sha1-M8UDmPcOp+uW0h97gXYwpVeRx+4=" - }, - "punycode": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", - "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=" - }, - "puppeteer": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/puppeteer/-/puppeteer-2.1.1.tgz", - "integrity": "sha512-LWzaDVQkk1EPiuYeTOj+CZRIjda4k2s5w4MK4xoH2+kgWV/SDlkYHmxatDdtYrciHUKSXTsGgPgPP8ILVdBsxg==", - "requires": { - "@types/mime-types": "2.1.0", - "debug": "4.1.1", - "extract-zip": "1.6.7", - "https-proxy-agent": "4.0.0", - "mime": "2.4.4", - "mime-types": "2.1.26", - "progress": "2.0.3", - "proxy-from-env": "1.0.0", - "rimraf": "2.7.1", - "ws": "6.2.1" - } - }, - "querystring": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", - "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA=" - }, - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "requires": { - "core-util-is": "1.0.2", - "inherits": "2.0.3", - "isarray": "1.0.0", - "process-nextick-args": "2.0.1", - "safe-buffer": "5.1.2", - "string_decoder": "1.1.1", - "util-deprecate": "1.0.2" - } - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "requires": { - "glob": "7.1.6" - } - }, - "safe-buffer": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" - }, - "sax": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", - "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==" - }, - "string_decoder": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", - "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", - "requires": { - "safe-buffer": "5.1.2" - } - }, - "typedarray": { - "version": "0.0.6", - "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", - "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=" - }, - "universalify": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", - "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==" - }, - "url": { - "version": "0.11.0", - "resolved": "https://registry.npmjs.org/url/-/url-0.11.0.tgz", - "integrity": "sha1-ODjpfPxgUh63PFJajlW/3Z4uKPE=", - "requires": { - "punycode": "1.3.2", - "querystring": "0.2.0" - } - }, - "util": { - "version": "0.10.4", - "resolved": "https://registry.npmjs.org/util/-/util-0.10.4.tgz", - "integrity": "sha512-0Pm9hTQ3se5ll1XihRic3FDIku70C+iHUdT/W926rSgHV5QgXsYbKZN8MSC3tJtSkhuROzvsQjAaFENRXr+19A==", - "requires": { - "inherits": "2.0.3" - } - }, - "util-deprecate": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" - }, - "wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" - }, - "ws": { - "version": "6.2.1", - "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz", - "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==", - "requires": { - "async-limiter": "1.0.1" - } - }, - "xml2js": { - "version": "0.4.23", - "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", - "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", - "requires": { - "sax": "1.2.4", - "xmlbuilder": "11.0.1" - } - }, - "xmlbuilder": { - "version": "11.0.1", - "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", - "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==" - }, - "yauzl": { - "version": "2.4.1", - "resolved": "https://registry.npmjs.org/yauzl/-/yauzl-2.4.1.tgz", - "integrity": "sha1-lSj0QtqxsihOWLQ3m7GU4i4MQAU=", - "requires": { - "fd-slicer": "1.0.1" - } - } - } -} diff --git a/projects/web-scraper/package.json b/projects/web-scraper/package.json deleted file mode 100644 index 08d9ae7..0000000 --- a/projects/web-scraper/package.json +++ /dev/null @@ -1,24 +0,0 @@ -{ - "name": "web-scraper", - "version": "0.0.1", - "main": "index.js", - "scripts": { - "start": "node index.js" - }, - "repository": { - "type": "git", - "url": "https://github.com/hackforla/gender-equity.git" - }, - "author": "", - "license": "GPL-2.0", - "description": "A headless webscraper", - "dependencies": { - "fs": "0.0.1-security", - "fs-extra": "^8.1.0", - "path": "^0.12.7", - "puppeteer": "^2.1.1", - "url": "^0.11.0", - "xml2js": "^0.4.23" - }, - "devDependencies": {} -} diff --git a/projects/web-scraper/siteMap.xml b/projects/web-scraper/siteMap.xml deleted file mode 100644 index 677ac3e..0000000 --- a/projects/web-scraper/siteMap.xml +++ /dev/null @@ -1,2512 +0,0 @@ - - - - - - - https://www.lamayor.org/ - 2020-02-13T03:28:54+00:00 - 1.00 - - - https://www.lamayor.org/how-can-we-help - 2020-02-13T03:28:55+00:00 - 0.80 - - - https://www.lamayor.org/about - 2020-02-13T03:28:56+00:00 - 0.80 - - - https://www.lamayor.org/our-team - 2020-02-13T03:28:57+00:00 - 0.80 - - - https://www.lamayor.org/OurWork - 2020-02-13T03:28:58+00:00 - 0.80 - - - https://www.lamayor.org/PressRoom - 2020-02-13T03:29:00+00:00 - 0.80 - - - https://www.lamayor.org/mayor-garcetti-launches-la%E2%80%99s-%E2%80%98decade-action%E2%80%99-fight-climate-crisis - 2020-02-13T03:29:00+00:00 - 0.80 - - - https://www.lamayor.org/Homelessness - 2020-02-13T03:29:02+00:00 - 0.80 - - - https://www.lamayor.org/mayor_garcetti_s_executive_directives - 2020-02-13T03:29:03+00:00 - 0.80 - - - https://www.lamayor.org/achievements - 2020-02-13T03:29:04+00:00 - 0.80 - - - https://www.lamayor.org/sustainability - 2020-02-13T03:29:06+00:00 - 0.80 - - - https://www.lamayor.org/Immigration - 2020-02-13T03:29:07+00:00 - 0.80 - - - https://www.lamayor.org/infrastructure - 2020-02-13T03:29:08+00:00 - 0.80 - - - https://www.lamayor.org/economic-growth - 2020-02-13T03:29:09+00:00 - 0.80 - - - https://www.lamayor.org/safe-neighborhoods - 2020-02-13T03:29:10+00:00 - 0.80 - - - https://www.lamayor.org/mayor-garcetti-convenes-faith-leaders-support-la%E2%80%99s-green-new-deal - 2020-02-13T03:29:11+00:00 - 0.80 - - - https://www.lamayor.org/mayor-garcetti-celebrates-two-years-evolve-entertainment-fund - 2020-02-13T03:29:12+00:00 - 0.80 - - - https://www.lamayor.org/mayor-garcetti-celebrates-launch-new-lafd-fast-response-vehicle-unit - 2020-02-13T03:29:13+00:00 - 0.80 - - - https://www.lamayor.org/terms - 2020-02-13T03:29:16+00:00 - 0.80 - - - https://www.lamayor.org/privacy - 2020-02-13T03:29:17+00:00 - 0.80 - - - https://www.lamayor.org/meet-your-area-representative - 2020-02-13T03:29:19+00:00 - 0.64 - - - https://www.lamayor.org/files/33081613638899593114807961192485181194240ojpg-0 - 2020-02-13T03:29:19+00:00 - 0.64 - - - https://www.lamayor.org/mayors-help-desk - 2020-02-13T03:29:20+00:00 - 0.64 - - - https://www.lamayor.org/files/website-card-full13png-0 - 2020-02-13T03:29:21+00:00 - 0.64 - - - https://www.lamayor.org/files/311jpg - 2020-02-13T03:29:22+00:00 - 0.64 - - - https://www.lamayor.org/files/6g1a0072a-webpng-0 - 2020-02-13T03:29:23+00:00 - 0.64 - - - https://www.lamayor.org/files/bphomejpg - 2020-02-13T03:29:24+00:00 - 0.64 - - - https://www.lamayor.org/files/city-govjpg - 2020-02-13T03:29:25+00:00 - 0.64 - - - https://www.lamayor.org/files/newsletterjpg-0 - 2020-02-13T03:29:26+00:00 - 0.64 - - - https://www.lamayor.org/general - 2020-02-13T03:29:27+00:00 - 0.64 - - - https://www.lamayor.org/files/bgherofpojpg-0 - 2020-02-13T03:29:28+00:00 - 0.64 - - - https://www.lamayor.org/about-spanish - 2020-02-13T03:29:29+00:00 - 0.64 - - - https://www.lamayor.org/about-korean - 2020-02-13T03:29:30+00:00 - 0.64 - - - https://www.lamayor.org/about-mandarin-chinese - 2020-02-13T03:29:31+00:00 - 0.64 - - - https://www.lamayor.org/about-armenian - 2020-02-13T03:29:33+00:00 - 0.64 - - - https://www.lamayor.org/about-tagalog - 2020-02-13T03:29:34+00:00 - 0.64 - - - https://www.lamayor.org/amy-elaine-wakeland - 2020-02-13T03:29:35+00:00 - 0.64 - - - https://www.lamayor.org/files/35721154406baa8a0031bk-webjpg-0 - 2020-02-13T03:29:36+00:00 - 0.64 - - - https://www.lamayor.org/files/tile-14x7jpg - 2020-02-13T03:29:39+00:00 - 0.64 - - - https://www.lamayor.org/files/edpng-1 - 2020-02-13T03:29:40+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcettis-la - 2020-02-13T03:29:41+00:00 - 0.64 - - - https://www.lamayor.org/files/funpng-0 - 2020-02-13T03:29:42+00:00 - 0.64 - - - https://www.lamayor.org/files/nk3a0492-card1jpg-0 - 2020-02-13T03:29:43+00:00 - 0.64 - - - https://www.lamayor.org/files/photos-2png-2 - 2020-02-13T03:29:44+00:00 - 0.64 - - - https://www.lamayor.org/files/26634819977fc27d5f65cojpg - 2020-02-13T03:29:45+00:00 - 0.64 - - - https://www.lamayor.org/mayors-offices - 2020-02-13T03:29:47+00:00 - 0.64 - - - https://www.lamayor.org/files/6g1a6187a-croppng-1 - 2020-02-13T03:29:48+00:00 - 0.64 - - - https://www.lamayor.org/senior-leadership - 2020-02-13T03:29:49+00:00 - 0.64 - - - https://www.lamayor.org/files/ckn00003a-14x7jpg - 2020-02-13T03:29:49+00:00 - 0.64 - - - https://www.lamayor.org/files/area-repjpg-0 - 2020-02-13T03:29:50+00:00 - 0.64 - - - https://www.lamayor.org/intern - 2020-02-13T03:29:51+00:00 - 0.64 - - - https://www.lamayor.org/files/6g1a5920a-webjpg - 2020-02-13T03:29:52+00:00 - 0.64 - - - https://www.lamayor.org/files/tile-14x7jpg-0 - 2020-02-13T03:29:53+00:00 - 0.64 - - - https://www.lamayor.org/education - 2020-02-13T03:29:54+00:00 - 0.64 - - - https://www.lamayor.org/files/edu-14x7png - 2020-02-13T03:29:55+00:00 - 0.64 - - - https://www.lamayor.org/files/solarpng-0 - 2020-02-13T03:29:56+00:00 - 0.64 - - - https://www.lamayor.org/files/38040116492fb57f29d77opng-1 - 2020-02-13T03:29:57+00:00 - 0.64 - - - https://www.lamayor.org/Housing - 2020-02-13T03:29:58+00:00 - 0.64 - - - https://www.lamayor.org/files/497a0599a-14x7jpg - 2020-02-13T03:29:59+00:00 - 0.64 - - - https://www.lamayor.org/files/econpng-0 - 2020-02-13T03:30:02+00:00 - 0.64 - - - https://www.lamayor.org/files/snl0png - 2020-02-13T03:30:03+00:00 - 0.64 - - - https://www.lamayor.org/files/laxpng-0 - 2020-02-13T03:30:04+00:00 - 0.64 - - - https://www.lamayor.org/files/moiapng-0 - 2020-02-13T03:30:05+00:00 - 0.64 - - - https://www.lamayor.org/transportation - 2020-02-13T03:30:06+00:00 - 0.64 - - - https://www.lamayor.org/files/infrastructure0png - 2020-02-13T03:30:07+00:00 - 0.64 - - - https://www.lamayor.org/GenderEquity - 2020-02-13T03:30:08+00:00 - 0.64 - - - https://www.lamayor.org/files/33095081610dcc3db7165ojpg-0 - 2020-02-13T03:30:09+00:00 - 0.64 - - - https://www.lamayor.org/improving-government - 2020-02-13T03:30:10+00:00 - 0.64 - - - https://www.lamayor.org/files/well-runpng-0 - 2020-02-13T03:30:11+00:00 - 0.64 - - - https://www.lamayor.org/international-affairs - 2020-02-13T03:30:12+00:00 - 0.64 - - - https://www.lamayor.org/files/11470-10jpg - 2020-02-13T03:30:13+00:00 - 0.64 - - - https://www.lamayor.org/veterans - 2020-02-13T03:30:14+00:00 - 0.64 - - - https://www.lamayor.org/files/veterans-button - 2020-02-13T03:30:16+00:00 - 0.64 - - - https://www.lamayor.org/files/dashboard-jpg - 2020-02-13T03:30:17+00:00 - 0.64 - - - https://www.lamayor.org/media/press_releases - 2020-02-13T03:30:17+00:00 - 0.64 - - - https://www.lamayor.org/files/website-card-flat-secondary-bluepng-2 - 2020-02-13T03:30:19+00:00 - 0.64 - - - https://www.lamayor.org/Speeches - 2020-02-13T03:30:19+00:00 - 0.64 - - - https://www.lamayor.org/files/inaug-2017-20png - 2020-02-13T03:30:21+00:00 - 0.64 - - - https://www.lamayor.org/files/photos-2png-0 - 2020-02-13T03:30:22+00:00 - 0.64 - - - https://www.lamayor.org/ask-my-anything-reddit - 2020-02-13T03:30:23+00:00 - 0.64 - - - https://www.lamayor.org/files/redditpng - 2020-02-13T03:30:24+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/20200210ExecutiveDirective25.pdf - 2020-02-11T01:25:03+00:00 - 0.64 - - - https://www.lamayor.org/HEAP - 2020-02-13T03:30:25+00:00 - 0.64 - - - https://www.lamayor.org/rising-challenge-helping-homeless-angelenos - 2020-02-13T03:30:26+00:00 - 0.64 - - - https://www.lamayor.org/momentum-solutions-homelessness - 2020-02-13T03:30:27+00:00 - 0.64 - - - https://www.lamayor.org/meet-blanca - 2020-02-13T03:30:28+00:00 - 0.64 - - - https://www.lamayor.org/CTC-Helping_our_Homeless_Neighbors - 2020-02-13T03:30:29+00:00 - 0.64 - - - https://www.lamayor.org/confronting-crisis-helping-our-homeless-neighbors - 2020-02-13T03:30:31+00:00 - 0.64 - - - https://www.lamayor.org/HomelessnessCausesAndResponses - 2020-02-13T03:30:32+00:00 - 0.64 - - - https://www.lamayor.org/files/277541644998ecc7993d5opng-1 - 2020-02-13T03:30:33+00:00 - 0.64 - - - https://www.lamayor.org/ABridgeHome - 2020-02-13T03:30:34+00:00 - 0.64 - - - https://www.lamayor.org/files/abhjpg - 2020-02-13T03:30:35+00:00 - 0.64 - - - https://www.lamayor.org/ConfrontingTheHomelessnessCrisis - 2020-02-13T03:30:36+00:00 - 0.64 - - - https://www.lamayor.org/files/32142783900fa82a1a375opng - 2020-02-13T03:30:37+00:00 - 0.64 - - - https://www.lamayor.org/street-strategies - 2020-02-13T03:30:38+00:00 - 0.64 - - - https://www.lamayor.org/files/ssjpg - 2020-02-13T03:30:40+00:00 - 0.64 - - - https://www.lamayor.org/files/14x7jpg-0 - 2020-02-13T03:30:40+00:00 - 0.64 - - - https://www.lamayor.org/HomelessnessHousing - 2020-02-13T03:30:41+00:00 - 0.64 - - - https://www.lamayor.org/files/384590444729fc6c29be2opng - 2020-02-13T03:30:42+00:00 - 0.64 - - - https://www.lamayor.org/HomelessnessTrackingHHH - 2020-02-13T03:30:43+00:00 - 0.64 - - - https://www.lamayor.org/files/39154395042c93ec8f9b1opng - 2020-02-13T03:30:44+00:00 - 0.64 - - - https://www.lamayor.org/how-help-homeless-angelenos - 2020-02-13T03:30:46+00:00 - 0.64 - - - https://www.lamayor.org/files/helpjpg-0 - 2020-02-13T03:30:47+00:00 - 0.64 - - - https://www.lamayor.org/homeless-help-desk - 2020-02-13T03:30:48+00:00 - 0.64 - - - https://www.lamayor.org/files/homeless-help-desk-2jpg - 2020-02-13T03:30:49+00:00 - 0.64 - - - https://www.lamayor.org/Homelessness-Data-and-Dashboards - 2020-02-13T03:30:50+00:00 - 0.64 - - - https://www.lamayor.org/files/14x7jpg - 2020-02-13T03:30:51+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Executive-Directive-1-Great-Streets-Initiative-1.pdf?1426619965 - 2016-02-06T22:04:27+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED2_with_signature_and_letterhead.pdf?1426620047 - 2016-02-06T21:09:47+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Executive-Directive-3-Open-Data.pdf?1426620075 - 2016-02-06T21:10:17+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED_4_-_Intergovernmental_Relations.pdf?1426619988 - 2016-02-06T21:09:11+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED_5_-_Emergency_Drought__Response_-_Creating_a_Water_Wise_City.pdf?1426620015 - 2016-02-06T21:09:17+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED6Film.pdf?1426620059 - 2016-02-06T21:09:55+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED7-SustainableCitypLAn.pdf - 2019-01-03T01:01:21+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED8-CleanStreets.pdf - 2019-01-03T01:01:12+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED9.pdf - 2019-01-03T00:59:51+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Mayor_Garcetti_File_Executive_Directive_10_Vision_Zero.pdf?1440454405 - 2016-02-06T21:10:25+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED_11.pdf?1440645063 - 2016-02-06T21:09:26+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/SCHM_305H_K15100111110.pdf?1443724774 - 2016-02-06T22:07:03+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED_13_-_Support_for_Affordable_Housing_Development.pdf?1445984955 - 2016-02-06T21:09:39+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Executive_Directive_14.pdf?1446858272 - 2016-02-06T21:10:06+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2018%20-%20Safe%20and%20Healthy%20Workforce.pdf - 2017-02-06T23:08:09+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ed19%20planning.pdf - 2017-03-09T21:13:37+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Exec.%20Dir.%20No.%2020--Standing%20with%20Immigrants.pdf - 2017-03-21T22:04:02+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/20180206%20ED%2021.pdf - 2018-02-07T00:10:17+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2022%20-%20Resilient%20Los%20Angeles.pdf - 2018-03-06T00:48:31+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2023%20-%20Harassment%20and%20Discrimination.pdf - 2018-05-01T19:20:07+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2024%20-%20Building%20%27A%20Bridge%20Home%27.pdf - 2018-06-02T01:20:31+00:00 - 0.64 - - - https://www.lamayor.org/statement-mayor-garcetti-minimum-wage-increase - 2020-02-13T03:31:05+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/2018_SOTC_AS_PREPARED_FOR_DELIVERY.pdf - 2018-04-16T16:01:50+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-cuts-costs-digital-filmmakers-new-pilot-program - 2020-02-13T03:31:07+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-record-breaking-year-la-economy-0 - 2020-02-13T03:31:08+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-unveils-plans-portsocall-revitalization - 2020-02-13T03:31:09+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-surpasses-10000-veterans-hiring-goal-ahead-schedule - 2020-02-13T03:31:10+00:00 - 0.64 - - - https://www.lamayor.org/statement-mayor-garcetti-signs-affordable-housing-linkage-fee-ordinance - 2020-02-13T03:31:11+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-6-million-state-grant-support-city-reentry-programs - 2020-02-13T03:31:12+00:00 - 0.64 - - - https://www.lamayor.org/la-secures-35-million-community-revitalization-watts - 2020-02-13T03:31:13+00:00 - 0.64 - - - https://www.lamayor.org/mayor_garcetti_marks_milestone_in_the_lowering_of_tunnel_boring_machine_for_crenshaw_lax_line - 2020-02-13T03:31:14+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-launches-complete-streets-program - 2020-02-13T03:31:16+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-releases-progress-report-sustainable-city-plan-two-thirds-all-2017-goals-accomplished - 2020-02-13T03:31:17+00:00 - 0.64 - - - https://www.lamayor.org/los-angeles-named-1-solar-city-america - 2020-02-13T03:31:18+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-breaks-ground-major-water-recycling-project - 2020-02-13T03:31:19+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-and-lapd-launch-recruitment-campaign-focused-diversity - 2020-02-13T03:31:21+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-new-expansion-community-safety-partnership - 2020-02-13T03:31:22+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-gun-buybacks-took-close-500-weapons-street - 2020-02-13T03:31:23+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-kicks-7th-year-summer-night-lights - 2020-02-13T03:31:24+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-expanded-domestic-violence-programs - 2020-02-13T03:31:25+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-grant-expand-domestic-violence-programs - 2020-02-13T03:31:26+00:00 - 0.64 - - - https://www.lamayor.org/Resilience - 2020-02-13T03:31:28+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-launches-nation%E2%80%99s-first-publicly-available-earthquake-early-warning-mobile-app - 2020-02-13T03:31:29+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-cleanstat-0 - 2020-02-13T03:31:31+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-signs-2018-19-budget-record-investments-ending-homelessness-improving-infrastructure - 2020-02-13T03:31:32+00:00 - 0.64 - - - https://www.lamayor.org/release-mayor-garcetti-unveils-new-ladwp-customer-bill-rights - 2020-02-13T03:31:34+00:00 - 0.64 - - - https://www.lamayor.org/larams - 2020-02-13T03:31:36+00:00 - 0.64 - - - https://www.lamayor.org/statement-mayor-garcetti-chargers-moving-los-angeles - 2020-02-13T03:31:37+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-joins-mayors-around-world-set-bold-new-sustainability-targets - 2020-02-13T03:31:38+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-leads-%E2%80%98climate-mayors%E2%80%99-oppose-us-withdrawal-paris-agreement - 2020-02-13T03:31:39+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-los-angeles-will-recycle-100-city%E2%80%99s-wastewater-2035 - 2020-02-13T03:31:40+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-ladwp-will-phase-out-natural-gas-operations-three-power-plants - 2020-02-13T03:31:41+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-completion-world%E2%80%99s-most-powerful-rooftop-solar-project - 2020-02-13T03:31:42+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-and-long-beach-mayor-robert-garcia-announce-zero-emissions-goals-san-pedro-bay-ports - 2020-02-13T03:31:43+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-and-state-leaders-launch-electric-vehicle-car-share-program-disadvantaged-communities - 2020-02-13T03:31:44+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-hosts-clean-transportation-forum-paris-mayor-hidalgo-announces-new-online-platform - 2020-02-13T03:31:46+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-leads-us-mayors%E2%80%99-call-accurate-2020-census - 2020-02-13T03:31:47+00:00 - 0.64 - - - https://www.lamayor.org/strong-families-la - 2020-02-13T03:31:48+00:00 - 0.64 - - - https://www.lamayor.org/immigration-news-and-events - 2020-02-13T03:31:49+00:00 - 0.64 - - - https://www.lamayor.org/files/37614181706b11febddd0opng - 2020-02-13T03:31:50+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Community-Resource-Guide-%28English3%29.pdf - 2019-08-28T16:41:09+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Community-Resource-Guide-%28Spanish3%29.pdf - 2019-08-28T16:41:23+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Community-Resource-Guide-%28Korean2%29.pdf - 2019-08-28T16:41:17+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Community-Resource-Guide-%28Chinese2%29.pdf - 2019-08-28T16:41:29+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/LAPDandImmigrationFAQ.pdf - 2018-05-30T22:33:12+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Immigration%20consumer-alert%20English.pdf - 2018-08-29T18:02:36+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Immigration%20consumer-alert-Spanish.pdf - 2018-08-29T18:02:45+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Immigration%20consumer-alert%20Chinese.PDF - 2018-08-29T18:03:13+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Immigration%20consumer-alert%20Vietnamese.PDF - 2018-08-29T18:03:20+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-2400-lane-miles-paved-over-last-year - 2020-02-13T03:31:57+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-and-los-angeles-world-airports-break-ground-historic-lax-airport-train - 2020-02-13T03:31:58+00:00 - 0.64 - - - https://www.lamayor.org/new-groundwater-treatment-project-will-help-la-reduce-dependence-imported-water - 2020-02-13T03:31:59+00:00 - 0.64 - - - https://www.lamayor.org/two-big-steps-forward-garcettis-economic-development-agenda - 2020-02-13T03:32:01+00:00 - 0.64 - - - https://www.lamayor.org/statement-mayor-garcetti-latest-round-film-tax-credit-allocations - 2020-02-13T03:32:02+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-opens-tech-job-opportunities-young-people - 2020-02-13T03:32:03+00:00 - 0.64 - - - https://www.lamayor.org/mayors-office-economic-development - 2020-02-13T03:32:04+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-launches-la-original - 2020-02-13T03:32:05+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-launches-evolve-entertainment-fund-boost-diversity-entertainment-industry - 2020-02-13T03:32:06+00:00 - 0.64 - - - https://www.lamayor.org/%E2%80%98new-roads-second-chances%E2%80%99-connects-hundreds-formerly-incarcerated-angelenos-employment - 2020-02-13T03:32:07+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-first-steps-comprehensive-strategy-combat-homelessness-crisis - 2020-02-13T03:32:08+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/AcceleratingSmallBusinessGrowth.pdf - 2018-12-05T19:14:23+00:00 - 0.64 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/LAWomenMeanBizBrochure.pdf - 2018-12-05T19:14:31+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-body-cameras-lapd - 2020-02-13T03:32:11+00:00 - 0.64 - - - https://www.lamayor.org/%E2%80%98summer-night-lights%E2%80%99-kicks-10th-year-providing-fun-safe-spaces-young-people-and-families - 2020-02-13T03:32:13+00:00 - 0.64 - - - https://www.lamayor.org/mayor_garcetti_unveils_los_angeles_fire_department_innovations_to_improve_emergency_response_patient_care - 2020-02-13T03:32:14+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-urges-earthquake-preparedness-day-great-shakeout - 2020-02-13T03:32:15+00:00 - 0.64 - - - https://www.lamayor.org/BuildingForwardLA - 2020-02-13T03:32:17+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-launches-nation%E2%80%99s-first-city-based-cyber-lab - 2020-02-13T03:32:18+00:00 - 0.64 - - - https://www.lamayor.org/mayor-garcetti-announces-opportunities-mayors-crisis-response-team - 2020-02-13T03:32:20+00:00 - 0.64 - - - https://www.lamayor.org/WestValleyAreaRep - 2020-02-13T03:32:20+00:00 - 0.51 - - - https://www.lamayor.org/CentralAreaRep - 2020-02-13T03:32:21+00:00 - 0.51 - - - https://www.lamayor.org/EastValleyAreaRep - 2020-02-13T03:32:22+00:00 - 0.51 - - - https://www.lamayor.org/SouthwestLAAreaRep - 2020-02-13T03:32:23+00:00 - 0.51 - - - https://www.lamayor.org/WestAreaRep - 2020-02-13T03:32:24+00:00 - 0.51 - - - https://www.lamayor.org/SoutheastLAAreaRep - 2020-02-13T03:32:25+00:00 - 0.51 - - - https://www.lamayor.org/EastAreaRep - 2020-02-13T03:32:26+00:00 - 0.51 - - - https://www.lamayor.org/HarborAreaRep - 2020-02-13T03:32:27+00:00 - 0.51 - - - https://www.lamayor.org/FreeTaxPrep - 2020-02-13T03:32:28+00:00 - 0.51 - - - https://www.lamayor.org/EstudiantesYFamilias - 2020-02-13T03:32:29+00:00 - 0.51 - - - https://www.lamayor.org/StudentsAndFamilies - 2020-02-13T03:32:30+00:00 - 0.51 - - - https://www.lamayor.org/free-rides-metro-and-other-election-day-essentials - 2020-02-13T03:32:32+00:00 - 0.51 - - - https://www.lamayor.org/CityHallTour - 2020-02-13T03:32:32+00:00 - 0.51 - - - https://www.lamayor.org/what-you-should-know-about-typhus - 2020-02-13T03:32:33+00:00 - 0.51 - - - https://www.lamayor.org/LAVotes - 2020-02-13T03:32:34+00:00 - 0.51 - - - https://www.lamayor.org/MYC - 2020-02-13T03:32:35+00:00 - 0.51 - - - https://www.lamayor.org/WiSTEM - 2020-02-13T03:32:36+00:00 - 0.51 - - - https://www.lamayor.org/mayors-office-public-safety - 2020-02-13T03:32:37+00:00 - 0.51 - - - https://www.lamayor.org/files/2406380406764fbf0e30copng - 2020-02-13T03:32:38+00:00 - 0.51 - - - https://www.lamayor.org/mayors-office-city-homelessness-initiatives - 2020-02-13T03:32:39+00:00 - 0.51 - - - https://www.lamayor.org/files/mochijpg - 2020-02-13T03:32:40+00:00 - 0.51 - - - https://www.lamayor.org/mayors-office-economic-opportunity - 2020-02-13T03:32:41+00:00 - 0.51 - - - https://www.lamayor.org/files/4113988404454a0779444opng - 2020-02-13T03:32:42+00:00 - 0.51 - - - https://www.lamayor.org/mayors-office-budget-and-innovation - 2020-02-13T03:32:43+00:00 - 0.51 - - - https://www.lamayor.org/files/41567263871ab63f4bd6eopng - 2020-02-13T03:32:44+00:00 - 0.51 - - - https://www.lamayor.org/mayors-office-city-services - 2020-02-13T03:32:45+00:00 - 0.51 - - - https://www.lamayor.org/files/32159890491717519b700ojpg - 2020-02-13T03:32:46+00:00 - 0.51 - - - https://www.lamayor.org/mayors-office-international-affairs - 2020-02-13T03:32:47+00:00 - 0.51 - - - https://www.lamayor.org/files/253401907572eb44d57a9opng - 2020-02-13T03:32:48+00:00 - 0.51 - - - https://www.lamayor.org/files/megmoedwebpng-0 - 2020-02-13T03:32:49+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-announces-program-provide-free-dash-bus-passes-students - 2020-02-13T03:32:50+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-launches-%E2%80%98connect-success%E2%80%99-provide-free-laptops-la-college-promise-students - 2020-02-13T03:32:51+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-launches-%E2%80%98college-corners%E2%80%99-help-ease-la-students%E2%80%99-path-higher-education - 2020-02-13T03:32:52+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-announces-record-investment-la%E2%80%99s-gear-college-readiness-program - 2020-02-13T03:32:53+00:00 - 0.51 - - - https://www.lamayor.org/expanding-supply-housing - 2020-02-13T03:32:53+00:00 - 0.51 - - - https://www.lamayor.org/files/img3555-14x7jpg - 2020-02-13T03:32:54+00:00 - 0.51 - - - https://www.lamayor.org/growing-our-housing-stock-equitable-and-inclusive-way - 2020-02-13T03:32:55+00:00 - 0.51 - - - https://www.lamayor.org/files/rendering-14x7jpg-1 - 2020-02-13T03:32:56+00:00 - 0.51 - - - https://www.lamayor.org/increasing-housing-stability-renters - 2020-02-13T03:32:57+00:00 - 0.51 - - - https://www.lamayor.org/files/renter-14x7jpg - 2020-02-13T03:32:58+00:00 - 0.51 - - - https://www.lamayor.org/files/welcome-home-14x7jpg - 2020-02-13T03:32:59+00:00 - 0.51 - - - https://www.lamayor.org/report-status-women-girls-los-angeles - 2020-02-13T03:33:00+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-welcomes-first-%E2%80%98pledge-patrol%E2%80%99-class - 2020-02-13T03:33:01+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-and-la-county-unveil-new-family-justice-center - 2020-02-13T03:33:02+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-signs-historic-minimum-wage-increase-law - 2020-02-13T03:33:03+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-wraps-asia-trade-mission-stops-vietnam-hong-kong - 2020-02-13T03:33:05+00:00 - 0.51 - - - https://www.lamayor.org/mayor%E2%80%99s-young-ambassador-program - 2020-02-13T03:33:06+00:00 - 0.51 - - - https://www.lamayor.org/mexico-foreign-minister-ebrard-and-la-mayor-garcetti-announce-mexico-la-commission - 2020-02-13T03:33:07+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-nominates-capri-maddox-lead-los-angeles-department-civil-and-human-rights - 2020-02-13T03:33:08+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-announces-program-provide-entrepreneurship-training-angelenos-experiencing - 2020-02-13T03:33:09+00:00 - 0.51 - - - https://www.lamayor.org/la-london-team-transportation-innovation - 2020-02-13T03:33:10+00:00 - 0.51 - - - https://www.lamayor.org/statement-mayor-garcetti-passing-kobe-bryant - 2020-02-13T03:33:11+00:00 - 0.51 - - - https://www.lamayor.org/statement-mayor-garcetti-talks-hud-secretary-ben-carson-federal-partnership-confront-homelessness - 2020-02-13T03:33:12+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-celebrates-metro-board-vote-advance-nextgen-bus-plan - 2020-02-13T03:33:13+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-announces-10-million-investment-jobs-program-formerly-incarcerated-angelenos - 2020-02-13T03:33:14+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-and-entertainment-industry-leaders-launch-%E2%80%98la-collab%E2%80%99-initiative-increase-latinx - 2020-02-13T03:33:15+00:00 - 0.51 - - - https://www.lamayor.org/media/press_releases?page=1 - 2020-02-13T03:33:16+00:00 - 0.51 - - - https://www.lamayor.org/media/press_releases?page=2 - 2020-02-13T03:33:17+00:00 - 0.51 - - - https://www.lamayor.org/media/press_releases?page=3 - 2020-02-13T03:33:18+00:00 - 0.51 - - - https://www.lamayor.org/media/press_releases?page=4 - 2020-02-13T03:33:19+00:00 - 0.51 - - - https://www.lamayor.org/media/press_releases?page=81 - 2020-02-13T03:33:21+00:00 - 0.51 - - - https://www.lamayor.org/state-city-2019 - 2020-02-13T03:33:22+00:00 - 0.51 - - - https://www.lamayor.org/files/sotc2019jpg - 2020-02-13T03:33:22+00:00 - 0.51 - - - https://www.lamayor.org/SOTC2018 - 2020-02-13T03:33:23+00:00 - 0.51 - - - https://www.lamayor.org/files/41505002341db4366c1f3opng-0 - 2020-02-13T03:33:24+00:00 - 0.51 - - - https://www.lamayor.org/files/inaug-2017-2png-0 - 2020-02-13T03:33:25+00:00 - 0.51 - - - https://www.lamayor.org/state-city-2017 - 2020-02-13T03:33:26+00:00 - 0.51 - - - https://www.lamayor.org/files/photospng-0 - 2020-02-13T03:33:28+00:00 - 0.51 - - - https://www.lamayor.org/SOTC2016 - 2020-02-13T03:33:29+00:00 - 0.51 - - - https://www.lamayor.org/files/nk3a0492-cardjpg-0 - 2020-02-13T03:33:30+00:00 - 0.51 - - - https://www.lamayor.org/files/2015png - 2020-02-13T03:33:31+00:00 - 0.51 - - - https://www.lamayor.org/files/2014png - 2020-02-13T03:33:32+00:00 - 0.51 - - - https://www.lamayor.org/files/inaug-2013png - 2020-02-13T03:33:32+00:00 - 0.51 - - - https://www.lamayor.org/files/videojpg - 2020-02-13T03:33:33+00:00 - 0.51 - - - https://www.lamayor.org/files/megpng - 2020-02-13T03:33:34+00:00 - 0.51 - - - https://www.lamayor.org/files/reddit-ama-cannabis-policypng-0 - 2020-02-13T03:33:35+00:00 - 0.51 - - - https://www.lamayor.org/files/la2028png - 2020-02-13T03:33:36+00:00 - 0.51 - - - https://www.lamayor.org/files/c7dfyqvmaacgtqjpg - 2020-02-13T03:33:37+00:00 - 0.51 - - - https://www.lamayor.org/files/rec-and-parks-amapng - 2020-02-13T03:33:38+00:00 - 0.51 - - - https://www.lamayor.org/files/housing-ama-2png - 2020-02-13T03:33:39+00:00 - 0.51 - - - https://www.lamayor.org/files/seleta-reynolds-ladot-amapng-0 - 2020-02-13T03:33:40+00:00 - 0.51 - - - https://www.lamayor.org/files/10-metro-amapng - 2020-02-13T03:33:41+00:00 - 0.51 - - - https://www.lamayor.org/files/09-voting-amajpg - 2020-02-13T03:33:42+00:00 - 0.51 - - - https://www.lamayor.org/files/08-reentry-amapng - 2020-02-13T03:33:43+00:00 - 0.51 - - - https://www.lamayor.org/files/07-lax-amapng - 2020-02-13T03:33:44+00:00 - 0.51 - - - https://www.lamayor.org/files/06-data-ama-redditpng-0 - 2020-02-13T03:33:45+00:00 - 0.51 - - - https://www.lamayor.org/files/05-dept-cultural-afairs-ama-redditpng-0 - 2020-02-13T03:33:46+00:00 - 0.51 - - - https://www.lamayor.org/files/04-matt-petersen-ama-redditpng - 2020-02-13T03:33:47+00:00 - 0.51 - - - https://www.lamayor.org/files/03-library-ama-redditjpg - 2020-02-13T03:33:48+00:00 - 0.51 - - - https://www.lamayor.org/files/02-transpo-amajpg - 2020-02-13T03:33:49+00:00 - 0.51 - - - https://www.lamayor.org/files/01-meg-amajpg - 2020-02-13T03:33:49+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/HEAP%20One%20Pager.pdf - 2018-10-24T15:37:42+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-quickly-directs-new-dollars-bridge-housing-homeless-angelenos-la-receives-85-million - 2020-02-13T03:33:52+00:00 - 0.51 - - - https://www.lamayor.org/confronting-crisis-update-our-progress-help-homeless-angelenos - 2020-02-13T03:33:54+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-hosts-job-fair-connect-angelenos-new-opportunities-homeless-services - 2020-02-13T03:33:55+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-partners-loyola-law-school-offer-free-legal-services-formerly-incarcerated-angelenos - 2020-02-13T03:33:56+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-proposes-fiscally-responsible-2020-budget-record-reserves-investments-homelessness - 2020-02-13T03:33:57+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2016%20-%20Implementation%20of%20the%20Comprehensive%20Homeless%20Strategy%20(1).pdf - 2016-05-05T22:13:22+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-opens-new-hygiene-center-skid-row - 2020-02-13T03:33:59+00:00 - 0.51 - - - https://www.lamayor.org/bridge-home-spanish - 2020-02-13T03:34:00+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-la-city-council-declare-shelter-crisis-mayor-signs-package-ordinances-address-la%E2%80%99s - 2020-02-13T03:34:01+00:00 - 0.51 - - - https://www.lamayor.org/la-opens-new-homeless-help-desk - 2020-02-13T03:34:02+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-announces-new-plan-deploy-new-sanitation-teams-deliver-services-homeless-encampments - 2020-02-13T03:34:03+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/2019.12.18-UHRCPolicyGroupAgenda1.pdf - 2019-12-13T17:50:50+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/2020.1.15-UHRCPolicyGroupAgenda.pdf - 2020-01-09T22:53:00+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/2020.1.29-UHRCPolicyGroupAgendaFINAL.pdf - 2020-01-24T20:10:33+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/2020.2.12-UHRCPolicyGroupAgenda.pdf - 2020-02-10T18:11:03+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/2019.12.18-UHRCPolicyGroupMeetingMinutesFINAL.pdf - 2020-01-24T20:09:09+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/2020.1.15-UHRCPolicyGroupMeetingMinutesFINAL.pdf - 2020-02-05T20:41:32+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/2020.1.29-UHRCPolicyGroupMeetingMinutesDRAFT.pdf - 2020-02-10T18:10:15+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-continues-tackle-housing-crisis-new-law-strengthening-rent-stabilization-ordinance - 2020-02-13T03:34:10+00:00 - 0.51 - - - https://www.lamayor.org/summary-supportive-housing-pipeline - 2020-02-13T03:34:11+00:00 - 0.51 - - - https://www.lamayor.org/summary-hhh-pipeline - 2020-02-13T03:34:13+00:00 - 0.51 - - - https://www.lamayor.org/summary-hhh-projects-development - 2020-02-13T03:34:14+00:00 - 0.51 - - - https://www.lamayor.org/hhh-projects-development-details - 2020-02-13T03:34:16+00:00 - 0.51 - - - https://www.lamayor.org/non-hhh-funded-projects-development-details - 2020-02-13T03:34:17+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2015%20-%20Equitable%20Workforce%20and%20Service%20Restoration.pdf - 2016-04-29T19:39:01+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Introducing%20Resilient%20Los%20Angeles.pdf - 2018-02-28T02:11:49+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Resilient%20Los%20Angeles.pdf - 2018-02-28T02:11:35+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/L.A.%20Resilience%20Report%20Spanish.pdf - 2018-03-02T17:24:49+00:00 - 0.51 - - - https://www.lamayor.org/resilience-design-building-stronger-los-angeles - 2020-02-13T03:34:20+00:00 - 0.51 - - - https://www.lamayor.org/statement-mayor-garcetti-presidents-proposal-defund-earthquake-early-warning-system - 2020-02-13T03:34:21+00:00 - 0.51 - - - https://www.lamayor.org/statement-mayor-garcetti-great-shakeout - 2020-02-13T03:34:22+00:00 - 0.51 - - - https://www.lamayor.org/mayor_garcetti_announces_award_for_resilience_by_design_press_release - 2020-02-13T03:34:23+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-signs-historic-earthquake-retrofit-measure-law-ensures-safety-thousands-angelenos - 2020-02-13T03:34:24+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-announces-new-local-earthquake-fault-study-zones-ensure-new-buildings-are-not-built - 2020-02-13T03:34:25+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-announces-new-chief-resilience-officer-coming-la - 2020-02-13T03:34:26+00:00 - 0.51 - - - https://www.lamayor.org/los-angeles-wins-rockefeller-foundation-grant - 2020-02-13T03:34:27+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/article/files/Fossil%20Fuel%20Free%20Streets%20Declaration.pdf - 2017-10-23T18:23:07+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/article/files/101917%20Me%CC%81morandum%20Paris%20LA%20Olympic%20ENG%20FINAL.pdf - 2017-10-23T18:22:54+00:00 - 0.51 - - - https://www.lamayor.org/los-angeles-named-1-energy-star-city-us - 2020-02-13T03:34:29+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-announces-new-100-million-insulation-rebate-program - 2020-02-13T03:34:30+00:00 - 0.51 - - - https://www.lamayor.org/strong-families-la-esp - 2020-02-13T03:34:31+00:00 - 0.51 - - - https://www.lamayor.org/files/sfla-icon-question-14x7jpg-1 - 2020-02-13T03:34:32+00:00 - 0.51 - - - https://www.lamayor.org/files/sfla-icon-local-14x7jpg-0 - 2020-02-13T03:34:33+00:00 - 0.51 - - - https://www.lamayor.org/files/sfla-nounimmigration31-14x7jpg-0 - 2020-02-13T03:34:34+00:00 - 0.51 - - - https://www.lamayor.org/files/sfla-icon-family0-14x70jpg - 2020-02-13T03:34:35+00:00 - 0.51 - - - https://www.lamayor.org/files/sfla-nounquestions55-14x7jpg - 2020-02-13T03:34:36+00:00 - 0.51 - - - https://www.lamayor.org/files/sfla-mail-icon-14x7jpg-0 - 2020-02-13T03:34:37+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Public%20Comment%20Letter_0.pdf - 2019-10-10T13:43:29+00:00 - 0.51 - - - https://www.lamayor.org/files/sfla-icon-megaphone-14x7jpg-0 - 2020-02-13T03:34:38+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_January_2018_0.pdf - 2018-08-01T22:39:53+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_February_2018_0.pdf - 2018-08-01T22:40:01+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_March_2018_0.pdf - 2018-08-01T22:40:25+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_April_2018_0.pdf - 2018-08-01T22:40:47+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_May_2018.pdf - 2018-08-01T22:41:15+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_June_2018_0.pdf - 2018-08-01T22:41:24+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_July_2018.pdf - 2018-08-01T22:41:33+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_AugustSeptember_2018.pdf - 2018-10-02T22:44:44+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_October_2018_0.pdf - 2018-11-02T21:10:42+00:00 - 0.51 - - - https://www.lamayor.org/statement-mayor-garcetti-federal-government-shutdown - 2020-02-13T03:34:45+00:00 - 0.51 - - - https://www.lamayor.org/mayor-garcetti-launches-new-initiative-expand-support-immigrant-angelenos - 2020-02-13T03:34:46+00:00 - 0.51 - - - https://www.lamayor.org/statement-mayor-garcetti-trump-administration-order-end-temporary-protected-status-el-salvador - 2020-02-13T03:34:47+00:00 - 0.51 - - - https://www.lamayor.org/sites/g/files/wph446/f/article/files/Drop-In-Registration-Form-Spanish.pdf - 2019-01-15T08:03:00+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/article/files/Drop-In-Registration-Form-English_0.pdf - 2019-01-17T02:00:53+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/article/files/LAUSDSchoolVoterRegistrationToolkit.pdf - 2018-09-29T02:52:24+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/article/files/WebinarSlides.pdf - 2018-09-20T16:44:45+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/file/_MYC%20One-Pager%202019-20.pdf - 2019-08-08T23:59:12+00:00 - 0.41 - - - https://www.lamayor.org/budget-resources - 2020-02-13T03:34:55+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/STATEOFTHECITY2019.pdf - 2019-04-18T01:45:43+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/image/LACity_Part1_Demographics.pdf - 2016-02-07T17:58:23+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/image/LACity_Part2_Leadership.pdf - 2016-02-07T17:58:24+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/image/LACity_Part3_Veterans.pdf - 2016-02-07T17:58:24+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/image/LACity_Part4_EducationWorkforceDevelopment.pdf - 2016-02-07T17:58:24+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/image/LACity_Part5_Public_Safety.pdf - 2016-02-07T17:58:24+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/page/image/ED_11.pdf - 2016-02-07T18:05:39+00:00 - 0.41 - - - https://www.lamayor.org/la-puts-higher-education-within-reach-all-students - 2020-02-13T03:35:02+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-reaches-youth-hiring-goal-connecting-more-15000-young-people-jobs-last-year - 2020-02-13T03:35:03+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-completes-trade-mission-trips-tokyo-seoul - 2020-02-13T03:35:04+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-celebrates-opening-first-hhh-funded-supportive-housing-development - 2020-02-13T03:35:05+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-nominates-carolyn-hull-general-manager-economic-and-workforce-development-department - 2020-02-13T03:35:06+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-la-public-library-will-end-late-fines - 2020-02-13T03:35:08+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-announces-major-infrastructure-investments-la-river - 2020-02-13T03:35:09+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-lausd-announce-expansion-program-assist-families-coping-homelessness-housing - 2020-02-13T03:35:10+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-announces-griffith-parkline-shuttle - 2020-02-13T03:35:11+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-councilmember-nury-martinez%E2%80%99s-election-city-council-president - 2020-02-13T03:35:12+00:00 - 0.41 - - - https://www.lamayor.org/mayor-eric-garcetti-meets-community-leaders-urge-neighborhood-census-action - 2020-02-13T03:35:12+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-la-open-new-emergency-shelters-ahead-expected-storm - 2020-02-13T03:35:13+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-joins-los-angeles-cleantech-incubator-announce-new-transportation-electrification - 2020-02-13T03:35:14+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-nominates-heather-repenning-metropolitan-water-district-board-directors - 2020-02-13T03:35:15+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-launches-streetlight-design-competition - 2020-02-13T03:35:16+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-launches-america%E2%80%99s-first-public-private-partnership-transportation-innovation - 2020-02-13T03:35:17+00:00 - 0.41 - - - https://www.lamayor.org/statement-mayor-garcetti-today%E2%80%99s-southern-california-association-governments-vote-housing - 2020-02-13T03:35:19+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-celebrates-final-approval-largest-solar-and-energy-storage-project-us - 2020-02-13T03:35:20+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-nominates-nicole-neeman-brady-ladwp-board-commissioners - 2020-02-13T03:35:21+00:00 - 0.41 - - - https://www.lamayor.org/statement-mayor-garcetti-welcomes-la-students-and-greta-thunberg-city-hall - 2020-02-13T03:35:22+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-kicks-tree-planting-season - 2020-02-13T03:35:23+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-signs-no-fault-eviction-moratorium-ordinance - 2020-02-13T03:35:23+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-kicks-%E2%80%98cool-streets-la%E2%80%99 - 2020-02-13T03:35:25+00:00 - 0.41 - - - https://www.lamayor.org/statement-mayor-garcetti-funding-approval-new-round-prop-hhh-projects - 2020-02-13T03:35:25+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-launches-%E2%80%9Cla-made-40%E2%80%9D-increase-upskilling-la-manufacturing-and-create-new-career - 2020-02-13T03:35:26+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-announces-evacuation-orders-saddleridge-fire-have-been-lifted - 2020-02-13T03:35:27+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcettis-speech-c40-chair-handover-ceremony-copenhagen-denmark - 2020-02-13T03:35:28+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-elected-chair-c40-cities-climate-leadership-group - 2020-02-13T03:35:30+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-celebrates-clean-air-day-unveils-new-zero-emission-equipment-port-los-angeles - 2020-02-13T03:35:31+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-announces-formation-international-gender-equity-network - 2020-02-13T03:35:32+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-proposes-first-ever-paid-parental-leave-la-municipal-employees - 2020-02-13T03:35:33+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-walks-class-students-celebrate-international-walk-school-day - 2020-02-13T03:35:34+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-convenes-inaugural-meeting-youth-council-climate-action - 2020-02-13T03:35:35+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-joins-irish-prime-minister-leo-varadkar-open-new-consulate-los-angeles - 2020-02-13T03:35:36+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-nominates-m-teresa-villegas-board-public-works - 2020-02-13T03:35:37+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-names-nora-adriana-preciado-director-immigrant-affairs - 2020-02-13T03:35:38+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-joins-la-students-global-climate-strike - 2020-02-13T03:35:39+00:00 - 0.41 - - - https://www.lamayor.org/media/press_releases?page=5 - 2020-02-13T03:35:40+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-announces-new-americans-citizenship-center-echo-park-public-library - 2020-02-13T03:35:41+00:00 - 0.41 - - - https://www.lamayor.org/statement-mayor-garcetti-president-trump%E2%80%99s-move-revoke-california%E2%80%99s-authority-regulate-tailpipe - 2020-02-13T03:35:42+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-launches-country%E2%80%99s-first-city-developed-threat-sharing-platform-and-public - 2020-02-13T03:35:43+00:00 - 0.41 - - - https://www.lamayor.org/statement-mayor-garcetti-confirmation-ladwp-general-manager-marty-adams - 2020-02-13T03:35:44+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-creates-dwp-inspector-general - 2020-02-13T03:35:45+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-and-los-angeles-world-airports-break-ground-historic-consolidated-rental-car-facility - 2020-02-13T03:35:46+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-and-mexican-undersecretary-jes%C3%BAs-seade-kick-first-mexla-meeting - 2020-02-13T03:35:47+00:00 - 0.41 - - - https://www.lamayor.org/garcetti-administration-approves-building-largest-solar-and-battery-storage-project-us - 2020-02-13T03:35:48+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-letter-president-trump-homelessness-crisis - 2020-02-13T03:35:49+00:00 - 0.41 - - - https://www.lamayor.org/la-leaders-join-meeting-governor%E2%80%99s-council-regional-homeless-advisors - 2020-02-13T03:35:50+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-appoints-shannon-hoppes-chief-procurement-officer - 2020-02-13T03:35:51+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-stresses-innovation-accountability-yearly-review-general-managers - 2020-02-13T03:35:52+00:00 - 0.41 - - - https://www.lamayor.org/media/press_releases?page=6 - 2020-02-13T03:35:53+00:00 - 0.41 - - - https://www.lamayor.org/mayor_garcetti_kicks_off_2013_hire_la_summer_youth_employment_program - 2020-02-13T03:35:54+00:00 - 0.41 - - - https://www.lamayor.org/mayor_garcetti_announces_appointments - 2020-02-13T03:35:55+00:00 - 0.41 - - - https://www.lamayor.org/media/press_releases?page=80 - 2020-02-13T03:35:56+00:00 - 0.41 - - - https://www.lamayor.org/media/press_releases?page=77 - 2020-02-13T03:35:57+00:00 - 0.41 - - - https://www.lamayor.org/media/press_releases?page=78 - 2020-02-13T03:35:58+00:00 - 0.41 - - - https://www.lamayor.org/media/press_releases?page=79 - 2020-02-13T03:35:59+00:00 - 0.41 - - - https://www.lamayor.org/files/46909171964d984286dceojpg - 2020-02-13T03:36:00+00:00 - 0.41 - - - https://www.lamayor.org/files/image-1jpg - 2020-02-13T03:36:01+00:00 - 0.41 - - - https://www.lamayor.org/files/aa1jpg - 2020-02-13T03:36:02+00:00 - 0.41 - - - https://www.lamayor.org/state-city-2019-data - 2020-02-13T03:36:03+00:00 - 0.41 - - - https://www.lamayor.org/files/datajpg - 2020-02-13T03:36:04+00:00 - 0.41 - - - https://www.lamayor.org/files/41505002341db4366c1f3opng - 2020-02-13T03:36:05+00:00 - 0.41 - - - https://www.lamayor.org/files/sotc-year-recap-tilepng - 2020-02-13T03:36:06+00:00 - 0.41 - - - https://www.lamayor.org/files/text-speechpng - 2020-02-13T03:36:07+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/2018_SOTC_AS_PREPARED_FOR_DELIVERY%20%28ESP%29.pdf.pdf - 2018-04-16T23:38:39+00:00 - 0.41 - - - https://www.lamayor.org/files/text-speechpng-0 - 2020-02-13T03:36:08+00:00 - 0.41 - - - https://www.lamayor.org/sotc-2018-data - 2020-02-13T03:36:09+00:00 - 0.41 - - - https://www.lamayor.org/files/data-tilepng - 2020-02-13T03:36:10+00:00 - 0.41 - - - https://www.lamayor.org/files/cisco-brotherspng - 2020-02-13T03:36:11+00:00 - 0.41 - - - https://www.lamayor.org/files/taliapng - 2020-02-13T03:36:12+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/StateoftheCity2017--ForMediaDistribution.pdf - 2017-04-20T18:18:09+00:00 - 0.41 - - - https://www.lamayor.org/files/textpng - 2020-02-13T03:36:14+00:00 - 0.41 - - - https://www.lamayor.org/files/datapng - 2020-02-13T03:36:15+00:00 - 0.41 - - - https://www.lamayor.org/files/photospng - 2020-02-13T03:36:16+00:00 - 0.41 - - - https://www.lamayor.org/files/0a7a0693jpg - 2020-02-13T03:36:16+00:00 - 0.41 - - - https://www.lamayor.org/files/nk3a0492-cardjpg - 2020-02-13T03:36:17+00:00 - 0.41 - - - https://www.lamayor.org/files/web-stuffsjpg - 2020-02-13T03:36:18+00:00 - 0.41 - - - https://www.lamayor.org/files/nbpng - 2020-02-13T03:36:19+00:00 - 0.41 - - - https://www.lamayor.org/files/brpng-1 - 2020-02-13T03:36:20+00:00 - 0.41 - - - https://www.lamayor.org/files/dapng-0 - 2020-02-13T03:36:21+00:00 - 0.41 - - - https://www.lamayor.org/files/497a4260jpg - 2020-02-13T03:36:22+00:00 - 0.41 - - - https://www.lamayor.org/files/garcetti-sotc-infographic-11-infrastructurepng-0 - 2020-02-13T03:36:23+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/2016%20State%20of%20the%20City%20Address.pdf - 2016-04-15T01:46:10+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-proposes-fiscally-responsible-2019-budget-boosts-investment-homelessness - 2020-02-13T03:36:25+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-signs-executive-directive-accelerate-construction-emergency-homeless-shelters - 2020-02-13T03:36:25+00:00 - 0.41 - - - https://www.lamayor.org/mayor-garcetti-launches-%E2%80%98save-energy-la%E2%80%99-campaign-and-asks-angelenos-conserve-energy-during-summer - 2020-02-13T03:36:27+00:00 - 0.41 - - - https://www.lamayor.org/files/icon-questionpng-0 - 2020-02-13T03:36:28+00:00 - 0.41 - - - https://www.lamayor.org/files/icon-localpng-0 - 2020-02-13T03:36:29+00:00 - 0.41 - - - https://www.lamayor.org/files/sfla-nounimmigration31-14x70jpg - 2020-02-13T03:36:30+00:00 - 0.41 - - - https://www.lamayor.org/files/icon-familypng-1 - 2020-02-13T03:36:31+00:00 - 0.41 - - - https://www.lamayor.org/files/sfla-nounquestions55-14x7jpg-0 - 2020-02-13T03:36:32+00:00 - 0.41 - - - https://www.lamayor.org/files/sfla-mail-icon-14x70jpg - 2020-02-13T03:36:33+00:00 - 0.41 - - - https://www.lamayor.org/files/icon-megaphonepng-1 - 2020-02-13T03:36:34+00:00 - 0.41 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/2021_Mayors_Budget_Policy_Letter.pdf - 2019-11-04T23:04:42+00:00 - 0.33 - - - https://www.lamayor.org/files/letterjpg - 2020-02-13T03:36:35+00:00 - 0.33 - - - https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/CAO_Budget_Instructions.pdf - 2019-11-04T23:04:34+00:00 - 0.33 - - - https://www.lamayor.org/files/caopng - 2020-02-13T03:36:37+00:00 - 0.33 - - - https://www.lamayor.org/files/city-hall2jpg-0 - 2020-02-13T03:36:38+00:00 - 0.33 - - - https://www.lamayor.org/sites/g/files/wph446/f/article/files/Council-Transmittal-Paid-Parental-Leave.pdf - 2019-10-02T20:30:18+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-welcomes-new-school-year-free-dash-bus-service-students - 2020-02-13T03:36:39+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-hosts-state-leaders-highlight-local-and-state-action-strengthen-water-resilience - 2020-02-13T03:36:40+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-nominates-raquel-beltr%C3%A1n-general-manager-department-neighborhood-empowerment - 2020-02-13T03:36:41+00:00 - 0.33 - - - https://www.lamayor.org/mayor%E2%80%99s-youth-council-end-gun-violence-convenes-response-el-paso-and-dayton-shootings - 2020-02-13T03:36:42+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-and-usgs-announce-key-updates-shakealertla - 2020-02-13T03:36:43+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-meets-pope-francis-vatican - 2020-02-13T03:36:44+00:00 - 0.33 - - - https://www.lamayor.org/statement-mayor-garcetti-announces-agreement-us-department-housing-and-urban-development-continued - 2020-02-13T03:36:45+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-names-rachel-malarich-la%E2%80%99s-first-ever-city-forest-officer - 2020-02-13T03:36:46+00:00 - 0.33 - - - https://www.lamayor.org/statement-office-mayor-data-event - 2020-02-13T03:36:47+00:00 - 0.33 - - - https://www.lamayor.org/statement-mayor-garcetti-death-lapd-officer-juan-diaz - 2020-02-13T03:36:48+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-and-councilmember-buscaino-announce-agreement-between-ilwu-apmt-and-pma-launch-blue - 2020-02-13T03:36:49+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-highlights-la%E2%80%99s-leadership-global-development-goals - 2020-02-13T03:36:50+00:00 - 0.33 - - - https://www.lamayor.org/media/press_releases?page=7 - 2020-02-13T03:36:51+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-nominates-tracy-quinn-metropolitan-water-district-board-directors - 2020-02-13T03:36:52+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-la-will-fight-keep-families-together-%E2%80%94-and-will-not-assist-immigration-enforcement - 2020-02-13T03:36:53+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-launches-initiative-strengthen-emergency-preparedness-la-neighborhoods - 2020-02-13T03:36:54+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-names-dae-levine-director-communications-andre-herndon-senior-communications-advisor - 2020-02-13T03:36:56+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-announces-ladwp-power-outage-notification-service - 2020-02-13T03:36:57+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-leads-coalition-mayors-calling-federal-action-confront-homelessness-crisis - 2020-02-13T03:36:58+00:00 - 0.33 - - - https://www.lamayor.org/statement-mayor-garcetti-supreme-court-ruling-census-citizenship-question - 2020-02-13T03:36:59+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-la-won%E2%80%99t-assist-immigration-enforcement-will-help-connect-families-resources - 2020-02-13T03:37:00+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-nominates-denise-verret-director-la-zoo-and-botanical-gardens - 2020-02-13T03:37:01+00:00 - 0.33 - - - https://www.lamayor.org/statement-mayor-garcetti-president-trump%E2%80%99s-threats-against-immigrants - 2020-02-13T03:37:01+00:00 - 0.33 - - - https://www.lamayor.org/media/press_releases?page=8 - 2020-02-13T03:37:02+00:00 - 0.33 - - - https://www.lamayor.org/mayor_garcetti_announces_grant_to_fund_nuclear_detection_halo - 2020-02-13T03:37:04+00:00 - 0.33 - - - https://www.lamayor.org/mayor_garcetti_welcomes_riot_games_to_los_angeles - 2020-02-13T03:37:05+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-issues-executive-directive-cyber-security - 2020-02-13T03:37:05+00:00 - 0.33 - - - https://www.lamayor.org/mayor_garcetti_announces_harbor_commission_appointments - 2020-02-13T03:37:07+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-issues-first-executive-directive-launch-great-streets-initiative - 2020-02-13T03:37:08+00:00 - 0.33 - - - https://www.lamayor.org/mayor_garcetti_discusses_first_100_days_and_previews_performance_metrics - 2020-02-13T03:37:08+00:00 - 0.33 - - - https://www.lamayor.org/mayor_garcetti_announces_dwp_commission_appointments - 2020-02-13T03:37:09+00:00 - 0.33 - - - https://www.lamayor.org/matt-petersen-appointed-new-position-chief-sustainability-officer - 2020-02-13T03:37:10+00:00 - 0.33 - - - https://www.lamayor.org/mayor_garcetti_opens_help_desk - 2020-02-13T03:37:11+00:00 - 0.33 - - - https://www.lamayor.org/mayor_garcetti_announces_mta_board_appointments - 2020-02-13T03:37:12+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-announces-appointments - 2020-02-13T03:37:13+00:00 - 0.33 - - - https://www.lamayor.org/latest-update-water-main-break - 2020-02-13T03:37:14+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-appoints-ralph-terrazas-chief-la-fire-department - 2020-02-13T03:37:15+00:00 - 0.33 - - - https://www.lamayor.org/mayor-garcetti-thanks-la-employers-hiring-10000-youth-summer-jobs-through-hire-las-youth-initiative - 2020-02-13T03:37:16+00:00 - 0.33 - - - - \ No newline at end of file From 33fdb56219821f08968d84b0667e50e27ddc9db5 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:10:25 -0700 Subject: [PATCH 02/14] Init new folder structure with setup for docker & testing --- projects/web-scraper/.dockerignore | 6 + projects/web-scraper/.eslintignore | 3 + projects/web-scraper/.gitignore | 17 +- .../.semaphore/create-kubeconfig.secrets.bash | 6 + .../web-scraper/.semaphore/deploy-prod.yml | 25 + .../.semaphore/native-integration.yml | 58 + .../.semaphore/prod.variables.secrets.yml | 18 + .../web-scraper/.semaphore/push-images.yml | 36 + projects/web-scraper/.semaphore/semaphore.yml | 100 + .../.semaphore/test.variables.secrets.yml | 18 + projects/web-scraper/Archive.zip | Bin 0 -> 33854 bytes projects/web-scraper/CONTRIBUTING.md | 90 + projects/web-scraper/Dockerfile | 15 + projects/web-scraper/LICENSE.md | 19 + projects/web-scraper/Procfile | 1 + projects/web-scraper/README.md | 180 ++ projects/web-scraper/api-deployment.yaml | 39 + projects/web-scraper/api-service.yaml | 21 + projects/web-scraper/app.js | 5 + .../web-scraper/build_image_if_not_exists.sh | 8 + projects/web-scraper/client-deployment.yaml | 31 + projects/web-scraper/client-service.yaml | 21 + projects/web-scraper/client/Dockerfile | 5 + projects/web-scraper/client/default.conf | 11 + projects/web-scraper/client/index.html | 25 + projects/web-scraper/docker-compose.yml | 48 + projects/web-scraper/docker-kompose.yml | 43 + projects/web-scraper/eslintrc.js | 43 + projects/web-scraper/export-image-tag.sh | 1 + projects/web-scraper/generate-node-env.sh | 8 + projects/web-scraper/kops.sh | 25 + .../web-scraper/lib/auth/AuthController.js | 21 + projects/web-scraper/lib/auth/AuthProvider.js | 128 + projects/web-scraper/lib/auth/VerifyToken.js | 26 + projects/web-scraper/lib/config.js | 17 + projects/web-scraper/lib/config/dev.js | 9 + projects/web-scraper/lib/config/prod.js | 9 + projects/web-scraper/lib/config/test.js | 9 + projects/web-scraper/lib/db.js | 12 + projects/web-scraper/lib/helpers/HttpError.js | 11 + projects/web-scraper/lib/note/Note.js | 9 + .../web-scraper/lib/note/NoteController.js | 27 + projects/web-scraper/lib/note/NoteProvider.js | 45 + projects/web-scraper/lib/note/validateNote.js | 9 + projects/web-scraper/lib/routes.js | 34 + projects/web-scraper/lib/server.js | 44 + projects/web-scraper/lib/spider/siteMap.xml | 2512 +++++++++++++++++ projects/web-scraper/lib/spider/spider.js | 9 + .../lib/spider/spiderController.js | 143 + .../web-scraper/lib/spider/spiderProvider.js | 45 + .../web-scraper/lib/spider/validateSpider.js | 9 + projects/web-scraper/lib/user/User.js | 9 + .../web-scraper/lib/user/UserController.js | 27 + projects/web-scraper/lib/user/UserProvider.js | 45 + projects/web-scraper/mongo-deployment.yaml | 39 + .../mongo-persistentvolumeclaim.yaml | 14 + projects/web-scraper/mongo-service.yaml | 20 + projects/web-scraper/package.json | 61 + projects/web-scraper/sample.variables.env | 4 + projects/web-scraper/smoke_test.sh | 14 + projects/web-scraper/test-integration.js | 2 + projects/web-scraper/test-unit.js | 2 + projects/web-scraper/test.js | 67 + projects/web-scraper/test/integration/Auth.js | 217 ++ projects/web-scraper/test/integration/Note.js | 118 + projects/web-scraper/test/integration/User.js | 142 + projects/web-scraper/test/unit/Auth.spec.js | 18 + .../web-scraper/test/unit/HttpError.spec.js | 17 + projects/web-scraper/test/unit/Note.spec.js | 18 + projects/web-scraper/test/unit/User.spec.js | 18 + 70 files changed, 4905 insertions(+), 1 deletion(-) create mode 100644 projects/web-scraper/.dockerignore create mode 100644 projects/web-scraper/.eslintignore create mode 100644 projects/web-scraper/.semaphore/create-kubeconfig.secrets.bash create mode 100644 projects/web-scraper/.semaphore/deploy-prod.yml create mode 100644 projects/web-scraper/.semaphore/native-integration.yml create mode 100644 projects/web-scraper/.semaphore/prod.variables.secrets.yml create mode 100644 projects/web-scraper/.semaphore/push-images.yml create mode 100644 projects/web-scraper/.semaphore/semaphore.yml create mode 100644 projects/web-scraper/.semaphore/test.variables.secrets.yml create mode 100644 projects/web-scraper/Archive.zip create mode 100644 projects/web-scraper/CONTRIBUTING.md create mode 100644 projects/web-scraper/Dockerfile create mode 100644 projects/web-scraper/LICENSE.md create mode 100644 projects/web-scraper/Procfile create mode 100644 projects/web-scraper/README.md create mode 100644 projects/web-scraper/api-deployment.yaml create mode 100644 projects/web-scraper/api-service.yaml create mode 100644 projects/web-scraper/app.js create mode 100755 projects/web-scraper/build_image_if_not_exists.sh create mode 100644 projects/web-scraper/client-deployment.yaml create mode 100644 projects/web-scraper/client-service.yaml create mode 100644 projects/web-scraper/client/Dockerfile create mode 100644 projects/web-scraper/client/default.conf create mode 100644 projects/web-scraper/client/index.html create mode 100644 projects/web-scraper/docker-compose.yml create mode 100644 projects/web-scraper/docker-kompose.yml create mode 100644 projects/web-scraper/eslintrc.js create mode 100644 projects/web-scraper/export-image-tag.sh create mode 100755 projects/web-scraper/generate-node-env.sh create mode 100644 projects/web-scraper/kops.sh create mode 100644 projects/web-scraper/lib/auth/AuthController.js create mode 100644 projects/web-scraper/lib/auth/AuthProvider.js create mode 100644 projects/web-scraper/lib/auth/VerifyToken.js create mode 100644 projects/web-scraper/lib/config.js create mode 100644 projects/web-scraper/lib/config/dev.js create mode 100644 projects/web-scraper/lib/config/prod.js create mode 100644 projects/web-scraper/lib/config/test.js create mode 100644 projects/web-scraper/lib/db.js create mode 100644 projects/web-scraper/lib/helpers/HttpError.js create mode 100644 projects/web-scraper/lib/note/Note.js create mode 100644 projects/web-scraper/lib/note/NoteController.js create mode 100644 projects/web-scraper/lib/note/NoteProvider.js create mode 100644 projects/web-scraper/lib/note/validateNote.js create mode 100644 projects/web-scraper/lib/routes.js create mode 100644 projects/web-scraper/lib/server.js create mode 100644 projects/web-scraper/lib/spider/siteMap.xml create mode 100644 projects/web-scraper/lib/spider/spider.js create mode 100644 projects/web-scraper/lib/spider/spiderController.js create mode 100644 projects/web-scraper/lib/spider/spiderProvider.js create mode 100644 projects/web-scraper/lib/spider/validateSpider.js create mode 100644 projects/web-scraper/lib/user/User.js create mode 100644 projects/web-scraper/lib/user/UserController.js create mode 100644 projects/web-scraper/lib/user/UserProvider.js create mode 100644 projects/web-scraper/mongo-deployment.yaml create mode 100644 projects/web-scraper/mongo-persistentvolumeclaim.yaml create mode 100644 projects/web-scraper/mongo-service.yaml create mode 100644 projects/web-scraper/package.json create mode 100644 projects/web-scraper/sample.variables.env create mode 100755 projects/web-scraper/smoke_test.sh create mode 100644 projects/web-scraper/test-integration.js create mode 100644 projects/web-scraper/test-unit.js create mode 100644 projects/web-scraper/test.js create mode 100644 projects/web-scraper/test/integration/Auth.js create mode 100644 projects/web-scraper/test/integration/Note.js create mode 100644 projects/web-scraper/test/integration/User.js create mode 100644 projects/web-scraper/test/unit/Auth.spec.js create mode 100644 projects/web-scraper/test/unit/HttpError.spec.js create mode 100644 projects/web-scraper/test/unit/Note.spec.js create mode 100644 projects/web-scraper/test/unit/User.spec.js diff --git a/projects/web-scraper/.dockerignore b/projects/web-scraper/.dockerignore new file mode 100644 index 0000000..9cad99b --- /dev/null +++ b/projects/web-scraper/.dockerignore @@ -0,0 +1,6 @@ +node_modules +coverage +.nyc_output +mongo +.vscode +logs \ No newline at end of file diff --git a/projects/web-scraper/.eslintignore b/projects/web-scraper/.eslintignore new file mode 100644 index 0000000..c52047e --- /dev/null +++ b/projects/web-scraper/.eslintignore @@ -0,0 +1,3 @@ +# ignore these files +test.js +test/* \ No newline at end of file diff --git a/projects/web-scraper/.gitignore b/projects/web-scraper/.gitignore index 7b0d57b..72735d1 100644 --- a/projects/web-scraper/.gitignore +++ b/projects/web-scraper/.gitignore @@ -1,2 +1,17 @@ ./output/** -./node_modules/** \ No newline at end of file +./node_modules/** +node_modules +coverage +.nyc_output +mongo +.vscode +logs +variables.env +prod.variables.env +.env +test.variables.env + +# semaphore secrets +.semaphore/docker.secrets.yml +.semaphore/kops.secrets.yml +.semaphore/ssh.secrets.yml \ No newline at end of file diff --git a/projects/web-scraper/.semaphore/create-kubeconfig.secrets.bash b/projects/web-scraper/.semaphore/create-kubeconfig.secrets.bash new file mode 100644 index 0000000..842c061 --- /dev/null +++ b/projects/web-scraper/.semaphore/create-kubeconfig.secrets.bash @@ -0,0 +1,6 @@ +KUBECONFIG=${HOME}/${KOPS_CLUSTER_NAME}-kubeconfig.yaml kops export kubecfg --name ${KOPS_CLUSTER_NAME} --state ${KOPS_STATE_STORE} + +sem create secret kubeconfig \ + --file ${HOME}/${KOPS_CLUSTER_NAME}-kubeconfig.yaml:/home/semaphore/.kube/config + +sem get secrets kubeconfig diff --git a/projects/web-scraper/.semaphore/deploy-prod.yml b/projects/web-scraper/.semaphore/deploy-prod.yml new file mode 100644 index 0000000..9fe0262 --- /dev/null +++ b/projects/web-scraper/.semaphore/deploy-prod.yml @@ -0,0 +1,25 @@ +version: v1.0 +name: Deploy to Production K8S Cluster +agent: + machine: + type: e1-standard-2 + os_image: ubuntu1804 + +blocks: + - name: "Deploy" + task: + jobs: + - name: kubectl apply + commands: + - checkout + - export GIT_HASH=$(git log --format=format:'%h' -1) + - export TAG=${SEMAPHORE_GIT_BRANCH}_${GIT_HASH}_${SEMAPHORE_WORKFLOW_ID} + - docker pull $CLIENT_IMAGE:$TAG + - docker pull $API_IMAGE:$TAG + - envsubst '${TAG}' api-deployment.prod.yaml + - envsubst '${TAG}' client-deployment.prod.yaml + - kubectl apply -f api-deployment.prod.yaml + - kubectl apply -f client-deployment.prod.yaml + secrets: + - name: kubeconfig + - name: prod-variables diff --git a/projects/web-scraper/.semaphore/native-integration.yml b/projects/web-scraper/.semaphore/native-integration.yml new file mode 100644 index 0000000..69589c9 --- /dev/null +++ b/projects/web-scraper/.semaphore/native-integration.yml @@ -0,0 +1,58 @@ +version: v1.0 +name: Native Integration +agent: + machine: + type: e1-standard-2 + os_image: ubuntu1804 + +blocks: + + # Native + + - name: "Smoke Tests" + task: + jobs: + - name: curl /api + commands: + - checkout + + - docker run -d -p 27017:27017 mongo:4.0 + - mkdir docker_images + - docker save mongo:4.0 -o docker_images/mongo:4.0.tar + - cache restore node_modules-$SEMAPHORE_GIT_BRANCH-$(checksum package-lock.json),node_modules-$SEMAPHORE_GIT_BRANCH-,node_modules + - cache restore docker-images-$SEMAPHORE_WORKFLOW_ID + + - npm i + - npm start & + - ./smoke_test.sh localhost:3000/api + + - cache store node_modules-$SEMAPHORE_GIT_BRANCH-$(checksum package-lock.json) node_modules + - cache store docker-images-$SEMAPHORE_WORKFLOW_ID docker_images + + - name: "Unit & Integration Tests" + task: + jobs: + - name: npm run lint + commands: + - checkout + - cache restore node_modules-$SEMAPHORE_GIT_BRANCH-$(checksum package-lock.json),node_modules- + - npm i + - npm run lint + - name: npm run test-unit + commands: + - checkout + - cache restore docker-images-$SEMAPHORE_WORKFLOW_ID + - docker load -i docker_images/mongo:4.0.tar + - docker run -d -p 27017:27017 mongo:4.0 + - cache restore node_modules-$SEMAPHORE_GIT_BRANCH-$(checksum package-lock.json),node_modules- + - npm i + - npm run test-unit + - name: npm run test-integration + commands: + - checkout + - cache restore docker-images-$SEMAPHORE_WORKFLOW_ID + - docker load -i docker_images/mongo:4.0.tar + - docker run -d -p 27017:27017 mongo:4.0 + - cache restore node_modules-$SEMAPHORE_GIT_BRANCH-$(checksum package-lock.json),node_modules- + - npm i + - npm run test-integration \ No newline at end of file diff --git a/projects/web-scraper/.semaphore/prod.variables.secrets.yml b/projects/web-scraper/.semaphore/prod.variables.secrets.yml new file mode 100644 index 0000000..b8cb25d --- /dev/null +++ b/projects/web-scraper/.semaphore/prod.variables.secrets.yml @@ -0,0 +1,18 @@ +apiVersion: v1beta +kind: Secret +metadata: + name: prod-variables +data: + env_vars: + - name: SECRET + value: secretkey + - name: NODE_ENV + value: prod + - name: DB + value: mongodb://mongo:27017/boilerplate_api + - name: PORT + value: 3000 + - name: API_IMAGE + value: adnanrahic/boilerplate-api + - name: CLIENT_IMAGE + value: adnanrahic/smoketest \ No newline at end of file diff --git a/projects/web-scraper/.semaphore/push-images.yml b/projects/web-scraper/.semaphore/push-images.yml new file mode 100644 index 0000000..a7c18e2 --- /dev/null +++ b/projects/web-scraper/.semaphore/push-images.yml @@ -0,0 +1,36 @@ +version: v1.0 +name: Push images to Docker Hub +agent: + machine: + type: e1-standard-2 + os_image: ubuntu1804 + +blocks: + - name: "Push Images" + task: + jobs: + - name: docker push images + commands: + - checkout + - export GIT_HASH=$(git log --format=format:'%h' -1) + - export TAG=${SEMAPHORE_GIT_BRANCH}_${GIT_HASH}_${SEMAPHORE_WORKFLOW_ID} + - docker login -u $DOCKER_USER -p $DOCKER_PASS + - cache restore docker-images-$SEMAPHORE_WORKFLOW_ID + - ./build_image_if_not_exists.sh + - echo "Push API image" + - docker push $API_IMAGE:$TAG + - echo "Push CLIENT image" + - docker push $CLIENT_IMAGE:$TAG + - docker images + - cache delete docker-images-$SEMAPHORE_WORKFLOW_ID + secrets: + - name: docker-secrets + - name: prod-variables + +promotions: + - name: Deploy Production + pipeline_file: deploy-prod.yml + auto_promote_on: + - result: passed + branch: + - master \ No newline at end of file diff --git a/projects/web-scraper/.semaphore/semaphore.yml b/projects/web-scraper/.semaphore/semaphore.yml new file mode 100644 index 0000000..7fad561 --- /dev/null +++ b/projects/web-scraper/.semaphore/semaphore.yml @@ -0,0 +1,100 @@ +version: v1.0 +name: Build & Test +agent: + machine: + type: e1-standard-2 + os_image: ubuntu1804 + +blocks: + + # Docker only + - name: "Build and cache images" + task: + jobs: + - name: docker-compose build && cache store + commands: + - checkout + - export GIT_HASH=$(git log --format=format:'%h' -1) + - export TAG=${SEMAPHORE_GIT_BRANCH}_${GIT_HASH}_${SEMAPHORE_WORKFLOW_ID} + - docker pull mongo:4.0 + - ./generate-node-env.sh + - docker-compose build + + - mkdir docker_images + - docker save $API_IMAGE:$TAG -o docker_images/api_image_${TAG}.tar + - docker save $CLIENT_IMAGE:$TAG -o docker_images/client_image_${TAG}.tar + - docker save mongo:4.0 -o docker_images/mongo_4.0.tar + + - cache store docker-images-$SEMAPHORE_WORKFLOW_ID docker_images + secrets: + - name: prod-variables + + - name: "Smoke tests" + task: + jobs: + - name: CURL /api + commands: + - checkout + - export GIT_HASH=$(git log --format=format:'%h' -1) + - export TAG=${SEMAPHORE_GIT_BRANCH}_${GIT_HASH}_${SEMAPHORE_WORKFLOW_ID} + - cache restore docker-images-$SEMAPHORE_WORKFLOW_ID + - ls -l docker_images + - docker load -i docker_images/api_image_${TAG}.tar + - docker load -i docker_images/client_image_${TAG}.tar + - docker load -i docker_images/mongo_4.0.tar + - docker images + - ./generate-node-env.sh + - docker-compose up -d --build + - sleep 1 + - ./smoke_test.sh localhost:3000/api + - ./smoke_test.sh localhost + - ./smoke_test.sh localhost/api + secrets: + - name: prod-variables + + - name: "Unit & Integration tests" + task: + jobs: + - name: npm run lint + commands: + - checkout + - cache restore node_modules-$SEMAPHORE_GIT_BRANCH-$(checksum package-lock.json),node_modules-$SEMAPHORE_GIT_BRANCH-,node_modules + - npm i + - npm run lint + - cache store node_modules-$SEMAPHORE_GIT_BRANCH-$(checksum package-lock.json) node_modules + - name: npm run test-unit + commands: + - checkout + - export GIT_HASH=$(git log --format=format:'%h' -1) + - export TAG=${SEMAPHORE_GIT_BRANCH}_${GIT_HASH}_${SEMAPHORE_WORKFLOW_ID} + - cache restore docker-images-$SEMAPHORE_WORKFLOW_ID + - ls -l docker_images + - docker load -i docker_images/api_image_${TAG}.tar + - docker load -i docker_images/client_image_${TAG}.tar + - docker load -i docker_images/mongo_4.0.tar + - docker images + - ./generate-node-env.sh + - docker-compose build + - docker-compose run api node test-unit.js bash + - name: npm run test-integration + commands: + - checkout + - export GIT_HASH=$(git log --format=format:'%h' -1) + - export TAG=${SEMAPHORE_GIT_BRANCH}_${GIT_HASH}_${SEMAPHORE_WORKFLOW_ID} + - cache restore docker-images-$SEMAPHORE_WORKFLOW_ID + - ls -l docker_images + - docker load -i docker_images/api_image_${TAG}.tar + - docker load -i docker_images/client_image_${TAG}.tar + - docker load -i docker_images/mongo_4.0.tar + - docker images + - ./generate-node-env.sh + - docker-compose build + - docker-compose run api node test-integration.js bash + secrets: + - name: test-variables + +promotions: + - name: Push Images + pipeline_file: push-images.yml + auto_promote_on: + - result: passed \ No newline at end of file diff --git a/projects/web-scraper/.semaphore/test.variables.secrets.yml b/projects/web-scraper/.semaphore/test.variables.secrets.yml new file mode 100644 index 0000000..97dcc17 --- /dev/null +++ b/projects/web-scraper/.semaphore/test.variables.secrets.yml @@ -0,0 +1,18 @@ +apiVersion: v1beta +kind: Secret +metadata: + name: test-variables +data: + env_vars: + - name: SECRET + value: secretkey + - name: NODE_ENV + value: test + - name: DB + value: mongodb://mongo:27017/boilerplate_api_test + - name: PORT + value: 3000 + - name: API_IMAGE + value: adnanrahic/boilerplate-api + - name: CLIENT_IMAGE + value: adnanrahic/smoketest \ No newline at end of file diff --git a/projects/web-scraper/Archive.zip b/projects/web-scraper/Archive.zip new file mode 100644 index 0000000000000000000000000000000000000000..f55631d91b7a018c3f313711a7775b29f7799f1f GIT binary patch literal 33854 zcmb4r1yq&mwl>|dknZl3?(XiC?(UNA2I=l@K_o@GJ0zu38fg&t7w)~!ai6=d_dhz8 zgP~(CzuC`x-uZrtGT;z!ATPg84O|p|{_y7?@E`yX24_=SLkA0cCsRf>O$3l>Tvt6O zG*`VVP&5!lpdcCu4Dk1V3R3@F(Ade;(8ZMA+SSO^*xt^}(j54uz`t{GW-u~zws`qk zZtQ=5?FEvnlk4@3^My;3TKVYjWSCg@Ygj}iN5se*X9&0hvs8*=ioLyJ z(~?6qirrl*lRZo{+9GwdZOmjU|DEY+bQjl)SI#Qv@4rD0qo&;EB8mDDrz} zO^>PJXjLQZq(me6nG=1`9|yKko;Xwnw2X(I1-dBR15^8(`SxDP0eFl&-QXaj^p|;X z>dCM&f6a8H1{fp#*9jqSfZ9XDIGRF|mI9513JUTYYvBCd8YZR=Huj$M4o>za44$?& zFY5FBqJB(*oOQP#AoTPOT|}ymugc<^kT89b729nD96^uS2de`L+-nh$0mCkLA^hk~yc{9rl^=5fV;zw!J;_ zEgk>dXz137xjpAa`)-XO;#fxknDQ9bLQKv)j;v4ZX`@c8E%)Pjy1BUfc6M`Q5T$iF zczK?QeM~ED7p^=h+KKzCOrKN^PO~$l5F4wSLA}-qQaa{X+CiBOj4SVN7Ef!I4Bj_7 z($WxWO$-DIyHlIO!&nwEV!`0z#aG}jKDD&>SoQ|`Qy2VqJ|+IUPwfm{EZt1$E$v)P z&7FWh+uQw$@Ab& z)>ftPe^5zi+h;y(9&a967}~_MB|#7AT_4cnS$ibNhU{wYWNmU3WOx|CNgm6qX}!r~ z!aSp@`rzgI*6yMe(1|c`z33Yu)lQw)Bh%6fDyc+$P_m&bQKS;TyRSRRJyAw$*y({t z2d>^dG!#aN zRM42>RY*ExuqiZh`$LpIM~@QY-J+i?g9K(!C9a!81BhjNU_vPux&iv$#%RHl+b6L= zwp>3ql|loyNVCS$js5F^l{4yZ`nBX1rZdH0EgR8ECU|&_cHObnz44qC1WVKa8a>&m z)?3!df)EgPw+H^wEyc#!`+Jbvc7o1v4Xp^=TL^S`R!KV*MM>=<02AX>-`f1eNNloj!4O)hb;@{k4X z{H}5kHz`Zv5U%nBG=$T<_STP^%)hVUP_ShP&(D2^KJ{_de01<70??d@LjGpEei!+)H08EZchL z&m5G9jTKnP+obnKkyU)O8M>tBu6UoSCla6QOQ7B7ctm|jQa!!(cESfaMA-C

_>? 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 P0{tF + +## License + +This project is licensed under the MIT License - see the [LICENSE.md](LICENSE.md) file for details + +## Acknowledgments + +* Mad props to [Luka Pejović](https://github.com/luka454) for helping me understand testing! \ No newline at end of file diff --git a/projects/web-scraper/api-deployment.yaml b/projects/web-scraper/api-deployment.yaml new file mode 100644 index 0000000..eabf093 --- /dev/null +++ b/projects/web-scraper/api-deployment.yaml @@ -0,0 +1,39 @@ +apiVersion: extensions/v1beta1 +kind: Deployment +metadata: + annotations: + kompose.cmd: kompose -f docker-kompose.yml convert + kompose.service.type: LoadBalancer + kompose.version: 1.1.0 (36652f6) + creationTimestamp: null + labels: + io.kompose.service: api + name: api +spec: + replicas: 1 + strategy: {} + template: + metadata: + creationTimestamp: null + labels: + io.kompose.service: api + spec: + containers: + - env: + - name: DB + value: mongodb://mongo:27017/boilerplate_api + - name: GET_HOSTS_FROM + value: dns + - name: NODE_ENV + value: prod + - name: PORT + value: "3000" + - name: SECRET + value: secretkey + image: adnanrahic/boilerplate-api:${TAG} + name: api + ports: + - containerPort: 3000 + resources: {} + restartPolicy: Always +status: {} diff --git a/projects/web-scraper/api-service.yaml b/projects/web-scraper/api-service.yaml new file mode 100644 index 0000000..f2eb468 --- /dev/null +++ b/projects/web-scraper/api-service.yaml @@ -0,0 +1,21 @@ +apiVersion: v1 +kind: Service +metadata: + annotations: + kompose.cmd: kompose -f docker-kompose.yml convert + kompose.service.type: LoadBalancer + kompose.version: 1.1.0 (36652f6) + creationTimestamp: null + labels: + io.kompose.service: api + name: api +spec: + ports: + - name: "3000" + port: 3000 + targetPort: 3000 + selector: + io.kompose.service: api + type: LoadBalancer +status: + loadBalancer: {} diff --git a/projects/web-scraper/app.js b/projects/web-scraper/app.js new file mode 100644 index 0000000..450feff --- /dev/null +++ b/projects/web-scraper/app.js @@ -0,0 +1,5 @@ +require('dotenv').config(); +const server = require('./lib/server'); +server.serve(process.env.NODE_ENV) + .then(app => logger.log('info', `Server running on port ${app.port}`)) + .catch(err => logger.log('error', `Error in app.js at: ${err}`)); \ No newline at end of file diff --git a/projects/web-scraper/build_image_if_not_exists.sh b/projects/web-scraper/build_image_if_not_exists.sh new file mode 100755 index 0000000..6ca8257 --- /dev/null +++ b/projects/web-scraper/build_image_if_not_exists.sh @@ -0,0 +1,8 @@ +if [ -d "docker_images" ]; then + docker load -i docker_images/api_image_${TAG}.tar + docker load -i docker_images/client_image_${TAG}.tar +else + if [[ "$(docker images -q $API_IMAGE:$TAG 2> /dev/null)" == "" ]]; then + docker-compose -f docker-compose.build.yml build + fi +fi \ No newline at end of file diff --git a/projects/web-scraper/client-deployment.yaml b/projects/web-scraper/client-deployment.yaml new file mode 100644 index 0000000..d7c4ffd --- /dev/null +++ b/projects/web-scraper/client-deployment.yaml @@ -0,0 +1,31 @@ +apiVersion: extensions/v1beta1 +kind: Deployment +metadata: + annotations: + kompose.cmd: kompose -f docker-kompose.yml convert + kompose.service.type: LoadBalancer + kompose.version: 1.1.0 (36652f6) + creationTimestamp: null + labels: + io.kompose.service: client + name: client +spec: + replicas: 1 + strategy: {} + template: + metadata: + creationTimestamp: null + labels: + io.kompose.service: client + spec: + containers: + - env: + - name: GET_HOSTS_FROM + value: dns + image: adnanrahic/smoketest:${TAG} + name: client + ports: + - containerPort: 80 + resources: {} + restartPolicy: Always +status: {} diff --git a/projects/web-scraper/client-service.yaml b/projects/web-scraper/client-service.yaml new file mode 100644 index 0000000..9cace33 --- /dev/null +++ b/projects/web-scraper/client-service.yaml @@ -0,0 +1,21 @@ +apiVersion: v1 +kind: Service +metadata: + annotations: + kompose.cmd: kompose -f docker-kompose.yml convert + kompose.service.type: LoadBalancer + kompose.version: 1.1.0 (36652f6) + creationTimestamp: null + labels: + io.kompose.service: client + name: client +spec: + ports: + - name: "80" + port: 80 + targetPort: 80 + selector: + io.kompose.service: client + type: LoadBalancer +status: + loadBalancer: {} diff --git a/projects/web-scraper/client/Dockerfile b/projects/web-scraper/client/Dockerfile new file mode 100644 index 0000000..c59dd28 --- /dev/null +++ b/projects/web-scraper/client/Dockerfile @@ -0,0 +1,5 @@ +FROM nginx:1.14-alpine +COPY . /usr/share/nginx/html +COPY ./default.conf /etc/nginx/conf.d/default.conf +EXPOSE 80 +CMD ["nginx", "-g", "daemon off;"] \ No newline at end of file diff --git a/projects/web-scraper/client/default.conf b/projects/web-scraper/client/default.conf new file mode 100644 index 0000000..d87db83 --- /dev/null +++ b/projects/web-scraper/client/default.conf @@ -0,0 +1,11 @@ +server { + listen 80; + server_name localhost; + location / { + root /usr/share/nginx/html; + index index.html index.htm; + } + location /api { + proxy_pass http://api:3000/api; + } +} \ No newline at end of file diff --git a/projects/web-scraper/client/index.html b/projects/web-scraper/client/index.html new file mode 100644 index 0000000..b1a65a4 --- /dev/null +++ b/projects/web-scraper/client/index.html @@ -0,0 +1,25 @@ + + + + + + Sample Client app + + + + +

Sample Client App

+

+ + + + + \ No newline at end of file diff --git a/projects/web-scraper/docker-compose.yml b/projects/web-scraper/docker-compose.yml new file mode 100644 index 0000000..e9ee914 --- /dev/null +++ b/projects/web-scraper/docker-compose.yml @@ -0,0 +1,48 @@ +version: "3" +services: + mongo: + image: mongo:4.0 + volumes: + - mongo:/data/db + networks: + backend: + aliases: + - mongo + ports: + - "27017" + + api: + build: + context: ./ + dockerfile: Dockerfile + image: "${API_IMAGE}:${TAG}" + networks: + backend: + aliases: + - api + frontend: + aliases: + - api + depends_on: + - mongo + ports: + - "3000:3000" + + client: + build: + context: ./client + dockerfile: Dockerfile + image: "${CLIENT_IMAGE}:${TAG}" + networks: + - frontend + depends_on: + - api + ports: + - "80:80" + +networks: + backend: + frontend: + +volumes: + mongo: \ No newline at end of file diff --git a/projects/web-scraper/docker-kompose.yml b/projects/web-scraper/docker-kompose.yml new file mode 100644 index 0000000..51dc359 --- /dev/null +++ b/projects/web-scraper/docker-kompose.yml @@ -0,0 +1,43 @@ +version: "3" +services: + mongo: + environment: + - GET_HOSTS_FROM=dns + image: mongo:4.0 + ports: + - "27017" + labels: + kompose.volume.size: 1Gi + volumes: + - mongo:/data/db + + api: + environment: + - SECRET=secretkey + - NODE_ENV=prod + - DB=mongodb://mongo:27017/boilerplate_api + - PORT=3000 + - GET_HOSTS_FROM=dns + image: adnanrahic/boilerplate-api + deploy: + replicas: 1 + ports: + - "3000" + labels: + kompose.service.type: LoadBalancer + + client: + environment: + - GET_HOSTS_FROM=dns + image: adnanrahic/smoketest + deploy: + replicas: 1 + ports: + - "80:80" + labels: + kompose.service.type: LoadBalancer + +volumes: + mongo: + + diff --git a/projects/web-scraper/eslintrc.js b/projects/web-scraper/eslintrc.js new file mode 100644 index 0000000..16996cb --- /dev/null +++ b/projects/web-scraper/eslintrc.js @@ -0,0 +1,43 @@ +module.exports = { + "env": { + "es6": true, + "node": true + }, + "globals": { + "__root": true, + "HttpError": true, + "logger": true + }, + "plugins": ["node"], + "extends": ["eslint:recommended", "plugin:node/recommended"], + "rules": { + "node/no-unpublished-require": ["error", { + "allowModules": [ + "mocha", + "chai", + "chai-http", + "chai-as-promised" + ] + }], + "node/exports-style": [ + "error", + "module.exports" + ], + "indent": [ + "error", + 2 + ], + "linebreak-style": [ + "error", + "unix" + ], + "quotes": [ + "error", + "single" + ], + "semi": [ + "error", + "always" + ] + } +}; \ No newline at end of file diff --git a/projects/web-scraper/export-image-tag.sh b/projects/web-scraper/export-image-tag.sh new file mode 100644 index 0000000..10f1f51 --- /dev/null +++ b/projects/web-scraper/export-image-tag.sh @@ -0,0 +1 @@ +export TAG=${SEMAPHORE_GIT_BRANCH}_${SEMAPHORE_WORKFLOW_ID} \ No newline at end of file diff --git a/projects/web-scraper/generate-node-env.sh b/projects/web-scraper/generate-node-env.sh new file mode 100755 index 0000000..be5bd1e --- /dev/null +++ b/projects/web-scraper/generate-node-env.sh @@ -0,0 +1,8 @@ +cat < .env +SECRET=$SECRET +NODE_ENV=$NODE_ENV +DB=$DB +PORT=$PORT +API_IMAGE=$API_IMAGE +CLIENT_IMAGE=$CLIENT_IMAGE +EOF \ No newline at end of file diff --git a/projects/web-scraper/kops.sh b/projects/web-scraper/kops.sh new file mode 100644 index 0000000..67036ec --- /dev/null +++ b/projects/web-scraper/kops.sh @@ -0,0 +1,25 @@ +export ORGANIZATION_NAME= + +# create state store +export BUCKET_NAME=${ORGANIZATION_NAME}-kops-state-store +aws s3api create-bucket --bucket ${BUCKET_NAME} --region eu-west-1 --create-bucket-configuration LocationConstraint=eu-west-1 +aws s3api put-bucket-versioning --bucket ${BUCKET_NAME} --versioning-configuration Status=Enabled + +# create cluster +export KOPS_CLUSTER_NAME=${ORGANIZATION_NAME}.k8s.local +export KOPS_STATE_STORE=s3://${BUCKET_NAME} + +# define cluster config +kops create cluster --master-count=1 --master-size=t2.micro --node-count=1 --node-size=t2.micro --zones=eu-west-1a --name=${KOPS_CLUSTER_NAME} + +# if you want to edit config +kops edit cluster --name ${KOPS_CLUSTER_NAME} + +# apply and create cluster +kops update cluster --name ${KOPS_CLUSTER_NAME} --yes + +# validate cluster is running +kops validate cluster + +# export kubectl config file +KUBECONFIG=${HOME}/${KOPS_CLUSTER_NAME}-kubeconfig.yaml kops export kubecfg --name ${KOPS_CLUSTER_NAME} --state ${KOPS_STATE_STORE} diff --git a/projects/web-scraper/lib/auth/AuthController.js b/projects/web-scraper/lib/auth/AuthController.js new file mode 100644 index 0000000..7e96444 --- /dev/null +++ b/projects/web-scraper/lib/auth/AuthController.js @@ -0,0 +1,21 @@ +module.exports = function (app) { + if (!app) throw new Error('Missing parameter: \'app\' not provided.'); + + const express = require('express'); + const router = express.Router(); + const VerifyToken = require('./VerifyToken')(app); + const AuthProvider = require('./AuthProvider')(app); + + router.post('/login', AuthProvider.login); + + router.post('/register', AuthProvider.register); + + router.get('/me', VerifyToken, AuthProvider.me); + + router.get('/asyncMe', VerifyToken, AuthProvider.asyncMe); + + router.post('/asyncRegister', AuthProvider.asyncRegister); + + return router; + +}; diff --git a/projects/web-scraper/lib/auth/AuthProvider.js b/projects/web-scraper/lib/auth/AuthProvider.js new file mode 100644 index 0000000..987e4d6 --- /dev/null +++ b/projects/web-scraper/lib/auth/AuthProvider.js @@ -0,0 +1,128 @@ +const User = require(__root + 'user/User'); +const jwt = require('jsonwebtoken'); +const bcrypt = require('bcryptjs-then'); + +module.exports = function (app) { + + return { + login, + register, + me, + asyncMe, + asyncRegister + }; + + function signToken(id) { + return jwt.sign({ id: id }, app.config.secret, { + expiresIn: 86400 // expires in 24 hours + }); + } + + function validateEmail(email) { + const re = /^(([^<>()[\]\\.,;:\s@"]+(\.[^<>()[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/; + return re.test(email); + } + + + function login(req, res, next) { + + const _user = {}; + return User.findOne({ email: req.body.email }) + .then(user => { + if (!user) return Promise.reject(new Error('User with that email does not exits.')); + _user._id = user._id; + return user.password; + }) + .then(userPassword => bcrypt.compare(req.body.password, userPassword)) + .then(passwordIsValid => passwordIsValid + ? signToken(_user._id) + : Promise.reject(new Error('The credentials do not match.'))) + .then(token => res.status(200).send({ auth: true, token: token })) + .catch(err => next(err)); + + } + + function register(req, res, next) { + + if ( + !(req.body.password && + req.body.password.length >= 7) + ) { + return next(new HttpError(400, 'Password error. Password needs to be longer than 8 characters.')); + } + + if ( + !(req.body.name && + req.body.name.length > 5 && + typeof req.body.name === 'string') + ) return next(new HttpError(400, 'Username error. Username needs to longer than 5 characters')); + + if ( + !(req.body.email && + validateEmail(req.body.email) && + typeof req.body.name === 'string') + ) return next(new HttpError(400, 'Email error. Email must have valid characters.')); + + return User.findOne({ email: req.body.email }) + .then(user => user ? + Promise.reject(new HttpError(400, 'User with that email exists.')) : + Promise.resolve()) + .then(bcrypt.hash.bind(this, req.body.password, 8)) + .then(hash => User.create({ name: req.body.name, email: req.body.email, password: hash })) + .then(user => res.status(200).send({ auth: true, token: signToken(user._id) })) + .catch(next); + } + + function me(req, res, next) { + return User.findById(req.userId, { password: 0 }) + .then(user => !user ? + next(new HttpError(404, 'No user found.')) : + res.status(200).send(user)) + .catch(err => next(new Error(err))); + } + + async function asyncMe(req, res, next) { + try { + let user = await User.findById(req.userId, { password: 0 }); + if (!user) + return res.status(404).send('No user found.'); + + res.status(200).send(user); + } catch (e) { + next(e); + } + } + + async function asyncRegister(req, res, next) { + + if ( + !(req.body.password && + req.body.password.length >= 7) + ) return next(new Error('Password error. Password needs to be longer than 8 characters.')); + + if ( + !(req.body.name && + req.body.name.length > 5 && + typeof req.body.name === 'string') + ) return next(new Error('Username error. Username needs to be longer than 5 characters')); + + if ( + !(req.body.email && + validateEmail(req.body.email) && + typeof req.body.name === 'string') + ) return next(new Error('Email error. Email must have valid characters.')); + + try { + const userExists = await User.findOne({ email: req.body.email }); + if (userExists) + throw new Error('User with that email exists.'); + + const hash = await bcrypt.hash(req.body.password, 8); + const user = await User.create({ name: req.body.name, email: req.body.email, password: hash }); + res.status(200).send({ auth: true, token: signToken(user._id) }); + } catch (e) { + next(e); + } + } + +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/auth/VerifyToken.js b/projects/web-scraper/lib/auth/VerifyToken.js new file mode 100644 index 0000000..a6b70b7 --- /dev/null +++ b/projects/web-scraper/lib/auth/VerifyToken.js @@ -0,0 +1,26 @@ +module.exports = function (app) { + + var jwt = require('jsonwebtoken'); // used to create, sign, and verify tokens + + function verifyToken(req, res, next) { + + // check header or url parameters or post parameters for token + var token = req.headers['x-access-token']; + if (!token) + return res.status(403).send({ auth: false, message: 'No token provided.' }); + + // verifies secret and checks exp + jwt.verify(token, app.config.secret, function(err, decoded) { + if (err) + return res.status(500).send({ auth: false, message: 'Failed to authenticate token.' }); + + // if everything is good, save to request for use in other routes + req.userId = decoded.id; + next(); + }); + + } + + return verifyToken; + +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/config.js b/projects/web-scraper/lib/config.js new file mode 100644 index 0000000..5b74954 --- /dev/null +++ b/projects/web-scraper/lib/config.js @@ -0,0 +1,17 @@ +module.exports = function (env) { + const test = require('./config/test'); + const dev = require('./config/dev'); + const prod = require('./config/prod'); + + switch (env) { + case 'test': + return test; + case 'dev': + return dev; + case 'prod': + return prod; + + default: + return dev; + } +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/config/dev.js b/projects/web-scraper/lib/config/dev.js new file mode 100644 index 0000000..01f0bed --- /dev/null +++ b/projects/web-scraper/lib/config/dev.js @@ -0,0 +1,9 @@ +const secret = process.env.SECRET || 'supersecret'; +const env = process.env.NODE_ENV || 'dev'; +const db = process.env.DB || 'mongodb://localhost:27017/boilerplate-api'; + +module.exports = { + 'secret': secret, + 'env': env, + 'db': db +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/config/prod.js b/projects/web-scraper/lib/config/prod.js new file mode 100644 index 0000000..d4879eb --- /dev/null +++ b/projects/web-scraper/lib/config/prod.js @@ -0,0 +1,9 @@ +const secret = process.env.SECRET || 'supersecret'; +const env = process.env.NODE_ENV || 'production'; +const db = process.env.DB || 'mongodb://localhost:27017/boilerplate-api'; + +module.exports = { + 'secret': secret, + 'env': env, + 'db': db +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/config/test.js b/projects/web-scraper/lib/config/test.js new file mode 100644 index 0000000..964a625 --- /dev/null +++ b/projects/web-scraper/lib/config/test.js @@ -0,0 +1,9 @@ +const secret = process.env.SECRET || 'supersecret'; +const env = process.env.NODE_ENV || 'test'; +const db = process.env.DB || 'mongodb://localhost:27017/boilerplate-api-test'; + +module.exports = { + 'secret': secret, + 'env': env, + 'db': db +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/db.js b/projects/web-scraper/lib/db.js new file mode 100644 index 0000000..61a91ff --- /dev/null +++ b/projects/web-scraper/lib/db.js @@ -0,0 +1,12 @@ +module.exports = function (app) { + + var mongoose = require('mongoose'); + /** + * need to plug in native promises to work with promises instead of callbacks + */ + mongoose.Promise = global.Promise; + mongoose.connect(app.config.db, { useNewUrlParser: true }); + + logger.log('info', app.config.db); + +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/helpers/HttpError.js b/projects/web-scraper/lib/helpers/HttpError.js new file mode 100644 index 0000000..e69e27c --- /dev/null +++ b/projects/web-scraper/lib/helpers/HttpError.js @@ -0,0 +1,11 @@ +'use strict'; + +module.exports = HttpError; +function HttpError(status, message) { + if (arguments.length !== 2) throw new Error('HttpError must take two parameters.'); + + Error.captureStackTrace(this); + this.status = status; + this.message = message; +} +HttpError.prototype = new Error; \ No newline at end of file diff --git a/projects/web-scraper/lib/note/Note.js b/projects/web-scraper/lib/note/Note.js new file mode 100644 index 0000000..7d79b7d --- /dev/null +++ b/projects/web-scraper/lib/note/Note.js @@ -0,0 +1,9 @@ +var mongoose = require('mongoose'); +var NoteSchema = new mongoose.Schema({ + title: String, + description: String, + pinned: { type: Boolean, default: false } +}); +mongoose.model('Note', NoteSchema); + +module.exports = mongoose.model('Note'); \ No newline at end of file diff --git a/projects/web-scraper/lib/note/NoteController.js b/projects/web-scraper/lib/note/NoteController.js new file mode 100644 index 0000000..546b828 --- /dev/null +++ b/projects/web-scraper/lib/note/NoteController.js @@ -0,0 +1,27 @@ +module.exports = function (app) { + if (!app) throw new Error('Missing parameter: \'app\' not provided.'); + + var express = require('express'); + var NoteController = express.Router(); + var NoteProvider = require('./NoteProvider'); + var validateNote = require('./validateNote'); + var VerifyToken = require(__root + 'auth/VerifyToken')(app); + + // CREATES A NEW NOTE + NoteController.post('/', VerifyToken, validateNote, NoteProvider.createNote); + + // RETURNS ALL THE NOTES IN THE DATABASE + NoteController.get('/', NoteProvider.getNotes); + + // GETS A SINGLE NOTE FROM THE DATABASE + NoteController.get('/:id', NoteProvider.getNote); + + // DELETES A NOTE FROM THE DATABASE + NoteController.delete('/:id', VerifyToken, NoteProvider.deleteNote); + + // UPDATES A SINGLE NOTE IN THE DATABASE + NoteController.put('/:id', VerifyToken, NoteProvider.putNote); + + return NoteController; + +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/note/NoteProvider.js b/projects/web-scraper/lib/note/NoteProvider.js new file mode 100644 index 0000000..d90e032 --- /dev/null +++ b/projects/web-scraper/lib/note/NoteProvider.js @@ -0,0 +1,45 @@ +var Note = require('./Note'); +module.exports = { + createNote: createNote, + getNotes: getNotes, + getNote: getNote, + deleteNote: deleteNote, + putNote: putNote +}; + +function createNote(req, res, next) { + return Note.create({ + title : req.body.title, + description : req.body.description, + pinned : req.body.pinned + }) + .then(note => res.status(200).send(note)) + .catch(err => next(new Error(err))); +} + +function getNotes(req, res, next) { + return Note.find({}) + .then(notes => res.status(200).send(notes)) + .catch(err => next(new Error(err))); +} + +function getNote(req, res, next) { + return Note.findById(req.params.id) + .then(note => { + if (!note) return res.status(404).send('No note found.'); + res.status(200).send(note); + }) + .catch(err => next(new Error(err))); +} + +function deleteNote(req, res, next) { + return Note.findByIdAndRemove(req.params.id) + .then(note => res.status(200).send(note)) + .catch(err => next(new Error(err))); +} + +function putNote(req, res, next) { + return Note.findByIdAndUpdate(req.params.id, req.body, {new: true}) + .then(note => res.status(200).send(note)) + .catch(err => next(new Error(err))); +} \ No newline at end of file diff --git a/projects/web-scraper/lib/note/validateNote.js b/projects/web-scraper/lib/note/validateNote.js new file mode 100644 index 0000000..45996a2 --- /dev/null +++ b/projects/web-scraper/lib/note/validateNote.js @@ -0,0 +1,9 @@ +function validateNote(req, res, next) { + if (typeof req.body.title === 'string' && + typeof req.body.description === 'string') + return next(); + + return res.status(500).send('Please enter a valid Note.'); +} + +module.exports = validateNote; \ No newline at end of file diff --git a/projects/web-scraper/lib/routes.js b/projects/web-scraper/lib/routes.js new file mode 100644 index 0000000..e717311 --- /dev/null +++ b/projects/web-scraper/lib/routes.js @@ -0,0 +1,34 @@ +module.exports = function (app) { + + const express = require('express'); + const router = express.Router(); + + /** + * routes + */ + const NoteController = require('./note/NoteController')(app); + router.use('/notes', NoteController); + + const AuthController = require('./auth/AuthController')(app); + router.use('/auth', AuthController); + + const UserController = require('./user/UserController')(app); + router.use('/users', UserController); + + /** + * default error handling + */ + function errorHandler(err, req, res, next) { + logger.error(err.stack); + res.status(err.status || 500).send(err.message); + next(); + } + router.use(errorHandler); + + /** + * default root route for api + */ + app.use('/api', router); + app.get('/api', (req, res) => res.status(200).send({ message: 'Api Works.', db: app.config.db })); + +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/server.js b/projects/web-scraper/lib/server.js new file mode 100644 index 0000000..8272507 --- /dev/null +++ b/projects/web-scraper/lib/server.js @@ -0,0 +1,44 @@ +const fs = require('fs'); +const express = require('express'); +const app = express(); +const bodyParser = require('body-parser'); +const winston = require('winston'); +const { combine, timestamp, printf, splat } = winston.format; +global.__root = __dirname + '/'; +global.HttpError = require('./helpers/HttpError'); +app.use(bodyParser.json()); +app.use(bodyParser.urlencoded({ extended: true })); + +if (!fs.existsSync('logs')) fs.mkdirSync('logs'); + +global.logger = winston.createLogger({ + format: combine( + timestamp(), + printf(info => `${info.timestamp} ${info.level}: ${info.message}`), + splat() + ), + transports: [ + new winston.transports.File({ filename: 'logs/error.log', level: 'error' }), + new winston.transports.File({ filename: 'logs/combined.log' })] +}); + +module.exports = { + serve(env) { + + if (env !== 'test') + logger.add(new winston.transports.Console, { prettyPrint: true }); + + app.config = require('./config')(env); + require('./db')(app); + require('./routes')(app); + app.port = process.env.PORT || 3000; + return new Promise((resolve, reject) => { + app.listen(app.port, function (err) { + if (err) + return reject(err); + + resolve(app); + }); + }); + } +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/spider/siteMap.xml b/projects/web-scraper/lib/spider/siteMap.xml new file mode 100644 index 0000000..677ac3e --- /dev/null +++ b/projects/web-scraper/lib/spider/siteMap.xml @@ -0,0 +1,2512 @@ + + + + + + + https://www.lamayor.org/ + 2020-02-13T03:28:54+00:00 + 1.00 + + + https://www.lamayor.org/how-can-we-help + 2020-02-13T03:28:55+00:00 + 0.80 + + + https://www.lamayor.org/about + 2020-02-13T03:28:56+00:00 + 0.80 + + + https://www.lamayor.org/our-team + 2020-02-13T03:28:57+00:00 + 0.80 + + + https://www.lamayor.org/OurWork + 2020-02-13T03:28:58+00:00 + 0.80 + + + https://www.lamayor.org/PressRoom + 2020-02-13T03:29:00+00:00 + 0.80 + + + https://www.lamayor.org/mayor-garcetti-launches-la%E2%80%99s-%E2%80%98decade-action%E2%80%99-fight-climate-crisis + 2020-02-13T03:29:00+00:00 + 0.80 + + + https://www.lamayor.org/Homelessness + 2020-02-13T03:29:02+00:00 + 0.80 + + + https://www.lamayor.org/mayor_garcetti_s_executive_directives + 2020-02-13T03:29:03+00:00 + 0.80 + + + https://www.lamayor.org/achievements + 2020-02-13T03:29:04+00:00 + 0.80 + + + https://www.lamayor.org/sustainability + 2020-02-13T03:29:06+00:00 + 0.80 + + + https://www.lamayor.org/Immigration + 2020-02-13T03:29:07+00:00 + 0.80 + + + https://www.lamayor.org/infrastructure + 2020-02-13T03:29:08+00:00 + 0.80 + + + https://www.lamayor.org/economic-growth + 2020-02-13T03:29:09+00:00 + 0.80 + + + https://www.lamayor.org/safe-neighborhoods + 2020-02-13T03:29:10+00:00 + 0.80 + + + https://www.lamayor.org/mayor-garcetti-convenes-faith-leaders-support-la%E2%80%99s-green-new-deal + 2020-02-13T03:29:11+00:00 + 0.80 + + + https://www.lamayor.org/mayor-garcetti-celebrates-two-years-evolve-entertainment-fund + 2020-02-13T03:29:12+00:00 + 0.80 + + + https://www.lamayor.org/mayor-garcetti-celebrates-launch-new-lafd-fast-response-vehicle-unit + 2020-02-13T03:29:13+00:00 + 0.80 + + + https://www.lamayor.org/terms + 2020-02-13T03:29:16+00:00 + 0.80 + + + https://www.lamayor.org/privacy + 2020-02-13T03:29:17+00:00 + 0.80 + + + https://www.lamayor.org/meet-your-area-representative + 2020-02-13T03:29:19+00:00 + 0.64 + + + https://www.lamayor.org/files/33081613638899593114807961192485181194240ojpg-0 + 2020-02-13T03:29:19+00:00 + 0.64 + + + https://www.lamayor.org/mayors-help-desk + 2020-02-13T03:29:20+00:00 + 0.64 + + + https://www.lamayor.org/files/website-card-full13png-0 + 2020-02-13T03:29:21+00:00 + 0.64 + + + https://www.lamayor.org/files/311jpg + 2020-02-13T03:29:22+00:00 + 0.64 + + + https://www.lamayor.org/files/6g1a0072a-webpng-0 + 2020-02-13T03:29:23+00:00 + 0.64 + + + https://www.lamayor.org/files/bphomejpg + 2020-02-13T03:29:24+00:00 + 0.64 + + + https://www.lamayor.org/files/city-govjpg + 2020-02-13T03:29:25+00:00 + 0.64 + + + https://www.lamayor.org/files/newsletterjpg-0 + 2020-02-13T03:29:26+00:00 + 0.64 + + + https://www.lamayor.org/general + 2020-02-13T03:29:27+00:00 + 0.64 + + + https://www.lamayor.org/files/bgherofpojpg-0 + 2020-02-13T03:29:28+00:00 + 0.64 + + + https://www.lamayor.org/about-spanish + 2020-02-13T03:29:29+00:00 + 0.64 + + + https://www.lamayor.org/about-korean + 2020-02-13T03:29:30+00:00 + 0.64 + + + https://www.lamayor.org/about-mandarin-chinese + 2020-02-13T03:29:31+00:00 + 0.64 + + + https://www.lamayor.org/about-armenian + 2020-02-13T03:29:33+00:00 + 0.64 + + + https://www.lamayor.org/about-tagalog + 2020-02-13T03:29:34+00:00 + 0.64 + + + https://www.lamayor.org/amy-elaine-wakeland + 2020-02-13T03:29:35+00:00 + 0.64 + + + https://www.lamayor.org/files/35721154406baa8a0031bk-webjpg-0 + 2020-02-13T03:29:36+00:00 + 0.64 + + + https://www.lamayor.org/files/tile-14x7jpg + 2020-02-13T03:29:39+00:00 + 0.64 + + + https://www.lamayor.org/files/edpng-1 + 2020-02-13T03:29:40+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcettis-la + 2020-02-13T03:29:41+00:00 + 0.64 + + + https://www.lamayor.org/files/funpng-0 + 2020-02-13T03:29:42+00:00 + 0.64 + + + https://www.lamayor.org/files/nk3a0492-card1jpg-0 + 2020-02-13T03:29:43+00:00 + 0.64 + + + https://www.lamayor.org/files/photos-2png-2 + 2020-02-13T03:29:44+00:00 + 0.64 + + + https://www.lamayor.org/files/26634819977fc27d5f65cojpg + 2020-02-13T03:29:45+00:00 + 0.64 + + + https://www.lamayor.org/mayors-offices + 2020-02-13T03:29:47+00:00 + 0.64 + + + https://www.lamayor.org/files/6g1a6187a-croppng-1 + 2020-02-13T03:29:48+00:00 + 0.64 + + + https://www.lamayor.org/senior-leadership + 2020-02-13T03:29:49+00:00 + 0.64 + + + https://www.lamayor.org/files/ckn00003a-14x7jpg + 2020-02-13T03:29:49+00:00 + 0.64 + + + https://www.lamayor.org/files/area-repjpg-0 + 2020-02-13T03:29:50+00:00 + 0.64 + + + https://www.lamayor.org/intern + 2020-02-13T03:29:51+00:00 + 0.64 + + + https://www.lamayor.org/files/6g1a5920a-webjpg + 2020-02-13T03:29:52+00:00 + 0.64 + + + https://www.lamayor.org/files/tile-14x7jpg-0 + 2020-02-13T03:29:53+00:00 + 0.64 + + + https://www.lamayor.org/education + 2020-02-13T03:29:54+00:00 + 0.64 + + + https://www.lamayor.org/files/edu-14x7png + 2020-02-13T03:29:55+00:00 + 0.64 + + + https://www.lamayor.org/files/solarpng-0 + 2020-02-13T03:29:56+00:00 + 0.64 + + + https://www.lamayor.org/files/38040116492fb57f29d77opng-1 + 2020-02-13T03:29:57+00:00 + 0.64 + + + https://www.lamayor.org/Housing + 2020-02-13T03:29:58+00:00 + 0.64 + + + https://www.lamayor.org/files/497a0599a-14x7jpg + 2020-02-13T03:29:59+00:00 + 0.64 + + + https://www.lamayor.org/files/econpng-0 + 2020-02-13T03:30:02+00:00 + 0.64 + + + https://www.lamayor.org/files/snl0png + 2020-02-13T03:30:03+00:00 + 0.64 + + + https://www.lamayor.org/files/laxpng-0 + 2020-02-13T03:30:04+00:00 + 0.64 + + + https://www.lamayor.org/files/moiapng-0 + 2020-02-13T03:30:05+00:00 + 0.64 + + + https://www.lamayor.org/transportation + 2020-02-13T03:30:06+00:00 + 0.64 + + + https://www.lamayor.org/files/infrastructure0png + 2020-02-13T03:30:07+00:00 + 0.64 + + + https://www.lamayor.org/GenderEquity + 2020-02-13T03:30:08+00:00 + 0.64 + + + https://www.lamayor.org/files/33095081610dcc3db7165ojpg-0 + 2020-02-13T03:30:09+00:00 + 0.64 + + + https://www.lamayor.org/improving-government + 2020-02-13T03:30:10+00:00 + 0.64 + + + https://www.lamayor.org/files/well-runpng-0 + 2020-02-13T03:30:11+00:00 + 0.64 + + + https://www.lamayor.org/international-affairs + 2020-02-13T03:30:12+00:00 + 0.64 + + + https://www.lamayor.org/files/11470-10jpg + 2020-02-13T03:30:13+00:00 + 0.64 + + + https://www.lamayor.org/veterans + 2020-02-13T03:30:14+00:00 + 0.64 + + + https://www.lamayor.org/files/veterans-button + 2020-02-13T03:30:16+00:00 + 0.64 + + + https://www.lamayor.org/files/dashboard-jpg + 2020-02-13T03:30:17+00:00 + 0.64 + + + https://www.lamayor.org/media/press_releases + 2020-02-13T03:30:17+00:00 + 0.64 + + + https://www.lamayor.org/files/website-card-flat-secondary-bluepng-2 + 2020-02-13T03:30:19+00:00 + 0.64 + + + https://www.lamayor.org/Speeches + 2020-02-13T03:30:19+00:00 + 0.64 + + + https://www.lamayor.org/files/inaug-2017-20png + 2020-02-13T03:30:21+00:00 + 0.64 + + + https://www.lamayor.org/files/photos-2png-0 + 2020-02-13T03:30:22+00:00 + 0.64 + + + https://www.lamayor.org/ask-my-anything-reddit + 2020-02-13T03:30:23+00:00 + 0.64 + + + https://www.lamayor.org/files/redditpng + 2020-02-13T03:30:24+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/20200210ExecutiveDirective25.pdf + 2020-02-11T01:25:03+00:00 + 0.64 + + + https://www.lamayor.org/HEAP + 2020-02-13T03:30:25+00:00 + 0.64 + + + https://www.lamayor.org/rising-challenge-helping-homeless-angelenos + 2020-02-13T03:30:26+00:00 + 0.64 + + + https://www.lamayor.org/momentum-solutions-homelessness + 2020-02-13T03:30:27+00:00 + 0.64 + + + https://www.lamayor.org/meet-blanca + 2020-02-13T03:30:28+00:00 + 0.64 + + + https://www.lamayor.org/CTC-Helping_our_Homeless_Neighbors + 2020-02-13T03:30:29+00:00 + 0.64 + + + https://www.lamayor.org/confronting-crisis-helping-our-homeless-neighbors + 2020-02-13T03:30:31+00:00 + 0.64 + + + https://www.lamayor.org/HomelessnessCausesAndResponses + 2020-02-13T03:30:32+00:00 + 0.64 + + + https://www.lamayor.org/files/277541644998ecc7993d5opng-1 + 2020-02-13T03:30:33+00:00 + 0.64 + + + https://www.lamayor.org/ABridgeHome + 2020-02-13T03:30:34+00:00 + 0.64 + + + https://www.lamayor.org/files/abhjpg + 2020-02-13T03:30:35+00:00 + 0.64 + + + https://www.lamayor.org/ConfrontingTheHomelessnessCrisis + 2020-02-13T03:30:36+00:00 + 0.64 + + + https://www.lamayor.org/files/32142783900fa82a1a375opng + 2020-02-13T03:30:37+00:00 + 0.64 + + + https://www.lamayor.org/street-strategies + 2020-02-13T03:30:38+00:00 + 0.64 + + + https://www.lamayor.org/files/ssjpg + 2020-02-13T03:30:40+00:00 + 0.64 + + + https://www.lamayor.org/files/14x7jpg-0 + 2020-02-13T03:30:40+00:00 + 0.64 + + + https://www.lamayor.org/HomelessnessHousing + 2020-02-13T03:30:41+00:00 + 0.64 + + + https://www.lamayor.org/files/384590444729fc6c29be2opng + 2020-02-13T03:30:42+00:00 + 0.64 + + + https://www.lamayor.org/HomelessnessTrackingHHH + 2020-02-13T03:30:43+00:00 + 0.64 + + + https://www.lamayor.org/files/39154395042c93ec8f9b1opng + 2020-02-13T03:30:44+00:00 + 0.64 + + + https://www.lamayor.org/how-help-homeless-angelenos + 2020-02-13T03:30:46+00:00 + 0.64 + + + https://www.lamayor.org/files/helpjpg-0 + 2020-02-13T03:30:47+00:00 + 0.64 + + + https://www.lamayor.org/homeless-help-desk + 2020-02-13T03:30:48+00:00 + 0.64 + + + https://www.lamayor.org/files/homeless-help-desk-2jpg + 2020-02-13T03:30:49+00:00 + 0.64 + + + https://www.lamayor.org/Homelessness-Data-and-Dashboards + 2020-02-13T03:30:50+00:00 + 0.64 + + + https://www.lamayor.org/files/14x7jpg + 2020-02-13T03:30:51+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Executive-Directive-1-Great-Streets-Initiative-1.pdf?1426619965 + 2016-02-06T22:04:27+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED2_with_signature_and_letterhead.pdf?1426620047 + 2016-02-06T21:09:47+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Executive-Directive-3-Open-Data.pdf?1426620075 + 2016-02-06T21:10:17+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED_4_-_Intergovernmental_Relations.pdf?1426619988 + 2016-02-06T21:09:11+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED_5_-_Emergency_Drought__Response_-_Creating_a_Water_Wise_City.pdf?1426620015 + 2016-02-06T21:09:17+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED6Film.pdf?1426620059 + 2016-02-06T21:09:55+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED7-SustainableCitypLAn.pdf + 2019-01-03T01:01:21+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED8-CleanStreets.pdf + 2019-01-03T01:01:12+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED9.pdf + 2019-01-03T00:59:51+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Mayor_Garcetti_File_Executive_Directive_10_Vision_Zero.pdf?1440454405 + 2016-02-06T21:10:25+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED_11.pdf?1440645063 + 2016-02-06T21:09:26+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/SCHM_305H_K15100111110.pdf?1443724774 + 2016-02-06T22:07:03+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED_13_-_Support_for_Affordable_Housing_Development.pdf?1445984955 + 2016-02-06T21:09:39+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Executive_Directive_14.pdf?1446858272 + 2016-02-06T21:10:06+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2018%20-%20Safe%20and%20Healthy%20Workforce.pdf + 2017-02-06T23:08:09+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ed19%20planning.pdf + 2017-03-09T21:13:37+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Exec.%20Dir.%20No.%2020--Standing%20with%20Immigrants.pdf + 2017-03-21T22:04:02+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/20180206%20ED%2021.pdf + 2018-02-07T00:10:17+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2022%20-%20Resilient%20Los%20Angeles.pdf + 2018-03-06T00:48:31+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2023%20-%20Harassment%20and%20Discrimination.pdf + 2018-05-01T19:20:07+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2024%20-%20Building%20%27A%20Bridge%20Home%27.pdf + 2018-06-02T01:20:31+00:00 + 0.64 + + + https://www.lamayor.org/statement-mayor-garcetti-minimum-wage-increase + 2020-02-13T03:31:05+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/2018_SOTC_AS_PREPARED_FOR_DELIVERY.pdf + 2018-04-16T16:01:50+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-cuts-costs-digital-filmmakers-new-pilot-program + 2020-02-13T03:31:07+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-record-breaking-year-la-economy-0 + 2020-02-13T03:31:08+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-unveils-plans-portsocall-revitalization + 2020-02-13T03:31:09+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-surpasses-10000-veterans-hiring-goal-ahead-schedule + 2020-02-13T03:31:10+00:00 + 0.64 + + + https://www.lamayor.org/statement-mayor-garcetti-signs-affordable-housing-linkage-fee-ordinance + 2020-02-13T03:31:11+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-6-million-state-grant-support-city-reentry-programs + 2020-02-13T03:31:12+00:00 + 0.64 + + + https://www.lamayor.org/la-secures-35-million-community-revitalization-watts + 2020-02-13T03:31:13+00:00 + 0.64 + + + https://www.lamayor.org/mayor_garcetti_marks_milestone_in_the_lowering_of_tunnel_boring_machine_for_crenshaw_lax_line + 2020-02-13T03:31:14+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-launches-complete-streets-program + 2020-02-13T03:31:16+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-releases-progress-report-sustainable-city-plan-two-thirds-all-2017-goals-accomplished + 2020-02-13T03:31:17+00:00 + 0.64 + + + https://www.lamayor.org/los-angeles-named-1-solar-city-america + 2020-02-13T03:31:18+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-breaks-ground-major-water-recycling-project + 2020-02-13T03:31:19+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-and-lapd-launch-recruitment-campaign-focused-diversity + 2020-02-13T03:31:21+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-new-expansion-community-safety-partnership + 2020-02-13T03:31:22+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-gun-buybacks-took-close-500-weapons-street + 2020-02-13T03:31:23+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-kicks-7th-year-summer-night-lights + 2020-02-13T03:31:24+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-expanded-domestic-violence-programs + 2020-02-13T03:31:25+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-grant-expand-domestic-violence-programs + 2020-02-13T03:31:26+00:00 + 0.64 + + + https://www.lamayor.org/Resilience + 2020-02-13T03:31:28+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-launches-nation%E2%80%99s-first-publicly-available-earthquake-early-warning-mobile-app + 2020-02-13T03:31:29+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-cleanstat-0 + 2020-02-13T03:31:31+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-signs-2018-19-budget-record-investments-ending-homelessness-improving-infrastructure + 2020-02-13T03:31:32+00:00 + 0.64 + + + https://www.lamayor.org/release-mayor-garcetti-unveils-new-ladwp-customer-bill-rights + 2020-02-13T03:31:34+00:00 + 0.64 + + + https://www.lamayor.org/larams + 2020-02-13T03:31:36+00:00 + 0.64 + + + https://www.lamayor.org/statement-mayor-garcetti-chargers-moving-los-angeles + 2020-02-13T03:31:37+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-joins-mayors-around-world-set-bold-new-sustainability-targets + 2020-02-13T03:31:38+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-leads-%E2%80%98climate-mayors%E2%80%99-oppose-us-withdrawal-paris-agreement + 2020-02-13T03:31:39+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-los-angeles-will-recycle-100-city%E2%80%99s-wastewater-2035 + 2020-02-13T03:31:40+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-ladwp-will-phase-out-natural-gas-operations-three-power-plants + 2020-02-13T03:31:41+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-completion-world%E2%80%99s-most-powerful-rooftop-solar-project + 2020-02-13T03:31:42+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-and-long-beach-mayor-robert-garcia-announce-zero-emissions-goals-san-pedro-bay-ports + 2020-02-13T03:31:43+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-and-state-leaders-launch-electric-vehicle-car-share-program-disadvantaged-communities + 2020-02-13T03:31:44+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-hosts-clean-transportation-forum-paris-mayor-hidalgo-announces-new-online-platform + 2020-02-13T03:31:46+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-leads-us-mayors%E2%80%99-call-accurate-2020-census + 2020-02-13T03:31:47+00:00 + 0.64 + + + https://www.lamayor.org/strong-families-la + 2020-02-13T03:31:48+00:00 + 0.64 + + + https://www.lamayor.org/immigration-news-and-events + 2020-02-13T03:31:49+00:00 + 0.64 + + + https://www.lamayor.org/files/37614181706b11febddd0opng + 2020-02-13T03:31:50+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Community-Resource-Guide-%28English3%29.pdf + 2019-08-28T16:41:09+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Community-Resource-Guide-%28Spanish3%29.pdf + 2019-08-28T16:41:23+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Community-Resource-Guide-%28Korean2%29.pdf + 2019-08-28T16:41:17+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Community-Resource-Guide-%28Chinese2%29.pdf + 2019-08-28T16:41:29+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/LAPDandImmigrationFAQ.pdf + 2018-05-30T22:33:12+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Immigration%20consumer-alert%20English.pdf + 2018-08-29T18:02:36+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Immigration%20consumer-alert-Spanish.pdf + 2018-08-29T18:02:45+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Immigration%20consumer-alert%20Chinese.PDF + 2018-08-29T18:03:13+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Immigration%20consumer-alert%20Vietnamese.PDF + 2018-08-29T18:03:20+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-2400-lane-miles-paved-over-last-year + 2020-02-13T03:31:57+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-and-los-angeles-world-airports-break-ground-historic-lax-airport-train + 2020-02-13T03:31:58+00:00 + 0.64 + + + https://www.lamayor.org/new-groundwater-treatment-project-will-help-la-reduce-dependence-imported-water + 2020-02-13T03:31:59+00:00 + 0.64 + + + https://www.lamayor.org/two-big-steps-forward-garcettis-economic-development-agenda + 2020-02-13T03:32:01+00:00 + 0.64 + + + https://www.lamayor.org/statement-mayor-garcetti-latest-round-film-tax-credit-allocations + 2020-02-13T03:32:02+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-opens-tech-job-opportunities-young-people + 2020-02-13T03:32:03+00:00 + 0.64 + + + https://www.lamayor.org/mayors-office-economic-development + 2020-02-13T03:32:04+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-launches-la-original + 2020-02-13T03:32:05+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-launches-evolve-entertainment-fund-boost-diversity-entertainment-industry + 2020-02-13T03:32:06+00:00 + 0.64 + + + https://www.lamayor.org/%E2%80%98new-roads-second-chances%E2%80%99-connects-hundreds-formerly-incarcerated-angelenos-employment + 2020-02-13T03:32:07+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-first-steps-comprehensive-strategy-combat-homelessness-crisis + 2020-02-13T03:32:08+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/AcceleratingSmallBusinessGrowth.pdf + 2018-12-05T19:14:23+00:00 + 0.64 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/LAWomenMeanBizBrochure.pdf + 2018-12-05T19:14:31+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-body-cameras-lapd + 2020-02-13T03:32:11+00:00 + 0.64 + + + https://www.lamayor.org/%E2%80%98summer-night-lights%E2%80%99-kicks-10th-year-providing-fun-safe-spaces-young-people-and-families + 2020-02-13T03:32:13+00:00 + 0.64 + + + https://www.lamayor.org/mayor_garcetti_unveils_los_angeles_fire_department_innovations_to_improve_emergency_response_patient_care + 2020-02-13T03:32:14+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-urges-earthquake-preparedness-day-great-shakeout + 2020-02-13T03:32:15+00:00 + 0.64 + + + https://www.lamayor.org/BuildingForwardLA + 2020-02-13T03:32:17+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-launches-nation%E2%80%99s-first-city-based-cyber-lab + 2020-02-13T03:32:18+00:00 + 0.64 + + + https://www.lamayor.org/mayor-garcetti-announces-opportunities-mayors-crisis-response-team + 2020-02-13T03:32:20+00:00 + 0.64 + + + https://www.lamayor.org/WestValleyAreaRep + 2020-02-13T03:32:20+00:00 + 0.51 + + + https://www.lamayor.org/CentralAreaRep + 2020-02-13T03:32:21+00:00 + 0.51 + + + https://www.lamayor.org/EastValleyAreaRep + 2020-02-13T03:32:22+00:00 + 0.51 + + + https://www.lamayor.org/SouthwestLAAreaRep + 2020-02-13T03:32:23+00:00 + 0.51 + + + https://www.lamayor.org/WestAreaRep + 2020-02-13T03:32:24+00:00 + 0.51 + + + https://www.lamayor.org/SoutheastLAAreaRep + 2020-02-13T03:32:25+00:00 + 0.51 + + + https://www.lamayor.org/EastAreaRep + 2020-02-13T03:32:26+00:00 + 0.51 + + + https://www.lamayor.org/HarborAreaRep + 2020-02-13T03:32:27+00:00 + 0.51 + + + https://www.lamayor.org/FreeTaxPrep + 2020-02-13T03:32:28+00:00 + 0.51 + + + https://www.lamayor.org/EstudiantesYFamilias + 2020-02-13T03:32:29+00:00 + 0.51 + + + https://www.lamayor.org/StudentsAndFamilies + 2020-02-13T03:32:30+00:00 + 0.51 + + + https://www.lamayor.org/free-rides-metro-and-other-election-day-essentials + 2020-02-13T03:32:32+00:00 + 0.51 + + + https://www.lamayor.org/CityHallTour + 2020-02-13T03:32:32+00:00 + 0.51 + + + https://www.lamayor.org/what-you-should-know-about-typhus + 2020-02-13T03:32:33+00:00 + 0.51 + + + https://www.lamayor.org/LAVotes + 2020-02-13T03:32:34+00:00 + 0.51 + + + https://www.lamayor.org/MYC + 2020-02-13T03:32:35+00:00 + 0.51 + + + https://www.lamayor.org/WiSTEM + 2020-02-13T03:32:36+00:00 + 0.51 + + + https://www.lamayor.org/mayors-office-public-safety + 2020-02-13T03:32:37+00:00 + 0.51 + + + https://www.lamayor.org/files/2406380406764fbf0e30copng + 2020-02-13T03:32:38+00:00 + 0.51 + + + https://www.lamayor.org/mayors-office-city-homelessness-initiatives + 2020-02-13T03:32:39+00:00 + 0.51 + + + https://www.lamayor.org/files/mochijpg + 2020-02-13T03:32:40+00:00 + 0.51 + + + https://www.lamayor.org/mayors-office-economic-opportunity + 2020-02-13T03:32:41+00:00 + 0.51 + + + https://www.lamayor.org/files/4113988404454a0779444opng + 2020-02-13T03:32:42+00:00 + 0.51 + + + https://www.lamayor.org/mayors-office-budget-and-innovation + 2020-02-13T03:32:43+00:00 + 0.51 + + + https://www.lamayor.org/files/41567263871ab63f4bd6eopng + 2020-02-13T03:32:44+00:00 + 0.51 + + + https://www.lamayor.org/mayors-office-city-services + 2020-02-13T03:32:45+00:00 + 0.51 + + + https://www.lamayor.org/files/32159890491717519b700ojpg + 2020-02-13T03:32:46+00:00 + 0.51 + + + https://www.lamayor.org/mayors-office-international-affairs + 2020-02-13T03:32:47+00:00 + 0.51 + + + https://www.lamayor.org/files/253401907572eb44d57a9opng + 2020-02-13T03:32:48+00:00 + 0.51 + + + https://www.lamayor.org/files/megmoedwebpng-0 + 2020-02-13T03:32:49+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-announces-program-provide-free-dash-bus-passes-students + 2020-02-13T03:32:50+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-launches-%E2%80%98connect-success%E2%80%99-provide-free-laptops-la-college-promise-students + 2020-02-13T03:32:51+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-launches-%E2%80%98college-corners%E2%80%99-help-ease-la-students%E2%80%99-path-higher-education + 2020-02-13T03:32:52+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-announces-record-investment-la%E2%80%99s-gear-college-readiness-program + 2020-02-13T03:32:53+00:00 + 0.51 + + + https://www.lamayor.org/expanding-supply-housing + 2020-02-13T03:32:53+00:00 + 0.51 + + + https://www.lamayor.org/files/img3555-14x7jpg + 2020-02-13T03:32:54+00:00 + 0.51 + + + https://www.lamayor.org/growing-our-housing-stock-equitable-and-inclusive-way + 2020-02-13T03:32:55+00:00 + 0.51 + + + https://www.lamayor.org/files/rendering-14x7jpg-1 + 2020-02-13T03:32:56+00:00 + 0.51 + + + https://www.lamayor.org/increasing-housing-stability-renters + 2020-02-13T03:32:57+00:00 + 0.51 + + + https://www.lamayor.org/files/renter-14x7jpg + 2020-02-13T03:32:58+00:00 + 0.51 + + + https://www.lamayor.org/files/welcome-home-14x7jpg + 2020-02-13T03:32:59+00:00 + 0.51 + + + https://www.lamayor.org/report-status-women-girls-los-angeles + 2020-02-13T03:33:00+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-welcomes-first-%E2%80%98pledge-patrol%E2%80%99-class + 2020-02-13T03:33:01+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-and-la-county-unveil-new-family-justice-center + 2020-02-13T03:33:02+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-signs-historic-minimum-wage-increase-law + 2020-02-13T03:33:03+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-wraps-asia-trade-mission-stops-vietnam-hong-kong + 2020-02-13T03:33:05+00:00 + 0.51 + + + https://www.lamayor.org/mayor%E2%80%99s-young-ambassador-program + 2020-02-13T03:33:06+00:00 + 0.51 + + + https://www.lamayor.org/mexico-foreign-minister-ebrard-and-la-mayor-garcetti-announce-mexico-la-commission + 2020-02-13T03:33:07+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-nominates-capri-maddox-lead-los-angeles-department-civil-and-human-rights + 2020-02-13T03:33:08+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-announces-program-provide-entrepreneurship-training-angelenos-experiencing + 2020-02-13T03:33:09+00:00 + 0.51 + + + https://www.lamayor.org/la-london-team-transportation-innovation + 2020-02-13T03:33:10+00:00 + 0.51 + + + https://www.lamayor.org/statement-mayor-garcetti-passing-kobe-bryant + 2020-02-13T03:33:11+00:00 + 0.51 + + + https://www.lamayor.org/statement-mayor-garcetti-talks-hud-secretary-ben-carson-federal-partnership-confront-homelessness + 2020-02-13T03:33:12+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-celebrates-metro-board-vote-advance-nextgen-bus-plan + 2020-02-13T03:33:13+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-announces-10-million-investment-jobs-program-formerly-incarcerated-angelenos + 2020-02-13T03:33:14+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-and-entertainment-industry-leaders-launch-%E2%80%98la-collab%E2%80%99-initiative-increase-latinx + 2020-02-13T03:33:15+00:00 + 0.51 + + + https://www.lamayor.org/media/press_releases?page=1 + 2020-02-13T03:33:16+00:00 + 0.51 + + + https://www.lamayor.org/media/press_releases?page=2 + 2020-02-13T03:33:17+00:00 + 0.51 + + + https://www.lamayor.org/media/press_releases?page=3 + 2020-02-13T03:33:18+00:00 + 0.51 + + + https://www.lamayor.org/media/press_releases?page=4 + 2020-02-13T03:33:19+00:00 + 0.51 + + + https://www.lamayor.org/media/press_releases?page=81 + 2020-02-13T03:33:21+00:00 + 0.51 + + + https://www.lamayor.org/state-city-2019 + 2020-02-13T03:33:22+00:00 + 0.51 + + + https://www.lamayor.org/files/sotc2019jpg + 2020-02-13T03:33:22+00:00 + 0.51 + + + https://www.lamayor.org/SOTC2018 + 2020-02-13T03:33:23+00:00 + 0.51 + + + https://www.lamayor.org/files/41505002341db4366c1f3opng-0 + 2020-02-13T03:33:24+00:00 + 0.51 + + + https://www.lamayor.org/files/inaug-2017-2png-0 + 2020-02-13T03:33:25+00:00 + 0.51 + + + https://www.lamayor.org/state-city-2017 + 2020-02-13T03:33:26+00:00 + 0.51 + + + https://www.lamayor.org/files/photospng-0 + 2020-02-13T03:33:28+00:00 + 0.51 + + + https://www.lamayor.org/SOTC2016 + 2020-02-13T03:33:29+00:00 + 0.51 + + + https://www.lamayor.org/files/nk3a0492-cardjpg-0 + 2020-02-13T03:33:30+00:00 + 0.51 + + + https://www.lamayor.org/files/2015png + 2020-02-13T03:33:31+00:00 + 0.51 + + + https://www.lamayor.org/files/2014png + 2020-02-13T03:33:32+00:00 + 0.51 + + + https://www.lamayor.org/files/inaug-2013png + 2020-02-13T03:33:32+00:00 + 0.51 + + + https://www.lamayor.org/files/videojpg + 2020-02-13T03:33:33+00:00 + 0.51 + + + https://www.lamayor.org/files/megpng + 2020-02-13T03:33:34+00:00 + 0.51 + + + https://www.lamayor.org/files/reddit-ama-cannabis-policypng-0 + 2020-02-13T03:33:35+00:00 + 0.51 + + + https://www.lamayor.org/files/la2028png + 2020-02-13T03:33:36+00:00 + 0.51 + + + https://www.lamayor.org/files/c7dfyqvmaacgtqjpg + 2020-02-13T03:33:37+00:00 + 0.51 + + + https://www.lamayor.org/files/rec-and-parks-amapng + 2020-02-13T03:33:38+00:00 + 0.51 + + + https://www.lamayor.org/files/housing-ama-2png + 2020-02-13T03:33:39+00:00 + 0.51 + + + https://www.lamayor.org/files/seleta-reynolds-ladot-amapng-0 + 2020-02-13T03:33:40+00:00 + 0.51 + + + https://www.lamayor.org/files/10-metro-amapng + 2020-02-13T03:33:41+00:00 + 0.51 + + + https://www.lamayor.org/files/09-voting-amajpg + 2020-02-13T03:33:42+00:00 + 0.51 + + + https://www.lamayor.org/files/08-reentry-amapng + 2020-02-13T03:33:43+00:00 + 0.51 + + + https://www.lamayor.org/files/07-lax-amapng + 2020-02-13T03:33:44+00:00 + 0.51 + + + https://www.lamayor.org/files/06-data-ama-redditpng-0 + 2020-02-13T03:33:45+00:00 + 0.51 + + + https://www.lamayor.org/files/05-dept-cultural-afairs-ama-redditpng-0 + 2020-02-13T03:33:46+00:00 + 0.51 + + + https://www.lamayor.org/files/04-matt-petersen-ama-redditpng + 2020-02-13T03:33:47+00:00 + 0.51 + + + https://www.lamayor.org/files/03-library-ama-redditjpg + 2020-02-13T03:33:48+00:00 + 0.51 + + + https://www.lamayor.org/files/02-transpo-amajpg + 2020-02-13T03:33:49+00:00 + 0.51 + + + https://www.lamayor.org/files/01-meg-amajpg + 2020-02-13T03:33:49+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/HEAP%20One%20Pager.pdf + 2018-10-24T15:37:42+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-quickly-directs-new-dollars-bridge-housing-homeless-angelenos-la-receives-85-million + 2020-02-13T03:33:52+00:00 + 0.51 + + + https://www.lamayor.org/confronting-crisis-update-our-progress-help-homeless-angelenos + 2020-02-13T03:33:54+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-hosts-job-fair-connect-angelenos-new-opportunities-homeless-services + 2020-02-13T03:33:55+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-partners-loyola-law-school-offer-free-legal-services-formerly-incarcerated-angelenos + 2020-02-13T03:33:56+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-proposes-fiscally-responsible-2020-budget-record-reserves-investments-homelessness + 2020-02-13T03:33:57+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2016%20-%20Implementation%20of%20the%20Comprehensive%20Homeless%20Strategy%20(1).pdf + 2016-05-05T22:13:22+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-opens-new-hygiene-center-skid-row + 2020-02-13T03:33:59+00:00 + 0.51 + + + https://www.lamayor.org/bridge-home-spanish + 2020-02-13T03:34:00+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-la-city-council-declare-shelter-crisis-mayor-signs-package-ordinances-address-la%E2%80%99s + 2020-02-13T03:34:01+00:00 + 0.51 + + + https://www.lamayor.org/la-opens-new-homeless-help-desk + 2020-02-13T03:34:02+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-announces-new-plan-deploy-new-sanitation-teams-deliver-services-homeless-encampments + 2020-02-13T03:34:03+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/2019.12.18-UHRCPolicyGroupAgenda1.pdf + 2019-12-13T17:50:50+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/2020.1.15-UHRCPolicyGroupAgenda.pdf + 2020-01-09T22:53:00+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/2020.1.29-UHRCPolicyGroupAgendaFINAL.pdf + 2020-01-24T20:10:33+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/2020.2.12-UHRCPolicyGroupAgenda.pdf + 2020-02-10T18:11:03+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/2019.12.18-UHRCPolicyGroupMeetingMinutesFINAL.pdf + 2020-01-24T20:09:09+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/2020.1.15-UHRCPolicyGroupMeetingMinutesFINAL.pdf + 2020-02-05T20:41:32+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/2020.1.29-UHRCPolicyGroupMeetingMinutesDRAFT.pdf + 2020-02-10T18:10:15+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-continues-tackle-housing-crisis-new-law-strengthening-rent-stabilization-ordinance + 2020-02-13T03:34:10+00:00 + 0.51 + + + https://www.lamayor.org/summary-supportive-housing-pipeline + 2020-02-13T03:34:11+00:00 + 0.51 + + + https://www.lamayor.org/summary-hhh-pipeline + 2020-02-13T03:34:13+00:00 + 0.51 + + + https://www.lamayor.org/summary-hhh-projects-development + 2020-02-13T03:34:14+00:00 + 0.51 + + + https://www.lamayor.org/hhh-projects-development-details + 2020-02-13T03:34:16+00:00 + 0.51 + + + https://www.lamayor.org/non-hhh-funded-projects-development-details + 2020-02-13T03:34:17+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/ED%2015%20-%20Equitable%20Workforce%20and%20Service%20Restoration.pdf + 2016-04-29T19:39:01+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Introducing%20Resilient%20Los%20Angeles.pdf + 2018-02-28T02:11:49+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Resilient%20Los%20Angeles.pdf + 2018-02-28T02:11:35+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/L.A.%20Resilience%20Report%20Spanish.pdf + 2018-03-02T17:24:49+00:00 + 0.51 + + + https://www.lamayor.org/resilience-design-building-stronger-los-angeles + 2020-02-13T03:34:20+00:00 + 0.51 + + + https://www.lamayor.org/statement-mayor-garcetti-presidents-proposal-defund-earthquake-early-warning-system + 2020-02-13T03:34:21+00:00 + 0.51 + + + https://www.lamayor.org/statement-mayor-garcetti-great-shakeout + 2020-02-13T03:34:22+00:00 + 0.51 + + + https://www.lamayor.org/mayor_garcetti_announces_award_for_resilience_by_design_press_release + 2020-02-13T03:34:23+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-signs-historic-earthquake-retrofit-measure-law-ensures-safety-thousands-angelenos + 2020-02-13T03:34:24+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-announces-new-local-earthquake-fault-study-zones-ensure-new-buildings-are-not-built + 2020-02-13T03:34:25+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-announces-new-chief-resilience-officer-coming-la + 2020-02-13T03:34:26+00:00 + 0.51 + + + https://www.lamayor.org/los-angeles-wins-rockefeller-foundation-grant + 2020-02-13T03:34:27+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/article/files/Fossil%20Fuel%20Free%20Streets%20Declaration.pdf + 2017-10-23T18:23:07+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/article/files/101917%20Me%CC%81morandum%20Paris%20LA%20Olympic%20ENG%20FINAL.pdf + 2017-10-23T18:22:54+00:00 + 0.51 + + + https://www.lamayor.org/los-angeles-named-1-energy-star-city-us + 2020-02-13T03:34:29+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-announces-new-100-million-insulation-rebate-program + 2020-02-13T03:34:30+00:00 + 0.51 + + + https://www.lamayor.org/strong-families-la-esp + 2020-02-13T03:34:31+00:00 + 0.51 + + + https://www.lamayor.org/files/sfla-icon-question-14x7jpg-1 + 2020-02-13T03:34:32+00:00 + 0.51 + + + https://www.lamayor.org/files/sfla-icon-local-14x7jpg-0 + 2020-02-13T03:34:33+00:00 + 0.51 + + + https://www.lamayor.org/files/sfla-nounimmigration31-14x7jpg-0 + 2020-02-13T03:34:34+00:00 + 0.51 + + + https://www.lamayor.org/files/sfla-icon-family0-14x70jpg + 2020-02-13T03:34:35+00:00 + 0.51 + + + https://www.lamayor.org/files/sfla-nounquestions55-14x7jpg + 2020-02-13T03:34:36+00:00 + 0.51 + + + https://www.lamayor.org/files/sfla-mail-icon-14x7jpg-0 + 2020-02-13T03:34:37+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/Public%20Comment%20Letter_0.pdf + 2019-10-10T13:43:29+00:00 + 0.51 + + + https://www.lamayor.org/files/sfla-icon-megaphone-14x7jpg-0 + 2020-02-13T03:34:38+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_January_2018_0.pdf + 2018-08-01T22:39:53+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_February_2018_0.pdf + 2018-08-01T22:40:01+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_March_2018_0.pdf + 2018-08-01T22:40:25+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_April_2018_0.pdf + 2018-08-01T22:40:47+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_May_2018.pdf + 2018-08-01T22:41:15+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_June_2018_0.pdf + 2018-08-01T22:41:24+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_July_2018.pdf + 2018-08-01T22:41:33+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_AugustSeptember_2018.pdf + 2018-10-02T22:44:44+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/Newsletter_October_2018_0.pdf + 2018-11-02T21:10:42+00:00 + 0.51 + + + https://www.lamayor.org/statement-mayor-garcetti-federal-government-shutdown + 2020-02-13T03:34:45+00:00 + 0.51 + + + https://www.lamayor.org/mayor-garcetti-launches-new-initiative-expand-support-immigrant-angelenos + 2020-02-13T03:34:46+00:00 + 0.51 + + + https://www.lamayor.org/statement-mayor-garcetti-trump-administration-order-end-temporary-protected-status-el-salvador + 2020-02-13T03:34:47+00:00 + 0.51 + + + https://www.lamayor.org/sites/g/files/wph446/f/article/files/Drop-In-Registration-Form-Spanish.pdf + 2019-01-15T08:03:00+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/article/files/Drop-In-Registration-Form-English_0.pdf + 2019-01-17T02:00:53+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/article/files/LAUSDSchoolVoterRegistrationToolkit.pdf + 2018-09-29T02:52:24+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/article/files/WebinarSlides.pdf + 2018-09-20T16:44:45+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/file/_MYC%20One-Pager%202019-20.pdf + 2019-08-08T23:59:12+00:00 + 0.41 + + + https://www.lamayor.org/budget-resources + 2020-02-13T03:34:55+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/STATEOFTHECITY2019.pdf + 2019-04-18T01:45:43+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/image/LACity_Part1_Demographics.pdf + 2016-02-07T17:58:23+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/image/LACity_Part2_Leadership.pdf + 2016-02-07T17:58:24+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/image/LACity_Part3_Veterans.pdf + 2016-02-07T17:58:24+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/image/LACity_Part4_EducationWorkforceDevelopment.pdf + 2016-02-07T17:58:24+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/image/LACity_Part5_Public_Safety.pdf + 2016-02-07T17:58:24+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/page/image/ED_11.pdf + 2016-02-07T18:05:39+00:00 + 0.41 + + + https://www.lamayor.org/la-puts-higher-education-within-reach-all-students + 2020-02-13T03:35:02+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-reaches-youth-hiring-goal-connecting-more-15000-young-people-jobs-last-year + 2020-02-13T03:35:03+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-completes-trade-mission-trips-tokyo-seoul + 2020-02-13T03:35:04+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-celebrates-opening-first-hhh-funded-supportive-housing-development + 2020-02-13T03:35:05+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-nominates-carolyn-hull-general-manager-economic-and-workforce-development-department + 2020-02-13T03:35:06+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-la-public-library-will-end-late-fines + 2020-02-13T03:35:08+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-announces-major-infrastructure-investments-la-river + 2020-02-13T03:35:09+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-lausd-announce-expansion-program-assist-families-coping-homelessness-housing + 2020-02-13T03:35:10+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-announces-griffith-parkline-shuttle + 2020-02-13T03:35:11+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-councilmember-nury-martinez%E2%80%99s-election-city-council-president + 2020-02-13T03:35:12+00:00 + 0.41 + + + https://www.lamayor.org/mayor-eric-garcetti-meets-community-leaders-urge-neighborhood-census-action + 2020-02-13T03:35:12+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-la-open-new-emergency-shelters-ahead-expected-storm + 2020-02-13T03:35:13+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-joins-los-angeles-cleantech-incubator-announce-new-transportation-electrification + 2020-02-13T03:35:14+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-nominates-heather-repenning-metropolitan-water-district-board-directors + 2020-02-13T03:35:15+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-launches-streetlight-design-competition + 2020-02-13T03:35:16+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-launches-america%E2%80%99s-first-public-private-partnership-transportation-innovation + 2020-02-13T03:35:17+00:00 + 0.41 + + + https://www.lamayor.org/statement-mayor-garcetti-today%E2%80%99s-southern-california-association-governments-vote-housing + 2020-02-13T03:35:19+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-celebrates-final-approval-largest-solar-and-energy-storage-project-us + 2020-02-13T03:35:20+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-nominates-nicole-neeman-brady-ladwp-board-commissioners + 2020-02-13T03:35:21+00:00 + 0.41 + + + https://www.lamayor.org/statement-mayor-garcetti-welcomes-la-students-and-greta-thunberg-city-hall + 2020-02-13T03:35:22+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-kicks-tree-planting-season + 2020-02-13T03:35:23+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-signs-no-fault-eviction-moratorium-ordinance + 2020-02-13T03:35:23+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-kicks-%E2%80%98cool-streets-la%E2%80%99 + 2020-02-13T03:35:25+00:00 + 0.41 + + + https://www.lamayor.org/statement-mayor-garcetti-funding-approval-new-round-prop-hhh-projects + 2020-02-13T03:35:25+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-launches-%E2%80%9Cla-made-40%E2%80%9D-increase-upskilling-la-manufacturing-and-create-new-career + 2020-02-13T03:35:26+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-announces-evacuation-orders-saddleridge-fire-have-been-lifted + 2020-02-13T03:35:27+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcettis-speech-c40-chair-handover-ceremony-copenhagen-denmark + 2020-02-13T03:35:28+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-elected-chair-c40-cities-climate-leadership-group + 2020-02-13T03:35:30+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-celebrates-clean-air-day-unveils-new-zero-emission-equipment-port-los-angeles + 2020-02-13T03:35:31+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-announces-formation-international-gender-equity-network + 2020-02-13T03:35:32+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-proposes-first-ever-paid-parental-leave-la-municipal-employees + 2020-02-13T03:35:33+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-walks-class-students-celebrate-international-walk-school-day + 2020-02-13T03:35:34+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-convenes-inaugural-meeting-youth-council-climate-action + 2020-02-13T03:35:35+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-joins-irish-prime-minister-leo-varadkar-open-new-consulate-los-angeles + 2020-02-13T03:35:36+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-nominates-m-teresa-villegas-board-public-works + 2020-02-13T03:35:37+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-names-nora-adriana-preciado-director-immigrant-affairs + 2020-02-13T03:35:38+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-joins-la-students-global-climate-strike + 2020-02-13T03:35:39+00:00 + 0.41 + + + https://www.lamayor.org/media/press_releases?page=5 + 2020-02-13T03:35:40+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-announces-new-americans-citizenship-center-echo-park-public-library + 2020-02-13T03:35:41+00:00 + 0.41 + + + https://www.lamayor.org/statement-mayor-garcetti-president-trump%E2%80%99s-move-revoke-california%E2%80%99s-authority-regulate-tailpipe + 2020-02-13T03:35:42+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-launches-country%E2%80%99s-first-city-developed-threat-sharing-platform-and-public + 2020-02-13T03:35:43+00:00 + 0.41 + + + https://www.lamayor.org/statement-mayor-garcetti-confirmation-ladwp-general-manager-marty-adams + 2020-02-13T03:35:44+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-creates-dwp-inspector-general + 2020-02-13T03:35:45+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-and-los-angeles-world-airports-break-ground-historic-consolidated-rental-car-facility + 2020-02-13T03:35:46+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-and-mexican-undersecretary-jes%C3%BAs-seade-kick-first-mexla-meeting + 2020-02-13T03:35:47+00:00 + 0.41 + + + https://www.lamayor.org/garcetti-administration-approves-building-largest-solar-and-battery-storage-project-us + 2020-02-13T03:35:48+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-letter-president-trump-homelessness-crisis + 2020-02-13T03:35:49+00:00 + 0.41 + + + https://www.lamayor.org/la-leaders-join-meeting-governor%E2%80%99s-council-regional-homeless-advisors + 2020-02-13T03:35:50+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-appoints-shannon-hoppes-chief-procurement-officer + 2020-02-13T03:35:51+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-stresses-innovation-accountability-yearly-review-general-managers + 2020-02-13T03:35:52+00:00 + 0.41 + + + https://www.lamayor.org/media/press_releases?page=6 + 2020-02-13T03:35:53+00:00 + 0.41 + + + https://www.lamayor.org/mayor_garcetti_kicks_off_2013_hire_la_summer_youth_employment_program + 2020-02-13T03:35:54+00:00 + 0.41 + + + https://www.lamayor.org/mayor_garcetti_announces_appointments + 2020-02-13T03:35:55+00:00 + 0.41 + + + https://www.lamayor.org/media/press_releases?page=80 + 2020-02-13T03:35:56+00:00 + 0.41 + + + https://www.lamayor.org/media/press_releases?page=77 + 2020-02-13T03:35:57+00:00 + 0.41 + + + https://www.lamayor.org/media/press_releases?page=78 + 2020-02-13T03:35:58+00:00 + 0.41 + + + https://www.lamayor.org/media/press_releases?page=79 + 2020-02-13T03:35:59+00:00 + 0.41 + + + https://www.lamayor.org/files/46909171964d984286dceojpg + 2020-02-13T03:36:00+00:00 + 0.41 + + + https://www.lamayor.org/files/image-1jpg + 2020-02-13T03:36:01+00:00 + 0.41 + + + https://www.lamayor.org/files/aa1jpg + 2020-02-13T03:36:02+00:00 + 0.41 + + + https://www.lamayor.org/state-city-2019-data + 2020-02-13T03:36:03+00:00 + 0.41 + + + https://www.lamayor.org/files/datajpg + 2020-02-13T03:36:04+00:00 + 0.41 + + + https://www.lamayor.org/files/41505002341db4366c1f3opng + 2020-02-13T03:36:05+00:00 + 0.41 + + + https://www.lamayor.org/files/sotc-year-recap-tilepng + 2020-02-13T03:36:06+00:00 + 0.41 + + + https://www.lamayor.org/files/text-speechpng + 2020-02-13T03:36:07+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/2018_SOTC_AS_PREPARED_FOR_DELIVERY%20%28ESP%29.pdf.pdf + 2018-04-16T23:38:39+00:00 + 0.41 + + + https://www.lamayor.org/files/text-speechpng-0 + 2020-02-13T03:36:08+00:00 + 0.41 + + + https://www.lamayor.org/sotc-2018-data + 2020-02-13T03:36:09+00:00 + 0.41 + + + https://www.lamayor.org/files/data-tilepng + 2020-02-13T03:36:10+00:00 + 0.41 + + + https://www.lamayor.org/files/cisco-brotherspng + 2020-02-13T03:36:11+00:00 + 0.41 + + + https://www.lamayor.org/files/taliapng + 2020-02-13T03:36:12+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/StateoftheCity2017--ForMediaDistribution.pdf + 2017-04-20T18:18:09+00:00 + 0.41 + + + https://www.lamayor.org/files/textpng + 2020-02-13T03:36:14+00:00 + 0.41 + + + https://www.lamayor.org/files/datapng + 2020-02-13T03:36:15+00:00 + 0.41 + + + https://www.lamayor.org/files/photospng + 2020-02-13T03:36:16+00:00 + 0.41 + + + https://www.lamayor.org/files/0a7a0693jpg + 2020-02-13T03:36:16+00:00 + 0.41 + + + https://www.lamayor.org/files/nk3a0492-cardjpg + 2020-02-13T03:36:17+00:00 + 0.41 + + + https://www.lamayor.org/files/web-stuffsjpg + 2020-02-13T03:36:18+00:00 + 0.41 + + + https://www.lamayor.org/files/nbpng + 2020-02-13T03:36:19+00:00 + 0.41 + + + https://www.lamayor.org/files/brpng-1 + 2020-02-13T03:36:20+00:00 + 0.41 + + + https://www.lamayor.org/files/dapng-0 + 2020-02-13T03:36:21+00:00 + 0.41 + + + https://www.lamayor.org/files/497a4260jpg + 2020-02-13T03:36:22+00:00 + 0.41 + + + https://www.lamayor.org/files/garcetti-sotc-infographic-11-infrastructurepng-0 + 2020-02-13T03:36:23+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/2016%20State%20of%20the%20City%20Address.pdf + 2016-04-15T01:46:10+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-proposes-fiscally-responsible-2019-budget-boosts-investment-homelessness + 2020-02-13T03:36:25+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-signs-executive-directive-accelerate-construction-emergency-homeless-shelters + 2020-02-13T03:36:25+00:00 + 0.41 + + + https://www.lamayor.org/mayor-garcetti-launches-%E2%80%98save-energy-la%E2%80%99-campaign-and-asks-angelenos-conserve-energy-during-summer + 2020-02-13T03:36:27+00:00 + 0.41 + + + https://www.lamayor.org/files/icon-questionpng-0 + 2020-02-13T03:36:28+00:00 + 0.41 + + + https://www.lamayor.org/files/icon-localpng-0 + 2020-02-13T03:36:29+00:00 + 0.41 + + + https://www.lamayor.org/files/sfla-nounimmigration31-14x70jpg + 2020-02-13T03:36:30+00:00 + 0.41 + + + https://www.lamayor.org/files/icon-familypng-1 + 2020-02-13T03:36:31+00:00 + 0.41 + + + https://www.lamayor.org/files/sfla-nounquestions55-14x7jpg-0 + 2020-02-13T03:36:32+00:00 + 0.41 + + + https://www.lamayor.org/files/sfla-mail-icon-14x70jpg + 2020-02-13T03:36:33+00:00 + 0.41 + + + https://www.lamayor.org/files/icon-megaphonepng-1 + 2020-02-13T03:36:34+00:00 + 0.41 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/2021_Mayors_Budget_Policy_Letter.pdf + 2019-11-04T23:04:42+00:00 + 0.33 + + + https://www.lamayor.org/files/letterjpg + 2020-02-13T03:36:35+00:00 + 0.33 + + + https://www.lamayor.org/sites/g/files/wph446/f/landing_pages/files/CAO_Budget_Instructions.pdf + 2019-11-04T23:04:34+00:00 + 0.33 + + + https://www.lamayor.org/files/caopng + 2020-02-13T03:36:37+00:00 + 0.33 + + + https://www.lamayor.org/files/city-hall2jpg-0 + 2020-02-13T03:36:38+00:00 + 0.33 + + + https://www.lamayor.org/sites/g/files/wph446/f/article/files/Council-Transmittal-Paid-Parental-Leave.pdf + 2019-10-02T20:30:18+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-welcomes-new-school-year-free-dash-bus-service-students + 2020-02-13T03:36:39+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-hosts-state-leaders-highlight-local-and-state-action-strengthen-water-resilience + 2020-02-13T03:36:40+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-nominates-raquel-beltr%C3%A1n-general-manager-department-neighborhood-empowerment + 2020-02-13T03:36:41+00:00 + 0.33 + + + https://www.lamayor.org/mayor%E2%80%99s-youth-council-end-gun-violence-convenes-response-el-paso-and-dayton-shootings + 2020-02-13T03:36:42+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-and-usgs-announce-key-updates-shakealertla + 2020-02-13T03:36:43+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-meets-pope-francis-vatican + 2020-02-13T03:36:44+00:00 + 0.33 + + + https://www.lamayor.org/statement-mayor-garcetti-announces-agreement-us-department-housing-and-urban-development-continued + 2020-02-13T03:36:45+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-names-rachel-malarich-la%E2%80%99s-first-ever-city-forest-officer + 2020-02-13T03:36:46+00:00 + 0.33 + + + https://www.lamayor.org/statement-office-mayor-data-event + 2020-02-13T03:36:47+00:00 + 0.33 + + + https://www.lamayor.org/statement-mayor-garcetti-death-lapd-officer-juan-diaz + 2020-02-13T03:36:48+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-and-councilmember-buscaino-announce-agreement-between-ilwu-apmt-and-pma-launch-blue + 2020-02-13T03:36:49+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-highlights-la%E2%80%99s-leadership-global-development-goals + 2020-02-13T03:36:50+00:00 + 0.33 + + + https://www.lamayor.org/media/press_releases?page=7 + 2020-02-13T03:36:51+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-nominates-tracy-quinn-metropolitan-water-district-board-directors + 2020-02-13T03:36:52+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-la-will-fight-keep-families-together-%E2%80%94-and-will-not-assist-immigration-enforcement + 2020-02-13T03:36:53+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-launches-initiative-strengthen-emergency-preparedness-la-neighborhoods + 2020-02-13T03:36:54+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-names-dae-levine-director-communications-andre-herndon-senior-communications-advisor + 2020-02-13T03:36:56+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-announces-ladwp-power-outage-notification-service + 2020-02-13T03:36:57+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-leads-coalition-mayors-calling-federal-action-confront-homelessness-crisis + 2020-02-13T03:36:58+00:00 + 0.33 + + + https://www.lamayor.org/statement-mayor-garcetti-supreme-court-ruling-census-citizenship-question + 2020-02-13T03:36:59+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-la-won%E2%80%99t-assist-immigration-enforcement-will-help-connect-families-resources + 2020-02-13T03:37:00+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-nominates-denise-verret-director-la-zoo-and-botanical-gardens + 2020-02-13T03:37:01+00:00 + 0.33 + + + https://www.lamayor.org/statement-mayor-garcetti-president-trump%E2%80%99s-threats-against-immigrants + 2020-02-13T03:37:01+00:00 + 0.33 + + + https://www.lamayor.org/media/press_releases?page=8 + 2020-02-13T03:37:02+00:00 + 0.33 + + + https://www.lamayor.org/mayor_garcetti_announces_grant_to_fund_nuclear_detection_halo + 2020-02-13T03:37:04+00:00 + 0.33 + + + https://www.lamayor.org/mayor_garcetti_welcomes_riot_games_to_los_angeles + 2020-02-13T03:37:05+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-issues-executive-directive-cyber-security + 2020-02-13T03:37:05+00:00 + 0.33 + + + https://www.lamayor.org/mayor_garcetti_announces_harbor_commission_appointments + 2020-02-13T03:37:07+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-issues-first-executive-directive-launch-great-streets-initiative + 2020-02-13T03:37:08+00:00 + 0.33 + + + https://www.lamayor.org/mayor_garcetti_discusses_first_100_days_and_previews_performance_metrics + 2020-02-13T03:37:08+00:00 + 0.33 + + + https://www.lamayor.org/mayor_garcetti_announces_dwp_commission_appointments + 2020-02-13T03:37:09+00:00 + 0.33 + + + https://www.lamayor.org/matt-petersen-appointed-new-position-chief-sustainability-officer + 2020-02-13T03:37:10+00:00 + 0.33 + + + https://www.lamayor.org/mayor_garcetti_opens_help_desk + 2020-02-13T03:37:11+00:00 + 0.33 + + + https://www.lamayor.org/mayor_garcetti_announces_mta_board_appointments + 2020-02-13T03:37:12+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-announces-appointments + 2020-02-13T03:37:13+00:00 + 0.33 + + + https://www.lamayor.org/latest-update-water-main-break + 2020-02-13T03:37:14+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-appoints-ralph-terrazas-chief-la-fire-department + 2020-02-13T03:37:15+00:00 + 0.33 + + + https://www.lamayor.org/mayor-garcetti-thanks-la-employers-hiring-10000-youth-summer-jobs-through-hire-las-youth-initiative + 2020-02-13T03:37:16+00:00 + 0.33 + + + + \ No newline at end of file diff --git a/projects/web-scraper/lib/spider/spider.js b/projects/web-scraper/lib/spider/spider.js new file mode 100644 index 0000000..dfccfb5 --- /dev/null +++ b/projects/web-scraper/lib/spider/spider.js @@ -0,0 +1,9 @@ +var mongoose = require('mongoose'); +var SpiderSchema = new mongoose.Schema({ + title: String, + description: String, + pinned: { type: Boolean, default: false } +}); +mongoose.model('Spider', NoteSchema); + +module.exports = mongoose.model('Spider'); \ No newline at end of file diff --git a/projects/web-scraper/lib/spider/spiderController.js b/projects/web-scraper/lib/spider/spiderController.js new file mode 100644 index 0000000..b6b7399 --- /dev/null +++ b/projects/web-scraper/lib/spider/spiderController.js @@ -0,0 +1,143 @@ +module.exports = function (app) { + if (!app) throw new Error('Missing parameter: \'app\' not provided.'); + + var express = require('express'); + var SpiderController = express.Router(); + var SpiderProvider = require('./SpiderProvider'); + var validateSpider = require('./validateSpider'); + var VerifyToken = require(__root + 'auth/VerifyToken')(app); + let z = 1; + + const puppeteer = require('puppeteer'); // v 1.1.0 + const { URL } = require('url'); + const fse = require('fs-extra'); // v 5.0.0 + const path = require('path'); + const xml = require('xml2js'); + const parser = new xml.Parser(); + + + // Puppeteer worker. + async function start(urlToFetch, siteName) { + /* 1 */ + const browser = await puppeteer.launch(); + const page = await browser.newPage(); + //await page.setRequestInterception(true); + /* 2 */ + page.on('response', async (response) => { + + try { + const status = response.status; + + page.on( "console", function (log) { + console.log(log.text()); + }); + const url = new URL(response.url()); + + // This intercept is a method of blocking potential URL redirects + // console.log(interceptedRequest.url(), url, status); + // if (interceptedRequest.url().endsWith('/api')) { + // interceptedRequest.respond({ + // status: 422, + // body: "FAKE" + // }) + // } else interceptedRequest.continue(); + + let filePath = path.resolve(`./output/` + siteName + '/' + `${url.pathname}`); + if (path.extname(url.pathname).trim() === '') { + console.log(url.pathname, `${filePath}/index.html`) + filePath = `${filePath}/index.html`; + } + + // Save Data + if ((status >= 300) && (status <= 399)) { + console.log('Redirect from', response.url(), 'to', response.headers()['location']); + await new Promise(function(resolve, reject) { + resolve(null); + }); + } else { + await fse.outputFile(filePath, await response.buffer()); + await fse.outputFile(filePath + z, await response.buffer()); + z++; + } + + } catch (err) { + console.log('Redirect from', response.url(), 'to', response.headers()['location'], err, response.status); + } + }); + + /* 3 */ + await page.goto(urlToFetch, { + waitUntil: 'load' + }); + + /* 4 */ + setTimeout(async () => { + await browser.close(); + }, 60000 * 4); + } + + // Headless chrome requires a lot or resources. + function createQueue(tasks, maxNumOfWorkers = 4) { + var numOfWorkers = 0; + var taskIndex = 0; + + return new Promise( done => { + const handleResult = index => result => { + tasks[index] = result; + numOfWorkers--; + getNextTask(); + }; + const getNextTask = () => { + if (numOfWorkers < maxNumOfWorkers && taskIndex < tasks.length) { + tasks[taskIndex]().then(handleResult(taskIndex)).catch(handleResult(taskIndex)); + taskIndex++; + numOfWorkers++; + getNextTask(); + } else if (numOfWorkers === 0 && taskIndex === tasks.length) { + done(tasks); + } + }; + getNextTask(); + }); + } + + let items = []; + + // Read sites generated by web indexer. + let xmlInjest = (siteName, path) => { + fse.readFile(__dirname + path, (err, data) => { + parser.parseString(data, (err, result) => { + result.urlset.url.forEach((item, index) => { + items.push(async () => { + //console.log(item.loc[0]); + start(item.loc[0]); + }); + }); + console.log(JSON.stringify(items[0])); + // Read X items, Y at a time. + createQueue(items.slice(0,2), 1); + }); + }); + }; + + //start('https://www.lamayor.org/', 'www'); + xmlInjest('www-lamayor', '/siteMap.xml'); + + // CREATES A NEW NOTE + SpiderController.post('/', VerifyToken, validateNote, SpiderProvider.createNote); + + // RETURNS ALL THE NOTES IN THE DATABASE + SpiderController.get('/', SpiderProvider.getNotes); + + // GETS A SINGLE NOTE FROM THE DATABASE + SpiderController.get('/:id', SpiderProvider.getNote); + + // DELETES A NOTE FROM THE DATABASE + SpiderController.delete('/:id', VerifyToken, SpiderProvider.deleteNote); + + // UPDATES A SINGLE NOTE IN THE DATABASE + SpiderController.put('/:id', VerifyToken, SpiderProvider.putNote); + + return SpiderController; + +}; \ No newline at end of file diff --git a/projects/web-scraper/lib/spider/spiderProvider.js b/projects/web-scraper/lib/spider/spiderProvider.js new file mode 100644 index 0000000..05c8f9b --- /dev/null +++ b/projects/web-scraper/lib/spider/spiderProvider.js @@ -0,0 +1,45 @@ +var Spider = require('./Spider'); +module.exports = { + createSpider: createSpider, + getSpiders: getSpiders, + getSpider: getSpider, + deleteSpider: deleteSpider, + putSpider: putSpider +}; + +function createSpider(req, res, next) { + return Spider.create({ + title : req.body.title, + description : req.body.description, + pinned : req.body.pinned + }) + .then(note => res.status(200).send(note)) + .catch(err => next(new Error(err))); +} + +function getSpiders(req, res, next) { + return Spider.find({}) + .then(Spiders => res.status(200).send(notes)) + .catch(err => next(new Error(err))); +} + +function getSpider(req, res, next) { + return Spider.findById(req.params.id) + .then(Spider => { + if (!Spider) return res.status(404).send('No Spider found.'); + res.status(200).send(Spider); + }) + .catch(err => next(new Error(err))); +} + +function deleteSpider(req, res, next) { + return Spider.findByIdAndRemove(req.params.id) + .then(Spider => res.status(200).send(Spider)) + .catch(err => next(new Error(err))); +} + +function putSpider(req, res, next) { + return Spider.findByIdAndUpdate(req.params.id, req.body, {new: true}) + .then(Spider => res.status(200).send(Spider)) + .catch(err => next(new Error(err))); +} \ No newline at end of file diff --git a/projects/web-scraper/lib/spider/validateSpider.js b/projects/web-scraper/lib/spider/validateSpider.js new file mode 100644 index 0000000..bc34e46 --- /dev/null +++ b/projects/web-scraper/lib/spider/validateSpider.js @@ -0,0 +1,9 @@ +function validateSpider(req, res, next) { + if (typeof req.body.title === 'string' && + typeof req.body.description === 'string') + return next(); + + return res.status(500).send('Please enter a valid Spider.'); +} + +module.exports = validateSpider; \ No newline at end of file diff --git a/projects/web-scraper/lib/user/User.js b/projects/web-scraper/lib/user/User.js new file mode 100644 index 0000000..49ef4dc --- /dev/null +++ b/projects/web-scraper/lib/user/User.js @@ -0,0 +1,9 @@ +var mongoose = require('mongoose'); +var UserSchema = new mongoose.Schema({ + name: String, + email: String, + password: String +}); +mongoose.model('User', UserSchema); + +module.exports = mongoose.model('User'); \ No newline at end of file diff --git a/projects/web-scraper/lib/user/UserController.js b/projects/web-scraper/lib/user/UserController.js new file mode 100644 index 0000000..390d25c --- /dev/null +++ b/projects/web-scraper/lib/user/UserController.js @@ -0,0 +1,27 @@ +module.exports = function (app) { + if (!app) throw new Error('Missing parameter: \'app\' not provided.'); + + var express = require('express'); + var UserController = express.Router(); + var UserProvider = require('./UserProvider'); + var VerifyToken = require(__root + 'auth/VerifyToken')(app); + + // CREATES A NEW USER + UserController.post('/', VerifyToken, UserProvider.createUser); + + // RETURNS ALL THE USERS IN THE DATABASE + UserController.get('/', UserProvider.getUsers); + + // GETS A SINGLE USER FROM THE DATABASE + UserController.get('/:id', UserProvider.getUser); + + // DELETES A USER FROM THE DATABASE + UserController.delete('/:id', VerifyToken, UserProvider.deleteUser); + + // UPDATES A SINGLE USER IN THE DATABASE + // Added VerifyToken middleware to make sure only an authenticated user can put to this route + UserController.put('/:id', VerifyToken, UserProvider.putUser); + + return UserController; + +}; diff --git a/projects/web-scraper/lib/user/UserProvider.js b/projects/web-scraper/lib/user/UserProvider.js new file mode 100644 index 0000000..f58561a --- /dev/null +++ b/projects/web-scraper/lib/user/UserProvider.js @@ -0,0 +1,45 @@ +var User = require('./User'); +module.exports = { + createUser: createUser, + getUsers: getUsers, + getUser: getUser, + deleteUser: deleteUser, + putUser: putUser +}; + +function createUser(req, res, next) { + return User.create({ + name: req.body.name, + email: req.body.email, + password: req.body.password + }) + .then(user => res.status(200).send(user)) + .catch(err => next(new Error(err))); +} + +function getUsers(req, res, next) { + return User.find({}) + .then(users => res.status(200).send(users)) + .catch(err => next(new Error(err))); +} + +function getUser(req, res, next) { + return User.findById(req.params.id) + .then(user => { + if (!user) return res.send(404).send('No user found.'); + res.status(200).send(user); + }) + .catch(err => next(new Error(err))); +} + +function deleteUser(req, res, next) { + return User.findByIdAndRemove(req.params.id) + .then(user => res.status(200).send(user)) + .catch(err => next(new Error(err))); +} + +function putUser(req, res, next) { + return User.findByIdAndUpdate(req.params.id, req.body, { new: true }) + .then(user => res.status(200).send(user)) + .catch(err => next(new Error(err))); +} \ No newline at end of file diff --git a/projects/web-scraper/mongo-deployment.yaml b/projects/web-scraper/mongo-deployment.yaml new file mode 100644 index 0000000..636cfe0 --- /dev/null +++ b/projects/web-scraper/mongo-deployment.yaml @@ -0,0 +1,39 @@ +apiVersion: extensions/v1beta1 +kind: Deployment +metadata: + annotations: + kompose.cmd: kompose -f docker-kompose.yml convert + kompose.version: 1.1.0 (36652f6) + kompose.volume.size: 1Gi + creationTimestamp: null + labels: + io.kompose.service: mongo + name: mongo +spec: + replicas: 1 + strategy: + type: Recreate + template: + metadata: + creationTimestamp: null + labels: + io.kompose.service: mongo + spec: + containers: + - env: + - name: GET_HOSTS_FROM + value: dns + image: mongo:4.0 + name: mongo + ports: + - containerPort: 27017 + resources: {} + volumeMounts: + - mountPath: /data/db + name: mongo + restartPolicy: Always + volumes: + - name: mongo + persistentVolumeClaim: + claimName: mongo +status: {} diff --git a/projects/web-scraper/mongo-persistentvolumeclaim.yaml b/projects/web-scraper/mongo-persistentvolumeclaim.yaml new file mode 100644 index 0000000..40131ef --- /dev/null +++ b/projects/web-scraper/mongo-persistentvolumeclaim.yaml @@ -0,0 +1,14 @@ +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + creationTimestamp: null + labels: + io.kompose.service: mongo + name: mongo +spec: + accessModes: + - ReadWriteOnce + resources: + requests: + storage: 100Mi +status: {} diff --git a/projects/web-scraper/mongo-service.yaml b/projects/web-scraper/mongo-service.yaml new file mode 100644 index 0000000..6a30cd2 --- /dev/null +++ b/projects/web-scraper/mongo-service.yaml @@ -0,0 +1,20 @@ +apiVersion: v1 +kind: Service +metadata: + annotations: + kompose.cmd: kompose -f docker-kompose.yml convert + kompose.version: 1.1.0 (36652f6) + kompose.volume.size: 1Gi + creationTimestamp: null + labels: + io.kompose.service: mongo + name: mongo +spec: + ports: + - name: "27017" + port: 27017 + targetPort: 27017 + selector: + io.kompose.service: mongo +status: + loadBalancer: {} diff --git a/projects/web-scraper/package.json b/projects/web-scraper/package.json new file mode 100644 index 0000000..557df9f --- /dev/null +++ b/projects/web-scraper/package.json @@ -0,0 +1,61 @@ +{ + "name": "Web-Scraper", + "version": "0.1.0", + "engines": { + "node": ">=8.0.0" + }, + "description": "Lightweight Node.js API with Docker support, test coverage and circle.ci support.", + "main": "app.js", + "scripts": { + "start": "node app.js", + "docker": "nodemon -L app.js", + "prod": "node app.js", + "dev": "nodemon app.js", + "watch": "nodemon --exec 'npm run lint && npm start'", + "test": "nodemon test.js", + "test-integration": "node test-integration.js", + "test-unit": "node test-unit.js", + "lint": "eslint ./", + "coverage": "npm run lint && nyc npm run integration" + }, + "repository": { + "type": "git", + "url": "https://github.com/hackforla/gender-equity.git" + }, + "keywords": [ + "node", + "api", + "express", + "boilerplate" + ], + "author": "hackforla", + "license": "GPL-2.0", + "description": "A headless webscraper", + "dependencies": { + "async": "^2.5.0", + "bcryptjs": "^2.4.3", + "bcryptjs-then": "^1.0.1", + "body-parser": "^1.18.3", + "dotenv": "^4.0.0", + "express": "^4.16.4", + "fs": "0.0.1-security", + "fs-extra": "^8.1.0", + "jsonwebtoken": "^8.3.0", + "mongoose": "^5.3.4", + "path": "^0.12.7", + "puppeteer": "^2.1.1", + "url": "^0.11.0", + "winston": "^3.0.0-rc1", + "xml2js": "^0.4.23" + }, + "devDependencies": { + "chai": "^4.1.0", + "chai-as-promised": "^7.1.1", + "chai-http": "^4.2.0", + "eslint": "^4.9.0", + "eslint-plugin-node": "^5.2.1", + "istanbul": "^0.4.5", + "mocha": "^5.2.0", + "nodemon": "^1.18.7" + } +} diff --git a/projects/web-scraper/sample.variables.env b/projects/web-scraper/sample.variables.env new file mode 100644 index 0000000..78337cb --- /dev/null +++ b/projects/web-scraper/sample.variables.env @@ -0,0 +1,4 @@ +SECRET=secretkey +NODE_ENV=dev +DB=mongodb://localhost:27017/your_database +PORT=8000 \ No newline at end of file diff --git a/projects/web-scraper/smoke_test.sh b/projects/web-scraper/smoke_test.sh new file mode 100755 index 0000000..fea5c40 --- /dev/null +++ b/projects/web-scraper/smoke_test.sh @@ -0,0 +1,14 @@ +attempt_counter=0 +max_attempts=20 +endpoint=$1 + +until $(curl --output /dev/null --silent --head --fail ${endpoint}); do + if [ ${attempt_counter} -eq ${max_attempts} ];then + echo "Max attempts reached" + exit 1 + fi + + printf '.' + attempt_counter=$(($attempt_counter+1)) + sleep 5 +done diff --git a/projects/web-scraper/test-integration.js b/projects/web-scraper/test-integration.js new file mode 100644 index 0000000..137ba8b --- /dev/null +++ b/projects/web-scraper/test-integration.js @@ -0,0 +1,2 @@ +const runTest = require('./test'); +runTest('test/integration'); \ No newline at end of file diff --git a/projects/web-scraper/test-unit.js b/projects/web-scraper/test-unit.js new file mode 100644 index 0000000..61db6ab --- /dev/null +++ b/projects/web-scraper/test-unit.js @@ -0,0 +1,2 @@ +const runTest = require('./test'); +runTest('test/unit'); \ No newline at end of file diff --git a/projects/web-scraper/test.js b/projects/web-scraper/test.js new file mode 100644 index 0000000..409c8a9 --- /dev/null +++ b/projects/web-scraper/test.js @@ -0,0 +1,67 @@ +module.exports = (testDir) => { + require('dotenv').config(); + const Mocha = require('mocha'); + const async = require('async'); + const mocha = new Mocha(); + + //// Add test files + var files = Mocha.utils.lookupFiles(testDir, ['js'], true); + files.forEach(function (file) { + mocha.addFile(file); + }); + + global.__root = __dirname + '/'; + + //// Global export of mutil + global.mutil = { + getApp: getApp, + clearDB: clearDB, + getModel: getModel, + parseJSON: parseJSON + }; + + //// Chai - Configure + const chai = require('chai'); + chai.use(require('chai-http')); + chai.use(require('chai-as-promised')); + + + //// run the server and get the app object + const server = require('./lib/server'); + let appToReturn; + server.serve('test') + .then(function (app) { + appToReturn = app; + mocha.ui('bdd').run(code => { + process.exit(code) + }); // exit the node process on test end + }) + .catch('Failed to start test server.'); + + /////////////////////////// + //// MUTIL functions + /////////////////////////// + function getApp() { + return appToReturn; + } + + function clearDB(done) { + let mongoose = require('mongoose'); + async.each(mongoose.models, function (model, next) { + model.remove(next); + }, done); + } + + function getModel(model_name) { + try { + return require('mongoose').model(model_name); + } catch (err) { + return null; + } + + } + + function parseJSON(obj) { + return JSON.parse(JSON.stringify(obj)); + } +} \ No newline at end of file diff --git a/projects/web-scraper/test/integration/Auth.js b/projects/web-scraper/test/integration/Auth.js new file mode 100644 index 0000000..ba57005 --- /dev/null +++ b/projects/web-scraper/test/integration/Auth.js @@ -0,0 +1,217 @@ +const expect = require('chai').expect; +const request = require('chai').request; + +/** + * Set up server and constants + */ +const app = mutil.getApp(); +const User = mutil.getModel('User'); +const parseJSON = mutil.parseJSON; + +let token; +const testUser = { + name: 'superAdmin', + email: 'admin@example.com', + password: 'superSecret' +}; +const userWithInvalidEmail = { + name: 'superAdmin', + email: 'adminexamplecom', + password: 'superSecret' +}; +const userWithInvalidUsername = { + name: 'super', + email: 'admin@example.com', + password: 'superSecret' +}; +const userWithInvalidPassword = { + name: 'super', + email: 'admin@example.com', + password: 'super' +}; +let returnedUser = {}; +function registerUserWithInvalidEmail() { + return request(app) + .post('/api/auth/asyncRegister') + .send(userWithInvalidEmail); +} +function registerUserWithInvalidUsername() { + return request(app) + .post('/api/auth/asyncRegister') + .send(userWithInvalidUsername); +} +function registerUserWithInvalidPassword() { + return request(app) + .post('/api/auth/asyncRegister') + .send(userWithInvalidPassword); +} +function registerUser() { + return request(app) + .post('/api/auth/asyncRegister') + .send(testUser); +} +function authUser() { + return request(app) + .post('/api/auth/login') + .send(testUser); +} +function me(token) { + return request(app) + .get('/api/auth/me') + .set('x-access-token', token) + .then(res => res.body); +} +function asyncMe(token) { + return request(app) + .get('/api/auth/asyncMe') + .set('x-access-token', token) + .then(res => res.body); +} +function deleteUser(token) { + return request(app) + .get('/api/auth/asyncMe') + .set('x-access-token', token) + .then(res => res.body) + .then(user => request(app) + .delete('/api/users/' + user._id) + .set('x-access-token', token)); +} + +describe('AuthController', function () { + + before(function () { + return new Promise((resolve, reject) => { + mutil.clearDB(function (err) { + if (err) + return reject(err); + + resolve(); + }) + }) + }); + after(function (done) { + mutil.clearDB(function () { + done(); + }); + }); + + + describe('AuthProvider', function () { + + describe('.register', function () { + it('should not allow a user to register with an invalid email', function () { + return registerUserWithInvalidEmail() + .catch(err => { + expect(err).to.have.status(500); + expect(err).to.have.property('message'); + expect(err).to.have.property('stack'); + }); + }); + it('should not allow a user to register with an invalid username', function () { + return registerUserWithInvalidUsername() + .catch(err => { + expect(err).to.have.status(500); + expect(err).to.have.property('message'); + expect(err).to.have.property('stack'); + }); + }); + it('should not allow a user to register with an invalid password', function () { + return registerUserWithInvalidPassword() + .catch(err => { + expect(err).to.have.status(500); + expect(err).to.have.property('message'); + expect(err).to.have.property('stack'); + }); + }); + it('should register a new user', function () { + return registerUser() + .then(res => { + + expect(res).to.have.status(200); + expect(res.body).to.not.be.null.and.not.to.be.undefined; + expect(res.body).to.have.property('token'); + token = res.body.token; + + return token; + + }) + .then(me) + .then(returnedUser => + User.findById(returnedUser._id, { password: 0 }) + .then(user => + expect(parseJSON(user)).to.eql(parseJSON(returnedUser)))) + }); + it('should not allow to register the same user twice', function () { + return registerUser() + .catch(err => { + expect(err).to.have.status(500); + expect(err).to.have.property('message'); + expect(err).to.have.property('stack'); + }); + }); + }); + + describe('.login', function () { + it('should authenticate a user', function () { + return authUser() + .then(res => { + + expect(res).to.have.status(200); + expect(res.body).to.not.be.null.and.not.to.be.undefined; + expect(res.body).to.have.property('token'); + token = res.body.token; + + return token; + + }) + .then(me) + .then(returnedUser => User.findById(returnedUser._id, { password: 0 }).then(user => expect(parseJSON(user)).to.eql(parseJSON(returnedUser)))) + .catch(err => expect(err).to.be.null); + }); + }); + + describe('.me', function () { + it('should return the authenticated user', function () { + return me(token) + .then(returnedUser => User.findById(returnedUser._id, { password: 0 }) + .then(user => expect(parseJSON(user)).to.eql(parseJSON(returnedUser)))); + }); + }); + + describe('.asyncMe', function () { + it('should return the authenticated user (written with async/await)', function () { + return asyncMe(token) + .then(returnedUser => User.findById(returnedUser._id, { password: 0 }) + .then(user => expect(parseJSON(user)).to.eql(parseJSON(returnedUser)))); + }); + it('should fail gracefully when no user exists with current token (written with async/await)', function () { + return deleteUser(token) + .then(() => asyncMe(token)) + .catch(err => { + expect(err).to.have.status(404); + expect(err).to.have.property('message'); + expect(err).to.have.property('stack'); + }); + }); + }); + }); + + describe('VerifyToken', function () { + it('should fail with 403 if \'no token provided\'', function () { + return asyncMe('') + .catch(err => { + expect(err).to.have.status(403); + expect(err).to.have.property('message'); + expect(err).to.have.property('stack'); + }); + }); + it('should fail with 500 if \'failed to authenticate token\'', function () { + return asyncMe('somerandomtoken') + .catch(err => { + expect(err).to.have.status(500); + expect(err).to.have.property('message'); + expect(err).to.have.property('stack'); + }); + }); + }); +}); \ No newline at end of file diff --git a/projects/web-scraper/test/integration/Note.js b/projects/web-scraper/test/integration/Note.js new file mode 100644 index 0000000..f93692a --- /dev/null +++ b/projects/web-scraper/test/integration/Note.js @@ -0,0 +1,118 @@ +const expect = require('chai').expect; +const request = require('chai').request; + +/** + * Set up server and constants + */ +const app = mutil.getApp(); +const Note = mutil.getModel('Note'); +const parseJSON = mutil.parseJSON; + +let token; +const testUser = { + name: 'somerandomuser', + email: 'somerandomuser@example.com', + password: 'supersecret' +}; +function registerUser() { + return request(app) + .post('/api/auth/register') + .send(testUser) + // .then(res => token = res.body.token); + .then(res => { + token = res.body.token; + }); +} + +let testNote = { + title: 'Buy bread', + description: 'This task is very important' +}; + +describe('NoteController', function () { + + before(function () { + return new Promise((resolve, reject) => { + mutil.clearDB(function(err){ + if(err) + return reject(err); + + resolve(); + }) + }) + .then(registerUser); + }); + after(function (done) { + mutil.clearDB(function () { + done(); + }); + }); + + describe('NoteProvider', function () { + + describe('.createNote', function () { + it('should add new note to the Notes collection', function () { + this.slow(100); + + return request(app) + .post('/api/notes') + .set('x-access-token', token) + .send(testNote) + .then(res => { + testNote = res.body; + + expect(res).to.have.status(200); + expect(testNote).to.not.be.null.and.not.to.be.undefined; + expect(testNote).to.have.property('_id'); + + const note = Note.findById(testNote._id); + return note.then(note => expect(parseJSON(note)).to.eql(parseJSON(testNote))); + + }); + }); + }); + + describe('.getNotes', function () { + it('should get all notes from the Notes collection', function () { + this.slow(200); + + return request(app) + .get('/api/notes') + // .set('x-access-token', token) + .then(res => { + let responseNotes = res.body; + + expect(res).to.have.status(200); + expect(responseNotes).to.not.be.null.and.not.to.be.undefined; + expect(responseNotes).to.have.lengthOf(1); + + const notes = Note.find({}); + return notes.then(notes => expect(parseJSON(notes)).to.eql(parseJSON(responseNotes))); + + }) + }); + }); + + describe('.getNote', function () { + it('should get a note from the Notes collection', function () { + this.slow(200); + + return request(app) + .get('/api/notes') + // .set('x-access-token', token) + .then(res => { + let responseNotes = res.body; + + expect(res).to.have.status(200); + expect(responseNotes).to.not.be.null.and.not.to.be.undefined; + expect(responseNotes).to.have.lengthOf(1); + + const notes = Note.find({}); + return notes.then(notes => expect(parseJSON(notes)).to.eql(parseJSON(responseNotes))); + + }) + }); + }); + + }); +}); \ No newline at end of file diff --git a/projects/web-scraper/test/integration/User.js b/projects/web-scraper/test/integration/User.js new file mode 100644 index 0000000..78c647d --- /dev/null +++ b/projects/web-scraper/test/integration/User.js @@ -0,0 +1,142 @@ +const expect = require('chai').expect; +const request = require('chai').request; + +/* + * Server setup +*/ +const app = mutil.getApp(); +const User = mutil.getModel('User'); +const parseJSON = mutil.parseJSON; + +let token; +let ID; +const testUser = { + name: 'user56', + email: 'user@provider.domain', + password: 'secret78' +}; + +function registerUser() { + return request(app) + .post('/api/auth/register') + .send(testUser) + .then(res => { + token = res.body.token; + }); +} + +describe('UserController', function () { + + before(function () { + + return new Promise((resolve, reject) => { + mutil.clearDB(function (err) { + if (err) { + return reject(err); + } + resolve(); + }); + }).then(registerUser); + }); + + after(function (done) { + + mutil.clearDB(function () { + done(); + }); + }); + + describe('UserProvider', function () { + + describe('.createUser', function () { + + it('should add a new User to the user collection', function () { + + this.slow(100); + + return request(app) + .post('/api/users') + .set('x-access-token', token) + .set('Content-Type', 'application/json') + .send(testUser) + .then((res) => { + + const user = res.body; + ID = user._id; + expect(res).to.have.status(200); + expect(user).to.not.be.null.and.not.be.undefined; + expect(user).to.have.property('_id'); + }); + }); + }); + + describe('.getUsers', function () { + + it('should return users from the user collection', function () { + + this.slow(200); + + return request(app) + .get('/api/users') + .then((res) => { + + expect(res).to.have.status(200); + expect(res.body).to.not.be.null.and.not.be.undefined; + }); + }); + }); + + describe('.getUser', function () { + + it('should return a particular user from the user collection', function () { + + this.slow(200); + + return request(app) + .get('/api/users/' +ID) + .then((res) => { + + expect(res).to.have.status(200); + expect(res.body).to.not.be.null.and.not.be.undefined; + }); + }); + }); + + describe('.putUsers', function () { + + it('should update user info in the database', function () { + + this.slow(200); + + return request(app) + .put('/api/users/' +ID) + .set('x-access-token', token) + .set('Content-Type', 'application/json') + .send({ name: 'randomuser' }) + .then((res) => { + + expect(res).to.have.status(200); + expect(res.body).to.not.be.null.and.not.be.undefined; + }); + }); + }); + + describe('.deleteUsers', function () { + + it('should delete a user from the user collection', function () { + + this.slow(200); + + return request(app) + .delete('/api/users/' +ID) + .set('x-access-token', token) + .then((res) => { + + expect(res).to.have.status(200); + expect(res.body).to.not.be.null.and.not.be.undefined; + }); + }); + }); + + }); +}); diff --git a/projects/web-scraper/test/unit/Auth.spec.js b/projects/web-scraper/test/unit/Auth.spec.js new file mode 100644 index 0000000..2016114 --- /dev/null +++ b/projects/web-scraper/test/unit/Auth.spec.js @@ -0,0 +1,18 @@ +const expect = require('chai').expect; +const app = mutil.getApp(); +const AuthController = require(__root + 'auth/AuthController'); + +describe('AuthController', function () { + it('should be a function', function () { + expect(AuthController).to.be.a('function'); + }); + it('should take one parameter', function () { + expect(AuthController.bind(this, app)).to.not.throw(Error); + }); + it('should throw error if app parameter is missing', function () { + expect(AuthController).to.throw(Error); + }); + it('should return an express router function', function () { + expect(AuthController(app)).to.be.a('function'); + }); +}); \ No newline at end of file diff --git a/projects/web-scraper/test/unit/HttpError.spec.js b/projects/web-scraper/test/unit/HttpError.spec.js new file mode 100644 index 0000000..4bbe05d --- /dev/null +++ b/projects/web-scraper/test/unit/HttpError.spec.js @@ -0,0 +1,17 @@ +const expect = require('chai').expect; +const HttpError = require(__root + 'helpers/HttpError'); + +describe('HttpError', function () { + it('should be a function', function () { + expect(HttpError).to.be.a('function'); + }); + it('should take two parameter', function () { + expect(HttpError.bind(this, 200, 'message')).to.not.throw(Error); + }); + it('should throw error if there are not exactly two parameters', function () { + expect(HttpError.bind(this, 'message')).to.throw(Error); + }); + it('should instantiate an Error object when called', function () { + expect(new HttpError(404, 'Not found.')).to.be.an('object'); + }); +}); \ No newline at end of file diff --git a/projects/web-scraper/test/unit/Note.spec.js b/projects/web-scraper/test/unit/Note.spec.js new file mode 100644 index 0000000..2be7fbf --- /dev/null +++ b/projects/web-scraper/test/unit/Note.spec.js @@ -0,0 +1,18 @@ +const expect = require('chai').expect; +const app = mutil.getApp(); +const NoteController = require(__root + 'note/NoteController'); + +describe('NoteController', function () { + it('should be a function', function () { + expect(NoteController).to.be.a('function'); // fail on purpose + }); + it('should take one parameter', function () { + expect(NoteController.bind(this, app)).to.not.throw(Error); + }); + it('should throw error if app parameter is missing', function () { + expect(NoteController).to.throw(Error); + }); + it('should return an express router function', function () { + expect(NoteController(app)).to.be.a('function'); + }); +}); \ No newline at end of file diff --git a/projects/web-scraper/test/unit/User.spec.js b/projects/web-scraper/test/unit/User.spec.js new file mode 100644 index 0000000..034f002 --- /dev/null +++ b/projects/web-scraper/test/unit/User.spec.js @@ -0,0 +1,18 @@ +const expect = require('chai').expect; +const app = mutil.getApp(); +const UserController = require(__root + 'user/UserController'); + +describe('UserController', function () { + it('should be a function', function () { + expect(UserController).to.be.a('function'); + }); + it('should take on parameter', function () { + expect(UserController.bind(this, app)).to.not.throw(Error); + }); + it('should throw error if app parameter is missing', function () { + expect(UserController).to.throw(Error); + }); + it('should return an express router function', function () { + expect(UserController(app)).to.be.a('function'); + }); +}); From d36a9cdc97ed7f594baced81a4254a8990e01f61 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:16:09 -0700 Subject: [PATCH 03/14] Update routing for spider funcs --- projects/web-scraper/lib/routes.js | 4 ++++ projects/web-scraper/lib/spider/spiderController.js | 2 +- 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/projects/web-scraper/lib/routes.js b/projects/web-scraper/lib/routes.js index e717311..a23202c 100644 --- a/projects/web-scraper/lib/routes.js +++ b/projects/web-scraper/lib/routes.js @@ -15,6 +15,10 @@ module.exports = function (app) { const UserController = require('./user/UserController')(app); router.use('/users', UserController); + const SpiderController = require('./user/SpiderController')(app); + router.use('/spider', SpiderController); + + /** * default error handling */ diff --git a/projects/web-scraper/lib/spider/spiderController.js b/projects/web-scraper/lib/spider/spiderController.js index b6b7399..e85fede 100644 --- a/projects/web-scraper/lib/spider/spiderController.js +++ b/projects/web-scraper/lib/spider/spiderController.js @@ -44,7 +44,7 @@ module.exports = function (app) { let filePath = path.resolve(`./output/` + siteName + '/' + `${url.pathname}`); if (path.extname(url.pathname).trim() === '') { - console.log(url.pathname, `${filePath}/index.html`) + console.log(url.pathname, `${filePath}/index.html`) filePath = `${filePath}/index.html`; } From 27e33bfa8c7f0d16dd26a4b9129444521891e1e2 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:24:16 -0700 Subject: [PATCH 04/14] Abstracted worker to PUT endpoint will remove to helpers later. --- .../lib/spider/spiderController.js | 115 --------------- .../web-scraper/lib/spider/spiderProvider.js | 135 +++++++++++++++++- 2 files changed, 130 insertions(+), 120 deletions(-) diff --git a/projects/web-scraper/lib/spider/spiderController.js b/projects/web-scraper/lib/spider/spiderController.js index e85fede..1a7601f 100644 --- a/projects/web-scraper/lib/spider/spiderController.js +++ b/projects/web-scraper/lib/spider/spiderController.js @@ -6,122 +6,7 @@ module.exports = function (app) { var SpiderProvider = require('./SpiderProvider'); var validateSpider = require('./validateSpider'); var VerifyToken = require(__root + 'auth/VerifyToken')(app); - let z = 1; - const puppeteer = require('puppeteer'); // v 1.1.0 - const { URL } = require('url'); - const fse = require('fs-extra'); // v 5.0.0 - const path = require('path'); - const xml = require('xml2js'); - const parser = new xml.Parser(); - - - // Puppeteer worker. - async function start(urlToFetch, siteName) { - /* 1 */ - const browser = await puppeteer.launch(); - const page = await browser.newPage(); - //await page.setRequestInterception(true); - /* 2 */ - page.on('response', async (response) => { - - try { - const status = response.status; - - page.on( "console", function (log) { - console.log(log.text()); - }); - const url = new URL(response.url()); - - // This intercept is a method of blocking potential URL redirects - // console.log(interceptedRequest.url(), url, status); - // if (interceptedRequest.url().endsWith('/api')) { - // interceptedRequest.respond({ - // status: 422, - // body: "FAKE" - // }) - // } else interceptedRequest.continue(); - - let filePath = path.resolve(`./output/` + siteName + '/' + `${url.pathname}`); - if (path.extname(url.pathname).trim() === '') { - console.log(url.pathname, `${filePath}/index.html`) - filePath = `${filePath}/index.html`; - } - - // Save Data - if ((status >= 300) && (status <= 399)) { - console.log('Redirect from', response.url(), 'to', response.headers()['location']); - await new Promise(function(resolve, reject) { - resolve(null); - }); - } else { - await fse.outputFile(filePath, await response.buffer()); - await fse.outputFile(filePath + z, await response.buffer()); - z++; - } - - } catch (err) { - console.log('Redirect from', response.url(), 'to', response.headers()['location'], err, response.status); - } - }); - - /* 3 */ - await page.goto(urlToFetch, { - waitUntil: 'load' - }); - - /* 4 */ - setTimeout(async () => { - await browser.close(); - }, 60000 * 4); - } - - // Headless chrome requires a lot or resources. - function createQueue(tasks, maxNumOfWorkers = 4) { - var numOfWorkers = 0; - var taskIndex = 0; - - return new Promise( done => { - const handleResult = index => result => { - tasks[index] = result; - numOfWorkers--; - getNextTask(); - }; - const getNextTask = () => { - if (numOfWorkers < maxNumOfWorkers && taskIndex < tasks.length) { - tasks[taskIndex]().then(handleResult(taskIndex)).catch(handleResult(taskIndex)); - taskIndex++; - numOfWorkers++; - getNextTask(); - } else if (numOfWorkers === 0 && taskIndex === tasks.length) { - done(tasks); - } - }; - getNextTask(); - }); - } - - let items = []; - - // Read sites generated by web indexer. - let xmlInjest = (siteName, path) => { - fse.readFile(__dirname + path, (err, data) => { - parser.parseString(data, (err, result) => { - result.urlset.url.forEach((item, index) => { - items.push(async () => { - //console.log(item.loc[0]); - start(item.loc[0]); - }); - }); - console.log(JSON.stringify(items[0])); - // Read X items, Y at a time. - createQueue(items.slice(0,2), 1); - }); - }); - }; - - //start('https://www.lamayor.org/', 'www'); - xmlInjest('www-lamayor', '/siteMap.xml'); // CREATES A NEW NOTE SpiderController.post('/', VerifyToken, validateNote, SpiderProvider.createNote); diff --git a/projects/web-scraper/lib/spider/spiderProvider.js b/projects/web-scraper/lib/spider/spiderProvider.js index 05c8f9b..4e52e99 100644 --- a/projects/web-scraper/lib/spider/spiderProvider.js +++ b/projects/web-scraper/lib/spider/spiderProvider.js @@ -1,4 +1,12 @@ -var Spider = require('./Spider'); +const puppeteer = require('puppeteer'); // v 1.1.0 +const { URL } = require('url'); +const fse = require('fs-extra'); // v 5.0.0 +const path = require('path'); +const xml = require('xml2js'); +const parser = new xml.Parser(); +const Spider = require('./Spider'); +let z = 1; + module.exports = { createSpider: createSpider, getSpiders: getSpiders, @@ -7,6 +15,112 @@ module.exports = { putSpider: putSpider }; +// Puppeteer worker. +async function start(urlToFetch, siteName) { + /* 1 */ + const browser = await puppeteer.launch(); + const page = await browser.newPage(); + //await page.setRequestInterception(true); + /* 2 */ + page.on('response', async (response) => { + + try { + const status = response.status; + + page.on( "console", function (log) { + console.log(log.text()); + }); + const url = new URL(response.url()); + + // This intercept is a method of blocking potential URL redirects + // console.log(interceptedRequest.url(), url, status); + // if (interceptedRequest.url().endsWith('/api')) { + // interceptedRequest.respond({ + // status: 422, + // body: "FAKE" + // }) + // } else interceptedRequest.continue(); + + let filePath = path.resolve(`./output/` + siteName + '/' + `${url.pathname}`); + if (path.extname(url.pathname).trim() === '') { + console.log(url.pathname, `${filePath}/index.html`) + filePath = `${filePath}/index.html`; + } + + // Save Data + if ((status >= 300) && (status <= 399)) { + console.log('Redirect from', response.url(), 'to', response.headers()['location']); + await new Promise(function(resolve, reject) { + resolve(null); + }); + } else { + await fse.outputFile(filePath, await response.buffer()); + await fse.outputFile(filePath + z, await response.buffer()); + z++; + } + + } catch (err) { + console.log('Redirect from', response.url(), 'to', response.headers()['location'], err, response.status); + } + }); + + /* 3 */ + await page.goto(urlToFetch, { + waitUntil: 'load' + }); + + /* 4 */ + setTimeout(async () => { + await browser.close(); + }, 60000 * 4); +}; + +// Headless chrome requires a lot or resources. +function createQueue(tasks, maxNumOfWorkers = 4) { + var numOfWorkers = 0; + var taskIndex = 0; + + return new Promise( done => { + const handleResult = index => result => { + tasks[index] = result; + numOfWorkers--; + getNextTask(); + }; + const getNextTask = () => { + if (numOfWorkers < maxNumOfWorkers && taskIndex < tasks.length) { + tasks[taskIndex]().then(handleResult(taskIndex)).catch(handleResult(taskIndex)); + taskIndex++; + numOfWorkers++; + getNextTask(); + } else if (numOfWorkers === 0 && taskIndex === tasks.length) { + done(tasks); + } + }; + getNextTask(); + }); +}; + +// Read sites generated by web indexer. +let xmlInjest = (siteName, path) => { + let items = []; + fse.readFile(__dirname + path, (err, data) => { + parser.parseString(data, (err, result) => { + result.urlset.url.forEach((item, index) => { + items.push(async () => { + //console.log(item.loc[0]); + start(item.loc[0]); + }); + }); + console.log(JSON.stringify(items[0])); + // Read X items, Y at a time. + createQueue(items.slice(0,2), 1); + }); + }); +}; + + + +// Base HTTP REST stubs. We could go for graphQL lets keep REST for now. function createSpider(req, res, next) { return Spider.create({ title : req.body.title, @@ -39,7 +153,18 @@ function deleteSpider(req, res, next) { } function putSpider(req, res, next) { - return Spider.findByIdAndUpdate(req.params.id, req.body, {new: true}) - .then(Spider => res.status(200).send(Spider)) - .catch(err => next(new Error(err))); -} \ No newline at end of file + // in the future we want to use mongo or some other DB to cache requests? + // return Spider.findByIdAndUpdate(req.params.id, req.body, {new: true}) + // .then(Spider => res.status(200).send(Spider)) + // .catch(err => next(new Error(err))); + + //start('https://www.lamayor.org/', 'www'); + return { + 'worker': xmlInjest('www-lamayor', '/siteMap.xml'), + 'input': { + 'params': req.params, + 'body': req.body + } + }; +} + From 162faa56f7c31b15ee6824a3de8df8bf93fc4f83 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:27:48 -0700 Subject: [PATCH 05/14] Adding package.lock will switch to yarn later. --- projects/web-scraper/package-lock.json | 4780 ++++++++++++++++++++++++ projects/web-scraper/package.json | 3 +- 2 files changed, 4781 insertions(+), 2 deletions(-) create mode 100644 projects/web-scraper/package-lock.json diff --git a/projects/web-scraper/package-lock.json b/projects/web-scraper/package-lock.json new file mode 100644 index 0000000..c7213fd --- /dev/null +++ b/projects/web-scraper/package-lock.json @@ -0,0 +1,4780 @@ +{ + "name": "Web-Scraper", + "version": "0.1.0", + "lockfileVersion": 1, + "requires": true, + "dependencies": { + "@types/chai": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/@types/chai/-/chai-4.2.11.tgz", + "integrity": "sha512-t7uW6eFafjO+qJ3BIV2gGUyZs27egcNRkUdalkud+Qa3+kg//f129iuOFivHDXQ+vnU3fDXuwgv0cqMCbcE8sw==", + "dev": true + }, + "@types/cookiejar": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/@types/cookiejar/-/cookiejar-2.1.1.tgz", + "integrity": "sha512-aRnpPa7ysx3aNW60hTiCtLHlQaIFsXFCgQlpakNgDNVFzbtusSY8PwjAQgRWfSk0ekNoBjO51eQRB6upA9uuyw==", + "dev": true + }, + "@types/mime-types": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@types/mime-types/-/mime-types-2.1.0.tgz", + "integrity": "sha1-nKUs2jY/aZxpRmwqbM2q2RPqenM=" + }, + "@types/node": { + "version": "13.9.2", + "resolved": "https://registry.npmjs.org/@types/node/-/node-13.9.2.tgz", + "integrity": "sha512-bnoqK579sAYrQbp73wwglccjJ4sfRdKU7WNEZ5FW4K2U6Kc0/eZ5kvXG0JKsEKFB50zrFmfFt52/cvBbZa7eXg==", + "dev": true + }, + "@types/superagent": { + "version": "3.8.7", + "resolved": "https://registry.npmjs.org/@types/superagent/-/superagent-3.8.7.tgz", + "integrity": "sha512-9KhCkyXv268A2nZ1Wvu7rQWM+BmdYUVkycFeNnYrUL5Zwu7o8wPQ3wBfW59dDP+wuoxw0ww8YKgTNv8j/cgscA==", + "dev": true, + "requires": { + "@types/cookiejar": "2.1.1", + "@types/node": "13.9.2" + } + }, + "abbrev": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.0.9.tgz", + "integrity": "sha1-kbR5JYinc4wl813W9jdSovh3YTU=", + "dev": true + }, + "accepts": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", + "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", + "requires": { + "mime-types": "2.1.26", + "negotiator": "0.6.2" + } + }, + "acorn": { + "version": "5.7.4", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-5.7.4.tgz", + "integrity": "sha512-1D++VG7BhrtvQpNbBzovKNc1FLGGEE/oGe7b9xJm/RFHMBeUaUGpluV9RLjZa47YFdPcDAenEYuq9pQPcMdLJg==", + "dev": true + }, + "acorn-jsx": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-3.0.1.tgz", + "integrity": "sha1-r9+UiPsezvyDSPb7IvRk4ypYs2s=", + "dev": true, + "requires": { + "acorn": "3.3.0" + }, + "dependencies": { + "acorn": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-3.3.0.tgz", + "integrity": "sha1-ReN/s56No/JbruP/U2niu18iAXo=", + "dev": true + } + } + }, + "agent-base": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-5.1.1.tgz", + "integrity": "sha512-TMeqbNl2fMW0nMjTEPOwe3J/PRFP4vqeoNuQMG0HlMrtm5QxKqdvAkZ1pRBQ/ulIyDD5Yq0nJ7YbdD8ey0TO3g==" + }, + "ajv": { + "version": "5.5.2", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-5.5.2.tgz", + "integrity": "sha1-c7Xuyj+rZT49P5Qis0GtQiBdyWU=", + "dev": true, + "requires": { + "co": "4.6.0", + "fast-deep-equal": "1.1.0", + "fast-json-stable-stringify": "2.1.0", + "json-schema-traverse": "0.3.1" + } + }, + "ajv-keywords": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-2.1.1.tgz", + "integrity": "sha1-YXmX/F9gV2iUxDX5QNgZ4TW4B2I=", + "dev": true + }, + "amdefine": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.1.tgz", + "integrity": "sha1-SlKCrBZHKek2Gbz9OtFR+BfOkfU=", + "dev": true, + "optional": true + }, + "ansi-align": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-2.0.0.tgz", + "integrity": "sha1-w2rsy6VjuJzrVW82kPCx2eNUf38=", + "dev": true, + "requires": { + "string-width": "2.1.1" + } + }, + "ansi-escapes": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.2.0.tgz", + "integrity": "sha512-cBhpre4ma+U0T1oM5fXg7Dy1Jw7zzwv7lt/GoCpr+hDQJoYnKVPLL4dCvSEFMmQurOQvSrwT7SL/DAlhBI97RQ==", + "dev": true + }, + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", + "dev": true + }, + "any-promise": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz", + "integrity": "sha1-q8av7tzqUugJzcA3au0845Y10X8=" + }, + "anymatch": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", + "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", + "dev": true, + "requires": { + "micromatch": "3.1.10", + "normalize-path": "2.1.1" + }, + "dependencies": { + "normalize-path": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", + "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", + "dev": true, + "requires": { + "remove-trailing-separator": "1.1.0" + } + } + } + }, + "argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "requires": { + "sprintf-js": "1.0.3" + } + }, + "arr-diff": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", + "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", + "dev": true + }, + "arr-flatten": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", + "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==", + "dev": true + }, + "arr-union": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", + "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=", + "dev": true + }, + "array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" + }, + "array-unique": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", + "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", + "dev": true + }, + "assertion-error": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-1.1.0.tgz", + "integrity": "sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==", + "dev": true + }, + "assign-symbols": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", + "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=", + "dev": true + }, + "async": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", + "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", + "requires": { + "lodash": "4.17.15" + } + }, + "async-each": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.3.tgz", + "integrity": "sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==", + "dev": true + }, + "async-limiter": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz", + "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==" + }, + "asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", + "dev": true + }, + "atob": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", + "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", + "dev": true + }, + "babel-code-frame": { + "version": "6.26.0", + "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", + "integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=", + "dev": true, + "requires": { + "chalk": "1.1.3", + "esutils": "2.0.3", + "js-tokens": "3.0.2" + }, + "dependencies": { + "chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "requires": { + "ansi-styles": "2.2.1", + "escape-string-regexp": "1.0.5", + "has-ansi": "2.0.0", + "strip-ansi": "3.0.1", + "supports-color": "2.0.0" + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "2.1.1" + } + } + } + }, + "balanced-match": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", + "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" + }, + "base": { + "version": "0.11.2", + "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz", + "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", + "dev": true, + "requires": { + "cache-base": "1.0.1", + "class-utils": "0.3.6", + "component-emitter": "1.3.0", + "define-property": "1.0.0", + "isobject": "3.0.1", + "mixin-deep": "1.3.2", + "pascalcase": "0.1.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "1.0.2" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "6.0.3" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "6.0.3" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "1.0.0", + "is-data-descriptor": "1.0.0", + "kind-of": "6.0.3" + } + } + } + }, + "bcryptjs": { + "version": "2.4.3", + "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz", + "integrity": "sha1-mrVie5PmBiH/fNrF2pczAn3x0Ms=" + }, + "bcryptjs-then": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/bcryptjs-then/-/bcryptjs-then-1.0.1.tgz", + "integrity": "sha1-DAOyKlAy3sfiUQn6ps5CXrUBUVg=", + "requires": { + "any-promise": "1.3.0", + "bcryptjs": "2.4.3" + } + }, + "binary-extensions": { + "version": "1.13.1", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", + "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", + "dev": true + }, + "bindings": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", + "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", + "dev": true, + "optional": true, + "requires": { + "file-uri-to-path": "1.0.0" + } + }, + "bl": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/bl/-/bl-2.2.0.tgz", + "integrity": "sha512-wbgvOpqopSr7uq6fJrLH8EsvYMJf9gzfo2jCsL2eTy75qXPukA4pCgHamOQkZtY5vmfVtjB+P3LNlMHW5CEZXA==", + "requires": { + "readable-stream": "2.3.7", + "safe-buffer": "5.1.2" + } + }, + "bluebird": { + "version": "3.5.1", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.5.1.tgz", + "integrity": "sha512-MKiLiV+I1AA596t9w1sQJ8jkiSr5+ZKi0WKrYGUn6d1Fx+Ij4tIj+m2WMQSGczs5jZVxV339chE8iwk6F64wjA==" + }, + "body-parser": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", + "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", + "requires": { + "bytes": "3.1.0", + "content-type": "1.0.4", + "debug": "2.6.9", + "depd": "1.1.2", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "on-finished": "2.3.0", + "qs": "6.7.0", + "raw-body": "2.4.0", + "type-is": "1.6.18" + } + }, + "boxen": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/boxen/-/boxen-1.3.0.tgz", + "integrity": "sha512-TNPjfTr432qx7yOjQyaXm3dSR0MH9vXp7eT1BFSl/C51g+EFnOR9hTg1IreahGBmDNCehscshe45f+C1TBZbLw==", + "dev": true, + "requires": { + "ansi-align": "2.0.0", + "camelcase": "4.1.0", + "chalk": "2.4.2", + "cli-boxes": "1.0.0", + "string-width": "2.1.1", + "term-size": "1.2.0", + "widest-line": "2.0.1" + } + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "requires": { + "balanced-match": "1.0.0", + "concat-map": "0.0.1" + } + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "requires": { + "arr-flatten": "1.1.0", + "array-unique": "0.3.2", + "extend-shallow": "2.0.1", + "fill-range": "4.0.0", + "isobject": "3.0.1", + "repeat-element": "1.1.3", + "snapdragon": "0.8.2", + "snapdragon-node": "2.1.1", + "split-string": "3.1.0", + "to-regex": "3.0.2" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "0.1.1" + } + } + } + }, + "browser-stdout": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz", + "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", + "dev": true + }, + "bson": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/bson/-/bson-1.1.3.tgz", + "integrity": "sha512-TdiJxMVnodVS7r0BdL42y/pqC9cL2iKynVwA0Ho3qbsQYr428veL3l7BQyuqiw+Q5SqqoT0m4srSY/BlZ9AxXg==" + }, + "buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" + }, + "buffer-from": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", + "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==" + }, + "bytes": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", + "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" + }, + "cache-base": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz", + "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==", + "dev": true, + "requires": { + "collection-visit": "1.0.0", + "component-emitter": "1.3.0", + "get-value": "2.0.6", + "has-value": "1.0.0", + "isobject": "3.0.1", + "set-value": "2.0.1", + "to-object-path": "0.3.0", + "union-value": "1.0.1", + "unset-value": "1.0.0" + } + }, + "caller-path": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-0.1.0.tgz", + "integrity": "sha1-lAhe9jWB7NPaqSREqP6U6CV3dR8=", + "dev": true, + "requires": { + "callsites": "0.2.0" + } + }, + "callsites": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-0.2.0.tgz", + "integrity": "sha1-r6uWJikQp/M8GaV3WCXGnzTjUMo=", + "dev": true + }, + "camelcase": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-4.1.0.tgz", + "integrity": "sha1-1UVjW+HjPFQmScaRc+Xeas+uNN0=", + "dev": true + }, + "capture-stack-trace": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/capture-stack-trace/-/capture-stack-trace-1.0.1.tgz", + "integrity": "sha512-mYQLZnx5Qt1JgB1WEiMCf2647plpGeQ2NMR/5L0HNZzGQo4fuSPnK+wjfPnKZV0aiJDgzmWqqkV/g7JD+DW0qw==", + "dev": true + }, + "chai": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/chai/-/chai-4.2.0.tgz", + "integrity": "sha512-XQU3bhBukrOsQCuwZndwGcCVQHyZi53fQ6Ys1Fym7E4olpIqqZZhhoFJoaKVvV17lWQoXYwgWN2nF5crA8J2jw==", + "dev": true, + "requires": { + "assertion-error": "1.1.0", + "check-error": "1.0.2", + "deep-eql": "3.0.1", + "get-func-name": "2.0.0", + "pathval": "1.1.0", + "type-detect": "4.0.8" + } + }, + "chai-as-promised": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/chai-as-promised/-/chai-as-promised-7.1.1.tgz", + "integrity": "sha512-azL6xMoi+uxu6z4rhWQ1jbdUhOMhis2PvscD/xjLqNMkv3BPPp2JyyuTHOrf9BOosGpNQ11v6BKv/g57RXbiaA==", + "dev": true, + "requires": { + "check-error": "1.0.2" + } + }, + "chai-http": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/chai-http/-/chai-http-4.3.0.tgz", + "integrity": "sha512-zFTxlN7HLMv+7+SPXZdkd5wUlK+KxH6Q7bIEMiEx0FK3zuuMqL7cwICAQ0V1+yYRozBburYuxN1qZstgHpFZQg==", + "dev": true, + "requires": { + "@types/chai": "4.2.11", + "@types/superagent": "3.8.7", + "cookiejar": "2.1.2", + "is-ip": "2.0.0", + "methods": "1.1.2", + "qs": "6.7.0", + "superagent": "3.8.3" + } + }, + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "requires": { + "ansi-styles": "3.2.1", + "escape-string-regexp": "1.0.5", + "supports-color": "5.5.0" + }, + "dependencies": { + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "1.9.3" + } + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "requires": { + "has-flag": "3.0.0" + } + } + } + }, + "chardet": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.4.2.tgz", + "integrity": "sha1-tUc7M9yXxCTl2Y3IfVXU2KKci/I=", + "dev": true + }, + "check-error": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/check-error/-/check-error-1.0.2.tgz", + "integrity": "sha1-V00xLt2Iu13YkS6Sht1sCu1KrII=", + "dev": true + }, + "chokidar": { + "version": "2.1.8", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", + "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", + "dev": true, + "requires": { + "anymatch": "2.0.0", + "async-each": "1.0.3", + "braces": "2.3.2", + "fsevents": "1.2.12", + "glob-parent": "3.1.0", + "inherits": "2.0.3", + "is-binary-path": "1.0.1", + "is-glob": "4.0.1", + "normalize-path": "3.0.0", + "path-is-absolute": "1.0.1", + "readdirp": "2.2.1", + "upath": "1.2.0" + } + }, + "ci-info": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-1.6.0.tgz", + "integrity": "sha512-vsGdkwSCDpWmP80ncATX7iea5DWQemg1UgCW5J8tqjU3lYw4FBYuj89J0CTVomA7BEfvSZd84GmHko+MxFQU2A==", + "dev": true + }, + "circular-json": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/circular-json/-/circular-json-0.3.3.tgz", + "integrity": "sha512-UZK3NBx2Mca+b5LsG7bY183pHWt5Y1xts4P3Pz7ENTwGVnJOUWbRb3ocjvX7hx9tq/yTAdclXm9sZ38gNuem4A==", + "dev": true + }, + "class-utils": { + "version": "0.3.6", + "resolved": "https://registry.npmjs.org/class-utils/-/class-utils-0.3.6.tgz", + "integrity": "sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==", + "dev": true, + "requires": { + "arr-union": "3.1.0", + "define-property": "0.2.5", + "isobject": "3.0.1", + "static-extend": "0.1.2" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "0.1.6" + } + } + } + }, + "cli-boxes": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-1.0.0.tgz", + "integrity": "sha1-T6kXw+WclKAEzWH47lCdplFocUM=", + "dev": true + }, + "cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", + "dev": true, + "requires": { + "restore-cursor": "2.0.0" + } + }, + "cli-width": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.0.tgz", + "integrity": "sha1-/xnt6Kml5XkyQUewwR8PvLq+1jk=", + "dev": true + }, + "co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=", + "dev": true + }, + "collection-visit": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz", + "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=", + "dev": true, + "requires": { + "map-visit": "1.0.0", + "object-visit": "1.0.1" + } + }, + "color": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/color/-/color-3.0.0.tgz", + "integrity": "sha512-jCpd5+s0s0t7p3pHQKpnJ0TpQKKdleP71LWcA0aqiljpiuAkOSUFN/dyH8ZwF0hRmFlrIuRhufds1QyEP9EB+w==", + "requires": { + "color-convert": "1.9.3", + "color-string": "1.5.3" + } + }, + "color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "requires": { + "color-name": "1.1.3" + } + }, + "color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" + }, + "color-string": { + "version": "1.5.3", + "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.5.3.tgz", + "integrity": "sha512-dC2C5qeWoYkxki5UAXapdjqO672AM4vZuPGRQfO8b5HKuKGBbKWpITyDYN7TOFKvRW7kOgAn3746clDBMDJyQw==", + "requires": { + "color-name": "1.1.3", + "simple-swizzle": "0.2.2" + } + }, + "colornames": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/colornames/-/colornames-1.1.1.tgz", + "integrity": "sha1-+IiQMGhcfE/54qVZ9Qd+t2qBb5Y=" + }, + "colors": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz", + "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==" + }, + "colorspace": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/colorspace/-/colorspace-1.1.2.tgz", + "integrity": "sha512-vt+OoIP2d76xLhjwbBaucYlNSpPsrJWPlBTtwCpQKIu6/CSMutyzX93O/Do0qzpH3YoHEes8YEFXyZ797rEhzQ==", + "requires": { + "color": "3.0.0", + "text-hex": "1.0.0" + } + }, + "combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dev": true, + "requires": { + "delayed-stream": "1.0.0" + } + }, + "commander": { + "version": "2.20.3", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", + "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", + "dev": true, + "optional": true + }, + "component-emitter": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", + "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==", + "dev": true + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" + }, + "concat-stream": { + "version": "1.6.2", + "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", + "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", + "requires": { + "buffer-from": "1.1.1", + "inherits": "2.0.3", + "readable-stream": "2.3.7", + "typedarray": "0.0.6" + } + }, + "configstore": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/configstore/-/configstore-3.1.2.tgz", + "integrity": "sha512-vtv5HtGjcYUgFrXc6Kx747B83MRRVS5R1VTEQoXvuP+kMI+if6uywV0nDGoiydJRy4yk7h9od5Og0kxx4zUXmw==", + "dev": true, + "requires": { + "dot-prop": "4.2.0", + "graceful-fs": "4.2.3", + "make-dir": "1.3.0", + "unique-string": "1.0.0", + "write-file-atomic": "2.4.3", + "xdg-basedir": "3.0.0" + } + }, + "content-disposition": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", + "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", + "requires": { + "safe-buffer": "5.1.2" + } + }, + "content-type": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", + "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" + }, + "cookie": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", + "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" + }, + "cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" + }, + "cookiejar": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/cookiejar/-/cookiejar-2.1.2.tgz", + "integrity": "sha512-Mw+adcfzPxcPeI+0WlvRrr/3lGVO0bD75SxX6811cxSh1Wbxx7xZBGK1eVtDf6si8rg2lhnUjsVLMFMfbRIuwA==", + "dev": true + }, + "copy-descriptor": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", + "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", + "dev": true + }, + "core-util-is": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", + "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" + }, + "create-error-class": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/create-error-class/-/create-error-class-3.0.2.tgz", + "integrity": "sha1-Br56vvlHo/FKMP1hBnHUAbyot7Y=", + "dev": true, + "requires": { + "capture-stack-trace": "1.0.1" + } + }, + "cross-spawn": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz", + "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=", + "dev": true, + "requires": { + "lru-cache": "4.1.5", + "shebang-command": "1.2.0", + "which": "1.3.1" + } + }, + "crypto-random-string": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-1.0.0.tgz", + "integrity": "sha1-ojD2T1aDEOFJgAmUB5DsmVRbyn4=", + "dev": true + }, + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + } + }, + "decode-uri-component": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", + "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", + "dev": true + }, + "deep-eql": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/deep-eql/-/deep-eql-3.0.1.tgz", + "integrity": "sha512-+QeIQyN5ZuO+3Uk5DYh6/1eKO0m0YmJFGNmFHGACpf1ClL1nmlV/p4gNgbl2pJGxgXb4faqo6UE+M5ACEMyVcw==", + "dev": true, + "requires": { + "type-detect": "4.0.8" + } + }, + "deep-extend": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", + "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", + "dev": true + }, + "deep-is": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", + "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", + "dev": true + }, + "define-property": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-2.0.2.tgz", + "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", + "dev": true, + "requires": { + "is-descriptor": "1.0.2", + "isobject": "3.0.1" + }, + "dependencies": { + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "6.0.3" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "6.0.3" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "1.0.0", + "is-data-descriptor": "1.0.0", + "kind-of": "6.0.3" + } + } + } + }, + "delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", + "dev": true + }, + "denque": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/denque/-/denque-1.4.1.tgz", + "integrity": "sha512-OfzPuSZKGcgr96rf1oODnfjqBFmr1DVoc/TrItj3Ohe0Ah1C5WX5Baquw/9U9KovnQ88EqmJbD66rKYUQYN1tQ==" + }, + "depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" + }, + "destroy": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", + "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" + }, + "diagnostics": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/diagnostics/-/diagnostics-1.1.1.tgz", + "integrity": "sha512-8wn1PmdunLJ9Tqbx+Fx/ZEuHfJf4NKSN2ZBj7SJC/OWRWha843+WsTjqMe1B5E3p28jqBlp+mJ2fPVxPyNgYKQ==", + "requires": { + "colorspace": "1.1.2", + "enabled": "1.0.2", + "kuler": "1.0.1" + } + }, + "diff": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", + "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", + "dev": true + }, + "doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dev": true, + "requires": { + "esutils": "2.0.3" + } + }, + "dot-prop": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-4.2.0.tgz", + "integrity": "sha512-tUMXrxlExSW6U2EXiiKGSBVdYgtV8qlHL+C10TsW4PURY/ic+eaysnSkwB4kA/mBlCyy/IKDJ+Lc3wbWeaXtuQ==", + "dev": true, + "requires": { + "is-obj": "1.0.1" + } + }, + "dotenv": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-4.0.0.tgz", + "integrity": "sha1-hk7xN5rO1Vzm+V3r7NzhefegzR0=" + }, + "duplexer3": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", + "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", + "dev": true + }, + "ecdsa-sig-formatter": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", + "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", + "requires": { + "safe-buffer": "5.1.2" + } + }, + "ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" + }, + "enabled": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/enabled/-/enabled-1.0.2.tgz", + "integrity": "sha1-ll9lE9LC0cX0ZStkouM5ZGf8L5M=", + "requires": { + "env-variable": "0.0.6" + } + }, + "encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" + }, + "env-variable": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/env-variable/-/env-variable-0.0.6.tgz", + "integrity": "sha512-bHz59NlBbtS0NhftmR8+ExBEekE7br0e01jw+kk0NDro7TtZzBYZ5ScGPs3OmwnpyfHTHOtr1Y6uedCdrIldtg==" + }, + "escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" + }, + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", + "dev": true + }, + "escodegen": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.8.1.tgz", + "integrity": "sha1-WltTr0aTEQvrsIZ6o0MN07cKEBg=", + "dev": true, + "requires": { + "esprima": "2.7.3", + "estraverse": "1.9.3", + "esutils": "2.0.3", + "optionator": "0.8.3", + "source-map": "0.2.0" + }, + "dependencies": { + "esprima": { + "version": "2.7.3", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-2.7.3.tgz", + "integrity": "sha1-luO3DVd59q1JzQMmc9HDEnZ7pYE=", + "dev": true + }, + "estraverse": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-1.9.3.tgz", + "integrity": "sha1-r2fy3JIlgkFZUJJgkaQAXSnJu0Q=", + "dev": true + } + } + }, + "eslint": { + "version": "4.19.1", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-4.19.1.tgz", + "integrity": "sha512-bT3/1x1EbZB7phzYu7vCr1v3ONuzDtX8WjuM9c0iYxe+cq+pwcKEoQjl7zd3RpC6YOLgnSy3cTN58M2jcoPDIQ==", + "dev": true, + "requires": { + "ajv": "5.5.2", + "babel-code-frame": "6.26.0", + "chalk": "2.4.2", + "concat-stream": "1.6.2", + "cross-spawn": "5.1.0", + "debug": "3.2.6", + "doctrine": "2.1.0", + "eslint-scope": "3.7.3", + "eslint-visitor-keys": "1.1.0", + "espree": "3.5.4", + "esquery": "1.1.0", + "esutils": "2.0.3", + "file-entry-cache": "2.0.0", + "functional-red-black-tree": "1.0.1", + "glob": "7.1.6", + "globals": "11.12.0", + "ignore": "3.3.10", + "imurmurhash": "0.1.4", + "inquirer": "3.3.0", + "is-resolvable": "1.1.0", + "js-yaml": "3.13.1", + "json-stable-stringify-without-jsonify": "1.0.1", + "levn": "0.3.0", + "lodash": "4.17.15", + "minimatch": "3.0.4", + "mkdirp": "0.5.1", + "natural-compare": "1.4.0", + "optionator": "0.8.3", + "path-is-inside": "1.0.2", + "pluralize": "7.0.0", + "progress": "2.0.3", + "regexpp": "1.1.0", + "require-uncached": "1.0.3", + "semver": "5.7.1", + "strip-ansi": "4.0.0", + "strip-json-comments": "2.0.1", + "table": "4.0.2", + "text-table": "0.2.0" + }, + "dependencies": { + "debug": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", + "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + } + } + }, + "eslint-plugin-node": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-node/-/eslint-plugin-node-5.2.1.tgz", + "integrity": "sha512-xhPXrh0Vl/b7870uEbaumb2Q+LxaEcOQ3kS1jtIXanBAwpMre1l5q/l2l/hESYJGEFKuI78bp6Uw50hlpr7B+g==", + "dev": true, + "requires": { + "ignore": "3.3.10", + "minimatch": "3.0.4", + "resolve": "1.15.1", + "semver": "5.3.0" + }, + "dependencies": { + "semver": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.3.0.tgz", + "integrity": "sha1-myzl094C0XxgEq0yaqa00M9U+U8=", + "dev": true + } + } + }, + "eslint-scope": { + "version": "3.7.3", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-3.7.3.tgz", + "integrity": "sha512-W+B0SvF4gamyCTmUc+uITPY0989iXVfKvhwtmJocTaYoc/3khEHmEmvfY/Gn9HA9VV75jrQECsHizkNw1b68FA==", + "dev": true, + "requires": { + "esrecurse": "4.2.1", + "estraverse": "4.3.0" + } + }, + "eslint-visitor-keys": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.1.0.tgz", + "integrity": "sha512-8y9YjtM1JBJU/A9Kc+SbaOV4y29sSWckBwMHa+FGtVj5gN/sbnKDf6xJUl+8g7FAij9LVaP8C24DUiH/f/2Z9A==", + "dev": true + }, + "espree": { + "version": "3.5.4", + "resolved": "https://registry.npmjs.org/espree/-/espree-3.5.4.tgz", + "integrity": "sha512-yAcIQxtmMiB/jL32dzEp2enBeidsB7xWPLNiw3IIkpVds1P+h7qF9YwJq1yUNzp2OKXgAprs4F61ih66UsoD1A==", + "dev": true, + "requires": { + "acorn": "5.7.4", + "acorn-jsx": "3.0.1" + } + }, + "esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true + }, + "esquery": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.1.0.tgz", + "integrity": "sha512-MxYW9xKmROWF672KqjO75sszsA8Mxhw06YFeS5VHlB98KDHbOSurm3ArsjO60Eaf3QmGMCP1yn+0JQkNLo/97Q==", + "dev": true, + "requires": { + "estraverse": "4.3.0" + } + }, + "esrecurse": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.2.1.tgz", + "integrity": "sha512-64RBB++fIOAXPw3P9cy89qfMlvZEXZkqqJkjqqXIvzP5ezRZjW+lPWjw35UX/3EhUPFYbg5ER4JYgDw4007/DQ==", + "dev": true, + "requires": { + "estraverse": "4.3.0" + } + }, + "estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true + }, + "esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true + }, + "etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" + }, + "execa": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-0.7.0.tgz", + "integrity": "sha1-lEvs00zEHuMqY6n68nrVpl/Fl3c=", + "dev": true, + "requires": { + "cross-spawn": "5.1.0", + "get-stream": "3.0.0", + "is-stream": "1.1.0", + "npm-run-path": "2.0.2", + "p-finally": "1.0.0", + "signal-exit": "3.0.2", + "strip-eof": "1.0.0" + } + }, + "expand-brackets": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", + "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", + "dev": true, + "requires": { + "debug": "2.6.9", + "define-property": "0.2.5", + "extend-shallow": "2.0.1", + "posix-character-classes": "0.1.1", + "regex-not": "1.0.2", + "snapdragon": "0.8.2", + "to-regex": "3.0.2" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "0.1.6" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "0.1.1" + } + } + } + }, + "express": { + "version": "4.17.1", + "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", + "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", + "requires": { + "accepts": "1.3.7", + "array-flatten": "1.1.1", + "body-parser": "1.19.0", + "content-disposition": "0.5.3", + "content-type": "1.0.4", + "cookie": "0.4.0", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "1.1.2", + "encodeurl": "1.0.2", + "escape-html": "1.0.3", + "etag": "1.8.1", + "finalhandler": "1.1.2", + "fresh": "0.5.2", + "merge-descriptors": "1.0.1", + "methods": "1.1.2", + "on-finished": "2.3.0", + "parseurl": "1.3.3", + "path-to-regexp": "0.1.7", + "proxy-addr": "2.0.6", + "qs": "6.7.0", + "range-parser": "1.2.1", + "safe-buffer": "5.1.2", + "send": "0.17.1", + "serve-static": "1.14.1", + "setprototypeof": "1.1.1", + "statuses": "1.5.0", + "type-is": "1.6.18", + "utils-merge": "1.0.1", + "vary": "1.1.2" + } + }, + "extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", + "dev": true + }, + "extend-shallow": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", + "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", + "dev": true, + "requires": { + "assign-symbols": "1.0.0", + "is-extendable": "1.0.1" + }, + "dependencies": { + "is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, + "requires": { + "is-plain-object": "2.0.4" + } + } + } + }, + "external-editor": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-2.2.0.tgz", + "integrity": "sha512-bSn6gvGxKt+b7+6TKEv1ZycHleA7aHhRHyAqJyp5pbUFuYYNIzpZnQDk7AsYckyWdEnTeAnay0aCy2aV6iTk9A==", + "dev": true, + "requires": { + "chardet": "0.4.2", + "iconv-lite": "0.4.24", + "tmp": "0.0.33" + } + }, + "extglob": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", + "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", + "dev": true, + "requires": { + "array-unique": "0.3.2", + "define-property": "1.0.0", + "expand-brackets": "2.1.4", + "extend-shallow": "2.0.1", + "fragment-cache": "0.2.1", + "regex-not": "1.0.2", + "snapdragon": "0.8.2", + "to-regex": "3.0.2" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "1.0.2" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "0.1.1" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "6.0.3" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "6.0.3" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "1.0.0", + "is-data-descriptor": "1.0.0", + "kind-of": "6.0.3" + } + } + } + }, + "extract-zip": { + "version": "1.6.7", + "resolved": "https://registry.npmjs.org/extract-zip/-/extract-zip-1.6.7.tgz", + "integrity": "sha1-qEC0uK9kAyZMjbV/Txp0Mz74H+k=", + "requires": { + "concat-stream": "1.6.2", + "debug": "2.6.9", + "mkdirp": "0.5.1", + "yauzl": "2.4.1" + } + }, + "fast-deep-equal": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz", + "integrity": "sha1-wFNHeBfIa1HaqFPIHgWbcz0CNhQ=", + "dev": true + }, + "fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", + "dev": true + }, + "fast-safe-stringify": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.0.7.tgz", + "integrity": "sha512-Utm6CdzT+6xsDk2m8S6uL8VHxNwI6Jub+e9NYTcAms28T84pTa25GJQV9j0CY0N1rM8hK4x6grpF2BQf+2qwVA==" + }, + "fd-slicer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/fd-slicer/-/fd-slicer-1.0.1.tgz", + "integrity": "sha1-i1vL2ewyfFBBv5qwI/1nUPEXfmU=", + "requires": { + "pend": "1.2.0" + } + }, + "fecha": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fecha/-/fecha-2.3.3.tgz", + "integrity": "sha512-lUGBnIamTAwk4znq5BcqsDaxSmZ9nDVJaij6NvRt/Tg4R69gERA+otPKbS86ROw9nxVMw2/mp1fnaiWqbs6Sdg==" + }, + "figures": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", + "dev": true, + "requires": { + "escape-string-regexp": "1.0.5" + } + }, + "file-entry-cache": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-2.0.0.tgz", + "integrity": "sha1-w5KZDD5oR4PYOLjISkXYoEhFg2E=", + "dev": true, + "requires": { + "flat-cache": "1.3.4", + "object-assign": "4.1.1" + } + }, + "file-uri-to-path": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", + "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", + "dev": true, + "optional": true + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "requires": { + "extend-shallow": "2.0.1", + "is-number": "3.0.0", + "repeat-string": "1.6.1", + "to-regex-range": "2.1.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "0.1.1" + } + } + } + }, + "finalhandler": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", + "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", + "requires": { + "debug": "2.6.9", + "encodeurl": "1.0.2", + "escape-html": "1.0.3", + "on-finished": "2.3.0", + "parseurl": "1.3.3", + "statuses": "1.5.0", + "unpipe": "1.0.0" + } + }, + "flat-cache": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-1.3.4.tgz", + "integrity": "sha512-VwyB3Lkgacfik2vhqR4uv2rvebqmDvFu4jlN/C1RzWoJEo8I7z4Q404oiqYCkq41mni8EzQnm95emU9seckwtg==", + "dev": true, + "requires": { + "circular-json": "0.3.3", + "graceful-fs": "4.2.3", + "rimraf": "2.6.3", + "write": "0.2.1" + }, + "dependencies": { + "rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "dev": true, + "requires": { + "glob": "7.1.6" + } + } + } + }, + "for-in": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", + "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", + "dev": true + }, + "form-data": { + "version": "2.5.1", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.5.1.tgz", + "integrity": "sha512-m21N3WOmEEURgk6B9GLOE4RuWOFf28Lhh9qGYeNlGq4VDXUlJy2th2slBNU8Gp8EzloYZOibZJ7t5ecIrFSjVA==", + "dev": true, + "requires": { + "asynckit": "0.4.0", + "combined-stream": "1.0.8", + "mime-types": "2.1.26" + } + }, + "formidable": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/formidable/-/formidable-1.2.2.tgz", + "integrity": "sha512-V8gLm+41I/8kguQ4/o1D3RIHRmhYFG4pnNyonvua+40rqcEmT4+V71yaZ3B457xbbgCsCfjSPi65u/W6vK1U5Q==", + "dev": true + }, + "forwarded": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", + "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" + }, + "fragment-cache": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", + "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=", + "dev": true, + "requires": { + "map-cache": "0.2.2" + } + }, + "fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" + }, + "fs": { + "version": "0.0.1-security", + "resolved": "https://registry.npmjs.org/fs/-/fs-0.0.1-security.tgz", + "integrity": "sha1-invTcYa23d84E/I4WLV+yq9eQdQ=" + }, + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "requires": { + "graceful-fs": "4.2.3", + "jsonfile": "4.0.0", + "universalify": "0.1.2" + } + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" + }, + "fsevents": { + "version": "1.2.12", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.12.tgz", + "integrity": "sha512-Ggd/Ktt7E7I8pxZRbGIs7vwqAPscSESMrCSkx2FtWeqmheJgCo2R74fTsZFCifr0VTPwqRpPv17+6b8Zp7th0Q==", + "dev": true, + "optional": true, + "requires": { + "bindings": "1.5.0", + "nan": "2.14.0", + "node-pre-gyp": "0.14.0" + }, + "dependencies": { + "abbrev": { + "version": "1.1.1", + "bundled": true, + "dev": true, + "optional": true + }, + "ansi-regex": { + "version": "2.1.1", + "bundled": true, + "dev": true + }, + "aproba": { + "version": "1.2.0", + "bundled": true, + "dev": true, + "optional": true + }, + "are-we-there-yet": { + "version": "1.1.5", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "delegates": "1.0.0", + "readable-stream": "2.3.7" + } + }, + "balanced-match": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "brace-expansion": { + "version": "1.1.11", + "bundled": true, + "dev": true, + "requires": { + "balanced-match": "1.0.0", + "concat-map": "0.0.1" + } + }, + "chownr": { + "version": "1.1.4", + "bundled": true, + "dev": true, + "optional": true + }, + "code-point-at": { + "version": "1.1.0", + "bundled": true, + "dev": true + }, + "concat-map": { + "version": "0.0.1", + "bundled": true, + "dev": true + }, + "console-control-strings": { + "version": "1.1.0", + "bundled": true, + "dev": true + }, + "core-util-is": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "optional": true + }, + "debug": { + "version": "3.2.6", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "ms": "2.1.2" + } + }, + "deep-extend": { + "version": "0.6.0", + "bundled": true, + "dev": true, + "optional": true + }, + "delegates": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "optional": true + }, + "detect-libc": { + "version": "1.0.3", + "bundled": true, + "dev": true, + "optional": true + }, + "fs-minipass": { + "version": "1.2.7", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "minipass": "2.9.0" + } + }, + "fs.realpath": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "optional": true + }, + "gauge": { + "version": "2.7.4", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "aproba": "1.2.0", + "console-control-strings": "1.1.0", + "has-unicode": "2.0.1", + "object-assign": "4.1.1", + "signal-exit": "3.0.2", + "string-width": "1.0.2", + "strip-ansi": "3.0.1", + "wide-align": "1.1.3" + } + }, + "glob": { + "version": "7.1.6", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "fs.realpath": "1.0.0", + "inflight": "1.0.6", + "inherits": "2.0.4", + "minimatch": "3.0.4", + "once": "1.4.0", + "path-is-absolute": "1.0.1" + } + }, + "has-unicode": { + "version": "2.0.1", + "bundled": true, + "dev": true, + "optional": true + }, + "iconv-lite": { + "version": "0.4.24", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "safer-buffer": "2.1.2" + } + }, + "ignore-walk": { + "version": "3.0.3", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "minimatch": "3.0.4" + } + }, + "inflight": { + "version": "1.0.6", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "once": "1.4.0", + "wrappy": "1.0.2" + } + }, + "inherits": { + "version": "2.0.4", + "bundled": true, + "dev": true + }, + "ini": { + "version": "1.3.5", + "bundled": true, + "dev": true, + "optional": true + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "requires": { + "number-is-nan": "1.0.1" + } + }, + "isarray": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "optional": true + }, + "minimatch": { + "version": "3.0.4", + "bundled": true, + "dev": true, + "requires": { + "brace-expansion": "1.1.11" + } + }, + "minimist": { + "version": "1.2.5", + "bundled": true, + "dev": true + }, + "minipass": { + "version": "2.9.0", + "bundled": true, + "dev": true, + "requires": { + "safe-buffer": "5.1.2", + "yallist": "3.1.1" + } + }, + "minizlib": { + "version": "1.3.3", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "minipass": "2.9.0" + } + }, + "mkdirp": { + "version": "0.5.3", + "bundled": true, + "dev": true, + "requires": { + "minimist": "1.2.5" + } + }, + "ms": { + "version": "2.1.2", + "bundled": true, + "dev": true, + "optional": true + }, + "needle": { + "version": "2.3.3", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "debug": "3.2.6", + "iconv-lite": "0.4.24", + "sax": "1.2.4" + } + }, + "node-pre-gyp": { + "version": "0.14.0", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "detect-libc": "1.0.3", + "mkdirp": "0.5.3", + "needle": "2.3.3", + "nopt": "4.0.3", + "npm-packlist": "1.4.8", + "npmlog": "4.1.2", + "rc": "1.2.8", + "rimraf": "2.7.1", + "semver": "5.7.1", + "tar": "4.4.13" + } + }, + "nopt": { + "version": "4.0.3", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "abbrev": "1.1.1", + "osenv": "0.1.5" + } + }, + "npm-bundled": { + "version": "1.1.1", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "npm-normalize-package-bin": "1.0.1" + } + }, + "npm-normalize-package-bin": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "npm-packlist": { + "version": "1.4.8", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "ignore-walk": "3.0.3", + "npm-bundled": "1.1.1", + "npm-normalize-package-bin": "1.0.1" + } + }, + "npmlog": { + "version": "4.1.2", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "are-we-there-yet": "1.1.5", + "console-control-strings": "1.1.0", + "gauge": "2.7.4", + "set-blocking": "2.0.0" + } + }, + "number-is-nan": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "object-assign": { + "version": "4.1.1", + "bundled": true, + "dev": true, + "optional": true + }, + "once": { + "version": "1.4.0", + "bundled": true, + "dev": true, + "requires": { + "wrappy": "1.0.2" + } + }, + "os-homedir": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "optional": true + }, + "os-tmpdir": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "optional": true + }, + "osenv": { + "version": "0.1.5", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "os-homedir": "1.0.2", + "os-tmpdir": "1.0.2" + } + }, + "path-is-absolute": { + "version": "1.0.1", + "bundled": true, + "dev": true, + "optional": true + }, + "process-nextick-args": { + "version": "2.0.1", + "bundled": true, + "dev": true, + "optional": true + }, + "rc": { + "version": "1.2.8", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "deep-extend": "0.6.0", + "ini": "1.3.5", + "minimist": "1.2.5", + "strip-json-comments": "2.0.1" + } + }, + "readable-stream": { + "version": "2.3.7", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "core-util-is": "1.0.2", + "inherits": "2.0.4", + "isarray": "1.0.0", + "process-nextick-args": "2.0.1", + "safe-buffer": "5.1.2", + "string_decoder": "1.1.1", + "util-deprecate": "1.0.2" + } + }, + "rimraf": { + "version": "2.7.1", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "glob": "7.1.6" + } + }, + "safe-buffer": { + "version": "5.1.2", + "bundled": true, + "dev": true + }, + "safer-buffer": { + "version": "2.1.2", + "bundled": true, + "dev": true, + "optional": true + }, + "sax": { + "version": "1.2.4", + "bundled": true, + "dev": true, + "optional": true + }, + "semver": { + "version": "5.7.1", + "bundled": true, + "dev": true, + "optional": true + }, + "set-blocking": { + "version": "2.0.0", + "bundled": true, + "dev": true, + "optional": true + }, + "signal-exit": { + "version": "3.0.2", + "bundled": true, + "dev": true, + "optional": true + }, + "string-width": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "requires": { + "code-point-at": "1.1.0", + "is-fullwidth-code-point": "1.0.0", + "strip-ansi": "3.0.1" + } + }, + "string_decoder": { + "version": "1.1.1", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "safe-buffer": "5.1.2" + } + }, + "strip-ansi": { + "version": "3.0.1", + "bundled": true, + "dev": true, + "requires": { + "ansi-regex": "2.1.1" + } + }, + "strip-json-comments": { + "version": "2.0.1", + "bundled": true, + "dev": true, + "optional": true + }, + "tar": { + "version": "4.4.13", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "chownr": "1.1.4", + "fs-minipass": "1.2.7", + "minipass": "2.9.0", + "minizlib": "1.3.3", + "mkdirp": "0.5.3", + "safe-buffer": "5.1.2", + "yallist": "3.1.1" + } + }, + "util-deprecate": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "optional": true + }, + "wide-align": { + "version": "1.1.3", + "bundled": true, + "dev": true, + "optional": true, + "requires": { + "string-width": "1.0.2" + } + }, + "wrappy": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "yallist": { + "version": "3.1.1", + "bundled": true, + "dev": true + } + } + }, + "functional-red-black-tree": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", + "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", + "dev": true + }, + "get-func-name": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/get-func-name/-/get-func-name-2.0.0.tgz", + "integrity": "sha1-6td0q+5y4gQJQzoGY2YCPdaIekE=", + "dev": true + }, + "get-stream": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz", + "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=", + "dev": true + }, + "get-value": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", + "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=", + "dev": true + }, + "glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "requires": { + "fs.realpath": "1.0.0", + "inflight": "1.0.6", + "inherits": "2.0.3", + "minimatch": "3.0.4", + "once": "1.4.0", + "path-is-absolute": "1.0.1" + } + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "3.1.0", + "path-dirname": "1.0.2" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "2.1.1" + } + } + } + }, + "global-dirs": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-0.1.1.tgz", + "integrity": "sha1-sxnA3UYH81PzvpzKTHL8FIxJ9EU=", + "dev": true, + "requires": { + "ini": "1.3.5" + } + }, + "globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "dev": true + }, + "got": { + "version": "6.7.1", + "resolved": "https://registry.npmjs.org/got/-/got-6.7.1.tgz", + "integrity": "sha1-JAzQV4WpoY5WHcG0S0HHY+8ejbA=", + "dev": true, + "requires": { + "create-error-class": "3.0.2", + "duplexer3": "0.1.4", + "get-stream": "3.0.0", + "is-redirect": "1.0.0", + "is-retry-allowed": "1.2.0", + "is-stream": "1.1.0", + "lowercase-keys": "1.0.1", + "safe-buffer": "5.1.2", + "timed-out": "4.0.1", + "unzip-response": "2.0.1", + "url-parse-lax": "1.0.0" + } + }, + "graceful-fs": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.3.tgz", + "integrity": "sha512-a30VEBm4PEdx1dRB7MFK7BejejvCvBronbLjht+sHuGYj8PHs7M/5Z+rt5lw551vZ7yfTCj4Vuyy3mSJytDWRQ==" + }, + "growl": { + "version": "1.10.5", + "resolved": "https://registry.npmjs.org/growl/-/growl-1.10.5.tgz", + "integrity": "sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==", + "dev": true + }, + "handlebars": { + "version": "4.7.3", + "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.3.tgz", + "integrity": "sha512-SRGwSYuNfx8DwHD/6InAPzD6RgeruWLT+B8e8a7gGs8FWgHzlExpTFMEq2IA6QpAfOClpKHy6+8IqTjeBCu6Kg==", + "dev": true, + "requires": { + "neo-async": "2.6.1", + "optimist": "0.6.1", + "source-map": "0.6.1", + "uglify-js": "3.8.0" + }, + "dependencies": { + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true + } + } + }, + "has-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", + "dev": true, + "requires": { + "ansi-regex": "2.1.1" + } + }, + "has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", + "dev": true + }, + "has-value": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz", + "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=", + "dev": true, + "requires": { + "get-value": "2.0.6", + "has-values": "1.0.0", + "isobject": "3.0.1" + } + }, + "has-values": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz", + "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=", + "dev": true, + "requires": { + "is-number": "3.0.0", + "kind-of": "4.0.0" + }, + "dependencies": { + "kind-of": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", + "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", + "dev": true, + "requires": { + "is-buffer": "1.1.6" + } + } + } + }, + "he": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/he/-/he-1.1.1.tgz", + "integrity": "sha1-k0EP0hsAlzUVH4howvJx80J+I/0=", + "dev": true + }, + "http-errors": { + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", + "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", + "requires": { + "depd": "1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.1", + "statuses": "1.5.0", + "toidentifier": "1.0.0" + } + }, + "https-proxy-agent": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-4.0.0.tgz", + "integrity": "sha512-zoDhWrkR3of1l9QAL8/scJZyLu8j/gBkcwcaQOZh7Gyh/+uJQzGVETdgT30akuwkpL8HTRfssqI3BZuV18teDg==", + "requires": { + "agent-base": "5.1.1", + "debug": "4.1.1" + }, + "dependencies": { + "debug": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", + "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", + "requires": { + "ms": "2.1.2" + } + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + } + } + }, + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "requires": { + "safer-buffer": "2.1.2" + } + }, + "ignore": { + "version": "3.3.10", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz", + "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", + "dev": true + }, + "ignore-by-default": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", + "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", + "dev": true + }, + "import-lazy": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", + "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", + "dev": true + }, + "imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", + "dev": true + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "requires": { + "once": "1.4.0", + "wrappy": "1.0.2" + } + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" + }, + "ini": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", + "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==", + "dev": true + }, + "inquirer": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-3.3.0.tgz", + "integrity": "sha512-h+xtnyk4EwKvFWHrUYsWErEVR+igKtLdchu+o0Z1RL7VU/jVMFbYir2bp6bAj8efFNxWqHX0dIss6fJQ+/+qeQ==", + "dev": true, + "requires": { + "ansi-escapes": "3.2.0", + "chalk": "2.4.2", + "cli-cursor": "2.1.0", + "cli-width": "2.2.0", + "external-editor": "2.2.0", + "figures": "2.0.0", + "lodash": "4.17.15", + "mute-stream": "0.0.7", + "run-async": "2.4.0", + "rx-lite": "4.0.8", + "rx-lite-aggregates": "4.0.8", + "string-width": "2.1.1", + "strip-ansi": "4.0.0", + "through": "2.3.8" + } + }, + "ip-regex": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/ip-regex/-/ip-regex-2.1.0.tgz", + "integrity": "sha1-+ni/XS5pE8kRzp+BnuUUa7bYROk=", + "dev": true + }, + "ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" + }, + "is-accessor-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", + "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", + "dev": true, + "requires": { + "kind-of": "3.2.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "1.1.6" + } + } + } + }, + "is-arrayish": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz", + "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==" + }, + "is-binary-path": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", + "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", + "dev": true, + "requires": { + "binary-extensions": "1.13.1" + } + }, + "is-buffer": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", + "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", + "dev": true + }, + "is-ci": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-1.2.1.tgz", + "integrity": "sha512-s6tfsaQaQi3JNciBH6shVqEDvhGut0SUXr31ag8Pd8BBbVVlcGfWhpPmEOoM6RJ5TFhbypvf5yyRw/VXW1IiWg==", + "dev": true, + "requires": { + "ci-info": "1.6.0" + } + }, + "is-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "dev": true, + "requires": { + "kind-of": "3.2.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "1.1.6" + } + } + } + }, + "is-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", + "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "0.1.6", + "is-data-descriptor": "0.1.4", + "kind-of": "5.1.0" + }, + "dependencies": { + "kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "dev": true + } + } + }, + "is-extendable": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", + "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=", + "dev": true + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "is-glob": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", + "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", + "dev": true, + "requires": { + "is-extglob": "2.1.1" + } + }, + "is-installed-globally": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.1.0.tgz", + "integrity": "sha1-Df2Y9akRFxbdU13aZJL2e/PSWoA=", + "dev": true, + "requires": { + "global-dirs": "0.1.1", + "is-path-inside": "1.0.1" + } + }, + "is-ip": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-ip/-/is-ip-2.0.0.tgz", + "integrity": "sha1-aO6gfooKCpTC0IDdZ0xzGrKkYas=", + "dev": true, + "requires": { + "ip-regex": "2.1.0" + } + }, + "is-npm": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-1.0.0.tgz", + "integrity": "sha1-8vtjpl5JBbQGyGBydloaTceTufQ=", + "dev": true + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "3.2.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "1.1.6" + } + } + } + }, + "is-obj": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-1.0.1.tgz", + "integrity": "sha1-PkcprB9f3gJc19g6iW2rn09n2w8=", + "dev": true + }, + "is-path-inside": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.1.tgz", + "integrity": "sha1-jvW33lBDej/cprToZe96pVy0gDY=", + "dev": true, + "requires": { + "path-is-inside": "1.0.2" + } + }, + "is-plain-object": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", + "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", + "dev": true, + "requires": { + "isobject": "3.0.1" + } + }, + "is-promise": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz", + "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=", + "dev": true + }, + "is-redirect": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-redirect/-/is-redirect-1.0.0.tgz", + "integrity": "sha1-HQPd7VO9jbDzDCbk+V02/HyH3CQ=", + "dev": true + }, + "is-resolvable": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.1.0.tgz", + "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==", + "dev": true + }, + "is-retry-allowed": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/is-retry-allowed/-/is-retry-allowed-1.2.0.tgz", + "integrity": "sha512-RUbUeKwvm3XG2VYamhJL1xFktgjvPzL0Hq8C+6yrWIswDy3BIXGqCxhxkc30N9jqK311gVU137K8Ei55/zVJRg==", + "dev": true + }, + "is-stream": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", + "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=" + }, + "is-windows": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", + "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", + "dev": true + }, + "isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" + }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", + "dev": true + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "istanbul": { + "version": "0.4.5", + "resolved": "https://registry.npmjs.org/istanbul/-/istanbul-0.4.5.tgz", + "integrity": "sha1-ZcfXPUxNqE1POsMQuRj7C4Azczs=", + "dev": true, + "requires": { + "abbrev": "1.0.9", + "async": "1.5.2", + "escodegen": "1.8.1", + "esprima": "2.7.3", + "glob": "5.0.15", + "handlebars": "4.7.3", + "js-yaml": "3.13.1", + "mkdirp": "0.5.1", + "nopt": "3.0.6", + "once": "1.4.0", + "resolve": "1.1.7", + "supports-color": "3.2.3", + "which": "1.3.1", + "wordwrap": "1.0.0" + }, + "dependencies": { + "async": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/async/-/async-1.5.2.tgz", + "integrity": "sha1-7GphrlZIDAw8skHJVhjiCJL5Zyo=", + "dev": true + }, + "esprima": { + "version": "2.7.3", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-2.7.3.tgz", + "integrity": "sha1-luO3DVd59q1JzQMmc9HDEnZ7pYE=", + "dev": true + }, + "glob": { + "version": "5.0.15", + "resolved": "https://registry.npmjs.org/glob/-/glob-5.0.15.tgz", + "integrity": "sha1-G8k2ueAvSmA/zCIuz3Yz0wuLk7E=", + "dev": true, + "requires": { + "inflight": "1.0.6", + "inherits": "2.0.3", + "minimatch": "3.0.4", + "once": "1.4.0", + "path-is-absolute": "1.0.1" + } + }, + "has-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-1.0.0.tgz", + "integrity": "sha1-nZ55MWXOAXoA8AQYxD+UKnsdEfo=", + "dev": true + }, + "resolve": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.1.7.tgz", + "integrity": "sha1-IDEU2CrSxe2ejgQRs5ModeiJ6Xs=", + "dev": true + }, + "supports-color": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-3.2.3.tgz", + "integrity": "sha1-ZawFBLOVQXHYpklGsq48u4pfVPY=", + "dev": true, + "requires": { + "has-flag": "1.0.0" + } + } + } + }, + "js-tokens": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", + "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls=", + "dev": true + }, + "js-yaml": { + "version": "3.13.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", + "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", + "dev": true, + "requires": { + "argparse": "1.0.10", + "esprima": "4.0.1" + } + }, + "json-schema-traverse": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz", + "integrity": "sha1-NJptRMU6Ud6JtAgFxdXlm0F9M0A=", + "dev": true + }, + "json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", + "dev": true + }, + "jsonfile": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", + "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", + "requires": { + "graceful-fs": "4.2.3" + } + }, + "jsonwebtoken": { + "version": "8.5.1", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz", + "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==", + "requires": { + "jws": "3.2.2", + "lodash.includes": "4.3.0", + "lodash.isboolean": "3.0.3", + "lodash.isinteger": "4.0.4", + "lodash.isnumber": "3.0.3", + "lodash.isplainobject": "4.0.6", + "lodash.isstring": "4.0.1", + "lodash.once": "4.1.1", + "ms": "2.1.2", + "semver": "5.7.1" + }, + "dependencies": { + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + } + } + }, + "jwa": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", + "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", + "requires": { + "buffer-equal-constant-time": "1.0.1", + "ecdsa-sig-formatter": "1.0.11", + "safe-buffer": "5.1.2" + } + }, + "jws": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", + "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", + "requires": { + "jwa": "1.4.1", + "safe-buffer": "5.1.2" + } + }, + "kareem": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/kareem/-/kareem-2.3.1.tgz", + "integrity": "sha512-l3hLhffs9zqoDe8zjmb/mAN4B8VT3L56EUvKNqLFVs9YlFA+zx7ke1DO8STAdDyYNkeSo1nKmjuvQeI12So8Xw==" + }, + "kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true + }, + "kuler": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/kuler/-/kuler-1.0.1.tgz", + "integrity": "sha512-J9nVUucG1p/skKul6DU3PUZrhs0LPulNaeUOox0IyXDi8S4CztTHs1gQphhuZmzXG7VOQSf6NJfKuzteQLv9gQ==", + "requires": { + "colornames": "1.1.1" + } + }, + "latest-version": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-3.1.0.tgz", + "integrity": "sha1-ogU4P+oyKzO1rjsYq+4NwvNW7hU=", + "dev": true, + "requires": { + "package-json": "4.0.1" + } + }, + "levn": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", + "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", + "dev": true, + "requires": { + "prelude-ls": "1.1.2", + "type-check": "0.3.2" + } + }, + "lodash": { + "version": "4.17.15", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", + "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==" + }, + "lodash.includes": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", + "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=" + }, + "lodash.isboolean": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", + "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=" + }, + "lodash.isinteger": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", + "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=" + }, + "lodash.isnumber": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", + "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=" + }, + "lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" + }, + "lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=" + }, + "lodash.once": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", + "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=" + }, + "logform": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/logform/-/logform-2.1.2.tgz", + "integrity": "sha512-+lZh4OpERDBLqjiwDLpAWNQu6KMjnlXH2ByZwCuSqVPJletw0kTWJf5CgSNAUKn1KUkv3m2cUz/LK8zyEy7wzQ==", + "requires": { + "colors": "1.4.0", + "fast-safe-stringify": "2.0.7", + "fecha": "2.3.3", + "ms": "2.1.2", + "triple-beam": "1.3.0" + }, + "dependencies": { + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + } + } + }, + "lowercase-keys": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", + "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", + "dev": true + }, + "lru-cache": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.5.tgz", + "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", + "dev": true, + "requires": { + "pseudomap": "1.0.2", + "yallist": "2.1.2" + } + }, + "make-dir": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-1.3.0.tgz", + "integrity": "sha512-2w31R7SJtieJJnQtGc7RVL2StM2vGYVfqUOvUDxH6bC6aJTxPxTF0GnIgCyu7tjockiUWAYQRbxa7vKn34s5sQ==", + "dev": true, + "requires": { + "pify": "3.0.0" + } + }, + "map-cache": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", + "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=", + "dev": true + }, + "map-visit": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz", + "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=", + "dev": true, + "requires": { + "object-visit": "1.0.1" + } + }, + "media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" + }, + "memory-pager": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/memory-pager/-/memory-pager-1.5.0.tgz", + "integrity": "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg==", + "optional": true + }, + "merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" + }, + "methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" + }, + "micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", + "dev": true, + "requires": { + "arr-diff": "4.0.0", + "array-unique": "0.3.2", + "braces": "2.3.2", + "define-property": "2.0.2", + "extend-shallow": "3.0.2", + "extglob": "2.0.4", + "fragment-cache": "0.2.1", + "kind-of": "6.0.3", + "nanomatch": "1.2.13", + "object.pick": "1.3.0", + "regex-not": "1.0.2", + "snapdragon": "0.8.2", + "to-regex": "3.0.2" + } + }, + "mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" + }, + "mime-db": { + "version": "1.43.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.43.0.tgz", + "integrity": "sha512-+5dsGEEovYbT8UY9yD7eE4XTc4UwJ1jBYlgaQQF38ENsKR3wj/8q8RFZrF9WIZpB2V1ArTVFUva8sAul1NzRzQ==" + }, + "mime-types": { + "version": "2.1.26", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.26.tgz", + "integrity": "sha512-01paPWYgLrkqAyrlDorC1uDwl2p3qZT7yl806vW7DvDoxwXi46jsjFbg+WdwotBIk6/MbEhO/dh5aZ5sNj/dWQ==", + "requires": { + "mime-db": "1.43.0" + } + }, + "mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true + }, + "minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "requires": { + "brace-expansion": "1.1.11" + } + }, + "minimist": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", + "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" + }, + "mixin-deep": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.2.tgz", + "integrity": "sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==", + "dev": true, + "requires": { + "for-in": "1.0.2", + "is-extendable": "1.0.1" + }, + "dependencies": { + "is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, + "requires": { + "is-plain-object": "2.0.4" + } + } + } + }, + "mkdirp": { + "version": "0.5.1", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", + "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", + "requires": { + "minimist": "0.0.8" + } + }, + "mocha": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/mocha/-/mocha-5.2.0.tgz", + "integrity": "sha512-2IUgKDhc3J7Uug+FxMXuqIyYzH7gJjXECKe/w43IGgQHTSj3InJi+yAA7T24L9bQMRKiUEHxEX37G5JpVUGLcQ==", + "dev": true, + "requires": { + "browser-stdout": "1.3.1", + "commander": "2.15.1", + "debug": "3.1.0", + "diff": "3.5.0", + "escape-string-regexp": "1.0.5", + "glob": "7.1.2", + "growl": "1.10.5", + "he": "1.1.1", + "minimatch": "3.0.4", + "mkdirp": "0.5.1", + "supports-color": "5.4.0" + }, + "dependencies": { + "commander": { + "version": "2.15.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.15.1.tgz", + "integrity": "sha512-VlfT9F3V0v+jr4yxPc5gg9s62/fIVWsd2Bk2iD435um1NlGMYdVCq+MjcXnhYq2icNOizHr1kK+5TI6H0Hy0ag==", + "dev": true + }, + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "glob": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz", + "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==", + "dev": true, + "requires": { + "fs.realpath": "1.0.0", + "inflight": "1.0.6", + "inherits": "2.0.3", + "minimatch": "3.0.4", + "once": "1.4.0", + "path-is-absolute": "1.0.1" + } + }, + "supports-color": { + "version": "5.4.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.4.0.tgz", + "integrity": "sha512-zjaXglF5nnWpsq470jSv6P9DwPvgLkuapYmfDm3JWOm0vkNTVF2tI4UrN2r6jH1qM/uc/WtxYY1hYoA2dOKj5w==", + "dev": true, + "requires": { + "has-flag": "3.0.0" + } + } + } + }, + "mongodb": { + "version": "3.5.5", + "resolved": "https://registry.npmjs.org/mongodb/-/mongodb-3.5.5.tgz", + "integrity": "sha512-GCjDxR3UOltDq00Zcpzql6dQo1sVry60OXJY3TDmFc2SWFY6c8Gn1Ardidc5jDirvJrx2GC3knGOImKphbSL3A==", + "requires": { + "bl": "2.2.0", + "bson": "1.1.3", + "denque": "1.4.1", + "require_optional": "1.0.1", + "safe-buffer": "5.1.2", + "saslprep": "1.0.3" + } + }, + "mongoose": { + "version": "5.9.5", + "resolved": "https://registry.npmjs.org/mongoose/-/mongoose-5.9.5.tgz", + "integrity": "sha512-2kMNZCZRWCMtww4f//CwdGH6BjO3+9/c3YdsC6nbzdJVyl8+GRtNfgrKUge3226VZXXLJa6LwxXN2K8/Dh4irg==", + "requires": { + "bson": "1.1.3", + "kareem": "2.3.1", + "mongodb": "3.5.5", + "mongoose-legacy-pluralize": "1.0.2", + "mpath": "0.6.0", + "mquery": "3.2.2", + "ms": "2.1.2", + "regexp-clone": "1.0.0", + "safe-buffer": "5.1.2", + "sift": "7.0.1", + "sliced": "1.0.1" + }, + "dependencies": { + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + } + } + }, + "mongoose-legacy-pluralize": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/mongoose-legacy-pluralize/-/mongoose-legacy-pluralize-1.0.2.tgz", + "integrity": "sha512-Yo/7qQU4/EyIS8YDFSeenIvXxZN+ld7YdV9LqFVQJzTLye8unujAWPZ4NWKfFA+RNjh+wvTWKY9Z3E5XM6ZZiQ==" + }, + "mpath": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/mpath/-/mpath-0.6.0.tgz", + "integrity": "sha512-i75qh79MJ5Xo/sbhxrDrPSEG0H/mr1kcZXJ8dH6URU5jD/knFxCVqVC/gVSW7GIXL/9hHWlT9haLbCXWOll3qw==" + }, + "mquery": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/mquery/-/mquery-3.2.2.tgz", + "integrity": "sha512-XB52992COp0KP230I3qloVUbkLUxJIu328HBP2t2EsxSFtf4W1HPSOBWOXf1bqxK4Xbb66lfMJ+Bpfd9/yZE1Q==", + "requires": { + "bluebird": "3.5.1", + "debug": "3.1.0", + "regexp-clone": "1.0.0", + "safe-buffer": "5.1.2", + "sliced": "1.0.1" + }, + "dependencies": { + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "requires": { + "ms": "2.0.0" + } + } + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "mute-stream": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", + "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=", + "dev": true + }, + "nan": { + "version": "2.14.0", + "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.0.tgz", + "integrity": "sha512-INOFj37C7k3AfaNTtX8RhsTw7qRy7eLET14cROi9+5HAVbbHuIWUHEauBv5qT4Av2tWasiTY1Jw6puUNqRJXQg==", + "dev": true, + "optional": true + }, + "nanomatch": { + "version": "1.2.13", + "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", + "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", + "dev": true, + "requires": { + "arr-diff": "4.0.0", + "array-unique": "0.3.2", + "define-property": "2.0.2", + "extend-shallow": "3.0.2", + "fragment-cache": "0.2.1", + "is-windows": "1.0.2", + "kind-of": "6.0.3", + "object.pick": "1.3.0", + "regex-not": "1.0.2", + "snapdragon": "0.8.2", + "to-regex": "3.0.2" + } + }, + "natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", + "dev": true + }, + "negotiator": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", + "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" + }, + "neo-async": { + "version": "2.6.1", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.1.tgz", + "integrity": "sha512-iyam8fBuCUpWeKPGpaNMetEocMt364qkCsfL9JuhjXX6dRnguRVOfk2GZaDpPjcOKiiXCPINZC1GczQ7iTq3Zw==", + "dev": true + }, + "nodemon": { + "version": "1.19.4", + "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-1.19.4.tgz", + "integrity": "sha512-VGPaqQBNk193lrJFotBU8nvWZPqEZY2eIzymy2jjY0fJ9qIsxA0sxQ8ATPl0gZC645gijYEc1jtZvpS8QWzJGQ==", + "dev": true, + "requires": { + "chokidar": "2.1.8", + "debug": "3.2.6", + "ignore-by-default": "1.0.1", + "minimatch": "3.0.4", + "pstree.remy": "1.1.7", + "semver": "5.7.1", + "supports-color": "5.5.0", + "touch": "3.1.0", + "undefsafe": "2.0.3", + "update-notifier": "2.5.0" + }, + "dependencies": { + "debug": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", + "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "requires": { + "has-flag": "3.0.0" + } + } + } + }, + "nopt": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/nopt/-/nopt-3.0.6.tgz", + "integrity": "sha1-xkZdvwirzU2zWTF/eaxopkayj/k=", + "dev": true, + "requires": { + "abbrev": "1.0.9" + } + }, + "normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true + }, + "npm-run-path": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz", + "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=", + "dev": true, + "requires": { + "path-key": "2.0.1" + } + }, + "object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", + "dev": true + }, + "object-copy": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", + "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=", + "dev": true, + "requires": { + "copy-descriptor": "0.1.1", + "define-property": "0.2.5", + "kind-of": "3.2.2" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "0.1.6" + } + }, + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "1.1.6" + } + } + } + }, + "object-visit": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz", + "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=", + "dev": true, + "requires": { + "isobject": "3.0.1" + } + }, + "object.pick": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", + "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=", + "dev": true, + "requires": { + "isobject": "3.0.1" + } + }, + "on-finished": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", + "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", + "requires": { + "ee-first": "1.1.1" + } + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "requires": { + "wrappy": "1.0.2" + } + }, + "one-time": { + "version": "0.0.4", + "resolved": "https://registry.npmjs.org/one-time/-/one-time-0.0.4.tgz", + "integrity": "sha1-+M33eISCb+Tf+T46nMN7HkSAdC4=" + }, + "onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", + "dev": true, + "requires": { + "mimic-fn": "1.2.0" + } + }, + "optimist": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/optimist/-/optimist-0.6.1.tgz", + "integrity": "sha1-2j6nRob6IaGaERwybpDrFaAZZoY=", + "dev": true, + "requires": { + "minimist": "0.0.8", + "wordwrap": "0.0.3" + }, + "dependencies": { + "wordwrap": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-0.0.3.tgz", + "integrity": "sha1-o9XabNXAvAAI03I0u68b7WMFkQc=", + "dev": true + } + } + }, + "optionator": { + "version": "0.8.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", + "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", + "dev": true, + "requires": { + "deep-is": "0.1.3", + "fast-levenshtein": "2.0.6", + "levn": "0.3.0", + "prelude-ls": "1.1.2", + "type-check": "0.3.2", + "word-wrap": "1.2.3" + } + }, + "os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", + "dev": true + }, + "p-finally": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", + "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", + "dev": true + }, + "package-json": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/package-json/-/package-json-4.0.1.tgz", + "integrity": "sha1-iGmgQBJTZhxMTKPabCEh7VVfXu0=", + "dev": true, + "requires": { + "got": "6.7.1", + "registry-auth-token": "3.4.0", + "registry-url": "3.1.0", + "semver": "5.7.1" + } + }, + "parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" + }, + "pascalcase": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", + "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", + "dev": true + }, + "path": { + "version": "0.12.7", + "resolved": "https://registry.npmjs.org/path/-/path-0.12.7.tgz", + "integrity": "sha1-1NwqUGxM4hl+tIHr/NWzbAFAsQ8=", + "requires": { + "process": "0.11.10", + "util": "0.10.4" + } + }, + "path-dirname": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", + "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", + "dev": true + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" + }, + "path-is-inside": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", + "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=", + "dev": true + }, + "path-key": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", + "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", + "dev": true + }, + "path-parse": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", + "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", + "dev": true + }, + "path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" + }, + "pathval": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/pathval/-/pathval-1.1.0.tgz", + "integrity": "sha1-uULm1L3mUwBe9rcTYd74cn0GReA=", + "dev": true + }, + "pend": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/pend/-/pend-1.2.0.tgz", + "integrity": "sha1-elfrVQpng/kRUzH89GY9XI4AelA=" + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "pluralize": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-7.0.0.tgz", + "integrity": "sha512-ARhBOdzS3e41FbkW/XWrTEtukqqLoK5+Z/4UeDaLuSW+39JPeFgs4gCGqsrJHVZX0fUrx//4OF0K1CUGwlIFow==", + "dev": true + }, + "posix-character-classes": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz", + "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=", + "dev": true + }, + "prelude-ls": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", + "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", + "dev": true + }, + "prepend-http": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz", + "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=", + "dev": true + }, + "process": { + "version": "0.11.10", + "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", + "integrity": "sha1-czIwDoQBYb2j5podHZGn1LwW8YI=" + }, + "process-nextick-args": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" + }, + "progress": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", + "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==" + }, + "proxy-addr": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", + "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", + "requires": { + "forwarded": "0.1.2", + "ipaddr.js": "1.9.1" + } + }, + "proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==" + }, + "pseudomap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", + "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=", + "dev": true + }, + "pstree.remy": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.7.tgz", + "integrity": "sha512-xsMgrUwRpuGskEzBFkH8NmTimbZ5PcPup0LA8JJkHIm2IMUbQcpo3yeLNWVrufEYjh8YwtSVh0xz6UeWc5Oh5A==", + "dev": true + }, + "punycode": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", + "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=" + }, + "puppeteer": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/puppeteer/-/puppeteer-2.1.1.tgz", + "integrity": "sha512-LWzaDVQkk1EPiuYeTOj+CZRIjda4k2s5w4MK4xoH2+kgWV/SDlkYHmxatDdtYrciHUKSXTsGgPgPP8ILVdBsxg==", + "requires": { + "@types/mime-types": "2.1.0", + "debug": "4.1.1", + "extract-zip": "1.6.7", + "https-proxy-agent": "4.0.0", + "mime": "2.4.4", + "mime-types": "2.1.26", + "progress": "2.0.3", + "proxy-from-env": "1.1.0", + "rimraf": "2.7.1", + "ws": "6.2.1" + }, + "dependencies": { + "debug": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", + "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", + "requires": { + "ms": "2.1.2" + } + }, + "mime": { + "version": "2.4.4", + "resolved": "https://registry.npmjs.org/mime/-/mime-2.4.4.tgz", + "integrity": "sha512-LRxmNwziLPT828z+4YkNzloCFC2YM4wrB99k+AV5ZbEyfGNWfG8SO1FUXLmLDBSo89NrJZ4DIWeLjy1CHGhMGA==" + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + } + } + }, + "qs": { + "version": "6.7.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", + "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" + }, + "querystring": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", + "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA=" + }, + "range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" + }, + "raw-body": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", + "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", + "requires": { + "bytes": "3.1.0", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + } + }, + "rc": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", + "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", + "dev": true, + "requires": { + "deep-extend": "0.6.0", + "ini": "1.3.5", + "minimist": "1.2.5", + "strip-json-comments": "2.0.1" + }, + "dependencies": { + "minimist": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", + "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", + "dev": true + } + } + }, + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "requires": { + "core-util-is": "1.0.2", + "inherits": "2.0.3", + "isarray": "1.0.0", + "process-nextick-args": "2.0.1", + "safe-buffer": "5.1.2", + "string_decoder": "1.1.1", + "util-deprecate": "1.0.2" + } + }, + "readdirp": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", + "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", + "dev": true, + "requires": { + "graceful-fs": "4.2.3", + "micromatch": "3.1.10", + "readable-stream": "2.3.7" + } + }, + "regex-not": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.2.tgz", + "integrity": "sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==", + "dev": true, + "requires": { + "extend-shallow": "3.0.2", + "safe-regex": "1.1.0" + } + }, + "regexp-clone": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/regexp-clone/-/regexp-clone-1.0.0.tgz", + "integrity": "sha512-TuAasHQNamyyJ2hb97IuBEif4qBHGjPHBS64sZwytpLEqtBQ1gPJTnOaQ6qmpET16cK14kkjbazl6+p0RRv0yw==" + }, + "regexpp": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-1.1.0.tgz", + "integrity": "sha512-LOPw8FpgdQF9etWMaAfG/WRthIdXJGYp4mJ2Jgn/2lpkbod9jPn0t9UqN7AxBOKNfzRbYyVfgc7Vk4t/MpnXgw==", + "dev": true + }, + "registry-auth-token": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-3.4.0.tgz", + "integrity": "sha512-4LM6Fw8eBQdwMYcES4yTnn2TqIasbXuwDx3um+QRs7S55aMKCBKBxvPXl2RiUjHwuJLTyYfxSpmfSAjQpcuP+A==", + "dev": true, + "requires": { + "rc": "1.2.8", + "safe-buffer": "5.1.2" + } + }, + "registry-url": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-3.1.0.tgz", + "integrity": "sha1-PU74cPc93h138M+aOBQyRE4XSUI=", + "dev": true, + "requires": { + "rc": "1.2.8" + } + }, + "remove-trailing-separator": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", + "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", + "dev": true + }, + "repeat-element": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.3.tgz", + "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==", + "dev": true + }, + "repeat-string": { + "version": "1.6.1", + "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", + "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=", + "dev": true + }, + "require-uncached": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/require-uncached/-/require-uncached-1.0.3.tgz", + "integrity": "sha1-Tg1W1slmL9MeQwEcS5WqSZVUIdM=", + "dev": true, + "requires": { + "caller-path": "0.1.0", + "resolve-from": "1.0.1" + }, + "dependencies": { + "resolve-from": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-1.0.1.tgz", + "integrity": "sha1-Jsv+k10a7uq7Kbw/5a6wHpPUQiY=", + "dev": true + } + } + }, + "require_optional": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/require_optional/-/require_optional-1.0.1.tgz", + "integrity": "sha512-qhM/y57enGWHAe3v/NcwML6a3/vfESLe/sGM2dII+gEO0BpKRUkWZow/tyloNqJyN6kXSl3RyyM8Ll5D/sJP8g==", + "requires": { + "resolve-from": "2.0.0", + "semver": "5.7.1" + } + }, + "resolve": { + "version": "1.15.1", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.15.1.tgz", + "integrity": "sha512-84oo6ZTtoTUpjgNEr5SJyzQhzL72gaRodsSfyxC/AXRvwu0Yse9H8eF9IpGo7b8YetZhlI6v7ZQ6bKBFV/6S7w==", + "dev": true, + "requires": { + "path-parse": "1.0.6" + } + }, + "resolve-from": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-2.0.0.tgz", + "integrity": "sha1-lICrIOlP+h2egKgEx+oUdhGWa1c=" + }, + "resolve-url": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", + "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=", + "dev": true + }, + "restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", + "dev": true, + "requires": { + "onetime": "2.0.1", + "signal-exit": "3.0.2" + } + }, + "ret": { + "version": "0.1.15", + "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", + "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==", + "dev": true + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "requires": { + "glob": "7.1.6" + } + }, + "run-async": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.4.0.tgz", + "integrity": "sha512-xJTbh/d7Lm7SBhc1tNvTpeCHaEzoyxPrqNlvSdMfBTYwaY++UJFyXUOxAtsRUXjlqOfj8luNaR9vjCh4KeV+pg==", + "dev": true, + "requires": { + "is-promise": "2.1.0" + } + }, + "rx-lite": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/rx-lite/-/rx-lite-4.0.8.tgz", + "integrity": "sha1-Cx4Rr4vESDbwSmQH6S2kJGe3lEQ=", + "dev": true + }, + "rx-lite-aggregates": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/rx-lite-aggregates/-/rx-lite-aggregates-4.0.8.tgz", + "integrity": "sha1-dTuHqJoRyVRnxKwWJsTvxOBcZ74=", + "dev": true, + "requires": { + "rx-lite": "4.0.8" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "safe-regex": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", + "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", + "dev": true, + "requires": { + "ret": "0.1.15" + } + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "saslprep": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/saslprep/-/saslprep-1.0.3.tgz", + "integrity": "sha512-/MY/PEMbk2SuY5sScONwhUDsV2p77Znkb/q3nSVstq/yQzYJOH/Azh29p9oJLsl3LnQwSvZDKagDGBsBwSooag==", + "optional": true, + "requires": { + "sparse-bitfield": "3.0.3" + } + }, + "sax": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", + "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==" + }, + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" + }, + "semver-diff": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-2.1.0.tgz", + "integrity": "sha1-S7uEN8jTfksM8aaP1ybsbWRdbTY=", + "dev": true, + "requires": { + "semver": "5.7.1" + } + }, + "send": { + "version": "0.17.1", + "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", + "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", + "requires": { + "debug": "2.6.9", + "depd": "1.1.2", + "destroy": "1.0.4", + "encodeurl": "1.0.2", + "escape-html": "1.0.3", + "etag": "1.8.1", + "fresh": "0.5.2", + "http-errors": "1.7.2", + "mime": "1.6.0", + "ms": "2.1.1", + "on-finished": "2.3.0", + "range-parser": "1.2.1", + "statuses": "1.5.0" + }, + "dependencies": { + "ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" + } + } + }, + "serve-static": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", + "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", + "requires": { + "encodeurl": "1.0.2", + "escape-html": "1.0.3", + "parseurl": "1.3.3", + "send": "0.17.1" + } + }, + "set-value": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.1.tgz", + "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", + "dev": true, + "requires": { + "extend-shallow": "2.0.1", + "is-extendable": "0.1.1", + "is-plain-object": "2.0.4", + "split-string": "3.1.0" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "0.1.1" + } + } + } + }, + "setprototypeof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", + "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" + }, + "shebang-command": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", + "dev": true, + "requires": { + "shebang-regex": "1.0.0" + } + }, + "shebang-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", + "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", + "dev": true + }, + "sift": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/sift/-/sift-7.0.1.tgz", + "integrity": "sha512-oqD7PMJ+uO6jV9EQCl0LrRw1OwsiPsiFQR5AR30heR+4Dl7jBBbDLnNvWiak20tzZlSE1H7RB30SX/1j/YYT7g==" + }, + "signal-exit": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", + "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", + "dev": true + }, + "simple-swizzle": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.2.tgz", + "integrity": "sha1-pNprY1/8zMoz9w0Xy5JZLeleVXo=", + "requires": { + "is-arrayish": "0.3.2" + } + }, + "slice-ansi": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-1.0.0.tgz", + "integrity": "sha512-POqxBK6Lb3q6s047D/XsDVNPnF9Dl8JSaqe9h9lURl0OdNqy/ujDrOiIHtsqXMGbWWTIomRzAMaTyawAU//Reg==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "2.0.0" + } + }, + "sliced": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/sliced/-/sliced-1.0.1.tgz", + "integrity": "sha1-CzpmK10Ewxd7GSa+qCsD+Dei70E=" + }, + "snapdragon": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz", + "integrity": "sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==", + "dev": true, + "requires": { + "base": "0.11.2", + "debug": "2.6.9", + "define-property": "0.2.5", + "extend-shallow": "2.0.1", + "map-cache": "0.2.2", + "source-map": "0.5.7", + "source-map-resolve": "0.5.3", + "use": "3.1.1" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "0.1.6" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "0.1.1" + } + }, + "source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", + "dev": true + } + } + }, + "snapdragon-node": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz", + "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==", + "dev": true, + "requires": { + "define-property": "1.0.0", + "isobject": "3.0.1", + "snapdragon-util": "3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "1.0.2" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "6.0.3" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "6.0.3" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "1.0.0", + "is-data-descriptor": "1.0.0", + "kind-of": "6.0.3" + } + } + } + }, + "snapdragon-util": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/snapdragon-util/-/snapdragon-util-3.0.1.tgz", + "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==", + "dev": true, + "requires": { + "kind-of": "3.2.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "1.1.6" + } + } + } + }, + "source-map": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.2.0.tgz", + "integrity": "sha1-2rc/vPwrqBm03gO9b26qSBZLP50=", + "dev": true, + "optional": true, + "requires": { + "amdefine": "1.0.1" + } + }, + "source-map-resolve": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.3.tgz", + "integrity": "sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==", + "dev": true, + "requires": { + "atob": "2.1.2", + "decode-uri-component": "0.2.0", + "resolve-url": "0.2.1", + "source-map-url": "0.4.0", + "urix": "0.1.0" + } + }, + "source-map-url": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.0.tgz", + "integrity": "sha1-PpNdfd1zYxuXZZlW1VEo6HtQhKM=", + "dev": true + }, + "sparse-bitfield": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/sparse-bitfield/-/sparse-bitfield-3.0.3.tgz", + "integrity": "sha1-/0rm5oZWBWuks+eSqzM004JzyhE=", + "optional": true, + "requires": { + "memory-pager": "1.5.0" + } + }, + "split-string": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz", + "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", + "dev": true, + "requires": { + "extend-shallow": "3.0.2" + } + }, + "sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", + "dev": true + }, + "stack-trace": { + "version": "0.0.10", + "resolved": "https://registry.npmjs.org/stack-trace/-/stack-trace-0.0.10.tgz", + "integrity": "sha1-VHxws0fo0ytOEI6hoqFZ5f3eGcA=" + }, + "static-extend": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz", + "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=", + "dev": true, + "requires": { + "define-property": "0.2.5", + "object-copy": "0.1.0" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "0.1.6" + } + } + } + }, + "statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" + }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "2.0.0", + "strip-ansi": "4.0.0" + } + }, + "string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "requires": { + "safe-buffer": "5.1.2" + } + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "3.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + } + } + }, + "strip-eof": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz", + "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=", + "dev": true + }, + "strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", + "dev": true + }, + "superagent": { + "version": "3.8.3", + "resolved": "https://registry.npmjs.org/superagent/-/superagent-3.8.3.tgz", + "integrity": "sha512-GLQtLMCoEIK4eDv6OGtkOoSMt3D+oq0y3dsxMuYuDvaNUvuT8eFBuLmfR0iYYzHC1e8hpzC6ZsxbuP6DIalMFA==", + "dev": true, + "requires": { + "component-emitter": "1.3.0", + "cookiejar": "2.1.2", + "debug": "3.2.6", + "extend": "3.0.2", + "form-data": "2.5.1", + "formidable": "1.2.2", + "methods": "1.1.2", + "mime": "1.6.0", + "qs": "6.7.0", + "readable-stream": "2.3.7" + }, + "dependencies": { + "debug": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", + "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + } + } + }, + "supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true + }, + "table": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/table/-/table-4.0.2.tgz", + "integrity": "sha512-UUkEAPdSGxtRpiV9ozJ5cMTtYiqz7Ni1OGqLXRCynrvzdtR1p+cfOWe2RJLwvUG8hNanaSRjecIqwOjqeatDsA==", + "dev": true, + "requires": { + "ajv": "5.5.2", + "ajv-keywords": "2.1.1", + "chalk": "2.4.2", + "lodash": "4.17.15", + "slice-ansi": "1.0.0", + "string-width": "2.1.1" + } + }, + "term-size": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/term-size/-/term-size-1.2.0.tgz", + "integrity": "sha1-RYuDiH8oj8Vtb/+/rSYuJmOO+mk=", + "dev": true, + "requires": { + "execa": "0.7.0" + } + }, + "text-hex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/text-hex/-/text-hex-1.0.0.tgz", + "integrity": "sha512-uuVGNWzgJ4yhRaNSiubPY7OjISw4sw4E5Uv0wbjp+OzcbmVU/rsT8ujgcXJhn9ypzsgr5vlzpPqP+MBBKcGvbg==" + }, + "text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", + "dev": true + }, + "through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", + "dev": true + }, + "timed-out": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/timed-out/-/timed-out-4.0.1.tgz", + "integrity": "sha1-8y6srFoXW+ol1/q1Zas+2HQe9W8=", + "dev": true + }, + "tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "requires": { + "os-tmpdir": "1.0.2" + } + }, + "to-object-path": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", + "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", + "dev": true, + "requires": { + "kind-of": "3.2.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "1.1.6" + } + } + } + }, + "to-regex": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/to-regex/-/to-regex-3.0.2.tgz", + "integrity": "sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==", + "dev": true, + "requires": { + "define-property": "2.0.2", + "extend-shallow": "3.0.2", + "regex-not": "1.0.2", + "safe-regex": "1.1.0" + } + }, + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", + "dev": true, + "requires": { + "is-number": "3.0.0", + "repeat-string": "1.6.1" + } + }, + "toidentifier": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", + "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" + }, + "touch": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", + "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", + "dev": true, + "requires": { + "nopt": "1.0.10" + }, + "dependencies": { + "nopt": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", + "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", + "dev": true, + "requires": { + "abbrev": "1.0.9" + } + } + } + }, + "triple-beam": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/triple-beam/-/triple-beam-1.3.0.tgz", + "integrity": "sha512-XrHUvV5HpdLmIj4uVMxHggLbFSZYIn7HEWsqePZcI50pco+MPqJ50wMGY794X7AOOhxOBAjbkqfAbEe/QMp2Lw==" + }, + "type-check": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", + "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", + "dev": true, + "requires": { + "prelude-ls": "1.1.2" + } + }, + "type-detect": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", + "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", + "dev": true + }, + "type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "requires": { + "media-typer": "0.3.0", + "mime-types": "2.1.26" + } + }, + "typedarray": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", + "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=" + }, + "uglify-js": { + "version": "3.8.0", + "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.8.0.tgz", + "integrity": "sha512-ugNSTT8ierCsDHso2jkBHXYrU8Y5/fY2ZUprfrJUiD7YpuFvV4jODLFmb3h4btQjqr5Nh4TX4XtgDfCU1WdioQ==", + "dev": true, + "optional": true, + "requires": { + "commander": "2.20.3", + "source-map": "0.6.1" + }, + "dependencies": { + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "optional": true + } + } + }, + "undefsafe": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz", + "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==", + "dev": true, + "requires": { + "debug": "2.6.9" + } + }, + "union-value": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.1.tgz", + "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", + "dev": true, + "requires": { + "arr-union": "3.1.0", + "get-value": "2.0.6", + "is-extendable": "0.1.1", + "set-value": "2.0.1" + } + }, + "unique-string": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-1.0.0.tgz", + "integrity": "sha1-nhBXzKhRq7kzmPizOuGHuZyuwRo=", + "dev": true, + "requires": { + "crypto-random-string": "1.0.0" + } + }, + "universalify": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", + "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==" + }, + "unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" + }, + "unset-value": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", + "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=", + "dev": true, + "requires": { + "has-value": "0.3.1", + "isobject": "3.0.1" + }, + "dependencies": { + "has-value": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", + "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", + "dev": true, + "requires": { + "get-value": "2.0.6", + "has-values": "0.1.4", + "isobject": "2.1.0" + }, + "dependencies": { + "isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", + "dev": true, + "requires": { + "isarray": "1.0.0" + } + } + } + }, + "has-values": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", + "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=", + "dev": true + } + } + }, + "unzip-response": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/unzip-response/-/unzip-response-2.0.1.tgz", + "integrity": "sha1-0vD3N9FrBhXnKmk17QQhRXLVb5c=", + "dev": true + }, + "upath": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/upath/-/upath-1.2.0.tgz", + "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==", + "dev": true + }, + "update-notifier": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-2.5.0.tgz", + "integrity": "sha512-gwMdhgJHGuj/+wHJJs9e6PcCszpxR1b236igrOkUofGhqJuG+amlIKwApH1IW1WWl7ovZxsX49lMBWLxSdm5Dw==", + "dev": true, + "requires": { + "boxen": "1.3.0", + "chalk": "2.4.2", + "configstore": "3.1.2", + "import-lazy": "2.1.0", + "is-ci": "1.2.1", + "is-installed-globally": "0.1.0", + "is-npm": "1.0.0", + "latest-version": "3.1.0", + "semver-diff": "2.1.0", + "xdg-basedir": "3.0.0" + } + }, + "urix": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", + "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", + "dev": true + }, + "url": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/url/-/url-0.11.0.tgz", + "integrity": "sha1-ODjpfPxgUh63PFJajlW/3Z4uKPE=", + "requires": { + "punycode": "1.3.2", + "querystring": "0.2.0" + } + }, + "url-parse-lax": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-1.0.0.tgz", + "integrity": "sha1-evjzA2Rem9eaJy56FKxovAYJ2nM=", + "dev": true, + "requires": { + "prepend-http": "1.0.4" + } + }, + "use": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz", + "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", + "dev": true + }, + "util": { + "version": "0.10.4", + "resolved": "https://registry.npmjs.org/util/-/util-0.10.4.tgz", + "integrity": "sha512-0Pm9hTQ3se5ll1XihRic3FDIku70C+iHUdT/W926rSgHV5QgXsYbKZN8MSC3tJtSkhuROzvsQjAaFENRXr+19A==", + "requires": { + "inherits": "2.0.3" + } + }, + "util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" + }, + "utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" + }, + "vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" + }, + "which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "requires": { + "isexe": "2.0.0" + } + }, + "widest-line": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-2.0.1.tgz", + "integrity": "sha512-Ba5m9/Fa4Xt9eb2ELXt77JxVDV8w7qQrH0zS/TWSJdLyAwQjWoOzpzj5lwVftDz6n/EOu3tNACS84v509qwnJA==", + "dev": true, + "requires": { + "string-width": "2.1.1" + } + }, + "winston": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/winston/-/winston-3.2.1.tgz", + "integrity": "sha512-zU6vgnS9dAWCEKg/QYigd6cgMVVNwyTzKs81XZtTFuRwJOcDdBg7AU0mXVyNbs7O5RH2zdv+BdNZUlx7mXPuOw==", + "requires": { + "async": "2.6.3", + "diagnostics": "1.1.1", + "is-stream": "1.1.0", + "logform": "2.1.2", + "one-time": "0.0.4", + "readable-stream": "3.6.0", + "stack-trace": "0.0.10", + "triple-beam": "1.3.0", + "winston-transport": "4.3.0" + }, + "dependencies": { + "readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "requires": { + "inherits": "2.0.3", + "string_decoder": "1.1.1", + "util-deprecate": "1.0.2" + } + } + } + }, + "winston-transport": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/winston-transport/-/winston-transport-4.3.0.tgz", + "integrity": "sha512-B2wPuwUi3vhzn/51Uukcao4dIduEiPOcOt9HJ3QeaXgkJ5Z7UwpBzxS4ZGNHtrxrUvTwemsQiSys0ihOf8Mp1A==", + "requires": { + "readable-stream": "2.3.7", + "triple-beam": "1.3.0" + } + }, + "word-wrap": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", + "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", + "dev": true + }, + "wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha1-J1hIEIkUVqQXHI0CJkQa3pDLyus=", + "dev": true + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" + }, + "write": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/write/-/write-0.2.1.tgz", + "integrity": "sha1-X8A4KOJkzqP+kUVUdvejxWbLB1c=", + "dev": true, + "requires": { + "mkdirp": "0.5.1" + } + }, + "write-file-atomic": { + "version": "2.4.3", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-2.4.3.tgz", + "integrity": "sha512-GaETH5wwsX+GcnzhPgKcKjJ6M2Cq3/iZp1WyY/X1CSqrW+jVNM9Y7D8EC2sM4ZG/V8wZlSniJnCKWPmBYAucRQ==", + "dev": true, + "requires": { + "graceful-fs": "4.2.3", + "imurmurhash": "0.1.4", + "signal-exit": "3.0.2" + } + }, + "ws": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz", + "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==", + "requires": { + "async-limiter": "1.0.1" + } + }, + "xdg-basedir": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-3.0.0.tgz", + "integrity": "sha1-SWsswQnsqNus/i3HK2A8F8WHCtQ=", + "dev": true + }, + "xml2js": { + "version": "0.4.23", + "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", + "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", + "requires": { + "sax": "1.2.4", + "xmlbuilder": "11.0.1" + } + }, + "xmlbuilder": { + "version": "11.0.1", + "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", + "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==" + }, + "yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=", + "dev": true + }, + "yauzl": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/yauzl/-/yauzl-2.4.1.tgz", + "integrity": "sha1-lSj0QtqxsihOWLQ3m7GU4i4MQAU=", + "requires": { + "fd-slicer": "1.0.1" + } + } + } +} diff --git a/projects/web-scraper/package.json b/projects/web-scraper/package.json index 557df9f..259ce46 100644 --- a/projects/web-scraper/package.json +++ b/projects/web-scraper/package.json @@ -4,7 +4,7 @@ "engines": { "node": ">=8.0.0" }, - "description": "Lightweight Node.js API with Docker support, test coverage and circle.ci support.", + "description": "A headless webscraper", "main": "app.js", "scripts": { "start": "node app.js", @@ -30,7 +30,6 @@ ], "author": "hackforla", "license": "GPL-2.0", - "description": "A headless webscraper", "dependencies": { "async": "^2.5.0", "bcryptjs": "^2.4.3", From ae2bbdb0d71c9036fa87ea75674b6d489c102f2b Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:30:11 -0700 Subject: [PATCH 06/14] Fix typo ;) --- projects/web-scraper/lib/routes.js | 2 +- projects/web-scraper/lib/spider/spiderController.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/projects/web-scraper/lib/routes.js b/projects/web-scraper/lib/routes.js index a23202c..85cfb7e 100644 --- a/projects/web-scraper/lib/routes.js +++ b/projects/web-scraper/lib/routes.js @@ -15,7 +15,7 @@ module.exports = function (app) { const UserController = require('./user/UserController')(app); router.use('/users', UserController); - const SpiderController = require('./user/SpiderController')(app); + const SpiderController = require('./user/spiderController')(app); router.use('/spider', SpiderController); diff --git a/projects/web-scraper/lib/spider/spiderController.js b/projects/web-scraper/lib/spider/spiderController.js index 1a7601f..db9a918 100644 --- a/projects/web-scraper/lib/spider/spiderController.js +++ b/projects/web-scraper/lib/spider/spiderController.js @@ -3,7 +3,7 @@ module.exports = function (app) { var express = require('express'); var SpiderController = express.Router(); - var SpiderProvider = require('./SpiderProvider'); + var SpiderProvider = require('./spiderProvider'); var validateSpider = require('./validateSpider'); var VerifyToken = require(__root + 'auth/VerifyToken')(app); From 272a837170c9cba965fb39e7d08d7b744714cc14 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:31:44 -0700 Subject: [PATCH 07/14] Folders have names. --- projects/web-scraper/lib/routes.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/projects/web-scraper/lib/routes.js b/projects/web-scraper/lib/routes.js index 85cfb7e..c9599c6 100644 --- a/projects/web-scraper/lib/routes.js +++ b/projects/web-scraper/lib/routes.js @@ -15,7 +15,7 @@ module.exports = function (app) { const UserController = require('./user/UserController')(app); router.use('/users', UserController); - const SpiderController = require('./user/spiderController')(app); + const SpiderController = require('./spider/spiderController')(app); router.use('/spider', SpiderController); From 291e3d9f4ba1e6e0e3eed63a3d93c3ad6ccc7aa0 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:33:11 -0700 Subject: [PATCH 08/14] Schemas also have names. --- projects/web-scraper/lib/spider/spider.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/projects/web-scraper/lib/spider/spider.js b/projects/web-scraper/lib/spider/spider.js index dfccfb5..4dffb25 100644 --- a/projects/web-scraper/lib/spider/spider.js +++ b/projects/web-scraper/lib/spider/spider.js @@ -4,6 +4,6 @@ var SpiderSchema = new mongoose.Schema({ description: String, pinned: { type: Boolean, default: false } }); -mongoose.model('Spider', NoteSchema); +mongoose.model('Spider', SpiderSchema); module.exports = mongoose.model('Spider'); \ No newline at end of file From f2229b1f188bf3efdc2eeab8badaafe5370587b5 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:34:24 -0700 Subject: [PATCH 09/14] Validation also has a name. --- projects/web-scraper/lib/spider/spiderController.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/projects/web-scraper/lib/spider/spiderController.js b/projects/web-scraper/lib/spider/spiderController.js index db9a918..a8009a7 100644 --- a/projects/web-scraper/lib/spider/spiderController.js +++ b/projects/web-scraper/lib/spider/spiderController.js @@ -9,7 +9,7 @@ module.exports = function (app) { // CREATES A NEW NOTE - SpiderController.post('/', VerifyToken, validateNote, SpiderProvider.createNote); + SpiderController.post('/', VerifyToken, validateSpider, SpiderProvider.createNote); // RETURNS ALL THE NOTES IN THE DATABASE SpiderController.get('/', SpiderProvider.getNotes); From 2b725f571ffc1aee7916bdaaa3566f957b873b87 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:36:48 -0700 Subject: [PATCH 10/14] Functions also have names. --- projects/web-scraper/lib/spider/spiderController.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/projects/web-scraper/lib/spider/spiderController.js b/projects/web-scraper/lib/spider/spiderController.js index a8009a7..72b899b 100644 --- a/projects/web-scraper/lib/spider/spiderController.js +++ b/projects/web-scraper/lib/spider/spiderController.js @@ -9,7 +9,7 @@ module.exports = function (app) { // CREATES A NEW NOTE - SpiderController.post('/', VerifyToken, validateSpider, SpiderProvider.createNote); + SpiderController.post('/', VerifyToken, validateSpider, SpiderProvider.createSpider); // RETURNS ALL THE NOTES IN THE DATABASE SpiderController.get('/', SpiderProvider.getNotes); From b473836aa175e7f93dc2007bb6622fc9909d64a6 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 13:38:42 -0700 Subject: [PATCH 11/14] There are more Callback functions that ALSO have names :D --- projects/web-scraper/lib/spider/spiderController.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/projects/web-scraper/lib/spider/spiderController.js b/projects/web-scraper/lib/spider/spiderController.js index 72b899b..a07ca42 100644 --- a/projects/web-scraper/lib/spider/spiderController.js +++ b/projects/web-scraper/lib/spider/spiderController.js @@ -12,16 +12,16 @@ module.exports = function (app) { SpiderController.post('/', VerifyToken, validateSpider, SpiderProvider.createSpider); // RETURNS ALL THE NOTES IN THE DATABASE - SpiderController.get('/', SpiderProvider.getNotes); + SpiderController.get('/', SpiderProvider.getSpiders); // GETS A SINGLE NOTE FROM THE DATABASE - SpiderController.get('/:id', SpiderProvider.getNote); + SpiderController.get('/:id', SpiderProvider.getSpiders); // DELETES A NOTE FROM THE DATABASE - SpiderController.delete('/:id', VerifyToken, SpiderProvider.deleteNote); + SpiderController.delete('/:id', VerifyToken, SpiderProvider.deleteSpider); // UPDATES A SINGLE NOTE IN THE DATABASE - SpiderController.put('/:id', VerifyToken, SpiderProvider.putNote); + SpiderController.put('/:id', VerifyToken, SpiderProvider.putSpider); return SpiderController; From 65fac086ab3262fdf1d42c841b1477d96a8905f7 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 14:03:28 -0700 Subject: [PATCH 12/14] Folder using proper naming conventions. Promises giving grief. --- .../lib/spider/spiderController.js | 2 +- .../web-scraper/lib/spider/spiderProvider.js | 23 +++++++++++-------- 2 files changed, 15 insertions(+), 10 deletions(-) diff --git a/projects/web-scraper/lib/spider/spiderController.js b/projects/web-scraper/lib/spider/spiderController.js index a07ca42..32bf9aa 100644 --- a/projects/web-scraper/lib/spider/spiderController.js +++ b/projects/web-scraper/lib/spider/spiderController.js @@ -21,7 +21,7 @@ module.exports = function (app) { SpiderController.delete('/:id', VerifyToken, SpiderProvider.deleteSpider); // UPDATES A SINGLE NOTE IN THE DATABASE - SpiderController.put('/:id', VerifyToken, SpiderProvider.putSpider); + SpiderController.put('/:id', SpiderProvider.putSpider); return SpiderController; diff --git a/projects/web-scraper/lib/spider/spiderProvider.js b/projects/web-scraper/lib/spider/spiderProvider.js index 4e52e99..da85440 100644 --- a/projects/web-scraper/lib/spider/spiderProvider.js +++ b/projects/web-scraper/lib/spider/spiderProvider.js @@ -108,10 +108,10 @@ let xmlInjest = (siteName, path) => { result.urlset.url.forEach((item, index) => { items.push(async () => { //console.log(item.loc[0]); - start(item.loc[0]); + start(item.loc[0], siteName); }); }); - console.log(JSON.stringify(items[0])); + //console.log(JSON.stringify(items[0])); // Read X items, Y at a time. createQueue(items.slice(0,2), 1); }); @@ -159,12 +159,17 @@ function putSpider(req, res, next) { // .catch(err => next(new Error(err))); //start('https://www.lamayor.org/', 'www'); - return { - 'worker': xmlInjest('www-lamayor', '/siteMap.xml'), - 'input': { - 'params': req.params, - 'body': req.body - } - }; + let stuff = xmlInjest('www-lamayor', '/siteMap.xml').then((data) => { + console.log(data, 'fin ?'); + + return { + 'worker': data, + 'input': { + 'params': req.params, + 'body': req.body + } + }; + }) + } From 0b1f51df0b012f51bc59d21301a1412b27bc0936 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 15:05:34 -0700 Subject: [PATCH 13/14] Abstracted inputs and promise returns for callback work --- .../web-scraper/lib/spider/spiderProvider.js | 68 ++++++++++--------- 1 file changed, 36 insertions(+), 32 deletions(-) diff --git a/projects/web-scraper/lib/spider/spiderProvider.js b/projects/web-scraper/lib/spider/spiderProvider.js index da85440..ba5bd2f 100644 --- a/projects/web-scraper/lib/spider/spiderProvider.js +++ b/projects/web-scraper/lib/spider/spiderProvider.js @@ -28,7 +28,8 @@ async function start(urlToFetch, siteName) { const status = response.status; page.on( "console", function (log) { - console.log(log.text()); + // This is for debugging puppeteer + // console.log(log.text()); }); const url = new URL(response.url()); @@ -43,13 +44,13 @@ async function start(urlToFetch, siteName) { let filePath = path.resolve(`./output/` + siteName + '/' + `${url.pathname}`); if (path.extname(url.pathname).trim() === '') { - console.log(url.pathname, `${filePath}/index.html`) + //console.log(url.pathname, `${filePath}/index.html`) filePath = `${filePath}/index.html`; } // Save Data if ((status >= 300) && (status <= 399)) { - console.log('Redirect from', response.url(), 'to', response.headers()['location']); + //console.log('Redirect from', response.url(), 'to', response.headers()['location']); await new Promise(function(resolve, reject) { resolve(null); }); @@ -103,17 +104,23 @@ function createQueue(tasks, maxNumOfWorkers = 4) { // Read sites generated by web indexer. let xmlInjest = (siteName, path) => { let items = []; - fse.readFile(__dirname + path, (err, data) => { - parser.parseString(data, (err, result) => { - result.urlset.url.forEach((item, index) => { - items.push(async () => { - //console.log(item.loc[0]); - start(item.loc[0], siteName); + return new Promise(function(resolve, reject) { + fse.readFile(__dirname + path, (err, data) => { + parser.parseString(data, (err, result) => { + result.urlset.url.forEach((item, index) => { + items.push(async () => { + //console.log(item.loc[0]); + return start(item.loc[0], siteName); + }); + }); + // Read X items, Y at a time. + createQueue(items.slice(0,2), 1).then((data) => { + resolve({ + 'siteName': siteName, + 'path': path + }); }); }); - //console.log(JSON.stringify(items[0])); - // Read X items, Y at a time. - createQueue(items.slice(0,2), 1); }); }); }; @@ -127,29 +134,29 @@ function createSpider(req, res, next) { description : req.body.description, pinned : req.body.pinned }) - .then(note => res.status(200).send(note)) - .catch(err => next(new Error(err))); + .then(note => res.status(200).send(note)) + .catch(err => next(new Error(err))); } function getSpiders(req, res, next) { return Spider.find({}) - .then(Spiders => res.status(200).send(notes)) - .catch(err => next(new Error(err))); + .then(Spiders => res.status(200).send(notes)) + .catch(err => next(new Error(err))); } function getSpider(req, res, next) { return Spider.findById(req.params.id) - .then(Spider => { - if (!Spider) return res.status(404).send('No Spider found.'); - res.status(200).send(Spider); - }) - .catch(err => next(new Error(err))); + .then(Spider => { + if (!Spider) return res.status(404).send('No Spider found.'); + res.status(200).send(Spider); + }) + .catch(err => next(new Error(err))); } function deleteSpider(req, res, next) { return Spider.findByIdAndRemove(req.params.id) - .then(Spider => res.status(200).send(Spider)) - .catch(err => next(new Error(err))); + .then(Spider => res.status(200).send(Spider)) + .catch(err => next(new Error(err))); } function putSpider(req, res, next) { @@ -158,18 +165,15 @@ function putSpider(req, res, next) { // .then(Spider => res.status(200).send(Spider)) // .catch(err => next(new Error(err))); - //start('https://www.lamayor.org/', 'www'); - let stuff = xmlInjest('www-lamayor', '/siteMap.xml').then((data) => { - console.log(data, 'fin ?'); - - return { - 'worker': data, + // examples xmlInjest('www-lamayor', '/siteMap.xml'); + xmlInjest('www-lamayor', '/siteMap.xml').then((data) => { + res.send({ + 'worker': data || ' ??????', 'input': { 'params': req.params, 'body': req.body } - }; - }) - + }); + }); } From 55d8e6ea069f4560578deb65756fe2b9a1bb0554 Mon Sep 17 00:00:00 2001 From: Ryan Busby Date: Fri, 20 Mar 2020 16:45:44 -0700 Subject: [PATCH 14/14] asdfghjkqwerty --- projects/web-scraper/lib/spider/spiderProvider.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/projects/web-scraper/lib/spider/spiderProvider.js b/projects/web-scraper/lib/spider/spiderProvider.js index ba5bd2f..3d04bc3 100644 --- a/projects/web-scraper/lib/spider/spiderProvider.js +++ b/projects/web-scraper/lib/spider/spiderProvider.js @@ -168,7 +168,7 @@ function putSpider(req, res, next) { // examples xmlInjest('www-lamayor', '/siteMap.xml'); xmlInjest('www-lamayor', '/siteMap.xml').then((data) => { res.send({ - 'worker': data || ' ??????', + 'worker': data || '??????', 'input': { 'params': req.params, 'body': req.body