Форум программистов, компьютерный форум, киберфорум
JavaScript: ReactJS
Войти
Регистрация
Восстановить пароль
Карта форума Темы раздела Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 5.00/5: Рейтинг темы: голосов - 5, средняя оценка - 5.00
9 / 6 / 6
Регистрация: 26.12.2021
Сообщений: 95
1

Проблемы с состоянием аутентификации MobX

14.12.2023, 06:45. Показов 864. Ответов 3

Author24 — интернет-сервис помощи студентам
Доброго времени суток, создаю spa веб-приложение, для хранения всех состояний выбрал mobx, создал корневое хранилище (RootStore), создал хранилище с фичами для авторизации (AuthStore),подключил AuthStore в RootStore, написал кастомный хук для передачи контекста в компонент. Далее в AuthStore создал запрос на авторизацию, которая получает в ответе от сервера токен (сохраняется в localStorage) и состояние isAuth переводит в true, второй запрос на проверку токена, если ответ 200, то состояние isAuth переводится в true, иначе токен удаляется из localStorage и isAuth = false. Создал роут обёртку PrivateRoute, в него передал контекст RootStore и вытащил из него AuthStore, вызываю проверку токена и после проверки возвращаю Outlet или редирект на страницу авторизации (тернарник, который проверяет isAuth). В App.tsx обернул необходимые path в роутере в PrivateRoute, но при попытке зайти по path после получения токена получаю редирект на страницу авторизации. Когда проверяю состояние isAuth, в компоненте PrivateRoute он равен false, хотя параллельно в AuthStore состояние равно true. Хотя экспортировал AuthStore в RootStore в формате синглтона, а сам RootStore передавал через контекст.
Весь код оставляю ниже, надеюсь найдётся человек, который натолкнёт меня на хорошую мысль. Заранее спасибо)

AuthStore
Javascript
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
class AuthStore {
    private isAuth: boolean = false
    private authInProgress: boolean = false
 
    constructor(){
        makeAutoObservable(this)
    }
 
    private setAuth(status: boolean){
        runInAction(() => {
            this.isAuth = status
        })
    }
 
    public getAuth(){
        return this.isAuth
    }
 
    private setProgress(status: boolean){
        runInAction(() => {
            this.authInProgress = status
        })
    }
 
    public getProgress(){
        return this.authInProgress
    }
 
    public async login(username: string, password: string){
        this.setProgress(true)
        try{
           await AuthService.login(username, password).then(response => {
                if(response.status === 202){
                    localStorage.setItem('token', response.headers.authorization)
                    this.setAuth(true)               
                    window.location.replace('/')
                }
            })
        }
        catch(e){
            const error = e as AxiosError
            console.log(error.message)
        }
        finally{
            this.setProgress(false)
        }
    }
 
    public async checkIsAuth(token: string | null){
        this.setProgress(true)
        if (token === null){
            window.location.replace('/auth')
        } else {
            try{
                await AuthService.checkAuth(token).then(response => {
                    if (response.status === 200){
                        this.setAuth(true)             
                    }
                    else {
                        this.setAuth(false)
                    }
                })
            }
            catch(e){
                const error = e as AxiosError
                if(error.response?.status === 401){
                    localStorage.removeItem('token')
                    window.location.replace('/auth')
                    this.setAuth(false)
                }
                else {
                    this.setAuth(false)
                }
            }
            finally{
                this.setProgress(false)
            }
    }
        
    }
 
}
 
export default new AuthStore()
RootStore
Javascript
1
2
3
export default class RootStore {
    authStore = AuthStore
}
RootStoreContext
Javascript
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
import { createContext, useContext } from "react";
import RootStore from "./rootStore";
 
 
export const RootStoreContext = createContext<RootStore | null>(null)
 
export const useStore = () => {
    const context = useContext(RootStoreContext)
    if (context === null) {
        throw new Error(
          "Не указан контекст возвращаемый из корневого хранилища"
        );
    }
    return context
}
Index
Javascript
1
2
3
4
5
ReactDOM.createRoot(document.getElementById('root')!).render(
    <RootStoreContext.Provider value={new RootStore()}>
        <App/>
    </RootStoreContext.Provider>
)
App
Javascript
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
const App: FunctionComponent = observer(() => {
    return(<>
    
        <BrowserRouter>
            <Routes>
            <Route path='/auth' element={<AuthPage/>}/>
            <Route path='/' element={<PrivateRouter/>}>
                <Route index element={<MainPage/>}/>
                <Route path='news' element={<NewsPage/>}/>
                <Route path='request' element={<UserRequests/>}/>
                <Route path='request/:id' element={<RequestByID/>}/>
                <Route path='deprequest' element={<DepartamentRequests/>}/>
            </Route>
            <Route path='/notimplemented' element={<NotImplemented/>}/>
            <Route path='/*' element={<NotFound404/>}/>
            </Routes>
        </BrowserRouter>
    </>)
})
 
