add initial

This commit is contained in:
Nareshkumar Rao 2025-04-02 00:17:25 +02:00
commit 792d71523c
39 changed files with 7525 additions and 0 deletions

41
.gitignore vendored Normal file
View File

@ -0,0 +1,41 @@
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
# dependencies
/node_modules
/.pnp
.pnp.*
.yarn/*
!.yarn/patches
!.yarn/plugins
!.yarn/releases
!.yarn/versions
# testing
/coverage
# next.js
/.next/
/out/
# production
/build
# misc
.DS_Store
*.pem
# debug
npm-debug.log*
yarn-debug.log*
yarn-error.log*
.pnpm-debug.log*
# env files (can opt-in for committing if needed)
.env*
# vercel
.vercel
# typescript
*.tsbuildinfo
next-env.d.ts

3
.vscode/settings.json vendored Normal file
View File

@ -0,0 +1,3 @@
{
"editor.formatOnSave": true
}

36
README.md Normal file
View File

@ -0,0 +1,36 @@
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).
## Getting Started
First, run the development server:
```bash
npm run dev
# or
yarn dev
# or
pnpm dev
# or
bun dev
```
Open [http://localhost:3000](http://localhost:3000) with your browser to see the result.
You can start editing the page by modifying `app/page.tsx`. The page auto-updates as you edit the file.
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.
## Learn More
To learn more about Next.js, take a look at the following resources:
- [Next.js Documentation](https://nextjs.org/docs) - learn about Next.js features and API.
- [Learn Next.js](https://nextjs.org/learn) - an interactive Next.js tutorial.
You can check out [the Next.js GitHub repository](https://github.com/vercel/next.js) - your feedback and contributions are welcome!
## Deploy on Vercel
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.
Check out our [Next.js deployment documentation](https://nextjs.org/docs/app/building-your-application/deploying) for more details.

26
cli/cli.ts Normal file
View File

@ -0,0 +1,26 @@
import { PrismaClient } from "@prisma/client";
import argon2 from "argon2";
async function main() {
const args = process.argv.slice(2);
const username = args[0];
const password = args[1];
if (!username || !password) {
console.error("Usage: node cli.js <username>");
process.exit(1);
}
const hash = await argon2.hash(password, {
type: argon2.argon2id,
memoryCost: 19456,
timeCost: 2,
parallelism: 1,
});
const prisma = new PrismaClient();
await prisma.user.deleteMany({ where: { username } });
await prisma.user.create({ data: { username, password: hash } });
}
main();

16
eslint.config.mjs Normal file
View File

@ -0,0 +1,16 @@
import { dirname } from "path";
import { fileURLToPath } from "url";
import { FlatCompat } from "@eslint/eslintrc";
const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);
const compat = new FlatCompat({
baseDirectory: __dirname,
});
const eslintConfig = [
...compat.extends("next/core-web-vitals", "next/typescript"),
];
export default eslintConfig;

5
next.config.ts Normal file
View File

@ -0,0 +1,5 @@
import type { NextConfig } from "next";
const nextConfig: NextConfig = {};
export default nextConfig;

5792
package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

41
package.json Normal file
View File

@ -0,0 +1,41 @@
{
"name": "mainsite",
"version": "0.1.0",
"private": true,
"scripts": {
"dev": "next dev --turbopack",
"build": "next build",
"start": "next start",
"lint": "next lint"
},
"dependencies": {
"@icons-pack/react-simple-icons": "^12.3.0",
"@prisma/client": "^6.5.0",
"argon2": "^0.41.1",
"cheerio": "^1.0.0",
"highlight.js": "^11.11.1",
"jose": "^6.0.10",
"lucide-react": "^0.485.0",
"markdown-it": "^14.1.0",
"next": "15.2.4",
"nodemailer": "^6.10.0",
"react": "^19.0.0",
"react-dom": "^19.0.0",
"react-google-recaptcha": "^3.1.0",
"react-social-icons": "^6.22.0",
"slugify": "^1.6.6"
},
"devDependencies": {
"@eslint/eslintrc": "^3",
"@types/highlight.js": "^9.12.4",
"@types/markdown-it": "^14.1.2",
"@types/node": "^20",
"@types/nodemailer": "^6.4.17",
"@types/react": "^19",
"@types/react-dom": "^19",
"@types/react-google-recaptcha": "^2.1.9",
"eslint": "^9",
"eslint-config-next": "15.2.4",
"typescript": "^5"
}
}

32
prisma/schema.prisma Normal file
View File

@ -0,0 +1,32 @@
datasource db {
provider = "postgresql"
url = env("DATABASE_URL")
}
generator client {
provider = "prisma-client-js"
}
model Post {
id Int @id @default(autoincrement())
slug String @unique @db.Text
title String @db.Text
contentRendered Json @db.Json
contentMarkdown String @db.Text
blurb String @db.Text
publishedDate DateTime @db.Timestamp(6)
is_draft Boolean
tags Tag[]
}
model Tag {
id Int @id @default(autoincrement())
name String @unique
posts Post[]
}
model User {
id Int @id @default(autoincrement())
username String @unique
password String
}

16
src/app/about/page.tsx Normal file
View File

@ -0,0 +1,16 @@
/* eslint-disable react/no-unescaped-entities */
export default function About() {
return (
<>
so hey, i'm naresh!
<br />
<br />
i'm an engineer and i love building things and tinkering on things.
<br />
<br /> i like going on runs and going for rides on my bike.
<br />
<br /> umm, maybe i'll write more here later.
</>
);
}

