2023-03-24 07:41:33 -04:00
|
|
|
import { Router, Response, Request, NextFunction } from 'express'
|
|
|
|
import { TwitchAPI } from '../util/scraping/extractor';
|
|
|
|
import ws, { WebSocket } from 'ws';
|
2023-03-27 08:33:51 -04:00
|
|
|
import { TwitchChatServer } from './chatIrcProxy'
|
2023-03-24 07:41:33 -04:00
|
|
|
|
|
|
|
const proxyRouter = Router();
|
|
|
|
const twitch = new TwitchAPI()
|
|
|
|
|
|
|
|
proxyRouter.get('/img/:base64Url', async (req: Request, res: Response, next: NextFunction) => {
|
|
|
|
const imageUrl = Buffer.from(req.params.base64Url, 'base64url').toString('utf-8')
|
|
|
|
if(!imageUrl) return;
|
|
|
|
|
|
|
|
const imageRes = await fetch(imageUrl)
|
|
|
|
.catch(next)
|
|
|
|
|
|
|
|
if(!imageRes) return;
|
|
|
|
|
|
|
|
if(imageRes.status !== 200) {
|
|
|
|
res.status(imageRes.status).send()
|
|
|
|
}
|
|
|
|
|
|
|
|
const arrayBuffer = await imageRes.arrayBuffer()
|
|
|
|
const buf = Buffer.from(arrayBuffer)
|
|
|
|
|
|
|
|
res.send(buf)
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
proxyRouter.get('/stream/:username/hls.m3u8', async (req: Request, res: Response, next: NextFunction) => {
|
|
|
|
let m3u8Data = await twitch.getStream(req.params.username)
|
|
|
|
const urlRegex =/(\b(https?|ftp|file):\/\/[-A-Z0-9+&@#\/%?=~_|!:,.;]*[-A-Z0-9+&@#\/%=~_|])/ig;
|
|
|
|
const matches = m3u8Data.match(urlRegex)
|
|
|
|
if (!matches) return next(new Error('Error proxying HLS'));
|
|
|
|
|
|
|
|
for (let url of matches) {
|
|
|
|
const base64data = Buffer.from(url).toString('base64url')
|
2023-03-25 15:28:14 -04:00
|
|
|
m3u8Data = m3u8Data.replace(url, `${process.env.URL}/proxy/stream/sub/${base64data}`)
|
2023-03-24 07:41:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
res.setHeader('Content-type','application/vnd.apple.mpegurl')
|
|
|
|
res.send(m3u8Data)
|
|
|
|
})
|
|
|
|
|
2023-03-25 15:28:14 -04:00
|
|
|
proxyRouter.get('/stream/sub/:encodedUrl' , async (req: Request, res: Response, next: NextFunction) => {
|
2023-03-24 07:41:33 -04:00
|
|
|
const unencodedUrl = Buffer.from(req.params.encodedUrl, 'base64url').toString()
|
|
|
|
const m3u8Fetch = await fetch(unencodedUrl)
|
2023-03-25 15:28:14 -04:00
|
|
|
let m3u8Data = await m3u8Fetch.text()
|
|
|
|
const urlRegex =/(\b(https?|ftp|file):\/\/[-A-Z0-9+&@#\/%?=~_|!:,.;]*[-A-Z0-9+&@#\/%=~_|])/ig;
|
|
|
|
const matches = m3u8Data.match(urlRegex)
|
|
|
|
if(!matches) return;
|
|
|
|
|
|
|
|
for (let url of matches) {
|
|
|
|
const base64data = Buffer.from(url).toString('base64url')
|
|
|
|
m3u8Data = m3u8Data.replace(url, `${process.env.URL}/proxy/stream/segment/${base64data}`)
|
|
|
|
}
|
|
|
|
|
|
|
|
res.setHeader('Content-type','application/vnd.apple.mpegurl')
|
2023-03-24 07:41:33 -04:00
|
|
|
res.send(m3u8Data)
|
|
|
|
})
|
|
|
|
|
2023-03-25 15:28:14 -04:00
|
|
|
proxyRouter.get('/stream/segment/:encodedUrl', async (req: Request, res: Response) => {
|
|
|
|
const unencodedUrl = Buffer.from(req.params.encodedUrl, 'base64url').toString()
|
|
|
|
const data = await fetch(unencodedUrl)
|
|
|
|
const arrayBuffer = await data.arrayBuffer()
|
|
|
|
const buf = Buffer.from(arrayBuffer)
|
|
|
|
res.send(buf)
|
|
|
|
})
|
|
|
|
|
2023-03-24 07:41:33 -04:00
|
|
|
|
|
|
|
|
2023-03-27 08:33:51 -04:00
|
|
|
const twitchChatServer = new TwitchChatServer();
|
2023-03-24 07:41:33 -04:00
|
|
|
export const wsServer = new ws.Server({ noServer: true });
|
2023-03-27 08:33:51 -04:00
|
|
|
twitchChatServer.startWebSocketServer(wsServer);
|
2023-03-24 07:41:33 -04:00
|
|
|
|
|
|
|
export default proxyRouter
|