Continued migrating more and more files

This commit is contained in:
Tiago Ribeiro
2024-09-07 18:29:20 +01:00
parent bb8dca69cf
commit 6251f8f4db
16 changed files with 254 additions and 356 deletions

View File

@@ -1,7 +1,6 @@
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction // Next.js API route support: https://nextjs.org/docs/api-routes/introduction
import type {NextApiRequest, NextApiResponse} from "next"; import type {NextApiRequest, NextApiResponse} from "next";
import {app} from "@/firebase"; import {app} from "@/firebase";
import {getFirestore, collection, getDocs, setDoc, doc, getDoc, deleteDoc, query} from "firebase/firestore";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {sessionOptions} from "@/lib/session"; import {sessionOptions} from "@/lib/session";
import {CorporateUser, Group} from "@/interfaces/user"; import {CorporateUser, Group} from "@/interfaces/user";

View File

@@ -4,7 +4,6 @@ import {app} from "@/firebase";
import {sessionOptions} from "@/lib/session"; import {sessionOptions} from "@/lib/session";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {User} from "@/interfaces/user"; import {User} from "@/interfaces/user";
import {getFirestore, getDoc, doc} from "firebase/firestore";
import client from "@/lib/mongodb"; import client from "@/lib/mongodb";
const auth = getAuth(app); const auth = getAuth(app);

View File

@@ -1,6 +1,5 @@
import type {NextApiRequest, NextApiResponse} from "next"; import type {NextApiRequest, NextApiResponse} from "next";
import {app} from "@/firebase"; import {app} from "@/firebase";
import {getFirestore, setDoc, doc, query, collection, where, getDocs, getDoc, deleteDoc, limit, updateDoc} from "firebase/firestore";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {sessionOptions} from "@/lib/session"; import {sessionOptions} from "@/lib/session";
import {v4} from "uuid"; import {v4} from "uuid";

View File