View File

@ -0,0 +1,53 @@
"use client";
import { usePathname, useRouter } from "next/navigation";
export default function Pagination({
numberOfPages,
pageNumber,
}: {
numberOfPages: number;
pageNumber: number;
}) {
const router = useRouter();
const pathname = usePathname();
const setPageNumber = (page: number) => {
router.push(
pathname + "?" + new URLSearchParams({ page: page.toString() })
);
};
if (numberOfPages <= 1) {
return null;
}
return (
<div
style={{
display: "flex",
justifyContent: "center",
marginTop: "1rem",
gap: "0.5rem",
fontSize: "0.9rem",
}}
>
{Array.from({ length: numberOfPages }, (_, i) => (
<span
key={i + 1}
style={{
width: "0.5rem",
padding: "0.5rem",
cursor: "pointer",
backgroundColor: pageNumber === i + 1 ? "#333" : "#eee",
color: "#222",
justifyContent: "center",
display: "flex",
}}
onClick={() => setPageNumber(i + 1)}
>
{i + 1}
</span>
))}
</div>
);
}

View File

@ -0,0 +1,134 @@
"use client";
import { useState } from "react";
import { useRouter } from "next/navigation";
import { Summary } from "../types";
export default function PostSummary({ metadata }: { metadata: Summary }) {
const [elementColor, setElementColor] = useState("#999");
const router = useRouter();
const hoverStart = () => {
setElementColor("#eee");
};
const hoverEnd = () => {
setElementColor("#999");
};
const navigateToPost = () => {
router.push(`/blog/post/${metadata.slug}`);
};
return (
<div
key={metadata.slug}
style={{
display: "flex",
flexDirection: "column",
borderColor: elementColor,
transition: "border-color 0.3s linear",
borderStyle: "solid",
marginTop: "1rem",
}}
>
<div
style={{
width: "100%",
backgroundColor: elementColor,
color: "#111",
display: "flex",
alignItems: "center",
padding: "0.5rem",
boxSizing: "border-box",
fontSize: "1rem",
cursor: "pointer",
transition: "background-color 0.3s linear",
}}
onMouseEnter={hoverStart}
onMouseLeave={hoverEnd}
onClick={navigateToPost}
>
<span>{metadata.title}</span>
</div>
<div
style={{
display: "flex",
flexDirection: "row",
justifyContent: "space-between",
fontSize: "0.9rem",
maxWidth: "100%",
flexWrap: "wrap",
}}
>
<div
style={{
display: "flex",
justifyContent: "flex-start",
padding: "0.5rem",
gap: 4,
}}
>
tags: {metadata.tags.join(", ")}
</div>
<div
style={{
display: "flex",
justifyContent: "flex-end",
padding: "0.5rem",
gap: 4,
}}
>
<span>Published: </span>
<span>{metadata.publishedDate.toLocaleDateString()}</span>
</div>
</div>
<div
style={{
width: "100%",
height: "2px",
backgroundColor: elementColor,
transition: "background-color 0.3s linear",
}}
></div>
<div
style={{
display: "flex",
flexDirection: "column",
padding: "1rem 0.5rem 0 0.5rem",
}}
>
<div style={{ fontSize: "0.9rem" }}>
{metadata.blurb}
<span style={{ color: "#999" }}>&hellip;</span>
</div>
</div>
<div
style={{
display: "flex",
justifyContent: "flex-end",
marginTop: "0.5rem",
}}
>
<span
style={{
textDecoration: "none",
color: "#222",
transition: "background-color 0.3s linear",
padding: "0.5rem 1rem",
fontSize: "0.9rem",
fontWeight: "500",
display: "inline-flex",
alignItems: "center",
gap: "0.25rem",
cursor: "pointer",
backgroundColor: elementColor,
}}
onMouseEnter={hoverStart}
onMouseLeave={hoverEnd}
onClick={navigateToPost}
>
Read more <span style={{ fontSize: "1rem" }}></span>
</span>
</div>
</div>
);
}

View File

@ -0,0 +1,93 @@
"use client";
import { useRouter } from "next/navigation";
import React from "react";
function Tag({ tag }: { tag: string }) {
const router = useRouter();
return (
<span
style={{
fontWeight: "bold",
cursor: "pointer",
padding: "8px 6px",
display: "inline-block",
transition: "color 0.3s linear, font-size 0.3s ease",
}}
onMouseEnter={(e) => {
e.currentTarget.style.color = "#fff";
}}
onMouseLeave={(e) => {
e.currentTarget.style.color = "";
}}
onClick={() => router.push(`/blog/${tag}`)}
>
{tag}
</span>
);
}
function TagInfo({ tag }: { tag: string }) {
const router = useRouter();
return (
<div
style={{
fontSize: 13,
marginTop: "1rem",
display: "flex",
flexDirection: "column",
gap: "0.5rem",
}}
>
<span>showing posts tagged &quot;{tag}&quot;</span>
<span
style={{
padding: "0.5rem 1rem",
backgroundColor: "#333",
width: "fit-content",
transition: "background-color 0.3s linear",
cursor: "pointer",
userSelect: "none",
}}
onMouseOver={(e) => {
e.currentTarget.style.backgroundColor = "#444";
}}
onMouseLeave={(e) => {
e.currentTarget.style.backgroundColor = "#333";
}}
onClick={() => {
router.push("/blog");
}}
>
see all posts
</span>
</div>
);
}
export default function TagOverview({
currentTag,
tags,
}: {
currentTag: string | null;
tags: string[];
}) {
return (
<>
{!currentTag && (
<div style={{ fontSize: 12, marginTop: "1rem" }}>
tags:&nbsp;
{tags.map((tag, i) => (
<React.Fragment key={tag}>
<Tag tag={tag} />
{i < tags.length - 1 && (
<span style={{ color: "#999" }}>&nbsp;|&nbsp;</span>
)}
</React.Fragment>
))}
</div>
)}
{currentTag && <TagInfo tag={currentTag} />}
</>
);
}

