Implement admin controls
This commit is contained in:
parent
5e43925633
commit
f582ae5a6c
27
functions/package-lock.json
generated
27
functions/package-lock.json
generated
@ -6,8 +6,9 @@
|
|||||||
"": {
|
"": {
|
||||||
"name": "functions",
|
"name": "functions",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"cors": "^2.8.5",
|
||||||
"firebase-admin": "^10.0.2",
|
"firebase-admin": "^10.0.2",
|
||||||
"firebase-functions": "^3.18.0"
|
"firebase-functions": "^4.0.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@typescript-eslint/eslint-plugin": "^5.12.0",
|
"@typescript-eslint/eslint-plugin": "^5.12.0",
|
||||||
@ -1972,25 +1973,24 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/firebase-functions": {
|
"node_modules/firebase-functions": {
|
||||||
"version": "3.24.1",
|
"version": "4.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/firebase-functions/-/firebase-functions-3.24.1.tgz",
|
"resolved": "https://registry.npmjs.org/firebase-functions/-/firebase-functions-4.0.1.tgz",
|
||||||
"integrity": "sha512-GYhoyOV0864HFMU1h/JNBXYNmDk2MlbvU7VO/5qliHX6u/6vhSjTJjlyCG4leDEI8ew8IvmkIC5QquQ1U8hAuA==",
|
"integrity": "sha512-U0dOqGPShLi0g3jUlZ3aZlVTPFO9cREJfIxMJIlfRz/vNbYoKdIVdI7OAS9RKPcqz99zxkN/A8Ro4kjI+ytT8A==",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@types/cors": "^2.8.5",
|
"@types/cors": "^2.8.5",
|
||||||
"@types/express": "4.17.3",
|
"@types/express": "4.17.3",
|
||||||
"cors": "^2.8.5",
|
"cors": "^2.8.5",
|
||||||
"express": "^4.17.1",
|
"express": "^4.17.1",
|
||||||
"lodash": "^4.17.14",
|
|
||||||
"node-fetch": "^2.6.7"
|
"node-fetch": "^2.6.7"
|
||||||
},
|
},
|
||||||
"bin": {
|
"bin": {
|
||||||
"firebase-functions": "lib/bin/firebase-functions.js"
|
"firebase-functions": "lib/bin/firebase-functions.js"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": "^8.13.0 || >=10.10.0"
|
"node": ">=14.10.0"
|
||||||
},
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"firebase-admin": "^8.0.0 || ^9.0.0 || ^10.0.0 || ^11.0.0"
|
"firebase-admin": "^10.0.0 || ^11.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/firebase-functions-test": {
|
"node_modules/firebase-functions-test": {
|
||||||
@ -2952,7 +2952,8 @@
|
|||||||
"node_modules/lodash": {
|
"node_modules/lodash": {
|
||||||
"version": "4.17.21",
|
"version": "4.17.21",
|
||||||
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
|
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
|
||||||
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="
|
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==",
|
||||||
|
"dev": true
|
||||||
},
|
},
|
||||||
"node_modules/lodash.camelcase": {
|
"node_modules/lodash.camelcase": {
|
||||||
"version": "4.3.0",
|
"version": "4.3.0",
|
||||||
@ -5960,15 +5961,14 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"firebase-functions": {
|
"firebase-functions": {
|
||||||
"version": "3.24.1",
|
"version": "4.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/firebase-functions/-/firebase-functions-3.24.1.tgz",
|
"resolved": "https://registry.npmjs.org/firebase-functions/-/firebase-functions-4.0.1.tgz",
|
||||||
"integrity": "sha512-GYhoyOV0864HFMU1h/JNBXYNmDk2MlbvU7VO/5qliHX6u/6vhSjTJjlyCG4leDEI8ew8IvmkIC5QquQ1U8hAuA==",
|
"integrity": "sha512-U0dOqGPShLi0g3jUlZ3aZlVTPFO9cREJfIxMJIlfRz/vNbYoKdIVdI7OAS9RKPcqz99zxkN/A8Ro4kjI+ytT8A==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"@types/cors": "^2.8.5",
|
"@types/cors": "^2.8.5",
|
||||||
"@types/express": "4.17.3",
|
"@types/express": "4.17.3",
|
||||||
"cors": "^2.8.5",
|
"cors": "^2.8.5",
|
||||||
"express": "^4.17.1",
|
"express": "^4.17.1",
|
||||||
"lodash": "^4.17.14",
|
|
||||||
"node-fetch": "^2.6.7"
|
"node-fetch": "^2.6.7"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -6687,7 +6687,8 @@
|
|||||||
"lodash": {
|
"lodash": {
|
||||||
"version": "4.17.21",
|
"version": "4.17.21",
|
||||||
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
|
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
|
||||||
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="
|
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==",
|
||||||
|
"dev": true
|
||||||
},
|
},
|
||||||
"lodash.camelcase": {
|
"lodash.camelcase": {
|
||||||
"version": "4.3.0",
|
"version": "4.3.0",
|
||||||
|
@ -15,8 +15,9 @@
|
|||||||
},
|
},
|
||||||
"main": "lib/index.js",
|
"main": "lib/index.js",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"cors": "^2.8.5",
|
||||||
"firebase-admin": "^10.0.2",
|
"firebase-admin": "^10.0.2",
|
||||||
"firebase-functions": "^3.18.0"
|
"firebase-functions": "^4.0.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@typescript-eslint/eslint-plugin": "^5.12.0",
|
"@typescript-eslint/eslint-plugin": "^5.12.0",
|
||||||
|
@ -1,163 +1,10 @@
|
|||||||
import * as functions from "firebase-functions";
|
import * as functions from "firebase-functions";
|
||||||
import {initializeApp} from "firebase-admin/app";
|
import {initializeApp} from "firebase-admin/app";
|
||||||
import {
|
|
||||||
DocumentReference,
|
|
||||||
DocumentSnapshot,
|
|
||||||
getFirestore,
|
|
||||||
QuerySnapshot,
|
|
||||||
} from "firebase-admin/firestore";
|
|
||||||
import express, {Request, Response} from "express";
|
|
||||||
|
|
||||||
type VoteEntry = {
|
|
||||||
username: string
|
|
||||||
voteIndex: number
|
|
||||||
}
|
|
||||||
|
|
||||||
type Vote = {
|
|
||||||
prompt: string
|
|
||||||
options: Array<string>
|
|
||||||
}
|
|
||||||
|
|
||||||
initializeApp();
|
initializeApp();
|
||||||
|
|
||||||
|
import votingRoute from "./route/voting";
|
||||||
|
import adminRoute from "./route/admin";
|
||||||
|
|
||||||
const db = getFirestore();
|
export const vote = functions.https.onRequest(votingRoute);
|
||||||
const app = express();
|
export const admin = functions.https.onRequest(adminRoute);
|
||||||
|
|
||||||
const getVote = async (id: string) =>
|
|
||||||
db.collection("votes")
|
|
||||||
.doc(id)
|
|
||||||
.get() as Promise<DocumentSnapshot<Vote>>;
|
|
||||||
const getVoteEntry = async (vote: DocumentReference<Vote>, username: string) =>
|
|
||||||
vote.collection("entries")
|
|
||||||
.where("username", "==", username)
|
|
||||||
.get() as Promise<QuerySnapshot<VoteEntry>>;
|
|
||||||
const makeVoteEntry = async (
|
|
||||||
vote: DocumentReference<Vote>,
|
|
||||||
username: string,
|
|
||||||
voteIndex: number
|
|
||||||
) =>
|
|
||||||
vote.collection("entries")
|
|
||||||
.add({username, voteIndex}) as Promise<DocumentReference<VoteEntry>>;
|
|
||||||
|
|
||||||
app.post("/", async (req: Request, res: Response) => {
|
|
||||||
const voteId = req.body.voteId as string | undefined;
|
|
||||||
const voter = req.body.voter as string | undefined;
|
|
||||||
const voteIndex = parseInt((req.body.voteIndex as string | undefined) ?? "");
|
|
||||||
|
|
||||||
if (!voteId) {
|
|
||||||
res.status(400).json({error: "Missing voteId"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!voter) {
|
|
||||||
res.status(400).json({error: "Missing voter"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const vote = await getVote(voteId);
|
|
||||||
if (!vote.exists) {
|
|
||||||
res.status(404).json({error: "Vote not found"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
Number.isNaN(voteIndex) ||
|
|
||||||
voteIndex < 0 ||
|
|
||||||
voteIndex >= (vote.data()?.options ?? []).length
|
|
||||||
) {
|
|
||||||
res.status(400).json({error: "Invalid vote index"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const entry = await getVoteEntry(vote.ref, voter);
|
|
||||||
if (entry.empty) {
|
|
||||||
await makeVoteEntry(vote.ref, voter, voteIndex);
|
|
||||||
res.json({success: true});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
entry.docs[0].ref.update({voteIndex});
|
|
||||||
res.json({success: true});
|
|
||||||
});
|
|
||||||
|
|
||||||
app.get("/", async (req: Request, res: Response) => {
|
|
||||||
const voteId = req.query.voteId as string | undefined;
|
|
||||||
if (!voteId) {
|
|
||||||
res.status(400).json({error: "Missing voteId"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const vote = await getVote(voteId);
|
|
||||||
if (!vote.exists) {
|
|
||||||
res.status(404).json({error: "Vote not found"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
res.json(vote.data());
|
|
||||||
});
|
|
||||||
|
|
||||||
app.get("/entries", async (req: Request, res: Response) => {
|
|
||||||
const voteId = req.query.voteId as string | undefined;
|
|
||||||
const voteIndexStr = req.query.voteIndex as string | undefined;
|
|
||||||
const voteIndex = parseInt(voteIndexStr ?? "");
|
|
||||||
if (!voteId) {
|
|
||||||
res.status(400).json({error: "Missing voteId"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Number.isNaN(voteIndex) && voteIndexStr) {
|
|
||||||
res.status(400).json({error: "Invalid voteIndex"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const vote = await getVote(voteId);
|
|
||||||
if (!vote.exists) {
|
|
||||||
res.status(404).json({error: "Vote not found"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
!Number.isNaN(voteIndex) &&
|
|
||||||
(voteIndex < 0 || voteIndex >= (vote.data()?.options ?? []).length)
|
|
||||||
) {
|
|
||||||
res.status(400).json({error: "Invalid vote index"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const entryCollection = vote.ref.collection("entries");
|
|
||||||
const entries = await (
|
|
||||||
Number.isNaN(voteIndex) ?
|
|
||||||
entryCollection :
|
|
||||||
entryCollection.where("voteIndex", "==", voteIndex)
|
|
||||||
).get();
|
|
||||||
res.json(entries.docs.map((d) => d.data()));
|
|
||||||
});
|
|
||||||
|
|
||||||
app.get("/count", async (req: Request, res: Response) => {
|
|
||||||
const voteId = req.query.voteId as string | undefined;
|
|
||||||
|
|
||||||
if (!voteId) {
|
|
||||||
res.status(400).json({error: "Missing voteId"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const vote = await getVote(voteId);
|
|
||||||
if (!vote.exists) {
|
|
||||||
res.status(404).json({error: "Vote not found"});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const collect = new Array<Promise<QuerySnapshot<VoteEntry>>>();
|
|
||||||
for (let i = 0; i < (vote.data()?.options ?? []).length; i++) {
|
|
||||||
collect.push(vote.ref
|
|
||||||
.collection("entries")
|
|
||||||
.where("voteIndex", "==", i)
|
|
||||||
.get() as Promise<QuerySnapshot<VoteEntry>>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
res.json((await Promise.all(collect)).map((query) => query.size));
|
|
||||||
});
|
|
||||||
|
|
||||||
export const vote = functions.https.onRequest(app);
|
|
||||||
|
48
functions/src/route/admin.ts
Normal file
48
functions/src/route/admin.ts
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
import express, {Request, Response} from "express";
|
||||||
|
import cors from "cors";
|
||||||
|
import {createVote, setActiveVote} from "../types/vote";
|
||||||
|
|
||||||
|
const app = express();
|
||||||
|
|
||||||
|
app.use(cors());
|
||||||
|
|
||||||
|
app.post("/create", async (req: Request, res: Response) => {
|
||||||
|
const prompt = req.body.prompt as string | undefined;
|
||||||
|
const options = req.body.options as Array<string> | undefined;
|
||||||
|
|
||||||
|
if (!prompt) {
|
||||||
|
res.status(400).json({error: "Missing prompt"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!options || options.length < 2) {
|
||||||
|
res.status(400).json({error: "Missing options"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
res.json({id: await createVote(prompt, options)});
|
||||||
|
});
|
||||||
|
|
||||||
|
app.put("/setActive", async (req: Request, res: Response) => {
|
||||||
|
const voteId = req.body.voteId as string | undefined;
|
||||||
|
|
||||||
|
if (!voteId) {
|
||||||
|
res.status(400).json({error: "Missing voteId"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
res.json({success: await setActiveVote(voteId)});
|
||||||
|
});
|
||||||
|
|
||||||
|
app.put("/closeVote", async (req: Request, res: Response) => {
|
||||||
|
const voteId = req.body.voteId as string | undefined;
|
||||||
|
|
||||||
|
if (!voteId) {
|
||||||
|
res.status(400).json({error: "Missing voteId"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
res.json({success: await setActiveVote(voteId, false)});
|
||||||
|
});
|
||||||
|
|
||||||
|
export default app;
|
125
functions/src/route/voting.ts
Normal file
125
functions/src/route/voting.ts
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
import express, {Request, Response} from "express";
|
||||||
|
import {
|
||||||
|
getEntriesFromSnapshot,
|
||||||
|
getVoteSnapshot,
|
||||||
|
getVoteCounts,
|
||||||
|
getVoteEntry,
|
||||||
|
makeVoteEntry,
|
||||||
|
updateVoteEntry,
|
||||||
|
getVote,
|
||||||
|
getActiveVote,
|
||||||
|
} from "../types/vote";
|
||||||
|
|
||||||
|
|
||||||
|
const app = express();
|
||||||
|
|
||||||
|
app.post("/", async (req: Request, res: Response) => {
|
||||||
|
const voteId = req.body.voteId as string | undefined;
|
||||||
|
const voter = req.body.voter as string | undefined;
|
||||||
|
const voteIndex = parseInt((req.body.voteIndex as string | undefined) ?? "");
|
||||||
|
|
||||||
|
if (!voteId) {
|
||||||
|
res.status(400).json({error: "Missing voteId"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!voter) {
|
||||||
|
res.status(400).json({error: "Missing voter"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const vote = await getVoteSnapshot(voteId);
|
||||||
|
if (!vote.exists) {
|
||||||
|
res.status(404).json({error: "Vote not found"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
Number.isNaN(voteIndex) ||
|
||||||
|
voteIndex < 0 ||
|
||||||
|
voteIndex >= (vote.data()?.options ?? []).length
|
||||||
|
) {
|
||||||
|
res.status(400).json({error: "Invalid vote index"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const entry = await getVoteEntry(vote.ref, voter);
|
||||||
|
if (!entry) {
|
||||||
|
await makeVoteEntry(vote.ref, voter, voteIndex);
|
||||||
|
res.json({success: true});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await updateVoteEntry(entry, voteIndex);
|
||||||
|
res.json({success: true});
|
||||||
|
});
|
||||||
|
|
||||||
|
app.get("/", async (req: Request, res: Response) => {
|
||||||
|
const voteId = req.query.voteId as string | undefined;
|
||||||
|
if (!voteId) {
|
||||||
|
res.status(400).json({error: "Missing voteId"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const vote = await getVote(voteId);
|
||||||
|
if (!vote) {
|
||||||
|
res.status(404).json({error: "Vote not found"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
res.json(vote);
|
||||||
|
});
|
||||||
|
|
||||||
|
app.get("/entries", async (req: Request, res: Response) => {
|
||||||
|
const voteId = req.query.voteId as string | undefined;
|
||||||
|
const voteIndexStr = req.query.voteIndex as string | undefined;
|
||||||
|
const voteIndex = parseInt(voteIndexStr ?? "");
|
||||||
|
if (!voteId) {
|
||||||
|
res.status(400).json({error: "Missing voteId"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Number.isNaN(voteIndex) && voteIndexStr) {
|
||||||
|
res.status(400).json({error: "Invalid voteIndex"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const vote = await getVoteSnapshot(voteId);
|
||||||
|
if (!vote.exists) {
|
||||||
|
res.status(404).json({error: "Vote not found"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
!Number.isNaN(voteIndex) &&
|
||||||
|
(voteIndex < 0 || voteIndex >= (vote.data()?.options ?? []).length)
|
||||||
|
) {
|
||||||
|
res.status(400).json({error: "Invalid vote index"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
res.json(await getEntriesFromSnapshot(vote, voteIndex));
|
||||||
|
});
|
||||||
|
|
||||||
|
app.get("/count", async (req: Request, res: Response) => {
|
||||||
|
const voteId = req.query.voteId as string | undefined;
|
||||||
|
|
||||||
|
if (!voteId) {
|
||||||
|
res.status(400).json({error: "Missing voteId"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const vote = await getVoteSnapshot(voteId);
|
||||||
|
if (!vote.exists) {
|
||||||
|
res.status(404).json({error: "Vote not found"});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
res.json((await Promise.all(getVoteCounts(vote))).map((query) => query.size));
|
||||||
|
});
|
||||||
|
|
||||||
|
app.get("/active", async (req: Request, res: Response) => {
|
||||||
|
res.json({id: (await getActiveVote())?.id});
|
||||||
|
});
|
||||||
|
|
||||||
|
export default app;
|
104
functions/src/types/vote.ts
Normal file
104
functions/src/types/vote.ts
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
import {
|
||||||
|
DocumentReference,
|
||||||
|
DocumentSnapshot,
|
||||||
|
getFirestore,
|
||||||
|
QuerySnapshot,
|
||||||
|
} from "firebase-admin/firestore";
|
||||||
|
|
||||||
|
const db = getFirestore();
|
||||||
|
const votesCollection = db.collection("votes");
|
||||||
|
const entriesCollectionName = "entries";
|
||||||
|
const usernameField = "username";
|
||||||
|
const voteIndexField = "voteIndex";
|
||||||
|
const optionsField = "options";
|
||||||
|
const activeField = "active";
|
||||||
|
|
||||||
|
export type VoteEntry = {
|
||||||
|
[usernameField]: string
|
||||||
|
[voteIndexField]: number
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Vote = {
|
||||||
|
prompt: string
|
||||||
|
[optionsField]: Array<string>
|
||||||
|
[activeField]?: boolean
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getEntriesCollection = (vote: DocumentReference<Vote>) =>
|
||||||
|
vote.collection(entriesCollectionName);
|
||||||
|
|
||||||
|
|
||||||
|
export const getVoteSnapshot = async (id: string) =>
|
||||||
|
votesCollection
|
||||||
|
.doc(id)
|
||||||
|
.get() as Promise<DocumentSnapshot<Vote>>;
|
||||||
|
export const getVote = async (id: string) => (await getVoteSnapshot(id)).data();
|
||||||
|
export const getVoteEntry = async (
|
||||||
|
vote: DocumentReference<Vote>,
|
||||||
|
username: string
|
||||||
|
) =>
|
||||||
|
(await getEntriesCollection(vote)
|
||||||
|
.where(usernameField, "==", username)
|
||||||
|
.get() as QuerySnapshot<VoteEntry>).docs[0]?.ref;
|
||||||
|
export const updateVoteEntry = async (
|
||||||
|
vote: DocumentReference<VoteEntry>,
|
||||||
|
voteIndex: number
|
||||||
|
) =>
|
||||||
|
vote.update({voteIndex});
|
||||||
|
export const makeVoteEntry = async (
|
||||||
|
vote: DocumentReference<Vote>,
|
||||||
|
username: string,
|
||||||
|
voteIndex: number
|
||||||
|
) =>
|
||||||
|
vote.collection(entriesCollectionName)
|
||||||
|
.add({username, voteIndex}) as Promise<DocumentReference<VoteEntry>>;
|
||||||
|
|
||||||
|
export const getVoteCounts = (
|
||||||
|
vote: DocumentSnapshot<Vote>
|
||||||
|
): Array<Promise<QuerySnapshot<VoteEntry>>> => {
|
||||||
|
const collect = new Array<Promise<QuerySnapshot<VoteEntry>>>();
|
||||||
|
for (let i = 0; i < (vote.data()?.options ?? []).length; i++) {
|
||||||
|
collect.push(getEntriesCollection(vote.ref)
|
||||||
|
.where(voteIndexField, "==", i)
|
||||||
|
.get() as Promise<QuerySnapshot<VoteEntry>>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return collect;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getEntriesFromSnapshot = async (
|
||||||
|
snapshot: DocumentSnapshot<Vote>,
|
||||||
|
voteIndex: number
|
||||||
|
): Promise<VoteEntry[]> => {
|
||||||
|
const entryCollection = getEntriesCollection(snapshot.ref);
|
||||||
|
return (await (
|
||||||
|
Number.isNaN(voteIndex) ?
|
||||||
|
entryCollection :
|
||||||
|
entryCollection.where(voteIndexField, "==", voteIndex)
|
||||||
|
).get()
|
||||||
|
).docs.map((doc) => doc.data() as VoteEntry);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createVote = async (prompt: string, options: Array<string>) => {
|
||||||
|
const vote = await votesCollection.add({prompt, options});
|
||||||
|
return vote.id;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getActiveVote = async () =>
|
||||||
|
(await votesCollection.where(activeField, "==", true)
|
||||||
|
.get()).docs[0]?.ref as DocumentReference<Vote> | undefined;
|
||||||
|
export const setActiveVote = async (id: string, active = true) => {
|
||||||
|
const activeVote = await getActiveVote();
|
||||||
|
if (activeVote) {
|
||||||
|
await activeVote.update({active: false});
|
||||||
|
}
|
||||||
|
|
||||||
|
const vote = await getVoteSnapshot(id);
|
||||||
|
if (!vote.exists) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
await vote.ref.update({active});
|
||||||
|
return true;
|
||||||
|
};
|
Loading…
x
Reference in New Issue
Block a user