authReducer.js 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. import { gql } from "graphql-request";
  2. import { createApi } from '@reduxjs/toolkit/query/react'
  3. import { graphqlRequestBaseQuery } from '@rtk-query/graphql-request-base-query' //npm install
  4. import { jwtDecode } from "../utills";
  5. import { createSlice, current } from "@reduxjs/toolkit";
  6. import { history } from "../App";
  7. //import { prepareHeaders } from "./index";
  8. export const prepareHeaders = (headers, { getState }) => {
  9. const token = getState().auth.token;
  10. if (token) {
  11. headers.set("Authorization", `Bearer ${token}`);
  12. }
  13. return headers;
  14. }
  15. export const loginApi = createApi({
  16. reducerPath: "authApi",
  17. baseQuery: graphqlRequestBaseQuery({
  18. url: '/graphql',
  19. prepareHeaders
  20. }),
  21. endpoints: (builder) => ({
  22. login: builder.mutation({
  23. query: ({ login, password }) => ({
  24. document: gql`
  25. query login($login: String, $password: String) {
  26. login(login: $login, password: $password)
  27. }
  28. `,
  29. variables: { login, password }
  30. })
  31. }),
  32. userFind: builder.query({
  33. query: (_id) => ({
  34. document: gql`
  35. query UserFind($q: String) {
  36. UserFindOne(query: $q){
  37. _id login nick avatar {url} createdAt
  38. }
  39. }
  40. `,
  41. variables: { q: JSON.stringify([{ _id }]) }
  42. }),
  43. providesTags: (result, error, id) => ([{ type: 'User', id }])
  44. }),
  45. setNick: builder.mutation({
  46. query: ({ _id, nick }) => ({
  47. document: gql`
  48. mutation SetNick($_id:String, $nick: String){
  49. UserUpsert(user: {_id: $_id, nick: $nick}){
  50. _id, nick
  51. }
  52. }
  53. `,
  54. variables: { _id, nick }
  55. }),
  56. invalidatesTags: (result, error, arg) => ([{ type: 'User', id: arg._id }])
  57. })
  58. }),
  59. })
  60. export let authReducerPath = 'auth';
  61. const authSlice = createSlice({
  62. name: authReducerPath,
  63. initialState: {},
  64. reducers: {
  65. logout(state) { //type - auth/logout
  66. // state.token = undefined
  67. history.push('/');
  68. return {}
  69. }
  70. },
  71. extraReducers: builder => {
  72. builder.addMatcher(loginApi.endpoints.login.matchFulfilled,
  73. (state, { payload }) => {
  74. const tokenPayload = jwtDecode(payload.login);
  75. if (tokenPayload) {
  76. state.token = payload.login;
  77. state.payload = tokenPayload;
  78. state.currentUser = { _id: tokenPayload.sub.id };
  79. history.push('/');
  80. }
  81. });
  82. builder.addMatcher(loginApi.endpoints.userFind.matchFulfilled,
  83. (state, { payload }) => {
  84. let retrievedUser = payload?.UserFindOne;
  85. if (retrievedUser?._id === state.currentUser?._id)
  86. state.currentUser = retrievedUser;
  87. })
  88. }
  89. })
  90. const actionAboutMe = () =>
  91. async (dispatch, getState) => {
  92. const auth = getState().auth
  93. if (auth.token) {
  94. dispatch(loginApi.endpoints.userFind.initiate(auth.currentUser._id))
  95. }
  96. }
  97. const { logout: actionAuthLogout } = authSlice.actions;
  98. let authApiReducer = loginApi.reducer;
  99. let authReducer = authSlice.reducer;
  100. let authApiReducerPath = loginApi.reducerPath;
  101. export const { useLoginMutation, useUserFindQuery } = loginApi;
  102. export { authApiReducer, authReducer, authApiReducerPath, actionAuthLogout, actionAboutMe }