View File

@ -0,0 +1,61 @@
import { getSummaries, getTags } from "../action";
import { titleFont } from "@/components/fonts";
import { notFound } from "next/navigation";
import React from "react";
import PostSummary from "./PostSummary";
import Pagination from "./Pagination";
import TagOverview from "./TagOverview";
async function getPageNumber(
searchParams: Promise<{ page: string | undefined }>
) {
const { page } = await searchParams;
const result = page ? parseInt(page) : 1;
if (isNaN(result) || result < 1) {
notFound();
}
return result;
}
export default async function Blog({
params,
searchParams,
}: {
params: Promise<{ tag: string[] | undefined }>;
searchParams: Promise<{ page: string | undefined }>;
}) {
const pageNumber = await getPageNumber(searchParams);
const { tag } = await params;
const currentTag = tag != null && tag.length >= 1 ? tag[0] : null;
if ((tag?.length ?? 0) > 1) {
notFound();
}
const { metadata, numberOfPages } = await getSummaries(
pageNumber,
currentTag
);
if (pageNumber > numberOfPages) {
notFound();
}
const tags = await getTags();
return (
<>
<div
style={{
display: "flex",
flexDirection: "column",
}}
className={titleFont.className}
>
<span style={{ fontSize: 18 }}>naresh writes</span>
<span style={{ fontSize: 12 }}>...occasionally</span>
</div>
<TagOverview tags={tags} currentTag={currentTag} />
{metadata.length > 0 &&
metadata.map((m) => <PostSummary metadata={m} key={m.slug} />)}
<Pagination numberOfPages={numberOfPages} pageNumber={pageNumber} />
</>
);
}

62
src/app/blog/action.ts Normal file
View File

@ -0,0 +1,62 @@
"use server";
import { notFound } from "next/navigation";
import { Post, Summary } from "./types";
import { PrismaClient } from "@prisma/client";
export async function getTags(): Promise<string[]> {
const prisma = new PrismaClient();
const tags = (await prisma.tag.findMany({ select: { name: true } })).map(
(tag) => tag.name
);
return tags;
}
const PAGE_SIZE = 10;
export async function getPost(slug: string): Promise<Post | null> {
const primsa = new PrismaClient();
const result = await primsa.post.findUnique({
where: { slug: slug },
include: { tags: { select: { name: true } } },
omit: {
contentMarkdown: true,
},
});
if (result == null) {
notFound();
}
const post = {
...result,
tags: result.tags.map((tag) => tag.name),
contentRendered: result.contentRendered as { __html: string },
contentMarkdown: undefined,
};
return post;
}
export async function getSummaries(
pageNumber: number,
tagFilter: string | null = null
): Promise<{ metadata: Summary[]; numberOfPages: number }> {
const prisma = new PrismaClient();
let filter = {};
if (tagFilter) {
filter = { where: { tags: { some: { name: { contains: tagFilter } } } } };
}
const numberOfPosts = await prisma.post.count(filter);
const numberOfPages = Math.ceil(numberOfPosts / PAGE_SIZE);
const posts = (
await prisma.post.findMany({
...filter,
omit: { contentMarkdown: true, contentRendered: true },
include: { tags: { select: { name: true } } },
orderBy: { publishedDate: "desc" },
skip: PAGE_SIZE * (pageNumber - 1),
take: PAGE_SIZE,
})
).map((obj) => {
return { ...obj, tags: obj.tags.map((tag) => tag.name) };
});
return { metadata: posts, numberOfPages: numberOfPages };
}

21
src/app/blog/error.tsx Normal file
View File

@ -0,0 +1,21 @@
"use client"; // Error boundaries must be Client Components
import { useEffect } from "react";
export default function Error({
error,
}: {
error: Error & { digest?: string };
reset: () => void;
}) {
useEffect(() => {
// Log the error to an error reporting service
console.error(error);
}, [error]);
return (
<div>
<h2>Something went wrong!</h2>
</div>
);
}

View File

