diff --git a/dashboard/.babelrc b/dashboard/.babelrc
new file mode 100644
index 000000000..9390d164e
--- /dev/null
+++ b/dashboard/.babelrc
@@ -0,0 +1,18 @@
+{
+ "presets": [
+ ["env", {
+ "modules": false,
+ "targets": {
+ "browsers": ["> 1%", "last 2 versions", "not ie <= 8"]
+ }
+ }],
+ "stage-2"
+ ],
+ "plugins": ["transform-vue-jsx", "transform-runtime"],
+ "env": {
+ "test": {
+ "presets": ["env", "stage-2"],
+ "plugins": ["transform-vue-jsx", "transform-es2015-modules-commonjs", "dynamic-import-node"]
+ }
+ }
+}
diff --git a/dashboard/.editorconfig b/dashboard/.editorconfig
new file mode 100644
index 000000000..9d08a1a82
--- /dev/null
+++ b/dashboard/.editorconfig
@@ -0,0 +1,9 @@
+root = true
+
+[*]
+charset = utf-8
+indent_style = space
+indent_size = 2
+end_of_line = lf
+insert_final_newline = true
+trim_trailing_whitespace = true
diff --git a/dashboard/.eslintignore b/dashboard/.eslintignore
new file mode 100644
index 000000000..e2192c5c2
--- /dev/null
+++ b/dashboard/.eslintignore
@@ -0,0 +1,5 @@
+/build/
+/config/
+/dist/
+/*.js
+/test/unit/coverage/
diff --git a/dashboard/.eslintrc.js b/dashboard/.eslintrc.js
new file mode 100644
index 000000000..2e2e21183
--- /dev/null
+++ b/dashboard/.eslintrc.js
@@ -0,0 +1,25 @@
+// https://eslint.org/docs/user-guide/configuring
+
+module.exports = {
+ root: true,
+ parser: 'babel-eslint',
+ parserOptions: {
+ sourceType: 'module'
+ },
+ env: {
+ browser: true,
+ },
+ // https://github.com/standard/standard/blob/master/docs/RULES-en.md
+ extends: 'standard',
+ // required to lint *.vue files
+ plugins: [
+ 'html'
+ ],
+ // add your custom rules here
+ rules: {
+ // allow async-await
+ 'generator-star-spacing': 'off',
+ // allow debugger during development
+ 'no-debugger': process.env.NODE_ENV === 'production' ? 'error' : 'off'
+ }
+}
diff --git a/dashboard/.gitignore b/dashboard/.gitignore
new file mode 100644
index 000000000..dfb4167f9
--- /dev/null
+++ b/dashboard/.gitignore
@@ -0,0 +1,17 @@
+.DS_Store
+node_modules/
+/dist/
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+/test/unit/coverage/
+/test/e2e/reports/
+selenium-debug.log
+
+# Editor directories and files
+.idea
+.vscode
+*.suo
+*.ntvs*
+*.njsproj
+*.sln
diff --git a/dashboard/.postcssrc.js b/dashboard/.postcssrc.js
new file mode 100644
index 000000000..eee3e92d7
--- /dev/null
+++ b/dashboard/.postcssrc.js
@@ -0,0 +1,10 @@
+// https://github.com/michael-ciniawsky/postcss-load-config
+
+module.exports = {
+ "plugins": {
+ "postcss-import": {},
+ "postcss-url": {},
+ // to edit target browsers: use "browserslist" field in package.json
+ "autoprefixer": {}
+ }
+}
diff --git a/dashboard/README.md b/dashboard/README.md
new file mode 100644
index 000000000..d371726c4
--- /dev/null
+++ b/dashboard/README.md
@@ -0,0 +1,30 @@
+# dashboard
+
+> CVPM Dashboard
+
+## Build Setup
+
+``` bash
+# install dependencies
+npm install
+
+# serve with hot reload at localhost:8080
+npm run dev
+
+# build for production with minification
+npm run build
+
+# build for production and view the bundle analyzer report
+npm run build --report
+
+# run unit tests
+npm run unit
+
+# run e2e tests
+npm run e2e
+
+# run all tests
+npm test
+```
+
+For a detailed explanation on how things work, check out the [guide](http://vuejs-templates.github.io/webpack/) and [docs for vue-loader](http://vuejs.github.io/vue-loader).
diff --git a/dashboard/config/dev.env.js b/dashboard/config/dev.env.js
new file mode 100644
index 000000000..1e22973ae
--- /dev/null
+++ b/dashboard/config/dev.env.js
@@ -0,0 +1,7 @@
+'use strict'
+const merge = require('webpack-merge')
+const prodEnv = require('./prod.env')
+
+module.exports = merge(prodEnv, {
+ NODE_ENV: '"development"'
+})
diff --git a/dashboard/config/index.js b/dashboard/config/index.js
new file mode 100644
index 000000000..591a543d4
--- /dev/null
+++ b/dashboard/config/index.js
@@ -0,0 +1,76 @@
+'use strict'
+// Template version: 1.2.8
+// see http://vuejs-templates.github.io/webpack for documentation.
+
+const path = require('path')
+
+module.exports = {
+ dev: {
+
+ // Paths
+ assetsSubDirectory: 'static',
+ assetsPublicPath: '/',
+ proxyTable: {},
+
+ // Various Dev Server settings
+ host: 'localhost', // can be overwritten by process.env.HOST
+ port: 8080, // can be overwritten by process.env.PORT, if port is in use, a free one will be determined
+ autoOpenBrowser: false,
+ errorOverlay: true,
+ notifyOnErrors: true,
+ poll: false, // https://webpack.js.org/configuration/dev-server/#devserver-watchoptions-
+
+ // Use Eslint Loader?
+ // If true, your code will be linted during bundling and
+ // linting errors and warnings will be shown in the console.
+ useEslint: true,
+ // If true, eslint errors and warnings will also be shown in the error overlay
+ // in the browser.
+ showEslintErrorsInOverlay: false,
+
+ /**
+ * Source Maps
+ */
+
+ // https://webpack.js.org/configuration/devtool/#development
+ devtool: 'cheap-module-eval-source-map',
+
+ // If you have problems debugging vue-files in devtools,
+ // set this to false - it *may* help
+ // https://vue-loader.vuejs.org/en/options.html#cachebusting
+ cacheBusting: true,
+
+ cssSourceMap: true,
+ },
+
+ build: {
+ // Template for index.html
+ index: path.resolve(__dirname, '../dist/index.html'),
+
+ // Paths
+ assetsRoot: path.resolve(__dirname, '../dist'),
+ assetsSubDirectory: 'static',
+ assetsPublicPath: '/',
+
+ /**
+ * Source Maps
+ */
+
+ productionSourceMap: true,
+ // https://webpack.js.org/configuration/devtool/#production
+ devtool: '#source-map',
+
+ // Gzip off by default as many popular static hosts such as
+ // Surge or Netlify already gzip all static assets for you.
+ // Before setting to `true`, make sure to:
+ // npm install --save-dev compression-webpack-plugin
+ productionGzip: false,
+ productionGzipExtensions: ['js', 'css'],
+
+ // Run the build command with an extra argument to
+ // View the bundle analyzer report after build finishes:
+ // `npm run build --report`
+ // Set to `true` or `false` to always turn it on or off
+ bundleAnalyzerReport: process.env.npm_config_report
+ }
+}
diff --git a/dashboard/config/prod.env.js b/dashboard/config/prod.env.js
new file mode 100644
index 000000000..a6f997616
--- /dev/null
+++ b/dashboard/config/prod.env.js
@@ -0,0 +1,4 @@
+'use strict'
+module.exports = {
+ NODE_ENV: '"production"'
+}
diff --git a/dashboard/config/test.env.js b/dashboard/config/test.env.js
new file mode 100644
index 000000000..c2824a307
--- /dev/null
+++ b/dashboard/config/test.env.js
@@ -0,0 +1,7 @@
+'use strict'
+const merge = require('webpack-merge')
+const devEnv = require('./dev.env')
+
+module.exports = merge(devEnv, {
+ NODE_ENV: '"testing"'
+})
diff --git a/dashboard/index.html b/dashboard/index.html
new file mode 100644
index 000000000..ae779dfd2
--- /dev/null
+++ b/dashboard/index.html
@@ -0,0 +1,13 @@
+
+
+
+
+
+ dashboard
+
+
+
+
+
+
+
diff --git a/dashboard/package.json b/dashboard/package.json
new file mode 100644
index 000000000..83dc742b4
--- /dev/null
+++ b/dashboard/package.json
@@ -0,0 +1,88 @@
+{
+ "name": "dashboard",
+ "version": "1.0.0",
+ "description": "CVPM Dashboard",
+ "author": "Xiaozhe Yao ",
+ "private": true,
+ "scripts": {
+ "dev": "webpack-dev-server --inline --progress --config build/webpack.dev.conf.js",
+ "start": "npm run dev",
+ "unit": "jest --config test/unit/jest.conf.js --coverage",
+ "e2e": "node test/e2e/runner.js",
+ "test": "npm run unit && npm run e2e",
+ "lint": "eslint --ext .js,.vue src test/unit/specs test/e2e/specs",
+ "build": "node build/build.js"
+ },
+ "dependencies": {
+ "vue": "^2.5.2",
+ "vue-router": "^3.0.1",
+ "vuetify": "^1.0.0"
+ },
+ "devDependencies": {
+ "autoprefixer": "^7.1.2",
+ "babel-core": "^6.22.1",
+ "babel-eslint": "^7.1.1",
+ "babel-helper-vue-jsx-merge-props": "^2.0.3",
+ "babel-jest": "^21.0.2",
+ "babel-loader": "^7.1.1",
+ "babel-plugin-dynamic-import-node": "^1.2.0",
+ "babel-plugin-syntax-jsx": "^6.18.0",
+ "babel-plugin-transform-es2015-modules-commonjs": "^6.26.0",
+ "babel-plugin-transform-runtime": "^6.22.0",
+ "babel-plugin-transform-vue-jsx": "^3.5.0",
+ "babel-preset-env": "^1.3.2",
+ "babel-preset-stage-2": "^6.22.0",
+ "babel-register": "^6.22.0",
+ "chalk": "^2.0.1",
+ "chromedriver": "^2.27.2",
+ "copy-webpack-plugin": "^4.0.1",
+ "cross-spawn": "^5.0.1",
+ "css-loader": "^0.28.0",
+ "eslint": "^3.19.0",
+ "eslint-config-standard": "^10.2.1",
+ "eslint-friendly-formatter": "^3.0.0",
+ "eslint-loader": "^1.7.1",
+ "eslint-plugin-html": "^3.0.0",
+ "eslint-plugin-import": "^2.7.0",
+ "eslint-plugin-node": "^5.2.0",
+ "eslint-plugin-promise": "^3.4.0",
+ "eslint-plugin-standard": "^3.0.1",
+ "extract-text-webpack-plugin": "^3.0.0",
+ "file-loader": "^1.1.4",
+ "friendly-errors-webpack-plugin": "^1.6.1",
+ "html-webpack-plugin": "^2.30.1",
+ "jest": "^21.2.0",
+ "jest-serializer-vue": "^0.3.0",
+ "nightwatch": "^0.9.12",
+ "node-notifier": "^5.1.2",
+ "optimize-css-assets-webpack-plugin": "^3.2.0",
+ "ora": "^1.2.0",
+ "portfinder": "^1.0.13",
+ "postcss-import": "^11.0.0",
+ "postcss-loader": "^2.0.8",
+ "postcss-url": "^7.2.1",
+ "rimraf": "^2.6.0",
+ "selenium-server": "^3.0.1",
+ "semver": "^5.3.0",
+ "shelljs": "^0.7.6",
+ "uglifyjs-webpack-plugin": "^1.1.1",
+ "url-loader": "^0.5.8",
+ "vue-jest": "^1.0.2",
+ "vue-loader": "^13.3.0",
+ "vue-style-loader": "^3.0.1",
+ "vue-template-compiler": "^2.5.2",
+ "webpack": "^3.6.0",
+ "webpack-bundle-analyzer": "^2.9.0",
+ "webpack-dev-server": "^2.9.1",
+ "webpack-merge": "^4.1.0"
+ },
+ "engines": {
+ "node": ">= 6.0.0",
+ "npm": ">= 3.0.0"
+ },
+ "browserslist": [
+ "> 1%",
+ "last 2 versions",
+ "not ie <= 8"
+ ]
+}
diff --git a/dashboard/src/.gitrepo b/dashboard/src/.gitrepo
new file mode 100644
index 000000000..8ba7ed824
--- /dev/null
+++ b/dashboard/src/.gitrepo
@@ -0,0 +1,12 @@
+; DO NOT EDIT (unless you know what you are doing)
+;
+; This subdirectory is a git "subrepo", and this file is maintained by the
+; git-subrepo command. See https://github.com/git-commands/git-subrepo#readme
+;
+[subrepo]
+ remote = https://github.com/vuetifyjs/templates-common.git
+ branch = subrepo/webpack-src
+ commit = 090741fa8ba4da0c6f85db64eff64550704123e1
+ parent = e05204fc0583a8c99f1963ce873eba1266838215
+ method = merge
+ cmdver = 0.4.0
diff --git a/dashboard/src/App.vue b/dashboard/src/App.vue
new file mode 100644
index 000000000..4143476bf
--- /dev/null
+++ b/dashboard/src/App.vue
@@ -0,0 +1,91 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ web
+
+
+ remove
+
+
+
+
+ menu
+
+
+
+
+
+
+
+
+
+ compare_arrows
+
+ Switch drawer (click me)
+
+
+
+
+ © 2017
+
+
+
+
+
diff --git a/dashboard/src/assets/logo.png b/dashboard/src/assets/logo.png
new file mode 100755
index 000000000..a2ce2353f
Binary files /dev/null and b/dashboard/src/assets/logo.png differ
diff --git a/dashboard/src/components/HelloWorld.vue b/dashboard/src/components/HelloWorld.vue
new file mode 100644
index 000000000..6bbe5cdbe
--- /dev/null
+++ b/dashboard/src/components/HelloWorld.vue
@@ -0,0 +1,35 @@
+
+
+
+
+
+
+ “First, solve the problem. Then, write the code.”
+
+
+
+
+
+
+
+
+
diff --git a/dashboard/src/main.js b/dashboard/src/main.js
new file mode 100644
index 000000000..18df910c8
--- /dev/null
+++ b/dashboard/src/main.js
@@ -0,0 +1,19 @@
+// The Vue build version to load with the `import` command
+// (runtime-only or standalone) has been set in webpack.base.conf with an alias.
+import Vue from 'vue'
+import App from './App'
+import router from './router'
+import Vuetify from 'vuetify'
+import 'vuetify/dist/vuetify.min.css'
+
+Vue.use(Vuetify)
+
+Vue.config.productionTip = false
+
+/* eslint-disable no-new */
+new Vue({
+ el: '#app',
+ router,
+ components: { App },
+ template: ''
+})
diff --git a/dashboard/src/router/index.js b/dashboard/src/router/index.js
new file mode 100644
index 000000000..5fa7f9d31
--- /dev/null
+++ b/dashboard/src/router/index.js
@@ -0,0 +1,15 @@
+import Vue from 'vue'
+import Router from 'vue-router'
+import HelloWorld from '@/components/HelloWorld'
+
+Vue.use(Router)
+
+export default new Router({
+ routes: [
+ {
+ path: '/',
+ name: 'HelloWorld',
+ component: HelloWorld
+ }
+ ]
+})
diff --git a/dashboard/static/.gitkeep b/dashboard/static/.gitkeep
new file mode 100644
index 000000000..e69de29bb
diff --git a/dashboard/test/e2e/custom-assertions/elementCount.js b/dashboard/test/e2e/custom-assertions/elementCount.js
new file mode 100644
index 000000000..818e60209
--- /dev/null
+++ b/dashboard/test/e2e/custom-assertions/elementCount.js
@@ -0,0 +1,27 @@
+// A custom Nightwatch assertion.
+// The assertion name is the filename.
+// Example usage:
+//
+// browser.assert.elementCount(selector, count)
+//
+// For more information on custom assertions see:
+// http://nightwatchjs.org/guide#writing-custom-assertions
+
+exports.assertion = function (selector, count) {
+ this.message = 'Testing if element <' + selector + '> has count: ' + count
+ this.expected = count
+ this.pass = function (val) {
+ return val === this.expected
+ }
+ this.value = function (res) {
+ return res.value
+ }
+ this.command = function (cb) {
+ var self = this
+ return this.api.execute(function (selector) {
+ return document.querySelectorAll(selector).length
+ }, [selector], function (res) {
+ cb.call(self, res)
+ })
+ }
+}
diff --git a/dashboard/test/e2e/nightwatch.conf.js b/dashboard/test/e2e/nightwatch.conf.js
new file mode 100644
index 000000000..f019c0ac4
--- /dev/null
+++ b/dashboard/test/e2e/nightwatch.conf.js
@@ -0,0 +1,46 @@
+require('babel-register')
+var config = require('../../config')
+
+// http://nightwatchjs.org/gettingstarted#settings-file
+module.exports = {
+ src_folders: ['test/e2e/specs'],
+ output_folder: 'test/e2e/reports',
+ custom_assertions_path: ['test/e2e/custom-assertions'],
+
+ selenium: {
+ start_process: true,
+ server_path: require('selenium-server').path,
+ host: '127.0.0.1',
+ port: 4444,
+ cli_args: {
+ 'webdriver.chrome.driver': require('chromedriver').path
+ }
+ },
+
+ test_settings: {
+ default: {
+ selenium_port: 4444,
+ selenium_host: 'localhost',
+ silent: true,
+ globals: {
+ devServerURL: 'http://localhost:' + (process.env.PORT || config.dev.port)
+ }
+ },
+
+ chrome: {
+ desiredCapabilities: {
+ browserName: 'chrome',
+ javascriptEnabled: true,
+ acceptSslCerts: true
+ }
+ },
+
+ firefox: {
+ desiredCapabilities: {
+ browserName: 'firefox',
+ javascriptEnabled: true,
+ acceptSslCerts: true
+ }
+ }
+ }
+}
diff --git a/dashboard/test/e2e/runner.js b/dashboard/test/e2e/runner.js
new file mode 100644
index 000000000..3a02f8e96
--- /dev/null
+++ b/dashboard/test/e2e/runner.js
@@ -0,0 +1,48 @@
+// 1. start the dev server using production config
+process.env.NODE_ENV = 'testing'
+
+const webpack = require('webpack')
+const DevServer = require('webpack-dev-server')
+
+const webpackConfig = require('../../build/webpack.prod.conf')
+const devConfigPromise = require('../../build/webpack.dev.conf')
+
+let server
+
+devConfigPromise.then(devConfig => {
+ const devServerOptions = devConfig.devServer
+ const compiler = webpack(webpackConfig)
+ server = new DevServer(compiler, devServerOptions)
+ const port = devServerOptions.port
+ const host = devServerOptions.host
+ return server.listen(port, host)
+})
+.then(() => {
+ // 2. run the nightwatch test suite against it
+ // to run in additional browsers:
+ // 1. add an entry in test/e2e/nightwatch.conf.json under "test_settings"
+ // 2. add it to the --env flag below
+ // or override the environment flag, for example: `npm run e2e -- --env chrome,firefox`
+ // For more information on Nightwatch's config file, see
+ // http://nightwatchjs.org/guide#settings-file
+ let opts = process.argv.slice(2)
+ if (opts.indexOf('--config') === -1) {
+ opts = opts.concat(['--config', 'test/e2e/nightwatch.conf.js'])
+ }
+ if (opts.indexOf('--env') === -1) {
+ opts = opts.concat(['--env', 'chrome'])
+ }
+
+ const spawn = require('cross-spawn')
+ const runner = spawn('./node_modules/.bin/nightwatch', opts, { stdio: 'inherit' })
+
+ runner.on('exit', function (code) {
+ server.close()
+ process.exit(code)
+ })
+
+ runner.on('error', function (err) {
+ server.close()
+ throw err
+ })
+})
diff --git a/dashboard/test/e2e/specs/test.js b/dashboard/test/e2e/specs/test.js
new file mode 100644
index 000000000..a7b1bd920
--- /dev/null
+++ b/dashboard/test/e2e/specs/test.js
@@ -0,0 +1,19 @@
+// For authoring Nightwatch tests, see
+// http://nightwatchjs.org/guide#usage
+
+module.exports = {
+ 'default e2e tests': function (browser) {
+ // automatically uses dev Server port from /config.index.js
+ // default: http://localhost:8080
+ // see nightwatch.conf.js
+ const devServer = browser.globals.devServerURL
+
+ browser
+ .url(devServer)
+ .waitForElementVisible('#app', 5000)
+ .assert.elementPresent('.hello')
+ .assert.containsText('h1', 'Welcome to Your Vue.js App')
+ .assert.elementCount('img', 1)
+ .end()
+ }
+}
diff --git a/dashboard/test/unit/.eslintrc b/dashboard/test/unit/.eslintrc
new file mode 100644
index 000000000..4d5d98f3e
--- /dev/null
+++ b/dashboard/test/unit/.eslintrc
@@ -0,0 +1,7 @@
+{
+ "env": {
+ "jest": true
+ },
+ "globals": {
+ }
+}
diff --git a/dashboard/test/unit/jest.conf.js b/dashboard/test/unit/jest.conf.js
new file mode 100644
index 000000000..4f7da1519
--- /dev/null
+++ b/dashboard/test/unit/jest.conf.js
@@ -0,0 +1,30 @@
+const path = require('path')
+
+module.exports = {
+ rootDir: path.resolve(__dirname, '../../'),
+ moduleFileExtensions: [
+ 'js',
+ 'json',
+ 'vue'
+ ],
+ moduleNameMapper: {
+ '^@/(.*)$': '/src/$1'
+ },
+ transform: {
+ '^.+\\.js$': '/node_modules/babel-jest',
+ '.*\\.(vue)$': '/node_modules/vue-jest'
+ },
+ testPathIgnorePatterns: [
+ '/test/e2e'
+ ],
+ snapshotSerializers: ['/node_modules/jest-serializer-vue'],
+ setupFiles: ['/test/unit/setup'],
+ mapCoverage: true,
+ coverageDirectory: '/test/unit/coverage',
+ collectCoverageFrom: [
+ 'src/**/*.{js,vue}',
+ '!src/main.js',
+ '!src/router/index.js',
+ '!**/node_modules/**'
+ ]
+}
diff --git a/dashboard/test/unit/setup.js b/dashboard/test/unit/setup.js
new file mode 100644
index 000000000..edbdbae81
--- /dev/null
+++ b/dashboard/test/unit/setup.js
@@ -0,0 +1,3 @@
+import Vue from 'vue'
+
+Vue.config.productionTip = false
diff --git a/dashboard/test/unit/specs/HelloWorld.spec.js b/dashboard/test/unit/specs/HelloWorld.spec.js
new file mode 100644
index 000000000..d3e6a58e9
--- /dev/null
+++ b/dashboard/test/unit/specs/HelloWorld.spec.js
@@ -0,0 +1,11 @@
+import Vue from 'vue'
+import HelloWorld from '@/components/HelloWorld'
+
+describe('HelloWorld.vue', () => {
+ it('should render correct contents', () => {
+ const Constructor = Vue.extend(HelloWorld)
+ const vm = new Constructor().$mount()
+ expect(vm.$el.querySelector('.hello h1').textContent)
+ .toEqual('Welcome to Your Vue.js App')
+ })
+})