advplyr.audiobookshelf-app/Server.js

248 lines
6.3 KiB
JavaScript
Raw Normal View History

2021-09-01 20:07:11 -05:00
import { io } from 'socket.io-client'
import { Storage } from '@capacitor/storage'
2021-09-01 20:07:11 -05:00
import axios from 'axios'
import EventEmitter from 'events'
class Server extends EventEmitter {
constructor(store) {
super()
this.store = store
this.url = null
this.socket = null
this.user = null
this.connected = false
this.stream = null
this.isConnectingSocket = false
2021-09-01 20:07:11 -05:00
}
get token() {
return this.user ? this.user.token : null
}
getAxiosConfig() {
return { headers: { Authorization: `Bearer ${this.token}` } }
}
getServerUrl(url) {
if (!url) return null
try {
var urlObject = new URL(url)
return `${urlObject.protocol}//${urlObject.hostname}:${urlObject.port}`
} catch (error) {
console.error('Invalid URL', error)
return null
}
2021-09-01 20:07:11 -05:00
}
setUser(user) {
this.user = user
this.store.commit('user/setUser', user)
if (user) {
// this.store.commit('user/setSettings', user.settings)
Storage.set({ key: 'token', value: user.token })
2021-09-01 20:07:11 -05:00
} else {
Storage.remove({ key: 'token' })
2021-09-01 20:07:11 -05:00
}
}
setServerUrl(url) {
this.url = url
this.store.commit('setServerUrl', url)
if (url) {
Storage.set({ key: 'serverUrl', value: url })
} else {
Storage.remove({ key: 'serverUrl' })
}
2021-09-01 20:07:11 -05:00
}
async connect(url, token) {
if (this.connected) {
console.warn('[SOCKET] Connection already established for ' + this.url)
return true
}
if (!url) {
console.error('Invalid url to connect')
return false
}
2021-09-01 20:07:11 -05:00
var serverUrl = this.getServerUrl(url)
var res = await this.ping(serverUrl)
2021-09-01 20:07:11 -05:00
if (!res || !res.success) {
//this.setServerUrl(null)
2021-09-01 20:07:11 -05:00
return false
}
var authRes = await this.authorize(serverUrl, token)
if (!authRes || !authRes.user) {
return false
}
this.setServerUrl(serverUrl)
2021-09-01 20:07:11 -05:00
this.setUser(authRes.user)
this.connectSocket()
return true
}
async check(url) {
var serverUrl = this.getServerUrl(url)
if (!serverUrl) {
return false
}
2021-09-01 20:07:11 -05:00
var res = await this.ping(serverUrl)
if (!res || !res.success) {
return false
}
return serverUrl
}
async login(url, username, password) {
var serverUrl = this.getServerUrl(url)
var authUrl = serverUrl + '/login'
return axios.post(authUrl, { username, password }).then((res) => {
if (!res.data || !res.data.user) {
console.error(res.data.error)
return {
error: res.data.error || 'Unknown Error'
}
}
this.setServerUrl(serverUrl)
this.setUser(res.data.user)
this.connectSocket()
return {
user: res.data.user
}
}).catch(error => {
console.error('[Server] Server auth failed', error)
return {
error: 'Request Failed'
}
})
}
logout() {
this.setUser(null)
if (this.socket) {
this.socket.disconnect()
}
2021-09-01 20:07:11 -05:00
}
authorize(serverUrl, token) {
var authUrl = serverUrl + '/api/authorize'
return axios.post(authUrl, null, { headers: { Authorization: `Bearer ${token}` } }).then((res) => {
return res.data
}).catch(error => {
console.error('[Server] Server auth failed', error)
return false
})
}
ping(url) {
var pingUrl = url + '/ping'
console.log('[Server] Check server', pingUrl)
return axios.get(pingUrl, { timeout: 1000 }).then((res) => {
2021-09-01 20:07:11 -05:00
return res.data
}).catch(error => {
console.error('Server check failed', error)
return false
})
}
connectSocket() {
if (this.connected || this.socket) {
if (this.socket) console.error('[SOCKET] Socket already established', this.url)
else console.error('[SOCKET] Already connected to socket', this.url)
return
}
2021-09-01 20:07:11 -05:00
console.log('[SOCKET] Connect Socket', this.url)
const socketOptions = {
transports: ['websocket'],
upgrade: false,
// reconnectionAttempts: 3
}
this.socket = io(this.url, socketOptions)
2021-09-01 20:07:11 -05:00
this.socket.on('connect', () => {
console.log('[SOCKET] Socket Connected ' + this.socket.id)
2021-09-01 20:07:11 -05:00
// Authenticate socket with token
this.socket.emit('auth', this.token)
this.connected = true
this.emit('connected', true)
this.store.commit('setSocketConnected', true)
2021-09-01 20:07:11 -05:00
})
this.socket.on('disconnect', (reason) => {
console.log('[SOCKET] Socket Disconnected: ' + reason)
this.connected = false
this.emit('connected', false)
this.store.commit('setSocketConnected', false)
// this.socket.removeAllListeners()
// if (this.socket.io && this.socket.io.removeAllListeners) {
// console.log(`[SOCKET] Removing ALL IO listeners`)
// this.socket.io.removeAllListeners()
// }
2021-09-01 20:07:11 -05:00
})
this.socket.on('init', (data) => {
console.log('[SOCKET] Initial socket data received', data)
2021-09-01 20:07:11 -05:00
if (data.stream) {
this.stream = data.stream
this.store.commit('setStreamAudiobook', data.stream.audiobook)
this.emit('initialStream', data.stream)
}
})
this.socket.on('user_updated', (user) => {
if (this.user && user.id === this.user.id) {
this.setUser(user)
}
})
this.socket.on('current_user_audiobook_update', (payload) => {
this.emit('currentUserAudiobookUpdate', payload)
})
this.socket.on('show_error_toast', (payload) => {
this.emit('show_error_toast', payload)
})
this.socket.on('show_success_toast', (payload) => {
this.emit('show_success_toast', payload)
})
this.socket.onAny((evt, args) => {
console.log(`[SOCKET] ${this.socket.id}: ${evt} ${JSON.stringify(args)}`)
})
this.socket.on('connect_error', (err) => {
console.error('[SOCKET] connection failed', err)
this.emit('socketConnectionFailed', err)
})
this.socket.io.on("reconnect_attempt", (attempt) => {
console.log(`[SOCKET] Reconnect Attempt ${this.socket.id}: ${attempt}`)
})
this.socket.io.on("reconnect_error", (err) => {
console.log(`[SOCKET] Reconnect Error ${this.socket.id}: ${err}`)
})
this.socket.io.on("reconnect_failed", () => {
console.log(`[SOCKET] Reconnect Failed ${this.socket.id}`)
})
this.socket.io.on("reconnect", () => {
console.log(`[SOCKET] Reconnect Success ${this.socket.id}`)
})
2021-09-01 20:07:11 -05:00
}
}
export default Server