Skip to content

Commit 69f7b05

Browse files
authored
feat: Add sensitive info example (#153)
* feat: Add sensitive info example * Remove site key from async method * Message adjustments
1 parent 327b738 commit 69f7b05

23 files changed

+410
-60
lines changed

.trunk/trunk.yaml

+4-4
Original file line numberDiff line numberDiff line change
@@ -17,14 +17,14 @@ runtimes:
1717
# This is the section where you manage your linters. (https://docs.trunk.io/check/configuration)
1818
lint:
1919
enabled:
20-
20+
2121
2222
- git-diff-check
2323
24-
- osv-scanner@1.8.5
24+
- osv-scanner@1.9.0
2525
26-
- trivy@0.55.2
27-
26+
- trivy@0.56.2
27+
2828
2929
actions:
3030
enabled:

README.md

+7-4
Original file line numberDiff line numberDiff line change
@@ -8,10 +8,11 @@
88
# Arcjet example app
99

1010
[Arcjet](https://arcjet.com) helps developers protect their apps in just a few
11-
lines of code. This is an example application demonstrating the use of multiple
12-
features.
11+
lines of code. Bot detection. Rate limiting. Email validation. Attack
12+
protection. Data redaction. A developer-first approach to security.
1313

14-
This example is deployed at
14+
This is an example Next.js application demonstrating the use of multiple
15+
features. It is deployed at
1516
[https://example.arcjet.com](https://example.arcjet.com).
1617

1718
## Features
@@ -28,6 +29,9 @@ This example is deployed at
2829
- [Attack protection](https://example.arcjet.com/attack) demonstrates Arcjet
2930
Shield, which detects suspicious behavior, such as SQL injection and
3031
cross-site scripting attacks.
32+
- [Sensitive info](https://example.arcjet.com/sensitive-info) protects against
33+
clients sending you sensitive information such as PII that you do not wish to
34+
handle.
3135

3236
## Deploy it now
3337

@@ -68,7 +72,6 @@ npm run dev
6872
example](https://github.com/arcjet/arcjet-js/tree/main/examples/nextjs-14-react-hook-form))
6973
- Client-side validation: [Zod](https://zod.dev/)
7074
- Security: [Arcjet](https://arcjet.com/)
71-
- Platform: [Vercel](https://vercel.com/) (see [our integration](https://vercel.com/integrations/arcjet))
7275

7376
[vercel_deploy]: https://vercel.com/new/clone?repository-url=https%3A%2F%2Fgithub.com%2Farcjet%2Farcjet-js-example&project-name=arcjet-example&repository-name=arcjet-example&developer-id=oac_1GEcKBuKBilVnjToj1QUwdb8&demo-title=Arcjet%20Example%20&demo-description=Example%20rate%20limiting%2C%20bot%20protection%2C%20email%20verification%20%26%20form%20protection.&demo-url=https%3A%2F%2Fgithub.com%2Farcjet%2Farcjet-js-example&demo-image=https%3A%2F%2Fapp.arcjet.com%2Fimg%2Fexample-apps%2Fvercel%2Fdemo-image.jpg&integration-ids=oac_1GEcKBuKBilVnjToj1QUwdb8&external-id=arcjet-js-example◊
7477
[vercel_button]: https://vercel.com/button

app/attack/page.tsx

+3-4
Original file line numberDiff line numberDiff line change
@@ -2,15 +2,14 @@ import VisitDashboard from "@/components/compositions/VisitDashboard";
22
import WhatNext from "@/components/compositions/WhatNext";
33
import useSiteKey from "@/components/effects/useSiteKey";
44
import Divider from "@/components/elements/Divider";
5+
import styles from "@/components/elements/PageShared.module.scss";
56
import type { Metadata } from "next";
67
import Link from "next/link";
78

8-
import styles from "@/components/elements/PageShared.module.scss";
9-
109
export const metadata: Metadata = {
11-
title: "Arcjet attack protection",
10+
title: "Attack protection example",
1211
description:
13-
"Arcjet Shield detects suspicious behavior, such as SQL injection and cross-site scripting attacks.",
12+
"An example of Arcjet's attack protection for Next.js. Protect Next.js against SQL injection, cross-site scripting, and other attacks.",
1413
};
1514

1615
export default function IndexPage() {

app/bots/page.tsx

+2-3
Original file line numberDiff line numberDiff line change
@@ -8,9 +8,8 @@ import Link from "next/link";
88
import styles from "@/components/elements/PageShared.module.scss";
99

1010
export const metadata: Metadata = {
11-
title: "Arcjet bot protection example",
12-
description:
13-
"An example of Arcjet's bot protection configured to block automated clients.",
11+
title: "Bot protection example",
12+
description: "An example of Arcjet's bot protection for Next.js.",
1413
};
1514

1615
export default function IndexPage() {

app/layout.tsx

-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,6 @@ export const metadata: Metadata = {
2121
template: `%s - ${siteConfig.name}`,
2222
},
2323
description: siteConfig.description,
24-
2524
icons: [
2625
{
2726
rel: "icon",

app/page.tsx

+13-4
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,9 @@ export default function IndexPage() {
1313
<section className={styles.Content}>
1414
<div className={styles.Section}>
1515
<div className="flex max-w-[980px] flex-col items-start gap-6">
16+
<h1 className="text-3xl font-extrabold leading-tight tracking-tighter md:text-4xl">
17+
Arcjet Next.js example app
18+
</h1>
1619
<p className="max-w-[700px] text-lg">
1720
<Link
1821
href="https://arcjet.com"
@@ -21,12 +24,12 @@ export default function IndexPage() {
2124
>
2225
Arcjet
2326
</Link>{" "}
24-
helps developers protect their apps in just a few lines of code.
25-
Implement rate limiting, bot protection, email verification &
26-
defense against common attacks.
27+
helps developers protect their apps in just a few lines of code. Bot
28+
detection. Rate limiting. Email validation. Attack protection. Data
29+
redaction. A developer-first approach to security.
2730
</p>
2831
<p className="max-w-[700px] text-secondary-foreground">
29-
This application shows of the key features of Arcjet. The code is{" "}
32+
This is an example Next.js application using Arcjet. The code is{" "}
3033
<Link
3134
href="https://github.com/arcjet/arcjet-js-example"
3235
target="_blank"
@@ -66,6 +69,12 @@ export default function IndexPage() {
6669
>
6770
Attack protection
6871
</Link>
72+
<Link
73+
href="/sensitive-info"
74+
className={buttonVariants({ variant: "default" })}
75+
>
76+
Sensitive info
77+
</Link>
6978
</div>
7079
</div>
7180
</div>

app/rate-limiting/page.tsx

+3-8
Original file line numberDiff line numberDiff line change
@@ -12,16 +12,13 @@ import Link from "next/link";
1212
import styles from "@/components/elements/PageShared.module.scss";
1313

1414
export const metadata: Metadata = {
15-
title: "Arcjet rate limit example",
16-
description:
17-
"An example of Arcjet's rate limiting with different limits depending on authentication.",
15+
title: "Rate limiting example",
16+
description: "An example of Arcjet's rate limiting for Next.js.",
1817
};
1918

2019
export default async function IndexPage() {
2120
const session = await auth();
2221

23-
const { siteKey } = useSiteKey();
24-
2522
return (
2623
<section className={styles.Content}>
2724
<div className={styles.Section}>
@@ -80,8 +77,6 @@ export default async function IndexPage() {
8077
</p>
8178

8279
{session?.user ? <SignOut /> : <SignIn />}
83-
84-
{siteKey && <VisitDashboard />}
8580
</div>
8681

8782
<Divider />
@@ -113,7 +108,7 @@ export default async function IndexPage() {
113108

114109
<Divider />
115110

116-
<WhatNext deployed={siteKey != null} />
111+
<WhatNext />
117112
</section>
118113
);
119114
}

app/sensitive-info/page.tsx

+87
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,87 @@
1+
import { SupportForm } from "@/components/SuppportForm";
2+
import VisitDashboard from "@/components/compositions/VisitDashboard";
3+
import WhatNext from "@/components/compositions/WhatNext";
4+
import useSiteKey from "@/components/effects/useSiteKey";
5+
import Divider from "@/components/elements/Divider";
6+
import styles from "@/components/elements/PageShared.module.scss";
7+
import type { Metadata } from "next";
8+
import Link from "next/link";
9+
10+
export const metadata: Metadata = {
11+
title: "Sensitive info detection example",
12+
description:
13+
"An example of Arcjet's sensitive info detection for Next.js. Detect credit card numbers and other PII with Next.js.",
14+
};
15+
16+
export default function IndexPage() {
17+
const { siteKey } = useSiteKey();
18+
19+
return (
20+
<section className={styles.Content}>
21+
<div className={styles.Section}>
22+
<h1 className="text-3xl font-extrabold leading-tight tracking-tighter md:text-4xl">
23+
Arcjet sensitive info detection example
24+
</h1>
25+
<p className="max-w-[700px] text-lg">
26+
This form uses{" "}
27+
<Link
28+
href="https://docs.arcjet.com/sensitive-info/concepts"
29+
className="font-bold decoration-1 underline-offset-2 hover:underline"
30+
>
31+
Arcjet&apos;s sensitive info detection
32+
</Link>{" "}
33+
feature which is configured to detect credit card numbers. It can be
34+
configured to detect other types of sensitive information and custom
35+
patterns.
36+
</p>
37+
<p className="max-w-[700px] text-secondary-foreground">
38+
The request is analyzed entirely on your server so no sensitive
39+
information is sent to Arcjet.
40+
</p>
41+
</div>
42+
43+
<Divider />
44+
45+
<div className={styles.Section}>
46+
<h2 className="text-xl font-bold">Try it</h2>
47+
48+
<div className="flex gap-4">
49+
<SupportForm />
50+
</div>
51+
52+
{siteKey && <VisitDashboard />}
53+
</div>
54+
55+
<Divider />
56+
57+
<div className={styles.Section}>
58+
<h2 className="text-xl font-bold">See the code</h2>
59+
<p className="text-secondary-foreground">
60+
The{" "}
61+
<Link
62+
href="https://github.com/arcjet/arcjet-js-example/blob/main/app/sensitive-info/test/route.ts"
63+
target="_blank"
64+
rel="noreferrer"
65+
className="font-bold decoration-1 underline-offset-2 hover:underline"
66+
>
67+
API route
68+
</Link>{" "}
69+
imports a{" "}
70+
<Link
71+
href="https://github.com/arcjet/arcjet-js-example/blob/main/lib/arcjet.ts"
72+
target="_blank"
73+
rel="noreferrer"
74+
className="font-bold decoration-1 underline-offset-2 hover:underline"
75+
>
76+
centralized Arcjet client
77+
</Link>{" "}
78+
which sets base rules.
79+
</p>
80+
</div>
81+
82+
<Divider />
83+
84+
<WhatNext deployed={siteKey != null} />
85+
</section>
86+
);
87+
}

app/sensitive-info/schema.ts

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
import { z } from "zod";
2+
3+
// Zod schema for client-side validation of the form fields. Arcjet will do
4+
// server-side validation as well because you can't trust the client.
5+
// Client-side validation improves the UX by providing immediate feedback
6+
// whereas server-side validation is necessary for security.
7+
export const formSchema = z.object({
8+
supportMessage: z
9+
.string()
10+
.min(5, { message: "Your message must be at least 5 characters." })
11+
.max(1000, {
12+
message:
13+
"Your message is too long. Please shorten it to 1000 characters.",
14+
}),
15+
});
16+
17+
export const emptyFormSchema = z.object({});

app/sensitive-info/submitted/page.tsx

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import { SupportForm } from "@/components/SuppportForm";
2+
import VisitDashboard from "@/components/compositions/VisitDashboard";
3+
import WhatNext from "@/components/compositions/WhatNext";
4+
import useSiteKey from "@/components/effects/useSiteKey";
5+
import Divider from "@/components/elements/Divider";
6+
import Link from "next/link";
7+
8+
import styles from "@/components/elements/PageShared.module.scss";
9+
10+
export default function IndexPage() {
11+
const { siteKey } = useSiteKey();
12+
13+
return (
14+
<section className={styles.Content}>
15+
<div className={styles.Section}>
16+
<h1 className="text-3xl font-extrabold leading-tight tracking-tighter md:text-4xl">
17+
Form submitted
18+
</h1>
19+
<p className="max-w-[700px] text-lg">
20+
If this were a real form, your message would have been submitted.
21+
</p>
22+
</div>
23+
24+
<Divider />
25+
26+
<WhatNext deployed={siteKey != null} />
27+
</section>
28+
);
29+
}

app/sensitive-info/test/route.ts

+77
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,77 @@
1+
import { formSchema } from "@/app/sensitive-info/schema";
2+
import arcjet, { sensitiveInfo, shield } from "@/lib/arcjet";
3+
import { NextRequest, NextResponse } from "next/server";
4+
5+
// Add rules to the base Arcjet instance outside of the handler function
6+
const aj = arcjet
7+
.withRule(
8+
// Arcjet's protectSignup rule is a combination of email validation, bot
9+
// protection and rate limiting. Each of these can also be used separately
10+
// on other routes e.g. rate limiting on a login route. See
11+
// https://docs.arcjet.com/get-started
12+
sensitiveInfo({
13+
mode: "LIVE", // Will block requests, use "DRY_RUN" to log only
14+
deny: ["CREDIT_CARD_NUMBER", "EMAIL"], // Deny requests with credit card numbers
15+
}),
16+
)
17+
// You can chain multiple rules, so we'll include shield
18+
.withRule(
19+
// Shield detects suspicious behavior, such as SQL injection and cross-site
20+
// scripting attacks.
21+
shield({
22+
mode: "LIVE",
23+
}),
24+
);
25+
26+
export async function POST(req: NextRequest) {
27+
const fingerprint = req.ip!;
28+
const decision = await aj.protect(req, { fingerprint });
29+
30+
console.log("Arcjet decision: ", decision);
31+
32+
if (decision.isDenied()) {
33+
if (decision.reason.isSensitiveInfo()) {
34+
const message =
35+
"please do not include credit card numbers in your message.";
36+
37+
return NextResponse.json(
38+
{ message, reason: decision.reason },
39+
{ status: 400 },
40+
);
41+
} else {
42+
return NextResponse.json({ message: "Forbidden" }, { status: 403 });
43+
}
44+
} else if (decision.isErrored()) {
45+
console.error("Arcjet error:", decision.reason);
46+
if (decision.reason.message == "[unauthenticated] invalid key") {
47+
return NextResponse.json(
48+
{
49+
message:
50+
"invalid Arcjet key. Is the ARCJET_KEY environment variable set?",
51+
},
52+
{ status: 500 },
53+
);
54+
} else {
55+
return NextResponse.json(
56+
{ message: "Internal server error: " + decision.reason.message },
57+
{ status: 500 },
58+
);
59+
}
60+
}
61+
62+
const json = await req.json();
63+
const data = formSchema.safeParse(json);
64+
65+
if (!data.success) {
66+
const { error } = data;
67+
68+
return NextResponse.json(
69+
{ message: "invalid request", error },
70+
{ status: 400 },
71+
);
72+
}
73+
74+
return NextResponse.json({
75+
ok: true,
76+
});
77+
}

app/signup/page.tsx

+7
Original file line numberDiff line numberDiff line change
@@ -3,10 +3,17 @@ import VisitDashboard from "@/components/compositions/VisitDashboard";
33
import WhatNext from "@/components/compositions/WhatNext";
44
import useSiteKey from "@/components/effects/useSiteKey";
55
import Divider from "@/components/elements/Divider";
6+
import type { Metadata } from "next";
67
import Link from "next/link";
78

89
import styles from "@/components/elements/PageShared.module.scss";
910

11+
export const metadata: Metadata = {
12+
title: "Signup form protection example",
13+
description:
14+
"An example of Arcjet's signup form protection for Next.js which includes email verification, rate limiting, and bot protection.",
15+
};
16+
1017
export default function IndexPage() {
1118
const { siteKey } = useSiteKey();
1219

File renamed without changes.
File renamed without changes.

0 commit comments

Comments
 (0)