Skip to content

Instantly share code, notes, and snippets.

@OmkarK45
Last active April 3, 2022 10:16
Show Gist options
  • Save OmkarK45/713b0e620c1475d7342c66eac4853c76 to your computer and use it in GitHub Desktop.
Save OmkarK45/713b0e620c1475d7342c66eac4853c76 to your computer and use it in GitHub Desktop.
Next Auth with custom backend that returns JWT
import NextAuth from "next-auth";
import CredentialsProvider from "next-auth/providers/credentials";
type TA_Token = {
token: string;
user_id: number;
};
export default NextAuth({
providers: [
CredentialsProvider({
// The name to display on the sign in form (e.g. 'Sign in with...')
name: "Tube Archivist",
// The credentials is used to generate a suitable form on the sign in page.
// You can specify whatever fields you are expecting to be submitted.
// e.g. domain, username, password, 2FA token, etc.
// You can pass any HTML attribute to the <input> tag through the object.
credentials: {
username: {
label: "username",
type: "text",
placeholder: "[email protected]",
},
password: { label: "Password", type: "password" },
},
async authorize(credentials, req) {
const payload = {
username: credentials.username,
password: credentials.password,
};
const res = await fetch("http://localhost:8000/api/login/", {
method: "POST",
body: JSON.stringify(payload),
headers: {
"Content-Type": "application/json",
"Accept-Language": "en-US",
},
});
const ta_token: TA_Token = await res.json();
// If no error and we have user data, return it
if (res.ok && ta_token) {
// debug
return {
name: "TA User",
email: "[email protected]",
image: "test",
ta_token,
};
}
// Return null if user data could not be retrieved
return null;
},
}),
// ...add more providers here
],
session: {
maxAge: 30 * 24 * 60 * 60,
strategy: "jwt",
},
// customer sign in page
// pages: {
// signIn: "/login",
// },
callbacks: {
async session({ session, token, user }) {
session.ta_token = token.ta_token;
return session;
},
async jwt({ token, user }) {
if (user?.ta_token) {
// @ts-ignore
token.ta_token = user?.ta_token;
}
return token;
},
},
theme: {
colorScheme: "auto", // "auto" | "dark" | "light"
brandColor: "", // Hex color code #33FF5D
logo: "/img/banner-tube-archivist-dark.png", // Absolute URL to image
},
// Enable debug messages in the console if you are having problems
debug: process.env.NODE_ENV === "development",
});
import type { AppProps } from "next/app";
import { SessionProvider } from "next-auth/react";
import "../styles/globals.css";
import "../styles/dark.css";
function MyApp({ Component, pageProps: { session, ...pageProps } }: AppProps) {
return (
<SessionProvider session={session}>
<Component {...pageProps} />
</SessionProvider>
);
}
export default MyApp;
import type { NextPage } from "next";
import { signIn, signOut, useSession } from "next-auth/react";
import dynamic from "next/dynamic";
import Head from "next/head";
import { Suspense } from "react";
const DynamicHeader = dynamic(() => import("../components/Header"), {
suspense: true,
});
const SignInOutButton = ({ isSignedIn }: { isSignedIn: boolean }) => {
if (isSignedIn) {
return <button onClick={() => signOut()}>Sign Out</button>;
}
return <button onClick={() => signIn()}>Sign in</button>;
};
const Home: NextPage = () => {
const { data: session, status } = useSession();
const authData = {
session,
status,
};
console.log(status);
return (
<>
<Head>
<title>Tube Archivist</title>
<meta name="description" content="Generated by create next app" />
<link rel="icon" href="/favicon/favicon.ico" />
</Head>
<Suspense fallback={<h1>Loading</h1>}>
<DynamicHeader authData={authData} />
</Suspense>
<div
style={{
display: "flex",
flexDirection: "column",
maxWidth: "100px",
}}
>
<SignInOutButton isSignedIn={!!session?.user} />
</div>
</>
);
};
export default Home;
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment