mirror of
https://github.com/pa11y/pa11y-dashboard.git
synced 2025-09-25 14:51:28 +00:00
Compare commits
35 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
fe2630e9cc | ||
![]() |
1794e72d0a | ||
![]() |
2ebcafece3 | ||
![]() |
aadb791d47 | ||
![]() |
73b20f14f0 | ||
![]() |
eff26a55a4 | ||
![]() |
ea48d6a6fd | ||
![]() |
90328cfa39 | ||
![]() |
d766029bbb | ||
![]() |
7ea14497d9 | ||
![]() |
4675d2200c | ||
![]() |
f00a1ec426 | ||
![]() |
3841c36889 | ||
![]() |
f39df5b33e | ||
![]() |
6575c7cf6e | ||
![]() |
708e4b997e | ||
![]() |
573b624a32 | ||
![]() |
129e48e0ce | ||
![]() |
ca9551a3e6 | ||
![]() |
d93f8af20e | ||
![]() |
cc7f769653 | ||
![]() |
fc510b328a | ||
![]() |
77ed85eb87 | ||
![]() |
816eecb256 | ||
![]() |
ed1b9830fd | ||
![]() |
b5d3542670 | ||
![]() |
31bbb7229b | ||
![]() |
27cdf51258 | ||
![]() |
b3b2cd21da | ||
![]() |
537b93c671 | ||
![]() |
32a8ec55e1 | ||
![]() |
0829482020 | ||
![]() |
eb8f35779e | ||
![]() |
3cdbba00c1 | ||
![]() |
fd2dd62478 |
88
.jscsrc
Normal file
88
.jscsrc
Normal file
@@ -0,0 +1,88 @@
|
||||
{
|
||||
"disallowEmptyBlocks": true,
|
||||
"disallowImplicitTypeConversion": [
|
||||
"binary",
|
||||
"numeric",
|
||||
"string"
|
||||
],
|
||||
"disallowKeywordsOnNewLine": [
|
||||
"catch",
|
||||
"else"
|
||||
],
|
||||
"disallowMixedSpacesAndTabs": true,
|
||||
"disallowMultipleSpaces": true,
|
||||
"disallowMultipleVarDecl": true,
|
||||
"disallowNewlineBeforeBlockStatements": true,
|
||||
"disallowQuotedKeysInObjects": true,
|
||||
"disallowSpaceAfterObjectKeys": true,
|
||||
"disallowSpaceAfterPrefixUnaryOperators": true,
|
||||
"disallowSpaceBeforeComma": true,
|
||||
"disallowSpaceBeforeSemicolon": true,
|
||||
"disallowSpacesInCallExpression": true,
|
||||
"disallowSpacesInFunction": {
|
||||
"beforeOpeningRoundBrace": true
|
||||
},
|
||||
"disallowSpacesInsideArrayBrackets": true,
|
||||
"disallowSpacesInsideBrackets": true,
|
||||
"disallowSpacesInsideObjectBrackets": true,
|
||||
"disallowSpacesInsideParentheses": true,
|
||||
"disallowTrailingComma": true,
|
||||
"disallowTrailingWhitespace": true,
|
||||
"disallowYodaConditions": true,
|
||||
"maximumLineLength": 200,
|
||||
"requireBlocksOnNewline": true,
|
||||
"requireCapitalizedConstructors": true,
|
||||
"requireCommaBeforeLineBreak": true,
|
||||
"requireCurlyBraces": [
|
||||
"if",
|
||||
"else",
|
||||
"for",
|
||||
"while",
|
||||
"do",
|
||||
"switch",
|
||||
"try",
|
||||
"catch"
|
||||
],
|
||||
"requireDotNotation": true,
|
||||
"requireLineBreakAfterVariableAssignment": true,
|
||||
"requireLineFeedAtFileEnd": true,
|
||||
"requireObjectKeysOnNewLine": true,
|
||||
"requireParenthesesAroundIIFE": true,
|
||||
"requireSemicolons": true,
|
||||
"requireSpaceAfterBinaryOperators": true,
|
||||
"requireSpaceAfterKeywords": [
|
||||
"if",
|
||||
"else",
|
||||
"for",
|
||||
"while",
|
||||
"do",
|
||||
"switch",
|
||||
"try",
|
||||
"catch"
|
||||
],
|
||||
"requireSpaceAfterLineComment": {
|
||||
"allExcept": [
|
||||
"="
|
||||
]
|
||||
},
|
||||
"requireSpaceBeforeBinaryOperators": true,
|
||||
"requireSpaceBeforeBlockStatements": true,
|
||||
"requireSpaceBeforeObjectValues": true,
|
||||
"requireSpaceBetweenArguments": true,
|
||||
"requireSpacesInConditionalExpression": true,
|
||||
"requireSpacesInForStatement": true,
|
||||
"requireSpacesInFunction": {
|
||||
"beforeOpeningCurlyBrace": true
|
||||
},
|
||||
"validateIndentation": "\t",
|
||||
"validateLineBreaks": "LF",
|
||||
"validateNewlineAfterArrayElements": true,
|
||||
"validateParameterSeparator": ", ",
|
||||
"validateQuoteMarks": "'",
|
||||
|
||||
"excludeFiles": [
|
||||
"coverage",
|
||||
"node_modules",
|
||||
"public/js"
|
||||
]
|
||||
}
|
3
.jshintignore
Normal file
3
.jshintignore
Normal file
@@ -0,0 +1,3 @@
|
||||
coverage
|
||||
node_modules
|
||||
public/js
|
29
.jshintrc
Normal file
29
.jshintrc
Normal file
@@ -0,0 +1,29 @@
|
||||
{
|
||||
"browser": true,
|
||||
"curly": true,
|
||||
"eqeqeq": true,
|
||||
"forin": true,
|
||||
"globals": {
|
||||
"after": true,
|
||||
"afterEach": true,
|
||||
"before": true,
|
||||
"beforeEach": true,
|
||||
"describe": true,
|
||||
"it": true
|
||||
},
|
||||
"latedef": "nofunc",
|
||||
"maxcomplexity": 6,
|
||||
"maxdepth": 2,
|
||||
"maxparams": 4,
|
||||
"noarg": true,
|
||||
"node": true,
|
||||
"nonew": true,
|
||||
"nonstandard": true,
|
||||
"regexp": true,
|
||||
"shadow": true,
|
||||
"strict": true,
|
||||
"sub": true,
|
||||
"trailing": true,
|
||||
"undef": true,
|
||||
"unused": true
|
||||
}
|
26
.travis.yml
26
.travis.yml
@@ -1,8 +1,18 @@
|
||||
|
||||
# Language/versions
|
||||
language: node_js
|
||||
node_js:
|
||||
- "0.10"
|
||||
matrix:
|
||||
include:
|
||||
|
||||
# Run tests in Node.js 0.10 (unsupported)
|
||||
- node_js: '0.10'
|
||||
|
||||
# Run tests in Node.js 0.12
|
||||
- node_js: '0.12'
|
||||
|
||||
# Allow Node.js 0.10 to fail – it's unsupported
|
||||
allow_failures:
|
||||
- node_js: '0.10'
|
||||
|
||||
# Build only master (and pull-requests)
|
||||
branches:
|
||||
@@ -15,15 +25,7 @@ services:
|
||||
|
||||
# Build script
|
||||
before_script:
|
||||
- npm install -g grunt-cli
|
||||
- cp config/test.sample.json config/test.json
|
||||
- grunt start-test &
|
||||
- NODE_ENV=test node index.js &
|
||||
- sleep 5 # give server time to start
|
||||
script: 'grunt ci'
|
||||
|
||||
# Notifications
|
||||
notifications:
|
||||
email:
|
||||
- j.robinson@nature.com
|
||||
- perry.harlock@nature.com
|
||||
- rowan.manning@nature.com
|
||||
script: 'make ci'
|
||||
|
@@ -1,53 +0,0 @@
|
||||
|
||||
How To Contribute
|
||||
=================
|
||||
|
||||
pa11y-dashboard accepts contributions from anyone, as long as you follow the guidelines below. If you'd like to contribute but aren't sure what there is for you to do, check the issue tracker for [things ready to be worked on][ready] and [known bugs][bugs].
|
||||
|
||||
It might be an idea to focus efforts on the goal of the [next milestone][milestones] before jumping onto anything too far ahead on the roadmap.
|
||||
|
||||
|
||||
Features
|
||||
--------
|
||||
|
||||
We won't accept features without prior discussion in the [issue tracker][issues]. Two heads are always better than one – this blanket rule stops you from spending your valuable time on features which may not make it back into pa11y-dashboard.
|
||||
|
||||
If you want to fork the project and build on it by yourself, of course that's absolutely fine! Just don't expect your code to me merged back upstream :)
|
||||
|
||||
|
||||
Refactoring/Rewriting
|
||||
---------------------
|
||||
|
||||
We will accept refactors where it makes an improvement to the maintainability of the code-base or makes code more readable/understandable. If there's an argument about what's readable or not, chat about it in a pull-request.
|
||||
|
||||
|
||||
Coding Guidelines
|
||||
-----------------
|
||||
|
||||
* No trailing whitespace please (except in Markdown)
|
||||
* Generally follow the style that is currently present in the code – consistency is important
|
||||
* Keep indentation consistent (tabs)
|
||||
* Don't commit code with lint errors (run `grunt lint` to run JSHint with the correct configurations)
|
||||
* Don't commit code without passing tests (run `grunt test`).
|
||||
|
||||
|
||||
Versioning
|
||||
----------
|
||||
|
||||
We use [Semantic Versioning][semver] in this project. The process for releasing a new version is as follows; this should only be done by core contributors – you don't need to include a tagged version in your pull-requests.
|
||||
|
||||
* Switch to `master` and merge the `develop` branch into it
|
||||
* Update the version number in `package.json` and `README.md`
|
||||
* Commit the changes with the message: "Version x.x.x" (x.x.x being the new version number)
|
||||
* Tag the commit with the version number (just the numbers, no "version" or "v"): `git tag x.x.x`
|
||||
* Push with tags: `git push && git push --tags`
|
||||
* Check out the `develop` branch, merge `master` into it, and push
|
||||
* On GitHub, add [release notes][release-notes] for the new version. The title should be "Version x.x.x", and the description should be a list of new features/fixes
|
||||
|
||||
|
||||
[bugs]: https://github.com/nature/pa11y-dashboard/issues?labels=bug&state=open
|
||||
[ready]: https://github.com/nature/pa11y-dashboard/issues?labels=ready&state=open
|
||||
[issues]: https://github.com/nature/pa11y-dashboard/issues
|
||||
[milestones]: https://github.com/nature/pa11y-dashboard/issues/milestones
|
||||
[release-notes]: https://github.com/nature/pa11y-dashboard/releases
|
||||
[semver]: http://semver.org/
|
134
Gruntfile.js
134
Gruntfile.js
@@ -1,134 +0,0 @@
|
||||
// This file is part of pa11y-dashboard.
|
||||
//
|
||||
// pa11y-dashboard is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// pa11y-dashboard is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
module.exports = function (grunt) {
|
||||
|
||||
grunt.initConfig({
|
||||
|
||||
jshint: {
|
||||
all: [
|
||||
'**/*.js',
|
||||
'!node_modules/**/*.js',
|
||||
'!public/js/vendor/**/*.js',
|
||||
'!public/js/site.min.js'
|
||||
],
|
||||
options: {
|
||||
es3: false,
|
||||
indent: 4,
|
||||
latedef: false,
|
||||
maxcomplexity: 4,
|
||||
maxdepth: 2,
|
||||
maxlen: 100,
|
||||
maxparams: 4,
|
||||
maxstatements: false,
|
||||
node: true,
|
||||
quotmark: 'single'
|
||||
}
|
||||
},
|
||||
|
||||
less: {
|
||||
all: {
|
||||
options: {
|
||||
cleancss: true
|
||||
},
|
||||
files: {
|
||||
'public/css/site.min.css': 'public/less/main.less'
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
mochaTest: {
|
||||
functional: {
|
||||
src: ['test/functional/**/*.js'],
|
||||
options: {
|
||||
reporter: 'spec',
|
||||
timeout: 4000
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
nodemon: {
|
||||
development: {
|
||||
options: {
|
||||
cwd: __dirname,
|
||||
file: 'index.js',
|
||||
env: {
|
||||
NODE_ENV: 'development'
|
||||
}
|
||||
}
|
||||
},
|
||||
test: {
|
||||
options: {
|
||||
cwd: __dirname,
|
||||
file: 'index.js',
|
||||
env: {
|
||||
NODE_ENV: 'test'
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
uglify: {
|
||||
options: {
|
||||
mangle: false
|
||||
},
|
||||
all: {
|
||||
files: {
|
||||
'public/js/site.min.js': [
|
||||
'public/js/vendor/jquery/jquery.min.js',
|
||||
'public/js/vendor/bootstrap/js/alert.js',
|
||||
'public/js/vendor/bootstrap/js/dropdown.js',
|
||||
'public/js/vendor/bootstrap/js/tooltip.js',
|
||||
'public/js/vendor/bootstrap/js/transition.js',
|
||||
'public/js/vendor/bootstrap/js/collapse.js',
|
||||
'public/js/vendor/flot/jquery.flot.js',
|
||||
'public/js/vendor/flot/jquery.flot.time.js',
|
||||
'public/js/vendor/flot/jquery.flot.selection.js',
|
||||
'public/js/vendor/flot/jquery.flot.resize.js',
|
||||
'public/js/site.js'
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
watch: {
|
||||
less: {
|
||||
files: ['public/less/**/*.less'],
|
||||
tasks: ['less']
|
||||
},
|
||||
js: {
|
||||
files: ['public/js/**/*.js', '!public/js/site.min.js'],
|
||||
tasks: ['uglify']
|
||||
}
|
||||
}
|
||||
|
||||
});
|
||||
|
||||
grunt.loadNpmTasks('grunt-contrib-jshint');
|
||||
grunt.loadNpmTasks('grunt-contrib-less');
|
||||
grunt.loadNpmTasks('grunt-contrib-uglify');
|
||||
grunt.loadNpmTasks('grunt-contrib-watch');
|
||||
grunt.loadNpmTasks('grunt-mocha-test');
|
||||
grunt.loadNpmTasks('grunt-nodemon');
|
||||
|
||||
grunt.registerTask('lint', ['jshint']);
|
||||
grunt.registerTask('test', ['mochaTest']);
|
||||
grunt.registerTask('compile', ['less', 'uglify']);
|
||||
grunt.registerTask('start', ['nodemon:development']);
|
||||
grunt.registerTask('start-test', ['nodemon:test']);
|
||||
grunt.registerTask('default', ['compile', 'lint', 'test']);
|
||||
grunt.registerTask('ci', ['lint', 'test']);
|
||||
|
||||
};
|
59
Makefile
Normal file
59
Makefile
Normal file
@@ -0,0 +1,59 @@
|
||||
|
||||
# Color helpers
|
||||
C_CYAN=\x1b[34;01m
|
||||
C_RESET=\x1b[0m
|
||||
|
||||
# Group targets
|
||||
all: deps lint test
|
||||
ci: lint test
|
||||
|
||||
# Install dependencies
|
||||
deps:
|
||||
@echo "$(C_CYAN)> installing dependencies$(C_RESET)"
|
||||
@npm install
|
||||
|
||||
# Lint JavaScript
|
||||
lint: jshint jscs
|
||||
|
||||
# Run JSHint
|
||||
jshint:
|
||||
@echo "$(C_CYAN)> linting javascript$(C_RESET)"
|
||||
@./node_modules/.bin/jshint .
|
||||
|
||||
# Run JavaScript Code Style
|
||||
jscs:
|
||||
@echo "$(C_CYAN)> checking javascript code style$(C_RESET)"
|
||||
@./node_modules/.bin/jscs .
|
||||
|
||||
# Run all tests
|
||||
test: test-integration
|
||||
|
||||
# Run integration tests
|
||||
test-integration:
|
||||
@echo "$(C_CYAN)> running integration tests$(C_RESET)"
|
||||
@./node_modules/.bin/mocha ./test/integration --reporter spec --recursive --timeout 5000 --slow 50
|
||||
|
||||
# Compile LESS
|
||||
less:
|
||||
@echo "$(C_CYAN)> compiling less$(C_RESET)"
|
||||
@./node_modules/.bin/lessc -x ./public/less/main.less ./public/css/site.min.css
|
||||
|
||||
# Compile client-side JavaScript
|
||||
uglify:
|
||||
@echo "$(C_CYAN)> compiling client-side JavaScript$(C_RESET)"
|
||||
@./node_modules/.bin/uglifyjs \
|
||||
public/js/vendor/jquery/jquery.min.js \
|
||||
public/js/vendor/bootstrap/js/alert.js \
|
||||
public/js/vendor/bootstrap/js/dropdown.js \
|
||||
public/js/vendor/bootstrap/js/tooltip.js \
|
||||
public/js/vendor/bootstrap/js/transition.js \
|
||||
public/js/vendor/bootstrap/js/collapse.js \
|
||||
public/js/vendor/flot/jquery.flot.js \
|
||||
public/js/vendor/flot/jquery.flot.dashes.js \
|
||||
public/js/vendor/flot/jquery.flot.time.js \
|
||||
public/js/vendor/flot/jquery.flot.selection.js \
|
||||
public/js/vendor/flot/jquery.flot.resize.js \
|
||||
public/js/site.js \
|
||||
-o ./public/js/site.min.js
|
||||
|
||||
.PHONY: test
|
50
README.md
50
README.md
@@ -3,7 +3,7 @@ pa11y-dashboard
|
||||
|
||||
pa11y-dashboard is a web interface to the [pa11y][pa11y] accessibility reporter; allowing you to focus on *fixing* issues rather than hunting them down.
|
||||
|
||||
**Current Version:** *1.6.0*
|
||||
**Current Version:** *1.9.0*
|
||||
**Build Status:** [![Build Status][travis-img]][travis]
|
||||
**Node Version Support:** *0.10*
|
||||
|
||||
@@ -22,9 +22,9 @@ You'll then need to clone this repo locally and install dependencies with `npm i
|
||||
Once you have a local clone, you'll need to copy some sample configuration files in order to run the application. From within the repo, run the following commands:
|
||||
|
||||
```sh
|
||||
$ cp config/development.sample.json config/development.json
|
||||
$ cp config/production.sample.json config/production.json
|
||||
$ cp config/test.sample.json config/test.json
|
||||
cp config/development.sample.json config/development.json
|
||||
cp config/production.sample.json config/production.json
|
||||
cp config/test.sample.json config/test.json
|
||||
```
|
||||
|
||||
Each of these files defines configurations for a different environment. If you're just running the application locally, then you should be OK with just development configurations. The [available configurations are documented here](#configurations).
|
||||
@@ -32,9 +32,9 @@ Each of these files defines configurations for a different environment. If you'r
|
||||
Now that you've got your application configured, you can run in each mode with the following commands:
|
||||
|
||||
```sh
|
||||
$ NODE_ENV=production node . # Run in production
|
||||
$ NODE_ENV=development node . # Run in development
|
||||
$ NODE_ENV=test node . # Run in test
|
||||
NODE_ENV=production node index.js # Run in production
|
||||
NODE_ENV=development node index.js # Run in development
|
||||
NODE_ENV=test node index.js # Run in test
|
||||
```
|
||||
|
||||
Check the [development instructions](#development) for more information about running locally (and restarting automatically when files change).
|
||||
@@ -64,46 +64,46 @@ This can either be an object containing [pa11y-webservice configurations][pa11y-
|
||||
Development
|
||||
-----------
|
||||
|
||||
To develop pa11y-dashboard, you'll need to clone the repo and get set up as outlined in the [setup guide](#setup). You'll also need [Grunt][grunt] to be installed globally in order to run tests, you can do this with `npm install -g grunt-cli`.
|
||||
To develop pa11y-dashboard, you'll need to clone the repo and get set up as outlined in the [setup guide](#setup).
|
||||
|
||||
Once you've done this, you'll need to start the application in test mode with:
|
||||
You'll need to start the application in test mode with:
|
||||
|
||||
```sh
|
||||
$ grunt start-test
|
||||
NODE_ENV=test node index.js
|
||||
```
|
||||
|
||||
Now you'll be able to run the following commands:
|
||||
|
||||
```sh
|
||||
$ grunt # Run the lint and test tasks together
|
||||
$ grunt lint # Run JSHint with the correct config
|
||||
$ grunt compile # Compile front-end assets
|
||||
$ grunt start # Run app in development mode, restarting if files change
|
||||
$ grunt start-test # Run app in test mode, restarting if files change
|
||||
$ grunt test # Run functional tests
|
||||
$ grunt watch # Watch for file changes and compile assets
|
||||
make # Run the lint and test tasks together
|
||||
make lint # Run linters with the correct config
|
||||
make test # Run integration tests
|
||||
```
|
||||
|
||||
Code with lint errors or failing tests will not be accepted, please use the build tools outlined above.
|
||||
|
||||
For users with push-access, don't commit to the master branch. Code should be in `develop` until it's ready to be released.
|
||||
To compile the client-side JavaScript and CSS, you'll need the following commands. Compiled code is committed to the repository.
|
||||
|
||||
```sh
|
||||
make css # Compile the site CSS from LESS files
|
||||
make uglify # Compile and uglify the client-side JavaScript
|
||||
```
|
||||
|
||||
|
||||
License
|
||||
-------
|
||||
|
||||
[Copyright 2013 Nature Publishing Group](LICENSE.txt).
|
||||
[Copyright 2013 Springer Nature](LICENSE.txt).
|
||||
pa11y-dashboard is licensed under the [GNU General Public License 3.0][gpl].
|
||||
|
||||
|
||||
|
||||
[gpl]: http://www.gnu.org/licenses/gpl-3.0.html
|
||||
[grunt]: http://gruntjs.com/
|
||||
[mongo]: http://www.mongodb.org/
|
||||
[node]: http://nodejs.org/
|
||||
[pa11y]: https://github.com/nature/pa11y
|
||||
[pa11y-webservice]: https://github.com/nature/pa11y-webservice
|
||||
[pa11y-webservice-config]: https://github.com/nature/pa11y-webservice#configurations
|
||||
[pa11y]: https://github.com/springernature/pa11y
|
||||
[pa11y-webservice]: https://github.com/springernature/pa11y-webservice
|
||||
[pa11y-webservice-config]: https://github.com/springernature/pa11y-webservice#configurations
|
||||
[phantom]: http://phantomjs.org/
|
||||
[travis]: https://travis-ci.org/nature/pa11y-dashboard
|
||||
[travis-img]: https://travis-ci.org/nature/pa11y-dashboard.png?branch=master
|
||||
[travis]: https://travis-ci.org/springernature/pa11y-dashboard
|
||||
[travis-img]: https://travis-ci.org/springernature/pa11y-dashboard.png?branch=master
|
||||
|
16
app.js
16
app.js
@@ -25,11 +25,11 @@ var pkg = require('./package.json');
|
||||
module.exports = initApp;
|
||||
|
||||
// Initialise the application
|
||||
function initApp (config, callback) {
|
||||
function initApp(config, callback) {
|
||||
config = defaultConfig(config);
|
||||
|
||||
var webserviceUrl = config.webservice;
|
||||
if (typeof webserviceUrl == 'object') {
|
||||
if (typeof webserviceUrl === 'object') {
|
||||
webserviceUrl = 'http://' + webserviceUrl.host + ':' + webserviceUrl.port + '/';
|
||||
}
|
||||
|
||||
@@ -58,7 +58,7 @@ function initApp (config, callback) {
|
||||
contentHelperName: 'content',
|
||||
layoutsDir: __dirname + '/view/layout',
|
||||
partialsDir: __dirname + '/view/partial',
|
||||
defaultLayout: __dirname + '/view/layout/default',
|
||||
defaultLayout: __dirname + '/view/layout/default'
|
||||
}));
|
||||
app.express.set('view engine', 'html');
|
||||
|
||||
@@ -79,7 +79,7 @@ function initApp (config, callback) {
|
||||
siteMessage: config.siteMessage
|
||||
});
|
||||
|
||||
app.express.use(function (req, res, next) {
|
||||
app.express.use(function(req, res, next) {
|
||||
res.locals.isHomePage = (req.path === '/');
|
||||
res.locals.host = req.host;
|
||||
next();
|
||||
@@ -100,11 +100,11 @@ function initApp (config, callback) {
|
||||
}
|
||||
|
||||
// Error handling
|
||||
app.express.get('*', function (req, res) {
|
||||
app.express.get('*', function(req, res) {
|
||||
res.status(404);
|
||||
res.render('404');
|
||||
});
|
||||
app.express.use(function (err, req, res, next) {
|
||||
app.express.use(function(err, req, res, next) {
|
||||
/* jshint unused: false */
|
||||
if (err.code === 'ECONNREFUSED') {
|
||||
err = new Error('Could not connect to pa11y-webservice');
|
||||
@@ -117,7 +117,7 @@ function initApp (config, callback) {
|
||||
res.render('500');
|
||||
});
|
||||
|
||||
app.server.listen(config.port, function (err) {
|
||||
app.server.listen(config.port, function(err) {
|
||||
var address = app.server.address();
|
||||
app.address = 'http://' + address.address + ':' + address.port;
|
||||
callback(err, app);
|
||||
@@ -126,7 +126,7 @@ function initApp (config, callback) {
|
||||
}
|
||||
|
||||
// Get default configurations
|
||||
function defaultConfig (config) {
|
||||
function defaultConfig(config) {
|
||||
if (typeof config.noindex !== 'boolean') {
|
||||
config.noindex = true;
|
||||
}
|
||||
|
@@ -13,12 +13,12 @@
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
/* jshint maxlen: false */
|
||||
// jscs:disable maximumLineLength
|
||||
'use strict';
|
||||
|
||||
module.exports = getStandards;
|
||||
|
||||
function getStandards () {
|
||||
function getStandards() {
|
||||
return [
|
||||
{
|
||||
title: 'Section508',
|
||||
@@ -656,10 +656,12 @@ function getStandards () {
|
||||
{
|
||||
name: 'WCAG2AA.Principle1.Guideline1_1.1_1_1.H30.2',
|
||||
description: 'Img element is the only content of the link, but is missing alt text. The alt text should describe the purpose of the link.'
|
||||
}, {
|
||||
},
|
||||
{
|
||||
name: 'WCAG2AA.Principle1.Guideline1_1.1_1_1.H37',
|
||||
description: 'Img element missing an alt attribute. Use the alt attribute to specify a short text alternative.'
|
||||
}, {
|
||||
},
|
||||
{
|
||||
name: 'WCAG2AA.Principle1.Guideline1_1.1_1_1.H67.1',
|
||||
description: 'Img element with empty alt text must have absent or empty title attribute.'
|
||||
},
|
||||
|
11
index.js
11
index.js
@@ -18,14 +18,19 @@
|
||||
var chalk = require('chalk');
|
||||
var config = require('./config/' + (process.env.NODE_ENV || 'development') + '.json');
|
||||
|
||||
require('./app')(config, function (err, app) {
|
||||
process.on('SIGINT', function() {
|
||||
console.log('\nGracefully shutting down from SIGINT (Ctrl-C)');
|
||||
process.exit();
|
||||
});
|
||||
|
||||
require('./app')(config, function(err, app) {
|
||||
|
||||
console.log('');
|
||||
console.log(chalk.underline.magenta('pa11y-dashboard started'));
|
||||
console.log(chalk.grey('mode: %s'), process.env.NODE_ENV);
|
||||
console.log(chalk.grey('uri: %s'), app.address);
|
||||
|
||||
app.on('route-error', function (err) {
|
||||
app.on('route-error', function(err) {
|
||||
var stack = (err.stack ? err.stack.split('\n') : [err.message]);
|
||||
var msg = chalk.red(stack.shift());
|
||||
console.error('');
|
||||
@@ -35,7 +40,7 @@ require('./app')(config, function (err, app) {
|
||||
|
||||
// Start the webservice if required
|
||||
if (typeof config.webservice === 'object') {
|
||||
require('pa11y-webservice')(config.webservice, function (err, webservice) {
|
||||
require('pa11y-webservice')(config.webservice, function(err, webservice) {
|
||||
console.log('');
|
||||
console.log(chalk.underline.cyan('pa11y-webservice started'));
|
||||
console.log(chalk.grey('mode: %s'), process.env.NODE_ENV);
|
||||
|
31
package.json
31
package.json
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "pa11y-dashboard",
|
||||
"version": "1.6.0",
|
||||
"version": "1.9.0",
|
||||
"private": true,
|
||||
|
||||
"description": "pa11y-dashboard is a visual web interface to the pa11y accessibility reporter",
|
||||
@@ -12,10 +12,10 @@
|
||||
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/nature/pa11y-dashboard.git"
|
||||
"url": "https://github.com/springernature/pa11y-dashboard.git"
|
||||
},
|
||||
"homepage": "https://github.com/nature/pa11y-dashboard",
|
||||
"bugs": "https://github.com/nature/pa11y-dashboard/issues",
|
||||
"homepage": "https://github.com/springernature/pa11y-dashboard",
|
||||
"bugs": "https://github.com/springernature/pa11y-dashboard/issues",
|
||||
|
||||
"engines": {
|
||||
"node": ">=0.10"
|
||||
@@ -25,25 +25,24 @@
|
||||
"express": "~3.4",
|
||||
"express-hbs": "~0.2",
|
||||
"moment": "~2.2",
|
||||
"pa11y-webservice": "~1.6",
|
||||
"pa11y-webservice": "~1.8",
|
||||
"pa11y-webservice-client-node": "~1.1",
|
||||
"underscore": "~1.5"
|
||||
},
|
||||
"devDependencies": {
|
||||
"bower": "~1.2",
|
||||
"grunt": "~0.4",
|
||||
"grunt-contrib-jshint": "~0.7",
|
||||
"grunt-contrib-less": "~0.8",
|
||||
"grunt-contrib-uglify": "~0.2",
|
||||
"grunt-contrib-watch": "~0.5",
|
||||
"grunt-mocha-test": "~0.7",
|
||||
"grunt-nodemon": "~0.1",
|
||||
"jsdom": "~0.8",
|
||||
"proclaim": "~2.0",
|
||||
"request": "~2.27"
|
||||
"jscs": "^2",
|
||||
"jsdom": "^3",
|
||||
"jshint": "^2",
|
||||
"less": "~1.5",
|
||||
"mocha": "^2",
|
||||
"proclaim": "^3",
|
||||
"request": "~2.27",
|
||||
"uglify-js": "~2.4"
|
||||
},
|
||||
|
||||
"scripts": {
|
||||
"start": "node ."
|
||||
"start": "node index.js",
|
||||
"test": "make ci"
|
||||
}
|
||||
}
|
||||
|
2
public/css/site.min.css
vendored
2
public/css/site.min.css
vendored
File diff suppressed because one or more lines are too long
@@ -24,9 +24,11 @@ $(document).ready(function(){
|
||||
var zoomResetButton = $('[data-role="zoom-reset"]');
|
||||
var graphContainer = $('[data-role="graph"]');
|
||||
var dateSelectDropdownMenu = $('[data-role="date-select-dropdown-menu"]');
|
||||
var legend = graphContainer.parent('.graph-container').find('.dashedLegend');
|
||||
|
||||
var graphOptions = {
|
||||
series: {
|
||||
dashes: { show: false, lineWidth: 3 },
|
||||
lines: { show: true },
|
||||
points: { show: true },
|
||||
hoverable: true
|
||||
@@ -43,6 +45,12 @@ $(document).ready(function(){
|
||||
lines: {
|
||||
lineWidth: 3
|
||||
},
|
||||
points: {
|
||||
fill: true,
|
||||
radius:4,
|
||||
lineWidth:3
|
||||
},
|
||||
shadowSize: 0,
|
||||
grid: {
|
||||
backgroundColor: '#fff',
|
||||
borderColor: '#808080',
|
||||
@@ -60,17 +68,38 @@ $(document).ready(function(){
|
||||
}
|
||||
};
|
||||
|
||||
// have we declared a custom legend
|
||||
if (legend.length === 1) {
|
||||
$('body').addClass('custom-legend');
|
||||
}
|
||||
|
||||
// Toggle appearance of lists of error/warnings/notices
|
||||
expandLink.click( function(){
|
||||
$(this).parent().next().slideToggle('slow', function(){});
|
||||
if ($(this).parent().hasClass('showing')) {
|
||||
$(this).html('+');
|
||||
expandLink.click( function(){
|
||||
$(this).next().slideToggle('slow', function(){});
|
||||
if ($(this).hasClass('showing')) {
|
||||
$(this).find('span.expander').html('+');
|
||||
$(this).attr('aria-expanded', false);
|
||||
}
|
||||
else {
|
||||
$(this).html('-');
|
||||
$(this).find('span.expander').html('-');
|
||||
$(this).attr('aria-expanded', true);
|
||||
}
|
||||
$(this).parent().toggleClass('showing');
|
||||
$(this).toggleClass('showing');
|
||||
});
|
||||
$(document).on('keydown.lists', '[data-role="expander"]', function (e) {
|
||||
var $this = $(this);
|
||||
var k = e.which || e.keyCode;
|
||||
|
||||
if (!/(13|32)/.test(k)) {
|
||||
return;
|
||||
}
|
||||
if (k === 13 || k === 32) {
|
||||
$this.click();
|
||||
}
|
||||
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
});
|
||||
|
||||
// Back to top links
|
||||
toTopLinks.click( function(e){
|
||||
@@ -94,7 +123,7 @@ $(document).ready(function(){
|
||||
target = $(this).attr('href');
|
||||
animateSection($(target), -25);
|
||||
if (!$(target).hasClass('showing')) {
|
||||
$(target).children('[data-role="expander"]').click();
|
||||
$(target).click();
|
||||
}
|
||||
});
|
||||
|
||||
@@ -152,9 +181,25 @@ $(document).ready(function(){
|
||||
|
||||
function getData() {
|
||||
return [
|
||||
{ color: 'rgb(216, 61, 45)', label: 'Errors', data: data.error },
|
||||
{ color: 'rgb(168, 103, 0)', label: 'Warnings', data: data.warning },
|
||||
{ color: 'rgb(23, 123, 190)', label: 'Notices', data: data.notice }
|
||||
{
|
||||
color: 'rgb(216, 61, 45)',
|
||||
label: 'Errors',
|
||||
data: data.error
|
||||
},
|
||||
{
|
||||
color: 'rgb(168, 103, 0)',
|
||||
label: 'Warnings',
|
||||
data: data.warning,
|
||||
lines: { show: false },
|
||||
dashes: { show: true, dashLength: [10, 5] }
|
||||
},
|
||||
{
|
||||
color: 'rgb(23, 123, 190)',
|
||||
label: 'Notices',
|
||||
data: data.notice,
|
||||
lines: { show: false },
|
||||
dashes: { show: true, dashLength: 5 }
|
||||
}
|
||||
];
|
||||
}
|
||||
|
||||
@@ -209,13 +254,25 @@ $(document).ready(function(){
|
||||
|
||||
function plotAccordingToChoices() {
|
||||
var data = [];
|
||||
var labels = [];
|
||||
choiceContainer.find('input:checked').each(function () {
|
||||
var key = $(this).attr('name');
|
||||
if (key && datasets[key]) {
|
||||
labels.push(datasets[key].label);
|
||||
data.push(datasets[key]);
|
||||
}
|
||||
});
|
||||
|
||||
if (labels.length && legend.length === 1) {
|
||||
legend.find('tr').hide();
|
||||
$.each(labels, function (index, value) {
|
||||
$('.legend' + value).parents('tr').show();
|
||||
});
|
||||
legend.show();
|
||||
} else {
|
||||
legend.hide();
|
||||
}
|
||||
|
||||
if (data.length > -1) {
|
||||
$.plot(graphContainer, data, graphOptions);
|
||||
}
|
||||
@@ -288,4 +345,29 @@ $(document).ready(function(){
|
||||
});
|
||||
}
|
||||
|
||||
// Extend public/js/vendor/bootstrap/js/collapse.js
|
||||
// Add keyboard control for filters
|
||||
|
||||
$.fn.collapse.Constructor.prototype.keydown = function (e) {
|
||||
var $this = $(this);
|
||||
var k = e.which || e.keyCode;
|
||||
|
||||
if (!/(13|32)/.test(k)) {
|
||||
return;
|
||||
}
|
||||
if (k === 13 || k === 32) {
|
||||
$this.click();
|
||||
}
|
||||
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
};
|
||||
|
||||
$('[data-toggle="collapse"]').attr('role', 'button').attr('tabindex', 0);
|
||||
$(document).on(
|
||||
'keydown.collapse.data-api',
|
||||
'[data-toggle="collapse"]',
|
||||
$.fn.collapse.Constructor.prototype.keydown
|
||||
);
|
||||
|
||||
});
|
||||
|
4
public/js/site.min.js
vendored
4
public/js/site.min.js
vendored
File diff suppressed because one or more lines are too long
228
public/js/vendor/flot/jquery.flot.dashes.js
vendored
Normal file
228
public/js/vendor/flot/jquery.flot.dashes.js
vendored
Normal file
@@ -0,0 +1,228 @@
|
||||
/*
|
||||
* jQuery.flot.dashes
|
||||
*
|
||||
* options = {
|
||||
* series: {
|
||||
* dashes: {
|
||||
*
|
||||
* // show
|
||||
* // default: false
|
||||
* // Whether to show dashes for the series.
|
||||
* show: <boolean>,
|
||||
*
|
||||
* // lineWidth
|
||||
* // default: 2
|
||||
* // The width of the dashed line in pixels.
|
||||
* lineWidth: <number>,
|
||||
*
|
||||
* // dashLength
|
||||
* // default: 10
|
||||
* // Controls the length of the individual dashes and the amount of
|
||||
* // space between them.
|
||||
* // If this is a number, the dashes and spaces will have that length.
|
||||
* // If this is an array, it is read as [ dashLength, spaceLength ]
|
||||
* dashLength: <number> or <array[2]>
|
||||
* }
|
||||
* }
|
||||
* }
|
||||
*/
|
||||
(function($){
|
||||
|
||||
function init(plot) {
|
||||
plot.hooks.drawSeries.push(function(plot, ctx, series) {
|
||||
if (!series.dashes.show) return;
|
||||
|
||||
var plotOffset = plot.getPlotOffset(),
|
||||
axisx = series.xaxis,
|
||||
axisy = series.yaxis;
|
||||
|
||||
function plotDashes(xoffset, yoffset) {
|
||||
|
||||
var points = series.datapoints.points,
|
||||
ps = series.datapoints.pointsize,
|
||||
prevx = null,
|
||||
prevy = null,
|
||||
dashRemainder = 0,
|
||||
dashOn = true,
|
||||
dashOnLength,
|
||||
dashOffLength;
|
||||
|
||||
if (series.dashes.dashLength[0]) {
|
||||
dashOnLength = series.dashes.dashLength[0];
|
||||
if (series.dashes.dashLength[1]) {
|
||||
dashOffLength = series.dashes.dashLength[1];
|
||||
} else {
|
||||
dashOffLength = dashOnLength;
|
||||
}
|
||||
} else {
|
||||
dashOffLength = dashOnLength = series.dashes.dashLength;
|
||||
}
|
||||
|
||||
ctx.beginPath();
|
||||
|
||||
for (var i = ps; i < points.length; i += ps) {
|
||||
|
||||
var x1 = points[i - ps],
|
||||
y1 = points[i - ps + 1],
|
||||
x2 = points[i],
|
||||
y2 = points[i + 1];
|
||||
|
||||
if (x1 == null || x2 == null) continue;
|
||||
|
||||
// clip with ymin
|
||||
if (y1 <= y2 && y1 < axisy.min) {
|
||||
if (y2 < axisy.min) continue; // line segment is outside
|
||||
// compute new intersection point
|
||||
x1 = (axisy.min - y1) / (y2 - y1) * (x2 - x1) + x1;
|
||||
y1 = axisy.min;
|
||||
} else if (y2 <= y1 && y2 < axisy.min) {
|
||||
if (y1 < axisy.min) continue;
|
||||
x2 = (axisy.min - y1) / (y2 - y1) * (x2 - x1) + x1;
|
||||
y2 = axisy.min;
|
||||
}
|
||||
|
||||
// clip with ymax
|
||||
if (y1 >= y2 && y1 > axisy.max) {
|
||||
if (y2 > axisy.max) continue;
|
||||
x1 = (axisy.max - y1) / (y2 - y1) * (x2 - x1) + x1;
|
||||
y1 = axisy.max;
|
||||
} else if (y2 >= y1 && y2 > axisy.max) {
|
||||
if (y1 > axisy.max) continue;
|
||||
x2 = (axisy.max - y1) / (y2 - y1) * (x2 - x1) + x1;
|
||||
y2 = axisy.max;
|
||||
}
|
||||
|
||||
// clip with xmin
|
||||
if (x1 <= x2 && x1 < axisx.min) {
|
||||
if (x2 < axisx.min) continue;
|
||||
y1 = (axisx.min - x1) / (x2 - x1) * (y2 - y1) + y1;
|
||||
x1 = axisx.min;
|
||||
} else if (x2 <= x1 && x2 < axisx.min) {
|
||||
if (x1 < axisx.min) continue;
|
||||
y2 = (axisx.min - x1) / (x2 - x1) * (y2 - y1) + y1;
|
||||
x2 = axisx.min;
|
||||
}
|
||||
|
||||
// clip with xmax
|
||||
if (x1 >= x2 && x1 > axisx.max) {
|
||||
if (x2 > axisx.max) continue;
|
||||
y1 = (axisx.max - x1) / (x2 - x1) * (y2 - y1) + y1;
|
||||
x1 = axisx.max;
|
||||
} else if (x2 >= x1 && x2 > axisx.max) {
|
||||
if (x1 > axisx.max) continue;
|
||||
y2 = (axisx.max - x1) / (x2 - x1) * (y2 - y1) + y1;
|
||||
x2 = axisx.max;
|
||||
}
|
||||
|
||||
if (x1 != prevx || y1 != prevy) {
|
||||
ctx.moveTo(axisx.p2c(x1) + xoffset, axisy.p2c(y1) + yoffset);
|
||||
}
|
||||
|
||||
var ax1 = axisx.p2c(x1) + xoffset,
|
||||
ay1 = axisy.p2c(y1) + yoffset,
|
||||
ax2 = axisx.p2c(x2) + xoffset,
|
||||
ay2 = axisy.p2c(y2) + yoffset,
|
||||
dashOffset;
|
||||
|
||||
function lineSegmentOffset(segmentLength) {
|
||||
|
||||
var c = Math.sqrt(Math.pow(ax2 - ax1, 2) + Math.pow(ay2 - ay1, 2));
|
||||
|
||||
if (c <= segmentLength) {
|
||||
return {
|
||||
deltaX: ax2 - ax1,
|
||||
deltaY: ay2 - ay1,
|
||||
distance: c,
|
||||
remainder: segmentLength - c
|
||||
}
|
||||
} else {
|
||||
var xsign = ax2 > ax1 ? 1 : -1,
|
||||
ysign = ay2 > ay1 ? 1 : -1;
|
||||
return {
|
||||
deltaX: xsign * Math.sqrt(Math.pow(segmentLength, 2) / (1 + Math.pow((ay2 - ay1)/(ax2 - ax1), 2))),
|
||||
deltaY: ysign * Math.sqrt(Math.pow(segmentLength, 2) - Math.pow(segmentLength, 2) / (1 + Math.pow((ay2 - ay1)/(ax2 - ax1), 2))),
|
||||
distance: segmentLength,
|
||||
remainder: 0
|
||||
};
|
||||
}
|
||||
}
|
||||
//-end lineSegmentOffset
|
||||
|
||||
do {
|
||||
|
||||
dashOffset = lineSegmentOffset(
|
||||
dashRemainder > 0 ? dashRemainder :
|
||||
dashOn ? dashOnLength : dashOffLength);
|
||||
|
||||
if (dashOffset.deltaX != 0 || dashOffset.deltaY != 0) {
|
||||
if (dashOn) {
|
||||
ctx.lineTo(ax1 + dashOffset.deltaX, ay1 + dashOffset.deltaY);
|
||||
} else {
|
||||
ctx.moveTo(ax1 + dashOffset.deltaX, ay1 + dashOffset.deltaY);
|
||||
}
|
||||
}
|
||||
|
||||
dashOn = !dashOn;
|
||||
dashRemainder = dashOffset.remainder;
|
||||
ax1 += dashOffset.deltaX;
|
||||
ay1 += dashOffset.deltaY;
|
||||
|
||||
} while (dashOffset.distance > 0);
|
||||
|
||||
prevx = x2;
|
||||
prevy = y2;
|
||||
}
|
||||
|
||||
ctx.stroke();
|
||||
}
|
||||
//-end plotDashes
|
||||
|
||||
ctx.save();
|
||||
ctx.translate(plotOffset.left, plotOffset.top);
|
||||
ctx.lineJoin = 'round';
|
||||
|
||||
var lw = series.dashes.lineWidth,
|
||||
sw = series.shadowSize;
|
||||
|
||||
// FIXME: consider another form of shadow when filling is turned on
|
||||
if (lw > 0 && sw > 0) {
|
||||
// draw shadow as a thick and thin line with transparency
|
||||
ctx.lineWidth = sw;
|
||||
ctx.strokeStyle = "rgba(0,0,0,0.1)";
|
||||
// position shadow at angle from the mid of line
|
||||
var angle = Math.PI/18;
|
||||
plotDashes(Math.sin(angle) * (lw/2 + sw/2), Math.cos(angle) * (lw/2 + sw/2));
|
||||
ctx.lineWidth = sw/2;
|
||||
plotDashes(Math.sin(angle) * (lw/2 + sw/4), Math.cos(angle) * (lw/2 + sw/4));
|
||||
}
|
||||
|
||||
ctx.lineWidth = lw;
|
||||
ctx.strokeStyle = series.color;
|
||||
|
||||
if (lw > 0) {
|
||||
plotDashes(0, 0);
|
||||
}
|
||||
|
||||
ctx.restore();
|
||||
|
||||
});
|
||||
//-end draw hook
|
||||
}
|
||||
//-end init
|
||||
|
||||
$.plot.plugins.push({
|
||||
init: init,
|
||||
options: {
|
||||
series: {
|
||||
dashes: {
|
||||
show: false,
|
||||
lineWidth: 2,
|
||||
dashLength: 10
|
||||
}
|
||||
}
|
||||
},
|
||||
name: 'dashes',
|
||||
version: '0.1'
|
||||
});
|
||||
|
||||
})(jQuery)
|
@@ -17,7 +17,6 @@ code {
|
||||
font-size: 90%;
|
||||
color: @code-color;
|
||||
background-color: @code-bg;
|
||||
white-space: nowrap;
|
||||
border-radius: @border-radius-base;
|
||||
}
|
||||
|
||||
|
@@ -252,7 +252,7 @@
|
||||
word-wrap:break-word;
|
||||
}
|
||||
.h4 {
|
||||
margin-bottom:22px;
|
||||
margin-bottom:6px;
|
||||
}
|
||||
}
|
||||
.date {
|
||||
@@ -347,19 +347,35 @@ ul.date-links {
|
||||
z-index:10;
|
||||
}
|
||||
.date-selector {
|
||||
margin-top:-125px;
|
||||
margin-bottom:5px;
|
||||
|
||||
.btn-group > .btn {
|
||||
float:none;
|
||||
.show-stats {
|
||||
margin-top: 0;
|
||||
}
|
||||
|
||||
&.single-result {
|
||||
margin-top:-52px;
|
||||
|
||||
.show-stats {
|
||||
display:none;
|
||||
}
|
||||
h2 {
|
||||
margin-top: 0;
|
||||
}
|
||||
|
||||
.dates-list {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
list-style-type: none;
|
||||
}
|
||||
|
||||
.dates-list > li {
|
||||
list-style-type: none;
|
||||
padding: 0;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.dates-list a {
|
||||
color: #fff;
|
||||
text-decoration: underline;
|
||||
}
|
||||
}
|
||||
.single-result .date-selector-row {
|
||||
display: none;
|
||||
}
|
||||
|
||||
/* Graph */
|
||||
@@ -368,11 +384,11 @@ ul.date-links {
|
||||
width:100%;
|
||||
}
|
||||
.graph-spacer {
|
||||
margin-bottom:60px;
|
||||
padding-bottom:60px;
|
||||
margin-bottom:30px;
|
||||
padding-bottom:30px;
|
||||
}
|
||||
.graph-table {
|
||||
margin-bottom:50px;
|
||||
margin-bottom:0;
|
||||
|
||||
td {
|
||||
width:25%;
|
||||
@@ -413,8 +429,7 @@ ul.date-links {
|
||||
}
|
||||
}
|
||||
.btn-reset {
|
||||
margin-top:-24px;
|
||||
margin-right:35px
|
||||
margin-top:12px;
|
||||
}
|
||||
.flot-x-axis {
|
||||
.flot-tick-label {
|
||||
@@ -424,6 +439,68 @@ ul.date-links {
|
||||
.tooltip-graph {
|
||||
font-size:12px;
|
||||
}
|
||||
.custom-legend .legend {
|
||||
display:none !important;
|
||||
}
|
||||
.dashedLegend {
|
||||
position:absolute;
|
||||
top:17px;
|
||||
right:40px;
|
||||
font-size:smaller;
|
||||
color:#545454;
|
||||
background-color: #fff;
|
||||
background-color: rgba(255, 255, 255, 0.75);
|
||||
display:none;
|
||||
}
|
||||
.dashedContainer {
|
||||
background: #fff;
|
||||
border: 1px solid #808080;
|
||||
margin: 5px;
|
||||
padding-top: 5px;
|
||||
}
|
||||
.dashedLegend tr {
|
||||
display: none;
|
||||
}
|
||||
.dashedLegend .legendColorBox > div:first-child {
|
||||
border: 1px solid rgb(204, 204, 204);
|
||||
padding: 3px;
|
||||
}
|
||||
.dashedLegend .legendIcon div {
|
||||
height: 0px;
|
||||
border-width: 3px 0px 0px;
|
||||
border-top-style: solid;
|
||||
overflow: hidden;
|
||||
}
|
||||
.dashedLegend .legendErrors div {
|
||||
width: 25px;
|
||||
border-top-color: rgb(216, 61, 45);
|
||||
}
|
||||
.dashedLegend .legendWarnings div {
|
||||
width: 10px;
|
||||
border-top-color: rgb(168, 103, 0);
|
||||
float: left;
|
||||
}
|
||||
.dashedLegend .legendWarnings div:first-child {
|
||||
margin-right: 5px;
|
||||
}
|
||||
.dashedLegend .legendNotices div {
|
||||
width: 5px;
|
||||
border-top-color: rgb(23, 123, 190);
|
||||
float: left;
|
||||
margin-left: 5px;
|
||||
}
|
||||
.dashedLegend .legendNotices div:first-child {
|
||||
margin-left: 0;
|
||||
}
|
||||
.dashedLegend td.legendColorBox {
|
||||
padding-right: 5px;
|
||||
padding-bottom: 5px;
|
||||
padding-left: 10px;
|
||||
}
|
||||
.dashedLegend td.legendLabel {
|
||||
padding-right: 10px;
|
||||
padding-bottom: 5px;
|
||||
}
|
||||
|
||||
/* New task page */
|
||||
.standards-lists {
|
||||
@@ -478,3 +555,21 @@ ul.date-links {
|
||||
z-index:-1;
|
||||
}
|
||||
}
|
||||
|
||||
/* inline link list */
|
||||
|
||||
.inline-list {
|
||||
display: inline-block;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
}
|
||||
.inline-list > li {
|
||||
display: inline-block;
|
||||
border-right: 1px solid @dropdown-fallback-border; // IE8 fallback
|
||||
border-right: 1px solid @dropdown-border;
|
||||
padding: 0 4px 0 0;
|
||||
margin: 0 4px 0 0;
|
||||
}
|
||||
.inline-list + div.date {
|
||||
display: inline-block;
|
||||
}
|
||||
|
@@ -43,3 +43,11 @@
|
||||
.affix {
|
||||
position: fixed;
|
||||
}
|
||||
|
||||
|
||||
// Cursors
|
||||
// -------------------------
|
||||
|
||||
.pointer {
|
||||
cursor: pointer;
|
||||
}
|
||||
|
@@ -545,6 +545,7 @@
|
||||
// Wells
|
||||
// -------------------------
|
||||
@well-bg: #f5f5f5;
|
||||
@well-bg-drk: #2c3e50;
|
||||
|
||||
|
||||
// Badges
|
||||
|
@@ -27,3 +27,11 @@
|
||||
padding: 9px;
|
||||
border-radius: @border-radius-small;
|
||||
}
|
||||
|
||||
// Dark well
|
||||
|
||||
.dark-well {
|
||||
background-color: @well-bg-drk;
|
||||
border-color: darken(@well-bg-drk, 7%);
|
||||
color: #fff;
|
||||
}
|
||||
|
@@ -20,9 +20,9 @@ var presentTask = require('../view/presenter/task');
|
||||
module.exports = route;
|
||||
|
||||
// Route definition
|
||||
function route (app) {
|
||||
app.express.get('/', function (req, res, next) {
|
||||
app.webservice.tasks.get({lastres: true}, function (err, tasks) {
|
||||
function route(app) {
|
||||
app.express.get('/', function(req, res, next) {
|
||||
app.webservice.tasks.get({lastres: true}, function(err, tasks) {
|
||||
if (err) {
|
||||
return next(err);
|
||||
}
|
||||
|
14
route/new.js
14
route/new.js
@@ -20,10 +20,10 @@ var getStandards = require('../data/standards');
|
||||
module.exports = route;
|
||||
|
||||
// Route definition
|
||||
function route (app) {
|
||||
function route(app) {
|
||||
|
||||
app.express.get('/new', function (req, res) {
|
||||
var standards = getStandards().map(function (standard) {
|
||||
app.express.get('/new', function(req, res) {
|
||||
var standards = getStandards().map(function(standard) {
|
||||
if (standard.title === 'WCAG2AA') {
|
||||
standard.selected = true;
|
||||
}
|
||||
@@ -35,7 +35,7 @@ function route (app) {
|
||||
});
|
||||
});
|
||||
|
||||
app.express.post('/new', function (req, res) {
|
||||
app.express.post('/new', function(req, res) {
|
||||
var newTask = {
|
||||
name: req.body.name,
|
||||
url: req.body.url,
|
||||
@@ -45,13 +45,13 @@ function route (app) {
|
||||
username: req.body.username,
|
||||
password: req.body.password
|
||||
};
|
||||
app.webservice.tasks.create(newTask, function (err, task) {
|
||||
app.webservice.tasks.create(newTask, function(err, task) {
|
||||
if (err) {
|
||||
var standards = getStandards().map(function (standard) {
|
||||
var standards = getStandards().map(function(standard) {
|
||||
if (standard.title === newTask.standard) {
|
||||
standard.selected = true;
|
||||
}
|
||||
standard.rules = standard.rules.map(function (rule) {
|
||||
standard.rules = standard.rules.map(function(rule) {
|
||||
if (newTask.ignore.indexOf(rule.name) !== -1) {
|
||||
rule.ignored = true;
|
||||
}
|
||||
|
@@ -20,67 +20,69 @@ var moment = require('moment');
|
||||
module.exports = route;
|
||||
|
||||
// Route definition
|
||||
function route (app) {
|
||||
function route(app) {
|
||||
|
||||
function getTaskAndResult (req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function (err, task) {
|
||||
if (err) {
|
||||
return next('route');
|
||||
}
|
||||
app.webservice
|
||||
.task(req.params.id)
|
||||
.result(req.params.rid)
|
||||
.get({full: true}, function (err, result) {
|
||||
if (err) {
|
||||
return next('route');
|
||||
}
|
||||
res.locals.task = task;
|
||||
res.locals.result = result;
|
||||
next();
|
||||
});
|
||||
});
|
||||
}
|
||||
function getTaskAndResult(req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function(err, task) {
|
||||
if (err) {
|
||||
return next('route');
|
||||
}
|
||||
app.webservice
|
||||
.task(req.params.id)
|
||||
.result(req.params.rid)
|
||||
.get({full: true}, function(err, result) {
|
||||
if (err) {
|
||||
return next('route');
|
||||
}
|
||||
res.locals.task = task;
|
||||
res.locals.result = result;
|
||||
next();
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
function getDownloadFileName (task, result, extension) {
|
||||
return [
|
||||
'pa11y',
|
||||
'--',
|
||||
task.url
|
||||
.replace(/^https?:\/\//i, '')
|
||||
.replace(/\/$/, '')
|
||||
.replace(/[^a-z0-9\.\-\_]+/gi, '-'),
|
||||
'--',
|
||||
task.standard.toLowerCase(),
|
||||
'--',
|
||||
moment(result.date).format('YYYY-MM-DD'),
|
||||
'.',
|
||||
extension
|
||||
].join('');
|
||||
}
|
||||
function getDownloadFileName(task, result, extension) {
|
||||
return [
|
||||
'pa11y',
|
||||
'--',
|
||||
task.url
|
||||
.replace(/^https?:\/\//i, '')
|
||||
.replace(/\/$/, '')
|
||||
.replace(/[^a-z0-9\.\-\_]+/gi, '-'),
|
||||
'--',
|
||||
task.standard.toLowerCase(),
|
||||
'--',
|
||||
moment(result.date).format('YYYY-MM-DD'),
|
||||
'.',
|
||||
extension
|
||||
].join('');
|
||||
}
|
||||
|
||||
app.express.get('/:id/:rid.csv', getTaskAndResult, function (req, res) {
|
||||
var task = res.locals.task;
|
||||
var result = res.locals.result;
|
||||
var rows = ['"code","message","type"'];
|
||||
result.results.forEach(function (msg) {
|
||||
rows.push([
|
||||
JSON.stringify(msg.code),
|
||||
JSON.stringify(msg.message),
|
||||
JSON.stringify(msg.type)
|
||||
].join(','));
|
||||
});
|
||||
res.attachment(getDownloadFileName(task, result, 'csv'));
|
||||
res.send(rows.join('\n'));
|
||||
});
|
||||
app.express.get('/:id/:rid.csv', getTaskAndResult, function(req, res) {
|
||||
var task = res.locals.task;
|
||||
var result = res.locals.result;
|
||||
var rows = ['"code","message","type","context","selector"'];
|
||||
result.results.forEach(function(msg) {
|
||||
rows.push([
|
||||
JSON.stringify(msg.code),
|
||||
JSON.stringify(msg.message),
|
||||
JSON.stringify(msg.type),
|
||||
JSON.stringify(msg.context),
|
||||
JSON.stringify(msg.selector)
|
||||
].join(','));
|
||||
});
|
||||
res.attachment(getDownloadFileName(task, result, 'csv'));
|
||||
res.send(rows.join('\n'));
|
||||
});
|
||||
|
||||
app.express.get('/:id/:rid.json', getTaskAndResult, function (req, res) {
|
||||
var task = res.locals.task;
|
||||
var result = res.locals.result;
|
||||
res.attachment(getDownloadFileName(task, result, 'json'));
|
||||
delete task.id;
|
||||
delete result.id;
|
||||
result.task = task;
|
||||
res.send(result);
|
||||
});
|
||||
app.express.get('/:id/:rid.json', getTaskAndResult, function(req, res) {
|
||||
var task = res.locals.task;
|
||||
var result = res.locals.result;
|
||||
res.attachment(getDownloadFileName(task, result, 'json'));
|
||||
delete task.id;
|
||||
delete result.id;
|
||||
result.task = task;
|
||||
res.send(result);
|
||||
});
|
||||
|
||||
}
|
||||
|
@@ -21,27 +21,27 @@ var presentResult = require('../../view/presenter/result');
|
||||
module.exports = route;
|
||||
|
||||
// Route definition
|
||||
function route (app) {
|
||||
function route(app) {
|
||||
|
||||
app.express.get('/:id/:rid', function (req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function (err, task) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
app.webservice
|
||||
.task(req.params.id)
|
||||
.result(req.params.rid)
|
||||
.get({full: true}, function (err, result) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
res.render('result', {
|
||||
task: presentTask(task),
|
||||
mainResult: presentResult(result),
|
||||
isResultPage: true
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
app.express.get('/:id/:rid', function(req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function(err, task) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
app.webservice
|
||||
.task(req.params.id)
|
||||
.result(req.params.rid)
|
||||
.get({full: true}, function(err, result) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
res.render('result', {
|
||||
task: presentTask(task),
|
||||
mainResult: presentResult(result),
|
||||
isResultPage: true
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
}
|
||||
|
@@ -15,16 +15,15 @@
|
||||
|
||||
'use strict';
|
||||
|
||||
var _ = require('underscore');
|
||||
var presentTask = require('../../view/presenter/task');
|
||||
|
||||
module.exports = route;
|
||||
|
||||
// Route definition
|
||||
function route (app) {
|
||||
function route(app) {
|
||||
|
||||
app.express.get('/:id/delete', function (req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function (err, task) {
|
||||
app.express.get('/:id/delete', function(req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function(err, task) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
@@ -35,8 +34,8 @@ function route (app) {
|
||||
});
|
||||
});
|
||||
|
||||
app.express.post('/:id/delete', function (req, res, next) {
|
||||
app.webservice.task(req.params.id).remove(function (err) {
|
||||
app.express.post('/:id/delete', function(req, res, next) {
|
||||
app.webservice.task(req.params.id).remove(function(err) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
|
@@ -15,25 +15,24 @@
|
||||
|
||||
'use strict';
|
||||
|
||||
var _ = require('underscore');
|
||||
var presentTask = require('../../view/presenter/task');
|
||||
var getStandards = require('../../data/standards');
|
||||
|
||||
module.exports = route;
|
||||
|
||||
// Route definition
|
||||
function route (app) {
|
||||
function route(app) {
|
||||
|
||||
app.express.get('/:id/edit', function (req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function (err, task) {
|
||||
app.express.get('/:id/edit', function(req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function(err, task) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
var standards = getStandards().map(function (standard) {
|
||||
var standards = getStandards().map(function(standard) {
|
||||
if (standard.title === task.standard) {
|
||||
standard.selected = true;
|
||||
}
|
||||
standard.rules = standard.rules.map(function (rule) {
|
||||
standard.rules = standard.rules.map(function(rule) {
|
||||
if (task.ignore.indexOf(rule.name) !== -1) {
|
||||
rule.ignored = true;
|
||||
}
|
||||
@@ -50,24 +49,24 @@ function route (app) {
|
||||
});
|
||||
});
|
||||
|
||||
app.express.post('/:id/edit', function (req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function (err, task) {
|
||||
app.express.post('/:id/edit', function(req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function(err, task) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
req.body.ignore = req.body.ignore || [];
|
||||
app.webservice.task(req.params.id).edit(req.body, function (err) {
|
||||
app.webservice.task(req.params.id).edit(req.body, function(err) {
|
||||
if (err) {
|
||||
task.name = req.body.name;
|
||||
task.ignore = req.body.ignore;
|
||||
task.timeout = req.body.timeout;
|
||||
task.username = req.body.username;
|
||||
task.password = req.body.password;
|
||||
var standards = getStandards().map(function (standard) {
|
||||
var standards = getStandards().map(function(standard) {
|
||||
if (standard.title === task.standard) {
|
||||
standard.selected = true;
|
||||
}
|
||||
standard.rules = standard.rules.map(function (rule) {
|
||||
standard.rules = standard.rules.map(function(rule) {
|
||||
if (task.ignore.indexOf(rule.name) !== -1) {
|
||||
rule.ignored = true;
|
||||
}
|
||||
|
@@ -1,16 +1,12 @@
|
||||
'use strict';
|
||||
|
||||
var _ = require('underscore');
|
||||
var presentTask = require('../../view/presenter/task');
|
||||
var getStandards = require('../../data/standards');
|
||||
|
||||
module.exports = route;
|
||||
|
||||
// Route definition
|
||||
function route (app) {
|
||||
function route(app) {
|
||||
|
||||
app.express.post('/:id/ignore', function (req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function (err, task) {
|
||||
app.express.post('/:id/ignore', function(req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function(err, task) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
@@ -21,7 +17,7 @@ function route (app) {
|
||||
if (typeof req.body.rule === 'string') {
|
||||
edit.ignore.push(req.body.rule);
|
||||
}
|
||||
app.webservice.task(req.params.id).edit(edit, function () {
|
||||
app.webservice.task(req.params.id).edit(edit, function() {
|
||||
res.redirect('/' + req.params.id + '?rule-ignored');
|
||||
});
|
||||
});
|
||||
|
@@ -22,14 +22,14 @@ var presentResultList = require('../../view/presenter/result-list');
|
||||
module.exports = route;
|
||||
|
||||
// Route definition
|
||||
function route (app) {
|
||||
function route(app) {
|
||||
|
||||
app.express.get('/:id', function (req, res, next) {
|
||||
app.webservice.task(req.params.id).get({lastres: true}, function (err, task) {
|
||||
app.express.get('/:id', function(req, res, next) {
|
||||
app.webservice.task(req.params.id).get({lastres: true}, function(err, task) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
app.webservice.task(req.params.id).results({}, function (err, results) {
|
||||
app.webservice.task(req.params.id).results({}, function(err, results) {
|
||||
if (err) {
|
||||
return next(err);
|
||||
}
|
||||
|
@@ -18,10 +18,10 @@
|
||||
module.exports = route;
|
||||
|
||||
// Route definition
|
||||
function route (app) {
|
||||
function route(app) {
|
||||
|
||||
app.express.get('/:id/run', function (req, res, next) {
|
||||
app.webservice.task(req.params.id).run(function (err, task) {
|
||||
app.express.get('/:id/run', function(req, res, next) {
|
||||
app.webservice.task(req.params.id).run(function(err) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
|
@@ -1,16 +1,12 @@
|
||||
'use strict';
|
||||
|
||||
var _ = require('underscore');
|
||||
var presentTask = require('../../view/presenter/task');
|
||||
var getStandards = require('../../data/standards');
|
||||
|
||||
module.exports = route;
|
||||
|
||||
// Route definition
|
||||
function route (app) {
|
||||
function route(app) {
|
||||
|
||||
app.express.post('/:id/unignore', function (req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function (err, task) {
|
||||
app.express.post('/:id/unignore', function(req, res, next) {
|
||||
app.webservice.task(req.params.id).get({}, function(err, task) {
|
||||
if (err) {
|
||||
return next();
|
||||
}
|
||||
@@ -22,7 +18,7 @@ function route (app) {
|
||||
if (typeof req.body.rule === 'string' && indexOfRule !== -1) {
|
||||
edit.ignore.splice(indexOfRule, 1);
|
||||
}
|
||||
app.webservice.task(req.params.id).edit(edit, function () {
|
||||
app.webservice.task(req.params.id).edit(edit, function() {
|
||||
res.redirect('/' + req.params.id + '?rule-unignored');
|
||||
});
|
||||
});
|
||||
|
@@ -21,8 +21,8 @@ var request = require('request');
|
||||
module.exports = createNavigator;
|
||||
|
||||
// Create a navigate function
|
||||
function createNavigator (baseUrl, store) {
|
||||
return function (opts, callback) {
|
||||
function createNavigator(baseUrl, store) {
|
||||
return function(opts, callback) {
|
||||
|
||||
store.body = null;
|
||||
store.dom = null;
|
||||
@@ -38,7 +38,7 @@ function createNavigator (baseUrl, store) {
|
||||
json: true,
|
||||
qs: opts.query,
|
||||
followAllRedirects: true
|
||||
}, function (err, res, body) {
|
||||
}, function(err, res, body) {
|
||||
|
||||
store.body = body;
|
||||
store.request = res.request;
|
||||
@@ -52,7 +52,7 @@ function createNavigator (baseUrl, store) {
|
||||
} else {
|
||||
jsdom.env(
|
||||
store.body,
|
||||
function (err, window) {
|
||||
function(err, window) {
|
||||
store.window = window;
|
||||
store.dom = window.document;
|
||||
callback();
|
@@ -20,9 +20,9 @@ var createClient = require('pa11y-webservice-client-node');
|
||||
module.exports = createWebserviceClient;
|
||||
|
||||
// Create a webservice client
|
||||
function createWebserviceClient (config) {
|
||||
function createWebserviceClient(config) {
|
||||
var webserviceUrl = config.webservice;
|
||||
if (typeof webserviceUrl == 'object') {
|
||||
if (typeof webserviceUrl === 'object') {
|
||||
webserviceUrl = 'http://' + webserviceUrl.host + ':' + webserviceUrl.port + '/';
|
||||
}
|
||||
return createClient(webserviceUrl);
|
@@ -13,15 +13,13 @@
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
/* global beforeEach, describe, it */
|
||||
/* jshint maxlen: false, maxstatements: false */
|
||||
'use strict';
|
||||
|
||||
var assert = require('proclaim');
|
||||
|
||||
describe('GET /', function () {
|
||||
describe('GET /', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'GET',
|
||||
endpoint: '/'
|
||||
@@ -29,17 +27,17 @@ describe('GET /', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should display an "Add new URL" button', function () {
|
||||
it('should display an "Add new URL" button', function() {
|
||||
var elem = this.last.dom.querySelectorAll('[data-test=add-task]');
|
||||
assert.strictEqual(elem.length, 1);
|
||||
assert.strictEqual(elem[0].getAttribute('href'), '/new');
|
||||
});
|
||||
|
||||
it('should display all of the expected tasks', function () {
|
||||
it('should display all of the expected tasks', function() {
|
||||
var tasks = this.last.dom.querySelectorAll('[data-test=task]');
|
||||
assert.strictEqual(tasks.length, 3);
|
||||
assert.match(tasks[0].textContent, /npg home\s+\(wcag2aa\)/i);
|
||||
@@ -47,47 +45,47 @@ describe('GET /', function () {
|
||||
assert.match(tasks[2].textContent, /nature news\s+\(section508\)/i);
|
||||
});
|
||||
|
||||
it('should have links to each task', function () {
|
||||
it('should have links to each task', function() {
|
||||
var tasks = this.last.dom.querySelectorAll('[data-test=task]');
|
||||
assert.strictEqual(tasks[0].querySelectorAll('[href="/abc000000000000000000001"]').length, 1);
|
||||
assert.strictEqual(tasks[1].querySelectorAll('[href="/abc000000000000000000002"]').length, 1);
|
||||
assert.strictEqual(tasks[2].querySelectorAll('[href="/abc000000000000000000003"]').length, 1);
|
||||
});
|
||||
|
||||
it('should display an "Edit" button for each task', function () {
|
||||
it('should display an "Edit" button for each task', function() {
|
||||
var tasks = this.last.dom.querySelectorAll('[data-test=task]');
|
||||
assert.strictEqual(tasks[0].querySelectorAll('[href="/abc000000000000000000001/edit"]').length, 1);
|
||||
assert.strictEqual(tasks[1].querySelectorAll('[href="/abc000000000000000000002/edit"]').length, 1);
|
||||
assert.strictEqual(tasks[2].querySelectorAll('[href="/abc000000000000000000003/edit"]').length, 1);
|
||||
});
|
||||
|
||||
it('should display a "Delete" button for each task', function () {
|
||||
it('should display a "Delete" button for each task', function() {
|
||||
var tasks = this.last.dom.querySelectorAll('[data-test=task]');
|
||||
assert.strictEqual(tasks[0].querySelectorAll('[href="/abc000000000000000000001/delete"]').length, 1);
|
||||
assert.strictEqual(tasks[1].querySelectorAll('[href="/abc000000000000000000002/delete"]').length, 1);
|
||||
assert.strictEqual(tasks[2].querySelectorAll('[href="/abc000000000000000000003/delete"]').length, 1);
|
||||
});
|
||||
|
||||
it('should display a "Run" button for each task', function () {
|
||||
it('should display a "Run" button for each task', function() {
|
||||
var tasks = this.last.dom.querySelectorAll('[data-test=task]');
|
||||
assert.strictEqual(tasks[0].querySelectorAll('[href="/abc000000000000000000001/run"]').length, 1);
|
||||
assert.strictEqual(tasks[1].querySelectorAll('[href="/abc000000000000000000002/run"]').length, 1);
|
||||
assert.strictEqual(tasks[2].querySelectorAll('[href="/abc000000000000000000003/run"]').length, 1);
|
||||
});
|
||||
|
||||
it('should display the task result counts if the task has been run', function () {
|
||||
it('should display the task result counts if the task has been run', function() {
|
||||
var tasks = this.last.dom.querySelectorAll('[data-test=task]');
|
||||
assert.match(tasks[0].textContent, /1\s*errors/i);
|
||||
assert.match(tasks[0].textContent, /2\s*warnings/i);
|
||||
assert.match(tasks[0].textContent, /3\s*notices/i);
|
||||
});
|
||||
|
||||
it('should display a message indicating that there are no results if the task has not been run', function () {
|
||||
it('should display a message indicating that there are no results if the task has not been run', function() {
|
||||
var tasks = this.last.dom.querySelectorAll('[data-test=task]');
|
||||
assert.match(tasks[2].textContent, /no results/i);
|
||||
});
|
||||
|
||||
it('should not display an alert message', function () {
|
||||
it('should not display an alert message', function() {
|
||||
assert.strictEqual(this.last.dom.querySelectorAll('[data-test=alert]').length, 0);
|
||||
});
|
||||
|
@@ -13,15 +13,13 @@
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
/* global beforeEach, describe, it */
|
||||
/* jshint maxlen: false, maxstatements: false */
|
||||
'use strict';
|
||||
|
||||
var assert = require('proclaim');
|
||||
|
||||
describe('GET /new', function () {
|
||||
describe('GET /new', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'GET',
|
||||
endpoint: '/new'
|
||||
@@ -29,62 +27,62 @@ describe('GET /new', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should not display an error message', function () {
|
||||
it('should not display an error message', function() {
|
||||
assert.strictEqual(this.last.dom.querySelectorAll('[data-test=error]').length, 0);
|
||||
});
|
||||
|
||||
it('should have an "Add new URL" form', function () {
|
||||
it('should have an "Add new URL" form', function() {
|
||||
var form = this.last.dom.querySelectorAll('[data-test=new-url-form]')[0];
|
||||
assert.isDefined(form);
|
||||
assert.strictEqual(form.getAttribute('action'), '/new');
|
||||
assert.strictEqual(form.getAttribute('method'), 'post');
|
||||
});
|
||||
|
||||
describe('"Add New URL" form', function () {
|
||||
describe('"Add New URL" form', function() {
|
||||
|
||||
beforeEach(function () {
|
||||
beforeEach(function() {
|
||||
this.form = this.last.dom.querySelectorAll('[data-test=new-url-form]')[0];
|
||||
});
|
||||
|
||||
it('should have a "name" field', function () {
|
||||
it('should have a "name" field', function() {
|
||||
var field = this.form.querySelectorAll('input[name=name]')[0];
|
||||
assert.isDefined(field);
|
||||
assert.strictEqual(field.getAttribute('type'), 'text');
|
||||
assert.strictEqual(field.getAttribute('value'), '');
|
||||
});
|
||||
|
||||
it('should have a "url" field', function () {
|
||||
it('should have a "url" field', function() {
|
||||
var field = this.form.querySelectorAll('input[name=url]')[0];
|
||||
assert.isDefined(field);
|
||||
assert.strictEqual(field.getAttribute('type'), 'url');
|
||||
assert.strictEqual(field.getAttribute('value'), '');
|
||||
});
|
||||
|
||||
it('should have a "username" field', function () {
|
||||
it('should have a "username" field', function() {
|
||||
var field = this.form.querySelectorAll('input[name=username]')[0];
|
||||
assert.isDefined(field);
|
||||
assert.strictEqual(field.getAttribute('type'), 'text');
|
||||
assert.strictEqual(field.getAttribute('value'), '');
|
||||
});
|
||||
|
||||
it('should have a "password" field', function () {
|
||||
it('should have a "password" field', function() {
|
||||
var field = this.form.querySelectorAll('input[name=password]')[0];
|
||||
assert.isDefined(field);
|
||||
assert.strictEqual(field.getAttribute('type'), 'text');
|
||||
assert.strictEqual(field.getAttribute('value'), '');
|
||||
});
|
||||
|
||||
it('should have a "standard" field', function () {
|
||||
it('should have a "standard" field', function() {
|
||||
var field = this.form.querySelectorAll('select[name=standard]')[0];
|
||||
assert.isDefined(field);
|
||||
assert.strictEqual(field.querySelectorAll('option').length, 4);
|
||||
});
|
||||
|
||||
it('should have "ignore" fields', function () {
|
||||
it('should have "ignore" fields', function() {
|
||||
var fields = this.form.querySelectorAll('input[name="ignore[]"]');
|
||||
assert.isDefined(fields);
|
||||
assert.notStrictEqual(fields.length, 0);
|
||||
@@ -94,11 +92,11 @@ describe('GET /new', function () {
|
||||
|
||||
});
|
||||
|
||||
describe('POST /new', function () {
|
||||
describe('POST /new', function() {
|
||||
|
||||
describe('with invalid query', function () {
|
||||
describe('with invalid query', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'POST',
|
||||
endpoint: '/new',
|
||||
@@ -110,19 +108,19 @@ describe('POST /new', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should display an error message', function () {
|
||||
it('should display an error message', function() {
|
||||
assert.strictEqual(this.last.dom.querySelectorAll('[data-test=error]').length, 1);
|
||||
});
|
||||
|
||||
});
|
||||
|
||||
describe('with valid query', function () {
|
||||
describe('with valid query', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'POST',
|
||||
endpoint: '/new',
|
||||
@@ -135,26 +133,26 @@ describe('POST /new', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should create the task', function (done) {
|
||||
this.webservice.tasks.get({}, function (err, tasks) {
|
||||
it('should create the task', function(done) {
|
||||
this.webservice.tasks.get({}, function(err, tasks) {
|
||||
assert.strictEqual(tasks.length, 4);
|
||||
done();
|
||||
});
|
||||
});
|
||||
|
||||
it('should redirect me to the new URL page', function () {
|
||||
it('should redirect me to the new URL page', function() {
|
||||
assert.match(this.last.request.uri.pathname, /^\/[a-z0-9]{24}$/i);
|
||||
});
|
||||
|
||||
it('should not display an error message', function () {
|
||||
it('should not display an error message', function() {
|
||||
assert.strictEqual(this.last.dom.querySelectorAll('[data-test=error]').length, 0);
|
||||
});
|
||||
|
||||
it('should display a success message', function () {
|
||||
it('should display a success message', function() {
|
||||
var alert = this.last.dom.querySelectorAll('[data-test=alert]')[0];
|
||||
assert.isDefined(alert);
|
||||
assert.match(alert.textContent, /url has been added/i);
|
@@ -13,15 +13,13 @@
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
/* global beforeEach, describe, it */
|
||||
/* jshint maxlen: false, maxstatements: false */
|
||||
'use strict';
|
||||
|
||||
var assert = require('proclaim');
|
||||
|
||||
describe('GET /<task-id>/<result-id>.csv', function () {
|
||||
describe('GET /<task-id>/<result-id>.csv', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'GET',
|
||||
endpoint: '/abc000000000000000000001/def000000000000000000001.csv',
|
||||
@@ -30,19 +28,19 @@ describe('GET /<task-id>/<result-id>.csv', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should output CSV results', function () {
|
||||
it('should output CSV results', function() {
|
||||
assert.match(this.last.body, /^"code","message","type"/);
|
||||
});
|
||||
|
||||
});
|
||||
|
||||
describe('GET /<task-id>/<result-id>.json', function () {
|
||||
describe('GET /<task-id>/<result-id>.json', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'GET',
|
||||
endpoint: '/abc000000000000000000001/def000000000000000000001.json',
|
||||
@@ -51,11 +49,11 @@ describe('GET /<task-id>/<result-id>.json', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should output JSON results', function () {
|
||||
it('should output JSON results', function() {
|
||||
var json = this.last.body;
|
||||
assert.strictEqual(json.task.name, 'NPG Home');
|
||||
assert.strictEqual(json.task.url, 'nature.com');
|
@@ -13,15 +13,13 @@
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
/* global beforeEach, describe, it */
|
||||
/* jshint maxlen: false, maxstatements: false */
|
||||
'use strict';
|
||||
|
||||
var assert = require('proclaim');
|
||||
|
||||
describe('GET /<task-id>/<result-id>', function () {
|
||||
describe('GET /<task-id>/<result-id>', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'GET',
|
||||
endpoint: '/abc000000000000000000001/def000000000000000000001'
|
||||
@@ -29,39 +27,39 @@ describe('GET /<task-id>/<result-id>', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should display a "Download CSV" button', function () {
|
||||
it('should display a "Download CSV" button', function() {
|
||||
var elem = this.last.dom.querySelectorAll('[data-test=download-csv]');
|
||||
assert.strictEqual(elem.length, 1);
|
||||
assert.strictEqual(elem[0].getAttribute('href'), '/abc000000000000000000001/def000000000000000000001.csv');
|
||||
});
|
||||
|
||||
it('should display a "Download JSON" button', function () {
|
||||
it('should display a "Download JSON" button', function() {
|
||||
var elem = this.last.dom.querySelectorAll('[data-test=download-json]');
|
||||
assert.strictEqual(elem.length, 1);
|
||||
assert.strictEqual(elem[0].getAttribute('href'), '/abc000000000000000000001/def000000000000000000001.json');
|
||||
});
|
||||
|
||||
it('should display a link back to the task', function () {
|
||||
it('should display a link back to the task', function() {
|
||||
assert.isDefined(this.last.dom.querySelectorAll('[href="/abc000000000000000000001"]')[0]);
|
||||
});
|
||||
|
||||
it('should display errors', function () {
|
||||
it('should display errors', function() {
|
||||
var elem = this.last.dom.querySelectorAll('[data-test=task-errors]')[0];
|
||||
assert.isDefined(elem);
|
||||
assert.match(elem.textContent, /errors \( 1 \)/i);
|
||||
});
|
||||
|
||||
it('should display warnings', function () {
|
||||
it('should display warnings', function() {
|
||||
var elem = this.last.dom.querySelectorAll('[data-test=task-warnings]')[0];
|
||||
assert.isDefined(elem);
|
||||
assert.match(elem.textContent, /warnings \( 2 \)/i);
|
||||
});
|
||||
|
||||
it('should display notices', function () {
|
||||
it('should display notices', function() {
|
||||
var elem = this.last.dom.querySelectorAll('[data-test=task-notices]')[0];
|
||||
assert.isDefined(elem);
|
||||
assert.match(elem.textContent, /notices \( 3 \)/i);
|
@@ -13,15 +13,13 @@
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
/* global beforeEach, describe, it */
|
||||
/* jshint maxlen: false, maxstatements: false */
|
||||
'use strict';
|
||||
|
||||
var assert = require('proclaim');
|
||||
|
||||
describe('GET /<task-id>/delete', function () {
|
||||
describe('GET /<task-id>/delete', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'GET',
|
||||
endpoint: '/abc000000000000000000001/delete'
|
||||
@@ -29,26 +27,26 @@ describe('GET /<task-id>/delete', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should have a "Delete URL" form', function () {
|
||||
it('should have a "Delete URL" form', function() {
|
||||
var form = this.last.dom.querySelectorAll('[data-test=delete-url-form]')[0];
|
||||
assert.isDefined(form);
|
||||
assert.strictEqual(form.getAttribute('action'), '/abc000000000000000000001/delete');
|
||||
assert.strictEqual(form.getAttribute('method'), 'post');
|
||||
});
|
||||
|
||||
it('should display a link back to the task page', function () {
|
||||
it('should display a link back to the task page', function() {
|
||||
assert.greaterThan(this.last.dom.querySelectorAll('[href="/abc000000000000000000001"]').length, 0);
|
||||
});
|
||||
|
||||
});
|
||||
|
||||
describe('POST /<task-id>/delete', function () {
|
||||
describe('POST /<task-id>/delete', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'POST',
|
||||
endpoint: '/abc000000000000000000001/delete'
|
||||
@@ -56,22 +54,22 @@ describe('POST /<task-id>/delete', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should delete the task', function (done) {
|
||||
this.webservice.task('abc000000000000000000001').get({}, function (err) {
|
||||
it('should delete the task', function(done) {
|
||||
this.webservice.task('abc000000000000000000001').get({}, function(err) {
|
||||
assert.strictEqual(err.message, 'Error 404');
|
||||
done();
|
||||
});
|
||||
});
|
||||
|
||||
it('should redirect me to the home page', function () {
|
||||
it('should redirect me to the home page', function() {
|
||||
assert.strictEqual(this.last.request.uri.pathname, '/');
|
||||
});
|
||||
|
||||
it('should display a success message', function () {
|
||||
it('should display a success message', function() {
|
||||
var alert = this.last.dom.querySelectorAll('[data-test=alert]')[0];
|
||||
assert.isDefined(alert);
|
||||
assert.match(alert.textContent, /been deleted/i);
|
@@ -13,15 +13,13 @@
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
/* global beforeEach, describe, it */
|
||||
/* jshint maxlen: false, maxstatements: false */
|
||||
'use strict';
|
||||
|
||||
var assert = require('proclaim');
|
||||
|
||||
describe('GET /<task-id>/edit', function () {
|
||||
describe('GET /<task-id>/edit', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'GET',
|
||||
endpoint: '/abc000000000000000000001/edit'
|
||||
@@ -29,35 +27,35 @@ describe('GET /<task-id>/edit', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should have an "Edit URL" form', function () {
|
||||
it('should have an "Edit URL" form', function() {
|
||||
var form = this.last.dom.querySelectorAll('[data-test=edit-url-form]')[0];
|
||||
assert.isDefined(form);
|
||||
assert.strictEqual(form.getAttribute('action'), '/abc000000000000000000001/edit');
|
||||
assert.strictEqual(form.getAttribute('method'), 'post');
|
||||
});
|
||||
|
||||
it('should display a link back to the task page', function () {
|
||||
it('should display a link back to the task page', function() {
|
||||
assert.greaterThan(this.last.dom.querySelectorAll('[href="/abc000000000000000000001"]').length, 0);
|
||||
});
|
||||
|
||||
describe('"Edit URL" form', function () {
|
||||
describe('"Edit URL" form', function() {
|
||||
|
||||
beforeEach(function () {
|
||||
beforeEach(function() {
|
||||
this.form = this.last.dom.querySelectorAll('[data-test=edit-url-form]')[0];
|
||||
});
|
||||
|
||||
it('should have a "name" field', function () {
|
||||
it('should have a "name" field', function() {
|
||||
var field = this.form.querySelectorAll('input[name=name]')[0];
|
||||
assert.isDefined(field);
|
||||
assert.strictEqual(field.getAttribute('type'), 'text');
|
||||
assert.strictEqual(field.getAttribute('value'), 'NPG Home');
|
||||
});
|
||||
|
||||
it('should have a disabled "url" field', function () {
|
||||
it('should have a disabled "url" field', function() {
|
||||
var field = this.form.querySelectorAll('input[name=url]')[0];
|
||||
assert.isDefined(field);
|
||||
assert.strictEqual(field.getAttribute('type'), 'url');
|
||||
@@ -65,27 +63,27 @@ describe('GET /<task-id>/edit', function () {
|
||||
assert.isDefined(field.getAttribute('disabled'));
|
||||
});
|
||||
|
||||
it('should have a disabled "standard" field', function () {
|
||||
it('should have a disabled "standard" field', function() {
|
||||
var field = this.form.querySelectorAll('select[name=standard]')[0];
|
||||
assert.isDefined(field);
|
||||
assert.isDefined(field.getAttribute('disabled'));
|
||||
});
|
||||
|
||||
it('should have a "username" field', function () {
|
||||
it('should have a "username" field', function() {
|
||||
var field = this.form.querySelectorAll('input[name=username]')[0];
|
||||
assert.isDefined(field);
|
||||
assert.strictEqual(field.getAttribute('type'), 'text');
|
||||
assert.strictEqual(field.getAttribute('value'), 'user');
|
||||
});
|
||||
|
||||
it('should have a "password" field', function () {
|
||||
it('should have a "password" field', function() {
|
||||
var field = this.form.querySelectorAll('input[name=password]')[0];
|
||||
assert.isDefined(field);
|
||||
assert.strictEqual(field.getAttribute('type'), 'text');
|
||||
assert.strictEqual(field.getAttribute('value'), 'access');
|
||||
});
|
||||
|
||||
it('should have "ignore" fields', function () {
|
||||
it('should have "ignore" fields', function() {
|
||||
var fields = this.form.querySelectorAll('input[name="ignore[]"]');
|
||||
assert.isDefined(fields);
|
||||
assert.notStrictEqual(fields.length, 0);
|
||||
@@ -95,9 +93,9 @@ describe('GET /<task-id>/edit', function () {
|
||||
|
||||
});
|
||||
|
||||
describe('POST /<task-id>/edit', function () {
|
||||
describe('POST /<task-id>/edit', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'POST',
|
||||
endpoint: '/abc000000000000000000001/edit',
|
||||
@@ -111,12 +109,12 @@ describe('POST /<task-id>/edit', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should edit the task', function (done) {
|
||||
this.webservice.task('abc000000000000000000001').get({}, function (err, task) {
|
||||
it('should edit the task', function(done) {
|
||||
this.webservice.task('abc000000000000000000001').get({}, function(err, task) {
|
||||
assert.strictEqual(task.name, 'foo');
|
||||
assert.strictEqual(task.username, 'newuser');
|
||||
assert.strictEqual(task.password, 'secure');
|
||||
@@ -125,11 +123,10 @@ describe('POST /<task-id>/edit', function () {
|
||||
});
|
||||
});
|
||||
|
||||
it('should display a success message', function () {
|
||||
it('should display a success message', function() {
|
||||
var alert = this.last.dom.querySelectorAll('[data-test=alert]')[0];
|
||||
assert.isDefined(alert);
|
||||
assert.match(alert.textContent, /been saved/i);
|
||||
});
|
||||
|
||||
});
|
||||
|
@@ -13,17 +13,15 @@
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
/* global beforeEach, describe, it */
|
||||
/* jshint maxlen: false, maxstatements: false */
|
||||
'use strict';
|
||||
|
||||
var assert = require('proclaim');
|
||||
|
||||
describe('GET /<task-id>', function () {
|
||||
describe('GET /<task-id>', function() {
|
||||
|
||||
describe('when task has results', function () {
|
||||
describe('when task has results', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'GET',
|
||||
endpoint: '/abc000000000000000000001'
|
||||
@@ -31,48 +29,48 @@ describe('GET /<task-id>', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should display an "Edit" button', function () {
|
||||
it('should display an "Edit" button', function() {
|
||||
assert.strictEqual(this.last.dom.querySelectorAll('[href="/abc000000000000000000001/edit"]').length, 1);
|
||||
});
|
||||
|
||||
it('should display a "Delete" button', function () {
|
||||
it('should display a "Delete" button', function() {
|
||||
assert.strictEqual(this.last.dom.querySelectorAll('[href="/abc000000000000000000001/delete"]').length, 1);
|
||||
});
|
||||
|
||||
it('should display a "Run" button', function () {
|
||||
it('should display a "Run" button', function() {
|
||||
assert.strictEqual(this.last.dom.querySelectorAll('[href="/abc000000000000000000001/run"]').length, 1);
|
||||
});
|
||||
|
||||
it('should display a "Download CSV" button for the latest result', function () {
|
||||
it('should display a "Download CSV" button for the latest result', function() {
|
||||
assert.strictEqual(this.last.dom.querySelectorAll('[href="/abc000000000000000000001/def000000000000000000001.csv"]').length, 1);
|
||||
});
|
||||
|
||||
it('should display a "Download JSON" button for the latest result', function () {
|
||||
it('should display a "Download JSON" button for the latest result', function() {
|
||||
assert.strictEqual(this.last.dom.querySelectorAll('[href="/abc000000000000000000001/def000000000000000000001.json"]').length, 1);
|
||||
});
|
||||
|
||||
it('should display links to all results', function () {
|
||||
it('should display links to all results', function() {
|
||||
assert.isDefined(this.last.dom.querySelectorAll('[href="/abc000000000000000000001/def000000000000000000001"]')[0]);
|
||||
assert.isDefined(this.last.dom.querySelectorAll('[href="/abc000000000000000000001/def000000000000000000003"]')[0]);
|
||||
});
|
||||
|
||||
it('should display errors', function () {
|
||||
it('should display errors', function() {
|
||||
var elem = this.last.dom.querySelectorAll('[data-test=task-errors]')[0];
|
||||
assert.isDefined(elem);
|
||||
assert.match(elem.textContent, /errors \( 1 \)/i);
|
||||
});
|
||||
|
||||
it('should display warnings', function () {
|
||||
it('should display warnings', function() {
|
||||
var elem = this.last.dom.querySelectorAll('[data-test=task-warnings]')[0];
|
||||
assert.isDefined(elem);
|
||||
assert.match(elem.textContent, /warnings \( 2 \)/i);
|
||||
});
|
||||
|
||||
it('should display notices', function () {
|
||||
it('should display notices', function() {
|
||||
var elem = this.last.dom.querySelectorAll('[data-test=task-notices]')[0];
|
||||
assert.isDefined(elem);
|
||||
assert.match(elem.textContent, /notices \( 3 \)/i);
|
||||
@@ -80,9 +78,9 @@ describe('GET /<task-id>', function () {
|
||||
|
||||
});
|
||||
|
||||
describe('when task has no results', function () {
|
||||
describe('when task has no results', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'GET',
|
||||
endpoint: '/abc000000000000000000003'
|
||||
@@ -90,17 +88,17 @@ describe('GET /<task-id>', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should display a "Run" button', function () {
|
||||
it('should display a "Run" button', function() {
|
||||
var elem = this.last.dom.querySelectorAll('[data-test=run-task]');
|
||||
assert.strictEqual(elem.length, 1);
|
||||
assert.strictEqual(elem[0].getAttribute('href'), '/abc000000000000000000003/run');
|
||||
});
|
||||
|
||||
it('should display a message indicating that there are no results', function () {
|
||||
it('should display a message indicating that there are no results', function() {
|
||||
var alert = this.last.dom.querySelectorAll('[data-test=alert]')[0];
|
||||
assert.isDefined(alert);
|
||||
assert.match(alert.textContent, /there are no results to show/i);
|
@@ -13,15 +13,13 @@
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
/* global beforeEach, describe, it */
|
||||
/* jshint maxlen: false, maxstatements: false */
|
||||
'use strict';
|
||||
|
||||
var assert = require('proclaim');
|
||||
|
||||
describe('GET /<task-id>/run', function () {
|
||||
describe('GET /<task-id>/run', function() {
|
||||
|
||||
beforeEach(function (done) {
|
||||
beforeEach(function(done) {
|
||||
var req = {
|
||||
method: 'GET',
|
||||
endpoint: '/abc000000000000000000001/run'
|
||||
@@ -29,15 +27,15 @@ describe('GET /<task-id>/run', function () {
|
||||
this.navigate(req, done);
|
||||
});
|
||||
|
||||
it('should send a 200 status', function () {
|
||||
it('should send a 200 status', function() {
|
||||
assert.strictEqual(this.last.status, 200);
|
||||
});
|
||||
|
||||
it('should redirect me to the task page', function () {
|
||||
it('should redirect me to the task page', function() {
|
||||
assert.strictEqual(this.last.request.uri.pathname, '/abc000000000000000000001');
|
||||
});
|
||||
|
||||
it('should display a success message', function () {
|
||||
it('should display a success message', function() {
|
||||
var alert = this.last.dom.querySelectorAll('[data-test=alert]')[0];
|
||||
assert.isDefined(alert);
|
||||
assert.match(alert.textContent, /new results are being generated/i);
|
@@ -13,8 +13,6 @@
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
/* global afterEach, before */
|
||||
/* jshint maxlen: false, maxstatements: false */
|
||||
'use strict';
|
||||
|
||||
var config = require('../../config/test.json');
|
||||
@@ -24,26 +22,26 @@ var loadFixtures = require('pa11y-webservice/data/fixture/load');
|
||||
var request = require('request');
|
||||
|
||||
// Run before all tests
|
||||
before(function (done) {
|
||||
before(function(done) {
|
||||
this.baseUrl = 'http://localhost:' + config.port;
|
||||
this.last = {};
|
||||
this.navigate = createNavigator(this.baseUrl, this.last);
|
||||
this.webservice = createWebserviceClient(config);
|
||||
assertTestAppIsRunning(this.baseUrl, function () {
|
||||
assertTestAppIsRunning(this.baseUrl, function() {
|
||||
loadFixtures('test', config.webservice, done);
|
||||
});
|
||||
});
|
||||
|
||||
// Run after each test
|
||||
afterEach(function (done) {
|
||||
afterEach(function(done) {
|
||||
loadFixtures('test', config.webservice, done);
|
||||
});
|
||||
|
||||
// Check that the test application is running, and exit if not
|
||||
function assertTestAppIsRunning (url, done) {
|
||||
request(url, function (err) {
|
||||
function assertTestAppIsRunning(url, done) {
|
||||
request(url, function(err) {
|
||||
if (err) {
|
||||
console.error('Error: Test app not started; run with `grunt start-test`');
|
||||
console.error('Error: Test app not started; run with `NODE_ENV=test node index.js`');
|
||||
process.exit(1);
|
||||
}
|
||||
done();
|
@@ -19,20 +19,20 @@ var moment = require('moment');
|
||||
|
||||
module.exports = helper;
|
||||
|
||||
function helper (register) {
|
||||
function helper(register) {
|
||||
|
||||
// Format a date with Moment
|
||||
register('date-format', function (context, block) {
|
||||
register('date-format', function(context, block) {
|
||||
var format = block.hash.format || 'YYYY-MM-DD HH:mm:ss';
|
||||
return moment(context).format(format);
|
||||
});
|
||||
|
||||
// Get a relative date
|
||||
register('date-relative', function (context) {
|
||||
register('date-relative', function(context) {
|
||||
return moment(context).fromNow();
|
||||
});
|
||||
|
||||
register('date-timestamp', function (context) {
|
||||
register('date-timestamp', function(context) {
|
||||
return moment(context).valueOf();
|
||||
});
|
||||
|
||||
|
@@ -2,10 +2,10 @@
|
||||
|
||||
module.exports = helper;
|
||||
|
||||
function helper (register) {
|
||||
function helper(register) {
|
||||
|
||||
// Convert a string to lower-case
|
||||
register('lowercase', function (context) {
|
||||
register('lowercase', function(context) {
|
||||
return context.toLowerCase();
|
||||
});
|
||||
|
||||
|
@@ -17,10 +17,10 @@
|
||||
|
||||
module.exports = helper;
|
||||
|
||||
function helper (register) {
|
||||
function helper(register) {
|
||||
|
||||
// Simplify url by removing (eg http://, https://, trailing slashes) from url
|
||||
register('simplify-url', function (context) {
|
||||
register('simplify-url', function(context) {
|
||||
return context.replace(/^https?:\/\//i, '').replace(/\/$/, '').toLowerCase();
|
||||
});
|
||||
|
||||
|
@@ -92,7 +92,7 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<p class="control-label"><b>Ignore these rules</b> <a target="_blank" href="https://github.com/nature/pa11y/wiki/HTML-CodeSniffer-Rules">(full list of rules here)</a></p>
|
||||
<p class="control-label"><b>Ignore these rules</b> <a target="_blank" href="https://github.com/springernature/pa11y/wiki/HTML-CodeSniffer-Rules">(full list of rules here)</a></p>
|
||||
|
||||
<div class="standards-lists">
|
||||
{{#standards}}
|
||||
|
@@ -15,10 +15,42 @@ You should have received a copy of the GNU General Public License
|
||||
along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
}}
|
||||
<div class="col-md-12 clearfix">
|
||||
<div class="graph-container graph-spacer ruled">
|
||||
<div class="graph-container graph-spacer ruled clearfix">
|
||||
<div data-role="graph" class="graph"></div>
|
||||
<div class="row">
|
||||
<ul class="list-unstyled floated-list series-checkboxes clearfix crunch-bottom col-md-3 col-sm-6 col-xs-12" data-role="series-checkboxes"></ul>
|
||||
<ul class="list-unstyled floated-list series-checkboxes clearfix crunch-bottom col-md-3 col-sm-6 col-xs-12 pull-right" data-role="series-checkboxes"></ul>
|
||||
</div>
|
||||
<div class="dashedLegend">
|
||||
<div class="dashedContainer">
|
||||
<table>
|
||||
<tbody>
|
||||
<tr>
|
||||
<td class="legendColorBox">
|
||||
<div class="clearfix legendIcon legendErrors">
|
||||
<div></div>
|
||||
</div>
|
||||
</td>
|
||||
<td class="legendLabel">Errors</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td class="legendColorBox">
|
||||
<div class="clearfix legendIcon legendWarnings">
|
||||
<div></div><div></div>
|
||||
</div>
|
||||
</td>
|
||||
<td class="legendLabel">Warnings</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td class="legendColorBox">
|
||||
<div class="clearfix legendIcon legendNotices">
|
||||
<div></div><div></div><div></div>
|
||||
</div>
|
||||
</td>
|
||||
<td class="legendLabel">Notices</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
<button data-role='zoom-reset' class="btn btn-xs btn-primary pull-right btn-reset hidden">Reset Zoom <i class="glyphicon glyphicon-zoom-out"></i></button>
|
||||
</div>
|
||||
|
@@ -18,7 +18,7 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
<div class="footer" role="contentinfo">
|
||||
<div class="container">
|
||||
<div class="col-md-5">
|
||||
<small>© {{year}} Nature Publishing Group.<br/>pa11y dashboard is licensed under the GNU General Public License 3.0.<br/>Version {{version}}</small>
|
||||
<small>© {{year}} Springer Nature.<br/>pa11y dashboard is licensed under the GNU General Public License 3.0.<br/>Version {{version}}</small>
|
||||
</div>
|
||||
<div class="col-md-7 clearfix">
|
||||
<ul class="crunch-bottom floated-list nav">
|
||||
|
@@ -16,7 +16,7 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
}}
|
||||
<div class="col-md-12">
|
||||
<div class="ruled task-header">
|
||||
<div class="row clearfix">
|
||||
<div class="row clearfix task-header">
|
||||
<div class="col-md-12">
|
||||
<div class="h3 crunch well-med well pull-right"><span class="glyphicon glyphicon-calendar"></span> {{date-format mainResult.date format="DD MMM YYYY"}}</div>
|
||||
<h1 class="h2 crunch-top">{{task.name}}</h1>
|
||||
|
@@ -1,35 +0,0 @@
|
||||
{{!
|
||||
This file is part of pa11y-dashboard.
|
||||
|
||||
pa11y-dashboard is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
pa11y-dashboard is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
}}
|
||||
<div class="col-md-12 zfix">
|
||||
<div class="row">
|
||||
<div class="col-md-4 col-md-offset-4 col-sm-6 col-sm-offset-3">
|
||||
<div class="date-selector{{#if hasOneResult}} single-result{{/if}}">
|
||||
<h4 class="show-stats text-center">Select a date to show stats for</h4>
|
||||
<ul class="list-unstyled">
|
||||
<li class="btn-group block-level clearfix">
|
||||
<button data-toggle="dropdown" class="btn-full-width btn btn-primary dropdown-toggle" type="button">{{date-format task.lastResult.date format="DD MMM YYYY"}} <span class="glyphicon glyphicon-calendar"></span> <span class="caret"></span></button>
|
||||
<ul role="navigation" class="date-links list-group hidden" data-role="date-select-dropdown-menu">
|
||||
{{#results}}
|
||||
<li><a class="list-group-item text-center" href="{{href}}">{{date-format date format="DD MMM YYYY"}}</a></li>
|
||||
{{/results}}
|
||||
</ul>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
@@ -14,9 +14,10 @@ GNU General Public License for more details.
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
}}
|
||||
|
||||
<div class="col-md-3 aside">
|
||||
<div class="row">
|
||||
<div id="top" class="col-md-12 col-sm-6 col-xs-12">
|
||||
<div class="col-md-12 col-sm-6 col-xs-12">
|
||||
<ul data-role="task-list" class="clearfix list-unstyled floated-list task-stats">
|
||||
{{#mainResult}}
|
||||
<li class="danger"><a href="#errors" title="See errors">{{count.error}}<span class="stat-type">Errors</span></a></li>
|
||||
@@ -40,6 +41,25 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row date-selector-row">
|
||||
<div id="top" class="col-md-12 col-sm-12 clearfix">
|
||||
<div class="well dark-well">
|
||||
<div class="date-selector">
|
||||
<div class="btn-group block-level clearfix">
|
||||
<h2 class="h4">
|
||||
<span class="glyphicon glyphicon-calendar"></span> {{date-format task.lastResult.date format="DD MMM YYYY"}}
|
||||
</h2>
|
||||
<h3 class="h5 show-stats">Select a date to show stats for:</h3>
|
||||
<ul role="navigation" class="dates-list">
|
||||
{{#results}}
|
||||
<li><a class="" href="{{href}}">{{date-format date format="DD MMM YYYY"}}</a></li>
|
||||
{{/results}}
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row">
|
||||
<div class="col-md-12 col-sm-12 clearfix">
|
||||
<div class="well">
|
||||
@@ -50,19 +70,26 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="col-md-9">
|
||||
<div class="col-md-9" data-role="expandable-results" role="main">
|
||||
{{#if mainResult.count.error}}
|
||||
<div class="heading label-danger showing first" id="errors" data-test="task-errors">
|
||||
<span data-role="expander" class="pull-right expander"> - </span>
|
||||
<div class="heading label-danger pointer showing first" id="errors" data-test="task-errors" data-role="expander" role="button" tabindex="0" aria-expanded="true" aria-controls="errors-list">
|
||||
<span class="pull-right expander"> - <span class="hide">(close panel)</span></span>
|
||||
Errors ( {{mainResult.count.error}} )
|
||||
</div>
|
||||
<div class="task-danger tasks-list collapse clearfix in">
|
||||
<div class="task-danger tasks-list collapse clearfix in" id="errors-list">
|
||||
|
||||
<ul class="list-unstyled">
|
||||
{{#mainResult.errors}}
|
||||
<li>
|
||||
<p class="crunch rule-name">{{code}} <span class="badge">{{count}}</span></p>
|
||||
<p>{{message}}</p>
|
||||
<p><em>First result:</em> {{message}}</p>
|
||||
<p>
|
||||
<b>Selector:</b>
|
||||
<code style="text-wrap:pre-wrap">{{#if selector}}{{selector}}{{else}}-{{/if}}</code>
|
||||
<br/>
|
||||
<b>Context:</b>
|
||||
<code style="text-wrap:pre-wrap">{{#if context}}{{context}}{{else}}-{{/if}}</code>
|
||||
</p>
|
||||
{{#unless readonly}}
|
||||
{{#if ../../isTaskPage}}
|
||||
<form action="{{../../../task.hrefIgnore}}" method="post">
|
||||
@@ -81,16 +108,23 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
{{/if}}
|
||||
|
||||
{{#if mainResult.count.warning}}
|
||||
<div class="heading label-warning" id="warnings" data-test="task-warnings">
|
||||
<span data-role="expander" class="pull-right expander"> + </span>
|
||||
<div class="heading label-warning pointer" id="warnings" data-test="task-warnings" data-role="expander" role="button" tabindex="0" aria-expanded="false" aria-controls="warnings-list">
|
||||
<span class="pull-right expander"> + <span class="hide">(open panel)</span></span>
|
||||
Warnings ( {{mainResult.count.warning}} )
|
||||
</div>
|
||||
<div class="task-warning tasks-list collapse clearfix">
|
||||
<div class="task-warning tasks-list collapse clearfix" id="warnings-list">
|
||||
<ul class="list-unstyled">
|
||||
{{#mainResult.warnings}}
|
||||
<li>
|
||||
<p class="crunch rule-name">{{code}} <span class="badge">{{count}}</span></p>
|
||||
<p>{{message}}</p>
|
||||
<p><em>First result:</em> {{message}}</p>
|
||||
<p>
|
||||
<b>Selector:</b>
|
||||
<code style="text-wrap:pre-wrap">{{#if selector}}{{selector}}{{else}}-{{/if}}</code>
|
||||
<br/>
|
||||
<b>Context:</b>
|
||||
<code style="text-wrap:pre-wrap">{{#if context}}{{context}}{{else}}-{{/if}}</code>
|
||||
</p>
|
||||
{{#unless readonly}}
|
||||
{{#if ../../isTaskPage}}
|
||||
<form action="{{../../../task.hrefIgnore}}" method="post">
|
||||
@@ -110,16 +144,23 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
{{/if}}
|
||||
|
||||
{{#if mainResult.count.notice}}
|
||||
<div class="heading label-info" id="notices" data-test="task-notices">
|
||||
<span data-role="expander" class="pull-right expander"> + </span>
|
||||
<div class="heading label-info pointer" id="notices" data-test="task-notices" data-role="expander" role="button" tabindex="0" aria-expanded="false" aria-controls="notices-list">
|
||||
<span class="pull-right expander"> + <span class="hide">(open panel)</span></span>
|
||||
Notices ( {{mainResult.count.notice}} )
|
||||
</div>
|
||||
<div class="task-info tasks-list collapse clearfix">
|
||||
<div class="task-info tasks-list collapse clearfix" id="notices-list">
|
||||
<ul class="list-unstyled">
|
||||
{{#mainResult.notices}}
|
||||
<li>
|
||||
<p class="crunch rule-name">{{code}} <span class="badge">{{count}}</span></p>
|
||||
<p>{{message}}</p>
|
||||
<p><em>First result:</em> {{message}}</p>
|
||||
<p>
|
||||
<b>Selector:</b>
|
||||
<code style="text-wrap:pre-wrap">{{#if selector}}{{selector}}{{else}}-{{/if}}</code>
|
||||
<br/>
|
||||
<b>Context:</b>
|
||||
<code style="text-wrap:pre-wrap">{{#if context}}{{context}}{{else}}-{{/if}}</code>
|
||||
</p>
|
||||
{{#unless readonly}}
|
||||
{{#if ../../isTaskPage}}
|
||||
<form action="{{../../../task.hrefIgnore}}" method="post">
|
||||
@@ -138,11 +179,11 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
{{/if}}
|
||||
|
||||
{{#if mainResult.ignore.length}}
|
||||
<div class="heading label-default">
|
||||
<span data-role="expander" class="pull-right expander"> + </span>
|
||||
<div class="heading label-default pointer" id="ignore" data-role="expander" role="button" tabindex="0" aria-expanded="false" aria-controls="ignore-list">
|
||||
<span class="pull-right expander"> + <span class="hide">(open panel)</span></span>
|
||||
Ignored Rules ( {{mainResult.ignore.length}} )
|
||||
</div>
|
||||
<div class="task-default tasks-list collapse clearfix">
|
||||
<div class="task-default tasks-list collapse clearfix" id="ignore-list">
|
||||
<ul class="list-unstyled">
|
||||
{{#mainResult.ignore}}
|
||||
<li>
|
||||
|
@@ -17,27 +17,25 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
<div class="col-md-12 zfix">
|
||||
<div class="ruled task-header">
|
||||
<div class="row clearfix">
|
||||
<div class="col-md-9 col-sm-9">
|
||||
<div class="col-md-12">
|
||||
<h1 class="h2 crunch-top">{{task.name}}</h1>
|
||||
<p class="h4">{{simplify-url task.url}}<span class="h5"> ({{task.standard}})</span></p>
|
||||
</div>
|
||||
<div class="col-md-3 col-sm-3 text-right run-details">
|
||||
{{#unless readonly}}
|
||||
<div class="btn-group">
|
||||
<button type="button" class="btn btn-info dropdown-toggle" data-toggle="dropdown">Options <span class="caret"></span></button>
|
||||
<ul class="dropdown-menu pull-right" role="menu">
|
||||
<div class="run-details task-header clearfix">
|
||||
<div class="col-md-12 clearfix">
|
||||
{{#unless readonly}}
|
||||
<ul class="inline-list" role="menu">
|
||||
<li><a href="/{{task.id}}/edit">Edit this task</a></li>
|
||||
<li><a href="/{{task.id}}/delete">Delete this task</a></li>
|
||||
<li class="divider"></li>
|
||||
<li><a href="{{task.hrefRun}}" data-test="run-task">Run pa11y</a></li>
|
||||
</ul>
|
||||
</div>
|
||||
{{/unless}}
|
||||
{{#if mainResult}}
|
||||
<div class="date">Last run : {{date-format mainResult.date format="DD MMM YYYY"}}</div>
|
||||
{{else}}
|
||||
<div class="date">Not yet run</div>
|
||||
{{/if}}
|
||||
{{/unless}}
|
||||
{{#if mainResult}}
|
||||
<div class="date">Last run: <strong>{{date-format mainResult.date format="DD MMM YYYY"}}</strong></div>
|
||||
{{else}}
|
||||
<div class="date">Not yet run</div>
|
||||
{{/if}}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
@@ -20,21 +20,21 @@ var rules = createStandardDescriptionMap(standardsArray);
|
||||
|
||||
module.exports = presentIgnoreRules;
|
||||
|
||||
function presentIgnoreRules (ignore) {
|
||||
return ignore.map(function (name) {
|
||||
return {
|
||||
name: name,
|
||||
description: rules[name]
|
||||
};
|
||||
});
|
||||
function presentIgnoreRules(ignore) {
|
||||
return ignore.map(function(name) {
|
||||
return {
|
||||
name: name,
|
||||
description: rules[name]
|
||||
};
|
||||
});
|
||||
}
|
||||
|
||||
function createStandardDescriptionMap (standards) {
|
||||
var map = {};
|
||||
standards.forEach(function (standard) {
|
||||
standard.rules.forEach(function (rule) {
|
||||
map[rule.name] = rule.description;
|
||||
});
|
||||
});
|
||||
return map;
|
||||
function createStandardDescriptionMap(standards) {
|
||||
var map = {};
|
||||
standards.forEach(function(standard) {
|
||||
standard.rules.forEach(function(rule) {
|
||||
map[rule.name] = rule.description;
|
||||
});
|
||||
});
|
||||
return map;
|
||||
}
|
||||
|
@@ -20,13 +20,13 @@ var moment = require('moment');
|
||||
|
||||
module.exports = presentResultList;
|
||||
|
||||
function presentResultList (results) {
|
||||
var resultsByDay = _.groupBy(results, function (result) {
|
||||
return moment(result.date).format('YYYY-MM-DD');
|
||||
});
|
||||
var uniqueDayResults = [];
|
||||
_.keys(resultsByDay).forEach(function (day) {
|
||||
uniqueDayResults.push(resultsByDay[day][0]);
|
||||
});
|
||||
return uniqueDayResults;
|
||||
function presentResultList(results) {
|
||||
var resultsByDay = _.groupBy(results, function(result) {
|
||||
return moment(result.date).format('YYYY-MM-DD');
|
||||
});
|
||||
var uniqueDayResults = [];
|
||||
_.keys(resultsByDay).forEach(function(day) {
|
||||
uniqueDayResults.push(resultsByDay[day][0]);
|
||||
});
|
||||
return uniqueDayResults;
|
||||
}
|
||||
|
@@ -20,33 +20,33 @@ var presentIgnoreRules = require('./ignore');
|
||||
|
||||
module.exports = presentResult;
|
||||
|
||||
function presentResult (result) {
|
||||
function presentResult(result) {
|
||||
|
||||
// Add additional info
|
||||
result.href = '/' + result.task + '/' + result.id;
|
||||
result.hrefCsv = '/' + result.task + '/' + result.id + '.csv';
|
||||
result.hrefJson = '/' + result.task + '/' + result.id + '.json';
|
||||
// Add additional info
|
||||
result.href = '/' + result.task + '/' + result.id;
|
||||
result.hrefCsv = '/' + result.task + '/' + result.id + '.csv';
|
||||
result.hrefJson = '/' + result.task + '/' + result.id + '.json';
|
||||
|
||||
// Parse date
|
||||
result.date = new Date(result.date);
|
||||
// Parse date
|
||||
result.date = new Date(result.date);
|
||||
|
||||
// Enhance the ignored rules
|
||||
result.ignore = presentIgnoreRules(result.ignore);
|
||||
// Enhance the ignored rules
|
||||
result.ignore = presentIgnoreRules(result.ignore);
|
||||
|
||||
// Split out message types
|
||||
if (result.results) {
|
||||
var groupedByType = _.groupBy(result.results, 'type');
|
||||
['error', 'warning', 'notice'].forEach(function (type) {
|
||||
var pluralType = type + 's';
|
||||
var results = groupedByType[type] || [];
|
||||
var groupedByCode = _.groupBy(results, 'code');
|
||||
result[pluralType] = _.keys(groupedByCode).map(function (group) {
|
||||
var firstMessage = groupedByCode[group][0];
|
||||
firstMessage.count = groupedByCode[group].length;
|
||||
return firstMessage;
|
||||
});
|
||||
});
|
||||
}
|
||||
// Split out message types
|
||||
if (result.results) {
|
||||
var groupedByType = _.groupBy(result.results, 'type');
|
||||
['error', 'warning', 'notice'].forEach(function(type) {
|
||||
var pluralType = type + 's';
|
||||
var results = groupedByType[type] || [];
|
||||
var groupedByCode = _.groupBy(results, 'code');
|
||||
result[pluralType] = _.keys(groupedByCode).map(function(group) {
|
||||
var firstMessage = groupedByCode[group][0];
|
||||
firstMessage.count = groupedByCode[group].length;
|
||||
return firstMessage;
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
return result;
|
||||
return result;
|
||||
}
|
||||
|
@@ -15,31 +15,30 @@
|
||||
|
||||
'use strict';
|
||||
|
||||
var _ = require('underscore');
|
||||
var presentIgnoreRules = require('./ignore');
|
||||
var presentResult = require('./result');
|
||||
|
||||
module.exports = presentTask;
|
||||
|
||||
function presentTask (task) {
|
||||
function presentTask(task) {
|
||||
|
||||
// Add additional info
|
||||
task.href = '/' + task.id;
|
||||
task.hrefDelete = '/' + task.id + '/delete';
|
||||
task.hrefRun = '/' + task.id + '/run';
|
||||
task.hrefJson = '/' + task.id + '.json';
|
||||
task.hrefEdit = '/' + task.id + '/edit';
|
||||
task.hrefIgnore = '/' + task.id + '/ignore';
|
||||
task.hrefUnignore = '/' + task.id + '/unignore';
|
||||
// Add additional info
|
||||
task.href = '/' + task.id;
|
||||
task.hrefDelete = '/' + task.id + '/delete';
|
||||
task.hrefRun = '/' + task.id + '/run';
|
||||
task.hrefJson = '/' + task.id + '.json';
|
||||
task.hrefEdit = '/' + task.id + '/edit';
|
||||
task.hrefIgnore = '/' + task.id + '/ignore';
|
||||
task.hrefUnignore = '/' + task.id + '/unignore';
|
||||
|
||||
// Enhance the ignored rules
|
||||
task.ignore = presentIgnoreRules(task.ignore);
|
||||
// Enhance the ignored rules
|
||||
task.ignore = presentIgnoreRules(task.ignore);
|
||||
|
||||
// Present the last result if present
|
||||
if (task.last_result) {
|
||||
task.lastResult = presentResult(task.last_result);
|
||||
delete task.last_result;
|
||||
}
|
||||
// Present the last result if present
|
||||
if (task.last_result) {
|
||||
task.lastResult = presentResult(task.last_result);
|
||||
delete task.last_result;
|
||||
}
|
||||
|
||||
return task;
|
||||
return task;
|
||||
}
|
||||
|
@@ -21,4 +21,6 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
{{> result-header}}
|
||||
|
||||
{{> result}}
|
||||
<div class="single-result">
|
||||
{{> result}}
|
||||
</div>
|
||||
|
@@ -104,7 +104,7 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<p class="control-label"><b>Ignore these rules</b> <a target="_blank" href="https://github.com/nature/pa11y/wiki/HTML-CodeSniffer-Rules">(full list of rules here)</a></p>
|
||||
<p class="control-label"><b>Ignore these rules</b> <a target="_blank" href="https://github.com/springernature/pa11y/wiki/HTML-CodeSniffer-Rules">(full list of rules here)</a></p>
|
||||
|
||||
<div class="standards-lists">
|
||||
{{#standards}}
|
||||
|
@@ -76,8 +76,6 @@ along with pa11y-dashboard. If not, see <http://www.gnu.org/licenses/>.
|
||||
{{> graph}}
|
||||
{{/unless}}
|
||||
|
||||
{{> result-selector}}
|
||||
|
||||
{{/if}}
|
||||
|
||||
{{#if mainResult}}
|
||||
|
Reference in New Issue
Block a user