@ -0,0 +1,74 @@
"use client";
import Form from "next/form";
import { handleLogin } from "./action";
export default function FormComponent() {
return (
<Form action={handleLogin}>
<div
style={{
display: "flex",
flexDirection: "column",
gap: "1rem",
fontSize: 14,
}}
>
<div style={{ display: "flex", flexDirection: "column" }}>
Username:
<input
type="text"
style={{
height: "2rem",
backgroundColor: "#333",
borderStyle: "none",
color: "#eee",
fontSize: 16,
padding: "0.5rem 1rem",
}}
name="username"
required
/>
</div>
<div style={{ display: "flex", flexDirection: "column" }}>
Password:
<input
type="password"
style={{
height: "2rem",
backgroundColor: "#333",
borderStyle: "none",
color: "#eee",
fontSize: 16,
padding: "0.5rem 1rem",
}}
name="password"
required
/>
</div>
<div>
<button
style={{
backgroundColor: "#333",
borderStyle: "none",
color: "#eee",
fontSize: "1rem",
padding: "0.5rem 1rem",
transition: "background-color 0.3s linear",
cursor: "pointer",
}}
onMouseOver={(e) => {
e.currentTarget.style.backgroundColor = "#444";
}}
onMouseOut={(e) => {
e.currentTarget.style.backgroundColor = "#333";
}}
type="submit"
>
Login
</button>
</div>
</div>
</Form>
);
}

View File

@ -0,0 +1,27 @@
"use server";
import { PrismaClient } from "@prisma/client";
import argon2 from "argon2";
import { setSession } from "../write/auth";
import { redirect, RedirectType } from "next/navigation";
export async function handleLogin(data: FormData) {
const prisma = new PrismaClient();
const username = data.get("username")?.toString();
const password = data.get("password")?.toString();
if (!username || !password) {
throw new Error("Missing username or password");
}
const user = await prisma.user.findUnique({ where: { username } });
if (!user) {
redirect("/blog/login?error=Invalid%20credentials", RedirectType.replace);
}
if (await argon2.verify(user.password, password)) {
setSession();
redirect("/blog/write", RedirectType.replace);
} else {
redirect("/blog/login?error=Invalid%20credentials", RedirectType.replace);
}
}

View File

@ -0,0 +1,12 @@
"use server";
import { redirect } from "next/navigation";
import { isLoggedIn } from "../write/auth";
import FormComponent from "./Form";
export default async function Login() {
if (await isLoggedIn()) {
redirect("/blog/write");
}
return <FormComponent />;
}

View File

@ -0,0 +1,117 @@
"use client";
import { useRouter } from "next/navigation";
import * as Types from "../../types";
import "highlight.js/styles/github-dark.css";
export default function PostDisplay({ post }: { post: Types.Post }) {
const router = useRouter();
return (
<>
<div
style={{
display: "flex",
flexDirection: "column",
borderColor: "#999",
transition: "border-color 0.3s linear",
borderStyle: "solid",
marginTop: "1rem",
}}
>
<div
style={{
width: "100%",
backgroundColor: "#999",
color: "#111",
display: "flex",
alignItems: "center",
padding: "0.5rem",
boxSizing: "border-box",
fontSize: "1rem",
}}
>
<span>{post.title}</span>
</div>
<div
style={{
display: "flex",
flexDirection: "row",
justifyContent: "space-between",
fontSize: "0.9rem",
maxWidth: "100%",
flexWrap: "wrap",
}}
>
<div
style={{
display: "flex",
justifyContent: "flex-start",
padding: "0.5rem",
gap: 4,
}}
>
tags: {post.tags.join(", ")}
</div>
<div
style={{
display: "flex",
justifyContent: "flex-end",
padding: "0.5rem",
gap: 4,
}}
>
<span>Published: </span>
<span>{post.publishedDate.toLocaleDateString()}</span>
</div>
</div>
<div
style={{
width: "100%",
height: "2px",
backgroundColor: "#999",
}}
></div>
<div
style={{
display: "flex",
flexDirection: "column",
padding: "1rem 0.5rem 0 0.5rem",
}}
>
<div style={{ fontSize: "0.9rem" }}>
<div dangerouslySetInnerHTML={post.contentRendered} />
</div>
</div>
</div>
<div
style={{
display: "flex",
justifyContent: "flex-end",
marginTop: "1rem",
userSelect: "none",
cursor: "pointer",
backgroundColor: "transparent",
}}
>
<span
style={{
backgroundColor: "#999",
color: "#111",
fontSize: "0.8rem",
padding: "0.5rem",
transition: "background-color 0.3s linear",
}}
onMouseOver={(e) => {
e.currentTarget.style.backgroundColor = "#eee";
}}
onMouseOut={(e) => {
e.currentTarget.style.backgroundColor = "#999";
}}
onClick={() => router.push("/blog")}
>
return to all posts
</span>
</div>
</>
);
}

View File

@ -0,0 +1,20 @@
import { notFound } from "next/navigation";
import { getPost } from "../../action";
import PostDisplay from "./PostDisplay";
export default async function Post({
params,
}: {
params: Promise<{ slug: string }>;
}) {
const { slug } = await params;
const post = await getPost(slug);
if (!post) {
notFound();
}
return (
<>
<PostDisplay post={post} />
</>
);
}

16
src/app/blog/types.ts Normal file
View File

@ -0,0 +1,16 @@
export interface PostBase {
title: string;
publishedDate: Date;
slug: string;
tags: string[];
is_draft: boolean;
}
export interface Summary extends PostBase {
blurb: string;
}
export interface Post extends PostBase {
contentMarkdown: string | undefined;
contentRendered: { __html: string } | undefined;
}

