From 8cff767bc3bbe895d97f764451174fe93aa90e04 Mon Sep 17 00:00:00 2001 From: Joyce Er Date: Wed, 24 Mar 2021 13:08:30 -0700 Subject: [PATCH] Point fix for duplicated data in data viewer (#5259) * More data viewer style tweaks (#5055) * Add slice data functional tests (#5057) * Changelog and news entry * Fix smoke tests * Lint Co-authored-by: Don Jayamanne --- CHANGELOG.md | 12 + news/2 Fixes/5200.md | 1 + news/3 Code Health/5066.md | 1 + .../dataframes/vscodeDataFrame.py | 10 +- .../datascience/jupyter/kernelVariables.ts | 4 - .../data-explorer/cellFormatter.css | 4 + .../data-explorer/cellFormatter.tsx | 21 +- src/datascience-ui/data-explorer/index.tsx | 2 - .../data-explorer/mainPanel.css | 7 + .../data-explorer/mainPanel.tsx | 11 +- .../data-explorer/reactSlickGrid.css | 47 +- .../data-explorer/reactSlickGridFilterBox.tsx | 16 + .../data-explorer/sliceControl.css | 64 ++- .../data-explorer/sliceControl.tsx | 31 +- .../dataviewer.functional.test.tsx | 405 +++++++++++++++++- src/test/smoke/datascience.smoke.test.ts | 6 +- 16 files changed, 555 insertions(+), 87 deletions(-) create mode 100644 news/2 Fixes/5200.md create mode 100644 news/3 Code Health/5066.md diff --git a/CHANGELOG.md b/CHANGELOG.md index 0855db9eb33..b52da36c1da 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,17 @@ # Changelog +## 2021.3.1 (24 March 2021) + +### Fixes + +1. Fix duplicate rows being fetched into data viewer for large data. + ([#5200](https://github.com/Microsoft/vscode-jupyter/issues/5200)) + +### Code Health + +1. Add tests for data viewer slice data functionality. + ([#5066](https://github.com/Microsoft/vscode-jupyter/issues/5066)) + ## 2021.3.0 (3 March 2021) ### Enhancements diff --git a/news/2 Fixes/5200.md b/news/2 Fixes/5200.md new file mode 100644 index 00000000000..2b5536d8011 --- /dev/null +++ b/news/2 Fixes/5200.md @@ -0,0 +1 @@ +Fix duplicate rows being fetched into data viewer for large data. \ No newline at end of file diff --git a/news/3 Code Health/5066.md b/news/3 Code Health/5066.md new file mode 100644 index 00000000000..d32abe7541a --- /dev/null +++ b/news/3 Code Health/5066.md @@ -0,0 +1 @@ +Add tests for data viewer slice data functionality. \ No newline at end of file diff --git a/pythonFiles/vscode_datascience_helpers/dataframes/vscodeDataFrame.py b/pythonFiles/vscode_datascience_helpers/dataframes/vscodeDataFrame.py index 3210cc61585..9b6800900b5 100644 --- a/pythonFiles/vscode_datascience_helpers/dataframes/vscodeDataFrame.py +++ b/pythonFiles/vscode_datascience_helpers/dataframes/vscodeDataFrame.py @@ -91,14 +91,14 @@ def _VSCODE_convertTensorToDataFrame(tensor, start=None, end=None): def _VSCODE_convertToDataFrame(df, start=None, end=None): vartype = type(df) if isinstance(df, list): - df = _VSCODE_pd.DataFrame(df) + df = _VSCODE_pd.DataFrame(df).iloc[start:end] elif isinstance(df, _VSCODE_pd.Series): - df = _VSCODE_pd.Series.to_frame(df) + df = _VSCODE_pd.Series.to_frame(df).iloc[start:end] elif isinstance(df, dict): df = _VSCODE_pd.Series(df) - df = _VSCODE_pd.Series.to_frame(df) + df = _VSCODE_pd.Series.to_frame(df).iloc[start:end] elif hasattr(df, "toPandas"): - df = df.toPandas() + df = df.toPandas().iloc[start:end] elif ( hasattr(vartype, "__name__") and vartype.__name__ in _VSCODE_allowedTensorTypes ): @@ -109,7 +109,7 @@ def _VSCODE_convertToDataFrame(df, start=None, end=None): """Disabling bandit warning for try, except, pass. We want to swallow all exceptions here to not crash on variable fetching""" try: - temp = _VSCODE_pd.DataFrame(df) + temp = _VSCODE_pd.DataFrame(df).iloc[start:end] df = temp except: # nosec pass diff --git a/src/client/datascience/jupyter/kernelVariables.ts b/src/client/datascience/jupyter/kernelVariables.ts index d4e02b7adcc..912e193aad2 100644 --- a/src/client/datascience/jupyter/kernelVariables.ts +++ b/src/client/datascience/jupyter/kernelVariables.ts @@ -164,10 +164,6 @@ export class KernelVariables implements IJupyterVariables { // Import the data frame script directory if we haven't already await this.importDataFrameScripts(notebook); - if (targetVariable.rowCount) { - end = Math.min(end, targetVariable.rowCount); - } - let expression = targetVariable.name; if (sliceExpression) { expression = `${targetVariable.name}${sliceExpression}`; diff --git a/src/datascience-ui/data-explorer/cellFormatter.css b/src/datascience-ui/data-explorer/cellFormatter.css index f60b3fb5cde..7f1dd164548 100644 --- a/src/datascience-ui/data-explorer/cellFormatter.css +++ b/src/datascience-ui/data-explorer/cellFormatter.css @@ -11,3 +11,7 @@ text-overflow: ellipsis; overflow: hidden; } + +.index-column-formatter { + text-align: right; +} \ No newline at end of file diff --git a/src/datascience-ui/data-explorer/cellFormatter.tsx b/src/datascience-ui/data-explorer/cellFormatter.tsx index c7cc75a48cf..601ad7bedaa 100644 --- a/src/datascience-ui/data-explorer/cellFormatter.tsx +++ b/src/datascience-ui/data-explorer/cellFormatter.tsx @@ -24,9 +24,6 @@ class CellFormatter extends React.Component { // eslint-disable-next-line @typescript-eslint/no-explicit-any const columnType = (this.props.columnDef as any).type; switch (columnType) { - case ColumnType.Bool: - return this.renderBool(this.props.value as boolean); - case ColumnType.Number: return this.renderNumber(this.props.value as number); @@ -34,7 +31,7 @@ class CellFormatter extends React.Component { break; } } - // Otherwise an unknown type or a string + // Otherwise an unknown type, boolean, or a string const val = this.props.value?.toString() ?? ''; return (
@@ -43,18 +40,16 @@ class CellFormatter extends React.Component { ); } - private renderBool(value: boolean) { - return ( -
- {value.toString()} -
- ); - } - private renderNumber(value: number) { let val = generateDisplayValue(value); + const isIndexColumn = this.props.columnDef.id === '0'; + return ( -
+
{val}
); diff --git a/src/datascience-ui/data-explorer/index.tsx b/src/datascience-ui/data-explorer/index.tsx index fc3b639bd68..7c2a3b53cc5 100644 --- a/src/datascience-ui/data-explorer/index.tsx +++ b/src/datascience-ui/data-explorer/index.tsx @@ -11,7 +11,6 @@ import '../common/index.css'; import * as React from 'react'; import * as ReactDOM from 'react-dom'; -import { initializeIcons } from '@fluentui/react'; import { IVsCodeApi } from '../react-common/postOffice'; import { detectBaseTheme } from '../react-common/themeDetector'; @@ -21,7 +20,6 @@ import { MainPanel } from './mainPanel'; export declare function acquireVsCodeApi(): IVsCodeApi; const baseTheme = detectBaseTheme(); -initializeIcons(); /* eslint-disable */ ReactDOM.render( diff --git a/src/datascience-ui/data-explorer/mainPanel.css b/src/datascience-ui/data-explorer/mainPanel.css index 8c3ea88f219..9f2c0b766cd 100644 --- a/src/datascience-ui/data-explorer/mainPanel.css +++ b/src/datascience-ui/data-explorer/mainPanel.css @@ -16,6 +16,9 @@ padding-left: 16px; overflow: auto; white-space: nowrap; + border-bottom-color: var(--vscode-editor-inactiveSelectionBackground); + border-bottom-style: solid; + border-bottom-width: 1px; } .breadcrumb { @@ -39,3 +42,7 @@ font-size: 18px; padding-right: 2px; } + +.breadcrumb-codicon { + color: var(--vscode-breadcrumb-foreground); +} \ No newline at end of file diff --git a/src/datascience-ui/data-explorer/mainPanel.tsx b/src/datascience-ui/data-explorer/mainPanel.tsx index ef4dd9dc240..dbc4db2bbcc 100644 --- a/src/datascience-ui/data-explorer/mainPanel.tsx +++ b/src/datascience-ui/data-explorer/mainPanel.tsx @@ -27,13 +27,15 @@ import { StyleInjector } from '../react-common/styleInjector'; import { cellFormatterFunc } from './cellFormatter'; import { ISlickGridAdd, ISlickGridSlice, ISlickRow, ReactSlickGrid } from './reactSlickGrid'; import { generateTestData } from './testData'; -import { Image, ImageName } from '../react-common/image'; import '../react-common/codicon/codicon.css'; import '../react-common/seti/seti.less'; import { SliceControl } from './sliceControl'; import { debounce } from 'lodash'; +import { initializeIcons } from '@fluentui/react'; +initializeIcons(); // Register all FluentUI icons being used to prevent developer console errors + const SliceableTypes: Set = new Set(['ndarray', 'Tensor', 'EagerTensor']); // Our css has to come after in order to override body styles @@ -192,12 +194,7 @@ export class MainPanel extends React.Component
{this.state.fileName} {this.state.fileName ? ( - +
) : undefined} {breadcrumbText}
diff --git a/src/datascience-ui/data-explorer/reactSlickGrid.css b/src/datascience-ui/data-explorer/reactSlickGrid.css index 421fc7deca7..7478bd4a0e8 100644 --- a/src/datascience-ui/data-explorer/reactSlickGrid.css +++ b/src/datascience-ui/data-explorer/reactSlickGrid.css @@ -4,9 +4,7 @@ } .react-grid-container { - border-color: var(--vscode-editor-inactiveSelectionBackground); - border-style: solid; - border-width: 1px; + border: none; } .react-grid-measure { @@ -17,19 +15,17 @@ .react-grid-header-cell { padding: 0px 4px; background-color: var(--vscode-menu-background); - color: var(--vscode-editor-foreground); + color: var(--vscode-menu-foreground); text-align: left; font-weight: bold; - border-right-color: var(--vscode-editor-inactiveSelectionBackground); + border-right: 1px solid transparent; } .react-grid-cell { padding: 4px; background-color: var(--vscode-editor-background); color: var(--vscode-editor-foreground); - border-bottom-color: var(--vscode-editor-inactiveSelectionBackground); - border-right-color: var(--vscode-editor-inactiveSelectionBackground); - border-right-style: solid; + border-right: 1px solid transparent; box-sizing: border-box; } @@ -41,9 +37,8 @@ /* Some overrides necessary to get the colors we want */ .slick-headerrow-column { background-color: var(--vscode-menu-background); - border-right-color: var(--vscode-editor-inactiveSelectionBackground); - border-right-style: solid; - border-bottom-color: var(--vscode-menu-background); + color: var(--vscode-menu-foreground); + border-right: 1px solid transparent; } .slick-headerrow-column.ui-state-default { @@ -55,7 +50,7 @@ .slick-header-column.ui-state-default, .slick-group-header-column.ui-state-default { - border-right-color: var(--vscode-editor-inactiveSelectionBackground); + border-right: 1px solid transparent; display: flex; } @@ -76,21 +71,31 @@ } .react-grid-header-cell:hover { - background-color: var(--vscode-editor-inactiveSelectionBackground); + background-color: var(--override-selection-background, var(--vscode-list-hoverBackground)); } .react-grid-header-cell > .slick-sort-indicator-asc::before { background: none; - font: normal normal normal 16px/1 codicon; + font: normal normal normal 10px/1 codicon; content: '\eaa1'; /* VS Code arrow-up codicon */ align-items: center; + text-align: right; + display: flex; + padding: 3px; + color: var(--vscode-menu-foreground); + opacity: 0.4; } .react-grid-header-cell > .slick-sort-indicator-desc::before { background: none; - font: normal normal normal 16px/1 codicon; + font: normal normal normal 10px/1 codicon; content: '\ea9a'; /* VS Code arrow-down codicon */ align-items: center; + text-align: right; + display: flex; + padding: 3px; + color: var(--vscode-menu-foreground); + opacity: 0.4; } .slick-row:hover > .react-grid-cell { @@ -108,7 +113,7 @@ input.editor-text { outline: 0 none; border-style: solid; background-color: var(--vscode-list-activeSelectionBackground); - color: var(--vscode-button-foreground); + color: var(--vscode-list-activeSelectionForeground); text-align: left; /* input does not inherit font from body */ font-family: var(--vscode-editor-font-family); @@ -116,6 +121,10 @@ input.editor-text { font-size: var(--vscode-editor-font-size); } +.slick-cell { + border: 1px solid var(--vscode-editor-lineHighlightBorder); +} + .slick-cell.editable { border-color: none; border-style: none; @@ -125,9 +134,6 @@ input.editor-text { } .control-container { - border-bottom-color: var(--vscode-editor-inactiveSelectionBackground); - border-bottom-style: solid; - border-bottom-width: 1px; padding: 6px; display: flex; justify-content: start; @@ -136,6 +142,9 @@ input.editor-text { .codicon-button { cursor: pointer; + padding-left: 3px; + padding-top: 3px; + color: var(--vscode-menu-foreground); } .header-cell-button { diff --git a/src/datascience-ui/data-explorer/reactSlickGridFilterBox.tsx b/src/datascience-ui/data-explorer/reactSlickGridFilterBox.tsx index d441da213a9..a992e14470c 100644 --- a/src/datascience-ui/data-explorer/reactSlickGridFilterBox.tsx +++ b/src/datascience-ui/data-explorer/reactSlickGridFilterBox.tsx @@ -18,6 +18,21 @@ const filterIcon: IIconProps = { } }; +const styles = { + iconContainer: { + opacity: 0.4, + ':active': { + opacity: 0 + } + }, + root: { + '::after': { + borderRadius: '0px', + border: '1px solid var(--vscode-focusBorder)' + } + } +}; + interface IFilterProps { column: Slick.Column; fontSize: number; @@ -39,6 +54,7 @@ export class ReactSlickGridFilterBox extends React.Component { tabIndex={0} ariaLabel={this.props.column.name} className="filter-box" + styles={styles} value={this.props.filter} /> ); diff --git a/src/datascience-ui/data-explorer/sliceControl.css b/src/datascience-ui/data-explorer/sliceControl.css index 2eb378ab0e2..ee964183a3a 100644 --- a/src/datascience-ui/data-explorer/sliceControl.css +++ b/src/datascience-ui/data-explorer/sliceControl.css @@ -1,12 +1,12 @@ .slice-data { - background-color: var(--vscode-input-background); - color: var(--vscode-input-foreground); - padding: 0px 4px; + background-color: var(--vscode-settings-textInputBackground); + color: var(--vscode-settings-textInputForeground) !important; + padding: 0px 4px 0px 6px; /* input does not inherit font from body */ font-family: var(--vscode-font-family); font-weight: var(--vscode-font-weight); font-size: var(--vscode-font-size); - height: 30px; + height: 26px; margin: 0px; border: none; } @@ -27,10 +27,12 @@ .slice-summary { display: flex; flex-direction: row; + cursor: pointer; } .slice-summary-detail { padding-bottom: 2px; + padding-left: 2px; } .current-slice { @@ -39,13 +41,14 @@ padding-right: 5px; padding-top: 0px; background-color: var(--vscode-input-background); + border-radius: 2px; } .slice-form { align-self: center; flex-direction: column; justify-content: space-between; - padding: 4px; + padding: 4px 14px; } .slice-control-row { @@ -78,26 +81,21 @@ details[open] > summary::before { } .submit-slice-button { - color: var(--vscode-button-foreground); + color: var(--vscode-button-foreground) !important; background-color: var(--vscode-button-background); border: none; margin-left: 10px; padding: 4px 8px; - height: 30px; + height: 26px; font-family: var(--vscode-font-family); font-weight: var(--vscode-font-weight); font-size: var(--vscode-font-size); max-width: fit-content; -} - -.slice-enablement-checkbox { - margin-right: 6px; - width: 20px; + cursor: pointer; } .slice-enablement-checkbox-container { padding-top: 4px; - padding-bottom: 4px; } :focus { @@ -105,13 +103,6 @@ details[open] > summary::before { } /* Overrides for Fluent UI controls */ -[class*='text'] { - color: var(--vscode-editor-foreground); - font-family: var(--vscode-font-family); - font-weight: var(--vscode-font-weight); - font-size: var(--vscode-font-size); -} - [class*='ms-Dropdown-label'] { color: var(--vscode-editor-foreground); font-family: var(--vscode-font-family); @@ -119,6 +110,7 @@ details[open] > summary::before { font-size: var(--vscode-font-size); } +/* This needs to match VS Code settings dropdowns */ .dropdownTitleOverrides .ms-Dropdown-title { /* Usually want to avoid !important but in this case it's necessary to ensure that when the dropdown @@ -130,6 +122,34 @@ details[open] > summary::before { font-size: var(--vscode-font-size); background-color: var(--vscode-dropdown-background); border: var(--vscode-dropdown-border); + border-radius: 0px; + height: 26px; + line-height: 26px; +} + +[class*='dropdown'] { + border: 1px solid var(--vscode-dropdown-border), +} + +[class*='dropdown']:active { + border: 1px solid var(--vscode-dropdown-border), +} + +[class*='dropdown']:focus::after { + border-radius: 0px; + border: 1px solid var(--vscode-focusBorder) !important; +} + +[class*='ms-Dropdown-caretDownWrapper']::after { + color: var(--vscode-dropdown-foreground); + content: "\eab4"; + pointer-events: none; + font-family: codicon; + font-size: 14px; + width: 14px; + height: 14px; + line-height: 26px; + opacity: 0.8; } .slice-data.input-invalid { @@ -141,13 +161,13 @@ details[open] > summary::before { } .slice-data:focus { - outline: 1px solid var(--vscode-button-background) + outline: 1px solid var(--vscode-button-background); } .error-message { outline: 1px solid var(--vscode-inputValidation-errorBorder); background-color: var(--vscode-inputValidation-errorBackground); - color: var(--vscode-settings-textInputForeground); + color: var(--vscode-inputValidation-errorForeground); font-family: var(--vscode-font-family); font-weight: var(--vscode-font-weight); font-size: var(--vscode-font-size); diff --git a/src/datascience-ui/data-explorer/sliceControl.tsx b/src/datascience-ui/data-explorer/sliceControl.tsx index 0f57e1fbbf6..777819bab47 100644 --- a/src/datascience-ui/data-explorer/sliceControl.tsx +++ b/src/datascience-ui/data-explorer/sliceControl.tsx @@ -1,4 +1,4 @@ -import { Dropdown, IDropdownOption, ResponsiveMode } from '@fluentui/react'; +import { Checkbox, Dropdown, IDropdownOption, ResponsiveMode } from '@fluentui/react'; import * as React from 'react'; import { IGetSliceRequest } from '../../client/datascience/data-viewing/types'; import { getLocString } from '../react-common/locReactSide'; @@ -13,6 +13,21 @@ import { import './sliceControl.css'; +const checkboxStyles = { + checkbox: { + color: 'var(--vscode-checkbox-foreground)', + backgroundColor: 'var(--vscode-checkbox-background) !important', + border: 'var(--vscode-checkbox-border)' + }, + text: { + fontFamily: 'var(--vscode-font-family)', + fontWeight: 'var(--vscode-font-weight)', + fontSize: 'var(--vscode-font-size)', + color: 'var(--vscode-editor-foreground) !important', + paddingLeft: 2 + } +}; + // These styles are passed to the FluentUI dropdown controls const styleOverrides = { color: 'var(--vscode-dropdown-foreground)', @@ -20,6 +35,7 @@ const styleOverrides = { fontFamily: 'var(--vscode-font-family)', fontWeight: 'var(--vscode-font-weight)', fontSize: 'var(--vscode-font-size)', + border: 'var(--vscode-dropdown-border)', ':focus': { color: 'var(--vscode-dropdown-foreground)' }, @@ -43,6 +59,9 @@ const dropdownStyles = { } } }, + caretDown: { + visibility: 'hidden' // Override the FluentUI caret and use ::after selector on the caretDownWrapper in order to match VS Code. See sliceContro.css + }, callout: styleOverrides, dropdownItem: styleOverrides, dropdownItemSelected: { @@ -108,16 +127,12 @@ export class SliceControl extends React.Component
- -
diff --git a/src/test/datascience/dataviewer.functional.test.tsx b/src/test/datascience/dataviewer.functional.test.tsx index b60fc1ca78f..fbe181ad3fe 100644 --- a/src/test/datascience/dataviewer.functional.test.tsx +++ b/src/test/datascience/dataviewer.functional.test.tsx @@ -32,6 +32,13 @@ import { noop, sleep } from '../core'; import { DataScienceIocContainer } from './dataScienceIocContainer'; import { takeSnapshot, writeDiffSnapshot } from './helpers'; import { IMountedWebView } from './mountedWebView'; +import { SliceControl } from '../../datascience-ui/data-explorer/sliceControl'; +import { Dropdown } from '@fluentui/react'; + +interface ISliceControlTestInterface { + toggleEnablement: () => void; + applyDropdownsToInputBox: () => void; +} // import { asyncDump } from '../common/asyncDump'; suite('DataScience DataViewer tests', () => { @@ -95,7 +102,7 @@ suite('DataScience DataViewer tests', () => { delete (global as any).ascquireVsCodeApi; }); - function createJupyterVariable(variable: string, type: string): IJupyterVariable { + function createJupyterVariable(variable: string, type: string, shape: string): IJupyterVariable { return { name: variable, value: '', @@ -103,7 +110,7 @@ suite('DataScience DataViewer tests', () => { type, size: 0, truncated: true, - shape: '', + shape, count: 0 }; } @@ -118,8 +125,12 @@ suite('DataScience DataViewer tests', () => { return dataViewerFactory.create(dataProvider, title); } - async function createJupyterVariableDataViewer(variable: string, type: string): Promise { - const jupyterVariable: IJupyterVariable = createJupyterVariable(variable, type); + async function createJupyterVariableDataViewer( + variable: string, + type: string, + shape: string = '' + ): Promise { + const jupyterVariable: IJupyterVariable = createJupyterVariable(variable, type, shape); const jupyterVariableDataProvider: IDataViewerDataProvider = await createJupyterVariableDataProvider( jupyterVariable ); @@ -515,6 +526,392 @@ suite('DataScience DataViewer tests', () => { verifyRows(wrapper.wrapper, [0, 4, 5, 6]); }); + suite('Data viewer slice data', async () => { + function findSliceControlPanel(wrapper: ReactWrapper, React.Component>) { + const sliceControlWrapper = wrapper.find(SliceControl); + sliceControlWrapper.update(); + assert.ok(sliceControlWrapper && sliceControlWrapper.length > 0, 'Slice control not found'); + return sliceControlWrapper; + } + + function verifyReadonlyIndicator( + wrapper: ReactWrapper, React.Component>, + currentSlice: string + ) { + const sliceControl = wrapper.find(SliceControl); + const html = sliceControl.html(); + const root = parse(html) as any; + wrapper.render(); + const cells = root.querySelectorAll('.current-slice') as HTMLSpanElement[]; + assert.ok(cells.length === 1, 'No readonly indicator found'); + assert.ok(cells[0].innerHTML === currentSlice, 'Readonly indicator contents did not match'); + } + + function verifyDropdowns(wrapper: ReactWrapper, React.Component>, rows: (string | number)[]) { + const sliceControl = wrapper.find(SliceControl); + const html = sliceControl.html(); + const root = parse(html) as any; + const cells = root.querySelectorAll('.ms-Dropdown-title'); + assert.ok(cells.length >= rows.length, 'Not enough dropdowns found'); + // Now verify the list of dropdowns have the expected values + for (let i = 0; i < rows.length; i += 1) { + // Span reflects the dropdown's current selection + const span = cells[i] as HTMLSpanElement; + assert.ok(span, `Span ${i} not found`); + const val = rows[i].toString(); + assert.equal(span.innerHTML, val, `Dropdown ${i} selection not matching. ${span.innerHTML} !== ${val}`); + } + } + + function toggleCheckbox(wrapper: ReactWrapper, React.Component>) { + const sliceControl = findSliceControlPanel(wrapper); + // Enable slicing by toggling checkbox + const instance = (sliceControl.instance() as any) as ISliceControlTestInterface; + instance.toggleEnablement(); // simulate('click') doesn't suffice: https://github.com/facebook/react/issues/4950#issuecomment-255408709 + wrapper.render(); + } + + function verifyControlsDisabled( + wrapper: ReactWrapper, React.Component>, + expectedNumberOfDropdowns: number, + initialReadonlyIndicator: string + ) { + // Open the slice panel + findSliceControlPanel(wrapper); + // Verify that all controls are initially disabled + let input = wrapper.find('.slice-data'); + const html = input.html(); + assert.ok(html.includes('disabled'), 'Input field was not initially disabled'); + const dropdowns = wrapper.find(Dropdown); + assert.ok(dropdowns.length === expectedNumberOfDropdowns, 'Unexpected number of dropdowns found'); + // Verify no readonly indicator as we're not slicing yet + assert.throws( + () => verifyReadonlyIndicator(wrapper, initialReadonlyIndicator), + 'Readonly indicator rendered when not slicing' + ); + } + + function editInputValue(wrapper: IMountedWebView, slice: string) { + const inputElement = wrapper.wrapper.find('.slice-data').getDOMNode() as HTMLInputElement; + inputElement.value = slice; + wrapper.wrapper.find('.slice-data').simulate('change'); + } + + async function applySliceAndVerifyReadonlyIndicator(wrapper: IMountedWebView, slice: string) { + // Apply a slice to input box + const gotSlice = getCompletedPromise(wrapper); + editInputValue(wrapper, slice); + wrapper.wrapper.find('form').first().simulate('submit'); + await gotSlice; + // Ensure readonly indicator updates after slicing + verifyReadonlyIndicator(wrapper.wrapper, slice); + } + + async function changeDropdown( + wrapper: IMountedWebView, + dropdownType: 'Axis' | 'Index', + dropdownRow: number, + newValue: number | string + ) { + const gotSlice = getCompletedPromise(wrapper); + const sliceControl = findSliceControlPanel(wrapper.wrapper); + // Do a setstate because we don't have direct access to the dropdown selection change handler + const newState = { [`selected${dropdownType}${dropdownRow}`]: newValue }; + sliceControl.setState(newState); + const instance = (sliceControl.instance() as any) as ISliceControlTestInterface; + // This is what gets called in the dropdown change handler. Manually call it because + // simulating a change event on the Dropdown node doesn't seem to do anything + instance.applyDropdownsToInputBox(); + wrapper.wrapper.render(); + await gotSlice; + } + + runMountedTest('Slice 2D', async (wrapper) => { + const code = `import torch +import numpy as np +arr = np.arange(6).reshape(2, 3) +foo = torch.tensor(arr)`; + + // Create data viewer + await injectCode(code); + const gotAllRows = getCompletedPromise(wrapper); + const dv = await createJupyterVariableDataViewer('foo', 'Tensor', '(2, 3)'); + assert.ok(dv, 'DataViewer not created'); + await gotAllRows; + verifyRows(wrapper.wrapper, [0, 0, 1, 2, 1, 3, 4, 5]); + verifyControlsDisabled(wrapper.wrapper, 2, ''); + + // Apply a slice via input box and verify that dropdowns update + toggleCheckbox(wrapper.wrapper); + await applySliceAndVerifyReadonlyIndicator(wrapper, '[1, :]'); + verifyRows(wrapper.wrapper, [0, 3, 1, 4, 2, 5]); + verifyDropdowns(wrapper.wrapper, [0, 1]); // Axis 0, index 1 + + // Change the dropdowns and verify that the slice expression updates + await changeDropdown(wrapper, 'Axis', 0, 1); + verifyReadonlyIndicator(wrapper.wrapper, '[:, 1]'); + verifyDropdowns(wrapper.wrapper, [1, 1]); + verifyRows(wrapper.wrapper, [0, 1, 1, 4]); + assert.ok( + (wrapper.wrapper.find('.slice-data').getDOMNode() as HTMLInputElement).value === '[:, 1]', + 'Input box did not update to match slice' + ); + + // Apply a slice with no corresponding dropdown + await applySliceAndVerifyReadonlyIndicator(wrapper, '[:, :2]'); + verifyRows(wrapper.wrapper, [0, 0, 1, 1, 3, 4]); + verifyDropdowns(wrapper.wrapper, ['', '']); // Dropdowns should be unset + + // Uncheck slice checkbox and verify original contents are restored + const disableSlicing = getCompletedPromise(wrapper); + toggleCheckbox(wrapper.wrapper); + await disableSlicing; + verifyRows(wrapper.wrapper, [0, 0, 1, 2, 1, 3, 4, 5]); + verifyControlsDisabled(wrapper.wrapper, 2, ''); + + // Recheck slice checkbox and verify slice expression is restored + const reenableSlicing = getCompletedPromise(wrapper); + toggleCheckbox(wrapper.wrapper); + await reenableSlicing; + verifyRows(wrapper.wrapper, [0, 0, 1, 1, 3, 4]); + + // Enter an invalid slice expression and verify error message is rendered + editInputValue(wrapper, '[:]'); + assert.ok( + wrapper.wrapper.find('.error-message').length === 1, + 'No error message rendered for invalid slice' + ); + }); + + runMountedTest('Slice 3D', async (wrapper) => { + const code = `import torch +import numpy as np +arr = np.arange(24).reshape(2,4,3) +foo = torch.tensor(arr)`; + // Create data viewer + await injectCode(code); + const gotAllRows = getCompletedPromise(wrapper); + const dv = await createJupyterVariableDataViewer('foo', 'Tensor', '(2, 4, 3)'); + assert.ok(dv, 'DataViewer not created'); + await gotAllRows; + verifyRows(wrapper.wrapper, [ + 0, + '[0, 1, 2]', + '[3, 4, 5]', + '[6, 7, 8]', + '[9, 10, 11]', + 1, + '[12, 13, 14]', + '[15, 16, 17]', + '[18, 19, 20]', + '[21, 22, 23]' + ]); + verifyControlsDisabled(wrapper.wrapper, 2, ''); + + // Toggle on slicing. Slice should immediately be applied + const enableSlicing = getCompletedPromise(wrapper); + toggleCheckbox(wrapper.wrapper); + await enableSlicing; + verifyReadonlyIndicator(wrapper.wrapper, '[0, :, :]'); + verifyRows(wrapper.wrapper, [0, 0, 1, 2, 1, 3, 4, 5, 2, 6, 7, 8, 3, 9, 10, 11]); + + // Change the dropdowns and verify that the slice expression updates + await changeDropdown(wrapper, 'Axis', 0, 1); + verifyReadonlyIndicator(wrapper.wrapper, '[:, 0, :]'); + verifyDropdowns(wrapper.wrapper, [1, 0]); + verifyRows(wrapper.wrapper, [0, 0, 1, 2, 1, 12, 13, 14]); + assert.ok( + (wrapper.wrapper.find('.slice-data').getDOMNode() as HTMLInputElement).value === '[:, 0, :]', + 'Input box did not update to match slice' + ); + + // Apply a slice via input box and verify that dropdowns update + await applySliceAndVerifyReadonlyIndicator(wrapper, '[:, :, 2]'); + verifyRows(wrapper.wrapper, [0, 2, 5, 8, 11, 1, 14, 17, 20, 23]); + verifyDropdowns(wrapper.wrapper, [2, 2]); // Axis 2, index 2 + + // Apply a slice with no corresponding dropdown + await applySliceAndVerifyReadonlyIndicator(wrapper, '[:, :1, :]'); + verifyRows(wrapper.wrapper, [0, '[0, 1, 2]', 1, '[12, 13, 14]']); + verifyDropdowns(wrapper.wrapper, ['', '']); // Dropdowns should be unset + + // Uncheck slice checkbox and verify original contents are restored + const disableSlicing = getCompletedPromise(wrapper); + toggleCheckbox(wrapper.wrapper); + await disableSlicing; + verifyRows(wrapper.wrapper, [ + 0, + '[0, 1, 2]', + '[3, 4, 5]', + '[6, 7, 8]', + '[9, 10, 11]', + 1, + '[12, 13, 14]', + '[15, 16, 17]', + '[18, 19, 20]', + '[21, 22, 23]' + ]); + verifyControlsDisabled(wrapper.wrapper, 2, ''); + + // Recheck slice checkbox and verify slice expression is restored + const reenableSlicing = getCompletedPromise(wrapper); + toggleCheckbox(wrapper.wrapper); + await reenableSlicing; + verifyRows(wrapper.wrapper, [0, '[0, 1, 2]', 1, '[12, 13, 14]']); + verifyReadonlyIndicator(wrapper.wrapper, '[:, :1, :]'); + + // Enter an invalid slice expression and verify error message is rendered + editInputValue(wrapper, '[:]'); + assert.ok( + wrapper.wrapper.find('.error-message').length === 1, + 'No error message rendered for invalid slice' + ); + }); + + runMountedTest('Slice 4D', async (wrapper) => { + const code = `import torch +import numpy as np +arr = np.arange(30).reshape(3, 5, 1, 2) +foo = torch.tensor(arr)`; + // Create data viewer + await injectCode(code); + const gotAllRows = getCompletedPromise(wrapper); + const dv = await createJupyterVariableDataViewer('foo', 'Tensor', '(3, 5, 1, 2)'); + assert.ok(dv, 'DataViewer not created'); + await gotAllRows; + verifyRows(wrapper.wrapper, [ + 0, + '[[0, 1]]', + `[[2, 3]]`, + '[[4, 5]]', + '[[6, 7]]', + '[[8, 9]]', + 1, + '[[10, 11]]', + '[[12, 13]]', + '[[14, 15]]', + '[[16, 17]]', + '[[18, 19]]', + 2, + '[[20, 21]]', + '[[22, 23]]', + '[[24, 25]]', + '[[26, 27]]', + '[[28, 29]]' + ]); + verifyControlsDisabled(wrapper.wrapper, 4, ''); + + // Toggle on slicing. Slice should immediately be applied + const enableSlicing = getCompletedPromise(wrapper); + toggleCheckbox(wrapper.wrapper); + await enableSlicing; + verifyReadonlyIndicator(wrapper.wrapper, '[0, 0, :, :]'); + verifyRows(wrapper.wrapper, [0, 0, 1]); + + // Change the dropdowns and verify that the slice expression updates + await changeDropdown(wrapper, 'Index', 1, 2); + verifyReadonlyIndicator(wrapper.wrapper, '[0, 2, :, :]'); + verifyDropdowns(wrapper.wrapper, [0, 0, 1, 2]); + verifyRows(wrapper.wrapper, [0, 4, 5]); + assert.ok( + (wrapper.wrapper.find('.slice-data').getDOMNode() as HTMLInputElement).value === '[0, 2, :, :]', + 'Input box did not update to match slice' + ); + + // Apply a slice via input box and verify that dropdowns update + await applySliceAndVerifyReadonlyIndicator(wrapper, '[:, 4, :, 1]'); + verifyRows(wrapper.wrapper, [0, 9, 1, 19, 2, 29]); + verifyDropdowns(wrapper.wrapper, [1, 4, 3, 1]); // Axis 1 index 4, axis 3 index 1 + + // Apply a slice with no corresponding dropdown + await applySliceAndVerifyReadonlyIndicator(wrapper, '[1, 2, 0, :]'); + verifyRows(wrapper.wrapper, [0, 14, 1, 15]); + verifyDropdowns(wrapper.wrapper, ['', '', '', '']); // Dropdowns should be unset + + // Uncheck slice checkbox and verify original contents are restored + const disableSlicing = getCompletedPromise(wrapper); + toggleCheckbox(wrapper.wrapper); + await disableSlicing; + verifyRows(wrapper.wrapper, [ + 0, + '[[0, 1]]', + `[[2, 3]]`, + '[[4, 5]]', + '[[6, 7]]', + '[[8, 9]]', + 1, + '[[10, 11]]', + '[[12, 13]]', + '[[14, 15]]', + '[[16, 17]]', + '[[18, 19]]', + 2, + '[[20, 21]]', + '[[22, 23]]', + '[[24, 25]]', + '[[26, 27]]', + '[[28, 29]]' + ]); + verifyControlsDisabled(wrapper.wrapper, 4, ''); + + // Recheck slice checkbox and verify slice expression is restored + const reenableSlicing = getCompletedPromise(wrapper); + toggleCheckbox(wrapper.wrapper); + await reenableSlicing; + verifyRows(wrapper.wrapper, [0, 14, 1, 15]); + verifyDropdowns(wrapper.wrapper, ['', '', '', '']); // Dropdowns should be unset + + // Enter an invalid slice expression and verify error message is rendered + editInputValue(wrapper, '[:]'); + assert.ok( + wrapper.wrapper.find('.error-message').length === 1, + 'No error message rendered for invalid slice' + ); + }); + + runMountedTest('Refresh with slice applied', async (wrapper) => { + // Same shape, old slice is still valid, ensure update in place + const code = `import torch +foo = torch.tensor([[[0, 1, 2], [3, 4, 5]]])`; + // Create data viewer + const notebook = await injectCode(code); + const gotAllRows = getCompletedPromise(wrapper); + const dv = await createJupyterVariableDataViewer('foo', 'Tensor', '(1, 2, 3)'); + assert.ok(dv, 'DataViewer not created'); + await gotAllRows; + + // Toggle on slicing. Slice should immediately be applied + const enableSlicing = getCompletedPromise(wrapper); + toggleCheckbox(wrapper.wrapper); + await enableSlicing; + verifyReadonlyIndicator(wrapper.wrapper, '[0, :, :]'); + verifyRows(wrapper.wrapper, [0, 0, 1, 2, 1, 3, 4, 5]); + + // Apply a slice via input box and verify that dropdowns update + await applySliceAndVerifyReadonlyIndicator(wrapper, '[:, 1, :]'); + verifyRows(wrapper.wrapper, [0, 3, 4, 5]); + verifyDropdowns(wrapper.wrapper, [1, 1]); // Axis 1 index 1 + + // New variable value but same shape. Ensure slice updates in-place + await executeCode('foo = torch.tensor([[[6, 7, 8], [9, 10, 11]]])', notebook!); + const refreshPromise = getCompletedPromise(wrapper); + await dv.refreshData(); + await refreshPromise; + verifyReadonlyIndicator(wrapper.wrapper, '[:, 1, :]'); + verifyRows(wrapper.wrapper, [0, 9, 10, 11]); + + // New variable shape invalidates old slice + await executeCode('foo = torch.tensor([[[0, 1]], [[2, 3]]])', notebook!); + // Ensure data updates + const invalidateSlicePromise = getCompletedPromise(wrapper); + await dv.refreshData(); + await invalidateSlicePromise; + // Preselected slice is applied + verifyReadonlyIndicator(wrapper.wrapper, '[0, :, :]'); + verifyRows(wrapper.wrapper, [0, 0, 1]); + }); + }); + // https://github.com/microsoft/vscode-jupyter/issues/4706 // Disabled for now. Root cause is that pd.replace isn't recursive over objects in DataFrames, // so our current inf/nan handling does not work for DataFrames whose cells are Series, ndarray, or list diff --git a/src/test/smoke/datascience.smoke.test.ts b/src/test/smoke/datascience.smoke.test.ts index ac5d5a3a61a..e75a175601c 100644 --- a/src/test/smoke/datascience.smoke.test.ts +++ b/src/test/smoke/datascience.smoke.test.ts @@ -12,7 +12,7 @@ import * as vscode from 'vscode'; import { ISystemPseudoRandomNumberGenerator } from '../../client/datascience/types'; import { IExtensionTestApi, openFile, setAutoSaveDelayInWorkspaceRoot, waitForCondition } from '../common'; import { EXTENSION_ROOT_DIR_FOR_TESTS, IS_SMOKE_TEST } from '../constants'; -import { noop, sleep } from '../core'; +import { sleep } from '../core'; import { closeActiveWindows, initialize, initializeTest } from '../initialize'; const timeoutForCellToRun = 3 * 60 * 1_000; @@ -87,8 +87,8 @@ suite('Smoke Tests', () => { if (await fs.pathExists(outputFile)) { await fs.unlink(outputFile); } - // Ignore exceptions (as native editor closes the document as soon as its opened); - await openFile(file).catch(noop); + + await vscode.commands.executeCommand('jupyter.opennotebook', vscode.Uri.file(file)); // Wait for 15 seconds for notebook to launch. // Unfortunately there's no way to know for sure it has completely loaded.