Files
shields/services/codeclimate/codeclimate-analysis.service.js
dependabot[bot] b9d96755ec chore(deps-dev): bump prettier from 2.8.8 to 3.0.0 (#9357)
* chore(deps-dev): bump prettier from 2.8.8 to 3.0.0

Bumps [prettier](https://github.com/prettier/prettier) from 2.8.8 to 3.0.0.
- [Release notes](https://github.com/prettier/prettier/releases)
- [Changelog](https://github.com/prettier/prettier/blob/main/CHANGELOG.md)
- [Commits](https://github.com/prettier/prettier/compare/2.8.8...3.0.0)

---
updated-dependencies:
- dependency-name: prettier
  dependency-type: direct:development
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>

* reformat all the things (prettier 3)

* update tests to await calls to prettier.format()

---------

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
Co-authored-by: chris48s <git@chris-shaw.dev>
2023-07-10 09:27:51 +00:00

172 lines
4.6 KiB
JavaScript

import Joi from 'joi'
import { colorScale, letterScore } from '../color-formatters.js'
import { nonNegativeInteger } from '../validators.js'
import { BaseJsonService, NotFound } from '../index.js'
import { keywords, isLetterGrade, fetchRepo } from './codeclimate-common.js'
const schema = Joi.object({
data: Joi.object({
meta: Joi.object({
issues_count: nonNegativeInteger,
}).required(),
attributes: Joi.object({
ratings: Joi.array()
.items(
Joi.object({
letter: isLetterGrade,
measure: Joi.object({
value: Joi.number().required(),
}).required(),
}),
)
.length(1)
.required(),
}).required(),
}).required(),
}).required()
const maintainabilityColorScale = colorScale(
[50, 80, 90, 95],
['red', 'yellow', 'yellowgreen', 'green', 'brightgreen'],
)
const techDebtColorScale = colorScale(
[5, 10, 20, 50],
['brightgreen', 'green', 'yellowgreen', 'yellow', 'red'],
)
const issueColorScale = colorScale(
[1, 5, 10, 20],
['brightgreen', 'green', 'yellowgreen', 'yellow', 'red'],
)
const variantMap = {
maintainability: {
transform: data => ({
maintainabilityLetter: data.attributes.ratings[0].letter,
}),
render: ({ maintainabilityLetter }) => ({
label: 'maintainability',
message: maintainabilityLetter,
color: letterScore(maintainabilityLetter),
}),
},
'maintainability-percentage': {
transform: data => ({
techDebtPercentage: data.attributes.ratings[0].measure.value,
}),
render: ({ techDebtPercentage }) => {
// maintainability = 100 - technical debt.
const maintainabilityPercentage = 100 - techDebtPercentage
return {
label: 'maintainability',
message: `${maintainabilityPercentage.toFixed(0)}%`,
color: maintainabilityColorScale(maintainabilityPercentage),
}
},
},
'tech-debt': {
transform: data => ({
techDebtPercentage: data.attributes.ratings[0].measure.value,
}),
render: ({ techDebtPercentage }) => ({
label: 'technical debt',
message: `${techDebtPercentage.toFixed(0)}%`,
color: techDebtColorScale(techDebtPercentage),
}),
},
issues: {
transform: data => ({
issueCount: data.meta.issues_count,
}),
render: ({ issueCount }) => ({
label: 'issues',
message: `${issueCount}`,
color: issueColorScale(issueCount),
}),
},
}
export default class CodeclimateAnalysis extends BaseJsonService {
static category = 'analysis'
static route = {
base: 'codeclimate',
pattern:
':variant(maintainability|maintainability-percentage|tech-debt|issues)/:user/:repo',
}
static examples = [
{
title: 'Code Climate maintainability',
pattern:
':format(maintainability|maintainability-percentage)/:user/:repo',
namedParams: {
format: 'maintainability',
user: 'tensorflow',
repo: 'models',
},
staticPreview: this.render({
variant: 'maintainability',
maintainabilityLetter: 'F',
}),
keywords,
},
{
title: 'Code Climate issues',
pattern: 'issues/:user/:repo',
namedParams: { user: 'twbs', repo: 'bootstrap' },
staticPreview: this.render({
variant: 'issues',
issueCount: '89',
}),
keywords,
},
{
title: 'Code Climate technical debt',
pattern: 'tech-debt/:user/:repo',
namedParams: { user: 'tensorflow', repo: 'models' },
staticPreview: this.render({
variant: 'tech-debt',
techDebtPercentage: 3.0,
}),
keywords,
},
]
static render({ variant, ...props }) {
const { render } = variantMap[variant]
return render(props)
}
async fetch({ user, repo }) {
const repoInfos = await fetchRepo(this, { user, repo })
const repoInfosWithSnapshot = repoInfos.filter(
repoInfo => repoInfo.relationships.latest_default_branch_snapshot.data,
)
if (repoInfosWithSnapshot.length === 0) {
throw new NotFound({ prettyMessage: 'snapshot not found' })
}
const {
id: repoId,
relationships: {
latest_default_branch_snapshot: { data: snapshotInfo },
},
} = repoInfosWithSnapshot[0]
const { data } = await this._requestJson({
schema,
url: `https://api.codeclimate.com/v1/repos/${repoId}/snapshots/${snapshotInfo.id}`,
})
return data
}
async handle({ variant, user, repo }) {
const { transform } = variantMap[variant]
const data = await this.fetch({ user, repo })
const props = transform(data)
return this.constructor.render({
variant,
...props,
})
}
}