Polarcraft/webv2/server/api/auth.ts

94 lines
2.4 KiB
TypeScript
Raw Normal View History

2023-06-05 10:17:35 +02:00
import * as jwt from 'jsonwebtoken'
type AccessTokenResponse = {
access_token: string,
token_type: string,
expires_in: number,
refresh_token: string,
scope: string
}
type DiscordUser = {
id: string,
username: string,
discriminator: string,
avatar: string,
bot?: boolean,
system?: boolean,
mfa_enabled?: boolean,
banner?: string,
accent_color?: number,
locale?: string,
verified?: boolean,
email?: string,
flags?: number,
premium_type?: number,
public_flags?: number
}
2023-06-01 22:06:11 +02:00
export default defineEventHandler(async (event) => {
const { code }: { code?: string } = getQuery(event)
if (!code) {
return sendRedirect(event, '/', 302)
2023-06-01 22:06:11 +02:00
}
let tokenResponse: AccessTokenResponse
let authorizationResponse: DiscordUser
try {
tokenResponse = await $fetch('https://discord.com/api/oauth2/token', {
method: 'POST',
body: new URLSearchParams({
client_id: config.discordClientId,
client_secret: config.discordClientSecret,
code,
grant_type: 'authorization_code',
redirect_uri: config.oauthRedirectUri,
scope: 'identify'
}),
headers: {
'Content-Type': 'application/x-www-form-urlencoded'
}
})
authorizationResponse = await $fetch('https://discord.com/api/users/@me', {
headers: {
authorization: `Bearer ${tokenResponse.access_token}`
}
})
} catch (e) {
console.error('Discord authentication failed', e)
throw createError({ statusCode: 500, statusMessage: 'Discord authentication failed' })
}
try {
await UserModel.updateOne({ 'discord.id': authorizationResponse.id }, {
$set: {
discord: {
id: authorizationResponse.id,
username: authorizationResponse.username,
avatarHash: authorizationResponse.avatar
}
},
$setOnInsert: {
username: authorizationResponse.username
}
}, { upsert: true })
const token = jwt.sign({
accessToken: tokenResponse.access_token,
refreshToken: tokenResponse.refresh_token,
discordId: authorizationResponse.id
}, config.jwtSecret, { expiresIn: tokenResponse.expires_in })
setCookie(event, 'jwt', token, { httpOnly: true, maxAge: tokenResponse.expires_in * 1000 })
} catch (e) {
console.error('Updating user information failed', e)
throw createError({ statusCode: 500, statusMessage: 'Updating user information failed' })
}
return sendRedirect(event, '/', 302)
2023-06-01 22:06:11 +02:00
})