diff --git a/dashboard-api/.gitignore b/dashboard-api/.gitignore new file mode 100644 index 0000000000..ce8000b297 --- /dev/null +++ b/dashboard-api/.gitignore @@ -0,0 +1,82 @@ +#### joe made this: http://goel.io/joe + +#### node #### +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# Runtime data +pids +*.pid +*.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover +lib-cov + +# Coverage directory used by tools like istanbul +coverage + +# nyc test coverage +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) +.grunt + +# Bower dependency directory (https://bower.io/) +bower_components + +# node-waf configuration +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) +build/Release + +# Dependency directories +node_modules/ +jspm_packages/ + +# TypeScript v1 declaration files +typings/ + +# Optional npm cache directory +.npm + +# Optional eslint cache +.eslintcache + +# Optional REPL history +.node_repl_history + +# Output of 'npm pack' +*.tgz + +# Yarn Integrity file +.yarn-integrity + +# dotenv environment variables file +.env + +# parcel-bundler cache (https://parceljs.org/) +.cache + +# next.js build output +.next + +# nuxt.js build output +.nuxt + +# vuepress build output +.vuepress/dist + +# Serverless directories +.serverless/ + +# FuseBox cache +.fusebox/ + +#DynamoDB Local files +.dynamodb/ diff --git a/dashboard-api/package-lock.json b/dashboard-api/package-lock.json new file mode 100644 index 0000000000..11f0f7f03d --- /dev/null +++ b/dashboard-api/package-lock.json @@ -0,0 +1,357 @@ +{ + "name": "@freecodecamp/mods-dashboard-server", + "requires": true, + "lockfileVersion": 1, + "dependencies": { + "accepts": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.5.tgz", + "integrity": "sha1-63d99gEXI6OxTopywIBcjoZ0a9I=", + "requires": { + "mime-types": "~2.1.18", + "negotiator": "0.6.1" + } + }, + "array-flatten": { + "version": "1.1.1", + "resolved": "http://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" + }, + "body-parser": { + "version": "1.18.3", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.3.tgz", + "integrity": "sha1-WykhmP/dVTs6DyDe0FkrlWlVyLQ=", + "requires": { + "bytes": "3.0.0", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "~1.1.2", + "http-errors": "~1.6.3", + "iconv-lite": "0.4.23", + "on-finished": "~2.3.0", + "qs": "6.5.2", + "raw-body": "2.3.3", + "type-is": "~1.6.16" + } + }, + "bytes": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", + "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=" + }, + "content-disposition": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz", + "integrity": "sha1-DPaLud318r55YcOoUXjLhdunjLQ=" + }, + "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.3.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", + "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=" + }, + "cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" + }, + "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" + } + }, + "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=" + }, + "ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" + }, + "encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" + }, + "escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" + }, + "etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" + }, + "express": { + "version": "4.16.4", + "resolved": "https://registry.npmjs.org/express/-/express-4.16.4.tgz", + "integrity": "sha512-j12Uuyb4FMrd/qQAm6uCHAkPtO8FDTRJZBDd5D2KOL2eLaz1yUNdUB/NOIyq0iU4q4cFarsUCrnFDPBcnksuOg==", + "requires": { + "accepts": "~1.3.5", + "array-flatten": "1.1.1", + "body-parser": "1.18.3", + "content-disposition": "0.5.2", + "content-type": "~1.0.4", + "cookie": "0.3.1", + "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.1", + "fresh": "0.5.2", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "~2.3.0", + "parseurl": "~1.3.2", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.4", + "qs": "6.5.2", + "range-parser": "~1.2.0", + "safe-buffer": "5.1.2", + "send": "0.16.2", + "serve-static": "1.13.2", + "setprototypeof": "1.1.0", + "statuses": "~1.4.0", + "type-is": "~1.6.16", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + } + }, + "finalhandler": { + "version": "1.1.1", + "resolved": "http://registry.npmjs.org/finalhandler/-/finalhandler-1.1.1.tgz", + "integrity": "sha512-Y1GUDo39ez4aHAw7MysnUD5JzYX+WaIj8I57kO3aEPT1fFRL4sr7mjei97FgnwhAyyzRYmQZaTHb2+9uZ1dPtg==", + "requires": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.2", + "statuses": "~1.4.0", + "unpipe": "~1.0.0" + } + }, + "forwarded": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", + "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" + }, + "fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" + }, + "http-errors": { + "version": "1.6.3", + "resolved": "http://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", + "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.0", + "statuses": ">= 1.4.0 < 2" + } + }, + "iconv-lite": { + "version": "0.4.23", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.23.tgz", + "integrity": "sha512-neyTUVFtahjf0mB3dZT77u+8O0QB89jFdnBkd5P1JgYPbPaia3gXXOVL2fq8VyU2gMMD7SaN7QukTB/pmXYvDA==", + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" + }, + "ipaddr.js": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.8.0.tgz", + "integrity": "sha1-6qM9bd16zo9/b+DJygRA5wZzix4=" + }, + "media-typer": { + "version": "0.3.0", + "resolved": "http://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" + }, + "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=" + }, + "mime": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.1.tgz", + "integrity": "sha512-KI1+qOZu5DcW6wayYHSzR/tXKCDC5Om4s1z2QJjDULzLcmf3DvzS7oluY4HCTrc+9FiKmWUgeNLg7W3uIQvxtQ==" + }, + "mime-db": { + "version": "1.37.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.37.0.tgz", + "integrity": "sha512-R3C4db6bgQhlIhPU48fUtdVmKnflq+hRdad7IyKhtFj06VPNVdk2RhiYL3UjQIlso8L+YxAtFkobT0VK+S/ybg==" + }, + "mime-types": { + "version": "2.1.21", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.21.tgz", + "integrity": "sha512-3iL6DbwpyLzjR3xHSFNFeb9Nz/M8WDkX33t1GFQnFOllWk8pOrh/LSrB5OXlnlW5P9LH73X6loW/eogc+F5lJg==", + "requires": { + "mime-db": "~1.37.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "negotiator": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.1.tgz", + "integrity": "sha1-KzJxhOiZIQEXeyhWP7XnECrNDKk=" + }, + "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" + } + }, + "parseurl": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.2.tgz", + "integrity": "sha1-/CidTtiZMRlGDBViUyYs3I3mW/M=" + }, + "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=" + }, + "proxy-addr": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.4.tgz", + "integrity": "sha512-5erio2h9jp5CHGwcybmxmVqHmnCBZeewlfJ0pex+UW7Qny7OOZXTtH56TGNyBizkgiOwhJtMKrVzDTeKcySZwA==", + "requires": { + "forwarded": "~0.1.2", + "ipaddr.js": "1.8.0" + } + }, + "qs": { + "version": "6.5.2", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", + "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==" + }, + "range-parser": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.0.tgz", + "integrity": "sha1-9JvmtIeJTdxA3MlKMi9hEJLgDV4=" + }, + "raw-body": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.3.tgz", + "integrity": "sha512-9esiElv1BrZoI3rCDuOuKCBRbuApGGaDPQfjSflGxdy4oyzqghxu6klEkkVIvBje+FF0BX9coEv8KqW6X/7njw==", + "requires": { + "bytes": "3.0.0", + "http-errors": "1.6.3", + "iconv-lite": "0.4.23", + "unpipe": "1.0.0" + } + }, + "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==" + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "send": { + "version": "0.16.2", + "resolved": "https://registry.npmjs.org/send/-/send-0.16.2.tgz", + "integrity": "sha512-E64YFPUssFHEFBvpbbjr44NCLtI1AohxQ8ZSiJjQLskAdKuriYEP6VyGEsRDH8ScozGpkaX1BGvhanqCwkcEZw==", + "requires": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.6.2", + "mime": "1.4.1", + "ms": "2.0.0", + "on-finished": "~2.3.0", + "range-parser": "~1.2.0", + "statuses": "~1.4.0" + } + }, + "serve-static": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.13.2.tgz", + "integrity": "sha512-p/tdJrO4U387R9oMjb1oj7qSMaMfmOyd4j9hOFoxZe2baQszgHcSWjuya/CiT5kgZZKRudHNOA0pYXOl8rQ5nw==", + "requires": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.2", + "send": "0.16.2" + } + }, + "setprototypeof": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", + "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==" + }, + "statuses": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.4.0.tgz", + "integrity": "sha512-zhSCtt8v2NDrRlPQpCNtw/heZLtfUDqxBM1udqikb/Hbk52LK4nQSwr10u77iopCW5LsyHpuXS0GnEc48mLeew==" + }, + "type-is": { + "version": "1.6.16", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.16.tgz", + "integrity": "sha512-HRkVv/5qY2G6I8iab9cI7v1bOIdhm94dVjQCPFElW9W+3GeDOSHmy2EBYe4VTApuzolPcmgFTN3ftVJRKR2J9Q==", + "requires": { + "media-typer": "0.3.0", + "mime-types": "~2.1.18" + } + }, + "unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" + }, + "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=" + } + } +} diff --git a/dashboard-api/package.json b/dashboard-api/package.json new file mode 100644 index 0000000000..6cf126dfdb --- /dev/null +++ b/dashboard-api/package.json @@ -0,0 +1,13 @@ +{ + "name": "@freecodecamp/mods-dashboard-server", + "private": true, + "main": "server.js", + "scripts": { + "start": "node server.js" + }, + "dependencies": { + "express": "^4.16.3" + }, + "devDependencies": {}, + "license": "BSD-3-Clause" +} diff --git a/dashboard-api/routes/catchAll.js b/dashboard-api/routes/catchAll.js new file mode 100644 index 0000000000..c87e1bb1fc --- /dev/null +++ b/dashboard-api/routes/catchAll.js @@ -0,0 +1,8 @@ +const { indices, prs } = require('../data.json'); +const router = require('express').Router(); + +router.get('/', (request, response) => { + response.json({ ok: true, foundPRs: [] }); +}); + +module.exports = router; \ No newline at end of file diff --git a/dashboard-api/routes/getCurrData.js b/dashboard-api/routes/getCurrData.js new file mode 100644 index 0000000000..950924bdeb --- /dev/null +++ b/dashboard-api/routes/getCurrData.js @@ -0,0 +1,8 @@ +const data = require('../data.json'); +const router = require('express').Router(); + +router.get('/', (request, response) => { + response.json(data); +}); + +module.exports = router; \ No newline at end of file diff --git a/dashboard-api/routes/index.js b/dashboard-api/routes/index.js new file mode 100644 index 0000000000..775927d31a --- /dev/null +++ b/dashboard-api/routes/index.js @@ -0,0 +1,8 @@ +const catchAll = require('./catchAll'); +const pareto = require('./pareto'); +const pr = require('./pr'); +const search = require('./search'); +const info = require('./info'); +const getCurrData = require('./getCurrData'); + +module.exports = { catchAll, pareto, pr, search, info, getCurrData }; \ No newline at end of file diff --git a/dashboard-api/routes/info.js b/dashboard-api/routes/info.js new file mode 100644 index 0000000000..0ef240aefc --- /dev/null +++ b/dashboard-api/routes/info.js @@ -0,0 +1,15 @@ +const { prs, startTime } = require('../data.json'); +const router = require('express').Router(); + +const firstPR = prs[0].number; +const lastPR = prs[prs.length - 1].number; +router.get('/', (request, response) => { + response.json({ + ok: true, + lastUpdate: startTime, + numPRs: prs.length, + prRange: `${firstPR}-${lastPR}` + }); +}); + +module.exports = router; \ No newline at end of file diff --git a/dashboard-api/routes/pareto.js b/dashboard-api/routes/pareto.js new file mode 100644 index 0000000000..03bfb8959b --- /dev/null +++ b/dashboard-api/routes/pareto.js @@ -0,0 +1,31 @@ +const router = require('express').Router(); + +const { indices, prs } = require('../data.json'); + +const reportObj = prs.reduce((obj, pr) => { + const { number, filenames, username } = pr; + + filenames.forEach((filename) => { + if (obj[filename]) { + const { count, prs } = obj[filename]; + obj[filename] = { count: count + 1, prs: prs.concat({ number, username } ) }; + } + else { + obj[filename] = { count: 1, prs: [ { number, username } ] }; + } + }); + return obj; +}, {}); +const pareto = Object.keys(reportObj) + .map((filename) => { + const { count, prs } = reportObj[filename]; + return { filename, count, prs }; + }) + .sort((a, b) => b.count - a.count); + + +router.get('/', (reqeust, response) => { + response.json({ ok: true, pareto }); +}); + +module.exports = router; \ No newline at end of file diff --git a/dashboard-api/routes/pr.js b/dashboard-api/routes/pr.js new file mode 100644 index 0000000000..418288914a --- /dev/null +++ b/dashboard-api/routes/pr.js @@ -0,0 +1,37 @@ +const { indices, prs } = require('../data.json'); +const router = require('express').Router(); + +router.get('/:number', (request, response) => { + const { number: refNumber } = request.params; + const index = indices[refNumber]; + + if (!index) { + response.json({ ok: true, message: 'Not a valid PR #.', results: [] }); + return; + } + + const pr = prs[index]; + const results = []; + const { filenames: refFilenames } = pr; + + prs.forEach(({ number, filenames, username }) => { + if (number != refNumber) { + const matchedFilenames = filenames.filter((filename) => { + return refFilenames.includes(filename); + }); + + if (matchedFilenames.length) { + results.push({ number, filenames: matchedFilenames, username }); + } + } + }); + + if (!results.length) { + response.json({ ok: true, message: 'No matching results.', results: [] }); + return; + } + + response.json({ ok: true, results }); +}); + +module.exports = router; \ No newline at end of file diff --git a/dashboard-api/routes/search.js b/dashboard-api/routes/search.js new file mode 100644 index 0000000000..dd69395132 --- /dev/null +++ b/dashboard-api/routes/search.js @@ -0,0 +1,42 @@ + +const { indices, prs } = require('../data.json'); +const router = require('express').Router(); + +router.get('/', (request, response) => { + const value = request.query.value; + + if (value) { + const filesFound = {}; + + prs.forEach(({ number, filenames }) => { + filenames.forEach((filename) => { + if (filename.toLowerCase().includes(value.toLowerCase())) { + const prObj = { + number, + fileCount: prs[indices[number]].filenames.length + }; + + if (filesFound.hasOwnProperty(filename)) { + filesFound[filename].push(prObj); + } + else { + filesFound[filename] = [prObj] + } + } + }); + }); + + let results = Object.keys(filesFound) + .map((filename) => ({ filename, prs: filesFound[filename] })) + .sort((a, b) => a.filename === b.filename ? 0 : a.filename < b.filename ? -1 : 1); + + if (!results.length) { + response.json({ ok: true, message: 'No matching results.', results: [] }); + return; + } + + response.json({ ok: true, results }); + } +}); + +module.exports = router; \ No newline at end of file diff --git a/dashboard-api/server.js b/dashboard-api/server.js new file mode 100644 index 0000000000..d0e7ad021c --- /dev/null +++ b/dashboard-api/server.js @@ -0,0 +1,26 @@ +const fs = require('fs'); +const express = require('express'); + +const app = express(); +const { catchAll, pareto, pr, search, info, getCurrData } = require('./routes'); + +app.use(express.static('public')); +app.use((request, response, next) => { + response.header('Access-Control-Allow-Origin', '*'); + response.header('Access-Control-Allow-Headers', 'Origin, X-Requested-With, Content-Type, Accept'); + response.header('Access-Control-Allow-Methods', 'GET'); + next(); +}); + +app.get('/', (request, response) => response.sendFile(__dirname + '/views/index.html')); + +app.use('/pr', pr); +app.use('/search', search); +app.use('/pareto', pareto); +app.use('/info', info); +app.use('/getCurrData', getCurrData); +app.use('*', catchAll); + +const listener = app.listen(process.env.PORT, () => { + console.log('Your app is listening on port ' + listener.address().port); +}); \ No newline at end of file diff --git a/lerna.json b/lerna.json index 809c3a0b6e..237b9d4985 100644 --- a/lerna.json +++ b/lerna.json @@ -1,6 +1,8 @@ { "packages": [ - "sweeper" + "sweeper", + "dashboard-api", + "dashboard-client" ], "version": "independent" } diff --git a/package.json b/package.json index e78b3ccfe6..3aa0db9756 100644 --- a/package.json +++ b/package.json @@ -1,5 +1,6 @@ { "name": "@freecodecamp/github-tools", + "description": "freeCodeCamp.org's tools for helping maintain its Open Source codebase", "private": true, "scripts": { "postinstall": "lerna bootstrap" diff --git a/sweeper/package-lock.json b/sweeper/package-lock.json index 904ebed5f9..3cabab7086 100644 --- a/sweeper/package-lock.json +++ b/sweeper/package-lock.json @@ -1,8 +1,7 @@ { "name": "@freecodecamp/sweeper", - "version": "1.0.0", - "lockfileVersion": 1, "requires": true, + "lockfileVersion": 1, "dependencies": { "@octokit/rest": { "version": "15.18.0", diff --git a/sweeper/package.json b/sweeper/package.json index e16f34015d..b9ff496cdd 100644 --- a/sweeper/package.json +++ b/sweeper/package.json @@ -1,7 +1,6 @@ { "name": "@freecodecamp/sweeper", - "version": "1.0.0", - "description": "freeCodeCamp.org's tools for helping maintain its Open Source codebase", + "private": true, "main": "index.js", "dependencies": { "@octokit/rest": "^15.18.0", @@ -20,14 +19,5 @@ "scripts": { "test": "echo \"Error: no test specified\" && exit 1" }, - "repository": { - "type": "git", - "url": "git+https://github.com/freeCodeCamp/github-tools.git" - }, - "author": "", - "license": "BSD-3-Clause", - "bugs": { - "url": "https://github.com/freeCodeCamp/github-tools/issues" - }, - "homepage": "https://github.com/freeCodeCamp/github-tools#readme" + "license": "BSD-3-Clause" }