support for bearer tokens

This commit is contained in:
daniel31x13 2023-11-02 14:59:31 -04:00
parent b458fad567
commit ae1889e757
27 changed files with 148 additions and 436 deletions

View File

@ -9,7 +9,6 @@ import {
} from "@fortawesome/free-solid-svg-icons"; } from "@fortawesome/free-solid-svg-icons";
import useCollectionStore from "@/store/collections"; import useCollectionStore from "@/store/collections";
import { CollectionIncludingMembersAndLinkCount, Member } from "@/types/global"; import { CollectionIncludingMembersAndLinkCount, Member } from "@/types/global";
import { useSession } from "next-auth/react";
import addMemberToCollection from "@/lib/client/addMemberToCollection"; import addMemberToCollection from "@/lib/client/addMemberToCollection";
import Checkbox from "../../Checkbox"; import Checkbox from "../../Checkbox";
import SubmitButton from "@/components/SubmitButton"; import SubmitButton from "@/components/SubmitButton";
@ -18,6 +17,7 @@ import usePermissions from "@/hooks/usePermissions";
import { toast } from "react-hot-toast"; import { toast } from "react-hot-toast";
import getPublicUserData from "@/lib/client/getPublicUserData"; import getPublicUserData from "@/lib/client/getPublicUserData";
import TextInput from "@/components/TextInput"; import TextInput from "@/components/TextInput";
import useAccountStore from "@/store/account";
type Props = { type Props = {
toggleCollectionModal: Function; toggleCollectionModal: Function;
@ -34,6 +34,7 @@ export default function TeamManagement({
collection, collection,
method, method,
}: Props) { }: Props) {
const { account } = useAccountStore();
const permissions = usePermissions(collection.id as number); const permissions = usePermissions(collection.id as number);
const currentURL = new URL(document.URL); const currentURL = new URL(document.URL);
@ -59,8 +60,6 @@ export default function TeamManagement({
const { addCollection, updateCollection } = useCollectionStore(); const { addCollection, updateCollection } = useCollectionStore();
const session = useSession();
const setMemberState = (newMember: Member) => { const setMemberState = (newMember: Member) => {
if (!collection) return null; if (!collection) return null;
@ -158,7 +157,7 @@ export default function TeamManagement({
onKeyDown={(e) => onKeyDown={(e) =>
e.key === "Enter" && e.key === "Enter" &&
addMemberToCollection( addMemberToCollection(
session.data?.user.username as string, account.username as string,
memberUsername || "", memberUsername || "",
collection, collection,
setMemberState setMemberState
@ -169,7 +168,7 @@ export default function TeamManagement({
<div <div
onClick={() => onClick={() =>
addMemberToCollection( addMemberToCollection(
session.data?.user.username as string, account.username as string,
memberUsername || "", memberUsername || "",
collection, collection,
setMemberState setMemberState

View File

@ -49,6 +49,7 @@ export default function AddOrEditLink({
screenshotPath: "", screenshotPath: "",
pdfPath: "", pdfPath: "",
readabilityPath: "", readabilityPath: "",
textContent: "",
collection: { collection: {
name: "", name: "",
ownerId: data?.user.id as number, ownerId: data?.user.id as number,

View File

@ -21,5 +21,5 @@ export default function useInitialData() {
// setLinks(); // setLinks();
setAccount(data.user.id); setAccount(data.user.id);
} }
}, [status]); }, [status, data]);
} }

View File

@ -4,6 +4,7 @@ import Loader from "../components/Loader";
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import useInitialData from "@/hooks/useInitialData"; import useInitialData from "@/hooks/useInitialData";
import useAccountStore from "@/store/account";
interface Props { interface Props {
children: ReactNode; children: ReactNode;
@ -13,6 +14,7 @@ export default function AuthRedirect({ children }: Props) {
const router = useRouter(); const router = useRouter();
const { status, data } = useSession(); const { status, data } = useSession();
const [redirect, setRedirect] = useState(true); const [redirect, setRedirect] = useState(true);
const { account } = useAccountStore();
const emailEnabled = process.env.NEXT_PUBLIC_EMAIL_PROVIDER; const emailEnabled = process.env.NEXT_PUBLIC_EMAIL_PROVIDER;
@ -25,7 +27,8 @@ export default function AuthRedirect({ children }: Props) {
status === "authenticated" && status === "authenticated" &&
(data.user.isSubscriber === true || (data.user.isSubscriber === true ||
data.user.isSubscriber === undefined) && data.user.isSubscriber === undefined) &&
!data.user.username account.id &&
!account.username
) { ) {
router.push("/choose-username").then(() => { router.push("/choose-username").then(() => {
setRedirect(false); setRedirect(false);
@ -66,7 +69,7 @@ export default function AuthRedirect({ children }: Props) {
} else { } else {
setRedirect(false); setRedirect(false);
} }
}, [status]); }, [status, account]);
if (status !== "loading" && !redirect) return <>{children}</>; if (status !== "loading" && !redirect) return <>{children}</>;
else return <></>; else return <></>;

View File

@ -1,21 +1,31 @@
import { NextApiRequest } from "next"; import { NextApiRequest, NextApiResponse } from "next";
import { getToken } from "next-auth/jwt"; import { getToken } from "next-auth/jwt";
import { prisma } from "./db";
import { User } from "@prisma/client";
type Props = { type Props = {
req: NextApiRequest; req: NextApiRequest;
res: NextApiResponse;
}; };
export default async function authenticateUser({ req }: Props) { export default async function authenticateUser({
req,
res,
}: Props): Promise<User | null> {
const token = await getToken({ req }); const token = await getToken({ req });
const userId = token?.id;
if (!token?.id) { if (!userId) {
return { response: "You must be logged in.", status: 401 }; res.status(401).json({ message: "You must be logged in." });
} else if (token.isSubscriber === false) return null;
return { } else if (token.isSubscriber === false) {
response: res.status(401).json({
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.", message:
status: 401, "You are not a subscriber, feel free to reach out to us at support@linkwarden.app if you think this is an issue.",
}; });
return null;
}
return token; const user = await prisma.user.findUnique({ where: { id: userId } });
return user;
} }

View File

@ -10,12 +10,7 @@ const emailEnabled =
process.env.EMAIL_FROM && process.env.EMAIL_SERVER ? true : false; process.env.EMAIL_FROM && process.env.EMAIL_SERVER ? true : false;
export default async function updateUserById( export default async function updateUserById(
sessionUser: { userId: number,
id: number;
username: string;
email: string;
isSubscriber: boolean;
},
data: AccountSettings data: AccountSettings
) { ) {
if (emailEnabled && !data.email) if (emailEnabled && !data.email)
@ -49,7 +44,7 @@ export default async function updateUserById(
const userIsTaken = await prisma.user.findFirst({ const userIsTaken = await prisma.user.findFirst({
where: { where: {
id: { not: sessionUser.id }, id: { not: userId },
OR: emailEnabled OR: emailEnabled
? [ ? [
{ {
@ -97,7 +92,7 @@ export default async function updateUserById(
createFolder({ filePath: `uploads/avatar` }); createFolder({ filePath: `uploads/avatar` });
await createFile({ await createFile({
filePath: `uploads/avatar/${sessionUser.id}.jpg`, filePath: `uploads/avatar/${userId}.jpg`,
data: base64Data, data: base64Data,
isBase64: true, isBase64: true,
}); });
@ -112,9 +107,13 @@ export default async function updateUserById(
}; };
} }
} else if (data.image == "") { } else if (data.image == "") {
removeFile({ filePath: `uploads/avatar/${sessionUser.id}.jpg` }); removeFile({ filePath: `uploads/avatar/${userId}.jpg` });
} }
const previousEmail = (
await prisma.user.findUnique({ where: { id: userId } })
)?.email;
// Other settings // Other settings
const saltRounds = 10; const saltRounds = 10;
@ -122,14 +121,14 @@ export default async function updateUserById(
const updatedUser = await prisma.user.update({ const updatedUser = await prisma.user.update({
where: { where: {
id: sessionUser.id, id: userId,
}, },
data: { data: {
name: data.name, name: data.name,
username: data.username.toLowerCase().trim(), username: data.username.toLowerCase().trim(),
email: data.email?.toLowerCase().trim(), email: data.email?.toLowerCase().trim(),
isPrivate: data.isPrivate, isPrivate: data.isPrivate,
image: data.image ? `uploads/avatar/${sessionUser.id}.jpg` : "", image: data.image ? `uploads/avatar/${userId}.jpg` : "",
archiveAsScreenshot: data.archiveAsScreenshot, archiveAsScreenshot: data.archiveAsScreenshot,
archiveAsPDF: data.archiveAsPDF, archiveAsPDF: data.archiveAsPDF,
archiveAsWaybackMachine: data.archiveAsWaybackMachine, archiveAsWaybackMachine: data.archiveAsWaybackMachine,
@ -167,7 +166,7 @@ export default async function updateUserById(
// Delete whitelistedUsers that are not present in the new list // Delete whitelistedUsers that are not present in the new list
await prisma.whitelistedUser.deleteMany({ await prisma.whitelistedUser.deleteMany({
where: { where: {
userId: sessionUser.id, userId: userId,
username: { username: {
in: usernamesToDelete, in: usernamesToDelete,
}, },
@ -179,17 +178,17 @@ export default async function updateUserById(
await prisma.whitelistedUser.create({ await prisma.whitelistedUser.create({
data: { data: {
username, username,
userId: sessionUser.id, userId: userId,
}, },
}); });
} }
const STRIPE_SECRET_KEY = process.env.STRIPE_SECRET_KEY; const STRIPE_SECRET_KEY = process.env.STRIPE_SECRET_KEY;
if (STRIPE_SECRET_KEY && emailEnabled && sessionUser.email !== data.email) if (STRIPE_SECRET_KEY && emailEnabled && previousEmail !== data.email)
await updateCustomerEmail( await updateCustomerEmail(
STRIPE_SECRET_KEY, STRIPE_SECRET_KEY,
sessionUser.email, previousEmail as string,
data.email as string data.email as string
); );

View File

@ -1,28 +1,20 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import getPermission from "@/lib/api/getPermission"; import getPermission from "@/lib/api/getPermission";
import readFile from "@/lib/api/storage/readFile"; import readFile from "@/lib/api/storage/readFile";
import authenticateUser from "@/lib/api/authenticateUser";
export default async function Index(req: NextApiRequest, res: NextApiResponse) { export default async function Index(req: NextApiRequest, res: NextApiResponse) {
if (!req.query.params) if (!req.query.params)
return res.status(401).json({ response: "Invalid parameters." }); return res.status(401).json({ response: "Invalid parameters." });
const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
const collectionId = req.query.params[0]; const collectionId = req.query.params[0];
const linkId = req.query.params[1]; const linkId = req.query.params[1];
const session = await getServerSession(req, res, authOptions);
if (!session?.user?.username)
return res.status(401).json({ response: "You must be logged in." });
else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
const collectionIsAccessible = await getPermission({ const collectionIsAccessible = await getPermission({
userId: session.user.id, userId: user.id,
collectionId: Number(collectionId), collectionId: Number(collectionId),
}); });

View File

@ -18,7 +18,7 @@ const providers: Provider[] = [
type: "credentials", type: "credentials",
credentials: {}, credentials: {},
async authorize(credentials, req) { async authorize(credentials, req) {
console.log("User logged in..."); console.log("User logged in attempt...");
if (!credentials) return null; if (!credentials) return null;
const { username, password } = credentials as { const { username, password } = credentials as {
@ -51,7 +51,7 @@ const providers: Provider[] = [
} }
if (passwordMatches) { if (passwordMatches) {
return findUser; return { id: findUser?.id };
} else return null as any; } else return null as any;
}, },
}), }),
@ -101,9 +101,15 @@ export const authOptions: AuthOptions = {
STRIPE_SECRET_KEY && STRIPE_SECRET_KEY &&
(trigger || subscriptionIsTimesUp || !token.isSubscriber) (trigger || subscriptionIsTimesUp || !token.isSubscriber)
) { ) {
const user = await prisma.user.findUnique({
where: {
id: Number(token.sub),
},
});
const subscription = await checkSubscription( const subscription = await checkSubscription(
STRIPE_SECRET_KEY, STRIPE_SECRET_KEY,
token.email as string user?.email as string
); );
if (subscription.subscriptionCanceledAt) { if (subscription.subscriptionCanceledAt) {
@ -115,27 +121,22 @@ export const authOptions: AuthOptions = {
if (trigger === "signIn") { if (trigger === "signIn") {
token.id = user.id as number; token.id = user.id as number;
token.username = (user as any).username;
} else if (trigger === "update" && token.id) { } else if (trigger === "update" && token.id) {
console.log(token);
const user = await prisma.user.findUnique({ const user = await prisma.user.findUnique({
where: { where: {
id: token.id as number, id: token.id as number,
}, },
}); });
if (user?.name && user.username && user.email) { if (user?.name) {
token.name = user.name; token.name = user.name;
token.username = user.username?.toLowerCase();
token.email = user.email?.toLowerCase();
} }
} }
return token; return token;
}, },
async session({ session, token }) { async session({ session, token }) {
session.user.id = token.id; session.user.id = token.id;
session.user.username = token.username;
session.user.isSubscriber = token.isSubscriber; session.user.isSubscriber = token.isSubscriber;
return session; return session;

View File

@ -1,26 +1,13 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import { prisma } from "@/lib/api/db"; import { prisma } from "@/lib/api/db";
import readFile from "@/lib/api/storage/readFile"; import readFile from "@/lib/api/storage/readFile";
import authenticateUser from "@/lib/api/authenticateUser";
export default async function Index(req: NextApiRequest, res: NextApiResponse) { export default async function Index(req: NextApiRequest, res: NextApiResponse) {
const session = await getServerSession(req, res, authOptions);
const userId = session?.user.id;
const username = session?.user.username?.toLowerCase();
const queryId = Number(req.query.id); const queryId = Number(req.query.id);
if (!userId || !username) const user = await authenticateUser({ req, res });
return res if (!user) return res.status(404).json({ response: "User not found." });
.setHeader("Content-Type", "text/plain")
.status(401)
.send("You must be logged in.");
else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
if (!queryId) if (!queryId)
return res return res
@ -28,7 +15,7 @@ export default async function Index(req: NextApiRequest, res: NextApiResponse) {
.status(401) .status(401)
.send("Invalid parameters."); .send("Invalid parameters.");
if (userId !== queryId) { if (user.id !== queryId) {
const targetUser = await prisma.user.findUnique({ const targetUser = await prisma.user.findUnique({
where: { where: {
id: queryId, id: queryId,
@ -42,7 +29,11 @@ export default async function Index(req: NextApiRequest, res: NextApiResponse) {
(whitelistedUsername) => whitelistedUsername.username (whitelistedUsername) => whitelistedUsername.username
); );
if (targetUser?.isPrivate && !whitelistedUsernames?.includes(username)) { if (
targetUser?.isPrivate &&
user.username &&
!whitelistedUsernames?.includes(user.username)
) {
return res return res
.setHeader("Content-Type", "text/plain") .setHeader("Content-Type", "text/plain")
.status(400) .status(400)

View File

@ -1,33 +1,25 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import updateCollectionById from "@/lib/api/controllers/collections/collectionId/updateCollectionById"; import updateCollectionById from "@/lib/api/controllers/collections/collectionId/updateCollectionById";
import deleteCollectionById from "@/lib/api/controllers/collections/collectionId/deleteCollectionById"; import deleteCollectionById from "@/lib/api/controllers/collections/collectionId/deleteCollectionById";
import authenticateUser from "@/lib/api/authenticateUser";
export default async function collections( export default async function collections(
req: NextApiRequest, req: NextApiRequest,
res: NextApiResponse res: NextApiResponse
) { ) {
const session = await getServerSession(req, res, authOptions); const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
if (!session?.user?.id) {
return res.status(401).json({ response: "You must be logged in." });
} else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
if (req.method === "PUT") { if (req.method === "PUT") {
const updated = await updateCollectionById( const updated = await updateCollectionById(
session.user.id, user.id,
Number(req.query.id) as number, Number(req.query.id) as number,
req.body req.body
); );
return res.status(updated.status).json({ response: updated.response }); return res.status(updated.status).json({ response: updated.response });
} else if (req.method === "DELETE") { } else if (req.method === "DELETE") {
const deleted = await deleteCollectionById( const deleted = await deleteCollectionById(
session.user.id, user.id,
Number(req.query.id) as number Number(req.query.id) as number
); );
return res.status(deleted.status).json({ response: deleted.response }); return res.status(deleted.status).json({ response: deleted.response });

View File

@ -1,30 +1,22 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import getCollections from "@/lib/api/controllers/collections/getCollections"; import getCollections from "@/lib/api/controllers/collections/getCollections";
import postCollection from "@/lib/api/controllers/collections/postCollection"; import postCollection from "@/lib/api/controllers/collections/postCollection";
import authenticateUser from "@/lib/api/authenticateUser";
export default async function collections( export default async function collections(
req: NextApiRequest, req: NextApiRequest,
res: NextApiResponse res: NextApiResponse
) { ) {
const session = await getServerSession(req, res, authOptions); const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
if (!session?.user?.id) {
return res.status(401).json({ response: "You must be logged in." });
} else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
if (req.method === "GET") { if (req.method === "GET") {
const collections = await getCollections(session.user.id); const collections = await getCollections(user.id);
return res return res
.status(collections.status) .status(collections.status)
.json({ response: collections.response }); .json({ response: collections.response });
} else if (req.method === "POST") { } else if (req.method === "POST") {
const newCollection = await postCollection(req.body, session.user.id); const newCollection = await postCollection(req.body, user.id);
return res return res
.status(newCollection.status) .status(newCollection.status)
.json({ response: newCollection.response }); .json({ response: newCollection.response });

View File

@ -1,19 +1,11 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import { LinkRequestQuery } from "@/types/global"; import { LinkRequestQuery } from "@/types/global";
import getDashboardData from "@/lib/api/controllers/dashboard/getDashboardData"; import getDashboardData from "@/lib/api/controllers/dashboard/getDashboardData";
import authenticateUser from "@/lib/api/authenticateUser";
export default async function links(req: NextApiRequest, res: NextApiResponse) { export default async function links(req: NextApiRequest, res: NextApiResponse) {
const session = await getServerSession(req, res, authOptions); const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
if (!session?.user?.id) {
return res.status(401).json({ response: "You must be logged in." });
} else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
if (req.method === "GET") { if (req.method === "GET") {
const convertedData: LinkRequestQuery = { const convertedData: LinkRequestQuery = {
@ -21,7 +13,7 @@ export default async function links(req: NextApiRequest, res: NextApiResponse) {
cursor: req.query.cursor ? Number(req.query.cursor as string) : undefined, cursor: req.query.cursor ? Number(req.query.cursor as string) : undefined,
}; };
const links = await getDashboardData(session.user.id, convertedData); const links = await getDashboardData(user.id, convertedData);
return res.status(links.status).json({ response: links.response }); return res.status(links.status).json({ response: links.response });
} }
} }

View File

@ -4,7 +4,7 @@ import { getToken } from "next-auth/jwt";
export default async (req: NextApiRequest, res: NextApiResponse) => { export default async (req: NextApiRequest, res: NextApiResponse) => {
// if using `NEXTAUTH_SECRET` env variable, we detect it, and you won't actually need to `secret` // if using `NEXTAUTH_SECRET` env variable, we detect it, and you won't actually need to `secret`
// const token = await getToken({ req }) // const token = await getToken({ req })
const token = await getToken({ req }); // const token = await getToken({ req });
console.log("JSON Web Token", token); // console.log("JSON Web Token", token);
res.end(); // res.end();
}; };

View File

@ -1,21 +1,13 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import archive from "@/lib/api/archive"; import archive from "@/lib/api/archive";
import { prisma } from "@/lib/api/db"; import { prisma } from "@/lib/api/db";
import authenticateUser from "@/lib/api/authenticateUser";
const RE_ARCHIVE_LIMIT = Number(process.env.RE_ARCHIVE_LIMIT) || 5; const RE_ARCHIVE_LIMIT = Number(process.env.RE_ARCHIVE_LIMIT) || 5;
export default async function links(req: NextApiRequest, res: NextApiResponse) { export default async function links(req: NextApiRequest, res: NextApiResponse) {
const session = await getServerSession(req, res, authOptions); const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
if (!session?.user?.id) {
return res.status(401).json({ response: "You must be logged in." });
} else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
const link = await prisma.link.findUnique({ const link = await prisma.link.findUnique({
where: { where: {
@ -29,7 +21,7 @@ export default async function links(req: NextApiRequest, res: NextApiResponse) {
response: "Link not found.", response: "Link not found.",
}); });
if (link.collection.ownerId !== session.user.id) if (link.collection.ownerId !== user.id)
return res.status(401).json({ return res.status(401).json({
response: "Permission denied.", response: "Permission denied.",
}); });
@ -49,7 +41,7 @@ export default async function links(req: NextApiRequest, res: NextApiResponse) {
} minutes or create a new one.`, } minutes or create a new one.`,
}); });
archive(link.id, link.url, session.user.id); archive(link.id, link.url, user.id);
return res.status(200).json({ return res.status(200).json({
response: "Link is being archived.", response: "Link is being archived.",
}); });

View File

@ -1,29 +1,21 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import deleteLinkById from "@/lib/api/controllers/links/linkId/deleteLinkById"; import deleteLinkById from "@/lib/api/controllers/links/linkId/deleteLinkById";
import updateLinkById from "@/lib/api/controllers/links/linkId/updateLinkById"; import updateLinkById from "@/lib/api/controllers/links/linkId/updateLinkById";
import getLinkById from "@/lib/api/controllers/links/linkId/getLinkById"; import getLinkById from "@/lib/api/controllers/links/linkId/getLinkById";
import authenticateUser from "@/lib/api/authenticateUser";
export default async function links(req: NextApiRequest, res: NextApiResponse) { export default async function links(req: NextApiRequest, res: NextApiResponse) {
const session = await getServerSession(req, res, authOptions); const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
if (!session?.user?.id) {
return res.status(401).json({ response: "You must be logged in." });
} else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
if (req.method === "GET") { if (req.method === "GET") {
const updated = await getLinkById(session.user.id, Number(req.query.id)); const updated = await getLinkById(user.id, Number(req.query.id));
return res.status(updated.status).json({ return res.status(updated.status).json({
response: updated.response, response: updated.response,
}); });
} else if (req.method === "PUT") { } else if (req.method === "PUT") {
const updated = await updateLinkById( const updated = await updateLinkById(
session.user.id, user.id,
Number(req.query.id), Number(req.query.id),
req.body req.body
); );
@ -31,7 +23,7 @@ export default async function links(req: NextApiRequest, res: NextApiResponse) {
response: updated.response, response: updated.response,
}); });
} else if (req.method === "DELETE") { } else if (req.method === "DELETE") {
const deleted = await deleteLinkById(session.user.id, Number(req.query.id)); const deleted = await deleteLinkById(user.id, Number(req.query.id));
return res.status(deleted.status).json({ return res.status(deleted.status).json({
response: deleted.response, response: deleted.response,
}); });

View File

@ -1,25 +1,12 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import getLinks from "@/lib/api/controllers/links/getLinks"; import getLinks from "@/lib/api/controllers/links/getLinks";
import postLink from "@/lib/api/controllers/links/postLink"; import postLink from "@/lib/api/controllers/links/postLink";
import { LinkRequestQuery } from "@/types/global"; import { LinkRequestQuery } from "@/types/global";
import { getToken } from "next-auth/jwt"; import authenticateUser from "@/lib/api/authenticateUser";
export default async function links(req: NextApiRequest, res: NextApiResponse) { export default async function links(req: NextApiRequest, res: NextApiResponse) {
const token = await getToken({ req }); const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
// const session = await getServerSession(req, res, authOptions);
return res.status(200).json(token);
if (!session?.user?.id) {
return res.status(401).json({ response: "You must be logged in." });
} else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
if (req.method === "GET") { if (req.method === "GET") {
// Convert the type of the request query to "LinkRequestQuery" // Convert the type of the request query to "LinkRequestQuery"
@ -45,10 +32,10 @@ export default async function links(req: NextApiRequest, res: NextApiResponse) {
searchByTags: req.query.searchByTags === "true" ? true : undefined, searchByTags: req.query.searchByTags === "true" ? true : undefined,
}; };
const links = await getLinks(session.user.id, convertedData); const links = await getLinks(user.id, convertedData);
return res.status(links.status).json({ response: links.response }); return res.status(links.status).json({ response: links.response });
} else if (req.method === "POST") { } else if (req.method === "POST") {
const newlink = await postLink(req.body, session.user.id); const newlink = await postLink(req.body, user.id);
return res.status(newlink.status).json({ return res.status(newlink.status).json({
response: newlink.response, response: newlink.response,
}); });

View File

@ -1,10 +1,9 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import exportData from "@/lib/api/controllers/migration/exportData"; import exportData from "@/lib/api/controllers/migration/exportData";
import importFromHTMLFile from "@/lib/api/controllers/migration/importFromHTMLFile"; import importFromHTMLFile from "@/lib/api/controllers/migration/importFromHTMLFile";
import importFromLinkwarden from "@/lib/api/controllers/migration/importFromLinkwarden"; import importFromLinkwarden from "@/lib/api/controllers/migration/importFromLinkwarden";
import { MigrationFormat, MigrationRequest } from "@/types/global"; import { MigrationFormat, MigrationRequest } from "@/types/global";
import authenticateUser from "@/lib/api/authenticateUser";
export const config = { export const config = {
api: { api: {
@ -15,18 +14,11 @@ export const config = {
}; };
export default async function users(req: NextApiRequest, res: NextApiResponse) { export default async function users(req: NextApiRequest, res: NextApiResponse) {
const session = await getServerSession(req, res, authOptions); const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
if (!session?.user.id) {
return res.status(401).json({ response: "You must be logged in." });
} else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
if (req.method === "GET") { if (req.method === "GET") {
const data = await exportData(session.user.id); const data = await exportData(user.id);
if (data.status === 200) if (data.status === 200)
return res return res
@ -39,10 +31,10 @@ export default async function users(req: NextApiRequest, res: NextApiResponse) {
let data; let data;
if (request.format === MigrationFormat.htmlFile) if (request.format === MigrationFormat.htmlFile)
data = await importFromHTMLFile(session.user.id, request.data); data = await importFromHTMLFile(user.id, request.data);
if (request.format === MigrationFormat.linkwarden) if (request.format === MigrationFormat.linkwarden)
data = await importFromLinkwarden(session.user.id, request.data); data = await importFromLinkwarden(user.id, request.data);
if (data) return res.status(data.status).json({ response: data.response }); if (data) return res.status(data.status).json({ response: data.response });
} }

View File

@ -1,21 +1,20 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import paymentCheckout from "@/lib/api/paymentCheckout"; import paymentCheckout from "@/lib/api/paymentCheckout";
import { Plan } from "@/types/global"; import { Plan } from "@/types/global";
import authenticateUser from "@/lib/api/authenticateUser";
export default async function users(req: NextApiRequest, res: NextApiResponse) { export default async function users(req: NextApiRequest, res: NextApiResponse) {
const STRIPE_SECRET_KEY = process.env.STRIPE_SECRET_KEY; const STRIPE_SECRET_KEY = process.env.STRIPE_SECRET_KEY;
const MONTHLY_PRICE_ID = process.env.MONTHLY_PRICE_ID; const MONTHLY_PRICE_ID = process.env.MONTHLY_PRICE_ID;
const YEARLY_PRICE_ID = process.env.YEARLY_PRICE_ID; const YEARLY_PRICE_ID = process.env.YEARLY_PRICE_ID;
const session = await getServerSession(req, res, authOptions);
if (!session?.user?.id) if (!STRIPE_SECRET_KEY || !MONTHLY_PRICE_ID || !YEARLY_PRICE_ID) {
return res.status(401).json({ response: "You must be logged in." });
else if (!STRIPE_SECRET_KEY || !MONTHLY_PRICE_ID || !YEARLY_PRICE_ID) {
return res.status(400).json({ response: "Payment is disabled." }); return res.status(400).json({ response: "Payment is disabled." });
} }
const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
let PRICE_ID = MONTHLY_PRICE_ID; let PRICE_ID = MONTHLY_PRICE_ID;
if ((Number(req.query.plan) as unknown as Plan) === Plan.monthly) if ((Number(req.query.plan) as unknown as Plan) === Plan.monthly)
@ -26,7 +25,7 @@ export default async function users(req: NextApiRequest, res: NextApiResponse) {
if (req.method === "GET") { if (req.method === "GET") {
const users = await paymentCheckout( const users = await paymentCheckout(
STRIPE_SECRET_KEY, STRIPE_SECRET_KEY,
session?.user.email, user.email as string,
PRICE_ID PRICE_ID
); );
return res.status(users.status).json({ response: users.response }); return res.status(users.status).json({ response: users.response });

View File

@ -1,23 +1,15 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import updateTag from "@/lib/api/controllers/tags/tagId/updeteTagById"; import updateTag from "@/lib/api/controllers/tags/tagId/updeteTagById";
import authenticateUser from "@/lib/api/authenticateUser";
export default async function tags(req: NextApiRequest, res: NextApiResponse) { export default async function tags(req: NextApiRequest, res: NextApiResponse) {
const session = await getServerSession(req, res, authOptions); const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
if (!session?.user?.username) {
return res.status(401).json({ response: "You must be logged in." });
} else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
const tagId = Number(req.query.id); const tagId = Number(req.query.id);
if (req.method === "PUT") { if (req.method === "PUT") {
const tags = await updateTag(session.user.id, tagId, req.body); const tags = await updateTag(user.id, tagId, req.body);
return res.status(tags.status).json({ response: tags.response }); return res.status(tags.status).json({ response: tags.response });
} }
} }

View File

@ -1,21 +1,13 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import getTags from "@/lib/api/controllers/tags/getTags"; import getTags from "@/lib/api/controllers/tags/getTags";
import authenticateUser from "@/lib/api/authenticateUser";
export default async function tags(req: NextApiRequest, res: NextApiResponse) { export default async function tags(req: NextApiRequest, res: NextApiResponse) {
const session = await getServerSession(req, res, authOptions); const user = await authenticateUser({ req, res });
if (!user) return res.status(404).json({ response: "User not found." });
if (!session?.user?.username) {
return res.status(401).json({ response: "You must be logged in." });
} else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
if (req.method === "GET") { if (req.method === "GET") {
const tags = await getTags(session.user.id); const tags = await getTags(user.id);
return res.status(tags.status).json({ response: tags.response }); return res.status(tags.status).json({ response: tags.response });
} }
} }

View File

@ -1,15 +1,23 @@
import type { NextApiRequest, NextApiResponse } from "next"; import type { NextApiRequest, NextApiResponse } from "next";
import { getServerSession } from "next-auth/next";
import { authOptions } from "@/pages/api/v1/auth/[...nextauth]";
import getUserById from "@/lib/api/controllers/users/userId/getUserById"; import getUserById from "@/lib/api/controllers/users/userId/getUserById";
import getPublicUserById from "@/lib/api/controllers/users/userId/getPublicUserById"; import getPublicUserById from "@/lib/api/controllers/users/userId/getPublicUserById";
import updateUserById from "@/lib/api/controllers/users/userId/updateUserById"; import updateUserById from "@/lib/api/controllers/users/userId/updateUserById";
import deleteUserById from "@/lib/api/controllers/users/userId/deleteUserById"; import deleteUserById from "@/lib/api/controllers/users/userId/deleteUserById";
import authenticateUser from "@/lib/api/authenticateUser";
import { prisma } from "@/lib/api/db";
import { getToken } from "next-auth/jwt";
export default async function users(req: NextApiRequest, res: NextApiResponse) { export default async function users(req: NextApiRequest, res: NextApiResponse) {
const session = await getServerSession(req, res, authOptions); const token = await getToken({ req });
const userId = session?.user.id; const userId = token?.id;
const username = session?.user.username;
if (!token?.id)
return res.status(400).json({ response: "Invalid parameters." });
const username = (await prisma.user.findUnique({ where: { id: token.id } }))
?.username;
if (!username) return res.status(404).json({ response: "User not found." });
const lookupId = req.query.id as string; const lookupId = req.query.id as string;
const isSelf = const isSelf =
@ -23,26 +31,18 @@ export default async function users(req: NextApiRequest, res: NextApiResponse) {
return res.status(users.status).json({ response: users.response }); return res.status(users.status).json({ response: users.response });
} }
if (!userId) { const user = await authenticateUser({ req, res });
return res.status(401).json({ response: "You must be logged in." }); if (!user) return res.status(404).json({ response: "User not found." });
} else if (session?.user?.isSubscriber === false)
return res.status(401).json({
response:
"You are not a subscriber, feel free to reach out to us at support@linkwarden.app in case of any issues.",
});
if (req.method === "GET") { if (req.method === "GET") {
const users = await getUserById(session.user.id); const users = await getUserById(user.id);
return res.status(users.status).json({ response: users.response }); return res.status(users.status).json({ response: users.response });
} else if (req.method === "PUT") { } else if (req.method === "PUT") {
const updated = await updateUserById(session.user, req.body); const updated = await updateUserById(user.id, req.body);
return res.status(updated.status).json({ response: updated.response }); return res.status(updated.status).json({ response: updated.response });
} else if ( } else if (req.method === "DELETE" && user.id === Number(req.query.id)) {
req.method === "DELETE" &&
session.user.id === Number(req.query.id)
) {
console.log(req.body); console.log(req.body);
const updated = await deleteUserById(session.user.id, req.body); const updated = await deleteUserById(user.id, req.body);
return res.status(updated.status).json({ response: updated.response }); return res.status(updated.status).json({ response: updated.response });
} }
} }

View File

@ -1,62 +0,0 @@
// TODO - Stripe webhooks for user cancellation...
// import { NextApiRequest, NextApiResponse } from "next";
// import Stripe from "stripe";
// import { buffer } from "micro";
// import { prisma } from "@/lib/api/db";
// const stripe = new Stripe(process.env.STRIPE_SECRET_KEY as string, {
// apiVersion: "2022-11-15",
// });
// const endpointSecret =
// "whsec_7c144bcd924041257e3d83eac1e2fba9c8a938b240fd8adb1c902f079e0cdee0";
// export const config = {
// api: {
// bodyParser: false,
// },
// };
// export default async function handler(
// req: NextApiRequest,
// res: NextApiResponse
// ) {
// if (req.method === "POST") {
// const buf = await buffer(req);
// const sig = req.headers["stripe-signature"];
// let event: Stripe.Event;
// try {
// if (!sig) throw new Error("Stripe Signature is not defined.");
// event = stripe.webhooks.constructEvent(buf, sig, endpointSecret);
// } catch (err) {
// console.log(err);
// return res.status(400).send({ response: "Error..." });
// }
// // Handle the event
// switch (event.type) {
// case "customer.subscription.deleted":
// const customerSubscriptionDeleted = event.data.object as any;
// // Revoke all the token under the customers email...
// const customer = (await stripe.customers.retrieve(
// customerSubscriptionDeleted.customer
// )) as any;
// if (customer?.email) {
// // Revoke tokens inside the database
// }
// break;
// // ... handle other event types
// default:
// console.log(`Unhandled event type ${event.type}`);
// }
// return res.status(200).send({ response: "Done!" });
// }
// }

View File

@ -12,7 +12,6 @@ import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import MainLayout from "@/layouts/MainLayout"; import MainLayout from "@/layouts/MainLayout";
import { useSession } from "next-auth/react";
import ProfilePhoto from "@/components/ProfilePhoto"; import ProfilePhoto from "@/components/ProfilePhoto";
import SortDropdown from "@/components/SortDropdown"; import SortDropdown from "@/components/SortDropdown";
import useModalStore from "@/store/modals"; import useModalStore from "@/store/modals";

View File

@ -1,8 +1,6 @@
import SubmitButton from "@/components/SubmitButton";
import { signOut } from "next-auth/react"; import { signOut } from "next-auth/react";
import { useState } from "react"; import { useState } from "react";
import { toast } from "react-hot-toast"; import { toast } from "react-hot-toast";
import { useSession } from "next-auth/react";
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import CenteredForm from "@/layouts/CenteredForm"; import CenteredForm from "@/layouts/CenteredForm";
import { Plan } from "@/types/global"; import { Plan } from "@/types/global";
@ -12,7 +10,6 @@ export default function Subscribe() {
const [plan, setPlan] = useState<Plan>(1); const [plan, setPlan] = useState<Plan>(1);
const { data, status } = useSession();
const router = useRouter(); const router = useRouter();
async function submit() { async function submit() {

View File

@ -1,125 +0,0 @@
import SubmitButton from "@/components/SubmitButton";
import TextInput from "@/components/TextInput";
import CenteredForm from "@/layouts/CenteredForm";
import { signIn } from "next-auth/react";
import Link from "next/link";
import { useState, FormEvent } from "react";
import { toast } from "react-hot-toast";
interface FormData {
username: string;
password: string;
}
const emailEnabled = process.env.NEXT_PUBLIC_EMAIL_PROVIDER;
export default function Login() {
const [submitLoader, setSubmitLoader] = useState(false);
const [form, setForm] = useState<FormData>({
username: "",
password: "",
});
async function loginUser(event: FormEvent<HTMLFormElement>) {
event.preventDefault();
if (form.username !== "" && form.password !== "") {
setSubmitLoader(true);
const load = toast.loading("Authenticating...");
const res = await signIn("credentials", {
username: form.username,
password: form.password,
redirect: false,
});
console.log(res);
toast.dismiss(load);
setSubmitLoader(false);
if (!res?.ok) {
toast.error("Invalid login.");
}
} else {
toast.error("Please fill out all the fields.");
}
}
return (
<CenteredForm text="Sign in to your account">
<form onSubmit={loginUser}>
<div className="p-4 mx-auto flex flex-col gap-3 justify-between max-w-[30rem] min-w-80 w-full bg-slate-50 dark:bg-neutral-800 rounded-2xl shadow-md border border-sky-100 dark:border-neutral-700">
<p className="text-3xl text-black dark:text-white text-center font-extralight">
Enter your credentials
</p>
<hr className="border-1 border-sky-100 dark:border-neutral-700" />
<div>
<p className="text-sm text-black dark:text-white w-fit font-semibold mb-1">
Username
{emailEnabled ? " or Email" : undefined}
</p>
<TextInput
autoFocus={true}
placeholder="johnny"
value={form.username}
className="bg-white"
onChange={(e) => setForm({ ...form, username: e.target.value })}
/>
</div>
<div className="w-full">
<p className="text-sm text-black dark:text-white w-fit font-semibold mb-1">
Password
</p>
<TextInput
type="password"
placeholder="••••••••••••••"
value={form.password}
className="bg-white"
onChange={(e) => setForm({ ...form, password: e.target.value })}
/>
{emailEnabled && (
<div className="w-fit ml-auto mt-1">
<Link
href={"/forgot"}
className="text-gray-500 dark:text-gray-400 font-semibold"
>
Forgot Password?
</Link>
</div>
)}
</div>
<SubmitButton
type="submit"
label="Login"
className=" w-full text-center"
loading={submitLoader}
/>
{process.env.NEXT_PUBLIC_DISABLE_REGISTRATION ===
"true" ? undefined : (
<div className="flex items-baseline gap-1 justify-center">
<p className="w-fit text-gray-500 dark:text-gray-400">
New here?
</p>
<Link
href={"/register"}
className="block text-black dark:text-white font-semibold"
>
Sign Up
</Link>
</div>
)}
</div>
</form>
</CenteredForm>
);
}

View File

@ -5,7 +5,10 @@ type OptionalExcluding<T, TRequired extends keyof T> = Partial<T> &
Pick<T, TRequired>; Pick<T, TRequired>;
export interface LinkIncludingShortenedCollectionAndTags export interface LinkIncludingShortenedCollectionAndTags
extends Omit<Link, "id" | "createdAt" | "collectionId" | "updatedAt"> { extends Omit<
Link,
"id" | "createdAt" | "collectionId" | "updatedAt" | "lastPreserved"
> {
id?: number; id?: number;
createdAt?: string; createdAt?: string;
collectionId?: number; collectionId?: number;

20
types/next-auth.d.ts vendored
View File

@ -5,38 +5,20 @@ declare module "next-auth" {
interface Session { interface Session {
user: { user: {
id: number; id: number;
username: string;
email: string;
isSubscriber: boolean; isSubscriber: boolean;
}; };
} }
interface User { interface User {
id: number; id: number;
name: string;
username: string;
email: string;
emailVerified: Date;
image: string;
password: string;
archiveAsScreenshot: boolean;
archiveAsPDF: boolean;
archiveAsWaybackMachine: boolean;
isPrivate: boolean;
createdAt: Date;
updatedAt: Date;
} }
} }
declare module "next-auth/jwt" { declare module "next-auth/jwt" {
interface JWT { interface JWT {
name: string;
email: string;
picture: string;
sub: string; sub: string;
isSubscriber: boolean;
id: number; id: number;
username: string; isSubscriber: boolean;
iat: number; iat: number;
exp: number; exp: number;
jti: string; jti: string;