Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix serialization and mutation of <textarea> elements #1351

Merged
merged 12 commits into from
Dec 1, 2023

Conversation

eoghanmurray
Copy link
Contributor

@eoghanmurray eoghanmurray commented Nov 9, 2023

Bug report: #1301

Fix serialization and mutation of <textarea> elements, taking account the duality that the value can be set in either the child node, or in the value parameter (not attribute)

This was prompted by the #1326 PR but goes much further by adding tests and clarifying that textarea values should always be serialized in the 'value' attribute in the snapshot, and rebuilt as a childNode (textareas are unusual compared with where the value is always in the value attribute/property)

So

  • always read from textarea.value property
  • when writing static html, always write the value property into the (single) childNode

In snapshots, I've chosen to always serialize the childNodes into a 'value' attribute, so that masking can be done consistently with other elements (I actually first went with the alternative of masking childNodes; the extra complexity can be seen in the addition of a new maskInputValue call in d5b93ac)

When text is typed into the textarea by the user, the textarea value is changed but the childnodes stay the same;
if a childNode mutation happens it may or may not affect the value property depending on whether text has already been typed by user. To satisfy this, value mutations are rebuilt into childNodes, and don't modify the .value property directly; that property should only be modified by a text IncrementalSource.Input event.

All of this should be covered by examples in the tests.

Copy link

changeset-bot bot commented Nov 9, 2023

🦋 Changeset detected

Latest commit: 9df40e4

The changes in this PR will be included in the next version bump.

This PR includes changesets to release 8 packages
Name Type
rrweb-snapshot Patch
rrweb Patch
rrdom Patch
rrdom-nodejs Patch
rrweb-player Patch
@rrweb/types Patch
@rrweb/web-extension Patch
rrvideo Patch

Not sure what this means? Click here to learn what changesets are.

Click here if you're a maintainer who wants to add another changeset to this PR

@eoghanmurray
Copy link
Contributor Author

I've added a new aspect to rrweb/test/integration.test.ts in that it now tests the replayer as well as checking the snapshot; the replayer is needed to read out what the actual values of the textarea are after specific mutations are applied.

@josephmathew900
Copy link

#1301 (comment)

@eoghanmurray Do we have any blockers to getting this PR merged? After merging this PR, in which version will the fix be available?

…the duality that the value can be set in either the child node, or in the value _parameter_ (not attribute)
 - this is to fix up 'historical' recordings, as duplicate textarea content should no longer be being created at record time (see previous commit in this PR)
 - new test shows what the snapshot generated by previous versions of rrweb used to look like, hence 'bad'
 - original 0efe23f fix either didn't work or no longer works due to childNodes being appended subsequent to this part of the code
…he value was recorded as a child node

 - I didn't notice that form.html was used in other tests, so lucky that I noticed that those tests also should have the 'pre value' masked out
…te (from it's DOM property) and not as a childNode. It should still be rebuilt as a childNode rather than a property
…e was extremely painful to get right from a typings point of view)
@eoghanmurray
Copy link
Contributor Author

I've rebased based on upstream; we need a review from another core team member.

return document.querySelector('textarea')!;
};

it('should record textarea values once', () => {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe we can add some comments here to document that we are doing the inverse of browser behaviour for our serialising (we are using the textarea's value attribute, and ignoring child nodes during serialization, where the browser ignores the value attribute but uses the child text nodes)

…prehension of why we are doing things this way
Comment on lines 53 to 64
id: 6,
type: 2,
tagName: 'textarea',
attributes: {
value: 'dupe',
},
childNodes: [
{
type: 3,
textContent: 'dupe should ignore',
id: 7,
},
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe we can add some comments here to document that we are doing the inverse of browser behaviour for our serialising (we are using the textarea's value attribute, and ignoring child nodes during serialization, where the browser ignores the value attribute but uses the child text nodes)

type: 2,
tagName: 'textarea',
attributes: {
value: 'dupe',
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
value: 'dupe',
value: "text that is used in rrweb's replay",

childNodes: [
{
type: 3,
textContent: 'dupe should ignore',
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
textContent: 'dupe should ignore',
textContent: 'ignored when rrweb's replay reads this serialized child node',

@eoghanmurray eoghanmurray merged commit a2be77b into rrweb-io:master Dec 1, 2023
billyvg added a commit to getsentry/rrweb that referenced this pull request Apr 26, 2024
jxiwang pushed a commit to amplitude/rrweb that referenced this pull request Jul 31, 2024
* Fix serialization and mutation of <textarea> elements taking account the duality that the value can be set in either the child node, or in the value _parameter_ (not attribute)

* Backwards compatibility: Bug fix and regression test for rrweb-io#112
 - this is to fix up 'historical' recordings, as duplicate textarea content should no longer be being created at record time
 - new test shows what the snapshot generated by previous versions of rrweb used to look like, hence 'bad'
 - original 0efe23f fix either didn't work or no longer works due to childNodes being appended subsequent to this part of the code
 - during review, we also verified that the `_cssText` case should still be handled okay, as there's currently no scenario where csstext is present with css child nodes of a <style>

* Masking: Fix that textarea values were being missed by the masking system if the value was recorded as a child node
 - I didn't notice that form.html was used in other tests, so lucky that I noticed that those tests also should have the 'pre value' masked out

* Simplify by always storing the textarea value in the `.value` attribute (from it's DOM property) and not as a childNode. It should still be rebuilt as a childNode rather than a property
---------

Authored-by: eoghanmurray <[email protected]>
lewgordon-amplitude pushed a commit to amplitude/rrweb that referenced this pull request Aug 1, 2024
* Fix serialization and mutation of <textarea> elements taking account the duality that the value can be set in either the child node, or in the value _parameter_ (not attribute)

* Backwards compatibility: Bug fix and regression test for rrweb-io#112
 - this is to fix up 'historical' recordings, as duplicate textarea content should no longer be being created at record time
 - new test shows what the snapshot generated by previous versions of rrweb used to look like, hence 'bad'
 - original 0efe23f fix either didn't work or no longer works due to childNodes being appended subsequent to this part of the code
 - during review, we also verified that the `_cssText` case should still be handled okay, as there's currently no scenario where csstext is present with css child nodes of a <style>

* Masking: Fix that textarea values were being missed by the masking system if the value was recorded as a child node
 - I didn't notice that form.html was used in other tests, so lucky that I noticed that those tests also should have the 'pre value' masked out

* Simplify by always storing the textarea value in the `.value` attribute (from it's DOM property) and not as a childNode. It should still be rebuilt as a childNode rather than a property
---------

Authored-by: eoghanmurray <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants