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

[data grid] invalid gridState in local storage causes crash on page load #15940

Open
Tracked by #9328
hdale94 opened this issue Dec 19, 2024 · 15 comments
Open
Tracked by #9328

[data grid] invalid gridState in local storage causes crash on page load #15940

hdale94 opened this issue Dec 19, 2024 · 15 comments
Labels
component: data grid This is the name of the generic UI component, not the React module! feature: Row grouping Related to the data grid Row grouping feature recipe support: premium standard Support request from a Premium standard plan user. https://mui.com/legal/technical-support-sla/ waiting for 👍 Waiting for upvotes

Comments

@hdale94
Copy link

hdale94 commented Dec 19, 2024

Steps to reproduce

The issue

Steps:

  1. Open this link to live example: https://codesandbox.io/p/sandbox/dry-microservice-forked-927pcx?workspaceId=ws_SzhXwxFRYxciKVso4R9bwN
  2. Open sandbox in a new tab (icon in upper right corner)
  3. Group name column, which should cause crash
  4. Refresh page, and it will forever crash because the invalid gridState was saved in local storage

Current behavior

Invalid grid state in local storage crashes application when initializing grid state.

Expected behavior

Allow us to catch DataGrid errors, e.g. with ErrorBoundary so we can handle errors gracefully.

Context

No response

Your environment

npx @mui/envinfo
  Don't forget to mention which browser you used.
  Output from `npx @mui/envinfo` goes here.

Search keywords: DataGridPremium, Crash, Grouping

Order ID: 83335

@hdale94 hdale94 added bug 🐛 Something doesn't work status: waiting for maintainer These issues haven't been looked at yet by a maintainer labels Dec 19, 2024
@github-actions github-actions bot added component: data grid This is the name of the generic UI component, not the React module! support: premium standard Support request from a Premium standard plan user. https://mui.com/legal/technical-support-sla/ labels Dec 19, 2024
@hdale94 hdale94 changed the title DataGridPremium stores invalid gridState in local storage, causes crash on load DataGridPremium stores invalid gridState in local storage, causes crash on page load Dec 19, 2024
@michelengelen
Copy link
Member

The main problem here is the crash that is caused by an incorrect block of code.
valueGetter should either check if the row in question is autogenerated: Autogenerated rows

example:

valueGetter: (_, row: RowType) => {
  if (isAutogeneratedRow(row)) {
    return '[this is an autogenerated row]';
  }
  ...
},

Or you can adapt to something else and fix the error in your function:

valueGetter: (_, row: RowType) => {
  // Extract fieldNames from all questions based on the questionIds in the row
  const questionNames = questions
-   ?.filter((question) => row.questionIds.includes(question.fieldId))
+   ?.filter((question) => row.questionIds?.includes(question.fieldId))
    .map((question) => question.fieldName)
    .join(', ');
  return questionNames ?? '';
},

This should solve your issue

@michelengelen michelengelen added status: waiting for author Issue with insufficient information feature: Row grouping Related to the data grid Row grouping feature and removed status: waiting for maintainer These issues haven't been looked at yet by a maintainer labels Dec 19, 2024
@michelengelen michelengelen changed the title DataGridPremium stores invalid gridState in local storage, causes crash on page load [data grid] invalid gridState in local storage causes crash on page load Dec 19, 2024
@hdale94
Copy link
Author

hdale94 commented Dec 19, 2024

The main problem here is the crash that is caused by an incorrect block of code. valueGetter should either check if the row in question is autogenerated: Autogenerated rows

example:

valueGetter: (_, row: RowType) => {
  if (isAutogeneratedRow(row)) {
    return '[this is an autogenerated row]';
  }
  ...
},

Or you can adapt to something else and fix the error in your function:

valueGetter: (_, row: RowType) => {
  // Extract fieldNames from all questions based on the questionIds in the row
  const questionNames = questions
-   ?.filter((question) => row.questionIds.includes(question.fieldId))
+   ?.filter((question) => row.questionIds?.includes(question.fieldId))
    .map((question) => question.fieldName)
    .join(', ');
  return questionNames ?? '';
},

This should solve your issue

Thanks, I didn't know about the autogenerated rows.

While this solves the issue in this particular case, it doesn't solve the issue at its root, which is that the data grid will store an invalid state to local storage.

@github-actions github-actions bot added status: waiting for maintainer These issues haven't been looked at yet by a maintainer and removed status: waiting for author Issue with insufficient information labels Dec 19, 2024
@michelengelen
Copy link
Member

The main problem here is the crash that is caused by an incorrect block of code. valueGetter should either check if the row in question is autogenerated: Autogenerated rows

example:

valueGetter: (_, row: RowType) => {

if (isAutogeneratedRow(row)) {

return '[this is an autogenerated row]';

}

...

},

Or you can adapt to something else and fix the error in your function:

valueGetter: (_, row: RowType) => {

// Extract fieldNames from all questions based on the questionIds in the row

const questionNames = questions

  • ?.filter((question) => row.questionIds.includes(question.fieldId))
  • ?.filter((question) => row.questionIds?.includes(question.fieldId))
.map((question) => question.fieldName)
.join(', ');

return questionNames ?? '';

},

This should solve your issue

Thanks, I didn't know about the autogenerated rows.

While this solves the issue in this particular case, it doesn't solve the issue at its root, which is that the data grid will store an invalid state to local storage.

Why should the state be invalid? Once you resolve that problem with the row property not being available the error will go away. The state that is being stored to the local storage is not related to that.

@michelengelen michelengelen added status: waiting for author Issue with insufficient information and removed status: waiting for maintainer These issues haven't been looked at yet by a maintainer labels Dec 19, 2024
@hdale94
Copy link
Author

hdale94 commented Dec 19, 2024

The main problem here is the crash that is caused by an incorrect block of code. valueGetter should either check if the row in question is autogenerated: Autogenerated rows

example:

valueGetter: (_, row: RowType) => {

if (isAutogeneratedRow(row)) {

return '[this is an autogenerated row]';

}

...

},

Or you can adapt to something else and fix the error in your function:

valueGetter: (_, row: RowType) => {

// Extract fieldNames from all questions based on the questionIds in the row

const questionNames = questions

  • ?.filter((question) => row.questionIds.includes(question.fieldId))
  • ?.filter((question) => row.questionIds?.includes(question.fieldId))
.map((question) => question.fieldName)
.join(', ');

return questionNames ?? '';

},

This should solve your issue

Thanks, I didn't know about the autogenerated rows.
While this solves the issue in this particular case, it doesn't solve the issue at its root, which is that the data grid will store an invalid state to local storage.

Why should the state be invalid? Once you resolve that problem with the row property not being available the error will go away. The state that is being stored to the local storage is not related to that.

The fix you provided does solve the issue in this particular case, but when having large data-grids, 10-20 columns with some valueGetters, catching these potential errors is difficult.

It's really about the order of the operations, which should ensure validation before setting the grid-state to local storage.

This is how I experience it currently:

  1. Click group column
  2. New data-grid state is set to local storage
  3. Row grouping occurs and fails

Now you're stuck with a state in local storage that crashes the data-grid because the state was stored before the row grouping occurred.

It would IMO make more sense this way:

  1. Click group column
  2. Row grouping occurs and fails
  3. New data-grid state is not set to local storage because the application has already crashed

@github-actions github-actions bot added status: waiting for maintainer These issues haven't been looked at yet by a maintainer and removed status: waiting for author Issue with insufficient information labels Dec 19, 2024
@cherniavskii
Copy link
Member

Correct me if I'm wrong, but localStorage and state export are unrelated to the reported issue.
I've created this minimal demo that doesn't use the initial state at all and I can still reproduce the issue you reported: https://codesandbox.io/p/sandbox/dry-microservice-forked-dh6mwt

The Autogenerated rows section is indeed the current solution to avoid this.

Hope this helps!

@cherniavskii cherniavskii added status: waiting for author Issue with insufficient information and removed status: waiting for maintainer These issues haven't been looked at yet by a maintainer labels Dec 19, 2024
@hdale94
Copy link
Author

hdale94 commented Dec 19, 2024

Correct me if I'm wrong, but localStorage and state export are unrelated to the reported issue. I've created this minimal demo that doesn't use the initial state at all and I can still reproduce the issue you reported: https://codesandbox.io/p/sandbox/dry-microservice-forked-dh6mwt

The Autogenerated rows section is indeed the current solution to avoid this.

Hope this helps!

I agree partly, but the issue is really that there is seemingly no graceful handling in place before updating the grid-state, or when loading the grid state.

Does it make sense to push a complex object to local storage if any errors are thrown in the grid? In my opinion no, especially if I can't validate the incoming object when reading back from local storage.

If there was a function that could validate the grid-state to the rows, it would atleast allow us to handle these errors gracefully when reading the state.

@github-actions github-actions bot added status: waiting for maintainer These issues haven't been looked at yet by a maintainer and removed status: waiting for author Issue with insufficient information labels Dec 19, 2024
@michelengelen
Copy link
Member

