client/server: lint and add functionality for admin to update homepage

This commit is contained in:
Max Leiter 2022-04-20 01:49:34 -07:00
parent be6de7c796
commit b5024e3f45
No known key found for this signature in database
GPG key ID: A3512F2F2F17EBDA
26 changed files with 617 additions and 313 deletions

View file

@ -5,9 +5,11 @@ Drift is a self-hostable Gist clone. It's also a major work-in-progress, but is
You can try a demo at https://drift.maxleiter.com. The demo is built on master but has no database, so files and accounts can be wiped at any time. You can try a demo at https://drift.maxleiter.com. The demo is built on master but has no database, so files and accounts can be wiped at any time.
If you want to contribute, need support, or want to stay updated, you can join the IRC channel at #drift on irc.libera.chat or [reach me on twitter](https://twitter.com/Max_Leiter). If you don't have an IRC client yet, you can use a webclient [here](https://demo.thelounge.chat/#/connect?join=%23drift&nick=drift-user&realname=Drift%20User). If you want to contribute, need support, or want to stay updated, you can join the IRC channel at #drift on irc.libera.chat or [reach me on twitter](https://twitter.com/Max_Leiter). If you don't have an IRC client yet, you can use a webclient [here](https://demo.thelounge.chat/#/connect?join=%23drift&nick=drift-user&realname=Drift%20User).
<hr /> <hr />
**Contents:** **Contents:**
- [Setup](#setup) - [Setup](#setup)
- [Development](#development) - [Development](#development)
- [Production](#production) - [Production](#production)
@ -50,8 +52,6 @@ You can change these to your liking.
- `MEMORY_DB`: if `true`, a sqlite database will not be created and changes will only exist in memory. Mainly for the demo. - `MEMORY_DB`: if `true`, a sqlite database will not be created and changes will only exist in memory. Mainly for the demo.
- `REGISTRATION_PASSWORD`: if `true`, the user will be required to provide this password to sign-up, in addition to their username and account password. If it's not set, no additional password will be required. - `REGISTRATION_PASSWORD`: if `true`, the user will be required to provide this password to sign-up, in addition to their username and account password. If it's not set, no additional password will be required.
- `SECRET_KEY`: the same secret key as the client - `SECRET_KEY`: the same secret key as the client
- `WELCOME_CONTENT`: a markdown string that's rendered on the home page
- `WELCOME_TITLE`: the file title for the post on the homepage.
- `ENABLE_ADMIN`: the first account created is an administrator account - `ENABLE_ADMIN`: the first account created is an administrator account
- `DRIFT_HOME`: defaults to ~/.drift, the directory for storing the database and eventually images - `DRIFT_HOME`: defaults to ~/.drift, the directory for storing the database and eventually images

View file

@ -2,42 +2,37 @@ import { Popover, Button } from "@geist-ui/core"
import { MoreVertical } from "@geist-ui/icons" import { MoreVertical } from "@geist-ui/icons"
type Action = { type Action = {
title: string title: string
onClick: () => void onClick: () => void
} }
const ActionDropdown = ({ const ActionDropdown = ({
title = "Actions", title = "Actions",
actions, actions,
showTitle = false, showTitle = false
}: { }: {
title?: string, title?: string
showTitle?: boolean, showTitle?: boolean
actions: Action[] actions: Action[]
}) => { }) => {
return ( return (
<Popover <Popover
title={title} title={title}
content={ content={
<> <>
{showTitle && <Popover.Item title> {showTitle && <Popover.Item title>{title}</Popover.Item>}
{title} {actions.map((action) => (
</Popover.Item>} <Popover.Item onClick={action.onClick} key={action.title}>
{actions.map(action => ( {action.title}
<Popover.Item </Popover.Item>
onClick={action.onClick} ))}
key={action.title} </>
> }
{action.title} hideArrow
</Popover.Item> >
))} <Button iconRight={<MoreVertical />} auto></Button>
</> </Popover>
} )
hideArrow
>
<Button iconRight={<MoreVertical />} auto></Button>
</Popover>
)
} }
export default ActionDropdown export default ActionDropdown

View file

@ -2,6 +2,7 @@ import { Text, Spacer } from "@geist-ui/core"
import Cookies from "js-cookie" import Cookies from "js-cookie"
import styles from "./admin.module.css" import styles from "./admin.module.css"
import PostTable from "./post-table" import PostTable from "./post-table"
import ServerInfo from "./server-info"
import UserTable from "./user-table" import UserTable from "./user-table"
export const adminFetcher = async ( export const adminFetcher = async (
@ -24,6 +25,8 @@ const Admin = () => {
return ( return (
<div className={styles.adminWrapper}> <div className={styles.adminWrapper}>
<Text h2>Administration</Text> <Text h2>Administration</Text>
<ServerInfo />
<Spacer height={1} />
<UserTable /> <UserTable />
<Spacer height={1} /> <Spacer height={1} />
<PostTable /> <PostTable />

View file

@ -9,7 +9,7 @@ import ActionDropdown from "./action-dropdown"
const PostTable = () => { const PostTable = () => {
const [posts, setPosts] = useState<Post[]>() const [posts, setPosts] = useState<Post[]>()
const { setToast } = useToasts() // const { setToast } = useToasts()
useEffect(() => { useEffect(() => {
const fetchPosts = async () => { const fetchPosts = async () => {
@ -35,8 +35,8 @@ const PostTable = () => {
visibility: post.visibility, visibility: post.visibility,
size: post.files size: post.files
? byteToMB( ? byteToMB(
post.files.reduce((acc, file) => acc + file.html.length, 0) post.files.reduce((acc, file) => acc + file.html.length, 0)
) )
: 0, : 0,
actions: "" actions: ""
} }
@ -109,14 +109,14 @@ const PostTable = () => {
dataIndex: "", dataIndex: "",
key: "actions", key: "actions",
width: 50, width: 50,
render(post: Post) { render() {
return ( return (
<ActionDropdown <ActionDropdown
title="Actions" title="Actions"
actions={[ actions={[
{ {
title: "Delete", title: "Delete",
onClick: () => deletePost(post.id) onClick: () => deletePost()
} }
]} ]}
/> />
@ -128,7 +128,11 @@ const PostTable = () => {
return ( return (
<SettingsGroup title="Posts"> <SettingsGroup title="Posts">
{!posts && <Fieldset.Subtitle>Loading...</Fieldset.Subtitle>} {!posts && <Fieldset.Subtitle>Loading...</Fieldset.Subtitle>}
{posts && <Fieldset.Subtitle><h5>{posts.length} posts</h5></Fieldset.Subtitle>} {posts && (
<Fieldset.Subtitle>
<h5>{posts.length} posts</h5>
</Fieldset.Subtitle>
)}
{posts && <Table columns={tableColumns} data={tablePosts} />} {posts && <Table columns={tableColumns} data={tablePosts} />}
</SettingsGroup> </SettingsGroup>
) )

View file

@ -0,0 +1,75 @@
import SettingsGroup from "@components/settings-group"
import { Button, Input, Spacer, Textarea, useToasts } from "@geist-ui/core"
import { useEffect, useState } from "react"
import { adminFetcher } from "."
const Homepage = () => {
const [description, setDescription] = useState<string>("")
const [title, setTitle] = useState<string>("")
const { setToast } = useToasts()
useEffect(() => {
const fetchServerInfo = async () => {
const res = await adminFetcher("/server-info")
const data = await res.json()
setDescription(data.welcomeMessage)
setTitle(data.welcomeTitle)
}
fetchServerInfo()
}, [])
const onSubmit = async (e: React.FormEvent<HTMLFormElement>) => {
e.preventDefault()
const res = await adminFetcher("/server-info", {
method: "PUT",
body: {
description,
title
}
})
if (res.status === 200) {
setToast({
type: "success",
text: "Server info updated"
})
setDescription(description)
setTitle(title)
} else {
setToast({
text: "Something went wrong",
type: "error"
})
}
}
return (
<SettingsGroup title="Homepage">
<form onSubmit={onSubmit}>
<div>
<label htmlFor="title">Title</label>
<Input
id="title"
value={title}
onChange={(e) => setTitle(e.target.value)}
/>
</div>
<Spacer height={1} />
<div>
<label htmlFor="message">Description (markdown)</label>
<Textarea
width={"100%"}
height={10}
id="message"
value={description}
onChange={(e) => setDescription(e.target.value)}
/>
</div>
<Spacer height={1} />
<Button htmlType="submit">Update</Button>
</form>
</SettingsGroup>
)
}
export default Homepage

View file

@ -131,7 +131,8 @@ const UserTable = () => {
actions={[ actions={[
{ {
title: user.role === "admin" ? "Change role" : "Make admin", title: user.role === "admin" ? "Change role" : "Make admin",
onClick: () => toggleRole(user.id, user.role === "admin" ? "user" : "admin") onClick: () =>
toggleRole(user.id, user.role === "admin" ? "user" : "admin")
}, },
{ {
title: "Delete", title: "Delete",
@ -147,7 +148,11 @@ const UserTable = () => {
return ( return (
<SettingsGroup title="Users"> <SettingsGroup title="Users">
{!users && <Fieldset.Subtitle>Loading...</Fieldset.Subtitle>} {!users && <Fieldset.Subtitle>Loading...</Fieldset.Subtitle>}
{users && <Fieldset.Subtitle><h5>{users.length} users</h5></Fieldset.Subtitle>} {users && (
<Fieldset.Subtitle>
<h5>{users.length} users</h5>
</Fieldset.Subtitle>
)}
{users && <Table columns={usernameColumns} data={tableUsers} />} {users && <Table columns={usernameColumns} data={tableUsers} />}
</SettingsGroup> </SettingsGroup>
) )

View file

@ -89,10 +89,10 @@ const Header = () => {
href: "/mine" href: "/mine"
}, },
{ {
name: 'settings', name: "settings",
icon: <SettingsIcon />, icon: <SettingsIcon />,
value: 'settings', value: "settings",
href: '/settings' href: "/settings"
}, },
{ {
name: "sign out", name: "sign out",

View file

@ -1,9 +1,4 @@
import { import { Button, useToasts, ButtonDropdown, Input } from "@geist-ui/core"
Button,
useToasts,
ButtonDropdown,
Input,
} from "@geist-ui/core"
import { useRouter } from "next/router" import { useRouter } from "next/router"
import { useCallback, useEffect, useMemo, useRef, useState } from "react" import { useCallback, useEffect, useMemo, useRef, useState } from "react"
import generateUUID from "@lib/generate-uuid" import generateUUID from "@lib/generate-uuid"

View file

@ -97,10 +97,7 @@ const ListItem = ({
{post.files?.map((file: File) => { {post.files?.map((file: File) => {
return ( return (
<div key={file.id}> <div key={file.id}>
<Link <Link color href={`/post/${post.id}#${file.title}`}>
color
href={`/post/${post.id}#${file.title}`}
>
{file.title || "Untitled file"} {file.title || "Untitled file"}
</Link> </Link>
</div> </div>

View file

@ -85,9 +85,7 @@ const PostPage = ({ post: initialPost, isProtected }: Props) => {
} }
const viewParentClick = () => { const viewParentClick = () => {
router.push( router.push(`/post/${post.parent!.id}`)
`/post/${post.parent!.id}`
)
} }
if (isLoading) { if (isLoading) {
@ -123,11 +121,7 @@ const PostPage = ({ post: initialPost, isProtected }: Props) => {
Edit a Copy Edit a Copy
</Button> </Button>
{post.parent && ( {post.parent && (
<Button <Button auto icon={<Parent />} onClick={viewParentClick}>
auto
icon={<Parent />}
onClick={viewParentClick}
>
View Parent View Parent
</Button> </Button>
)} )}

View file

@ -1,24 +1,21 @@
import { Fieldset, Text, Divider } from "@geist-ui/core" import { Fieldset, Text, Divider } from "@geist-ui/core"
import styles from './settings-group.module.css' import styles from "./settings-group.module.css"
type Props = { type Props = {
title: string, title: string
children: React.ReactNode | React.ReactNode[], children: React.ReactNode | React.ReactNode[]
} }
const SettingsGroup = ({ const SettingsGroup = ({ title, children }: Props) => {
title, return (
children, <Fieldset>
}: Props) => { <Fieldset.Content>
return <Fieldset> <Text h4>{title}</Text>
<Fieldset.Content> </Fieldset.Content>
<Text h4>{title}</Text> <Divider />
</Fieldset.Content> <Fieldset.Content className={styles.content}>{children}</Fieldset.Content>
<Divider /> </Fieldset>
<Fieldset.Content className={styles.content}> )
{children}
</Fieldset.Content>
</Fieldset>
} }
export default SettingsGroup export default SettingsGroup

View file

@ -3,20 +3,24 @@ import Profile from "./sections/profile"
import SettingsGroup from "../settings-group" import SettingsGroup from "../settings-group"
const SettingsPage = () => { const SettingsPage = () => {
return (<div style={{ return (
display: 'flex', <div
flexDirection: 'column', style={{
gap: 'var(--gap)', display: "flex",
marginBottom: 'var(--gap)', flexDirection: "column",
}}> gap: "var(--gap)",
<h1>Settings</h1> marginBottom: "var(--gap)"
<SettingsGroup title="Profile"> }}
<Profile /> >
</SettingsGroup> <h1>Settings</h1>
<SettingsGroup title="Password"> <SettingsGroup title="Profile">
<Password /> <Profile />
</SettingsGroup> </SettingsGroup>
</div>) <SettingsGroup title="Password">
<Password />
</SettingsGroup>
</div>
)
} }
export default SettingsPage export default SettingsPage

View file

@ -3,95 +3,132 @@ import Cookies from "js-cookie"
import { useState } from "react" import { useState } from "react"
const Password = () => { const Password = () => {
const [password, setPassword] = useState<string>('') const [password, setPassword] = useState<string>("")
const [newPassword, setNewPassword] = useState<string>('') const [newPassword, setNewPassword] = useState<string>("")
const [confirmPassword, setConfirmPassword] = useState<string>('') const [confirmPassword, setConfirmPassword] = useState<string>("")
const { setToast } = useToasts() const { setToast } = useToasts()
const handlePasswordChange = (e: React.ChangeEvent<HTMLInputElement>) => { const handlePasswordChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setPassword(e.target.value) setPassword(e.target.value)
} }
const handleNewPasswordChange = (e: React.ChangeEvent<HTMLInputElement>) => { const handleNewPasswordChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setNewPassword(e.target.value) setNewPassword(e.target.value)
} }
const handleConfirmPasswordChange = (e: React.ChangeEvent<HTMLInputElement>) => { const handleConfirmPasswordChange = (
setConfirmPassword(e.target.value) e: React.ChangeEvent<HTMLInputElement>
} ) => {
setConfirmPassword(e.target.value)
}
const onSubmit = async (e: React.FormEvent<HTMLFormElement>) => { const onSubmit = async (e: React.FormEvent<HTMLFormElement>) => {
e.preventDefault() e.preventDefault()
if (!password || !newPassword || !confirmPassword) { if (!password || !newPassword || !confirmPassword) {
setToast({ setToast({
text: "Please fill out all fields", text: "Please fill out all fields",
type: "error", type: "error"
}) })
} }
if (newPassword !== confirmPassword) { if (newPassword !== confirmPassword) {
setToast({ setToast({
text: "New password and confirm password do not match", text: "New password and confirm password do not match",
type: "error", type: "error"
}) })
} }
const res = await fetch("/server-api/auth/change-password", { const res = await fetch("/server-api/auth/change-password", {
method: "PUT", method: "PUT",
headers: { headers: {
"Content-Type": "application/json", "Content-Type": "application/json",
"Authorization": `Bearer ${Cookies.get("drift-token")}`, Authorization: `Bearer ${Cookies.get("drift-token")}`
}, },
body: JSON.stringify({ body: JSON.stringify({
oldPassword: password, oldPassword: password,
newPassword, newPassword
}), })
}) })
if (res.status === 200) {
setToast({
text: "Password updated successfully",
type: "success"
})
setPassword("")
setNewPassword("")
setConfirmPassword("")
} else {
const data = await res.json()
if (res.status === 200) { setToast({
setToast({ text: data.error ?? "Failed to update password",
text: "Password updated successfully", type: "error"
type: "success", })
}) }
setPassword('') }
setNewPassword('')
setConfirmPassword('')
} else {
const data = await res.json()
setToast({ return (
text: data.error ?? "Failed to update password", <form
type: "error", style={{
}) display: "flex",
} flexDirection: "column",
} gap: "var(--gap)",
maxWidth: "300px"
return ( }}
<form onSubmit={onSubmit}
style={{ >
display: "flex", <div>
flexDirection: "column", <label htmlFor="current-password">Current password</label>
gap: "var(--gap)", <Input
maxWidth: "300px", onChange={handlePasswordChange}
}} minLength={6}
onSubmit={onSubmit} maxLength={128}
> value={password}
<div> id="current-password"
<label htmlFor="current-password">Current password</label> htmlType="password"
<Input onChange={handlePasswordChange} minLength={6} maxLength={128} value={password} id="current-password" htmlType="password" required autoComplete="current-password" placeholder="Current Password" width={"100%"} /> required
</div> autoComplete="current-password"
<div> placeholder="Current Password"
<label htmlFor="new-password">New password</label> width={"100%"}
<Input onChange={handleNewPasswordChange} minLength={6} maxLength={128} value={newPassword} id="new-password" htmlType="password" required autoComplete="new-password" placeholder="New Password" width={"100%"} /> />
</div> </div>
<div> <div>
<label htmlFor="confirm-password">Confirm password</label> <label htmlFor="new-password">New password</label>
<Input onChange={handleConfirmPasswordChange} minLength={6} maxLength={128} value={confirmPassword} id="confirm-password" htmlType="password" required autoComplete="confirm-password" placeholder="Confirm Password" width={"100%"} /> <Input
</div> onChange={handleNewPasswordChange}
<Button htmlType="submit" auto>Change Password</Button> minLength={6}
</form>) maxLength={128}
value={newPassword}
id="new-password"
htmlType="password"
required
autoComplete="new-password"
placeholder="New Password"
width={"100%"}
/>
</div>
<div>
<label htmlFor="confirm-password">Confirm password</label>
<Input
onChange={handleConfirmPasswordChange}
minLength={6}
maxLength={128}
value={confirmPassword}
id="confirm-password"
htmlType="password"
required
autoComplete="confirm-password"
placeholder="Confirm Password"
width={"100%"}
/>
</div>
<Button htmlType="submit" auto>
Change Password
</Button>
</form>
)
} }
export default Password export default Password

View file

@ -4,97 +4,121 @@ import Cookies from "js-cookie"
import { useEffect, useState } from "react" import { useEffect, useState } from "react"
const Profile = () => { const Profile = () => {
const user = useUserData() const user = useUserData()
const [name, setName] = useState<string>() const [name, setName] = useState<string>()
const [email, setEmail] = useState<string>() const [email, setEmail] = useState<string>()
const [bio, setBio] = useState<string>() const [bio, setBio] = useState<string>()
useEffect(() => { useEffect(() => {
console.log(user) if (user?.displayName) setName(user.displayName)
if (user?.displayName) setName(user.displayName) if (user?.email) setEmail(user.email)
if (user?.email) setEmail(user.email) if (user?.bio) setBio(user.bio)
if (user?.bio) setBio(user.bio) }, [user])
}, [user])
const { setToast } = useToasts() const { setToast } = useToasts()
const handleNameChange = (e: React.ChangeEvent<HTMLInputElement>) => { const handleNameChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setName(e.target.value) setName(e.target.value)
} }
const handleEmailChange = (e: React.ChangeEvent<HTMLInputElement>) => { const handleEmailChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setEmail(e.target.value) setEmail(e.target.value)
} }
const handleBioChange = (e: React.ChangeEvent<HTMLTextAreaElement>) => { const handleBioChange = (e: React.ChangeEvent<HTMLTextAreaElement>) => {
setBio(e.target.value) setBio(e.target.value)
} }
const onSubmit = async (e: React.FormEvent<HTMLFormElement>) => { const onSubmit = async (e: React.FormEvent<HTMLFormElement>) => {
e.preventDefault() e.preventDefault()
if (!name && !email && !bio) { if (!name && !email && !bio) {
setToast({ setToast({
text: "Please fill out at least one field", text: "Please fill out at least one field",
type: "error", type: "error"
}) })
return return
} }
const data = { const data = {
displayName: name, displayName: name,
email, email,
bio, bio
} }
const res = await fetch("/server-api/user/profile", { const res = await fetch("/server-api/user/profile", {
method: "PUT", method: "PUT",
headers: { headers: {
"Content-Type": "application/json", "Content-Type": "application/json",
"Authorization": `Bearer ${Cookies.get("drift-token")}`, Authorization: `Bearer ${Cookies.get("drift-token")}`
}, },
body: JSON.stringify(data), body: JSON.stringify(data)
}) })
if (res.status === 200) { if (res.status === 200) {
setToast({ setToast({
text: "Profile updated", text: "Profile updated",
type: "success", type: "success"
}) })
} else { } else {
setToast({ setToast({
text: "Something went wrong updating your profile", text: "Something went wrong updating your profile",
type: "error", type: "error"
}) })
} }
} }
return (<> return (
<Note type="warning" marginBottom={"var(--gap)"}> <>
This information will be publicly available on your profile <Note type="warning" marginBottom={"var(--gap)"}>
</Note> This information will be publicly available on your profile
<form </Note>
style={{ <form
display: "flex", style={{
flexDirection: "column", display: "flex",
gap: "var(--gap)", flexDirection: "column",
maxWidth: "300px", gap: "var(--gap)",
}} maxWidth: "300px"
onSubmit={onSubmit} }}
> onSubmit={onSubmit}
<div> >
<label htmlFor="displayName">Display name</label> <div>
<Input id="displayName" width={"100%"} placeholder="my name" value={name || ''} onChange={handleNameChange} /> <label htmlFor="displayName">Display name</label>
</div> <Input
<div> id="displayName"
<label htmlFor="email">Email</label> width={"100%"}
<Input id="email" htmlType="email" width={"100%"} placeholder="my@email.io" value={email || ''} onChange={handleEmailChange} /> placeholder="my name"
</div> value={name || ""}
<div> onChange={handleNameChange}
<label htmlFor="bio">Biography (max 250 characters)</label> />
<Textarea id="bio" width="100%" maxLength={250} placeholder="I enjoy..." value={bio || ''} onChange={handleBioChange} /> </div>
</div> <div>
<Button htmlType="submit" auto>Submit</Button> <label htmlFor="email">Email</label>
</form></>) <Input
id="email"
htmlType="email"
width={"100%"}
placeholder="my@email.io"
value={email || ""}
onChange={handleEmailChange}
/>
</div>
<div>
<label htmlFor="bio">Biography (max 250 characters)</label>
<Textarea
id="bio"
width="100%"
maxLength={250}
placeholder="I enjoy..."
value={bio || ""}
onChange={handleBioChange}
/>
</div>
<Button htmlType="submit" auto>
Submit
</Button>
</form>
</>
)
} }
export default Profile export default Profile

View file

@ -3,14 +3,16 @@ import PageSeo from "@components/page-seo"
import HomeComponent from "@components/home" import HomeComponent from "@components/home"
import { Page, Text } from "@geist-ui/core" import { Page, Text } from "@geist-ui/core"
import type { GetStaticProps } from "next" import type { GetStaticProps } from "next"
import { InferGetStaticPropsType } from 'next' import { InferGetStaticPropsType } from "next"
type Props = { type Props =
introContent: string | {
introTitle: string introContent: string
rendered: string introTitle: string
} | { rendered: string
error: boolean }
} | {
error: boolean
}
export const getStaticProps: GetStaticProps = async () => { export const getStaticProps: GetStaticProps = async () => {
try { try {
@ -32,21 +34,26 @@ export const getStaticProps: GetStaticProps = async () => {
// Next.js will attempt to re-generate the page: // Next.js will attempt to re-generate the page:
// - When a request comes in // - When a request comes in
// - At most every 60 seconds // - At most every 60 seconds
revalidate: 60, // In seconds revalidate: 60 // In seconds
} }
} catch (err) { } catch (err) {
// If there was an error, it's likely due to the server not running, so we attempt to regenerate the page // If there was an error, it's likely due to the server not running, so we attempt to regenerate the page
return { return {
props: { props: {
error: true, error: true
}, },
revalidate: 10, // In seconds revalidate: 10 // In seconds
} }
} }
} }
// TODO: fix props type // TODO: fix props type
const Home = ({ rendered, introContent, introTitle, error }: InferGetStaticPropsType<typeof getStaticProps>) => { const Home = ({
rendered,
introContent,
introTitle,
error
}: InferGetStaticPropsType<typeof getStaticProps>) => {
return ( return (
<Page className={styles.wrapper}> <Page className={styles.wrapper}>
<PageSeo /> <PageSeo />

View file

@ -69,7 +69,6 @@ export const getServerSideProps: GetServerSideProps = async ({
} }
} }
return { return {
props: { props: {
post: json, post: json,

View file

@ -1,15 +1,27 @@
import { Button, Divider, Text, Fieldset, Input, Page, Note, Textarea } from "@geist-ui/core" import {
Button,
Divider,
Text,
Fieldset,
Input,
Page,
Note,
Textarea
} from "@geist-ui/core"
import PageSeo from "@components/page-seo" import PageSeo from "@components/page-seo"
import styles from "@styles/Home.module.css" import styles from "@styles/Home.module.css"
import SettingsPage from "@components/settings" import SettingsPage from "@components/settings"
const Settings = () => ( const Settings = () => (
<Page width={"100%"}> <Page width={"100%"}>
<PageSeo title="Drift - Settings" /> <PageSeo title="Drift - Settings" />
<Page.Content className={styles.main} style={{ gap: 'var(--gap)', display: 'flex', flexDirection: 'column' }}> <Page.Content
<SettingsPage /> className={styles.main}
</Page.Content> style={{ gap: "var(--gap)", display: "flex", flexDirection: "column" }}
</Page> >
<SettingsPage />
</Page.Content>
</Page>
) )
export default Settings export default Settings

View file

@ -6,6 +6,7 @@ import { errors } from "celebrate"
import secretKey from "@lib/middleware/secret-key" import secretKey from "@lib/middleware/secret-key"
import markdown from "@lib/render-markdown" import markdown from "@lib/render-markdown"
import config from "@lib/config" import config from "@lib/config"
import { ServerInfo } from "@lib/models/ServerInfo"
export const app = express() export const app = express()
@ -19,17 +20,25 @@ app.use("/files", files)
app.use("/admin", admin) app.use("/admin", admin)
app.use("/health", health) app.use("/health", health)
app.get("/welcome", secretKey, (req, res) => { app.get("/welcome", secretKey, async (req, res) => {
const introContent = config.welcome_content const serverInfo = await ServerInfo.findOne({
const introTitle = config.welcome_title where: {
if (!introContent || !introTitle) { id: "1"
return res.status(500).json({ error: "Missing welcome content" }) }
})
if (!serverInfo) {
return res.status(500).json({
message: "Server info not found."
})
} }
const { welcomeMessage, welcomeTitle } = serverInfo
return res.json({ return res.json({
title: introTitle, title: welcomeTitle,
content: introContent, content: welcomeMessage,
rendered: markdown(introContent) rendered: markdown(welcomeMessage)
}) })
}) })

View file

@ -2,6 +2,7 @@ import config from "@lib/config"
import databasePath from "@lib/get-database-path" import databasePath from "@lib/get-database-path"
import { Sequelize } from "sequelize-typescript" import { Sequelize } from "sequelize-typescript"
import { SequelizeStorage, Umzug } from "umzug" import { SequelizeStorage, Umzug } from "umzug"
import { QueryTypes } from "sequelize"
export const sequelize = new Sequelize({ export const sequelize = new Sequelize({
dialect: "sqlite", dialect: "sqlite",
@ -11,10 +12,24 @@ export const sequelize = new Sequelize({
logging: console.log logging: console.log
}) })
if (config.memory_db) { export const initServerInfo = async () => {
console.log("Using in-memory database") const serverInfo = await sequelize.query(
} else { "SELECT * FROM `server-info` WHERE id = '1'",
console.log(`Database path: ${databasePath}`) {
type: QueryTypes.SELECT
}
)
if (serverInfo.length === 0) {
console.log("server-info table not found, creating...")
console.log(
"You can change the welcome message and title on the admin settings page."
)
await sequelize.query("INSERT INTO `server-info` (id) VALUES ('1')", {
type: QueryTypes.INSERT
})
console.log("server-info table created.")
}
} }
export const umzug = new Umzug({ export const umzug = new Umzug({
@ -30,6 +45,12 @@ export const umzug = new Umzug({
export type Migration = typeof umzug._types.migration export type Migration = typeof umzug._types.migration
if (config.memory_db) {
console.log("Using in-memory database")
} else {
console.log(`Database path: ${databasePath}`)
}
// If you're in a development environment, you can manually migrate with `yarn migrate:{up,down}` in the `server` folder // If you're in a development environment, you can manually migrate with `yarn migrate:{up,down}` in the `server` folder
if (config.is_production) { if (config.is_production) {
;(async () => { ;(async () => {

View file

@ -0,0 +1,33 @@
import {
Model,
Column,
Table,
IsUUID,
PrimaryKey,
DataType,
Unique
} from "sequelize-typescript"
@Table({
tableName: "server-info"
})
export class ServerInfo extends Model {
@IsUUID(4)
@PrimaryKey
@Unique
@Column({
type: DataType.UUID,
defaultValue: DataType.UUIDV4
})
id!: string
@Column({
type: DataType.STRING
})
welcomeMessage!: string
@Column({
type: DataType.STRING
})
welcomeTitle!: string
}

View file

@ -3,24 +3,24 @@ import { DataTypes } from "sequelize"
import type { Migration } from "../database" import type { Migration } from "../database"
export const up: Migration = async ({ context: queryInterface }) => export const up: Migration = async ({ context: queryInterface }) =>
Promise.all([ Promise.all([
queryInterface.addColumn("users", "email", { queryInterface.addColumn("users", "email", {
type: DataTypes.STRING, type: DataTypes.STRING,
allowNull: true allowNull: true
}), }),
queryInterface.addColumn("users", "displayName", { queryInterface.addColumn("users", "displayName", {
type: DataTypes.STRING, type: DataTypes.STRING,
allowNull: true, allowNull: true
}), }),
queryInterface.addColumn("users", "bio", { queryInterface.addColumn("users", "bio", {
type: DataTypes.STRING, type: DataTypes.STRING,
allowNull: true, allowNull: true
}), })
]) ])
export const down: Migration = async ({ context: queryInterface }) => export const down: Migration = async ({ context: queryInterface }) =>
Promise.all([ Promise.all([
queryInterface.removeColumn("users", "email"), queryInterface.removeColumn("users", "email"),
queryInterface.removeColumn("users", "displayName"), queryInterface.removeColumn("users", "displayName"),
queryInterface.removeColumn("users", "bio"), queryInterface.removeColumn("users", "bio")
]) ])

View file

@ -0,0 +1,37 @@
"use strict"
import { DataTypes } from "sequelize"
import type { Migration } from "../database"
export const up: Migration = async ({ context: queryInterface }) =>
queryInterface.createTable("server-info", {
id: {
type: DataTypes.INTEGER,
primaryKey: true,
autoIncrement: true,
defaultValue: 1
},
welcomeMessage: {
type: DataTypes.TEXT,
defaultValue:
"## Drift is a self-hostable clone of GitHub Gist. \nIt is a simple way to share code and text snippets with your friends, with support for the following:\n \n - Render GitHub Extended Markdown (including images)\n - User authentication\n - Private, public, and password protected posts\n - Markdown is rendered and stored on the server\n - Syntax highlighting and automatic language detection\n - Drag-and-drop file uploading\n\n If you want to signup, you can join at [/signup](/signup) as long as you have a passcode provided by the administrator (which you don't need for this demo). **This demo is on a memory-only database, so accounts and pastes can be deleted at any time.** \n\nYou can find the source code on [GitHub](https://github.com/MaxLeiter/drift).",
allowNull: true
},
welcomeTitle: {
type: DataTypes.TEXT,
defaultValue: "Welcome to Drift",
allowNull: true
},
createdAt: {
type: DataTypes.DATE,
defaultValue: DataTypes.NOW,
allowNull: true
},
updatedAt: {
type: DataTypes.DATE,
defaultValue: DataTypes.NOW,
allowNull: true
}
})
export const down: Migration = async ({ context: queryInterface }) =>
queryInterface.dropTable("server-info")

View file

@ -4,6 +4,7 @@ import { User } from "@lib/models/User"
import { File } from "@lib/models/File" import { File } from "@lib/models/File"
import { Router } from "express" import { Router } from "express"
import { celebrate, Joi } from "celebrate" import { celebrate, Joi } from "celebrate"
import { ServerInfo } from "@lib/models/ServerInfo"
export const admin = Router() export const admin = Router()
@ -197,3 +198,54 @@ admin.delete("/post/:id", async (req, res, next) => {
next(e) next(e)
} }
}) })
admin.get("/server-info", async (req, res, next) => {
try {
const info = await ServerInfo.findOne({
where: {
id: 1
}
})
res.json(info)
} catch (e) {
next(e)
}
})
admin.put(
"/server-info",
celebrate({
body: {
title: Joi.string().required(),
description: Joi.string().required()
}
}),
async (req, res, next) => {
try {
const { description, title } = req.body
const serverInfo = await ServerInfo.findOne({
where: {
id: 1
}
})
if (!serverInfo) {
return res.status(404).json({
error: "Server info not found"
})
}
await serverInfo.update({
welcomeMessage: description,
welcomeTitle: title
})
res.json({
success: true
})
} catch (e) {
next(e)
}
}
)

View file

@ -195,7 +195,8 @@ auth.post("/signout", secretKey, async (req, res, next) => {
} }
}) })
auth.put("/change-password", auth.put(
"/change-password",
jwt, jwt,
celebrate({ celebrate({
body: { body: {

View file

@ -30,13 +30,14 @@ user.get("/self", jwt, async (req: UserJwtRequest, res, next) => {
} }
}) })
user.put("/profile", user.put(
"/profile",
jwt, jwt,
celebrate({ celebrate({
body: { body: {
displayName: Joi.string().optional().allow(""), displayName: Joi.string().optional().allow(""),
bio: Joi.string().optional().allow(""), bio: Joi.string().optional().allow(""),
email: Joi.string().optional().email().allow(""), email: Joi.string().optional().email().allow("")
} }
}), }),
async (req: UserJwtRequest, res, next) => { async (req: UserJwtRequest, res, next) => {

View file

@ -2,8 +2,10 @@ import { createServer } from "http"
import { app } from "./app" import { app } from "./app"
import config from "./lib/config" import config from "./lib/config"
import "./database" import "./database"
import { initServerInfo } from "./database"
;(async () => { ;(async () => {
// await sequelize.sync() initServerInfo()
createServer(app).listen(config.port, () => createServer(app).listen(config.port, () =>
console.info(`Server running on port ${config.port}`) console.info(`Server running on port ${config.port}`)
) )