Skip to content

Commit 0f07d27

Browse files
authored
Add Changelog (#9063)
* Add DB Changelog * fix build errors * Create/publish/unpublish * Edit * Login with Google * Login * Protect actions * GCS bucket upload * Last change before killing seed data * Remove seed data * fix build errors * fix: Build error * fix: Env names * build prisma first * enable tracing * Add file upload for hero image * remove mdxcomponents and pray vercel error is gone * try to split stuff in functions * fix * Remove load mdx for notFound * remove navbar since it also loads the world * More nextjs style * Revert "More nextjs style" This reverts commit 780b988. * fix picture height * urlencode filename * Fix errors * Fix build error
1 parent d286236 commit 0f07d27

File tree

198 files changed

+3805
-11995
lines changed

Some content is hidden

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

198 files changed

+3805
-11995
lines changed

.envrc

-3
This file was deleted.

app/api/auth/[...nextauth]/route.ts

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
import {PrismaAdapter} from '@auth/prisma-adapter';
2+
import NextAuth from 'next-auth';
3+
import GoogleProvider from 'next-auth/providers/google';
4+
5+
import {prisma} from 'sentry-docs/prisma';
6+
7+
const handler = NextAuth({
8+
adapter: PrismaAdapter(prisma),
9+
providers: [
10+
GoogleProvider({
11+
clientId: process.env.GOOGLE_CLIENT_ID || '',
12+
clientSecret: process.env.GOOGLE_CLIENT_SECRET || '',
13+
}),
14+
],
15+
session: {
16+
strategy: 'jwt',
17+
},
18+
});
19+
20+
export {handler as GET, handler as POST};
+92
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
import {Fragment, Suspense} from 'react';
2+
import Link from 'next/link';
3+
4+
import {editChangelog} from 'sentry-docs/actions/changelog';
5+
import {FileUpload} from 'sentry-docs/components/changelog/fileUpload';
6+
import {ForwardRefEditor} from 'sentry-docs/components/changelog/forwardRefEditor';
7+
import {TitleSlug} from 'sentry-docs/components/changelog/titleSlug';
8+
import {Button} from 'sentry-docs/components/changelog/ui/Button';
9+
import {Select} from 'sentry-docs/components/changelog/ui/Select';
10+
import {prisma} from 'sentry-docs/prisma';
11+
12+
export default async function ChangelogCreatePage({params}) {
13+
const categories = await prisma.category.findMany();
14+
const changelog = await prisma.changelog.findUnique({
15+
where: {id: params.id},
16+
include: {
17+
author: true,
18+
categories: true,
19+
},
20+
});
21+
22+
if (!changelog) {
23+
return (
24+
<Fragment>
25+
<header>
26+
<h2>Changelog not found</h2>
27+
</header>
28+
<footer>
29+
<Link href="/changelogs">Return to Changelogs list</Link>
30+
</footer>
31+
</Fragment>
32+
);
33+
}
34+
35+
return (
36+
<section className="overflow-x-auto col-start-3 col-span-8">
37+
<form action={editChangelog} className="px-2 w-full">
38+
<input type="hidden" name="id" value={changelog.id} />
39+
<TitleSlug defaultSlug={changelog.slug} defaultTitle={changelog.title} />
40+
<FileUpload defaultFile={changelog.image || ''} />
41+
<div className="my-6">
42+
<label htmlFor="summary" className="block text-xs font-medium text-gray-700">
43+
Summary
44+
<Fragment>
45+
&nbsp;<span className="font-bold text-secondary">*</span>
46+
</Fragment>
47+
</label>
48+
<textarea name="summary" className="w-full" required>
49+
{changelog.summary}
50+
</textarea>
51+
<span className="text-xs text-gray-500 italic">
52+
This will be shown in the list
53+
</span>
54+
</div>
55+
<div>
56+
<Select
57+
name="categories"
58+
className="mt-1 mb-6"
59+
label="Category"
60+
placeholder="Select Category"
61+
defaultValue={changelog.categories.map(category => ({
62+
label: category.name,
63+
value: category.name,
64+
}))}
65+
options={categories.map(category => ({
66+
label: category.name,
67+
value: category.name,
68+
}))}
69+
isMulti
70+
/>
71+
</div>
72+
73+
<Suspense fallback={null}>
74+
<ForwardRefEditor
75+
name="content"
76+
defaultValue={changelog.content || ''}
77+
className="w-full"
78+
/>
79+
</Suspense>
80+
81+
<footer className="flex items-center justify-between mt-2 mb-8">
82+
<Link href="/changelog/_admin" className="underline text-gray-500">
83+
Return to Changelogs list
84+
</Link>
85+
<div>
86+
<Button type="submit">Update</Button>
87+
</div>
88+
</footer>
89+
</form>
90+
</section>
91+
);
92+
}

app/changelog/%5Fadmin/confirm.tsx

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
'use client';
2+
3+
export default function Confirm({changelog, action, children}) {
4+
return (
5+
<form
6+
action={action}
7+
className="inline-block"
8+
onSubmit={e => {
9+
e.preventDefault();
10+
// eslint-disable-next-line no-alert
11+
if (confirm('Are you sure?')) {
12+
action(new FormData(e.currentTarget));
13+
}
14+
}}
15+
>
16+
<input type="hidden" name="id" value={changelog.id} />
17+
<button
18+
type="submit"
19+
className="text-indigo-600 hover:bg-indigo-100 rounded-md px-1 py-2 text-xs whitespace-nowrap"
20+
>
21+
{children}
22+
</button>
23+
</form>
24+
);
25+
}
+61
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
import {Fragment} from 'react';
2+
import Link from 'next/link';
3+
4+
import {createChangelog} from 'sentry-docs/actions/changelog';
5+
import {FileUpload} from 'sentry-docs/components/changelog/fileUpload';
6+
import {ForwardRefEditor} from 'sentry-docs/components/changelog/forwardRefEditor';
7+
import {TitleSlug} from 'sentry-docs/components/changelog/titleSlug';
8+
import {Button} from 'sentry-docs/components/changelog/ui/Button';
9+
import {Select} from 'sentry-docs/components/changelog/ui/Select';
10+
import {prisma} from 'sentry-docs/prisma';
11+
12+
export default async function ChangelogCreatePage() {
13+
const categories = await prisma.category.findMany();
14+
15+
return (
16+
<section className="overflow-x-auto col-start-3 col-span-8">
17+
<form action={createChangelog} className="px-2 w-full">
18+
<TitleSlug />
19+
<FileUpload />
20+
<div className="my-6">
21+
<label htmlFor="summary" className="block text-xs font-medium text-gray-700">
22+
Summary
23+
<Fragment>
24+
&nbsp;<span className="font-bold text-secondary">*</span>
25+
</Fragment>
26+
</label>
27+
<textarea name="summary" className="w-full" required />
28+
<span className="text-xs text-gray-500 italic">
29+
This will be shown in the list
30+
</span>
31+
</div>
32+
<div>
33+
<Select
34+
name="categories"
35+
className="mt-1 mb-6"
36+
label="Category"
37+
placeholder="Select Category"
38+
options={categories.map(category => ({
39+
label: category.name,
40+
value: category.name,
41+
}))}
42+
isMulti
43+
/>
44+
</div>
45+
46+
<ForwardRefEditor name="content" className="w-full" />
47+
48+
<footer className="flex items-center justify-between mt-2">
49+
<Link href="/changelog/_admin" className="underline text-gray-500">
50+
Return to Changelogs list
51+
</Link>
52+
<div>
53+
<Button type="submit">Create (not published yet)</Button>
54+
<br />
55+
<span className="text-xs text-gray-500 italic">You can publish it later</span>
56+
</div>
57+
</footer>
58+
</form>
59+
</section>
60+
);
61+
}

app/changelog/%5Fadmin/layout.tsx

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
import {type ReactNode, Suspense} from 'react';
2+
import {GET} from 'app/api/auth/[...nextauth]/route';
3+
import {getServerSession} from 'next-auth/next';
4+
5+
import LoginButton from 'sentry-docs/components/changelog/loginButton';
6+
import NextAuthSessionProvider from 'sentry-docs/components/nextAuthSessionProvider';
7+
8+
export default async function Layout({children}: {children: ReactNode}) {
9+
const session = await getServerSession(GET);
10+
let content = (
11+
<div className="relative min-h-[calc(100vh-8rem)] w-full mx-auto bg-gray-200 pt-16 grid grid-cols-12">
12+
{children}
13+
<div className="fixed top-3 right-4 z-50">
14+
<Suspense fallback={null}>
15+
<LoginButton />
16+
</Suspense>
17+
</div>
18+
</div>
19+
);
20+
if (!session) {
21+
content = (
22+
<div className="relative min-h-[calc(100vh-8rem)] w-full mx-auto bg-gray-200 pt-16 flex items-center justify-center">
23+
<LoginButton />
24+
</div>
25+
);
26+
}
27+
return <NextAuthSessionProvider>{content}</NextAuthSessionProvider>;
28+
}

app/changelog/%5Fadmin/page.tsx

+121
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,121 @@
1+
import {Fragment} from 'react';
2+
import {PlusIcon} from '@radix-ui/react-icons';
3+
import {Button, Text} from '@radix-ui/themes';
4+
import Link from 'next/link';
5+
6+
import {
7+
deleteChangelog,
8+
publishChangelog,
9+
unpublishChangelog,
10+
} from 'sentry-docs/actions/changelog';
11+
import {prisma} from 'sentry-docs/prisma';
12+
13+
import Confirm from './confirm';
14+
15+
export default async function ChangelogsListPage() {
16+
const changelogs = await prisma.changelog.findMany({
17+
include: {
18+
categories: true,
19+
author: true,
20+
},
21+
orderBy: {
22+
createdAt: 'desc',
23+
},
24+
});
25+
26+
return (
27+
<Fragment>
28+
<header className="mb-4 col-start-3 col-span-2 text-left">
29+
<Button>
30+
<PlusIcon />
31+
<Link href="/changelog/_admin/create">New Changelog</Link>
32+
</Button>
33+
</header>
34+
<section className="overflow-x-auto col-start-3 col-span-8 shadow-md rounded-lg">
35+
<table className="w-full text-sm text-left text-gray-500">
36+
<thead className="text-xs text-gray-700 uppercase bg-gray-50">
37+
<tr>
38+
<th className="whitespace-nowrap px-4 py-2">Title</th>
39+
<th className="whitespace-nowrap px-4 py-2">Categories</th>
40+
<th className="whitespace-nowrap px-4 py-2">Published by</th>
41+
<th className="px-4 py-2" />
42+
</tr>
43+
</thead>
44+
<tbody className="divide-y divide-gray-200 bg-white border-b hover:bg-gray-50 dark:hover:bg-gray-600">
45+
{changelogs.length === 0 && (
46+
<tr>
47+
<td
48+
colSpan={10}
49+
className="text-center font-medium text-gray-900 whitespace-nowrap"
50+
>
51+
No changelogs found
52+
</td>
53+
</tr>
54+
)}
55+
56+
{changelogs.map(changelog => (
57+
<tr key={changelog.id} className="bg-white border-b hover:bg-gray-50">
58+
<td className="px-6 py-4 font-medium text-gray-900">{changelog.title}</td>
59+
60+
<td className="px-4 py-2">
61+
{changelog.categories.map(category => (
62+
<div
63+
key={category.id}
64+
className="inline whitespace-nowrap p-2 uppercase shadow-sm no-underline rounded-full text-red text-xs mr-1 bg-gray-100"
65+
>
66+
{category.name}
67+
</div>
68+
))}
69+
</td>
70+
<td className="px-4 py-2 text-center">
71+
{changelog.published && (
72+
<span className="text-gray-500">
73+
<Text size="1">
74+
{' '}
75+
{new Date(changelog.publishedAt || '').toLocaleDateString(
76+
undefined,
77+
{month: 'long', day: 'numeric'}
78+
)}
79+
</Text>
80+
<br />
81+
</span>
82+
)}
83+
<Text size="1">{changelog.author?.name}</Text>
84+
</td>
85+
86+
<td className="px-4 py-2">
87+
<div className="flex h-full justify-end">
88+
<Link
89+
href={`/changelog/${changelog.slug}`}
90+
className="text-indigo-600 hover:bg-indigo-100 rounded-md px-1 py-2 text-xs whitespace-nowrap"
91+
>
92+
👀 Show
93+
</Link>
94+
<Link
95+
href={`/changelog/_admin/${changelog.id}/edit`}
96+
className="text-indigo-600 hover:bg-indigo-100 rounded-md px-1 py-2 text-xs whitespace-nowrap"
97+
>
98+
📝 Edit
99+
</Link>
100+
{changelog.published ? (
101+
<Confirm action={unpublishChangelog} changelog={changelog}>
102+
⛔️ Unpublish?
103+
</Confirm>
104+
) : (
105+
<Confirm action={publishChangelog} changelog={changelog}>
106+
✅ Publish?
107+
</Confirm>
108+
)}
109+
<Confirm action={deleteChangelog} changelog={changelog}>
110+
💀 Delete?
111+
</Confirm>
112+
</div>
113+
</td>
114+
</tr>
115+
))}
116+
</tbody>
117+
</table>
118+
</section>
119+
</Fragment>
120+
);
121+
}

0 commit comments

Comments
 (0)