Authenticate your react applications easily with react-query.
Thanks to react-query we have been able to reduce our codebases by a lot by caching server state with it. However, we still have to think about where to store the user data. The user data can be considered as a global application state because we need to access it from lots of places in the application. On the other hand, it is also a server state since all the user data is expected to arrive from a server. With this library, we can manage user authentication in an easy way. It is agnostic of the method you are using for authenticating your application, it can be adjusted according to the API it is being used against. It just needs the configuration to be provided and the rest will be set up automatically.
It is also possible to try out the sample app locally via storybook by running following command:
$ npm run storybook
It is required to have react-query
library installed and configured.
$ npm install react-query-auth
Or if you use Yarn:
$ yarn add react-query-auth
First of all, AuthProvider
and useAuth
must be initialized and exported.
// src/lib/auth.ts
import { initReactQueryAuth } from 'react-query-auth';
import { loginUser, loginFn, registerFn, logoutFn } from '...';
interface User {
id: string;
name: string;
}
interface Error {
message: string;
}
const authConfig = {
loadUser,
loginFn,
registerFn,
logoutFn,
};
export const { AuthProvider, useAuth } = initReactQueryAuth<
User,
Error,
LoginCredentials,
RegisterCredentials
>(authConfig);
AuthProvider
should be rendered inside the QueryClientProvider
from react-query
.
// src/App.tsx
import { QueryClient } from 'react-query';
import { QueryClientProvider } from 'react-query/devtools';
import { AuthProvider } from 'src/lib/auth';
const queryClient = new QueryClient();
export const App = () => {
return (
<QueryClientProvider client={queryClient}>
<AuthProvider>
{//the rest the app goes here}
</AuthProvider>
</QueryClientProvider>
);
};
Then the user data is accessible from any component rendered inside the provider via the useAuth
hook:
// src/components/UserInfo.tsx
import { useAuth } from 'src/lib/auth';
export const UserInfo = () => {
const { user } = useAuth();
return <div>My Name is {user.name}</div>;
};
Function that initializes and returns AuthProvider
, AuthConsumer
and useAuth
.
// src/lib/auth.ts
export const { AuthProvider, useAuth } = initReactQueryAuth<User, Error>({
key,
loadUser,
loginFn,
registerFn,
logoutFn,
waitInitial,
LoaderComponent,
ErrorComponent,
});
-
key: string
- key that is being used by react-query.
- defaults to
'auth-user'
-
loadUser: (data:any) => Promise<User>
- Required
- function that handles user profile fetching
-
loginFn: (data:any) => Promise<User>
- Required
- function that handles user login
-
registerFn: (data:any) => Promise<User>
- Required
- function that handles user registration
-
logoutFn: (data:unknown) => Promise<any>
- Required
- function that handles user logout
-
logoutFn: () => Promise<any>
- Required
- function that handles user logout
-
waitInitial: boolean
- Flag for checking if the provider should show
LoaderComponent
while fetching the user. If set tofalse
it will fetch the user in the background. - defaults to
true
- Flag for checking if the provider should show
-
LoaderComponent: () => React.ReactNode
- component for the loader
- defaults to
() => <div>Loading...</div>
-
ErrorComponent: (error: Error) => React.ReactNode
- component for the error
- defaults to
(error) => (<div style={{color: 'tomato'}}>{JSON.stringify(error, null, 2)}</div>)
The provider wraps the app and allows useAuth
hook data to be available across the app. No further configuration required.
import { AuthProvider } from 'src/lib/auth';
export const App = () => {
return (
<QueryClientProvider client={queryClient}>
<AuthProvider>
{//the rest of the app}
</AuthProvider>
</QueryClientProvider>
);
};
The hook allows access of the user data across the app.
import { useAuth } from 'src/lib/auth';
export const UserInfo = () => {
const { user, login, logout, register, error, refetch } = useAuth();
return <div>My Name is {user.name}</div>;
};
-
user: User | undefined
- user data that was retrieved from server
- type can be provided by passing it to
initReactQueryAuth
generic
-
login: (variables: TVariables, { onSuccess, onSettled, onError }) => Promise<TData>
- function to login the user
-
logout: (variables: TVariables, { onSuccess, onSettled, onError }) => Promise<TData>
- function to logout the user
-
register: (variables: TVariables, { onSuccess, onSettled, onError }) => Promise<TData>
- function to register the user
-
isLoggingIn: boolean
- checks if is logging in is in progress
-
isLoggingOut: boolean
- checks if is logging out is in progress
-
isRegistering: boolean
- checks if is registering in is in progress
-
refetchUser: (options?: RefetchOptions | undefined) => Promise<QueryObserverResult<User, Error>>
- function for refetching user data. this can also be done by invalidating its query by
key
- function for refetching user data. this can also be done by invalidating its query by
-
error: Error | null
- error object
- type can be provided by passing it to
initReactQueryAuth
Exposes the same context value like the useAuth
hook but as render prop.
import { AuthProvider, AuthConsumer } from './lib/auth';
import { ReactQueryProvider } from './lib/react-query';
export const App = () => {
return (
<ReactQueryProvider>
<AuthProvider>
<AuthConsumer>
{({ user }) => <div>{JSON.stringify(user) || 'No User Found'}</div>}
</AuthConsumer>
</AuthProvider>
</ReactQueryProvider>
);
};
- Clone this repo
- Create a branch:
git checkout -b your-feature
- Make some changes
- Test your changes
- Push your branch and open a Pull Request
MIT