export default App
PrivateRouter
Javascript
1
2
3
4
5
6
7
8
9
10
const PrivateRoute: FunctionComponent = observer(() => {
    const {authStore} = useStore()
 
    console.log(`Статус авторизации: ${authStore.getAuth()}`);
 
    return authStore.getAuth()  ? <Outlet/> : <Navigate to='/auth'/>
    
});
    
export default PrivateRoute;
Если нужно что-то ещё, то дополню топ
0
Лучшие ответы (1)
Programming
Эксперт
94731 / 64177 / 26122
Регистрация: 12.04.2006
Сообщений: 116,782
14.12.2023, 06:45
Ответы с готовыми решениями:

Тип аутентификации для веб сервиса отличный от типа аутентификации ресурса
Всем привет! Есть IIS 8. На нем опубликована база 1С, для нее требуется доменная...

Mobx strict mode
Добрый день, объясните, пожалуйста, почему выходит предупреждение Since strict-mode is enabled,...

Mobx не загружается в проект
Доброго времени суток. Начал  изучение mobx, пытаюсь установить п проект npm i mobx-react-lite...

Как организовать mobx store?
Есть код, где в store в state хранится folder_id, employeesList и currentEmploySelect. То есть...

3
603 / 403 / 212
Регистрация: 30.04.2017
Сообщений: 743
17.12.2023, 15:38 2
Лучший ответ Сообщение было отмечено yegres_ как решение

Решение

Цитата Сообщение от yegres_ Посмотреть сообщение
window.location.replace('/')
я думаю проблема связана с тем что из-за таких вызовов все приложение по новой рендерится и состояние в контексте соответственно сбрасывается

стоит попробовать заменить код
1) в месте вызова login в функциональном компоненте вытащить
Javascript
1
2
3
const navigate = useNavigate()
...
navigate('/your-path')
2) можно вытащить history/router из react-router и прокинуть его в AuthStore
для старых версий в react-router можно было инициализировать глобально и прокидывать объект history из react-history
и затем в нужных местах делать history.push('/xxx')

сейчас в новых версиях (6 версия точно) можно делать как то так
Javascript
1
2
3
4
5
export const router = createBrowserRouter([
// ...
])
...
router.navigate("/xxx")
BrowserRouter прийдется поменять на <RouterProvider router={router} />
если лень переписывать Routes на элементы масива, то можно обойтись утилитой createRoutesFromElements
https://reactrouter.com/en/mai... m-elements
1
9 / 6 / 6
Регистрация: 26.12.2021
Сообщений: 95
17.12.2023, 21:00  [ТС] 3
Цитата Сообщение от Ovederax Посмотреть сообщение
2) можно вытащить history/router из react-router и прокинуть его в AuthStore
для старых версий в react-router можно было инициализировать глобально и прокидывать объект history из react-history
и затем в нужных местах делать history.push('/xxx')
Вот, я пытался сделать так, но компилятор ругался на то, что history.push можно использовать только в компонентах

Цитата Сообщение от Ovederax Посмотреть сообщение
BrowserRouter прийдется поменять на <RouterProvider router={router} />
У меня изначально было в виде массива, но я не разобрался как обернуть роуты в PrivateRoute и решил переделать на более легаси вариант)

Полезные советы, попробую использовать ваши предложенные решения и отпишу о результате
0
9 / 6 / 6
Регистрация: 26.12.2021
Сообщений: 95
18.12.2023, 06:02  [ТС] 4
Цитата Сообщение от Ovederax Посмотреть сообщение
сейчас в новых версиях (6 версия точно) можно делать как то так
Спасибо вам, действительно был перерендер и сброс состояний, использовал третий вариант с экспортирование роутра и вызовом navigate, всё работает как часы, даже сильно ничего не пришлось изменять!
0
18.12.2023, 06:02
IT_Exp
Эксперт
87844 / 49110 / 22898
Регистрация: 17.06.2006
Сообщений: 92,604
18.12.2023, 06:02
Помогаю со студенческими работами здесь

Не работает изменение состояния в MobX
Почему не работает изменения состояние в MobX? Код ниже: инициализирую объект, который будет...

Вычисляемое свойство MobX пытается обновить само себя
Всем привет. Имеется хранилище MobX (класс AppModalsStore) с computed-свойством...

WCF: Запрос HTTP не разрешен для схемы аутентификации клиента "Ntlm". От сервера получен заголовок аутентификации "NTLM ."
Добрый день! Борюсь с WCF. Цель- подключиться и работать с вебсервисом от Битрикса....

Циклы с состоянием в ФП
Внезапно возник сложный вопрос. Вступление. Известно, что циклы видаb = ; for(var i=0;...

Управление состоянием объекта
Например ситуация: при столкновении объектов один меняет цвет и через 10 секунд меняет на другой...

Управление состоянием объекта
Например ситуация: при столкновении объектов один меняет цвет и через 10 секунд меняет на другой...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
4
Ответ Создать тему
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2024, CyberForum.ru