View File

@ -0,0 +1,120 @@
import MarkdownIt from "markdown-it";
import hljs from "highlight.js";
import "highlight.js/styles/github-dark.css";
import { KeyboardEvent } from "react";
const md = new MarkdownIt({
linkify: true,
typographer: true,
highlight: (str, lang) => {
if (lang && hljs.getLanguage(lang)) {
try {
return hljs.highlight(str, { language: lang }).value;
} catch {}
}
return "";
},
});
export default function ContentArea({
content,
setContent,
}: {
content: string;
setContent: (e: string) => void;
}) {
const renderedContent =
content.trim() != ""
? { __html: md.render(content) }
: { __html: "content is previewed here!" };
return (
<div
style={{
display: "flex",
flexDirection: "column",
gap: 8,
fontSize: 14,
marginTop: "1rem",
}}
>
<span>content:</span>
<textarea
style={{
resize: "none",
width: "100%",
height: "10rem",
background: "#333",
borderStyle: "none",
color: "#eee",
}}
onChange={(e) => setContent(e.target.value)}
onKeyDown={(e) => {
handleTabbing(e);
setContent(e.currentTarget.value);
}}
value={content}
/>
<div
style={{
border: "1px solid #999",
width: "100%",
padding: "0.5rem",
boxSizing: "border-box",
fontSize: "0.8rem",
}}
dangerouslySetInnerHTML={renderedContent}
></div>
</div>
);
}
function handleTabbing(e: KeyboardEvent<HTMLTextAreaElement>) {
if (e.key == "Tab") {
e.preventDefault();
const start = e.currentTarget.selectionStart;
const end = e.currentTarget.selectionEnd;
const value = e.currentTarget.value;
if (e.shiftKey) {
const currentLineStart = value.lastIndexOf("\n", start - 1) + 1;
const currentLineEnd = value.indexOf("\n", start);
const currentLine = value.substring(
currentLineStart,
currentLineEnd === -1 ? value.length : currentLineEnd
);
const numSpaces = currentLine.search(/\S|$/);
let newSpaces = numSpaces;
if (numSpaces % 4 === 0) {
newSpaces = Math.max(0, numSpaces - 4);
} else {
newSpaces = Math.max(0, numSpaces - (numSpaces % 4));
}
const newLine = " ".repeat(newSpaces) + currentLine.trimStart();
const newValue =
value.substring(0, currentLineStart) +
newLine +
value.substring(currentLineEnd === -1 ? value.length : currentLineEnd);
e.currentTarget.value = newValue;
e.currentTarget.selectionStart = e.currentTarget.selectionEnd =
currentLineStart + newSpaces;
} else {
e.currentTarget.value =
value.substring(0, start) + " " + value.substring(end, value.length);
e.currentTarget.selectionStart = e.currentTarget.selectionEnd = start + 4;
}
} else if (e.key == "Enter") {
e.preventDefault();
const start = e.currentTarget.selectionStart;
const value = e.currentTarget.value;
const beforeCursor = value.substring(0, start);
const afterCursor = value.substring(start);
const lastLineStart = beforeCursor.lastIndexOf("\n") + 1;
const currentLine = beforeCursor.substring(lastLineStart);
const indentMatch = currentLine.match(/^(\s*)/);
const indent = indentMatch ? indentMatch[1] : "";
// Add new line with same indentation
e.currentTarget.value = beforeCursor + "\n" + indent + afterCursor;
e.currentTarget.selectionStart = e.currentTarget.selectionEnd =
start + 1 + indent.length;
}
}

View File

@ -0,0 +1,114 @@
"use client";
import { useState } from "react";
import ContentArea from "./ContentArea";
import { Post } from "../types";
import { savePostServer } from "./action";
export default function Write({ post }: { post?: Post }) {
const [title, setTitle] = useState(post?.title ?? "");
const [tagsString, setTagsString] = useState<string>(
post?.tags.join(", ") ?? ""
);
const tags = tagsString.split(",").map((tag) => tag.trim());
const [content, setContent] = useState(post?.contentMarkdown ?? "");
const enabled =
title.trim() !== "" && content.trim() !== "" && tags.length > 0;
const savePost = async (is_draft: boolean) => {
if (!enabled) return;
await savePostServer(title, content, tags, is_draft, post?.slug);
};
return (
<>
<div style={{ marginBottom: "1rem" }}>write your post</div>
<div
style={{
display: "flex",
flexDirection: "column",
gap: 8,
fontSize: 14,
}}
>
<span>title:</span>
<input
type="text"
value={title}
onChange={(e) => setTitle(e.target.value)}
style={{
borderStyle: "none",
backgroundColor: "#333",
color: "#eee",
fontSize: 16,
padding: 10,
}}
/>
<span>tags: (comma separated)</span>
<input
type="text"
value={tagsString}
onChange={(e) => setTagsString(e.target.value)}
style={{
borderStyle: "none",
backgroundColor: "#333",
color: "#eee",
fontSize: 15,
}}
/>
</div>
<ContentArea content={content} setContent={setContent} />
<div
style={{
display: "flex",
gap: 8,
flexDirection: "row",
justifyContent: "space-evenly",
marginTop: "1rem",
}}
>
<div
style={{
padding: "0.5rem 1rem",
backgroundColor: enabled ? "#333" : "#222",
userSelect: "none",
cursor: enabled ? "pointer" : "default",
transition: "background-color 0.3s linear",
}}
onMouseOver={(e) => {
if (!enabled) return;
e.currentTarget.style.backgroundColor = "#444";
}}
onMouseLeave={(e) => {
if (!enabled) return;
e.currentTarget.style.backgroundColor = "#333";
}}
onClick={() => savePost(true)}
>
save as draft
</div>
<div
style={{
padding: "0.5rem 1rem",
backgroundColor: enabled ? "#353" : "#222",
userSelect: "none",
cursor: enabled ? "pointer" : "default",
transition: "background-color 0.3s linear",
}}
onMouseOver={(e) => {
if (!enabled) return;
e.currentTarget.style.backgroundColor = "#464";
}}
onMouseLeave={(e) => {
if (!enabled) return;
e.currentTarget.style.backgroundColor = "#353";
}}
onClick={() => savePost(false)}
>
publish
</div>
</div>
</>
);
}

