[poeditor] Adding POEditor badge (#4812)

* Add POEditor badge for language translation progress

* Add example, tests, and documentation key for POEditor

* Remove redundant required() as indicated in PR review

* Throw errors instead of rendering red badge

* small wording tweak: mention read-only again

Co-authored-by: chris48s <chris.shaw480@gmail.com>
Co-authored-by: repo-ranger[bot] <39074581+repo-ranger[bot]@users.noreply.github.com>
This commit is contained in:
Cimbali
2020-03-27 22:35:36 +01:00
committed by GitHub
parent 831251cf16
commit 78fd9502a9
2 changed files with 209 additions and 0 deletions

View File

@@ -0,0 +1,113 @@
'use strict'
const Joi = require('@hapi/joi')
const { nonNegativeInteger } = require('../validators')
const { coveragePercentage } = require('../color-formatters')
const { BaseJsonService, InvalidResponse } = require('..')
const documentation = `
<p>
You must specify the read-only API token from the POEditor account to which the project belongs.
</p>
<p>
As per <a href="https://poeditor.com/docs/api">the POEditor API documentation</a>,
<q>all requests to the API must contain the parameter api_token. You can get a read-only key from your POEditor account.
You'll find it in <a href="https://poeditor.com/account/api">My Account > API Access</a>.</q>
</p>
`
const schema = Joi.object({
response: Joi.object({
code: nonNegativeInteger,
message: Joi.string().required(),
}).required(),
result: Joi.object({
languages: Joi.array()
.items({
name: Joi.string().required(),
code: Joi.string().required(),
percentage: Joi.number()
.min(0)
.max(100)
.required(),
})
.required(),
}),
}).required()
const queryParamSchema = Joi.object({
token: Joi.string().required(),
}).required()
module.exports = class POEditor extends BaseJsonService {
static get category() {
return 'other'
}
static get route() {
return {
base: 'poeditor',
pattern: 'progress/:projectId/:languageCode',
queryParamSchema,
}
}
static get examples() {
return [
{
title: 'POEditor',
namedParams: { projectId: '323337', languageCode: 'fr' },
queryParams: { token: 'abc123def456' },
staticPreview: this.render({
code: 200,
message: 'OK',
language: { percentage: 93, code: 'fr', name: 'French' },
}),
keywords: ['l10n'],
documentation,
},
]
}
static render({ code, message, language }) {
if (code !== 200) {
throw new InvalidResponse({ prettyMessage: message })
}
if (language === undefined) {
throw new InvalidResponse({ prettyMessage: 'Language not in project' })
}
return {
label: language.name,
message: `${language.percentage.toFixed(0)}%`,
color: coveragePercentage(language.percentage),
}
}
async fetch({ projectId, token }) {
return this._requestJson({
schema,
url: 'https://api.poeditor.com/v2/languages/list',
options: {
method: 'POST',
form: {
api_token: token,
id: projectId,
},
},
})
}
async handle({ projectId, languageCode }, { token }) {
const {
response: { code, message },
result: { languages } = { languages: [] },
} = await this.fetch({ projectId, token })
return this.constructor.render({
code,
message,
language: languages.find(lang => lang.code === languageCode),
})
}
}

View File

@@ -0,0 +1,96 @@
'use strict'
const { isIntegerPercentage } = require('../test-validators')
const t = (module.exports = require('../tester').createServiceTester())
t.create('gets POEditor progress online')
.get('/progress/323337/de.json?token=7a666b44c0985d16a7b59748f488275c')
.expectBadge({
label: 'German',
message: isIntegerPercentage,
})
t.create('gets POEditor progress online')
.get('/progress/1/zh.json?token=7a666b44c0985d16a7b59748f488275c')
.expectBadge({
label: 'other',
message: "You don't have permission to access this resource",
})
// https:/.com/docs/api#languages_list_response
const apiResponse = {
response: {
status: 'success',
code: '200',
message: 'OK',
},
result: {
languages: [
{
name: 'English',
code: 'en',
translations: 13,
percentage: 12.5,
updated: '2015-05-04T14:21:41+0000',
},
{
name: 'French',
code: 'fr',
translations: 70,
percentage: 68.75,
updated: '2015-04-30T08:59:34+0000',
},
],
},
}
t.create('gets mock POEditor progress')
.get('/progress/1234/fr.json?token=abc123def456')
.intercept(nock =>
nock('https://api.poeditor.com')
.post('/v2/languages/list', {
id: '1234',
api_token: 'abc123def456',
})
.reply(200, apiResponse)
)
.expectBadge({
label: 'French',
message: '69%',
})
t.create('handles requests for missing languages')
.get('/progress/1234/zh.json?token=abc123def456')
.intercept(nock =>
nock('https://api.poeditor.com')
.post('/v2/languages/list', {
id: '1234',
api_token: 'abc123def456',
})
.reply(200, apiResponse)
)
.expectBadge({
label: 'other',
message: 'Language not in project',
})
t.create('handles requests for wrong keys')
.get('/progress/1234/fr.json?token=abc123def456')
.intercept(nock =>
nock('https://api.poeditor.com')
.post('/v2/languages/list', {
id: '1234',
api_token: 'abc123def456',
})
.reply(200, {
response: {
status: 'fail',
code: '403',
message: "You don't have permission to access this resource",
},
})
)
.expectBadge({
label: 'other',
message: "You don't have permission to access this resource",
})