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

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

14.12.2023, 06:45. Показов 916. Ответов 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
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
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
Сообщений: 96
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
Сообщений: 96
18.12.2023, 06:02  [ТС] 4
Цитата Сообщение от Ovederax Посмотреть сообщение
сейчас в новых версиях (6 версия точно) можно делать как то так
Спасибо вам, действительно был перерендер и сброс состояний, использовал третий вариант с экспортирование роутра и вызовом navigate, всё работает как часы, даже сильно ничего не пришлось изменять!
0
18.12.2023, 06:02
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
18.12.2023, 06:02
Помогаю со студенческими работами здесь

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

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

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

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

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

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


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

Или воспользуйтесь поиском по форуму:
4
Ответ Создать тему
Новые блоги и статьи
Книги и учебные ресурсы по C#
InfoMaster 08.01.2025
Базовые учебники и руководства Одной из лучших книг для начинающих является "C# 10 и . NET 6 для начинающих" Эндрю Троелсена и Филиппа Джепикса . Книга последовательно раскрывает основные концепции. . .
Что такое NullReferenceEx­­­ception и как исправить?
InfoMaster 08.01.2025
NullReferenceException - одно из самых распространенных исключений, с которым сталкиваются разработчики на C#. Это исключение возникает при попытке обратиться к членам объекта (методам, свойствам или. . .
Что такое Null Pointer Exception (NPE) и как это исправить?
InfoMaster 08.01.2025
Null Pointer Exception (NPE) - это одно из самых распространенных исключений в Java, которое возникает при попытке использовать ссылку на объект, значение которой равно null. Это исключение относится. . .
Русский язык в консоли C++
InfoMaster 08.01.2025
При разработке программ на C++ одной из частых проблем, с которой сталкиваются русскоязычные программисты, является корректное отображение кириллицы в консольных приложениях. Эта проблема особенно. . .
Telegram бот на C#
InfoMaster 08.01.2025
Разработка ботов для Telegram стала неотъемлемой частью современной экосистемы мессенджеров. C# предоставляет мощный и удобный инструментарий для создания разнообразных ботов, от простых. . .
Использование GraphQL в Go (Golang)
InfoMaster 08.01.2025
Go (Golang) является одним из наиболее популярных языков программирования, используемых для создания высокопроизводительных серверных приложений. Его архитектурные особенности и встроенные. . .
Что лучше использовать при создании класса в Java: сеттеры или конструктор?
Alexander-7 08.01.2025
Вопрос подробнее: На вопрос: «Когда одновременно создаются конструктор и сеттеры в классе – это нормально?» куратор уточнил: «Ваш класс может вообще не иметь сеттеров, а только конструктор и геттеры. . .
Как работать с GraphQL на TypeScript
InfoMaster 08.01.2025
Введение в GraphQL и TypeScript В современной разработке веб-приложений GraphQL стал мощным инструментом для создания гибких и эффективных API. В сочетании с TypeScript, эта технология. . .
Счётчик на базе сумматоров + регистров и генератора сигналов согласования.
Hrethgir 07.01.2025
Создан с целью проверки скорости асинхронной логики: ранее описанного сумматора и предополагаемых fast регистров. Регистры созданы на базе ранее описанного, предполагаемого fast триггера. То-есть. . .
Как перейти с Options API на Composition API в Vue.js
BasicMan 06.01.2025
Почему переход на Composition API актуален В мире современной веб-разработки фреймворк Vue. js продолжает эволюционировать, предлагая разработчикам все более совершенные инструменты для создания. . .
Архитектура современных процессоров
inter-admin 06.01.2025
Процессор (центральный процессор, ЦП) является основным вычислительным устройством компьютера, которое выполняет обработку данных и управляет работой всех остальных компонентов системы. Архитектура. . .
История создания реляционной модели баз данных, правила Кодда
Programming 06.01.2025
Предпосылки создания реляционной модели В конце 1960-х годов компьютерная индустрия столкнулась с серьезными проблемами в области управления данными. Существовавшие на тот момент модели данных -. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru