This repository contains a custom OAuth2 strategy for authenticating with Discord using Passport.js. It facilitates user authentication via Discord and enables the retrieval of user data, including profile information, guilds, and connections.
To use this strategy, first install express, passport and then the discord-strategy:
npm install express passport discord-strategyIntegrate the strategy into your Express application as follows:
const express = require("express");
const passport = require("passport");
const { Strategy, DiscordScope } = require("discord-strategy");
const app = express();
// Define options for the Strategy
const options = {
clientID: "YOUR_CLIENT_ID",
clientSecret: "YOUR_CLIENT_SECRET",
callbackURL: "http://localhost:3000/auth/discord/callback",
scope: [
DiscordScope.Identify,
DiscordScope.Email,
DiscordScope.Guilds,
DiscordScope.Connections,
],
};
passport.use(new Strategy(options, verify));
/**
* Verify function for authentication
* @param {string} accessToken - The access token received from OAuth2
* @param {string} refreshToken - The refresh token to get a new access token
* @param {DiscordProfile} profile - The user information from the Discord API
* @param {VerifyCallback} done - The function to call after verification of the user.
* @param {ConsumableAPI} consume - Includes multiple abstractions for interacting with Discord's API.
* @returns {Promise<void>}
*
*
* @typedef {import('discord-strategy').VerifyCallback} VerifyCallback
* @typedef {import('discord-strategy').DiscordProfile} DiscordProfile
* @typedef {import('discord-strategy').ConsumableAPI} ConsumableAPI
*/
async function verify(accessToken, refreshToken, profile, done, consume) {
try {
// Fetch connections and guilds concurrently
await Promise.all([consume.connections(), consume.guilds()]);
console.log("Authentication successful!");
done(null, profile);
} catch (error) {
done(error?.data || error, null);
}
}
app.use(passport.initialize());
app.get("/auth/discord", passport.authenticate("discord"));
app.get(
"/auth/discord/callback",
passport.authenticate("discord", { session: false }),
(req, res) => {
res.send(`
<h1>Authentication successful!</h1>
<h2>User Profile:</h2>
<pre>${JSON.stringify(req.user, null, 2)}</pre>
`);
},
);
app.listen(3000, () => {
console.log("Login via http://localhost:3000/auth/discord");
});For scenarios where only basic user information is needed:
function verify(accessToken, refreshToken, profile, done, consume) {
console.log("Fetched", profile);
return done(null, profile);
}clientID: Your Discord application's Client ID.clientSecret: Your Discord application's Client Secret.callbackURL: The URL to which Discord will redirect after authorization.scope: An array of scopes specifying the level of access (default:[DiscordScope.Identify, DiscordScope.Email]).
List of Consumable Functions
guilds(callback?): Fetches the user's connections. Requires theconnectionsscope.
function verify(accessToken, refreshToken, profile, done, consume) {
try {
await consume.guilds();
console.log(profile.guild);
done(null, profile);
} catch (err) {
done(err, null);
}connections(callback?): Fetches the connections of the user. Requires theconnectionsscope.
await consume.connections();
console.log(profile.connections);guildJoin(botToken: string, serverId: string, nickname: string, roles: string[], callback): join the specified guild.
await consume.guildJoiner(
"botToken",
"serverId",
undefined,
undefined,
(err, result) =>
!err && !result ? console.log("Joined") : console.log(err, result)
);member(guild_id: string): Returns a guild member object for the current user and creates a member property inside the profile. Within the member property, there is a guild_id. If profile.member.guild_id is null, the user is not in that guild. This requires the guilds.members.read OAuth2 scope.
await consume.member("id");
profile = consume.profile();
done(null, profile);resolver(key, api): Fetches data from a specified API endpoint and stores it under the given key in the profile.
await consume.resolver("guilds", "users/@me/guilds");
profile = consume.profile();
done(null, profile);consume.profile(): Returns the user profile.
done(null, consume.profile());-
consume.linkedRole.get(): Returns the application role connection for the user. Requires anrole_connections.writescope. -
consume.linkedRole.set(platform_name?, platform_username?, metadata, done?): Updates and returns the application role connection for the user. Requires anrole_connections.writescope.
// Role register Example
// fetch(
// `https://discord.com/api/v10/applications/APPLICATION_ID_HERE/role-connections/metadata`,
// {
// method: "PUT",
// body: JSON.stringify([
// {
// key: "cool",
// name: "Cool ppl",
// description: "You are cool ppl",
// type: 7, //type 7: BOOLEAN_EQUAL
// },
// ]),
// headers: {
// "Content-Type": "application/json",
// Authorization: `Bot BOT_TOKEN`,
// },
// }
// )
// .then((r) => r.json())
// .then(console.log);
//
// After registration do not forget to place url in designated field.
// Bot setting -> General Information -> Linked Roles Verification URL
await consume.set(undefined, undefined, {
//key: value,
cool: 1, //for type 7, value 1 represent true
});
done(null, profile);async function verify(accessToken, refreshToken, profile, done, consume) {
try {
await Promise.all([consume.connections(), consume.guilds()]);
profile = consume.profile();
console.log("Authentication successful!", profile);
done(null, profile);
} catch (err) {
done(err, null);
}
}Callback based Data Fetching (Not Recommended // extreme-slow)
async function verify(accessToken, refreshToken, profile, done, consume) {
consume.connections((err) => {
if (err) return done(err, false);
consume.guilds((err) => {
if (err) return done(err, false);
console.log("Authentication successful!", profile);
done(null, profile);
});
});
}async function verify(accessToken, refreshToken, profile, done, consume) {
try {
await consume.resolver("guilds", "users/@me/guilds");
done(null, profile);
} catch (err) {
done(err, null);
}
}If you need to store the refreshToken, manage sessions, or handle other processes unrelated to Discord OAuth, Refer to the Passport.js documentation for more information on managing these tasks or explore other strategies that might be necessary for additional handling.
- Scopes has been turned into enums.
- Better LSP hints.
- Added
consume.linkedRole.get&consume.linkedRole.get. https://discord.com/developers/docs/resources/user#get-current-user-guild-member & https://discord.com/developers/docs/resources/user#get-current-user-application-role-connection
- Added
consume.member("guild_id"),Returns a guild member object for the current user. https://discord.com/developers/docs/resources/user#get-current-user-guild-member
- Fixed typo and doc error
- Introduced the
consumeparameter to encapsulate utility functions. - Removed the clean function, as the profile object is no longer need to be sanitize.
- No longer required to pass the access token to the consumable functions.
- Added two new consumable functions:
complexResolver()andguildJoiner().
- Bound the cleaner function to the
cleanproperty of the profile (profile.clean()).