-
-
Notifications
You must be signed in to change notification settings - Fork 5.3k
/
Copy pathCheckForApplicationUpdate.md
193 lines (147 loc) · 6.86 KB
/
CheckForApplicationUpdate.md
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
---
layout: default
title: "The CheckForApplicationUpdate component"
---
# `CheckForApplicationUpdate`
When your admin application is a Single Page Application, users who keep a browser tab open at all times might not use the most recent version of the application unless you tell them to refresh the page.
This component regularly checks whether the application source code has changed and prompts users to reload the page when an update is available. To detect updates, it fetches the current URL at regular intervals and compares the hash of the response content (usually the HTML source). This should be enough in most cases as bundlers usually update the links to the application bundles after an update.

## Usage
Include this component in a custom layout:
```tsx
// in src/MyLayout.tsx
import type { ReactNode } from 'react';
import { CheckForApplicationUpdate, Layout } from 'react-admin';
export const MyLayout = ({ children }: { children: ReactNode}) => (
<Layout>
{children}
<CheckForApplicationUpdate />
</Layout>
);
// in src/App.tsx
import { Admin, ListGuesser, Resource } from 'react-admin';
import { MyLayout } from './MyLayout';
export const App = () => (
<Admin layout={MyLayout}>
<Resource name="posts" list={ListGuesser} />
</Admin>
);
```
## Props
`<CheckForApplicationUpdate>` accepts the following props:
| Prop | Required | Type | Default | Description |
| --------------- | -------- | -------------- | ------------------ |-------------------------------------------------------------------- |
| `interval` | Optional | `number` | `3600000` (1 hour) | The interval in milliseconds between two checks |
| `disabled` | Optional | `boolean` | `false` | Whether the automatic check is disabled |
| `notification` | Optional | `ReactElement` | | The notification to display to the user when an update is available |
| `onNewVersion Available` | Optional | `function` | | The effect to execute when a new version is detected. |
| `url` | Optional | `string` | Current URL | The URL to download to check for code update |
| `fetchOptions` | Optional | `RequestInit | undefined` | `undefined` | The options passed to `fetch` when checking for an update |
## `interval`
You can customize the interval between each check by providing the `interval` prop. It accepts a number of milliseconds and is set to `3600000` (1 hour) by default.
```tsx
// in src/MyLayout.tsx
import type { ReactNode } from 'react';
import { CheckForApplicationUpdate, Layout } from 'react-admin';
const HALF_HOUR = 30 * 60 * 1000;
export const MyLayout = ({ children }: { children: ReactNode}) => (
<Layout>
{children}
<CheckForApplicationUpdate interval={HALF_HOUR} />
</Layout>
);
```
## `disabled`
You can dynamically disable the automatic application update detection by providing the `disabled` prop. By default, it's only enabled in `production` mode.
```tsx
// in src/MyLayout.tsx
import type { ReactNode } from 'react';
import { CheckForApplicationUpdate, Layout } from 'react-admin';
export const MyLayout = ({ children }: { children: ReactNode}) => (
<Layout>
{children}
<CheckForApplicationUpdate disabled={process.env.NODE_ENV !== 'production'} />
</Layout>
);
```
## `notification`
You can customize the notification shown to users when an update is available by passing your own element to the `notification` prop.
Note that you must wrap your component with `forwardRef`.
```tsx
// in src/MyLayout.tsx
import { forwardRef, ReactNode } from 'react';
import { Layout, CheckForApplicationUpdate } from 'react-admin';
const CustomAppUpdatedNotification = forwardRef((props, ref) => (
<Alert
ref={ref}
severity="info"
action={
<Button
color="inherit"
size="small"
onClick={() => window.location.reload()}
>
Update
</Button>
}
>
A new version of the application is available. Please update.
</Alert>
));
const MyLayout = ({ children }: { children: ReactNode}) => (
<Layout>
{children}
<CheckForApplicationUpdate notification={<CustomAppUpdatedNotification />}/>
</Layout>
);
```
If you want to customize the behavior when a new version is available, checkout the [`onNewVersionAvailable` section](#onnewversionavailable). If you just want to customize the notification texts, including the button, check out the [Internationalization section](#internationalization).
## `onNewVersionAvailable`
Advanced users who wish to customize the handling function other than just displaying a notification can leverage the `onNewVersionAvailable` prop:
```tsx
import { CheckForApplicationUpdate, useNotify } from "react-admin";
export const MyCheckForApplicationUpdate = () => {
const notify = useNotify();
const onNewVersionAvailable = () => {
// Perform a backup of user preference in localStorage in case bad things happen
const preference1 = localStorage.getItem("preference1");
const preference2 = localStorage.getItem("preference2");
const checkpointData = {
preference1,
preference2,
};
localStorage.setItem(
`checkpoint_${new Date().toISOString()}`,
JSON.stringify(checkpointData),
);
// Notify user
notify("New Version Ready to Update");
};
return (
<CheckForApplicationUpdate
onNewVersionAvailable={onNewVersionAvailable}
/>
);
};
```
## `url`
You can customize the URL fetched to detect updates by providing the `url` prop. By default, it's the current URL.
```tsx
// in src/MyLayout.tsx
import type { ReactNode } from 'react';
import { CheckForApplicationUpdate, Layout } from 'react-admin';
const MY_APP_ROOT_URL = 'https://admin.mycompany.com';
export const MyLayout = ({ children }: { children: ReactNode}) => (
<Layout>
{children}
<CheckForApplicationUpdate url={MY_APP_ROOT_URL} />
</Layout>
);
```
## `fetchOptions`
You can also customize the request [options](https://developer.mozilla.org/en-US/docs/Web/API/fetch#options) passed along to fetch function when detecting updates.
Tip: Depending on your server-side HTTP cache settings, you may want to set the fetchOptions to `{ cache: "no-cache" }` to check if the resource has changed.
## Internationalization
You can customize the texts of the default notification by overriding the following keys:
* `ra.notification.application_update_available`: the notification text
* `ra.action.update_application`: the reload button text