View File

@ -0,0 +1,25 @@
import { notFound, redirect } from "next/navigation";
import { getPost } from "../../action";
import { Post } from "../../types";
import Write from "../Write";
import { isLoggedIn } from "../auth";
export default async function WritePage({
params,
}: {
params: Promise<{ slug: string[] | undefined }>;
}) {
if (!(await isLoggedIn())) {
redirect("/blog/login");
}
const slug = (await params).slug?.[0];
let post: Post | undefined = undefined;
if (slug) {
post = (await getPost(slug)) ?? undefined;
if (post == null) {
notFound();
}
}
return <Write post={post} />;
}

View File

@ -0,0 +1,58 @@
"use server";
import { redirect } from "next/navigation";
import { PrismaClient } from "@prisma/client";
import MarkdownIt from "markdown-it";
import hljs from "highlight.js";
import * as cheerio from "cheerio";
import slugify from "slugify";
export async function savePostServer(
title: string,
contentMarkdown: string,
tags: string[],
is_draft: boolean,
existingSlug?: string
) {
const prisma = new PrismaClient();
const md = new MarkdownIt({
linkify: true,
typographer: true,
highlight: (str, lang) => {
if (lang && hljs.getLanguage(lang)) {
try {
return hljs.highlight(str, { language: lang }).value;
} catch {}
}
return "";
},
});
const contentRendered = { __html: md.render(contentMarkdown) };
const blurb = cheerio.load(contentRendered.__html).text().substring(0, 200);
const slug = slugify(title, { lower: true, strict: true });
if (existingSlug) {
await prisma.post.delete({ where: { slug: existingSlug } });
}
await prisma.post.create({
data: {
title,
slug,
publishedDate: new Date(),
tags: {
connectOrCreate: tags.map((tag) => ({
where: { name: tag },
create: { name: tag },
})),
},
blurb,
contentMarkdown,
contentRendered,
is_draft,
},
});
redirect("/blog");
}

View File

@ -0,0 +1,50 @@
"use server";
import { jwtVerify, SignJWT } from "jose";
import { cookies } from "next/headers";
const SECRET_KEY = process.env.SESSION_SECRET;
const encodedKey = new TextEncoder().encode(SECRET_KEY);
export type SessionPayload = { admin: true };
export async function encrypt(payload: SessionPayload) {
return new SignJWT(payload)
.setProtectedHeader({ alg: "HS256" })
.setIssuedAt()
.setExpirationTime("7d")
.sign(encodedKey);
}
export async function decrypt(
token: string | undefined = ""
): Promise<SessionPayload | null> {
try {
const { payload } = await jwtVerify(token, encodedKey, {
algorithms: ["HS256"],
});
return payload as SessionPayload;
} catch {
return null;
}
}
export async function isLoggedIn() {
const cookieStore = (await cookies()).get("session")?.value;
const session = await decrypt(cookieStore);
if (session != null && session.admin) {
setSession();
return true;
}
return false;
}
export async function setSession() {
const expiresAt = new Date(Date.now() + 7 * 24 * 60 * 60 * 1000);
(await cookies()).set("session", await encrypt({ admin: true }), {
httpOnly: true,
secure: true,
expires: expiresAt,
sameSite: "lax",
path: "/",
});
}

85
src/app/contact/page.tsx Normal file
View File

