Skip to content

Commit

Permalink
Merge pull request #5180 from marmelab/getIdentity
Browse files Browse the repository at this point in the history
Add user name and avatar on the top bar
  • Loading branch information
djhi authored Aug 24, 2020
2 parents 4abaeeb + 5275630 commit d580844
Show file tree
Hide file tree
Showing 9 changed files with 238 additions and 23 deletions.
66 changes: 56 additions & 10 deletions docs/Authentication.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,9 @@ title: "Authentication"

# Authentication

![Logout button](./img/login.gif)
![Login](./img/login.gif)

React-admin lets you secure your admin app with the authentication strategy of your choice. Since there are many possible strategies (Basic Auth, JWT, OAuth, etc.), react-admin simply provides hooks to execute your own authentication code.
React-admin lets you secure your admin app with the authentication strategy of your choice. Since there are many possible strategies (Basic Auth, JWT, OAuth, etc.), react-admin delegates authentication logic to your `authProvider`, and provides hooks to execute your authentication code.

## The `authProvider`

Expand All @@ -33,6 +33,7 @@ const authProvider = {
checkAuth: params => Promise.resolve(),
checkError: error => Promise.resolve(),
getPermissions: params => Promise.resolve(),
getIdentity: () => Promise.resolve(),
};
```

Expand Down Expand Up @@ -66,8 +67,8 @@ const authProvider = {
}
return response.json();
})
.then(({ token }) => {
localStorage.setItem('token', token);
.then(auth => {
localStorage.setItem('auth', JSON.stringify(auth));
});
},
// ...
Expand All @@ -94,7 +95,7 @@ const httpClient = (url, options = {}) => {
if (!options.headers) {
options.headers = new Headers({ Accept: 'application/json' });
}
const token = localStorage.getItem('token');
const { token } = JSON.parse(localStorage.getItem('auth'));
options.headers.set('Authorization', `Bearer ${token}`);
return fetchUtils.fetchJson(url, options);
};
Expand All @@ -111,6 +112,47 @@ Now the admin is secured: The user can be authenticated and use their credential

If you have a custom REST client, don't forget to add credentials yourself.

## User Identity

React-admin displays the current user name and avatar on the top right side of the screen. To enable this feature, implement the `getIdentity` method in the `authProvider`:

```js
// in src/authProvider.js
const authProvider = {
login: ({ username, password }) => { /* ... */ },
getIdentity: () => {
const { id, fullName, avatar } = JSON.parse(localStorage.getItem('auth'));
return { id, fullName, avatar };
}
// ...
};

export default authProvider;
```

React-admin uses the `fullName` and the `avatar` (an image source, or a data-uri) in the App Bar:

![User identity](./img/identity.png)

**Tip**: You can use the `id` field to identify the current user in your code, by calling the `useGetIdentity` hook:

```jsx
import { useGetIdentity, useGetOne } from 'react-admin';

const PostDetail = ({ id }) => {
const { data: post, loading: postLoading } = useGetOne('posts', id);
const { identity, loading: identityLoading } = useGetIdentity();
if (postLoading || identityLoading) return <>Loading...</>;
if (!post.lockedBy || post.lockedBy === identity.id) {
// post isn't locked, or is locked by me
return <PostEdit post={post} />
} else {
// post is locked by someone else and cannot be edited
return <PostShow post={post} />
}
}
```

## Logout Configuration

As soon as you provide an `authProvider` prop to `<Admin>`, react-admin displays a logout button in the top bar (or in the menu on mobile). When the user clicks on the logout button, this calls the `authProvider.logout()` method, and removes potentially sensitive data from the Redux store. Then the user gets redirected to the login page.
Expand All @@ -121,8 +163,9 @@ So it's the responsibility of the `authProvider` to clean up the current authent
// in src/authProvider.js
export default {
login: ({ username, password }) => { /* ... */ },
getIdentity: () => { /* ... */ },
logout: () => {
localStorage.removeItem('token');
localStorage.removeItem('auth');
return Promise.resolve();
},
// ...
Expand All @@ -147,11 +190,12 @@ For instance, to redirect the user to the login page for both 401 and 403 codes:
// in src/authProvider.js
export default {
login: ({ username, password }) => { /* ... */ },
getIdentity: () => { /* ... */ },
logout: () => { /* ... */ },
checkError: (error) => {
const status = error.status;
if (status === 401 || status === 403) {
localStorage.removeItem('token');
localStorage.removeItem('auth');
return Promise.reject();
}
return Promise.resolve();
Expand All @@ -168,15 +212,16 @@ Redirecting to the login page whenever a REST response uses a 401 status code is

Fortunately, each time the user navigates, react-admin calls the `authProvider.checkAuth()` method, so it's the ideal place to validate the credentials.

For instance, to check for the existence of the token in local storage:
For instance, to check for the existence of the authentication data in local storage:

```js
// in src/authProvider.js
export default {
login: ({ username, password }) => { /* ... */ },
getIdentity: () => { /* ... */ },
logout: () => { /* ... */ },
checkError: (error) => { /* ... */ },
checkAuth: () => localStorage.getItem('token')
checkAuth: () => localStorage.getItem('auth')
? Promise.resolve()
: Promise.reject(),
// ...
Expand All @@ -189,9 +234,10 @@ If the promise is rejected, react-admin redirects by default to the `/login` pag
// in src/authProvider.js
export default {
login: ({ username, password }) => { /* ... */ },
getIdentity: () => { /* ... */ },
logout: () => { /* ... */ },
checkError: (error) => { /* ... */ },
checkAuth: () => localStorage.getItem('token')
checkAuth: () => localStorage.getItem('auth')
? Promise.resolve()
: Promise.reject({ redirectTo: '/no-access' }),
// ...
Expand Down
Binary file added docs/img/identity.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
7 changes: 7 additions & 0 deletions examples/demo/src/authProvider.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,13 @@ const authProvider: AuthProvider = {
checkAuth: () =>
localStorage.getItem('username') ? Promise.resolve() : Promise.reject(),
getPermissions: () => Promise.reject('Unknown method'),
getIdentity: () =>
Promise.resolve({
id: 'user',
fullName: 'Jane Doe',
avatar:
'',
}),
};

export default authProvider;
Loading

0 comments on commit d580844

Please sign in to comment.