Remove charts from model direcory and add speed benchmarks

With speed benchmarks, charts ended up taking up too much space – and they were mostly data porn and not particularly useful anyways. Instead, we might add a "Compare" page that fetches all models and lets the user compare two or more models in terms of accuracy, speed etc.
This commit is contained in:
ines 2017-10-29 03:58:19 +01:00
parent 47fd254ba7
commit ae2ad5becc
6 changed files with 57 additions and 68 deletions

View File

@ -14,7 +14,7 @@
- MODEL_META = public.models._data.MODEL_META - MODEL_META = public.models._data.MODEL_META
- MODEL_LICENSES = public.models._data.MODEL_LICENSES - MODEL_LICENSES = public.models._data.MODEL_LICENSES
- MODEL_ACCURACY = public.models._data.MODEL_ACCURACY - MODEL_BENCHMARKS = public.models._data.MODEL_BENCHMARKS
- EXAMPLE_SENTENCES = public.models._data.EXAMPLE_SENTENCES - EXAMPLE_SENTENCES = public.models._data.EXAMPLE_SENTENCES
- IS_PAGE = (SECTION != "index") && !landing - IS_PAGE = (SECTION != "index") && !landing

View File

@ -26,7 +26,7 @@ for id in CURRENT_MODELS
| about this model, see the overview of the | about this model, see the overview of the
| #[+a(gh("spacy-models") + "/releases") latest model releases]. | #[+a(gh("spacy-models") + "/releases") latest model releases].
+table(data-tpl=id data-tpl-key="table") +table.o-block-small(data-tpl=id data-tpl-key="table")
+row +row
+cell #[+label Language] +cell #[+label Language]
+cell #[+tag=comps.lang] #{LANGUAGES[comps.lang]} +cell #[+tag=comps.lang] #{LANGUAGES[comps.lang]}
@ -56,12 +56,14 @@ for id in CURRENT_MODELS
select.o-field__select.u-text-small(data-tpl=id data-tpl-key="compat") select.o-field__select.u-text-small(data-tpl=id data-tpl-key="compat")
.o-empty(data-tpl=id data-tpl-key="compat-versions")   .o-empty(data-tpl=id data-tpl-key="compat-versions")  
section(data-tpl=id data-tpl-key="accuracy-wrapper" style="display: none") section(data-tpl=id data-tpl-key="benchmarks" style="display: none")
+grid.o-no-block +grid.o-block-small
+grid-col("third") for keys, label in MODEL_BENCHMARKS
+h(4) Accuracy .u-flex-full.u-padding-small(data-tpl=id data-tpl-key=label.toLowerCase() style="display: none")
+table.o-block-small +table.o-block-small
for label, field in MODEL_ACCURACY +row("head")
+head-cell(colspan="2")=(MODEL_META["benchmark_" + label] || label)
for label, field in keys
+row(style="display: none") +row(style="display: none")
+cell.u-nowrap +cell.u-nowrap
+label=label +label=label
@ -70,8 +72,4 @@ for id in CURRENT_MODELS
+cell.u-text-right(data-tpl=id data-tpl-key=field) +cell.u-text-right(data-tpl=id data-tpl-key=field)
| n/a | n/a
+grid-col("two-thirds")
+h(4) Comparison
+chart(id).u-padding-small
p.u-text-small.u-color-dark(data-tpl=id data-tpl-key="notes") p.u-text-small.u-color-dark(data-tpl=id data-tpl-key="notes")

View File

@ -6,9 +6,6 @@ if quickstart
if IS_PAGE if IS_PAGE
script(src="/assets/js/in-view.min.js") script(src="/assets/js/in-view.min.js")
if HAS_MODELS
script(src="/assets/js/chart.min.js")
if environment == "deploy" if environment == "deploy"
script(async src="https://www.google-analytics.com/analytics.js") script(async src="https://www.google-analytics.com/analytics.js")
@ -35,7 +32,7 @@ script
| }; | };
if HAS_MODELS if HAS_MODELS
| new ModelLoader('!{MODELS_REPO}', !{JSON.stringify(CURRENT_MODELS)}, !{JSON.stringify(MODEL_LICENSES)}, !{JSON.stringify(MODEL_ACCURACY)}); | new ModelLoader('!{MODELS_REPO}', !{JSON.stringify(CURRENT_MODELS)}, !{JSON.stringify(MODEL_LICENSES)}, !{JSON.stringify(MODEL_BENCHMARKS)});
if environment == "deploy" if environment == "deploy"
| window.ga=window.ga||function(){ | window.ga=window.ga||function(){

View File

@ -108,22 +108,12 @@ class ModelLoader {
* @param {Object} licenses - License IDs mapped to URLs. * @param {Object} licenses - License IDs mapped to URLs.
* @param {Object} accKeys - Available accuracy keys mapped to display labels. * @param {Object} accKeys - Available accuracy keys mapped to display labels.
*/ */
constructor(repo, models = [], licenses = {}, accKeys = {}) { constructor(repo, models = [], licenses = {}, benchmarkKeys = {}) {
this.url = `https://raw.githubusercontent.com/${repo}/master`; this.url = `https://raw.githubusercontent.com/${repo}/master`;
this.repo = `https://github.com/${repo}`; this.repo = `https://github.com/${repo}`;
this.modelIds = models; this.modelIds = models;
this.licenses = licenses; this.licenses = licenses;
this.accKeys = accKeys; this.benchKeys = benchmarkKeys;
this.chartColor = '#09a3d5';
this.chartOptions = {
type: 'bar',
options: { responsive: true, scales: {
yAxes: [{ label: 'Accuracy', ticks: { suggestedMin: 70 }}],
xAxes: [{ barPercentage: 0.425 }]
}}
}
Chart.defaults.global.legend.position = 'bottom';
Chart.defaults.global.defaultFontFamily = "-apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Helvetica, Arial, sans-serif, 'Apple Color Emoji', 'Segoe UI Emoji', 'Segoe UI Symbol'";
this.init(); this.init();
} }
@ -171,7 +161,7 @@ class ModelLoader {
/** /**
* Update model details in tables. Currently quite hacky :( * Update model details in tables. Currently quite hacky :(
*/ */
render({ lang, name, version, sources, pipeline, vectors, url, author, license, accuracy, size, description, notes }) { render({ lang, name, version, sources, pipeline, vectors, url, author, license, accuracy, speed, size, description, notes }) {
const modelId = `${lang}_${name}`; const modelId = `${lang}_${name}`;
const model = `${modelId}-${version}`; const model = `${modelId}-${version}`;
const template = new Templater(modelId); const template = new Templater(modelId);
@ -194,11 +184,31 @@ class ModelLoader {
if (license) template.fill('license', this.licenses[license] ? getLink(license, this.licenses[license]) : license, true); if (license) template.fill('license', this.licenses[license] ? getLink(license, this.licenses[license]) : license, true);
template.get('download').setAttribute('href', `${this.repo}/releases/tag/${model}`); template.get('download').setAttribute('href', `${this.repo}/releases/tag/${model}`);
if (accuracy) this.renderAccuracy(template, accuracy, modelId);
this.renderBenchmarks(template, accuracy, speed);
this.renderCompat(template, modelId); this.renderCompat(template, modelId);
template.get('table').removeAttribute('data-loading'); template.get('table').removeAttribute('data-loading');
} }
renderBenchmarks(template, accuracy = {}, speed = {}) {
if (!accuracy && !speed) return;
template.get('benchmarks').style.display = 'block';
this.renderTable(template, 'parser', accuracy, val => val.toFixed(2));
this.renderTable(template, 'ner', accuracy, val => val.toFixed(2));
this.renderTable(template, 'speed', speed, Math.round);
}
renderTable(template, id, benchmarks, convertVal = val => val) {
if (!this.benchKeys[id] || !Object.keys(this.benchKeys[id]).some(key => benchmarks[key])) return;
const keys = Object.keys(this.benchKeys[id]).map(k => benchmarks[k] ? k : false).filter(k => k);
template.get(id).style.display = 'block';
for (let key of keys) {
template
.fill(key, this.convertNumber(convertVal(benchmarks[key])))
.parentElement.style.display = 'table-row';
}
}
renderCompat(template, modelId) { renderCompat(template, modelId) {
template.get('compat-wrapper').style.display = 'table-row'; template.get('compat-wrapper').style.display = 'table-row';
const options = Object.keys(this.compat).map(v => `<option value="${v}">v${v}</option>`).join(''); const options = Object.keys(this.compat).map(v => `<option value="${v}">v${v}</option>`).join('');
@ -211,24 +221,6 @@ class ModelLoader {
}); });
} }
renderAccuracy(template, accuracy, modelId, compare=false) {
template.get('accuracy-wrapper').style.display = 'block';
const metaKeys = Object.keys(this.accKeys).map(k => accuracy[k] ? k : false).filter(k => k);
for (let key of metaKeys) {
template.fill(key, accuracy[key].toFixed(2)).parentElement.style.display = 'table-row';
}
this.chartOptions.options.legend = { display: compare }
new Chart(`chart_${modelId}`, Object.assign({}, this.chartOptions, { data: {
datasets: [{
label: modelId,
data: metaKeys.map(key => accuracy[key].toFixed(2)),
backgroundColor: this.chartColor
}],
labels: metaKeys.map(key => this.accKeys[key])
}}))
}
getLatestVersion(model, compat = {}) { getLatestVersion(model, compat = {}) {
for (let spacy_v of Object.keys(compat)) { for (let spacy_v of Object.keys(compat)) {
const models = compat[spacy_v]; const models = compat[spacy_v];

View File

@ -50,8 +50,13 @@
"ents_f": "Entities (F-score)", "ents_f": "Entities (F-score)",
"ents_p": "Entities (precision)", "ents_p": "Entities (precision)",
"ents_r": "Entities (recall)", "ents_r": "Entities (recall)",
"cpu": "words per second on CPU",
"gpu": "words per second on GPU",
"pipeline": "Processing pipeline components in order", "pipeline": "Processing pipeline components in order",
"sources": "Sources of training data" "sources": "Sources of training data",
"benchmark_parser": "Parser accuracy",
"benchmark_ner": "NER accuracy",
"benchmark_speed": "Speed"
}, },
"MODEL_LICENSES": { "MODEL_LICENSES": {
@ -62,13 +67,10 @@
"GPL": "http://www.gnu.de/documents/gpl.en.html" "GPL": "http://www.gnu.de/documents/gpl.en.html"
}, },
"MODEL_ACCURACY": { "MODEL_BENCHMARKS": {
"uas": "UAS", "parser": { "uas": "UAS", "las": "LAS", "tags_acc": "POS" },
"las": "LAS", "ner": { "ents_f": "NER F", "ents_p": "NER P", "ents_r": "NER R" },
"tags_acc": "POS", "speed": { "nwords": "Words", "cpu": "w/s CPU", "gpu": "w/s GPU" }
"ents_f": "NER F",
"ents_p": "NER P",
"ents_r": "NER R"
}, },
"LANGUAGES": { "LANGUAGES": {

View File

@ -609,8 +609,8 @@ include _includes/_mixins
+code(false, "json").o-no-block "CC BY-SA 3.0": "http://..." +code(false, "json").o-no-block "CC BY-SA 3.0": "http://..."
+row +row
+cell #[code MODEL_ACCURACY] +cell #[code MODEL_BENCHMARKS]
+cell Display labels for accuracy keys. +cell Display labels for accuracy and speed.
+cell +cell
+code(false, "json").o-no-block "ents_f": "NER F" +code(false, "json").o-no-block "ents_f": "NER F"