@ -0,0 +1,85 @@
/* eslint-disable react/no-unescaped-entities */
"use client";
import SubmitContact from "@/components/contact";
import { useActionState } from "react";
import ReCAPTCHA from "react-google-recaptcha";
const RECAPTCHA_SITE_KEY = process.env.RECAPTCHA_SITE_KEY;
export default function Contact() {
const inputStyle = {
backgroundColor: "#111",
color: "#eee",
borderWidth: 0,
padding: "1rem",
};
const [state, formAction, pending] = useActionState(SubmitContact, null);
if (!RECAPTCHA_SITE_KEY) {
throw new Error("ReCAPTCHA not configured correctly");
}
return (
<>
Need to get in touch?
<br />
<br />
<form
action={formAction}
style={{ display: "flex", flexDirection: "column", gap: "1rem" }}
>
<input
style={inputStyle}
type="text"
name="name"
placeholder="Your Name"
required
/>
<input
style={inputStyle}
type="email"
name="email"
placeholder="Your Email"
required
/>
<textarea
style={{ ...inputStyle, resize: "none", height: "5rem" }}
name="message"
placeholder="Your Message"
required
></textarea>
{state && (
<span
style={{
padding: "1rem",
backgroundColor: "error" in state ? "#f003" : "#0f03",
}}
>
{"error" in state && state.error}
{"success" in state && "Submitted successfully!"}
</span>
)}
<ReCAPTCHA sitekey={RECAPTCHA_SITE_KEY} theme="dark" />
<button
type="submit"
style={{
...inputStyle,
backgroundColor: pending ? "#333" : "#111",
cursor: pending ? "default" : "pointer",
}}
onMouseEnter={(e) => {
if (!pending) e.currentTarget.style.backgroundColor = "#333";
}}
onMouseLeave={(e) => {
if (!pending) e.currentTarget.style.backgroundColor = "#111";
}}
disabled={pending}
>
{pending ? "Sending..." : "Submit"}
</button>
</form>
</>
);
}

6
src/app/globals.css Normal file
View File

@ -0,0 +1,6 @@
body{
background-color: #222;
color: #eee;
margin: 0;
padding:0;
}

49
src/app/layout.tsx Normal file
View File

@ -0,0 +1,49 @@
import { Metadata } from "next";
import "./globals.css";
import Title from "@/components/Title";
import Navbar from "@/components/Navbar";
import { bodyFont } from "@/components/fonts";
export const metadata: Metadata = {
title: "nrx.sh",
description: "naresh's site",
};
export default function RootLayout({
children,
}: Readonly<{
children: React.ReactNode;
}>) {
return (
<html lang="en">
<body>
<div
style={{
display: "flex",
justifyContent: "center",
alignItems: "center",
flexDirection: "column",
}}
>
<Title />
<Navbar />
<div
className={bodyFont.className}
style={{
backgroundColor: "#1D1D1D",
marginTop: "2rem",
width: "600px",
maxWidth: "100vw",
padding: "2rem",
boxSizing: "border-box",
fontSize: "1.1rem",
}}
>
{children}
</div>
</div>
</body>
</html>
);
}

65
src/app/links/page.tsx Normal file
View File

@ -0,0 +1,65 @@
/* eslint-disable react/no-unescaped-entities */
"use client";
import { SiGitea } from "@icons-pack/react-simple-icons";
import { NotebookText } from "lucide-react";
import { SocialIcon } from "react-social-icons";
export default function Links() {
return (
<>
here are some of my links:
<br />
<br />
<div
style={{
display: "flex",
flexDirection: "column",
gap: "1rem",
}}
>
<Link href="https://blog.nrx.sh">
<NotebookText size={26} style={{ padding: 10 }} /> My Blog {"<3"}
</Link>
<Link href="https://github.com/naresh97">
<SocialIcon network="github" bgColor="#111" /> GitHub
</Link>
<Link href="https://linkedin.com/in/nareshkumar-rao">
<SocialIcon network="linkedin" bgColor="#111" /> LinkedIn
</Link>
<Link href="https://git.nrx.sh">
<SiGitea size={40} /> Private Git Repo
</Link>
</div>
</>
);
}
function Link({ href, children }: { href: string; children: React.ReactNode }) {
return (
<div
style={{
display: "flex",
gap: "10px",
alignItems: "center",
width: "100%",
backgroundColor: "#111",
padding: "1rem",
transition: "background-color 0.3s ease",
borderRadius: "0.3rem",
cursor: "pointer",
boxSizing: "border-box",
}}
onMouseEnter={(e) => {
e.currentTarget.style.backgroundColor = "#333";
}}
onMouseLeave={(e) => {
e.currentTarget.style.backgroundColor = "#111";
}}
onClick={() => window.open(href, "_blank")}
>
{children}
</div>
);
}

34
src/app/not-found.tsx Normal file
View File

@ -0,0 +1,34 @@
/* eslint-disable react/no-unescaped-entities */
import { titleFont } from "@/components/fonts";
import Link from "next/link";
export default function NotFound() {
return (
<>
<span style={{ fontSize: 32 }} className={titleFont.className}>
404
</span>
<br />
<br />
shit... something is fucked and now you're here
<br />
i'll look into it, at some point
<br />
but for now, maybe you'd like to
<br />
<br />
<Link
href="/"
style={{
textDecoration: "none",
backgroundColor: "#aaa",
color: "#222",
padding: "0.5rem",
}}
>
return home?
</Link>
</>
);
}

20
src/app/page.tsx Normal file
View File

@ -0,0 +1,20 @@
/* eslint-disable react/no-unescaped-entities */
"use client";
import React from "react";
export default function App() {
return (
<>
hi,
<br />
<br /> thanks for stopping by.
<br />
<br /> this is still a work in progress, so it's a little sparse.
<br />
<br />
<br /> naresh.
</>
);
}

59
src/components/Navbar.tsx Normal file
View File

@ -0,0 +1,59 @@
"use client";
import { useState } from "react";
import React from "react";
import { navBarFont } from "./fonts";
import {
PAGES,
Pages,
pathNameFromSelectedPage,
selectedPageFromPathName,
} from "./pages";
import { usePathname, useRouter } from "next/navigation";
export default function Navbar() {
const [hoveredPage, setHoveredPage] = useState<Pages | null>(null);
const pathName = usePathname();
const selectedPage = selectedPageFromPathName(pathName);
const router = useRouter();
const navbarItem = (page: Pages): React.CSSProperties => ({
padding: "0.3rem 0.7rem",
cursor: "pointer",
backgroundColor:
selectedPage == page || hoveredPage === page ? "#eee" : "transparent",
color: selectedPage == page || hoveredPage == page ? "#222" : "#eee",
transition: "all 0.3s ease",
});
return (
<nav
className={navBarFont.className}
style={{
fontWeight: "700",
display: "flex",
flexDirection: "row",
gap: "1rem",
fontSize: "1.1rem",
flexWrap: "wrap",
maxWidth: "90vw",
justifyContent: "center",
}}
>
{PAGES.map((page, index) => (
<React.Fragment key={page}>
<div
style={navbarItem(page)}
onClick={() => router.push(pathNameFromSelectedPage(page))}
onMouseEnter={() => setHoveredPage(page)}
onMouseLeave={() => setHoveredPage(null)}
>
<div style={{ userSelect: "none" }}>{page}</div>
</div>
{index < PAGES.length - 1 && <span>|</span>}
</React.Fragment>
))}
</nav>
);
}

5
src/components/Title.tsx Normal file
View File

@ -0,0 +1,5 @@
import { titleFont } from "./fonts";
export default function Title() {
return <h1 className={titleFont.className}>nrx.sh</h1>;
}

76
src/components/contact.ts Normal file
View File

@ -0,0 +1,76 @@
"use server";
import { createTransport } from "nodemailer";
const RECAPTCHA_SECRET_KEY = process.env.RECAPTCHA_SECRET_KEY;
export type SubmitContactReturn = { error: string } | { success: true };
export default async function SubmitContact(
prevState: unknown,
data: FormData
): Promise<SubmitContactReturn> {
if (!RECAPTCHA_SECRET_KEY) {
console.error(
"RECAPTCHA_SECRET_KEY is not set. Please check your environment variables."
);
throw new Error("Server error: RECAPTCHA not configure correctly.");
}
const name = data.get("name");
const email = data.get("email");
const message = data.get("message");
const recaptcha = data.get("g-recaptcha-response");
if (!name || !email || !message) {
return { error: "All fields are required." };
}
if (!recaptcha) {
return { error: "Please complete the reCAPTCHA." };
}
try {
const recaptchaResponse = await fetch(
`https://www.google.com/recaptcha/api/siteverify`,
{
method: "POST",
headers: {
"Content-Type": "application/x-www-form-urlencoded",
},
body: `secret=${RECAPTCHA_SECRET_KEY}&response=${recaptcha}`,
}
);
const recaptchaData = await recaptchaResponse.json();
if (!recaptchaData.success) {
return { error: "reCAPTCHA verification failed." };
}
} catch (e) {
console.error(e);
return { error: "Could not reach reCAPTCHA for verification." };
}
try {
const transporter = createTransport({
host: process.env.MAIL_HOST ?? "localhost",
port: process.env.MAIL_PORT ? parseInt(process.env.MAIL_PORT) : undefined,
secure: true,
auth: {
user: process.env.MAIL_USER,
pass: process.env.MAIL_PASS,
},
});
await transporter.sendMail({
from: process.env.MAIL_FROM,
to: process.env.MAIL_TO,
subject: `From ${name}`,
replyTo: email.toString(),
text: `Name:${name}\nEmail: ${email}\nMessage:\n ${message}`,
});
} catch (e) {
console.error(e);
return { error: "Failed to send email." };
}
return { success: true };
}

15
src/components/fonts.ts Normal file
View File

@ -0,0 +1,15 @@
import { Doto, Sixtyfour, Space_Grotesk } from "next/font/google";
export const titleFont = Sixtyfour({
subsets: ["latin"],
});
export const navBarFont = Doto({
subsets: ["latin"],
weight: "900",
});
export const bodyFont = Space_Grotesk({
subsets: ["latin"],
weight: "400",
});

19
src/components/pages.ts Normal file
View File

@ -0,0 +1,19 @@
export type Pages = "home" | "about" | "links" | "contact" | "blog";
export const PAGES: Pages[] = ["home", "about", "blog", "links", "contact"];
export function selectedPageFromPathName(pathName: string): Pages {
if (pathName === "/") {
return "home";
}
if (pathName.includes("/blog/")) {
return "blog";
}
return pathName.replace("/", "") as Pages;
}
export function pathNameFromSelectedPage(page: Pages): string {
if (page === "home") {
return "/";
}
return `/${page}`;
}

27
tsconfig.json Normal file
View File

@ -0,0 +1,27 @@
{
"compilerOptions": {
"target": "ES2017",
"lib": ["dom", "dom.iterable", "esnext"],
"allowJs": true,
"skipLibCheck": true,
"strict": true,
"noEmit": true,
"esModuleInterop": true,
"module": "esnext",
"moduleResolution": "bundler",
"resolveJsonModule": true,
"isolatedModules": true,
"jsx": "preserve",
"incremental": true,
"plugins": [
{
"name": "next"
}
],
"paths": {
"@/*": ["./src/*"]
}
},
"include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts"],
"exclude": ["node_modules"]
}