Skip to content

Commit 6ed3613

Browse files
init
1 parent 8dae7bd commit 6ed3613

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

55 files changed

+4196
-1111
lines changed

.env.example

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
ENCRYPTION_KEY=""

.eslintrc.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
{
2-
"extends": ["next/core-web-vitals", "next/typescript"]
2+
"extends": ["next/core-web-vitals", "next/typescript", "prettier"]
33
}

.gitignore

+3
Original file line numberDiff line numberDiff line change
@@ -34,3 +34,6 @@ yarn-error.log*
3434
# typescript
3535
*.tsbuildinfo
3636
next-env.d.ts
37+
38+
.env
39+
sqlite.db

.prettierrc

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
{
2+
"useTabs": true,
3+
"trailingComma": "none",
4+
"printWidth": 120
5+
}

README.md

+34-23
Original file line numberDiff line numberDiff line change
@@ -1,36 +1,47 @@
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).
1+
# Email and password example with 2FA in Next.js
22

3-
## Getting Started
3+
Built with SQLite.
44

5-
First, run the development server:
5+
- Password check with HaveIBeenPwned
6+
- Email verification
7+
- 2FA with TOTP
8+
- 2FA recovery codes
9+
- Password reset
10+
- Login throttling and rate limiting
611

7-
```bash
8-
npm run dev
9-
# or
10-
yarn dev
11-
# or
12-
pnpm dev
13-
# or
14-
bun dev
15-
```
12+
Emails are just logged to the console. Rate limiting is implemented using JavaScript `Map`.
1613

