This is the preferred way we’re handling server URLs: in the query string. While we still have a mix of these, I’ve argued this way is better for these reasons: 1. It allows us to make the URLs a bit more standardized across services. 2. It makes the routes unambiguous. 3. It requires less code. While it introduces a URL-encoding requirement on the parameter, I think these tradeoffs are worth it.
181 lines
4.8 KiB
JavaScript
181 lines
4.8 KiB
JavaScript
'use strict'
|
|
|
|
const { colorScale } = require('../color-formatters')
|
|
const { metric } = require('../text-formatters')
|
|
const SonarBase = require('./sonar-base')
|
|
const {
|
|
getLabel,
|
|
documentation,
|
|
keywords,
|
|
queryParamWithFormatSchema,
|
|
} = require('./sonar-helpers')
|
|
|
|
const violationsColorScale = colorScale(
|
|
[1, 2, 3, 5],
|
|
['brightgreen', 'yellowgreen', 'yellow', 'orange', 'red']
|
|
)
|
|
|
|
const violationCategoryColorMap = {
|
|
blocker_violations: 'red',
|
|
critical_violations: 'orange',
|
|
major_violations: 'yellow',
|
|
minor_violations: 'yellowgreen',
|
|
info_violations: 'green',
|
|
}
|
|
|
|
module.exports = class SonarViolations extends SonarBase {
|
|
static get category() {
|
|
return 'analysis'
|
|
}
|
|
|
|
static get route() {
|
|
return {
|
|
base: 'sonar',
|
|
pattern:
|
|
':metric(violations|blocker_violations|critical_violations|major_violations|minor_violations|info_violations)/:component',
|
|
queryParamSchema: queryParamWithFormatSchema,
|
|
}
|
|
}
|
|
|
|
static get examples() {
|
|
return [
|
|
{
|
|
title: 'Sonar Violations (short format)',
|
|
namedParams: {
|
|
component: 'swellaby:azdo-shellcheck',
|
|
metric: 'violations',
|
|
},
|
|
queryParams: {
|
|
server: 'https://sonarcloud.io',
|
|
format: 'short',
|
|
sonarVersion: '4.2',
|
|
},
|
|
staticPreview: this.render({
|
|
violations: 0,
|
|
metricName: 'violations',
|
|
format: 'short',
|
|
}),
|
|
keywords,
|
|
documentation,
|
|
},
|
|
{
|
|
title: 'Sonar Violations (long format)',
|
|
namedParams: {
|
|
component: 'org.ow2.petals:petals-se-ase',
|
|
metric: 'violations',
|
|
},
|
|
queryParams: {
|
|
server: 'http://sonar.petalslink.com',
|
|
format: 'long',
|
|
},
|
|
staticPreview: this.render({
|
|
violations: {
|
|
info_violations: 2,
|
|
minor_violations: 1,
|
|
},
|
|
metricName: 'violations',
|
|
format: 'long',
|
|
}),
|
|
keywords,
|
|
documentation,
|
|
},
|
|
]
|
|
}
|
|
|
|
static get defaultBadgeData() {
|
|
return { label: 'violations' }
|
|
}
|
|
|
|
static renderLongViolationsBadge(violations) {
|
|
if (violations.violations === 0) {
|
|
return {
|
|
message: 0,
|
|
color: 'brightgreen',
|
|
}
|
|
}
|
|
|
|
let color
|
|
const violationSummary = []
|
|
|
|
if (violations.info_violations > 0) {
|
|
violationSummary.push(`${violations.info_violations} info`)
|
|
color = violationCategoryColorMap.info_violations
|
|
}
|
|
if (violations.minor_violations > 0) {
|
|
violationSummary.unshift(`${violations.minor_violations} minor`)
|
|
color = violationCategoryColorMap.minor_violations
|
|
}
|
|
if (violations.major_violations > 0) {
|
|
violationSummary.unshift(`${violations.major_violations} major`)
|
|
color = violationCategoryColorMap.major_violations
|
|
}
|
|
if (violations.critical_violations > 0) {
|
|
violationSummary.unshift(`${violations.critical_violations} critical`)
|
|
color = violationCategoryColorMap.critical_violations
|
|
}
|
|
if (violations.blocker_violations > 0) {
|
|
violationSummary.unshift(`${violations.blocker_violations} blocker`)
|
|
color = violationCategoryColorMap.blocker_violations
|
|
}
|
|
|
|
return {
|
|
message: violationSummary.join(', '),
|
|
color,
|
|
}
|
|
}
|
|
|
|
static render({ violations, metricName, format }) {
|
|
if (metricName === 'violations') {
|
|
if (format === 'long') {
|
|
return this.renderLongViolationsBadge(violations)
|
|
}
|
|
return {
|
|
message: metric(violations),
|
|
color: violationsColorScale(violations),
|
|
}
|
|
}
|
|
|
|
const color =
|
|
violations === 0 ? 'brightgreen' : violationCategoryColorMap[metricName]
|
|
|
|
return {
|
|
label: getLabel({ metric: metricName }),
|
|
message: `${metric(violations)}`,
|
|
color,
|
|
}
|
|
}
|
|
|
|
transformViolations({ json, sonarVersion, metric, format }) {
|
|
const metrics = this.transform({ json, sonarVersion })
|
|
if (metric !== 'violations' || format !== 'long') {
|
|
return { violations: metrics[metric] }
|
|
}
|
|
|
|
return { violations: metrics }
|
|
}
|
|
|
|
async handle({ component, metric }, { server, sonarVersion, format }) {
|
|
// If the user has requested the long format for the violations badge
|
|
// then we need to include each individual violation metric in the call to the API
|
|
// in order to get the count breakdown per each violation category.
|
|
const metricKeys =
|
|
metric === 'violations' && format === 'long'
|
|
? 'violations,blocker_violations,critical_violations,major_violations,minor_violations,info_violations'
|
|
: metric
|
|
const json = await this.fetch({
|
|
sonarVersion,
|
|
server,
|
|
component,
|
|
metricName: metricKeys,
|
|
})
|
|
|
|
const { violations } = this.transformViolations({
|
|
json,
|
|
sonarVersion,
|
|
metric,
|
|
format,
|
|
})
|
|
return this.constructor.render({ violations, metricName: metric, format })
|
|
}
|
|
}
|