From 2d81e7b6244d4e3c6cf550b14417f2607ffbaf83 Mon Sep 17 00:00:00 2001
From: Joe Li <56132941+jlyaoyuli@users.noreply.github.com>
Date: Fri, 8 Jul 2022 16:58:57 -0700
Subject: [PATCH] feat(frontend) Support LIST, STRUCT type in RuntimeConfig
parameters (#7991)
* Convert string-type user input to real-type pararmeter. (Currently supporting num, bool, str)
* Delete unnecessary console.log
* Change incorrect type.
Add TODO
Move inputConvert out from function component.
Check parameter type by PipelineSpec before converting.
* Format.
* Move inputConverter out from function component.
Delete unnecessary console.log
* Change inputConverter from if-else to switch.
Assign invalid input to null.
* 1. Change paramTypeIdx to ParameterTypeEnum.
2. Add tests to validate the changes.
* Format
* Support LIST and STRUCT type.
* 1. Fixed default value bugs.
2. Added unit test for LIST and STRUCT type parameters.
* Added type into param.
* Added comment.
* Added unit test for invalid JSON form input.
* Removed incorrect merge conflicts.
* Support double type.
Added and modified unit tests.
* Clean unnecessary expect() in the unit tests.
* Added additional tests to check for the existence of display value 4.56
---
.../components/NewRunParametersV2.test.tsx | 233 ++++++++++++++++--
.../src/components/NewRunParametersV2.tsx | 77 ++++--
2 files changed, 281 insertions(+), 29 deletions(-)
diff --git a/frontend/src/components/NewRunParametersV2.test.tsx b/frontend/src/components/NewRunParametersV2.test.tsx
index 284650a05df..4a81f38efee 100644
--- a/frontend/src/components/NewRunParametersV2.test.tsx
+++ b/frontend/src/components/NewRunParametersV2.test.tsx
@@ -48,14 +48,14 @@ describe('NewRunParametersV2', () => {
,
);
- expect(screen.getByText('Run parameters'));
- expect(screen.getByText('Specify parameters required by the pipeline'));
- expect(screen.getByText('strParam - STRING'));
- expect(screen.getByDisplayValue('string value'));
- expect(screen.getByText('boolParam - BOOLEAN'));
- expect(screen.getByDisplayValue('true'));
- expect(screen.getByText('intParam - NUMBER_INTEGER'));
- expect(screen.getByDisplayValue('123'));
+ screen.getByText('Run parameters');
+ screen.getByText('Specify parameters required by the pipeline');
+ screen.getByText('strParam - STRING');
+ screen.getByDisplayValue('string value');
+ screen.getByText('boolParam - BOOLEAN');
+ screen.getByDisplayValue('true');
+ screen.getByText('intParam - NUMBER_INTEGER');
+ screen.getByDisplayValue('123');
});
it('edits parameters', () => {
@@ -116,6 +116,7 @@ describe('NewRunParametersV2', () => {
expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
strParam: 'new string',
});
+ screen.getByDisplayValue('new string');
});
it('test convertInput function for string type without default value', () => {
@@ -131,7 +132,7 @@ describe('NewRunParametersV2', () => {
handlePipelineRootChange: jest.fn(),
handleParameterChange: handleParameterChangeSpy,
};
- render();
+ render();
const strParam = screen.getByLabelText('strParam - STRING');
fireEvent.change(strParam, { target: { value: 'new string' } });
@@ -139,6 +140,7 @@ describe('NewRunParametersV2', () => {
expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
strParam: 'new string',
});
+ screen.getByDisplayValue('new string');
});
it('test convertInput function for boolean type with default value', () => {
@@ -155,7 +157,7 @@ describe('NewRunParametersV2', () => {
handlePipelineRootChange: jest.fn(),
handleParameterChange: handleParameterChangeSpy,
};
- render();
+ render();
const boolParam = screen.getByDisplayValue('true');
fireEvent.change(boolParam, { target: { value: 'false' } });
@@ -163,6 +165,7 @@ describe('NewRunParametersV2', () => {
expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
boolParam: false,
});
+ screen.getByDisplayValue('false');
});
it('test convertInput function for boolean type without default value', () => {
@@ -178,7 +181,7 @@ describe('NewRunParametersV2', () => {
handlePipelineRootChange: jest.fn(),
handleParameterChange: handleParameterChangeSpy,
};
- render();
+ render();
const boolParam = screen.getByLabelText('boolParam - BOOLEAN');
fireEvent.change(boolParam, { target: { value: 'true' } });
@@ -186,6 +189,7 @@ describe('NewRunParametersV2', () => {
expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
boolParam: true,
});
+ screen.getByDisplayValue('true');
});
it('test convertInput function for boolean type with invalid input (Uppercase)', () => {
@@ -201,7 +205,7 @@ describe('NewRunParametersV2', () => {
handlePipelineRootChange: jest.fn(),
handleParameterChange: handleParameterChangeSpy,
};
- render();
+ render();
const boolParam = screen.getByLabelText('boolParam - BOOLEAN');
fireEvent.change(boolParam, { target: { value: 'True' } });
@@ -209,6 +213,7 @@ describe('NewRunParametersV2', () => {
expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
boolParam: null,
});
+ screen.getByDisplayValue('True');
});
it('test convertInput function for integer type with default value', () => {
@@ -225,7 +230,7 @@ describe('NewRunParametersV2', () => {
handlePipelineRootChange: jest.fn(),
handleParameterChange: handleParameterChangeSpy,
};
- render();
+ render();
const intParam = screen.getByDisplayValue('123');
fireEvent.change(intParam, { target: { value: '456' } });
@@ -233,6 +238,7 @@ describe('NewRunParametersV2', () => {
expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
intParam: 456,
});
+ screen.getByDisplayValue('456');
});
it('test convertInput function for integer type without default value', () => {
@@ -248,7 +254,7 @@ describe('NewRunParametersV2', () => {
handlePipelineRootChange: jest.fn(),
handleParameterChange: handleParameterChangeSpy,
};
- render();
+ render();
const intParam = screen.getByLabelText('intParam - NUMBER_INTEGER');
fireEvent.change(intParam, { target: { value: '789' } });
@@ -256,6 +262,7 @@ describe('NewRunParametersV2', () => {
expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
intParam: 789,
});
+ screen.getByDisplayValue('789');
});
it('test convertInput function for integer type with invalid input (float)', () => {
@@ -271,7 +278,7 @@ describe('NewRunParametersV2', () => {
handlePipelineRootChange: jest.fn(),
handleParameterChange: handleParameterChangeSpy,
};
- render();
+ render();
const intParam = screen.getByLabelText('intParam - NUMBER_INTEGER');
fireEvent.change(intParam, { target: { value: '7.89' } });
@@ -279,6 +286,202 @@ describe('NewRunParametersV2', () => {
expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
intParam: null,
});
+ screen.getByDisplayValue('7.89');
+ });
+
+ it('test convertInput function for double type with default value', () => {
+ const handleParameterChangeSpy = jest.fn();
+ const props = {
+ titleMessage: 'default Title',
+ pipelineRoot: 'defalut pipelineRoot',
+ specParameters: {
+ doubleParam: {
+ parameterType: ParameterType_ParameterTypeEnum.NUMBER_DOUBLE,
+ defaultValue: 1.23,
+ },
+ },
+ handlePipelineRootChange: jest.fn(),
+ handleParameterChange: handleParameterChangeSpy,
+ };
+ render();
+
+ const doubleParam = screen.getByDisplayValue('1.23');
+ fireEvent.change(doubleParam, { target: { value: '4.56' } });
+ expect(handleParameterChangeSpy).toHaveBeenCalledTimes(1);
+ expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
+ doubleParam: 4.56,
+ });
+ screen.getByDisplayValue('4.56');
+ });
+
+ it('test convertInput function for double type without default value', () => {
+ const handleParameterChangeSpy = jest.fn();
+ const props = {
+ titleMessage: 'default Title',
+ pipelineRoot: 'defalut pipelineRoot',
+ specParameters: {
+ doubleParam: {
+ parameterType: ParameterType_ParameterTypeEnum.NUMBER_DOUBLE,
+ },
+ },
+ handlePipelineRootChange: jest.fn(),
+ handleParameterChange: handleParameterChangeSpy,
+ };
+ render();
+
+ const doubleParam = screen.getByLabelText('doubleParam - NUMBER_DOUBLE');
+ fireEvent.change(doubleParam, { target: { value: '7.89' } });
+ expect(handleParameterChangeSpy).toHaveBeenCalledTimes(1);
+ expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
+ doubleParam: 7.89,
+ });
+ screen.getByDisplayValue('7.89');
+ });
+
+ it('test convertInput function for LIST type with default value', () => {
+ const handleParameterChangeSpy = jest.fn();
+ const props = {
+ titleMessage: 'default Title',
+ pipelineRoot: 'defalut pipelineRoot',
+ specParameters: {
+ listParam: {
+ parameterType: ParameterType_ParameterTypeEnum.LIST,
+ defaultValue: [1, 2, 3],
+ },
+ },
+ handlePipelineRootChange: jest.fn(),
+ handleParameterChange: handleParameterChangeSpy,
+ };
+ render();
+
+ const listParam = screen.getByDisplayValue('[1,2,3]');
+ fireEvent.change(listParam, { target: { value: '[4,5,6]' } });
+ expect(handleParameterChangeSpy).toHaveBeenCalledTimes(1);
+ expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
+ listParam: [4, 5, 6],
+ });
+ screen.getByDisplayValue('[4,5,6]');
+ });
+
+ it('test convertInput function for LIST type without default value', () => {
+ const handleParameterChangeSpy = jest.fn();
+ const props = {
+ titleMessage: 'default Title',
+ pipelineRoot: 'defalut pipelineRoot',
+ specParameters: {
+ listParam: {
+ parameterType: ParameterType_ParameterTypeEnum.LIST,
+ },
+ },
+ handlePipelineRootChange: jest.fn(),
+ handleParameterChange: handleParameterChangeSpy,
+ };
+ render();
+
+ const listParam = screen.getByLabelText('listParam - LIST');
+ fireEvent.change(listParam, { target: { value: '[4,5,6]' } });
+ expect(handleParameterChangeSpy).toHaveBeenCalledTimes(1);
+ expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
+ listParam: [4, 5, 6],
+ });
+ screen.getByDisplayValue('[4,5,6]');
+ });
+
+ it('test convertInput function for LIST type with invalid input (invalid JSON form)', () => {
+ const handleParameterChangeSpy = jest.fn();
+ const props = {
+ titleMessage: 'default Title',
+ pipelineRoot: 'defalut pipelineRoot',
+ specParameters: {
+ listParam: {
+ parameterType: ParameterType_ParameterTypeEnum.LIST,
+ },
+ },
+ handlePipelineRootChange: jest.fn(),
+ handleParameterChange: handleParameterChangeSpy,
+ };
+ render();
+
+ const listParam = screen.getByLabelText('listParam - LIST');
+ fireEvent.change(listParam, { target: { value: '[4,5,6' } });
+ expect(handleParameterChangeSpy).toHaveBeenCalledTimes(1);
+ expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
+ listParam: null,
+ });
+ screen.getByDisplayValue('[4,5,6');
+ });
+
+ it('test convertInput function for STRUCT type with default value', () => {
+ const handleParameterChangeSpy = jest.fn();
+ const props = {
+ titleMessage: 'default Title',
+ pipelineRoot: 'defalut pipelineRoot',
+ specParameters: {
+ structParam: {
+ parameterType: ParameterType_ParameterTypeEnum.STRUCT,
+ defaultValue: { A: 1, B: 2 },
+ },
+ },
+ handlePipelineRootChange: jest.fn(),
+ handleParameterChange: handleParameterChangeSpy,
+ };
+ render();
+
+ const structParam = screen.getByDisplayValue('{"A":1,"B":2}');
+ fireEvent.change(structParam, { target: { value: '{"C":3,"D":4}' } });
+ expect(handleParameterChangeSpy).toHaveBeenCalledTimes(1);
+ expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
+ structParam: { C: 3, D: 4 },
+ });
+ screen.getByDisplayValue('{"C":3,"D":4}');
+ });
+
+ it('test convertInput function for STRUCT type without default value', () => {
+ const handleParameterChangeSpy = jest.fn();
+ const props = {
+ titleMessage: 'default Title',
+ pipelineRoot: 'defalut pipelineRoot',
+ specParameters: {
+ structParam: {
+ parameterType: ParameterType_ParameterTypeEnum.STRUCT,
+ },
+ },
+ handlePipelineRootChange: jest.fn(),
+ handleParameterChange: handleParameterChangeSpy,
+ };
+ render();
+
+ const structParam = screen.getByLabelText('structParam - STRUCT');
+ fireEvent.change(structParam, { target: { value: '{"A":1,"B":2}' } });
+ expect(handleParameterChangeSpy).toHaveBeenCalledTimes(1);
+ expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
+ structParam: { A: 1, B: 2 },
+ });
+ screen.getByDisplayValue('{"A":1,"B":2}');
+ });
+
+ it('test convertInput function for STRUCT type with invalid input (invalid JSON form)', () => {
+ const handleParameterChangeSpy = jest.fn();
+ const props = {
+ titleMessage: 'default Title',
+ pipelineRoot: 'defalut pipelineRoot',
+ specParameters: {
+ structParam: {
+ parameterType: ParameterType_ParameterTypeEnum.STRUCT,
+ },
+ },
+ handlePipelineRootChange: jest.fn(),
+ handleParameterChange: handleParameterChangeSpy,
+ };
+ render();
+
+ const structParam = screen.getByLabelText('structParam - STRUCT');
+ fireEvent.change(structParam, { target: { value: '"A":1,"B":2' } });
+ expect(handleParameterChangeSpy).toHaveBeenCalledTimes(1);
+ expect(handleParameterChangeSpy).toHaveBeenLastCalledWith({
+ structParam: null,
+ });
+ screen.getByDisplayValue('"A":1,"B":2');
});
it('does not display any text fields if there are no parameters', () => {
diff --git a/frontend/src/components/NewRunParametersV2.tsx b/frontend/src/components/NewRunParametersV2.tsx
index d388514dec4..58591b77f71 100644
--- a/frontend/src/components/NewRunParametersV2.tsx
+++ b/frontend/src/components/NewRunParametersV2.tsx
@@ -71,6 +71,18 @@ function convertInput(paramStr: string, paramType: ParameterType_ParameterTypeEn
return Number(paramStr);
}
return null;
+ case ParameterType_ParameterTypeEnum.NUMBER_DOUBLE:
+ if (!Number.isNaN(Number(paramStr))) {
+ return Number(paramStr);
+ }
+ return null;
+ case ParameterType_ParameterTypeEnum.LIST:
+ case ParameterType_ParameterTypeEnum.STRUCT:
+ try {
+ return JSON.parse(paramStr);
+ } catch (err) {
+ return null;
+ }
default:
// TODO: (jlyaoyuli) Validate if the type of parameters matches the value
// If it doesn't throw an error message next to the TextField.
@@ -89,7 +101,21 @@ function NewRunParametersV2(props: NewRunParametersProps) {
Object.keys(props.specParameters).map(key => {
if (props.specParameters[key].defaultValue) {
// TODO(zijianjoy): Make sure to consider all types of parameters.
- runtimeParametersWithDefault[key] = props.specParameters[key].defaultValue;
+ let defaultValStr; // Convert default to string type first to avoid error from convertInput
+ switch (props.specParameters[key].parameterType) {
+ case ParameterType_ParameterTypeEnum.STRUCT:
+ case ParameterType_ParameterTypeEnum.LIST:
+ defaultValStr = JSON.stringify(props.specParameters[key].defaultValue);
+ break;
+ case ParameterType_ParameterTypeEnum.BOOLEAN:
+ case ParameterType_ParameterTypeEnum.NUMBER_INTEGER:
+ case ParameterType_ParameterTypeEnum.NUMBER_DOUBLE:
+ defaultValStr = props.specParameters[key].defaultValue.toString();
+ break;
+ default:
+ defaultValStr = props.specParameters[key].defaultValue;
+ }
+ runtimeParametersWithDefault[key] = defaultValStr;
}
});
setUpdatedParameters(runtimeParametersWithDefault);
@@ -145,6 +171,7 @@ function NewRunParametersV2(props: NewRunParametersProps) {
const param = {
key: `${k} - ${ParameterType_ParameterTypeEnum[v.parameterType]}`,
value: updatedParameters[k],
+ type: v.parameterType,
};
return (
@@ -160,7 +187,7 @@ function NewRunParametersV2(props: NewRunParametersProps) {
let parametersInRealType: RuntimeParameters = {};
Object.entries(nextUpdatedParameters).map(([k, v1]) => {
parametersInRealType[k] = convertInput(
- v1.toString(),
+ v1,
props.specParameters[k].parameterType,
);
});
@@ -182,6 +209,7 @@ export default NewRunParametersV2;
interface Param {
key: string;
value: any;
+ type: ParameterType_ParameterTypeEnum;
}
interface ParamEditorProps {
@@ -202,16 +230,23 @@ class ParamEditor extends React.Component {
prevState: ParamEditorState,
): { isInJsonForm: boolean; isJsonField: boolean } {
let isJson = true;
- try {
- const displayValue = JSON.parse('');
- // Nulls, booleans, strings, and numbers can all be parsed as JSON, but we don't care
- // about rendering. Note that `typeOf null` returns 'object'
- if (displayValue === null || typeof displayValue !== 'object') {
- throw new Error('Parsed JSON was neither an array nor an object. Using default renderer');
- }
- } catch (err) {
- isJson = false;
+ let paramType = nextProps.param.type;
+
+ switch (paramType) {
+ case ParameterType_ParameterTypeEnum.LIST:
+ case ParameterType_ParameterTypeEnum.STRUCT:
+ isJson = true;
+ break;
+ case ParameterType_ParameterTypeEnum.STRING:
+ case ParameterType_ParameterTypeEnum.BOOLEAN:
+ case ParameterType_ParameterTypeEnum.NUMBER_INTEGER:
+ case ParameterType_ParameterTypeEnum.NUMBER_DOUBLE:
+ isJson = false;
+ break;
+ default:
+ isJson = false;
}
+
return {
isInJsonForm: isJson,
isJsonField: prevState.isJsonField || isJson,
@@ -229,8 +264,22 @@ class ParamEditor extends React.Component {
const onClick = () => {
if (this.state.isInJsonForm) {
+ let paramType = param.type;
+ let displayValue;
+ switch (paramType) {
+ case ParameterType_ParameterTypeEnum.LIST:
+ displayValue = JSON.parse(param.value || '[]');
+ break;
+ case ParameterType_ParameterTypeEnum.STRUCT:
+ displayValue = JSON.parse(param.value || '{}');
+ break;
+ default:
+ // TODO(jlyaoyuli): If the type from PipelineSpec is either LIST or STURCT,
+ // but the user-input or default value is invalid JSON form, show error message.
+ displayValue = JSON.parse('');
+ }
+
// TODO(zijianjoy): JSON format needs to be struct or list type.
- const displayValue = JSON.parse(param.value?.string_value || '');
if (this.state.isEditorOpen) {
onChange(JSON.stringify(displayValue) || '');
} else {
@@ -249,8 +298,8 @@ class ParamEditor extends React.Component {
id={id}
disabled={this.state.isEditorOpen}
variant='outlined'
- // label={param.name}
- // value={param.value || ''}
+ label={param.key}
+ value={param.value || ''}
onChange={ev => onChange(ev.target.value || '')}
className={classes(commonCss.textField, css.textfield)}
InputProps={{