17-
Open [http://localhost:3000](http://localhost:3000) with your browser to see the result.
14+
## Initialize project
1815

19-
You can start editing the page by modifying `app/page.tsx`. The page auto-updates as you edit the file.
16+
Create `sqlite.db` and run `setup.sql`.
2017

21-
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.
18+
```
19+
sqlite3 sqlite.db
20+
```
2221

23-
## Learn More
22+
Create a .env file. Generate a 128 bit (16 byte) string, base64 encode it, and set it as `ENCRYPTION_KEY`.
2423

25-
To learn more about Next.js, take a look at the following resources:
24+
```bash
25+
ENCRYPTION_KEY="L9pmqRJnO1ZJSQ2svbHuBA=="
26+
```
2627

27-
- [Next.js Documentation](https://nextjs.org/docs) - learn about Next.js features and API.
28-
- [Learn Next.js](https://nextjs.org/learn) - an interactive Next.js tutorial.
28+
> You can use OpenSSL to quickly generate a secure key.
29+
>
30+
> ```bash
31+
> openssl rand --base64 16
32+
> ```
2933
30-
You can check out [the Next.js GitHub repository](https://github.com/vercel/next.js) - your feedback and contributions are welcome!
34+
Run the application:
3135
32-
## Deploy on Vercel
36+
```
37+
pnpm dev
38+
```
3339
34-
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.
40+
## Notes
3541

36-
Check out our [Next.js deployment documentation](https://nextjs.org/docs/app/building-your-application/deploying) for more details.
42+
- We do not consider user enumeration to be a real vulnerability so please don't open issues on it. If you really need to prevent it, just don't use emails.
43+
- This example does not handle unexpected errors gracefully.
44+
- There are some major code duplications (specifically for 2FA) to keep the codebase simple.
45+
- TODO: You may need to rewrite some queries and use transactions to avoid race conditions when using MySQL, Postgres, etc.
46+
- TODO: This project relies on the `X-Forwarded-For` header for getting the client's IP address.
47+
- TODO: Logging should be implemented.

actions/2fa.ts

+187
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,187 @@
1+
"use server";
2+
3+
import { recoveryCodeBucket, resetUser2FAWithRecoveryCode, totpBucket } from "@/lib/server/2fa";
4+
import { RefillingTokenBucket } from "@/lib/server/rate-limit";
5+
import { getCurrentSession, setSessionAs2FAVerified } from "@/lib/server/session";
6+
import { getUserTOTPKey, updateUserTOTPKey } from "@/lib/server/user";
7+
import { decodeBase64 } from "@oslojs/encoding";
8+
import { verifyTOTP } from "@oslojs/otp";
9+
import { redirect } from "next/navigation";
10+
11+
export async function verify2FAAction(_prev: ActionResult, formData: FormData): Promise<ActionResult> {
12+
const { session, user } = getCurrentSession();
13+
if (session === null) {
14+
return {
15+
message: "Not authenticated"
16+
};
17+
}
18+
if (!user.emailVerified) {
19+
return {
20+
message: "Forbidden"
21+
};
22+
}
23+
if (!user.registered2FA) {
24+
return {
25+
message: "Forbidden"
26+
};
27+
}
28+
if (!totpBucket.check(user.id, 1)) {
29+
return {
30+
message: "Too many requests"
31+
};
32+
}
33+
34+
const code = formData.get("code");
35+
if (typeof code !== "string") {
36+
return {
37+
message: "Invalid or missing fields"
38+
};
39+
}
40+
if (code === "") {
41+
return {
42+
message: "Enter your code"
43+
};
44+
}
45+
if (!totpBucket.consume(user.id, 1)) {
46+
return {
47+
message: "Too many requests"
48+
};
49+
}
50+
const totpKey = getUserTOTPKey(user.id);
51+
if (totpKey === null) {
52+
return {
53+
message: "Forbidden"
54+
};
55+
}
56+
if (!verifyTOTP(totpKey, 30, 6, code)) {
57+
return {
58+
message: "Invalid code"
59+
};
60+
}
61+
totpBucket.reset(user.id);
62+
setSessionAs2FAVerified(session.id);
63+
return redirect("/");
64+
}
65+
66+
const totpUpdateBucket = new RefillingTokenBucket<number>(3, 60 * 10);
67+
68+
export async function setup2FAAction(_prev: ActionResult, formData: FormData): Promise<ActionResult> {
69+
const { session, user } = getCurrentSession();
70+
if (session === null) {
71+
return {
72+
message: "Not authenticated"
73+
};
74+
}
75+
if (!user.emailVerified) {
76+
return {
77+
message: "Forbidden"
78+
};
79+
}
80+
if (user.registered2FA && !session.twoFactorVerified) {
81+
return {
82+
message: "Forbidden"
83+
};
84+
}
85+
if (!totpUpdateBucket.check(user.id, 1)) {
86+
return {
87+
message: "Too many requests"
88+
};
89+
}
90+
91+
const encodedKey = formData.get("key");
92+
const code = formData.get("code");
93+
if (typeof encodedKey !== "string" || typeof code !== "string") {
94+
return {
95+
message: "Invalid or missing fields"
96+
};
97+
}
98+
if (code === "") {
99+
return {
100+
message: "Please enter your code"
101+
};
102+
}
103+
if (encodedKey.length !== 28) {
104+
return {
105+
message: "Please enter your code"
106+
};
107+
}
108+
let key: Uint8Array;
109+
try {
110+
key = decodeBase64(encodedKey);
111+
} catch {
112+
return {
113+
message: "Invalid key"
114+
};
115+
}
116+
if (key.byteLength !== 20) {
117+
return {
118+
message: "Invalid key"
119+
};
120+
}
121+
if (!totpUpdateBucket.consume(user.id, 1)) {
122+
return {
123+
message: "Too many requests"
124+
};
125+
}
126+
if (!verifyTOTP(key, 30, 6, code)) {
127+
return {
128+
message: "Invalid code"
129+
};
130+
}
131+
updateUserTOTPKey(session.userId, key);
132+
setSessionAs2FAVerified(session.id);
133+
return redirect("/recovery-code");
134+
}
135+
136+
export async function reset2FAAction(_prev: ActionResult, formData: FormData): Promise<ActionResult> {
137+
const { session, user } = getCurrentSession();
138+
if (session === null) {
139+
return {
140+
message: "Not authenticated"
141+
};
142+
}
143+
if (!user.emailVerified) {
144+
return {
145+
message: "Forbidden"
146+
};
147+
}
148+
if (!user.registered2FA) {
149+
return {
150+
message: "Forbidden"
151+
};
152+
}
153+
if (!recoveryCodeBucket.check(user.id, 1)) {
154+
return {
155+
message: "Too many requests"
156+
};
157+
}
158+
159+
const code = formData.get("code");
160+
if (typeof code !== "string") {
161+
return {
162+
message: "Invalid or missing fields"
163+
};
164+
}
165+
if (code === "") {
166+
return {
167+
message: "Please enter your code"
168+
};
169+
}
170+
if (!recoveryCodeBucket.consume(user.id, 1)) {
171+
return {
172+
message: "Too many requests"
173+
};
174+
}
175+
const valid = resetUser2FAWithRecoveryCode(user.id, code);
176+
if (!valid) {
177+
return {
178+
message: "Invalid recovery code"
179+
};
180+
}
181+
recoveryCodeBucket.reset(user.id);
182+
return redirect("/2fa/setup");
183+
}
184+
185+
interface ActionResult {
186+
message: string;
187+
}

0 commit comments

Comments
 (0)