/react-query-auth

⚛️ Authenticate your react applications easily with react-query.

Primary LanguageTypeScriptMIT LicenseMIT

react-query-auth

NPM

Authenticate your react applications easily with react-query.

Introduction

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.

Table of Contents

Demo:

CodeSandbox

It is also possible to try out the sample app locally via storybook by running following command:

$ npm run storybook

Prerequisites

It is required to have react-query library installed and configured.

Installation

$ npm install react-query-auth

Or if you use Yarn:

$ yarn add react-query-auth

Usage

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>;
};

API

initReactQueryAuth

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,
});
configuration
  • 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 to false it will fetch the user in the background.
    • defaults to true
  • 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>)

AuthProvider

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>
  );
};

useAuth

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>;
};
returns context value:
  • 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
  • error: Error | null

    • error object
    • type can be provided by passing it to initReactQueryAuth

AuthConsumer

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>
  );
};

Contributing

  1. Clone this repo
  2. Create a branch: git checkout -b your-feature
  3. Make some changes
  4. Test your changes
  5. Push your branch and open a Pull Request

LICENSE

MIT