support for bearer tokens
This commit is contained in:
parent
b458fad567
commit
ae1889e757
|
@ -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
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -21,5 +21,5 @@ export default function useInitialData() {
|
||||||
// setLinks();
|
// setLinks();
|
||||||
setAccount(data.user.id);
|
setAccount(data.user.id);
|
||||||
}
|
}
|
||||||
}, [status]);
|
}, [status, data]);
|
||||||
}
|
}
|
||||||
|
|
|
@ -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 <></>;
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -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),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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 });
|
||||||
|
|
|
@ -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 });
|
||||||
|
|
|
@ -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 });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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();
|
||||||
};
|
};
|
||||||
|
|
|
@ -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.",
|
||||||
});
|
});
|
||||||
|
|
|
@ -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,
|
||||||
});
|
});
|
||||||
|
|
|
@ -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,
|
||||||
});
|
});
|
||||||
|
|
|
@ -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 });
|
||||||
}
|
}
|
||||||
|
|
|
@ -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 });
|
||||||
|
|
|
@ -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 });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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 });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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 });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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!" });
|
|
||||||
// }
|
|
||||||
// }
|
|
|
@ -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";
|
||||||
|
|
|
@ -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() {
|
||||||
|
|
125
pages/test.tsx
125
pages/test.tsx
|
@ -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>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -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;
|
||||||
|
|
|
@ -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;
|
||||||
|
|
Ŝarĝante…
Reference in New Issue