@@ -7,8 +7,9 @@ import {sessionOptions} from "@/lib/session";
import {Group} from "@/interfaces/user"; import {Group} from "@/interfaces/user";
import {Payment} from "@/interfaces/paypal"; import {Payment} from "@/interfaces/paypal";
import {deleteObject, ref} from "firebase/storage"; import {deleteObject, ref} from "firebase/storage";
import client from "@/lib/mongodb";
const db = getFirestore(app); const db = client.db(process.env.MONGODB_DB);
export default withIronSessionApiRoute(handler, sessionOptions); export default withIronSessionApiRoute(handler, sessionOptions);
@@ -28,10 +29,10 @@ async function get(req: NextApiRequest, res: NextApiResponse) {
const {id} = req.query as {id: string}; const {id} = req.query as {id: string};
const snapshot = await getDoc(doc(db, "payments", id)); const payment = await db.collection("payments").findOne<Payment>({id});
if (snapshot.exists()) { if (!!payment) {
res.status(200).json({...snapshot.data(), id: snapshot.id}); res.status(200).json(payment);
} else { } else {
res.status(404).json(undefined); res.status(404).json(undefined);
} }
@@ -45,15 +46,15 @@ async function del(req: NextApiRequest, res: NextApiResponse) {
const {id} = req.query as {id: string}; const {id} = req.query as {id: string};
const snapshot = await getDoc(doc(db, "payments", id)); const payment = await db.collection("payments").findOne<Payment>({id});
const data = snapshot.data() as Payment; if (!payment) return res.status(404).json({ok: false});
const user = req.session.user; const user = req.session.user;
if (user.type === "admin" || user.type === "developer") { if (user.type === "admin" || user.type === "developer") {
if (data.commissionTransfer) await deleteObject(ref(storage, data.commissionTransfer)); if (payment.commissionTransfer) await deleteObject(ref(storage, payment.commissionTransfer));
if (data.corporateTransfer) await deleteObject(ref(storage, data.corporateTransfer)); if (payment.corporateTransfer) await deleteObject(ref(storage, payment.corporateTransfer));
await deleteDoc(snapshot.ref); await db.collection("payments").deleteOne({id: payment.id});
res.status(200).json({ok: true}); res.status(200).json({ok: true});
return; return;
} }
@@ -68,15 +69,17 @@ async function patch(req: NextApiRequest, res: NextApiResponse) {
} }
const {id} = req.query as {id: string}; const {id} = req.query as {id: string};
const snapshot = await getDoc(doc(db, "payments", id));
const payment = await db.collection("payments").findOne<Payment>({id});
if (!payment) return res.status(404).json({ok: false});
const user = req.session.user; const user = req.session.user;
if (user.type === "admin" || user.type === "developer") { if (user.type === "admin" || user.type === "developer") {
await setDoc(snapshot.ref, req.body, {merge: true}); await db.collection("payments").updateOne({id: payment.id}, {$set: req.body});
if (req.body.isPaid) { if (req.body.isPaid) {
const corporateID = req.body.corporate; const corporateID = req.body.corporate;
await setDoc(doc(db, "users", corporateID), {status: "active"}, {merge: true}); await db.collection("users").updateOne({id: corporateID}, {$set: {status: "active"}});
} }
return res.status(200).json({ok: true}); return res.status(200).json({ok: true});
} }

View File

@@ -1,19 +1,14 @@
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction // Next.js API route support: https://nextjs.org/docs/api-routes/introduction
import type {NextApiRequest, NextApiResponse} from "next"; import type {NextApiRequest, NextApiResponse} from "next";
import {app} from "@/firebase"; import {app} from "@/firebase";
import { import {getFirestore, collection, getDocs, query, where} from "firebase/firestore";
getFirestore,
collection,
getDocs,
query,
where,
} from "firebase/firestore";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {sessionOptions} from "@/lib/session"; import {sessionOptions} from "@/lib/session";
import {Payment} from "@/interfaces/paypal"; import {Payment} from "@/interfaces/paypal";
import {PaymentsStatus} from "@/interfaces/user.payments"; import {PaymentsStatus} from "@/interfaces/user.payments";
import client from "@/lib/mongodb";
const db = getFirestore(app); const db = client.db(process.env.MONGODB_DB);
export default withIronSessionApiRoute(handler, sessionOptions); export default withIronSessionApiRoute(handler, sessionOptions);
@@ -30,23 +25,12 @@ async function get(req: NextApiRequest, res: NextApiResponse) {
return; return;
} }
// if it's an admin, don't apply query filters const payments = await db
const whereClauses = ["admin", "developer"].includes(req.session.user.type) .collection("payments")
? [] .find(["admin", "developer"].includes(req.session.user.type) ? {} : {[req.session.user.type]: req.session.user.id})
: [ .toArray();
// where("agent", "==", "xRMirufz6PPQqxKBgvPTWiWKBD63"),
where(req.session.user.type, "==", req.session.user.id),
// Based on the logic of query we should be able to do this:
// where("isPaid", "==", paid === "paid"),
// but for some reason it is ignoring all but the first clause
// I opted into only fetching relevant content for the user
// and then filter it with JS
];
const codeQuery = query(collection(db, "payments"), ...whereClauses); if (payments.length === 0) {
const snapshot = await getDocs(codeQuery);
if (snapshot.empty) {
res.status(200).json({ res.status(200).json({
pending: [], pending: [],
done: [], done: [],
@@ -54,12 +38,7 @@ async function get(req: NextApiRequest, res: NextApiResponse) {
return; return;
} }
const docs = snapshot.docs.map((doc) => ({ const paidStatusEntries = payments.reduce(
id: doc.id,
...doc.data(),
})) as Payment[];
const paidStatusEntries = docs.reduce(
(acc: PaymentsStatus, doc) => { (acc: PaymentsStatus, doc) => {
if (doc.isPaid) { if (doc.isPaid) {
return { return {
@@ -76,7 +55,7 @@ async function get(req: NextApiRequest, res: NextApiResponse) {
{ {
pending: [], pending: [],
done: [], done: [],
} },
); );
res.status(200).json({ res.status(200).json({
pending: [...new Set(paidStatusEntries.pending)], pending: [...new Set(paidStatusEntries.pending)],

View File

@@ -8,8 +8,9 @@ import {Group} from "@/interfaces/user";
import {Payment} from "@/interfaces/paypal"; import {Payment} from "@/interfaces/paypal";
import {v4} from "uuid"; import {v4} from "uuid";
import ShortUniqueId from "short-unique-id"; import ShortUniqueId from "short-unique-id";
import client from "@/lib/mongodb";
const db = getFirestore(app); const db = client.db(process.env.MONGODB_DB);
export default withIronSessionApiRoute(handler, sessionOptions); export default withIronSessionApiRoute(handler, sessionOptions);
@@ -24,20 +25,15 @@ async function handler(req: NextApiRequest, res: NextApiResponse) {
} }
async function get(req: NextApiRequest, res: NextApiResponse) { async function get(req: NextApiRequest, res: NextApiResponse) {
const snapshot = await getDocs(collection(db, "payments")); const payments = await db.collection("payments").find({}).toArray();
res.status(200).json( res.status(200).json(payments);
snapshot.docs.map((doc) => ({
id: doc.id,
...doc.data(),
})),
);
} }
async function post(req: NextApiRequest, res: NextApiResponse) { async function post(req: NextApiRequest, res: NextApiResponse) {
const body = req.body as Payment; const body = req.body as Payment;
const shortUID = new ShortUniqueId(); const shortUID = new ShortUniqueId();
await setDoc(doc(db, "payments", shortUID.randomUUID(8)), body); await db.collection("payments").insertOne({...body, id: shortUID.randomUUID(8)});
res.status(200).json({ok: true}); res.status(200).json({ok: true});
} }

View File

@@ -1,23 +1,19 @@
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction // Next.js API route support: https://nextjs.org/docs/api-routes/introduction
import type {NextApiRequest, NextApiResponse} from "next"; import type {NextApiRequest, NextApiResponse} from "next";
import {app} from "@/firebase"; import {app} from "@/firebase";
import { import {getFirestore, getDocs, collection} from "firebase/firestore";
getFirestore,
getDocs,
collection,
} from "firebase/firestore";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {sessionOptions} from "@/lib/session"; import {sessionOptions} from "@/lib/session";
import client from "@/lib/mongodb";
const db = getFirestore(app); const db = client.db(process.env.MONGODB_DB);
export default withIronSessionApiRoute(handler, sessionOptions); export default withIronSessionApiRoute(handler, sessionOptions);
async function get(req: NextApiRequest, res: NextApiResponse) { async function get(req: NextApiRequest, res: NextApiResponse) {
const payments = await getDocs(collection(db, "paypalpayments")); const payments = await db.collection("paypalpayments").find({}).toArray();
const data = payments.docs.map((doc) => doc.data()); res.status(200).json(payments);
res.status(200).json(data);
} }
async function handler(req: NextApiRequest, res: NextApiResponse) { async function handler(req: NextApiRequest, res: NextApiResponse) {

View File

@@ -1,7 +1,5 @@
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction // Next.js API route support: https://nextjs.org/docs/api-routes/introduction
import type {NextApiRequest, NextApiResponse} from "next"; import type {NextApiRequest, NextApiResponse} from "next";
import { app } from "@/firebase";
import { getFirestore, collection, getDocs } from "firebase/firestore";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {sessionOptions} from "@/lib/session"; import {sessionOptions} from "@/lib/session";
import axios from "axios"; import axios from "axios";
@@ -9,18 +7,14 @@ import { v4 } from "uuid";
import {OrderResponseBody} from "@paypal/paypal-js"; import {OrderResponseBody} from "@paypal/paypal-js";
import {getAccessToken} from "@/utils/paypal"; import {getAccessToken} from "@/utils/paypal";
const db = getFirestore(app);
export default withIronSessionApiRoute(handler, sessionOptions); export default withIronSessionApiRoute(handler, sessionOptions);
async function handler(req: NextApiRequest, res: NextApiResponse) { async function handler(req: NextApiRequest, res: NextApiResponse) {
if (req.method !== "POST") if (req.method !== "POST") return res.status(404).json({ok: false, reason: "Method not supported!"});
return res.status(404).json({ ok: false, reason: "Method not supported!" });
if (!req.session.user) return res.status(401).json({ok: false}); if (!req.session.user) return res.status(401).json({ok: false});
const accessToken = await getAccessToken(); const accessToken = await getAccessToken();
if (!accessToken) if (!accessToken) return res.status(401).json({ok: false, reason: "Authorization failed!"});
return res.status(401).json({ ok: false, reason: "Authorization failed!" });
const {currencyCode, price, trackingId} = req.body as { const {currencyCode, price, trackingId} = req.body as {
currencyCode: string; currencyCode: string;
@@ -28,8 +22,7 @@ async function handler(req: NextApiRequest, res: NextApiResponse) {
trackingId: string; trackingId: string;
}; };
if (!trackingId) if (!trackingId) return res.status(401).json({ok: false, reason: "Missing tracking id!"});
return res.status(401).json({ ok: false, reason: "Missing tracking id!" });
const url = `${process.env.PAYPAL_ACCESS_TOKEN_URL}/v2/checkout/orders`; const url = `${process.env.PAYPAL_ACCESS_TOKEN_URL}/v2/checkout/orders`;
const amount = { const amount = {
@@ -95,7 +88,7 @@ async function handler(req: NextApiRequest, res: NextApiResponse) {
url, url,
data, data,
headers, headers,
}) }),
); );
axios axios

View File

@@ -1,7 +1,5 @@
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction // Next.js API route support: https://nextjs.org/docs/api-routes/introduction
import type {NextApiRequest, NextApiResponse} from "next"; import type {NextApiRequest, NextApiResponse} from "next";
import { app } from "@/firebase";
import { getFirestore, collection, getDocs } from "firebase/firestore";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {sessionOptions} from "@/lib/session"; import {sessionOptions} from "@/lib/session";
import axios from "axios"; import axios from "axios";
@@ -9,19 +7,15 @@ import { v4 } from "uuid";
import {OrderResponseBody} from "@paypal/paypal-js"; import {OrderResponseBody} from "@paypal/paypal-js";
import {getAccessToken} from "@/utils/paypal"; import {getAccessToken} from "@/utils/paypal";
const db = getFirestore(app);
export default withIronSessionApiRoute(handler, sessionOptions); export default withIronSessionApiRoute(handler, sessionOptions);
async function handler(req: NextApiRequest, res: NextApiResponse) { async function handler(req: NextApiRequest, res: NextApiResponse) {
if (req.method !== "PUT") if (req.method !== "PUT") return res.status(404).json({ok: false, reason: "Method not supported!"});
return res.status(404).json({ ok: false, reason: "Method not supported!" });
if (!req.session.user) return res.status(401).json({ok: false}); if (!req.session.user) return res.status(401).json({ok: false});
const accessToken = await getAccessToken(); const accessToken = await getAccessToken();
if (!accessToken) if (!accessToken) return res.status(401).json({ok: false, reason: "Authorization failed!"});
return res.status(401).json({ ok: false, reason: "Authorization failed!" });
const trackingId = `${req.session.user.id}-${Date.now()}`; const trackingId = `${req.session.user.id}-${Date.now()}`;
@@ -40,11 +34,13 @@ async function handler(req: NextApiRequest, res: NextApiResponse) {
Authorization: `Bearer ${accessToken}`, Authorization: `Bearer ${accessToken}`,
}, },
}; };
console.log(JSON.stringify({ console.log(
JSON.stringify({
url, url,
data, data,
headers, headers,
})); }),
);
try { try {
const request = await axios.put(url, data, headers); const request = await axios.put(url, data, headers);
@@ -54,8 +50,6 @@ async function handler(req: NextApiRequest, res: NextApiResponse) {
}); });
} catch (err) { } catch (err) {
console.error(url, err); console.error(url, err);
return res return res.status(500).json({ok: false, reason: "Failed to create tracking ID"});
.status(500)
.json({ ok: false, reason: "Failed to create tracking ID" });
} }
} }

View File

@@ -1,25 +1,10 @@
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction // Next.js API route support: https://nextjs.org/docs/api-routes/introduction
import type {NextApiRequest, NextApiResponse} from "next"; import type {NextApiRequest, NextApiResponse} from "next";
import { app } from "@/firebase";
import {
getFirestore,
collection,
getDocs,
query,
where,
doc,
setDoc,
addDoc,
getDoc,
deleteDoc,
} from "firebase/firestore";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {sessionOptions} from "@/lib/session"; import {sessionOptions} from "@/lib/session";
import {Permission} from "@/interfaces/permissions"; import {Permission} from "@/interfaces/permissions";
import {bootstrap} from "@/utils/permissions.be"; import {bootstrap} from "@/utils/permissions.be";
const db = getFirestore(app);
export default withIronSessionApiRoute(handler, sessionOptions); export default withIronSessionApiRoute(handler, sessionOptions);
async function handler(req: NextApiRequest, res: NextApiResponse) { async function handler(req: NextApiRequest, res: NextApiResponse) {

View File

@@ -3,7 +3,6 @@ import {createUserWithEmailAndPassword, getAuth} from "firebase/auth";
import {app} from "@/firebase"; import {app} from "@/firebase";
import {sessionOptions} from "@/lib/session"; import {sessionOptions} from "@/lib/session";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {getFirestore, doc, setDoc, query, collection, where, getDocs} from "firebase/firestore";
import {Code, CorporateInformation, DemographicInformation, Group, Type} from "@/interfaces/user"; import {Code, CorporateInformation, DemographicInformation, Group, Type} from "@/interfaces/user";
import {addUserToGroupOnCreation} from "@/utils/registration"; import {addUserToGroupOnCreation} from "@/utils/registration";
import moment from "moment"; import moment from "moment";

View File

@@ -1,7 +1,6 @@
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction // Next.js API route support: https://nextjs.org/docs/api-routes/introduction
import type {NextApiRequest, NextApiResponse} from "next"; import type {NextApiRequest, NextApiResponse} from "next";
import {app} from "@/firebase"; import {app} from "@/firebase";
import {getFirestore, doc, getDoc, deleteDoc} from "firebase/firestore";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {sessionOptions} from "@/lib/session"; import {sessionOptions} from "@/lib/session";
import {Session} from "@/hooks/useSessions"; import {Session} from "@/hooks/useSessions";

View File

@@ -7,11 +7,7 @@ import formidable from "formidable-serverless";
import {getDownloadURL, ref, uploadBytes} from "firebase/storage"; import {getDownloadURL, ref, uploadBytes} from "firebase/storage";
import fs from "fs"; import fs from "fs";
import {app, storage} from "@/firebase"; import {app, storage} from "@/firebase";
import {doc, getDoc, getFirestore, setDoc} from "firebase/firestore";
import {Stat} from "@/interfaces/user";
import {speakingReverseMarking} from "@/utils/score";
const db = getFirestore(app);
export default withIronSessionApiRoute(handler, sessionOptions); export default withIronSessionApiRoute(handler, sessionOptions);
async function handler(req: NextApiRequest, res: NextApiResponse) { async function handler(req: NextApiRequest, res: NextApiResponse) {

View File

@@ -7,9 +7,11 @@ import {getAuth} from "firebase-admin/auth";
import {withIronSessionApiRoute} from "iron-session/next"; import {withIronSessionApiRoute} from "iron-session/next";
import {NextApiRequest, NextApiResponse} from "next"; import {NextApiRequest, NextApiResponse} from "next";
import {getPermissions, getPermissionDocs} from "@/utils/permissions.be"; import {getPermissions, getPermissionDocs} from "@/utils/permissions.be";
import client from "@/lib/mongodb";
import {getGroupsForUser, getParticipantGroups} from "@/utils/groups.be";
const db = getFirestore(app);
const auth = getAuth(adminApp); const auth = getAuth(adminApp);
const db = client.db(process.env.MONGODB_DB);
export default withIronSessionApiRoute(user, sessionOptions); export default withIronSessionApiRoute(user, sessionOptions);
@@ -28,36 +30,26 @@ async function del(req: NextApiRequest, res: NextApiResponse) {
const {id} = req.query as {id: string}; const {id} = req.query as {id: string};
const docUser = await getDoc(doc(db, "users", req.session.user.id)); const user = await db.collection("users").findOne<User>({id: req.session.user.id});
if (!docUser.exists()) { if (!user) {
res.status(401).json({ok: false}); res.status(401).json({ok: false});
return; return;
} }
const user = docUser.data() as User; const targetUser = await db.collection("users").findOne<User>({id});
if (!targetUser) {
const docTargetUser = await getDoc(doc(db, "users", id));
if (!docTargetUser.exists()) {
res.status(404).json({ok: false}); res.status(404).json({ok: false});
return; return;
} }
const targetUser = {...docTargetUser.data(), id: docTargetUser.id} as User;
if (user.type === "corporate" && (targetUser.type === "student" || targetUser.type === "teacher")) { if (user.type === "corporate" && (targetUser.type === "student" || targetUser.type === "teacher")) {
const userParticipantGroup = await getDocs(query(collection(db, "groups"), where("participants", "array-contains", id))); const groups = await getGroupsForUser(user.id, targetUser.id);
await Promise.all([ await Promise.all([
...userParticipantGroup.docs ...groups
.filter((x) => (x.data() as Group).admin === user.id) .filter((x) => x.admin === user.id)
.map( .map(
async (x) => async (x) =>
await setDoc( await db.collection("groups").updateOne({id: x.id}, {$set: {participants: x.participants.filter((y: string) => y !== id)}}),
x.ref,
{
participants: x.data().participants.filter((y: string) => y !== id),
},
{merge: true},
),
), ),
]); ]);
@@ -65,41 +57,30 @@ async function del(req: NextApiRequest, res: NextApiResponse) {
} }
await auth.deleteUser(id); await auth.deleteUser(id);
await deleteDoc(doc(db, "users", id)); await db.collection("users").deleteOne({id: targetUser.id});
const userCodeDocs = await getDocs(query(collection(db, "codes"), where("userId", "==", id))); await db.collection("codes").deleteMany({userId: targetUser.id});
const userParticipantGroup = await getDocs(query(collection(db, "groups"), where("participants", "array-contains", id))); await db.collection("groups").deleteMany({admin: targetUser.id});
const userGroupAdminDocs = await getDocs(query(collection(db, "groups"), where("admin", "==", id))); await db.collection("stats").deleteMany({user: targetUser.id});
const userStatsDocs = await getDocs(query(collection(db, "stats"), where("user", "==", id)));
await Promise.all([ const groups = await getParticipantGroups(targetUser.id);
...userCodeDocs.docs.map(async (x) => await deleteDoc(x.ref)), await Promise.all(
...userGroupAdminDocs.docs.map(async (x) => await deleteDoc(x.ref)), groups.map(
...userStatsDocs.docs.map(async (x) => await deleteDoc(x.ref)), async (x) => await db.collection("groups").updateOne({id: x.id}, {$set: {participants: x.participants.filter((y: string) => y !== id)}}),
...userParticipantGroup.docs.map(
async (x) =>
await setDoc(
x.ref,
{
participants: x.data().participants.filter((y: string) => y !== id),
},
{merge: true},
), ),
), );
]);
res.json({ok: true}); res.json({ok: true});
} }
async function get(req: NextApiRequest, res: NextApiResponse) { async function get(req: NextApiRequest, res: NextApiResponse) {
if (req.session.user) { if (req.session.user) {
const docUser = await getDoc(doc(db, "users", req.session.user.id)); const user = await db.collection("users").findOne<User>({id: req.session.user.id});
if (!docUser.exists()) { if (!user) {
res.status(401).json(undefined); res.status(401).json(undefined);
return; return;
} }
const user = docUser.data() as User; await db.collection("users").updateOne({id: user.id}, {$set: {lastLogin: new Date().toISOString()}});
await setDoc(docUser.ref, {lastLogin: new Date().toISOString()}, {merge: true});
req.session.user = { req.session.user = {
...user, ...user,

View File

@@ -1,4 +1,3 @@
import {collection, doc, getDoc, getDocs, getFirestore, query, setDoc, where} from "firebase/firestore";
import {app} from "@/firebase"; import {app} from "@/firebase";
import {Group, Type, User} from "@/interfaces/user"; import {Group, Type, User} from "@/interfaces/user";
import {uuidv4} from "@firebase/util"; import {uuidv4} from "@firebase/util";

View File

@@ -1,22 +1,3 @@
import {app} from "@/firebase";
import {
collection,
doc,
documentId,
endAt,
endBefore,
getCountFromServer,
getDoc,
getDocs,
getFirestore,
limit,
orderBy,
query,
startAfter,
startAt,
where,
} from "firebase/firestore";
import {CorporateUser, Group, Type, User} from "@/interfaces/user"; import {CorporateUser, Group, Type, User} from "@/interfaces/user";
import {getGroupsForUser, getParticipantGroups, getUserGroups, getUsersGroups} from "./groups.be"; import {getGroupsForUser, getParticipantGroups, getUserGroups, getUsersGroups} from "./groups.be";
import {last, uniq, uniqBy} from "lodash"; import {last, uniq, uniqBy} from "lodash";