2022-08-31 20:44:33 +01:00
|
|
|
import type { AxiosInstance } from 'axios'
|
|
|
|
import axios from 'axios'
|
|
|
|
import type { Socket } from 'socket.io-client'
|
|
|
|
import { io } from 'socket.io-client'
|
2023-04-03 00:11:19 +02:00
|
|
|
import { isUnauthorizedError } from '@thream/socketio-jwt'
|
2021-10-24 06:09:43 +02:00
|
|
|
|
2021-10-26 16:38:55 +02:00
|
|
|
import { API_URL } from '../api'
|
|
|
|
import { cookies } from '../cookies'
|
2022-08-31 20:44:33 +01:00
|
|
|
import type { Tokens } from '.'
|
2021-10-24 06:09:43 +02:00
|
|
|
import { fetchRefreshToken } from './authenticationFromServerSide'
|
2022-08-30 18:42:21 +02:00
|
|
|
import { clearCache } from '../cache'
|
2021-10-24 06:09:43 +02:00
|
|
|
|
|
|
|
export class Authentication {
|
|
|
|
public tokens: Tokens
|
|
|
|
public accessTokenAge: number
|
2022-08-24 17:22:55 +02:00
|
|
|
public socket?: Socket
|
2021-10-24 06:09:43 +02:00
|
|
|
public api: AxiosInstance
|
|
|
|
|
2022-08-24 17:22:55 +02:00
|
|
|
constructor(tokens: Tokens, disableSocketIO: boolean = false) {
|
2021-10-24 06:09:43 +02:00
|
|
|
this.tokens = tokens
|
|
|
|
this.accessTokenAge = Date.now()
|
2022-08-30 21:30:06 +02:00
|
|
|
if (typeof window === 'undefined' || disableSocketIO) {
|
2022-08-24 17:22:55 +02:00
|
|
|
this.socket = undefined
|
|
|
|
} else {
|
|
|
|
this.socket = io(API_URL, {
|
|
|
|
auth: { token: `Bearer ${tokens.accessToken}` }
|
|
|
|
})
|
|
|
|
this.socket.on('connect', () => {
|
|
|
|
console.log(
|
|
|
|
`Connected to socket with clientId: ${this.socket?.id ?? 'undefined'}`
|
|
|
|
)
|
|
|
|
})
|
|
|
|
this.socket.on('connect_error', (error) => {
|
2023-04-03 00:11:19 +02:00
|
|
|
if (isUnauthorizedError(error)) {
|
2022-08-24 17:22:55 +02:00
|
|
|
fetchRefreshToken(this.tokens.refreshToken)
|
|
|
|
.then(({ accessToken }) => {
|
|
|
|
this.setAccessToken(accessToken)
|
|
|
|
})
|
|
|
|
.catch(async () => {
|
|
|
|
this.signout()
|
|
|
|
return await Promise.reject(error)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2021-10-24 06:09:43 +02:00
|
|
|
this.api = axios.create({
|
|
|
|
baseURL: API_URL,
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/json'
|
|
|
|
}
|
|
|
|
})
|
|
|
|
this.api.interceptors.request.use(
|
|
|
|
async (config) => {
|
|
|
|
const isValidAccessToken =
|
|
|
|
this.accessTokenAge + tokens.expiresIn > Date.now()
|
|
|
|
if (!isValidAccessToken) {
|
|
|
|
const { accessToken } = await fetchRefreshToken(
|
|
|
|
this.tokens.refreshToken
|
|
|
|
)
|
|
|
|
this.setAccessToken(accessToken)
|
|
|
|
}
|
2023-04-03 00:11:19 +02:00
|
|
|
config.headers.Authorization = `${this.tokens.type} ${this.tokens.accessToken}`
|
2021-10-24 06:09:43 +02:00
|
|
|
return config
|
|
|
|
},
|
|
|
|
async (error) => {
|
|
|
|
this.signout()
|
|
|
|
return await Promise.reject(error)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
this.api.interceptors.response.use(
|
|
|
|
(response) => {
|
|
|
|
return response
|
|
|
|
},
|
|
|
|
async (error) => {
|
|
|
|
if (error.response.status !== 403 || (error.config._retry as boolean)) {
|
|
|
|
return await Promise.reject(error)
|
|
|
|
}
|
|
|
|
error.config._retry = true
|
|
|
|
try {
|
|
|
|
const { accessToken } = await fetchRefreshToken(
|
|
|
|
this.tokens.refreshToken
|
|
|
|
)
|
|
|
|
this.setAccessToken(accessToken)
|
|
|
|
error.response.config.headers.Authorization = `${this.tokens.type} ${this.tokens.accessToken}`
|
|
|
|
return await this.api.request(error.response.config)
|
|
|
|
} catch {
|
|
|
|
this.signout()
|
|
|
|
return await Promise.reject(error)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
public setAccessToken(accessToken: string): void {
|
|
|
|
this.tokens.accessToken = accessToken
|
|
|
|
this.accessTokenAge = Date.now()
|
|
|
|
const token = `${this.tokens.type} ${this.tokens.accessToken}`
|
2022-08-24 17:22:55 +02:00
|
|
|
if (typeof this?.socket?.auth !== 'function' && this.socket != null) {
|
2023-01-11 17:39:09 +01:00
|
|
|
this.socket.auth['token'] = token
|
2021-10-24 06:09:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public signout(): void {
|
|
|
|
cookies.remove('refreshToken')
|
2022-08-30 18:42:21 +02:00
|
|
|
clearCache()
|
2021-10-24 06:09:43 +02:00
|
|
|
window.location.href = '/authentication/signin'
|
|
|
|
}
|
|
|
|
|
|
|
|
public async signoutServerSide(): Promise<void> {
|
2022-08-29 19:58:50 +02:00
|
|
|
try {
|
|
|
|
await this.api.post('/users/signout', {
|
|
|
|
refreshToken: this.tokens.refreshToken
|
|
|
|
})
|
|
|
|
} catch {}
|
2021-10-24 06:09:43 +02:00
|
|
|
this.signout()
|
|
|
|
}
|
|
|
|
|
2022-08-29 20:15:47 +02:00
|
|
|
public async signoutAllDevicesServerSide(): Promise<void> {
|
|
|
|
try {
|
|
|
|
await this.api.delete('/users/signout')
|
|
|
|
} catch {}
|
|
|
|
this.signout()
|
|
|
|
}
|
|
|
|
|
2021-10-24 06:09:43 +02:00
|
|
|
public signin(): void {
|
2022-08-30 18:42:21 +02:00
|
|
|
clearCache()
|
2021-10-24 06:09:43 +02:00
|
|
|
cookies.set('refreshToken', this.tokens.refreshToken)
|
|
|
|
}
|
|
|
|
}
|