RESOLVED FIXED 192388
Web Inspector: Table selection becomes corrupted when deleting selected cookies
https://bugs.webkit.org/show_bug.cgi?id=192388
Summary Web Inspector: Table selection becomes corrupted when deleting selected cookies
Matt Baker
Reported 2018-12-04 17:25:48 PST
Summary: Table selection becomes corrupted when deleting selected cookies. Steps to Reproduce: 1. Inspector https://daringfireball.net 2. Storage tab > Cookies 3. Select first row, hit delete => Cookie removed, next row selected 4. Hit delete Expected: => Cookie removed, next row selected Actual: => The cookie following the selected cookie is removed, and two rows are now selected Note: Table adjusts cached row indexes that move as a result of removing rows, but its SelectionController needs to be informed of the change so it can do the same. Instead of calling `didRemoveItem` for each removed index, taking care to enumerate indexes in reverse order, SelectionController should have a `didRemoveItems` method taking an IndexSet. This shields users of SelectionController from having to do the right thing.
Attachments
Patch (6.62 KB, patch)
2018-12-05 10:56 PST, Matt Baker
no flags
Archive of layout-test-results from ews101 for mac-sierra (2.43 MB, application/zip)
2018-12-05 12:09 PST, EWS Watchlist
no flags
Archive of layout-test-results from ews106 for mac-sierra-wk2 (3.03 MB, application/zip)
2018-12-05 12:09 PST, EWS Watchlist
no flags
Patch (7.80 KB, patch)
2018-12-05 12:43 PST, Matt Baker
no flags
Patch (7.87 KB, patch)
2018-12-05 17:53 PST, Matt Baker
no flags
Archive of layout-test-results from ews103 for mac-sierra (2.76 MB, application/zip)
2018-12-05 18:47 PST, EWS Watchlist
no flags
Patch (12.33 KB, patch)
2018-12-06 17:41 PST, Matt Baker
no flags
Patch (11.46 KB, patch)
2018-12-12 13:41 PST, Matt Baker
no flags
Patch (11.45 KB, patch)
2018-12-12 13:42 PST, Matt Baker
no flags
Archive of layout-test-results from ews100 for mac-sierra (2.42 MB, application/zip)
2018-12-12 14:44 PST, EWS Watchlist
no flags
Archive of layout-test-results from ews105 for mac-sierra-wk2 (3.15 MB, application/zip)
2018-12-12 14:53 PST, EWS Watchlist
no flags
Archive of layout-test-results from ews114 for mac-sierra (2.10 MB, application/zip)
2018-12-12 15:16 PST, EWS Watchlist
no flags
Patch (11.46 KB, patch)
2018-12-12 15:46 PST, Matt Baker
no flags
Patch (11.42 KB, patch)
2018-12-12 16:15 PST, Matt Baker
no flags
Patch (14.11 KB, patch)
2018-12-12 19:23 PST, Matt Baker
no flags
Radar WebKit Bug Importer
Comment 1 2018-12-04 17:26:21 PST
Matt Baker
Comment 2 2018-12-05 10:56:00 PST
EWS Watchlist
Comment 3 2018-12-05 12:09:30 PST
Comment on attachment 356619 [details] Patch Attachment 356619 [details] did not pass mac-ews (mac): Output: https://webkit-queues.webkit.org/results/10280853 New failing tests: inspector/table/table-remove-rows.html
EWS Watchlist
Comment 4 2018-12-05 12:09:32 PST
Created attachment 356636 [details] Archive of layout-test-results from ews101 for mac-sierra The attached test failures were seen while running run-webkit-tests on the mac-ews. Bot: ews101 Port: mac-sierra Platform: Mac OS X 10.12.6
EWS Watchlist
Comment 5 2018-12-05 12:09:55 PST
Comment on attachment 356619 [details] Patch Attachment 356619 [details] did not pass mac-wk2-ews (mac-wk2): Output: https://webkit-queues.webkit.org/results/10280784 New failing tests: inspector/table/table-remove-rows.html
EWS Watchlist
Comment 6 2018-12-05 12:09:56 PST
Created attachment 356637 [details] Archive of layout-test-results from ews106 for mac-sierra-wk2 The attached test failures were seen while running run-webkit-tests on the mac-wk2-ews. Bot: ews106 Port: mac-sierra-wk2 Platform: Mac OS X 10.12.6
Matt Baker
Comment 7 2018-12-05 12:43:32 PST
Matt Baker
Comment 8 2018-12-05 17:53:31 PST
Nikita Vasilyev
Comment 9 2018-12-05 18:46:22 PST
Comment on attachment 356687 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=356687&action=review > Source/WebInspectorUI/ChangeLog:14 > + (WI.SelectionController.prototype.didRemoveItems): > + (WI.SelectionController.prototype.didRemoveItem): Deleted. > + Replace `didRemoveItem` with a method taking an IndexSet. Calling the > + single-index version while iterating over multiple rows in ascending > + order is unsafe, a detail best left to the SelectionController. The bug description was about removing a single item. Is this a drive-by change?
EWS Watchlist
Comment 10 2018-12-05 18:47:49 PST
Comment on attachment 356687 [details] Patch Attachment 356687 [details] did not pass mac-ews (mac): Output: https://webkit-queues.webkit.org/results/10286551 New failing tests: webgl/1.0.2/conformance/textures/gl-teximage.html
EWS Watchlist
Comment 11 2018-12-05 18:47:50 PST
Created attachment 356692 [details] Archive of layout-test-results from ews103 for mac-sierra The attached test failures were seen while running run-webkit-tests on the mac-ews. Bot: ews103 Port: mac-sierra Platform: Mac OS X 10.12.6
Nikita Vasilyev
Comment 12 2018-12-05 19:07:14 PST
Comment on attachment 356687 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=356687&action=review > Source/WebInspectorUI/UserInterface/Controllers/SelectionController.js:228 > + if (this.hasSelectedItem(index)) > + this.deselectItem(index); Could it get called more than once in this loop? If so, can one didRemoveItems call cause more than one selectionControllerSelectionDidChange?
Devin Rousso
Comment 13 2018-12-05 21:38:48 PST
Comment on attachment 356687 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=356687&action=review >> Source/WebInspectorUI/UserInterface/Controllers/SelectionController.js:228 >> + this.deselectItem(index); > > Could it get called more than once in this loop? > If so, can one didRemoveItems call cause more than one selectionControllerSelectionDidChange? Along the lines of what Nikita said: This worries me, because I feel like there's a way for a caller (e.g. `WI.Table`) to remove an item, notify `WI.SelectionController`, who will then tell the caller that an item was removed, only to have the caller then affect something after the removed item. Table has rows 2 (A) and 3 (B) selected > row 2 (A) deleted > row 3 (B) becomes row 2 (B) > didRemoveItems([2 (A)]) > Table is told to deselect row 2 (A) > Table deselects row 2 (B) Tests for cases such as these couldn't hurt :) I don't want to beat a dead horse, but maybe this would be a good time to revisit the idea of using objects instead of indexes. I don't think this would be an issue there, as the `WI.SelectionController` would just say to the `WI.Table` "hey you should deselect this object" and `WI.Table` can look and see "oh hey this object isn't there anymore... ignore". > Source/WebInspectorUI/UserInterface/Views/TreeOutline.js:343 > - treeOutline._selectionController.didRemoveItem(childIndex); > + treeOutline._selectionController.didRemoveItems(new WI.IndexSet([removedIndex])); I think we should be doing this inside `_forgetTreeElement`, as I don't see any logic right now that would notify the `WI.SelectionController` when `removeChildren` is called. Additionally, if I have a parent P and child C selected, but call `removeChild(P)`, I'd also expect C to be removed. The same is true for `_rememberTreeElement`, as right now only P would trigger a `didInsertItem`.
Matt Baker
Comment 14 2018-12-06 16:04:16 PST
Comment on attachment 356687 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=356687&action=review >> Source/WebInspectorUI/UserInterface/Views/TreeOutline.js:343 >> + treeOutline._selectionController.didRemoveItems(new WI.IndexSet([removedIndex])); > > I think we should be doing this inside `_forgetTreeElement`, as I don't see any logic right now that would notify the `WI.SelectionController` when `removeChildren` is called. Additionally, if I have a parent P and child C selected, but call `removeChild(P)`, I'd also expect C to be removed. > > The same is true for `_rememberTreeElement`, as right now only P would trigger a `didInsertItem`. I agree we should move the `didRemoveItems` call into `_forgetTreeElement`. However, this will require a change to `removeChildren`: removeChildren() { for (let child of this.children) { // Deselect, forget, and detach } this.children = []; } Because this.children isn't updated until after the loop completes, detached elements (lacking parent/sibling references) remain in the tree while `_forgetTreeElement` is called, causing `_indexOfTreeElement` to fail. To address this we simply need to update the loop as follows: removeChildren() { while (this.children.length) { let child = this.children[0]; // Deselect, forget, and detach this.children.shift(); } }
Matt Baker
Comment 15 2018-12-06 17:24:22 PST
Comment on attachment 356687 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=356687&action=review >> Source/WebInspectorUI/ChangeLog:14 >> + order is unsafe, a detail best left to the SelectionController. > > The bug description was about removing a single item. Is this a drive-by change? Not a drive by. The bug was that Table failed to inform its SelectionController about deleted items. The method was updated to take multiple indexes so that Table.prototype._removeRows could do this efficiently and safely. >>> Source/WebInspectorUI/UserInterface/Controllers/SelectionController.js:228 >>> + this.deselectItem(index); >> >> Could it get called more than once in this loop? >> If so, can one didRemoveItems call cause more than one selectionControllerSelectionDidChange? > > Along the lines of what Nikita said: > > This worries me, because I feel like there's a way for a caller (e.g. `WI.Table`) to remove an item, notify `WI.SelectionController`, who will then tell the caller that an item was removed, only to have the caller then affect something after the removed item. > > Table has rows 2 (A) and 3 (B) selected In practice TreeOutline deselects items before removing, so I don't think this will occur. That said, it is worth considering since this creates the possibility for reentrancy where before there was none. Even if `didRemoveItems` was guaranteed to cause, at most, one selectionControllerSelectionDidChange, this is still a problem. The simplest solution is to not call selectionControllerSelectionDidChange inside of `didRemoveItems`. In practice this should be fine, since TreeOutline deselects items before removing.
Matt Baker
Comment 16 2018-12-06 17:41:07 PST
Devin Rousso
Comment 17 2018-12-11 10:59:17 PST
Comment on attachment 356772 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=356772&action=review r-, as I think there is another issue with this. If [1, 2, 3] are selected, and both [1, 2] are removed simultaneously, I'd expect that [3] becomes [1]. This can happen if we had a DOM tree like this: <0> <1> <2></2> </1> </0> <3></3> If <0> gets removed from the DOM tree via JavaScript, <1> and <2> would get removed from the DOM tree without the frontend/user actually explicitly calling delete. I'm pretty sure that right now, because we call `_forgetTreeElement` on the parent and _then_ the child, we'd hit the assertion that we couldn't get an index for <1> and <2> since they'd already have been removed from the `WI.TreeOutline`. > Source/WebInspectorUI/UserInterface/Views/TreeOutline.js:409 > + // Update the SelectionController before attaching the TreeElement. > + // Attaching the TreeElement can cause it to become selected, and > + // added to the SelectionController. NIT: is this comment strictly necessary, or is it mainly for the benefit of future programming, like as a "gotcha"? > Source/WebInspectorUI/UserInterface/Views/TreeOutline.js:410 > + let index = this._indexOfTreeElement(element) || 0; Should we even be calling `didInsertItem` if we're unable to get an index? You already assert inside `_indexOfTreeElement`, so this seems a bit counter to that expectation. > Source/WebInspectorUI/UserInterface/Views/TreeOutline.js:422 > + this._selectionController.didRemoveItems(new WI.IndexSet([index])); Ditto (>410).
Matt Baker
Comment 18 2018-12-12 13:41:13 PST
Matt Baker
Comment 19 2018-12-12 13:42:17 PST
EWS Watchlist
Comment 20 2018-12-12 14:44:35 PST
Comment on attachment 357162 [details] Patch Attachment 357162 [details] did not pass mac-ews (mac): Output: https://webkit-queues.webkit.org/results/10373138 New failing tests: inspector/table/table-remove-rows.html
EWS Watchlist
Comment 21 2018-12-12 14:44:37 PST
Created attachment 357168 [details] Archive of layout-test-results from ews100 for mac-sierra The attached test failures were seen while running run-webkit-tests on the mac-ews. Bot: ews100 Port: mac-sierra Platform: Mac OS X 10.12.6
EWS Watchlist
Comment 22 2018-12-12 14:53:30 PST
Comment on attachment 357162 [details] Patch Attachment 357162 [details] did not pass mac-wk2-ews (mac-wk2): Output: https://webkit-queues.webkit.org/results/10373160 New failing tests: inspector/table/table-remove-rows.html
EWS Watchlist
Comment 23 2018-12-12 14:53:32 PST
Created attachment 357169 [details] Archive of layout-test-results from ews105 for mac-sierra-wk2 The attached test failures were seen while running run-webkit-tests on the mac-wk2-ews. Bot: ews105 Port: mac-sierra-wk2 Platform: Mac OS X 10.12.6
EWS Watchlist
Comment 24 2018-12-12 15:16:41 PST
Comment on attachment 357162 [details] Patch Attachment 357162 [details] did not pass mac-debug-ews (mac): Output: https://webkit-queues.webkit.org/results/10373119 New failing tests: inspector/table/table-remove-rows.html
EWS Watchlist
Comment 25 2018-12-12 15:16:43 PST
Created attachment 357175 [details] Archive of layout-test-results from ews114 for mac-sierra The attached test failures were seen while running run-webkit-tests on the mac-debug-ews. Bot: ews114 Port: mac-sierra Platform: Mac OS X 10.12.6
Matt Baker
Comment 26 2018-12-12 15:46:47 PST
Matt Baker
Comment 27 2018-12-12 16:10:46 PST
Comment on attachment 356772 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=356772&action=review >> Source/WebInspectorUI/UserInterface/Views/TreeOutline.js:410 >> + let index = this._indexOfTreeElement(element) || 0; > > Should we even be calling `didInsertItem` if we're unable to get an index? You already assert inside `_indexOfTreeElement`, so this seems a bit counter to that expectation. Now that we call this in _rememberTreeElement we should always be able to find an index.
Matt Baker
Comment 28 2018-12-12 16:13:57 PST
(In reply to Devin Rousso from comment #17) > Comment on attachment 356772 [details] > Patch > > View in context: > https://bugs.webkit.org/attachment.cgi?id=356772&action=review > > r-, as I think there is another issue with this. > > If [1, 2, 3] are selected, and both [1, 2] are removed simultaneously, I'd > expect that [3] becomes [1]. This can happen if we had a DOM tree like this: > > <0> > <1> > <2></2> > </1> > </0> > <3></3> > > If <0> gets removed from the DOM tree via JavaScript, <1> and <2> would get > removed from the DOM tree without the frontend/user actually explicitly > calling delete. > > I'm pretty sure that right now, because we call `_forgetTreeElement` on the > parent and _then_ the child, we'd hit the assertion that we couldn't get an > index for <1> and <2> since they'd already have been removed from the > `WI.TreeOutline`. This is fixed by changed to TreeOutline `removeChildren` and `removeChildAtIndex`.
Matt Baker
Comment 29 2018-12-12 16:15:19 PST
Devin Rousso
Comment 30 2018-12-12 18:48:04 PST
(In reply to Matt Baker from comment #28) > This is fixed by changed to TreeOutline `removeChildren` and `removeChildAtIndex`. If that's the case, we should have a test for it. Also, it seems like your changes to the "inspector/table/table-remove-rows.html" test got lost across patches. That test seems to still be relevant, so we should add it back.
Matt Baker
Comment 31 2018-12-12 19:21:59 PST
(In reply to Devin Rousso from comment #30) > (In reply to Matt Baker from comment #28) > > This is fixed by changed to TreeOutline `removeChildren` and `removeChildAtIndex`. > If that's the case, we should have a test for it. I can look into writing DOMTreeOutline tests when I work on https://webkit.org/b/192044. It should be possible to write layout tests for DOMTreeOutline selection that exercise DOMAgent.removeNode and DOMTreeUpdater. > Also, it seems like your changes to the > "inspector/table/table-remove-rows.html" test got lost across patches. That > test seems to still be relevant, so we should add it back. Oops!
Matt Baker
Comment 32 2018-12-12 19:23:06 PST
Devin Rousso
Comment 33 2018-12-13 12:19:56 PST
Comment on attachment 357203 [details] Patch rs=me, please don't forget to write tests in <https://webkit.org/b/192044>. Indexes seems quite brittle, so I'd like to know for sure what our limitations/expectations are.
WebKit Commit Bot
Comment 34 2018-12-13 12:47:07 PST
Comment on attachment 357203 [details] Patch Clearing flags on attachment: 357203 Committed r239175: <https://trac.webkit.org/changeset/239175>
WebKit Commit Bot
Comment 35 2018-12-13 12:47:10 PST
All reviewed patches have been landed. Closing bug.
Note You need to log in before you can comment on or make changes to this bug.