Skip to content

Commit 583377b

Browse files
authored
release
1 parent d1a2de2 commit 583377b

27 files changed

+5717
-1
lines changed

.env

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
NEXT_PUBLIC_FB_APIKEY=
2+
NEXT_PUBLIC_FB_AUTHDOMAIN=
3+
NEXT_PUBLIC_FB_PROJECTID=
4+
NEXT_PUBLIC_FB_STORAGEBUCKET=
5+
NEXT_PUBLIC_FB_MESSAGINGSENDERID=
6+
NEXT_PUBLIC_FB_APPID=
7+
FB_ADMIN=

.gitignore

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
2+
3+
# dependencies
4+
/node_modules
5+
/.pnp
6+
.pnp.*
7+
.yarn/*
8+
!.yarn/patches
9+
!.yarn/plugins
10+
!.yarn/releases
11+
!.yarn/versions
12+
13+
# testing
14+
/coverage
15+
16+
# next.js
17+
/.next/
18+
/out/
19+
20+
# production
21+
/build
22+
23+
# misc
24+
.DS_Store
25+
*.pem
26+
27+
# debug
28+
npm-debug.log*
29+
yarn-debug.log*
30+
yarn-error.log*
31+
.pnpm-debug.log*
32+
33+
# env files (can opt-in for committing if needed)
34+
.env*
35+
36+
# vercel
37+
.vercel
38+
39+
# typescript
40+
*.tsbuildinfo
41+
next-env.d.ts

README.md

+40-1
Original file line numberDiff line numberDiff line change
@@ -1 +1,40 @@
1-
# nextjs-dec24
1+
This is a [Next.js](https://nextjs.org) project bootstrapped with [`create-next-app`](https://nextjs.org/docs/app/api-reference/cli/create-next-app).
2+
3+
Refer to the docs in [Notion](https://buildingblocs.notion.site/Next-js-docs-Server-bd7b1eed636b4f7eacdfc28885fc5460) if you want to recreate the app from scratch, or if you want to do it client-sided.
4+
5+
## Getting Started
6+
7+
Add your Firebase credentials in `.env`
8+
9+
Then, run the development server:
10+
11+
```bash
12+
npm run dev
13+
# or
14+
yarn dev
15+
# or
16+
pnpm dev
17+
# or
18+
bun dev
19+
```
20+
21+
Open [http://localhost:3000](http://localhost:3000) with your browser to see the result.
22+
23+
You can start editing the page by modifying `app/page.tsx`. The page auto-updates as you edit the file.
24+
25+
This project uses [`next/font`](https://nextjs.org/docs/app/building-your-application/optimizing/fonts) to automatically optimize and load [Geist](https://vercel.com/font), a new font family for Vercel.
26+
27+
## Learn More
28+
29+
To learn more about Next.js, take a look at the following resources:
30+
31+
- [Next.js Documentation](https://nextjs.org/docs) - learn about Next.js features and API.
32+
- [Learn Next.js](https://nextjs.org/learn) - an interactive Next.js tutorial.
33+
34+
You can check out [the Next.js GitHub repository](https://github.com/vercel/next.js) - your feedback and contributions are welcome!
35+
36+
## Deploy on Vercel
37+
38+
The easiest way to deploy your Next.js app is to use the [Vercel Platform](https://vercel.com/new?utm_medium=default-template&filter=next.js&utm_source=create-next-app&utm_campaign=create-next-app-readme) from the creators of Next.js.
39+
40+
Check out our [Next.js deployment documentation](https://nextjs.org/docs/app/building-your-application/deploying) for more details.

app/actions/createData.ts

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
"use server";
2+
3+
import { getFirebaseAdmin } from "@/app/actions/firebase";
4+
import { cookies } from "next/headers";
5+
6+
export async function createData(value: string) {
7+
const { adminFirestore, adminAuth } = await getFirebaseAdmin();
8+
const cookieStore = await cookies();
9+
10+
const token = cookieStore.get("token")?.value;
11+
if (token) {
12+
try {
13+
const decodedToken = await adminAuth.verifyIdToken(token);
14+
if (decodedToken?.email) {
15+
const docRef = adminFirestore.collection("users").doc(decodedToken.email);
16+
await docRef.update({
17+
[Math.random().toString(36).slice(2, 4)]: {
18+
name: value,
19+
done: false
20+
}
21+
});
22+
return 'OK';
23+
}
24+
} catch (e) {
25+
console.error(e);
26+
return { status: 401 };
27+
}
28+
}
29+
return { status: 403 };
30+
}
31+

app/actions/deleteData.ts

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
"use server";
2+
3+
import { getFirebaseAdmin } from "@/app/actions/firebase";
4+
import { FieldValue } from "firebase-admin/firestore";
5+
import { cookies } from "next/headers";
6+
7+
export async function deleteData(id: string) {
8+
const { adminFirestore, adminAuth } = await getFirebaseAdmin();
9+
const cookieStore = await cookies();
10+
11+
const token = cookieStore.get("token")?.value;
12+
if (token) {
13+
try {
14+
const decodedToken = await adminAuth.verifyIdToken(token);
15+
if (decodedToken?.email) {
16+
const docRef = adminFirestore.collection("users").doc(decodedToken.email);
17+
await docRef.update({
18+
[id]: FieldValue.delete(),
19+
});
20+
return 'OK';
21+
}
22+
} catch (e) {
23+
console.error(e);
24+
return { status: 401 };
25+
}
26+
}
27+
return { status: 403 };
28+
}

app/actions/fetchData.ts

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
"use server";
2+
3+
import { getFirebaseAdmin } from "@/app/actions/firebase";
4+
import { cookies } from "next/headers";
5+
6+
export async function fetchData() {
7+
const { adminFirestore, adminAuth } = await getFirebaseAdmin();
8+
const cookieStore = await cookies();
9+
const token = cookieStore.get("token")?.value;
10+
if (token) {
11+
try {
12+
const decodedToken = await adminAuth.verifyIdToken(token);
13+
if (decodedToken?.email) {
14+
const docRef = adminFirestore.collection("users").doc(decodedToken.email);
15+
const doc = await docRef.get();
16+
if (!doc.exists) {
17+
await adminFirestore.collection("users").doc(decodedToken.email).set({});
18+
return {};
19+
} else {
20+
return doc.data();
21+
}
22+
}
23+
} catch (e) {
24+
console.error(e);
25+
return {status: 401};
26+
}
27+
}
28+
return {status: 403};
29+
}

app/actions/firebase.ts

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
"use server";
2+
3+
import { getApps, initializeApp } from 'firebase-admin/app';
4+
import admin from 'firebase-admin';
5+
import { getFirestore } from "firebase-admin/firestore";
6+
import { getAuth } from "firebase-admin/auth";
7+
8+
let app: ReturnType<typeof initializeApp> | undefined;
9+
10+
async function initializeFirebaseAdmin() {
11+
if (!getApps().length && process.env.FB_ADMIN) {
12+
app = initializeApp({
13+
credential: admin.credential.cert(JSON.parse(process.env.FB_ADMIN)),
14+
});
15+
}
16+
17+
if (!app) {
18+
throw new Error("Firebase admin does not exist");
19+
}
20+
21+
return {
22+
adminAuth: getAuth(app),
23+
adminFirestore: getFirestore(app)
24+
};
25+
}
26+
27+
export const getFirebaseAdmin = initializeFirebaseAdmin;

app/actions/updateData.ts

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
"use server";
2+
3+
import { getFirebaseAdmin } from "@/app/actions/firebase";
4+
import { cookies } from "next/headers";
5+
6+
export async function updateData(type: "name" | "done", value: string | boolean, id: string) {
7+
const { adminFirestore, adminAuth } = await getFirebaseAdmin();
8+
const cookieStore = await cookies();
9+
10+
const token = cookieStore.get("token")?.value;
11+
if (token) {
12+
try {
13+
const decodedToken = await adminAuth.verifyIdToken(token);
14+
if (decodedToken?.email) {
15+
const docRef = adminFirestore.collection("users").doc(decodedToken.email);
16+
if (type === "name") {
17+
await docRef.update({
18+
[`${id}.name`]: value,
19+
});
20+
} else if (type === "done") {
21+
await docRef.update({
22+
[`${id}.done`]: value,
23+
});
24+
}
25+
return 'OK';
26+
}
27+
} catch (e) {
28+
console.error(e);
29+
return { status: 401 };
30+
}
31+
}
32+
return { status: 403 };
33+
}

app/favicon.ico

25.3 KB
Binary file not shown.

app/firebase.ts

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
// Import the functions you need from the SDKs you need
2+
import { initializeApp } from "firebase/app";
3+
import { getAuth } from "firebase/auth";
4+
import { getFirestore } from "firebase/firestore";
5+
// TODO: Add SDKs for Firebase products that you want to use
6+
// https://firebase.google.com/docs/web/setup#available-libraries
7+
8+
// Your web app's Firebase configuration
9+
const firebaseConfig = {
10+
apiKey: process.env.NEXT_PUBLIC_FB_APIKEY,
11+
authDomain: process.env.NEXT_PUBLIC_FB_AUTHDOMAIN,
12+
projectId: process.env.NEXT_PUBLIC_FB_PROJECTID,
13+
storageBucket: process.env.NEXT_PUBLIC_FB_STORAGEBUCKET,
14+
messagingSenderId: process.env.NEXT_PUBLIC_FB_MESSAGINGSENDERID,
15+
appId: process.env.FB_APPID
16+
};
17+
18+
// Initialize Firebase
19+
const app = initializeApp(firebaseConfig);
20+
const auth = getAuth(app);
21+
const db = getFirestore(app);
22+
23+
export { app, db, auth };

app/globals.css

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
@tailwind base;
2+
@tailwind components;
3+
@tailwind utilities;
4+
5+
:root {
6+
--background: #ffffff;
7+
--foreground: #171717;
8+
}
9+
10+
@media (prefers-color-scheme: dark) {
11+
:root {
12+
--background: #0a0a0a;
13+
--foreground: #ededed;
14+
}
15+
}
16+
17+
body {
18+
color: var(--foreground);
19+
background: var(--background);
20+
font-family: Arial, Helvetica, sans-serif;
21+
}

app/layout.tsx

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
import type { Metadata } from "next";
2+
import { Geist, Geist_Mono } from "next/font/google";
3+
import "./globals.css";
4+
5+
const geistSans = Geist({
6+
variable: "--font-geist-sans",
7+
subsets: ["latin"],
8+
});
9+
10+
const geistMono = Geist_Mono({
11+
variable: "--font-geist-mono",
12+
subsets: ["latin"],
13+
});
14+
15+
export const metadata: Metadata = {
16+
title: "Create Next App",
17+
description: "Generated by create next app",
18+
};
19+
20+
export default function RootLayout({
21+
children,
22+
}: Readonly<{
23+
children: React.ReactNode;
24+
}>) {
25+
return (
26+
<html lang="en">
27+
<body
28+
className={`${geistSans.variable} ${geistMono.variable} antialiased`}
29+
>
30+
{children}
31+
</body>
32+
</html>
33+
);
34+
}

app/login/page.tsx

+44
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
'use client'
2+
import {signInWithPopup, GoogleAuthProvider, onAuthStateChanged} from "firebase/auth";
3+
import {auth} from "@/app/firebase";
4+
import {useEffect, useState} from "react";
5+
import { useRouter } from "next/navigation";
6+
7+
export default function Page() {
8+
const provider = new GoogleAuthProvider();
9+
const [loading, setLoading] = useState(true);
10+
const router = useRouter();
11+
12+
useEffect(() => {
13+
const authCheck = onAuthStateChanged(auth, (currentUser) => {
14+
if (currentUser) {
15+
router.push("/");
16+
} else {
17+
setLoading(false)
18+
}
19+
});
20+
21+
return () => authCheck();
22+
}, [router]);
23+
24+
function signIn() {
25+
signInWithPopup(auth, provider)
26+
.then(async (result) => {
27+
const credential = GoogleAuthProvider.credentialFromResult(result);
28+
if (credential) {
29+
const user = result.user;
30+
const idToken = await user.getIdToken();
31+
document.cookie = `token=${idToken}`;
32+
alert(user.email)
33+
}
34+
}).catch((error) => {
35+
alert(error)
36+
});
37+
}
38+
39+
if (loading) {
40+
return <p>Loading...</p>;
41+
} else {
42+
return <button onClick={signIn}>Sign in</button>
43+
}
44+
}

0 commit comments

Comments
 (0)