-
Notifications
You must be signed in to change notification settings - Fork 105
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
8 changed files
with
178 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
export default oauth.discordEventHandler({ | ||
async onSuccess(event, { user }) { | ||
await setUserSession(event, { | ||
user: { | ||
discord: user, | ||
}, | ||
loggedInAt: Date.now() | ||
}) | ||
|
||
return sendRedirect(event, '/') | ||
} | ||
}) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,141 @@ | ||
import type { H3Event, H3Error } from 'h3' | ||
import { eventHandler, createError, getQuery, getRequestURL, sendRedirect } from 'h3' | ||
import { withQuery, parseURL, stringifyParsedURL } from 'ufo' | ||
import { ofetch } from 'ofetch' | ||
import { defu } from 'defu' | ||
import { useRuntimeConfig } from '#imports' | ||
|
||
export interface OAuthDiscordConfig { | ||
/** | ||
* Discord OAuth Client ID | ||
* @default process.env.NUXT_OAUTH_DISCORD_CLIENT_ID | ||
*/ | ||
clientId?: string | ||
/** | ||
* Discord OAuth Client Secret | ||
* @default process.env.NUXT_OAUTH_DISCORD_CLIENT_SECRET | ||
*/ | ||
clientSecret?: string | ||
/** | ||
* Discord OAuth Scope | ||
* @default [] | ||
* @see https://discord.com/developers/docs/topics/oauth2#shared-resources-oauth2-scopes | ||
* @example ['identify', 'email'] | ||
* Without the identify scope the user will not be returned. | ||
*/ | ||
scope?: string[] | ||
/** | ||
* Require email from user, adds the ['email'] scope if not present. | ||
* @default false | ||
*/ | ||
emailRequired?: boolean, | ||
/** | ||
* Require profile from user, adds the ['identify'] scope if not present. | ||
* @default true | ||
*/ | ||
profileRequired?: boolean | ||
/** | ||
* Discord OAuth Authorization URL | ||
* @default 'https://discord.com/oauth2/authorize' | ||
*/ | ||
authorizationURL?: string | ||
/** | ||
* Discord OAuth Token URL | ||
* @default 'https://discord.com/api/oauth2/token' | ||
*/ | ||
tokenURL?: string | ||
} | ||
|
||
interface OAuthConfig { | ||
config?: OAuthDiscordConfig | ||
onSuccess: (event: H3Event, result: { user: any, tokens: any }) => Promise<void> | void | ||
onError?: (event: H3Event, error: H3Error) => Promise<void> | void | ||
} | ||
|
||
export function discordEventHandler({ config, onSuccess, onError }: OAuthConfig) { | ||
return eventHandler(async (event: H3Event) => { | ||
// @ts-ignore | ||
config = defu(config, useRuntimeConfig(event).oauth?.discord, { | ||
authorizationURL: 'https://discord.com/oauth2/authorize', | ||
tokenURL: 'https://discord.com/api/oauth2/token', | ||
profileRequired: true | ||
}) as OAuthDiscordConfig | ||
const { code } = getQuery(event) | ||
|
||
if (!config.clientId || !config.clientSecret) { | ||
const error = createError({ | ||
statusCode: 500, | ||
message: 'Missing NUXT_OAUTH_DISCORD_CLIENT_ID or NUXT_OAUTH_DISCORD_CLIENT_SECRET env variables.' | ||
}) | ||
if (!onError) throw error | ||
return onError(event, error) | ||
} | ||
|
||
const redirectUrl = getRequestURL(event).href | ||
if (!code) { | ||
config.scope = config.scope || [] | ||
if (config.emailRequired && !config.scope.includes('email')) { | ||
config.scope.push('email') | ||
} | ||
if (config.profileRequired && !config.scope.includes('identify')) { | ||
config.scope.push('identify') | ||
} | ||
|
||
// Redirect to Discord Oauth page | ||
return sendRedirect( | ||
event, | ||
withQuery(config.authorizationURL as string, { | ||
response_type: 'code', | ||
client_id: config.clientId, | ||
redirect_uri: redirectUrl, | ||
scope: config.scope.join(' ') | ||
}) | ||
) | ||
} | ||
|
||
const parsedRedirectUrl = parseURL(redirectUrl) | ||
parsedRedirectUrl.search = '' | ||
const tokens: any = await ofetch( | ||
config.tokenURL as string, | ||
{ | ||
method: 'POST', | ||
headers: { | ||
'Content-Type': 'application/x-www-form-urlencoded' | ||
}, | ||
body: new URLSearchParams({ | ||
client_id: config.clientId, | ||
client_secret: config.clientSecret, | ||
grant_type: 'authorization_code', | ||
redirect_uri: stringifyParsedURL(parsedRedirectUrl), | ||
code: code as string, | ||
}).toString() | ||
} | ||
).catch(error => { | ||
return { error } | ||
}) | ||
if (tokens.error) { | ||
console.log(tokens) | ||
const error = createError({ | ||
statusCode: 401, | ||
message: `Discord login failed: ${tokens.error?.data?.error_description || 'Unknown error'}`, | ||
data: tokens | ||
}) | ||
|
||
if (!onError) throw error | ||
return onError(event, error) | ||
} | ||
|
||
const accessToken = tokens.access_token | ||
const user: any = await ofetch('https://discord.com/api/users/@me', { | ||
headers: { | ||
'user-agent': 'Nuxt Auth Utils', | ||
Authorization: `Bearer ${accessToken}` | ||
} | ||
}) | ||
|
||
return onSuccess(event, { | ||
tokens, | ||
user | ||
}) | ||
}) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters