diff --git a/firestore.rules b/firestore.rules index c38e3ae..c2277a2 100644 --- a/firestore.rules +++ b/firestore.rules @@ -1,6 +1,10 @@ rules_version = '2'; service cloud.firestore { match /databases/{database}/documents { + match /state/{document=**} { + allow read, write: if true; + } + match /{document=**} { allow read, write: if false; } diff --git a/functions/src/index.ts b/functions/src/index.ts index 15b9dc5..79c2ac3 100644 --- a/functions/src/index.ts +++ b/functions/src/index.ts @@ -4,8 +4,9 @@ import {initializeApp} from "firebase-admin/app"; initializeApp(); import votingRoute, {activeVoteState} from "./route/voting"; -import adminRoute from "./route/admin"; +import adminRoute, {getAllVotesCall} from "./route/admin"; export const vote = functions.https.onRequest(votingRoute); export const admin = functions.https.onRequest(adminRoute); export const activeVote = activeVoteState; +export const getAllVotes = getAllVotesCall; diff --git a/functions/src/route/admin.ts b/functions/src/route/admin.ts index bdd0bd5..16a3ac6 100644 --- a/functions/src/route/admin.ts +++ b/functions/src/route/admin.ts @@ -1,6 +1,8 @@ +import * as functions from "firebase-functions"; import express, {Request, Response} from "express"; import cors from "cors"; -import {createVote, setActiveVote} from "../types/vote"; +import {createVote, setActiveVote, getAllVotes} from "../types/vote"; +import { setState, updateState } from "../types/state"; const app = express(); @@ -26,7 +28,12 @@ app.post("/create", async (req: Request, res: Response) => { return; } - res.json({id: await createVote(prompt, description, options)}); + const id = await createVote(prompt, description, options); + await setState(id); + + console.log("set state") + + res.json({id}); }); app.put("/setActive", async (req: Request, res: Response) => { @@ -37,7 +44,12 @@ app.put("/setActive", async (req: Request, res: Response) => { return; } - res.json({success: await setActiveVote(voteId)}); + const success = await setActiveVote(voteId); + if (success) { + await updateState(); + } + + res.json({success}); }); app.put("/closeVote", async (req: Request, res: Response) => { @@ -48,7 +60,19 @@ app.put("/closeVote", async (req: Request, res: Response) => { return; } - res.json({success: await setActiveVote(voteId, false)}); + const success = await setActiveVote(voteId, false); + if (success) { + await updateState(); + } + + res.json({success}); }); +export const getAllVotesCall = functions.https.onCall(async () => + (await getAllVotes()).docs.map((vote) => ({ + ...vote.data(), + id: vote.id, + })) +); + export default app; diff --git a/functions/src/route/voting.ts b/functions/src/route/voting.ts index 044ae70..bbfcbd6 100644 --- a/functions/src/route/voting.ts +++ b/functions/src/route/voting.ts @@ -10,6 +10,7 @@ import { getVote, getActiveVote, } from "../types/vote"; +import { updateVoteCount } from "../types/state"; const app = express(); @@ -47,11 +48,13 @@ app.post("/", async (req: Request, res: Response) => { const entry = await getVoteEntry(vote.ref, voter); if (!entry) { await makeVoteEntry(vote.ref, voter, voteIndex); + await updateVoteCount(); res.json({success: true}); return; } await updateVoteEntry(entry, voteIndex); + await updateVoteCount(); res.json({success: true}); }); diff --git a/functions/src/types/state.ts b/functions/src/types/state.ts new file mode 100644 index 0000000..6173c4e --- /dev/null +++ b/functions/src/types/state.ts @@ -0,0 +1,27 @@ +import { + FieldValue, + getFirestore, + } from "firebase-admin/firestore"; + + const db = getFirestore(); + +export type State = { + currentVote: string | null + voteChanges: number + changes: number +}; + +const stateCollection = db.collection("state"); + +export const setState = (currentVote: string | null) => + stateCollection.doc("currentVote").set({ + currentVote, + changes: 0, + voteChanges: 0 + }); + +export const updateState = () => + stateCollection.doc("currentVote").update({changes: FieldValue.increment(1)}); + +export const updateVoteCount = () => + stateCollection.doc("currentVote").update({voteChanges: FieldValue.increment(1)}); \ No newline at end of file diff --git a/functions/src/types/vote.ts b/functions/src/types/vote.ts index b2cf027..7630f40 100644 --- a/functions/src/types/vote.ts +++ b/functions/src/types/vote.ts @@ -28,6 +28,7 @@ export type Vote = { export const getEntriesCollection = (vote: DocumentReference) => vote.collection(entriesCollectionName); +export const getAllVotes = async () => await votesCollection.get(); export const getVoteSnapshot = async (id: string) => votesCollection diff --git a/public/controls.html b/public/controls.html index bc0fe7f..1ac454c 100644 --- a/public/controls.html +++ b/public/controls.html @@ -23,6 +23,13 @@

Controls

- - -
-
-
- - - - -
-
- - - - -
-
- - - - -
-
- - - - -
-
- - - - -
-
- - - - -
- -
+
+ +
+
+
+ + + + +
+
+ + + + +
+
+ + + + +
+
+ + + + +
+
+ + + + +
+
+ + + + +
+ +
+
+
+
    + +
+