As mentioned already, the grid state is unrelated to the error. We could validate it, but it would still run into the error, since the state in itself is correct. The valueGetter implementation in this case is the source of the problem.

Additionally the local storage recipe is not a feature of the grid, but a recipe to get you started on a potential implementation. So if you want you can adjust the recipe and remove the object on error, or don't even store it in the first place.

IMHO adding a method to "test" the state against the current set of rows would mean an immense performance impact.

@michelengelen michelengelen added status: waiting for author Issue with insufficient information and removed status: waiting for maintainer These issues haven't been looked at yet by a maintainer labels Dec 20, 2024
@hdale94
Copy link
Author

hdale94 commented Dec 20, 2024

Yeah, I understand the performance impact.

Additionally the local storage recipe is not a feature of the grid, but a recipe to get you started on a potential implementation. So if you want you can adjust the recipe and remove the object on error, or don't even store it in the first place.

Do you have a suggestion how I can remove the object on error? A Try...Catch?

@github-actions github-actions bot added status: waiting for maintainer These issues haven't been looked at yet by a maintainer and removed status: waiting for author Issue with insufficient information labels Dec 20, 2024
@michelengelen
Copy link
Member

Yeah, I understand the performance impact.

Additionally the local storage recipe is not a feature of the grid, but a recipe to get you started on a potential implementation. So if you want you can adjust the recipe and remove the object on error, or don't even store it in the first place.

Do you have a suggestion how I can remove the object on error? A Try...Catch?

Typically you would do this with an ErrorBoundary. But it's been quite some time since i built something like that. Maybe @cherniavskii knows a good way to do this. I am not opposed to add this to the docs if we find a good enough solution for this as well to prevent problems like this beforehand! 👍🏼

@hdale94
Copy link
Author

hdale94 commented Dec 21, 2024

Yeah, I understand the performance impact.

Additionally the local storage recipe is not a feature of the grid, but a recipe to get you started on a potential implementation. So if you want you can adjust the recipe and remove the object on error, or don't even store it in the first place.

Do you have a suggestion how I can remove the object on error? A Try...Catch?

Typically you would do this with an ErrorBoundary. But it's been quite some time since i built something like that. Maybe @cherniavskii knows a good way to do this. I am not opposed to add this to the docs if we find a good enough solution for this as well to prevent problems like this beforehand! 👍🏼

I know atleast 1 other person using DataGridPremium who has faced the same issue. I think an example using ErrorBoundary in the docs would be useful for a lot of people.

@michelengelen
Copy link
Member

@mui/xgrid should we add this to the board?

@MBilalShafi MBilalShafi added waiting for 👍 Waiting for upvotes recipe and removed bug 🐛 Something doesn't work status: waiting for maintainer These issues haven't been looked at yet by a maintainer labels Dec 24, 2024
@MBilalShafi
Copy link
Member

MBilalShafi commented Dec 24, 2024

@mui/xgrid should we add this to the board?

Sure, let's keep it as a recipe waiting for upvotes, if we have demand for it in the community, we could make update the existing one or add a new recipe.

@hdale94 Feel free to upvote the issue description.

@hdale94
Copy link
Author

hdale94 commented Dec 24, 2024

@mui/xgrid should we add this to the board?

Sure, let's keep it as a recipe waiting for upvotes, if we have demand for it in the community, we could make update the existing one or add a new recipe.

@hdale94 Feel free to upvote the issue description.

Great, updated the expected behaviour in the issue.

@github-project-automation github-project-automation bot moved this to 🆕 Needs refinement in MUI X Data Grid Dec 26, 2024
@cherniavskii
Copy link
Member

@hdale94 Thanks for the additional details!
Can you give this demo a try? https://codesandbox.io/p/sandbox/dry-microservice-forked-rf84k7
The ErrorBoundary should catch the rendering errors, reset the state, and clear out the local storage.

@hdale94
Copy link
Author

hdale94 commented Dec 30, 2024

@hdale94 Thanks for the additional details! Can you give this demo a try? https://codesandbox.io/p/sandbox/dry-microservice-forked-rf84k7 The ErrorBoundary should catch the rendering errors, reset the state, and clear out the local storage.

Works great, thank you! Please consider including this in the docs, I'm sure other people will find catching errors useful for other scenarios too!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
component: data grid This is the name of the generic UI component, not the React module! feature: Row grouping Related to the data grid Row grouping feature recipe support: premium standard Support request from a Premium standard plan user. https://mui.com/legal/technical-support-sla/ waiting for 👍 Waiting for upvotes
Projects
Status: 🆕 Needs refinement
Development

No branches or pull requests

4 participants