From 0a875ca9ffa13fd2cb5ec2a21c642bb86f93c2c2 Mon Sep 17 00:00:00 2001 From: Maxime Beauchemin Date: Tue, 13 Nov 2018 14:37:07 -0800 Subject: [PATCH] [bugfix] prevent d3-format from raising Since https://github.com/apache/incubator-superset/pull/6287 and effectively moving to a new version of d3, d3-format and d3-time-format raises when receiving invalid input strings. This code wraps the potential issues inside `try` blocks that will effectively return an `ERROR` string as output to the formatting function. (cherry picked from commit 5b668bdd5619da5bcb8feff3624776241a0d3068) --- .../spec/javascripts/modules/utils_spec.jsx | 4 +++- superset/assets/src/modules/utils.js | 19 ++++++++++++++++--- .../assets/src/visualizations/nvd3/NVD3Vis.js | 4 +++- 3 files changed, 22 insertions(+), 5 deletions(-) diff --git a/superset/assets/spec/javascripts/modules/utils_spec.jsx b/superset/assets/spec/javascripts/modules/utils_spec.jsx index d1566c2dda988..0faaf31cb1959 100644 --- a/superset/assets/spec/javascripts/modules/utils_spec.jsx +++ b/superset/assets/spec/javascripts/modules/utils_spec.jsx @@ -29,6 +29,7 @@ describe('utils', () => { expect(d3format('.3s', 1234)).toBe('1.23k'); expect(d3format('.3s', 1237)).toBe('1.24k'); expect(d3format('', 1237)).toBe('1.24k'); + expect(d3format('.2efd2.ef.2.e', 1237)).toBe('ERROR'); }); }); @@ -45,8 +46,9 @@ describe('utils', () => { it('is a function', () => { expect(typeof d3TimeFormatPreset).toBe('function'); }); - it('returns a working time formatter', () => { + it('returns a working formatter', () => { expect(d3FormatPreset('smart_date')(0)).toBe('1970'); + expect(d3FormatPreset('%%GIBBERISH')(0)).toBe('ERROR'); }); }); diff --git a/superset/assets/src/modules/utils.js b/superset/assets/src/modules/utils.js index 18e4764aa1840..249d92cf79c02 100644 --- a/superset/assets/src/modules/utils.js +++ b/superset/assets/src/modules/utils.js @@ -6,6 +6,7 @@ import { timeFormat as d3TimeFormat } from 'd3-time-format'; import { formatDate, UTC } from './dates'; const siFormatter = d3Format('.3s'); +const ERROR_STRING = 'ERROR'; export function defaultNumberFormatter(n) { let si = siFormatter(n); @@ -22,7 +23,13 @@ export function d3FormatPreset(format) { return formatDate; } if (format) { - return d3Format(format); + try { + return d3Format(format); + } catch (e) { + // eslint-disable-next-line no-console + console.warn(e); + return () => ERROR_STRING; + } } return defaultNumberFormatter; } @@ -45,12 +52,18 @@ export function d3format(format, number) { format = format || '.3s'; // Formats a number and memoizes formatters to be reused if (!(format in formatters)) { - formatters[format] = d3Format(format); + try { + formatters[format] = d3Format(format); + } catch (e) { + // eslint-disable-next-line no-console + console.warn(e); + return ERROR_STRING; + } } try { return formatters[format](number); } catch (e) { - return 'ERR'; + return ERROR_STRING; } } diff --git a/superset/assets/src/visualizations/nvd3/NVD3Vis.js b/superset/assets/src/visualizations/nvd3/NVD3Vis.js index 12e8ca36d12fe..cdd938f8936e3 100644 --- a/superset/assets/src/visualizations/nvd3/NVD3Vis.js +++ b/superset/assets/src/visualizations/nvd3/NVD3Vis.js @@ -456,11 +456,13 @@ function nvd3Vis(element, props) { chart.xScale(d3.scale.log()); } - let xAxisFormatter = d3FormatPreset(xAxisFormat); + let xAxisFormatter; if (isTimeSeries) { xAxisFormatter = d3TimeFormatPreset(xAxisFormat); // In tooltips, always use the verbose time format chart.interactiveLayer.tooltip.headerFormatter(formatDateVerbose); + } else { + xAxisFormatter = d3FormatPreset(xAxisFormat); } if (chart.x2Axis && chart.x2Axis.tickFormat) { chart.x2Axis.tickFormat(xAxisFormatter);