mirror of
https://github.com/SrIzan10/hctv.git
synced 2026-06-06 00:56:56 +00:00
Compare commits
110 Commits
feat/proto
...
feat/sdk-m
| Author | SHA1 | Date | |
|---|---|---|---|
| 7548390c1d | |||
| 398d4113c8 | |||
| 32c101934d | |||
| 48e00bada4 | |||
| bc69136133 | |||
| a96939684b | |||
| ed1608b8e3 | |||
| f4f653614d | |||
| 5fca354c58 | |||
| b4ad29853a | |||
| cf2f0ac86d | |||
| f57dec65e0 | |||
| 4c7ddeeb72 | |||
| 2a4a1adcd8 | |||
| 107982dbec | |||
| a75d9e3795 | |||
| 5336541010 | |||
| dd71b822ed | |||
| d343335b8e | |||
| 892cb7ab87 | |||
| b274903dc1 | |||
| b1c20a374a | |||
| 0b6b23c42d | |||
| 008db9e2c8 | |||
| 24bfcff68a | |||
| dc2b01ae21 | |||
| ef4563cc7c | |||
| 936b853536 | |||
| 60387773bf | |||
| eac101764b | |||
| 5f43567d89 | |||
|
|
bf86bafbe8 | ||
|
|
8f2516cabe | ||
|
|
d2cd99d3d4 | ||
|
|
c8e3b63cc5 | ||
| 60e01d8207 | |||
| 9e0ca29a2c | |||
| ae4d88a9e0 | |||
| 5735074af9 | |||
| 899e8f1054 | |||
| acd8e0d980 | |||
| cb10ee1855 | |||
| b0eb5d4430 | |||
| d8b9803019 | |||
| 2ceb813a98 | |||
| 6222f9dafe | |||
| c7bb9aef72 | |||
| 7456e80473 | |||
| a0cabbfa63 | |||
| 2f8ac7d343 | |||
| 0157eff9f3 | |||
| ebcb062b6a | |||
| fdc8e0f33c | |||
| eeb44dfae7 | |||
|
|
0e9f0a54dd | ||
|
|
5d81d32276 | ||
|
|
fe21d19250 | ||
| eac736b9fb | |||
| 381f4fc523 | |||
| 7d350cfc04 | |||
| 2dfbab5d0e | |||
| 4eef997d63 | |||
| 7574b94933 | |||
| 6c26ca9d2f | |||
| a1727b9a3d | |||
| f486c3b28e | |||
| 8e86be97d1 | |||
| 099b321b79 | |||
| 6fdadbec28 | |||
| 92cde437af | |||
| 28cbe4e8ed | |||
| 09d099d0ee | |||
| 5c99fee95d | |||
| df845b5601 | |||
| d4a6516157 | |||
| 17bbba7df3 | |||
|
|
1e27c7e77a | ||
|
|
80595d6299 | ||
|
|
aa9d0c1ca5 | ||
|
|
45894fc900 | ||
| ddbdf3caf9 | |||
| 80a8e670e1 | |||
| 3e5824093e | |||
| 75d6e648f9 | |||
|
|
1fadaa3600 | ||
|
|
7262b0e5c2 | ||
|
|
70832c7de8 | ||
|
|
61972da255 | ||
| 221aff0050 | |||
| 5b6addac9a | |||
| 5add3b0e5d | |||
| b623de5bdd | |||
| cc15a06ffb | |||
| c0f3e9d52e | |||
| a22dcf0746 | |||
|
|
b4d3cd5bb8 | ||
|
|
d5c02889de | ||
| c0657cc1ce | |||
| d97add9659 | |||
| 8f07dbadf3 | |||
| 21ab8a5e4f | |||
| 689c410828 | |||
| 593baa6505 | |||
| 786a2afb6c | |||
| 75f25eb8fe | |||
| 0e500037c4 | |||
| b49318f9e6 | |||
| 927d7d1bda | |||
| d1f5cc7a6d | |||
| 0afc54f0bf |
2
.github/workflows/docker.yml
vendored
2
.github/workflows/docker.yml
vendored
@@ -46,7 +46,6 @@ jobs:
|
||||
|
||||
mkdir -p apps/web/src/lib/instrumentation/
|
||||
|
||||
export SLACK_TOKEN=${{ secrets.SLACK_TOKEN }}
|
||||
./slack-import-emojis-bin default
|
||||
|
||||
cp emojis.json apps/web/
|
||||
@@ -63,6 +62,7 @@ jobs:
|
||||
secrets: |
|
||||
TURBO_TOKEN=${{ secrets.TURBO_TOKEN }}
|
||||
TURBO_TEAM=${{ secrets.TURBO_TEAM }}
|
||||
SENTRY_AUTH_TOKEN=${{ secrets.SENTRY_AUTH_TOKEN }}
|
||||
chat:
|
||||
name: Push chat module to Docker Hub
|
||||
runs-on: ubuntu-latest
|
||||
|
||||
5
.gitignore
vendored
5
.gitignore
vendored
@@ -28,6 +28,7 @@ yarn-error.log*
|
||||
# local env files
|
||||
.env*.local
|
||||
.env*
|
||||
!.env.example
|
||||
|
||||
# vercel
|
||||
.vercel
|
||||
@@ -47,4 +48,6 @@ packages/db/generated/client
|
||||
slack-import-emojis/target
|
||||
**/*/emojis.json
|
||||
|
||||
.idea
|
||||
.idea
|
||||
|
||||
/apps/docs/src/content/docs/typedoc-sdk
|
||||
343
AGENTS.md
Normal file
343
AGENTS.md
Normal file
@@ -0,0 +1,343 @@
|
||||
# Agent Guidelines for HackClub.tv
|
||||
|
||||
This document provides essential information for AI coding agents working on the HackClub.tv codebase.
|
||||
|
||||
## Project Overview
|
||||
|
||||
HackClub.tv is a live streaming platform built with Next.js 16, Prisma, and Turbo monorepo architecture.
|
||||
|
||||
- **Monorepo**: Turborepo with pnpm workspaces
|
||||
- **Apps**: web (Next.js), chat (Hono), docs
|
||||
- **Packages**: db (Prisma), auth (Lucia), hono-ws, sdk
|
||||
- **Package Manager**: pnpm 10.6.5
|
||||
|
||||
## Build, Lint, and Test Commands
|
||||
|
||||
### Root Level Commands
|
||||
|
||||
```bash
|
||||
pnpm install # Install all dependencies
|
||||
pnpm build # Build all apps and packages (uses Turbo)
|
||||
pnpm dev # Start all apps in dev mode (uses Turbo)
|
||||
pnpm lint # Lint all apps (uses Turbo)
|
||||
```
|
||||
|
||||
### Database Commands
|
||||
|
||||
```bash
|
||||
pnpm db:migrate # Run Prisma migrations in dev
|
||||
pnpm prisma # Run any Prisma command in db package
|
||||
```
|
||||
|
||||
### App-Specific Commands
|
||||
|
||||
```bash
|
||||
# Web app (Next.js)
|
||||
pnpm --filter=@hctv/web dev # Start Next.js dev server
|
||||
pnpm --filter=@hctv/web build # Build Next.js app
|
||||
pnpm --filter=@hctv/web lint # Lint web app
|
||||
pnpm --filter=@hctv/web check-types # Type check (tsc --noEmit)
|
||||
pnpm --filter=@hctv/web ui:add # Add shadcn components
|
||||
|
||||
# Chat app (Hono)
|
||||
pnpm --filter=@hctv/chat dev # Start chat server with watch
|
||||
pnpm --filter=@hctv/chat build # Build chat server
|
||||
|
||||
# DB package (Prisma)
|
||||
pnpm --filter=@hctv/db db:generate # Generate Prisma client
|
||||
pnpm --filter=@hctv/db db:migrate # Run migrations
|
||||
pnpm --filter=@hctv/db build # Generate client and build
|
||||
```
|
||||
|
||||
### Running Single Tests
|
||||
|
||||
This project does not currently have a test suite configured. When adding tests:
|
||||
|
||||
- Use Vitest or Jest for unit tests
|
||||
- Use Playwright for E2E tests (recommended for Next.js)
|
||||
- Follow the pattern: `pnpm test -- <test-file-path>`
|
||||
|
||||
### Docker Commands
|
||||
|
||||
```bash
|
||||
pnpm docker:web # Build web app Docker image
|
||||
pnpm docker:chat # Build chat app Docker image
|
||||
pnpm r:rtmp # Restart RTMP server
|
||||
```
|
||||
|
||||
## Code Style Guidelines
|
||||
|
||||
### Formatting
|
||||
|
||||
- **Formatter**: Prettier (`.prettierrc.json`)
|
||||
- **Indentation**: 2 spaces (no tabs)
|
||||
- **Line Width**: 100 characters
|
||||
- **Quotes**: Single quotes
|
||||
- **Semicolons**: Required
|
||||
- **Trailing Commas**: ES5 style
|
||||
- **Linter**: ESLint with Next.js rules (`next/core-web-vitals`)
|
||||
|
||||
### Import Organization
|
||||
|
||||
Imports should be ordered as follows (no blank lines between groups):
|
||||
|
||||
1. React/Next.js core imports
|
||||
2. Third-party libraries
|
||||
3. Internal components (`@/components`)
|
||||
4. Internal utilities/libs (`@/lib`)
|
||||
5. Package imports (`@hctv/*`)
|
||||
6. Type imports (use `import type` keyword)
|
||||
7. Relative imports
|
||||
|
||||
```typescript
|
||||
import { useState, useEffect } from 'react';
|
||||
import { Send } from 'lucide-react';
|
||||
import { Button } from '@/components/ui/button';
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { prisma } from '@hctv/db';
|
||||
import type { User, Channel } from '@hctv/db';
|
||||
import { helper } from './utils';
|
||||
```
|
||||
|
||||
### TypeScript Usage
|
||||
|
||||
- **Strict mode**: Enabled
|
||||
- **Type over interface**: Use `type` for unions/intersections, `interface` for object shapes
|
||||
- **No implicit any**: Always type your variables
|
||||
- **Type imports**: Use `import type` for type-only imports
|
||||
- **Prisma types**: Import from `@hctv/db` and use type composition
|
||||
|
||||
```typescript
|
||||
// Interfaces for props and object shapes
|
||||
interface ChatMessage {
|
||||
user?: User;
|
||||
message: string;
|
||||
type: 'message' | 'systemMsg';
|
||||
}
|
||||
|
||||
// Type aliases for complex types
|
||||
type FormFieldConfig = {
|
||||
name: string;
|
||||
label?: string;
|
||||
};
|
||||
|
||||
// Prisma type composition
|
||||
type StreamWithChannel = StreamInfo & { channel: Channel };
|
||||
```
|
||||
|
||||
### Naming Conventions
|
||||
|
||||
**Files:**
|
||||
|
||||
- React components: `PascalCase.tsx` (e.g., `ChatPanel.tsx`)
|
||||
- Utilities/helpers: `camelCase.ts` (e.g., `validate.ts`)
|
||||
- Next.js pages: `page.tsx`, `route.ts`, `layout.tsx`
|
||||
- Client components: `page.client.tsx`
|
||||
|
||||
**Variables & Functions:**
|
||||
|
||||
- Components: `PascalCase`
|
||||
- Functions: `camelCase`
|
||||
- Constants: `SCREAMING_SNAKE_CASE`
|
||||
- Booleans: Use `is`, `has`, `should` prefixes
|
||||
- Refs: `Ref` suffix (e.g., `socketRef`)
|
||||
|
||||
```typescript
|
||||
const MESSAGE_HISTORY_SIZE = 15;
|
||||
const isFollowing = await checkFollowing();
|
||||
const socketRef = useRef<WebSocket | null>(null);
|
||||
```
|
||||
|
||||
### Error Handling
|
||||
|
||||
**API Routes:**
|
||||
|
||||
- Return `Response` objects with appropriate status codes
|
||||
- Use descriptive error messages
|
||||
- Status codes: 400 (bad request), 401 (unauthorized), 403 (forbidden), 404 (not found)
|
||||
|
||||
```typescript
|
||||
if (!user) {
|
||||
return new Response('Unauthorized', { status: 401 });
|
||||
}
|
||||
```
|
||||
|
||||
**Server Actions:**
|
||||
|
||||
- Return objects with `success` boolean and `error` or `data` fields
|
||||
- Use `zodVerify` helper for validation
|
||||
|
||||
```typescript
|
||||
const zod = await zodVerify(schema, formData);
|
||||
if (!zod.success) {
|
||||
return { success: false, error: zod.error };
|
||||
}
|
||||
return { success: true, data: result };
|
||||
```
|
||||
|
||||
**Client-side:**
|
||||
|
||||
- Use try-catch for async operations
|
||||
- Use toast notifications (sonner) for user feedback
|
||||
- Log errors with `console.error`
|
||||
|
||||
### Async Patterns
|
||||
|
||||
- **Prefer**: async/await over Promise chains
|
||||
- **Parallel operations**: Use `Promise.all()`
|
||||
- **No .then() chaining**: Except in utility functions like fetchers
|
||||
|
||||
```typescript
|
||||
// Standard async/await
|
||||
const data = await prisma.model.findMany();
|
||||
|
||||
// Parallel operations
|
||||
const [channelA, channelB] = await Promise.all([
|
||||
prisma.channel.findUnique({ where: { id: 'a' } }),
|
||||
prisma.channel.findUnique({ where: { id: 'b' } }),
|
||||
]);
|
||||
```
|
||||
|
||||
## React Component Patterns
|
||||
|
||||
### Component Structure
|
||||
|
||||
1. Directive (`'use client'` or `'use server'`)
|
||||
2. Imports
|
||||
3. Component function
|
||||
4. Helper functions (if needed)
|
||||
5. Type/interface definitions (at bottom)
|
||||
|
||||
```typescript
|
||||
'use client';
|
||||
|
||||
import { useState } from 'react';
|
||||
import { Button } from '@/components/ui/button';
|
||||
|
||||
export default function ChatPanel(props: Props) {
|
||||
const [message, setMessage] = useState('');
|
||||
|
||||
return <div>{/* JSX */}</div>;
|
||||
}
|
||||
|
||||
interface Props {
|
||||
username: string;
|
||||
}
|
||||
```
|
||||
|
||||
### Server vs Client Components
|
||||
|
||||
- **Server components**: Default (no directive), use for data fetching
|
||||
- **Client components**: Add `'use client'`, use for interactivity
|
||||
- Fetch data in server components, pass to client components as props
|
||||
|
||||
## Database Patterns (Prisma)
|
||||
|
||||
### Imports
|
||||
|
||||
```typescript
|
||||
import { prisma } from '@hctv/db';
|
||||
import type { User, Channel, StreamInfo } from '@hctv/db';
|
||||
```
|
||||
|
||||
### Common Queries
|
||||
|
||||
```typescript
|
||||
// FindUnique with relations
|
||||
const channel = await prisma.channel.findUnique({
|
||||
where: { name: channelName },
|
||||
include: { owner: true, streamInfo: true },
|
||||
});
|
||||
|
||||
// FindMany with dynamic filters
|
||||
const where: Prisma.StreamInfoWhereInput = {};
|
||||
if (isLive) where.isLive = true;
|
||||
const streams = await prisma.streamInfo.findMany({ where });
|
||||
|
||||
// Create with relations
|
||||
await prisma.channel.create({
|
||||
data: {
|
||||
name: channelName,
|
||||
ownerId: user.id,
|
||||
personalFor: { connect: { id: user.id } },
|
||||
},
|
||||
});
|
||||
|
||||
// Update
|
||||
await prisma.streamInfo.update({
|
||||
where: { username },
|
||||
data: { title: newTitle },
|
||||
});
|
||||
|
||||
// Upsert
|
||||
await prisma.streamKey.upsert({
|
||||
create: { key: newKey, channelId },
|
||||
update: { key: newKey },
|
||||
where: { channelId },
|
||||
});
|
||||
```
|
||||
|
||||
### Redis Usage
|
||||
|
||||
```typescript
|
||||
import { getRedisConnection } from '@hctv/db';
|
||||
|
||||
const redis = getRedisConnection();
|
||||
await redis.set('key', 'value');
|
||||
await redis.get('key');
|
||||
await redis.setex('key', 30, 'value'); // with expiry
|
||||
```
|
||||
|
||||
## API Route Patterns
|
||||
|
||||
### Next.js App Router
|
||||
|
||||
```typescript
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { prisma } from '@hctv/db';
|
||||
import { NextRequest } from 'next/server';
|
||||
|
||||
export async function GET(request: NextRequest) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user) {
|
||||
return new Response('Unauthorized', { status: 401 });
|
||||
}
|
||||
|
||||
const searchParams = request.nextUrl.searchParams;
|
||||
const param = searchParams.get('param');
|
||||
|
||||
const data = await prisma.model.findMany();
|
||||
return Response.json(data);
|
||||
}
|
||||
```
|
||||
|
||||
### Server Actions
|
||||
|
||||
```typescript
|
||||
'use server';
|
||||
|
||||
export async function createChannel(prev: any, formData: FormData) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
const zod = await zodVerify(createChannelSchema, formData);
|
||||
if (!zod.success) {
|
||||
return zod;
|
||||
}
|
||||
|
||||
// ... processing
|
||||
|
||||
return { success: true };
|
||||
}
|
||||
```
|
||||
|
||||
## Important Notes
|
||||
|
||||
- **Turbo caching**: Build outputs are cached. Use `--force` to bypass cache
|
||||
- **Environment variables**: Use `NEXT_PUBLIC_` prefix for client-side vars
|
||||
- **Styling**: Tailwind CSS with shadcn/ui components, use `cn()` for conditional classes
|
||||
- **Data fetching**: SWR for client-side, direct Prisma for server components
|
||||
- **Validation**: Zod schemas for form and API validation
|
||||
- **Cache invalidation**: Use `revalidatePath()` after mutations
|
||||
@@ -1,7 +1,7 @@
|
||||
# hackclub.tv
|
||||
|
||||
This is the source code for [hackclub.tv (hackclub.tv)](https://hackclub.tv), a livestreaming website for hackclubbers.
|
||||
This is the source code for [hackclub.tv](https://hackclub.tv), a livestreaming website for hackclubbers.
|
||||
|
||||
Development has been ongoing for a few months, and the site is now live! There are some half-baked features, but I'm all ears for feedback.
|
||||
The development setup guide can be read at <https://docs.hackclub.tv/guides/dev/>
|
||||
|
||||
Join [#hctv](https://hackclub.slack.com/archives/C08HGLXGXAB) on the HC Slack for discussion and updates!
|
||||
Join [#hctv](https://hackclub.slack.com/archives/C08HGLXGXAB) on the Hack Club Slack for discussion and updates!
|
||||
|
||||
@@ -1,10 +1,13 @@
|
||||
FROM node:lts-alpine AS base
|
||||
ENV PNPM_HOME="/pnpm"
|
||||
ENV PATH="$PNPM_HOME:$PATH"
|
||||
RUN corepack enable
|
||||
|
||||
FROM base AS builder
|
||||
RUN apk update
|
||||
RUN apk add --no-cache libc6-compat
|
||||
WORKDIR /app
|
||||
RUN yarn global add turbo@^2
|
||||
RUN pnpm add -g turbo@^2
|
||||
COPY . .
|
||||
|
||||
RUN turbo prune @hctv/chat --docker
|
||||
@@ -16,10 +19,10 @@ WORKDIR /app
|
||||
|
||||
# First install the dependencies
|
||||
COPY --from=builder /app/out/json/ .
|
||||
RUN yarn install --frozen-lockfile
|
||||
RUN pnpm install --frozen-lockfile
|
||||
|
||||
COPY --from=builder /app/out/full/ .
|
||||
RUN --mount=type=secret,id=TURBO_TOKEN --mount=type=secret,id=TURBO_TEAM TURBO_TOKEN=$(cat /run/secrets/TURBO_TOKEN) TURBO_TEAM=$(cat /run/secrets/TURBO_TEAM) yarn turbo run build --concurrency=1
|
||||
RUN --mount=type=secret,id=TURBO_TOKEN --mount=type=secret,id=TURBO_TEAM TURBO_TOKEN=$(cat /run/secrets/TURBO_TOKEN) TURBO_TEAM=$(cat /run/secrets/TURBO_TEAM) pnpm turbo run build --concurrency=1
|
||||
|
||||
FROM base AS runner
|
||||
WORKDIR /app
|
||||
|
||||
@@ -7,9 +7,9 @@
|
||||
"build": "tsc --build"
|
||||
},
|
||||
"dependencies": {
|
||||
"@hctv/auth": "*",
|
||||
"@hctv/db": "*",
|
||||
"@hctv/hono-ws": "*",
|
||||
"@hctv/auth": "workspace:*",
|
||||
"@hctv/db": "workspace:*",
|
||||
"@hctv/hono-ws": "workspace:*",
|
||||
"@hono/node-server": "^1.14.0",
|
||||
"@hono/node-ws": "^1.1.0",
|
||||
"@leeoniya/ufuzzy": "^1.0.18",
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -2,24 +2,44 @@
|
||||
import { defineConfig } from 'astro/config';
|
||||
import starlight from '@astrojs/starlight';
|
||||
import mermaid from 'astro-mermaid';
|
||||
import catppuccin from "@catppuccin/starlight";
|
||||
import catppuccin from '@catppuccin/starlight';
|
||||
import starlightTypeDoc, { typeDocSidebarGroup } from 'starlight-typedoc';
|
||||
|
||||
// https://astro.build/config
|
||||
export default defineConfig({
|
||||
integrations: [
|
||||
integrations: [
|
||||
mermaid({
|
||||
theme: 'base',
|
||||
autoTheme: true
|
||||
autoTheme: true,
|
||||
}),
|
||||
starlight({
|
||||
title: 'hctv docs',
|
||||
social: [{ icon: 'github', label: 'GitHub', href: 'https://github.com/SrIzan10/hctv' }],
|
||||
starlight({
|
||||
title: 'hctv docs',
|
||||
social: [{ icon: 'github', label: 'GitHub', href: 'https://github.com/SrIzan10/hctv' }],
|
||||
plugins: [
|
||||
catppuccin({
|
||||
dark: { flavor: "mocha", accent: "blue" },
|
||||
light: { flavor: "latte", accent: "blue" }
|
||||
dark: { flavor: 'mocha', accent: 'blue' },
|
||||
light: { flavor: 'latte', accent: 'blue' },
|
||||
}),
|
||||
]
|
||||
}),
|
||||
],
|
||||
starlightTypeDoc({
|
||||
entryPoints: ['../../packages/sdk/src/index.ts'],
|
||||
tsconfig: '../../packages/sdk/tsconfig.json',
|
||||
output: 'typedoc-sdk',
|
||||
sidebar: {
|
||||
label: 'SDK Reference',
|
||||
},
|
||||
}),
|
||||
],
|
||||
sidebar: [
|
||||
{
|
||||
label: 'API',
|
||||
autogenerate: { directory: 'api' },
|
||||
},
|
||||
{
|
||||
label: 'Guides',
|
||||
autogenerate: { directory: 'guides' },
|
||||
},
|
||||
typeDocSidebarGroup,
|
||||
],
|
||||
}),
|
||||
],
|
||||
});
|
||||
|
||||
@@ -15,6 +15,9 @@
|
||||
"astro": "^5.6.1",
|
||||
"astro-mermaid": "^1.0.4",
|
||||
"mermaid": "^11.10.1",
|
||||
"sharp": "^0.34.2"
|
||||
"sharp": "^0.34.2",
|
||||
"starlight-typedoc": "^0.21.5",
|
||||
"typedoc": "^0.28.16",
|
||||
"typedoc-plugin-markdown": "^4.9.0"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -9,26 +9,34 @@ The chat system is powered by a websocket server. Please read the entire page be
|
||||
|
||||
## Connection and messages
|
||||
|
||||
The websocket server is located at `wss://hackclub.tv/api/chat/ws/:username`, where `:username` is the channel you want to connect to.
|
||||
The websocket server is located at `wss://hackclub.tv/api/stream/chat/ws/:username`, where `:username` is the channel you want to connect to.
|
||||
|
||||
You'll need to provide authentication, which can be done by providing an `auth_session` cookie, just like the REST API.
|
||||
|
||||
You'll need to provide authentication, which can be done by providing an `auth_session` cookie, just like the REST API.
|
||||
<Aside type="tip">
|
||||
Bot accounts are now supported. You can choose to connect as a bot by providing a bot account's API key on the Authentication header: `Bearer hctvb_xxxxxxx`
|
||||
Bot accounts are now supported. You can choose to connect as a bot by providing a bot account's API key in one of two ways:
|
||||
- Using the `Authorization` header: `Bearer hctvb_xxxxxxx`
|
||||
- Using the `?botAuth=hctvb_xxxxxxx` query parameter
|
||||
|
||||
**Security Note:** When using the `?botAuth=` query parameter, be aware that query parameters may be logged in server logs, and/or proxy logs. Use the `Authorization` header method whenever possible. The query parameter method should only be used when connecting from an environment where headers cannot be set.
|
||||
|
||||
It is highly advised to use a bot account for any automated task, and to implement anything pointed out in this page.
|
||||
|
||||
</Aside>
|
||||
|
||||
Once connected, you must implement a subroutine in your code to send ping messages every 5 seconds. This is because of Cloudflare limitations.
|
||||
Once connected, you must implement a subroutine in your code to send ping messages every about 5 seconds. This is because of Cloudflare limitations.
|
||||
|
||||
Messages are sent and received in JSON format. The following message types are supported:
|
||||
- `message`: a chat message.
|
||||
- sent by client:
|
||||
|
||||
- `message`: a chat message.
|
||||
- sent by client:
|
||||
```json
|
||||
{
|
||||
"type": "message",
|
||||
"content": "Hello, world!"
|
||||
"message": "Hello, world!"
|
||||
}
|
||||
```
|
||||
- received by client:
|
||||
- received by client:
|
||||
```json
|
||||
{
|
||||
"user": {
|
||||
@@ -36,24 +44,24 @@ Messages are sent and received in JSON format. The following message types are s
|
||||
"username": "user_who_sent_message",
|
||||
"avatar": "https://emoji.slack-edge.com/avatar.png"
|
||||
},
|
||||
"message": "Hello, world!",
|
||||
"message": "Hello, world!"
|
||||
}
|
||||
```
|
||||
- `ping`: a ping message to keep the connection alive.
|
||||
- sent by client:
|
||||
- `ping`: a ping message to keep the connection alive.
|
||||
- sent by client:
|
||||
```json
|
||||
{
|
||||
"type": "ping"
|
||||
}
|
||||
```
|
||||
- received by client:
|
||||
- received by client:
|
||||
```json
|
||||
{
|
||||
"type": "ping"
|
||||
"type": "pong"
|
||||
}
|
||||
```
|
||||
- `history`: a message containing the chat history. This is sent upon connection.
|
||||
- received by client:
|
||||
- `history`: a message containing the chat history. This is sent upon connection.
|
||||
- received by client:
|
||||
```json
|
||||
{
|
||||
"type": "history",
|
||||
@@ -71,9 +79,114 @@ Messages are sent and received in JSON format. The following message types are s
|
||||
]
|
||||
}
|
||||
```
|
||||
|
||||
## Moderation messages
|
||||
|
||||
Chat moderation is available over the same websocket connection when the connected account is a channel moderator.
|
||||
|
||||
- `mod:deleteMessage`: delete a message from chat history.
|
||||
- sent by client:
|
||||
```json
|
||||
{
|
||||
"type": "mod:deleteMessage",
|
||||
"msgId": "chat_message_id"
|
||||
}
|
||||
```
|
||||
- received by clients in the same channel:
|
||||
```json
|
||||
{
|
||||
"type": "messageDeleted",
|
||||
"msgId": "chat_message_id"
|
||||
}
|
||||
```
|
||||
- `mod:timeoutUser`: temporarily remove a user's ability to send messages.
|
||||
- sent by client:
|
||||
```json
|
||||
{
|
||||
"type": "mod:timeoutUser",
|
||||
"targetUserId": "user_id",
|
||||
"targetUsername": "username",
|
||||
"durationSeconds": 300,
|
||||
"reason": "optional reason"
|
||||
}
|
||||
```
|
||||
- `mod:banUser`: ban a user from chat indefinitely.
|
||||
- sent by client:
|
||||
```json
|
||||
{
|
||||
"type": "mod:banUser",
|
||||
"targetUserId": "user_id",
|
||||
"targetUsername": "username",
|
||||
"reason": "optional reason"
|
||||
}
|
||||
```
|
||||
- `mod:liftTimeout` and `mod:unbanUser`: remove an existing timeout/ban.
|
||||
- sent by client:
|
||||
```json
|
||||
{
|
||||
"type": "mod:unbanUser",
|
||||
"targetUserId": "user_id",
|
||||
"targetUsername": "username"
|
||||
}
|
||||
```
|
||||
- `chatAccess`: tells a user whether they can currently send messages.
|
||||
- received by client:
|
||||
```json
|
||||
{
|
||||
"type": "chatAccess",
|
||||
"canSend": false,
|
||||
"restriction": {
|
||||
"type": "timeout",
|
||||
"reason": "spamming",
|
||||
"expiresAt": "2026-02-21T10:00:00.000Z"
|
||||
}
|
||||
}
|
||||
```
|
||||
- `moderationError`: returned when a moderation action is rejected.
|
||||
- received by client:
|
||||
```json
|
||||
{
|
||||
"type": "moderationError",
|
||||
"code": "INVALID_TARGET",
|
||||
"message": "Invalid moderation target."
|
||||
}
|
||||
```
|
||||
|
||||
## SDK moderation usage (`@hctv/sdk`)
|
||||
|
||||
If you're building bots, you can call moderation helpers directly:
|
||||
|
||||
```ts
|
||||
import { HctvSdk } from '@hctv/sdk';
|
||||
|
||||
const sdk = new HctvSdk({ botToken: process.env.HCTV_BOT_TOKEN! });
|
||||
await sdk.chat.connect('channel-name');
|
||||
|
||||
sdk.chat.timeoutUser('channel-name', 'target-user-id', 'target-username', 300, 'spam');
|
||||
sdk.chat.banUser('channel-name', 'target-user-id', 'target-username', 'severe abuse');
|
||||
sdk.chat.liftTimeout('channel-name', 'target-user-id', 'target-username');
|
||||
sdk.chat.unbanUser('channel-name', 'target-user-id', 'target-username');
|
||||
sdk.chat.deleteMessage('channel-name', 'message-id');
|
||||
|
||||
sdk.chat.onModerationError((error, channel) => {
|
||||
console.error(`[${channel}] moderation error`, error.code, error.message);
|
||||
});
|
||||
|
||||
sdk.chat.onChatAccess((access, channel) => {
|
||||
console.log(`[${channel}] canSend=${access.canSend}`);
|
||||
});
|
||||
|
||||
sdk.chat.onModerationEvent((event) => {
|
||||
if (event.type === 'messageDeleted') {
|
||||
console.log('deleted message', event.msgId);
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
## Emoji handling
|
||||
|
||||
*diagram source: devin deepwiki*
|
||||
_diagram source: devin deepwiki_
|
||||
|
||||
```mermaid
|
||||
graph TB
|
||||
subgraph "Emoji Processing Pipeline"
|
||||
@@ -111,6 +224,7 @@ The server then checks Redis for the emoji URL and returns it.
|
||||
When a user wants to look up an emoji (by typing `:(partial name)`), the server uses uFuzzy to find matching emojis in the Redis `emojis` hash key and returns the results.
|
||||
|
||||
Here's what gets sent on the websocket:
|
||||
|
||||
- `emojiMsg`: Looks up emojis
|
||||
- sent by client:
|
||||
```json
|
||||
|
||||
32
apps/docs/src/content/docs/guides/dev.mdx
Normal file
32
apps/docs/src/content/docs/guides/dev.mdx
Normal file
@@ -0,0 +1,32 @@
|
||||
---
|
||||
title: Development Setup
|
||||
description: Instructions to set up a local development environment for hackclub.tv
|
||||
---
|
||||
|
||||
1. clone repo
|
||||
2. `pnpm install`
|
||||
3. `cp apps/web/.env.example apps/web/.env && cp packages/db/.env.example packages/db/.env`
|
||||
4. `pnpm dev`
|
||||
5. `pnpm db:migrate` (RUN THIS AFTER POPULATING ENV)
|
||||
|
||||
- slack notifier app manifest is as follows:
|
||||
```
|
||||
display_information:
|
||||
# please change the name to something that can be linked to you if possible
|
||||
name: hctv notifier dev
|
||||
features:
|
||||
bot_user:
|
||||
# same with this :pray:
|
||||
display_name: hctv notifier dev
|
||||
always_online: false
|
||||
oauth_config:
|
||||
scopes:
|
||||
bot:
|
||||
- chat:write
|
||||
- users:read
|
||||
- channels:join
|
||||
settings:
|
||||
org_deploy_enabled: false
|
||||
socket_mode_enabled: false
|
||||
token_rotation_enabled: false
|
||||
```
|
||||
31
apps/web/.env.example
Normal file
31
apps/web/.env.example
Normal file
@@ -0,0 +1,31 @@
|
||||
DATABASE_URL=postgresql://postgres:skbiditoilet@localhost:5555/postgres
|
||||
|
||||
# make a slack app here: https://api.slack.com/apps
|
||||
SLACK_NOTIFIER_TOKEN=<make a bot for this, check app manifest below>
|
||||
|
||||
# invite your bot to the channel you created. below is #hctv-dev, so use that if you want!
|
||||
NOTIFICATION_CHANNEL_ID=C08M3MGE6PJ
|
||||
|
||||
REDIS_URL=redis://localhost:6379
|
||||
|
||||
# get from https://uploadthing.com/
|
||||
UPLOADTHING_TOKEN=<get from uploadthing>
|
||||
|
||||
# enable oauth mode on your hca account and make an app: https://auth.hackclub.com/identity/edit
|
||||
HCID_CLIENT=<auth.hackclub.com client>
|
||||
HCID_SECRET=<auth.hackclub.com secret>
|
||||
# make sure to put this as one of the redirect uri
|
||||
HCID_REDIRECT_URI=http://localhost:3000/auth/hackclub/callback
|
||||
|
||||
# mediamtx settings
|
||||
NEXT_PUBLIC_MEDIAMTX_URL_HQ=http://localhost:8891
|
||||
MEDIAMTX_API_HQ=http://localhost:9997
|
||||
NEXT_PUBLIC_MEDIAMTX_INGEST_ROUTE_HQ=localhost:8890
|
||||
|
||||
# commented because we don't have another ingest server as of right now
|
||||
# NEXT_PUBLIC_MEDIAMTX_URL_ASIA=http://localhost:8991
|
||||
# MEDIAMTX_API_ASIA=http://localhost:9999
|
||||
# NEXT_PUBLIC_MEDIAMTX_INGEST_ROUTE_ASIA=localhost:8990
|
||||
|
||||
# idt you should change this
|
||||
MEDIAMTX_PUBLISH_KEY=rjq1xdpCPA4qyt3jge
|
||||
@@ -1,4 +1,7 @@
|
||||
FROM node:lts-slim AS base
|
||||
FROM node:22-slim AS base
|
||||
ENV PNPM_HOME="/pnpm"
|
||||
ENV PATH="$PNPM_HOME:$PATH"
|
||||
RUN corepack enable
|
||||
|
||||
FROM base AS builder
|
||||
RUN apt-get update && apt-get install -y --no-install-recommends \
|
||||
@@ -7,9 +10,7 @@ RUN apt-get update && apt-get install -y --no-install-recommends \
|
||||
&& rm -rf /var/lib/apt/lists/*
|
||||
# Set working directory
|
||||
WORKDIR /app
|
||||
# Replace <your-major-version> with the major version installed in your repository. For example:
|
||||
# RUN yarn global add turbo@^2
|
||||
RUN yarn global add turbo@^2
|
||||
RUN pnpm add -g turbo@^2
|
||||
COPY . .
|
||||
|
||||
# Get the git commit hash before pruning (since .git might be removed)
|
||||
@@ -23,6 +24,7 @@ RUN turbo prune @hctv/web --docker
|
||||
FROM base AS installer
|
||||
RUN apt-get update && apt-get install -y --no-install-recommends \
|
||||
git \
|
||||
curl \
|
||||
libvips-dev \
|
||||
python3 \
|
||||
make \
|
||||
@@ -35,13 +37,34 @@ WORKDIR /app
|
||||
|
||||
# First install the dependencies (as they change less often)
|
||||
COPY --from=builder /app/out/json/ .
|
||||
RUN yarn install --frozen-lockfile
|
||||
RUN pnpm install --frozen-lockfile
|
||||
|
||||
# Install a standalone Prisma CLI for runtime migrations (no pnpm symlinks).
|
||||
RUN mkdir -p /opt/prisma-cli && cd /opt/prisma-cli && npm init -y && npm install prisma@6.5.0
|
||||
|
||||
COPY --from=builder /app/out/full/ .
|
||||
RUN --mount=type=secret,id=TURBO_TOKEN --mount=type=secret,id=TURBO_TEAM \
|
||||
|
||||
# Generate latest emojis.json during image build.
|
||||
RUN ARCH=$(dpkg --print-architecture) && \
|
||||
if [ "$ARCH" = "amd64" ]; then EMOJI_ARCH="x86_64"; \
|
||||
elif [ "$ARCH" = "arm64" ]; then EMOJI_ARCH="aarch64"; \
|
||||
else EMOJI_ARCH=""; fi && \
|
||||
RELEASE_JSON=$(curl -fsSL https://api.github.com/repos/srizan10/hctv/releases/latest || true) && \
|
||||
RELEASE_URL=$(printf '%s' "$RELEASE_JSON" | grep "browser_download_url.*slack-import-emojis-linux-${EMOJI_ARCH}" | cut -d '"' -f 4 || true) && \
|
||||
if [ -n "$RELEASE_URL" ] && \
|
||||
curl -fsSL -o /tmp/slack-import-emojis-bin "$RELEASE_URL" && \
|
||||
chmod +x /tmp/slack-import-emojis-bin && \
|
||||
/tmp/slack-import-emojis-bin default; then \
|
||||
cp /app/emojis.json /app/apps/web/emojis.json; \
|
||||
else \
|
||||
cp /app/apps/web/src/lib/instrumentation/emojis.json /app/apps/web/emojis.json; \
|
||||
fi
|
||||
|
||||
RUN --mount=type=secret,id=TURBO_TOKEN --mount=type=secret,id=TURBO_TEAM --mount=type=secret,id=SENTRY_AUTH_TOKEN \
|
||||
COMMIT=$(cat /tmp/commit_hash 2>/dev/null || echo "unknown") && \
|
||||
TURBO_TOKEN=$(cat /run/secrets/TURBO_TOKEN) TURBO_TEAM=$(cat /run/secrets/TURBO_TEAM) \
|
||||
commit=$COMMIT yarn turbo run build --env-mode=loose
|
||||
SENTRY_AUTH_TOKEN=$(cat /run/secrets/SENTRY_AUTH_TOKEN) \
|
||||
commit=$COMMIT pnpm turbo run build --env-mode=loose
|
||||
|
||||
FROM base AS runner
|
||||
WORKDIR /app
|
||||
@@ -52,20 +75,20 @@ RUN apt-get update && apt-get install -y --no-install-recommends \
|
||||
&& rm -rf /var/lib/apt/lists/*
|
||||
|
||||
RUN groupadd --system --gid 1001 nodejs
|
||||
RUN useradd --system --uid 1001 nextjs
|
||||
RUN useradd --system --uid 1001 nextjs --create-home
|
||||
|
||||
# Copy Prisma files for migrations
|
||||
COPY --from=installer --chown=nextjs:nodejs /app/packages/db/prisma ./packages/db/prisma
|
||||
COPY --from=installer --chown=nextjs:nodejs /app/packages/db/generated ./packages/db/generated
|
||||
COPY --from=installer --chown=nextjs:nodejs /app/packages/db/package.json ./packages/db/package.json
|
||||
COPY --from=installer --chown=nextjs:nodejs /app/node_modules ./node_modules
|
||||
# Ensure home directory and cache directories have proper permissions
|
||||
RUN mkdir -p /home/nextjs/.cache && \
|
||||
chown -R nextjs:nodejs /home/nextjs
|
||||
|
||||
COPY --from=installer /tmp/commit_hash /tmp/commit_hash
|
||||
RUN COMMIT_VALUE=$(cat /tmp/commit_hash 2>/dev/null || echo "unknown") && \
|
||||
echo "#!/bin/sh" > /usr/local/bin/start.sh && \
|
||||
echo "set -e" >> /usr/local/bin/start.sh && \
|
||||
echo "export COREPACK_ENABLE_DOWNLOAD_PROMPT=0" >> /usr/local/bin/start.sh && \
|
||||
echo "export HOME=/home/nextjs" >> /usr/local/bin/start.sh && \
|
||||
echo "echo 'Running database migrations...'" >> /usr/local/bin/start.sh && \
|
||||
echo "npx prisma migrate deploy --schema=/app/packages/db/prisma/schema.prisma" >> /usr/local/bin/start.sh && \
|
||||
echo "node /opt/prisma-cli/node_modules/prisma/build/index.js migrate deploy --schema /app/packages/db/prisma/schema.prisma" >> /usr/local/bin/start.sh && \
|
||||
echo "cd /app" >> /usr/local/bin/start.sh && \
|
||||
echo "export commit=$COMMIT_VALUE" >> /usr/local/bin/start.sh && \
|
||||
echo "echo 'Starting Next.js application...'" >> /usr/local/bin/start.sh && \
|
||||
@@ -79,6 +102,10 @@ USER nextjs
|
||||
COPY --from=installer --chown=nextjs:nodejs /app/apps/web/.next/standalone ./
|
||||
COPY --from=installer --chown=nextjs:nodejs /app/apps/web/.next/static ./apps/web/.next/static
|
||||
COPY --from=installer --chown=nextjs:nodejs /app/apps/web/public ./apps/web/public
|
||||
COPY --chown=nextjs:nodejs apps/web/emojis.json .
|
||||
|
||||
CMD ["/usr/local/bin/start.sh"]
|
||||
COPY --from=installer --chown=nextjs:nodejs /app/apps/web/emojis.json ./emojis.json
|
||||
|
||||
# Copy Prisma schema and migrations for prisma migrate deploy
|
||||
COPY --from=installer --chown=nextjs:nodejs /app/packages/db/prisma ./packages/db/prisma
|
||||
COPY --from=installer --chown=nextjs:nodejs /opt/prisma-cli /opt/prisma-cli
|
||||
|
||||
CMD ["/usr/local/bin/start.sh"]
|
||||
|
||||
@@ -69,6 +69,9 @@ export default withSentryConfig(nextConfig, {
|
||||
|
||||
project: "hctv",
|
||||
|
||||
// Auth token for uploading source maps
|
||||
authToken: process.env.SENTRY_AUTH_TOKEN,
|
||||
|
||||
// Only print logs for uploading source maps in CI
|
||||
silent: !process.env.CI,
|
||||
|
||||
@@ -78,6 +81,9 @@ export default withSentryConfig(nextConfig, {
|
||||
// Upload a larger set of source maps for prettier stack traces (increases build time)
|
||||
widenClientFileUpload: true,
|
||||
|
||||
// Hides source maps from generated client bundles
|
||||
hideSourceMaps: true,
|
||||
|
||||
// Route browser requests to Sentry through a Next.js rewrite to circumvent ad-blockers.
|
||||
// This can increase your server load as well as your hosting bill.
|
||||
// Note: Check that the configured route will not match with your Next.js middleware, otherwise reporting of client-
|
||||
|
||||
@@ -5,7 +5,7 @@
|
||||
"type": "module",
|
||||
"scripts": {
|
||||
"dd": "docker compose --file ../../dev/docker-compose.yml up -d",
|
||||
"dev": "next dev --turbo",
|
||||
"dev": "next dev --turbo -H 0.0.0.0",
|
||||
"donly": "docker compose --file ../../dev/docker-compose.yml up",
|
||||
"build": "next build",
|
||||
"start": "next start",
|
||||
@@ -15,8 +15,8 @@
|
||||
"genMtxTypes": "bunx openapi-typescript https://github.com/bluenviron/mediamtx/raw/refs/tags/v1.15.5/api/openapi.yaml -o ./src/lib/types/mediamtx.d.ts"
|
||||
},
|
||||
"dependencies": {
|
||||
"@hctv/auth": "*",
|
||||
"@hctv/db": "*",
|
||||
"@hctv/auth": "workspace:*",
|
||||
"@hctv/db": "workspace:*",
|
||||
"@hookform/resolvers": "^3.9.1",
|
||||
"@lucia-auth/adapter-prisma": "^4.0.1",
|
||||
"@node-rs/argon2": "^2.0.2",
|
||||
@@ -47,7 +47,10 @@
|
||||
"class-variance-authority": "^0.7.1",
|
||||
"clsx": "^2.1.0",
|
||||
"cmdk": "1.0.0",
|
||||
"date-fns": "^4.1.0",
|
||||
"embla-carousel-react": "^8.6.0",
|
||||
"hls-video-element": "^1.5.0",
|
||||
"hls.js": "^1.6.15",
|
||||
"lucia": "^3.2.2",
|
||||
"lucide-react": "^0.473.0",
|
||||
"media-chrome": "^4.8.0",
|
||||
@@ -58,6 +61,7 @@
|
||||
"pg": "^8.14.1",
|
||||
"pg-boss": "^10.1.6",
|
||||
"react": "^19.2.3",
|
||||
"react-day-picker": "^9.13.0",
|
||||
"react-dom": "^19.2.3",
|
||||
"react-hook-form": "^7.54.2",
|
||||
"rehype-raw": "^7.0.0",
|
||||
|
||||
@@ -5,11 +5,34 @@ export default async function Page({ params }: { params: Promise<{ username: str
|
||||
const { username } = await params;
|
||||
const streamInfo = await prisma.streamInfo.findUnique({
|
||||
where: { username },
|
||||
include: { channel: true },
|
||||
include: {
|
||||
channel: {
|
||||
include: {
|
||||
restriction: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
if (!streamInfo) {
|
||||
return <div>Stream not found</div>;
|
||||
}
|
||||
|
||||
if (streamInfo.channel.restriction) {
|
||||
const isExpired = streamInfo.channel.restriction.expiresAt &&
|
||||
new Date(streamInfo.channel.restriction.expiresAt) < new Date();
|
||||
|
||||
if (!isExpired) {
|
||||
return (
|
||||
<div className="flex flex-col items-center justify-center h-[calc(100vh-64px)] p-4">
|
||||
<h1 className="text-2xl font-bold text-destructive mb-2">Channel Restricted</h1>
|
||||
<p className="text-muted-foreground text-center max-w-md">
|
||||
This channel has been restricted by a moderator and is not currently available for viewing.
|
||||
</p>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
return (
|
||||
<LiveStream username={username} streamInfo={streamInfo} />
|
||||
);
|
||||
|
||||
1355
apps/web/src/app/(ui)/(protected)/admin/page.client.tsx
Normal file
1355
apps/web/src/app/(ui)/(protected)/admin/page.client.tsx
Normal file
File diff suppressed because it is too large
Load Diff
17
apps/web/src/app/(ui)/(protected)/admin/page.tsx
Normal file
17
apps/web/src/app/(ui)/(protected)/admin/page.tsx
Normal file
@@ -0,0 +1,17 @@
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { redirect } from 'next/navigation';
|
||||
import AdminPanelClient from './page.client';
|
||||
|
||||
export default async function AdminPage() {
|
||||
const { user } = await validateRequest();
|
||||
|
||||
if (!user) {
|
||||
redirect('/auth/hackclub');
|
||||
}
|
||||
|
||||
if (!user.isAdmin) {
|
||||
redirect('/');
|
||||
}
|
||||
|
||||
return <AdminPanelClient currentUser={user} />;
|
||||
}
|
||||
@@ -0,0 +1,559 @@
|
||||
'use client';
|
||||
|
||||
import { useState } from 'react';
|
||||
import { useRouter } from 'next/navigation';
|
||||
import { format, formatDistanceToNow } from 'date-fns';
|
||||
import { toast } from 'sonner';
|
||||
import {
|
||||
ArrowLeft,
|
||||
Gavel,
|
||||
Flag,
|
||||
User,
|
||||
MessageSquare,
|
||||
Clock,
|
||||
ShieldAlert,
|
||||
ShieldOff,
|
||||
ShieldCheck,
|
||||
Trash2,
|
||||
CheckCircle2,
|
||||
XCircle,
|
||||
AlertTriangle,
|
||||
Timer,
|
||||
Ban,
|
||||
Unlock,
|
||||
Info,
|
||||
} from 'lucide-react';
|
||||
import { Badge } from '@/components/ui/badge';
|
||||
import { Button } from '@/components/ui/button';
|
||||
import { Label } from '@/components/ui/label';
|
||||
import { Textarea } from '@/components/ui/textarea';
|
||||
import { Separator } from '@/components/ui/separator';
|
||||
import { cn } from '@/lib/utils';
|
||||
|
||||
type ReportModerationAction =
|
||||
| 'review'
|
||||
| 'dismiss'
|
||||
| 'delete_reported_message'
|
||||
| 'timeout_10m'
|
||||
| 'timeout_1h'
|
||||
| 'ban_chat'
|
||||
| 'lift_chat_ban'
|
||||
| 'ban_platform'
|
||||
| 'unban_platform';
|
||||
|
||||
type ActionSeverity = 'info' | 'moderate' | 'severe';
|
||||
|
||||
interface ActionOption {
|
||||
value: ReportModerationAction;
|
||||
label: string;
|
||||
description: string;
|
||||
severity: ActionSeverity;
|
||||
requiresNote?: boolean;
|
||||
icon: React.ReactNode;
|
||||
}
|
||||
|
||||
const ACTION_OPTIONS: ActionOption[] = [
|
||||
{
|
||||
value: 'review',
|
||||
label: 'Mark as reviewed',
|
||||
description: 'Acknowledge the report without further action.',
|
||||
severity: 'info',
|
||||
icon: <CheckCircle2 className="h-4 w-4" />,
|
||||
},
|
||||
{
|
||||
value: 'dismiss',
|
||||
label: 'Dismiss',
|
||||
description: 'Close this report as unfounded or resolved.',
|
||||
severity: 'info',
|
||||
icon: <XCircle className="h-4 w-4" />,
|
||||
},
|
||||
{
|
||||
value: 'delete_reported_message',
|
||||
label: 'Delete message',
|
||||
description: 'Remove the reported message from the chat.',
|
||||
severity: 'moderate',
|
||||
requiresNote: true,
|
||||
icon: <Trash2 className="h-4 w-4" />,
|
||||
},
|
||||
{
|
||||
value: 'timeout_10m',
|
||||
label: 'Timeout 10 minutes',
|
||||
description: 'Prevent user from chatting for 10 minutes.',
|
||||
severity: 'moderate',
|
||||
requiresNote: true,
|
||||
icon: <Timer className="h-4 w-4" />,
|
||||
},
|
||||
{
|
||||
value: 'timeout_1h',
|
||||
label: 'Timeout 1 hour',
|
||||
description: 'Prevent user from chatting for 1 hour.',
|
||||
severity: 'moderate',
|
||||
requiresNote: true,
|
||||
icon: <Timer className="h-4 w-4" />,
|
||||
},
|
||||
{
|
||||
value: 'ban_chat',
|
||||
label: 'Ban from chat',
|
||||
description: 'Permanently ban user from chat.',
|
||||
severity: 'severe',
|
||||
requiresNote: true,
|
||||
icon: <Ban className="h-4 w-4" />,
|
||||
},
|
||||
{
|
||||
value: 'lift_chat_ban',
|
||||
label: 'Lift chat ban',
|
||||
description: 'Restore chat access for this user.',
|
||||
severity: 'info',
|
||||
requiresNote: true,
|
||||
icon: <Unlock className="h-4 w-4" />,
|
||||
},
|
||||
{
|
||||
value: 'ban_platform',
|
||||
label: 'Ban from platform',
|
||||
description: 'Permanently ban user from the entire platform.',
|
||||
severity: 'severe',
|
||||
requiresNote: true,
|
||||
icon: <ShieldOff className="h-4 w-4" />,
|
||||
},
|
||||
{
|
||||
value: 'unban_platform',
|
||||
label: 'Unban from platform',
|
||||
description: 'Restore platform access for this user.',
|
||||
severity: 'info',
|
||||
requiresNote: true,
|
||||
icon: <ShieldCheck className="h-4 w-4" />,
|
||||
},
|
||||
];
|
||||
|
||||
const SEVERITY_STYLES: Record<
|
||||
ActionSeverity,
|
||||
{ card: string; selected: string; icon: string; ring: string }
|
||||
> = {
|
||||
info: {
|
||||
card: 'border-border hover:border-muted-foreground/40 hover:bg-muted/30',
|
||||
selected: 'border-primary bg-primary/5',
|
||||
icon: 'text-muted-foreground',
|
||||
ring: 'ring-primary/30',
|
||||
},
|
||||
moderate: {
|
||||
card: 'border-border hover:border-amber-500/40 hover:bg-amber-500/5',
|
||||
selected: 'border-amber-500 bg-amber-500/5',
|
||||
icon: 'text-amber-500',
|
||||
ring: 'ring-amber-500/30',
|
||||
},
|
||||
severe: {
|
||||
card: 'border-border hover:border-destructive/40 hover:bg-destructive/5',
|
||||
selected: 'border-destructive bg-destructive/5',
|
||||
icon: 'text-destructive',
|
||||
ring: 'ring-destructive/30',
|
||||
},
|
||||
};
|
||||
|
||||
const STATUS_CONFIG = {
|
||||
OPEN: { label: 'Open', variant: 'destructive' as const, icon: <Flag className="h-3 w-3" /> },
|
||||
REVIEWED: {
|
||||
label: 'Reviewed',
|
||||
variant: 'secondary' as const,
|
||||
icon: <CheckCircle2 className="h-3 w-3" />,
|
||||
},
|
||||
DISMISSED: {
|
||||
label: 'Dismissed',
|
||||
variant: 'outline' as const,
|
||||
icon: <XCircle className="h-3 w-3" />,
|
||||
},
|
||||
};
|
||||
|
||||
const ACTION_LABELS: Record<NonNullable<ChatReportCase['lastAction']>, string> = {
|
||||
REVIEW: 'Marked as reviewed',
|
||||
DISMISS: 'Dismissed',
|
||||
DELETE_REPORTED_MESSAGE: 'Message deleted',
|
||||
TIMEOUT_10M: 'User timed out (10m)',
|
||||
TIMEOUT_1H: 'User timed out (1h)',
|
||||
BAN_CHAT: 'Chat banned',
|
||||
LIFT_CHAT_BAN: 'Chat ban lifted',
|
||||
BAN_PLATFORM: 'Platform banned',
|
||||
UNBAN_PLATFORM: 'Platform ban lifted',
|
||||
};
|
||||
|
||||
function InfoRow({ label, children }: { label: string; children: React.ReactNode }) {
|
||||
return (
|
||||
<div className="flex flex-col gap-0.5">
|
||||
<span className="text-[11px] font-medium uppercase tracking-wider text-muted-foreground">
|
||||
{label}
|
||||
</span>
|
||||
<span className="text-sm">{children}</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
function SectionLabel({ icon, children }: { icon: React.ReactNode; children: React.ReactNode }) {
|
||||
return (
|
||||
<div className="flex items-center gap-2 mb-3">
|
||||
<span className="text-muted-foreground">{icon}</span>
|
||||
<span className="text-xs font-semibold uppercase tracking-widest text-muted-foreground">
|
||||
{children}
|
||||
</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default function ReportCasePageClient({ report }: ReportCasePageClientProps) {
|
||||
const router = useRouter();
|
||||
const [selectedAction, setSelectedAction] = useState<ReportModerationAction>('review');
|
||||
const [note, setNote] = useState('');
|
||||
const [isSubmitting, setIsSubmitting] = useState(false);
|
||||
|
||||
const selectedMeta = ACTION_OPTIONS.find((o) => o.value === selectedAction)!;
|
||||
const requiresNote = Boolean(selectedMeta?.requiresNote);
|
||||
const isSevere = selectedMeta?.severity === 'severe';
|
||||
|
||||
const statusConfig = STATUS_CONFIG[report.status];
|
||||
|
||||
const submitAction = async () => {
|
||||
if (requiresNote && note.trim().length < 10) {
|
||||
toast.error('Please include at least 10 characters for enforcement context.');
|
||||
return;
|
||||
}
|
||||
|
||||
setIsSubmitting(true);
|
||||
try {
|
||||
const res = await fetch('/api/admin/reports', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({
|
||||
reportId: report.id,
|
||||
action: selectedAction,
|
||||
note: note.trim() || undefined,
|
||||
}),
|
||||
});
|
||||
|
||||
if (!res.ok) {
|
||||
const errorText = await res.text();
|
||||
toast.error(errorText || 'Failed to apply action');
|
||||
return;
|
||||
}
|
||||
|
||||
toast.success('Report action applied');
|
||||
setNote('');
|
||||
router.refresh();
|
||||
} catch {
|
||||
toast.error('Failed to apply action');
|
||||
} finally {
|
||||
setIsSubmitting(false);
|
||||
}
|
||||
};
|
||||
|
||||
const actionGroups: Array<{ label: string; actions: ActionOption[] }> = [
|
||||
{
|
||||
label: 'Informational',
|
||||
actions: ACTION_OPTIONS.filter((a) => a.severity === 'info'),
|
||||
},
|
||||
{
|
||||
label: 'Moderation',
|
||||
actions: ACTION_OPTIONS.filter((a) => a.severity === 'moderate'),
|
||||
},
|
||||
{
|
||||
label: 'Severe',
|
||||
actions: ACTION_OPTIONS.filter((a) => a.severity === 'severe'),
|
||||
},
|
||||
];
|
||||
|
||||
return (
|
||||
<div className="container max-w-5xl mx-auto py-8 px-4">
|
||||
<div className="flex items-start justify-between gap-4 mb-8">
|
||||
<div className="space-y-1">
|
||||
<div className="flex items-center gap-3">
|
||||
<h1 className="text-2xl font-bold tracking-tight">Report Case</h1>
|
||||
<Badge
|
||||
variant={statusConfig.variant}
|
||||
className="flex items-center gap-1.5 text-xs px-2 py-0.5"
|
||||
>
|
||||
{statusConfig.icon}
|
||||
{statusConfig.label}
|
||||
</Badge>
|
||||
</div>
|
||||
<p className="text-xs text-muted-foreground font-mono">{report.id}</p>
|
||||
</div>
|
||||
<Button
|
||||
variant="ghost"
|
||||
size="sm"
|
||||
onClick={() => router.push(`/admin?tab=reports&reportId=${report.id}`)}
|
||||
className="shrink-0 gap-1.5"
|
||||
>
|
||||
<ArrowLeft className="h-4 w-4" />
|
||||
Back to reports
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
<div className="grid grid-cols-1 md:grid-cols-5 gap-6">
|
||||
<div className="md:col-span-3 space-y-5">
|
||||
{report.reportedMessage ? (
|
||||
<div className="rounded-lg border border-border bg-muted/20 overflow-hidden">
|
||||
<div className="px-4 py-3 border-b border-border bg-muted/30 flex items-center gap-2">
|
||||
<MessageSquare className="h-3.5 w-3.5 text-muted-foreground" />
|
||||
<span className="text-xs font-semibold uppercase tracking-wider text-muted-foreground">
|
||||
Reported message
|
||||
</span>
|
||||
</div>
|
||||
<div className="px-4 py-4">
|
||||
<p className="text-sm leading-relaxed break-words">{report.reportedMessage}</p>
|
||||
</div>
|
||||
</div>
|
||||
) : (
|
||||
<div className="rounded-lg border border-dashed border-border px-4 py-5 flex items-center gap-3 text-muted-foreground">
|
||||
<Info className="h-4 w-4 shrink-0" />
|
||||
<span className="text-sm">No message content was captured with this report.</span>
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="rounded-lg border border-border overflow-hidden">
|
||||
<div className="px-4 py-3 border-b border-border bg-muted/20">
|
||||
<SectionLabel icon={<User className="h-3.5 w-3.5" />}>Parties</SectionLabel>
|
||||
</div>
|
||||
<div className="px-4 py-4 grid grid-cols-2 gap-x-6 gap-y-4">
|
||||
<InfoRow label="Reporter">
|
||||
<span className="font-medium">{report.reporter}</span>
|
||||
</InfoRow>
|
||||
<InfoRow label="Target">
|
||||
<div className="flex flex-wrap items-center gap-1.5">
|
||||
<span className="font-medium">{report.target}</span>
|
||||
{report.targetIsAdmin && (
|
||||
<Badge
|
||||
variant="outline"
|
||||
className="text-[10px] py-0 px-1.5 text-amber-500 border-amber-500/40"
|
||||
>
|
||||
Admin
|
||||
</Badge>
|
||||
)}
|
||||
{report.targetIsPlatformBanned && (
|
||||
<Badge
|
||||
variant="outline"
|
||||
className="text-[10px] py-0 px-1.5 text-destructive border-destructive/40"
|
||||
>
|
||||
Platform banned
|
||||
</Badge>
|
||||
)}
|
||||
</div>
|
||||
</InfoRow>
|
||||
<InfoRow label="Channel">
|
||||
<span className="font-mono text-xs bg-muted px-1.5 py-0.5 rounded">
|
||||
{report.channelName}
|
||||
</span>
|
||||
</InfoRow>
|
||||
<InfoRow label="Reason">
|
||||
<span>{report.reason}</span>
|
||||
</InfoRow>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="rounded-lg border border-border overflow-hidden">
|
||||
<div className="px-4 py-3 border-b border-border bg-muted/20">
|
||||
<SectionLabel icon={<Clock className="h-3.5 w-3.5" />}>Timeline</SectionLabel>
|
||||
</div>
|
||||
<div className="px-4 py-4 space-y-4">
|
||||
<InfoRow label="Filed">
|
||||
<span>
|
||||
{format(new Date(report.createdAt), 'PPP p')}{' '}
|
||||
<span className="text-muted-foreground text-xs">
|
||||
({formatDistanceToNow(new Date(report.createdAt), { addSuffix: true })})
|
||||
</span>
|
||||
</span>
|
||||
</InfoRow>
|
||||
{report.handledAt ? (
|
||||
<InfoRow label="Last handled">
|
||||
<span>
|
||||
{format(new Date(report.handledAt), 'PPP p')}{' '}
|
||||
<span className="text-muted-foreground text-xs">
|
||||
({formatDistanceToNow(new Date(report.handledAt), { addSuffix: true })})
|
||||
</span>
|
||||
</span>
|
||||
</InfoRow>
|
||||
) : null}
|
||||
<InfoRow label="Last action">
|
||||
{report.lastAction ? (
|
||||
<span className="font-medium">{ACTION_LABELS[report.lastAction]}</span>
|
||||
) : (
|
||||
<span className="text-muted-foreground">None</span>
|
||||
)}
|
||||
</InfoRow>
|
||||
<InfoRow label="Handled by">
|
||||
{report.handledBy ? (
|
||||
<span className="font-medium">{report.handledBy}</span>
|
||||
) : (
|
||||
<span className="text-muted-foreground">—</span>
|
||||
)}
|
||||
</InfoRow>
|
||||
{report.handlingNote ? (
|
||||
<div className="pt-1">
|
||||
<span className="text-[11px] font-medium uppercase tracking-wider text-muted-foreground block mb-1.5">
|
||||
Last note
|
||||
</span>
|
||||
<p className="text-sm bg-muted/40 rounded-md px-3 py-2.5 leading-relaxed border border-border">
|
||||
{report.handlingNote}
|
||||
</p>
|
||||
</div>
|
||||
) : null}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{report.targetIsAdmin && (
|
||||
<div className="flex items-start gap-3 rounded-lg border border-amber-500/30 bg-amber-500/5 px-4 py-3">
|
||||
<AlertTriangle className="h-4 w-4 text-amber-500 shrink-0 mt-0.5" />
|
||||
<div className="text-sm text-amber-700 dark:text-amber-400 leading-snug">
|
||||
<span className="font-semibold">Caution:</span> The reported user is a platform
|
||||
admin. Enforcement actions will still apply.
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div className="md:col-span-2">
|
||||
<div className="rounded-lg border border-border overflow-hidden sticky top-6">
|
||||
<div className="px-4 py-3 border-b border-border bg-muted/20 flex items-center gap-2">
|
||||
<Gavel className="h-3.5 w-3.5 text-muted-foreground" />
|
||||
<span className="text-xs font-semibold uppercase tracking-wider text-muted-foreground">
|
||||
Enforcement
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div className="px-4 py-4 space-y-5">
|
||||
{/* Action selector */}
|
||||
<div className="space-y-3">
|
||||
{actionGroups.map((group) => (
|
||||
<div key={group.label}>
|
||||
<p className="text-[10px] font-semibold uppercase tracking-widest text-muted-foreground mb-2">
|
||||
{group.label}
|
||||
</p>
|
||||
<div className="space-y-1.5">
|
||||
{group.actions.map((action) => {
|
||||
const isSelected = selectedAction === action.value;
|
||||
const styles = SEVERITY_STYLES[action.severity];
|
||||
return (
|
||||
<button
|
||||
key={action.value}
|
||||
type="button"
|
||||
onClick={() => setSelectedAction(action.value)}
|
||||
className={cn(
|
||||
'w-full flex items-start gap-3 rounded-md border px-3 py-2.5 text-left transition-all cursor-pointer',
|
||||
isSelected ? `${styles.selected} ring-1 ${styles.ring}` : styles.card
|
||||
)}
|
||||
>
|
||||
<span
|
||||
className={cn(
|
||||
'mt-0.5 shrink-0',
|
||||
isSelected ? styles.icon : 'text-muted-foreground'
|
||||
)}
|
||||
>
|
||||
{action.icon}
|
||||
</span>
|
||||
<div className="min-w-0">
|
||||
<p
|
||||
className={cn(
|
||||
'text-sm font-medium leading-tight',
|
||||
isSelected && action.severity === 'severe' && 'text-destructive',
|
||||
isSelected &&
|
||||
action.severity === 'moderate' &&
|
||||
'text-amber-600 dark:text-amber-400'
|
||||
)}
|
||||
>
|
||||
{action.label}
|
||||
</p>
|
||||
<p className="text-[11px] text-muted-foreground mt-0.5 leading-snug">
|
||||
{action.description}
|
||||
</p>
|
||||
</div>
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
|
||||
<Separator />
|
||||
|
||||
<div className="space-y-2">
|
||||
<Label htmlFor="note" className="text-xs">
|
||||
Moderator note
|
||||
{requiresNote ? (
|
||||
<span className="text-destructive ml-1">*</span>
|
||||
) : (
|
||||
<span className="text-muted-foreground ml-1">(optional)</span>
|
||||
)}
|
||||
</Label>
|
||||
<Textarea
|
||||
id="note"
|
||||
value={note}
|
||||
onChange={(e) => setNote(e.target.value)}
|
||||
placeholder="Explain why this action is being taken."
|
||||
rows={3}
|
||||
className="text-sm resize-none"
|
||||
/>
|
||||
{requiresNote && (
|
||||
<p className="text-[11px] text-muted-foreground">Min. 10 characters required.</p>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{isSevere && (
|
||||
<div className="flex items-start gap-2.5 rounded-md border border-destructive/30 bg-destructive/5 px-3 py-2.5">
|
||||
<ShieldAlert className="h-4 w-4 text-destructive shrink-0 mt-0.5" />
|
||||
<p className="text-[12px] text-destructive leading-snug">
|
||||
This is a severe action may often not be easily undone. Double-check before
|
||||
applying.
|
||||
</p>
|
||||
</div>
|
||||
)}
|
||||
|
||||
<Button
|
||||
onClick={submitAction}
|
||||
disabled={isSubmitting}
|
||||
variant={isSevere ? 'destructive' : 'default'}
|
||||
className="w-full gap-2"
|
||||
size="sm"
|
||||
>
|
||||
<Gavel className="h-3.5 w-3.5" />
|
||||
{isSubmitting ? 'Applying…' : 'Apply action'}
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
// ─── Types ───────────────────────────────────────────────────────────────────
|
||||
|
||||
interface ReportCasePageClientProps {
|
||||
report: ChatReportCase;
|
||||
}
|
||||
|
||||
interface ChatReportCase {
|
||||
id: string;
|
||||
status: 'OPEN' | 'REVIEWED' | 'DISMISSED';
|
||||
reason: string;
|
||||
reportedMessage: string | null;
|
||||
reportedMessageId: string | null;
|
||||
targetUsername: string | null;
|
||||
channelName: string;
|
||||
createdAt: string;
|
||||
handledAt: string | null;
|
||||
handlingNote: string | null;
|
||||
lastAction:
|
||||
| 'REVIEW'
|
||||
| 'DISMISS'
|
||||
| 'DELETE_REPORTED_MESSAGE'
|
||||
| 'TIMEOUT_10M'
|
||||
| 'TIMEOUT_1H'
|
||||
| 'BAN_CHAT'
|
||||
| 'LIFT_CHAT_BAN'
|
||||
| 'BAN_PLATFORM'
|
||||
| 'UNBAN_PLATFORM'
|
||||
| null;
|
||||
reporter: string;
|
||||
target: string;
|
||||
targetUserId: string | null;
|
||||
targetIsAdmin: boolean;
|
||||
targetIsPlatformBanned: boolean;
|
||||
handledBy: string | null;
|
||||
}
|
||||
@@ -0,0 +1,96 @@
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { prisma } from '@hctv/db';
|
||||
import { notFound, redirect } from 'next/navigation';
|
||||
import ReportCasePageClient from './page.client';
|
||||
|
||||
export default async function ReportCasePage({ params }: ReportCasePageProps) {
|
||||
const { reportId } = await params;
|
||||
const { user } = await validateRequest();
|
||||
|
||||
if (!user) {
|
||||
redirect('/auth/hackclub');
|
||||
}
|
||||
|
||||
if (!user.isAdmin) {
|
||||
redirect('/');
|
||||
}
|
||||
|
||||
const report = await prisma.chatUserReport.findUnique({
|
||||
where: {
|
||||
id: reportId,
|
||||
},
|
||||
include: {
|
||||
channel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
reporter: {
|
||||
include: {
|
||||
personalChannel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
targetUser: {
|
||||
include: {
|
||||
personalChannel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
ban: true,
|
||||
},
|
||||
},
|
||||
handledBy: {
|
||||
include: {
|
||||
personalChannel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
if (!report) {
|
||||
notFound();
|
||||
}
|
||||
|
||||
return (
|
||||
<ReportCasePageClient
|
||||
report={{
|
||||
id: report.id,
|
||||
status: report.status,
|
||||
reason: report.reason,
|
||||
reportedMessage: report.reportedMessage,
|
||||
reportedMessageId: report.reportedMessageId,
|
||||
targetUsername: report.targetUsername,
|
||||
channelName: report.channel.name,
|
||||
createdAt: report.createdAt.toISOString(),
|
||||
handledAt: report.handledAt?.toISOString() ?? null,
|
||||
handlingNote: report.handlingNote,
|
||||
lastAction: report.lastAction,
|
||||
reporter: report.reporter.personalChannel?.name ?? report.reporter.slack_id,
|
||||
target:
|
||||
report.targetUser?.personalChannel?.name ??
|
||||
report.targetUsername ??
|
||||
report.targetUserId ??
|
||||
'unknown',
|
||||
targetUserId: report.targetUserId,
|
||||
targetIsAdmin: Boolean(report.targetUser?.isAdmin),
|
||||
targetIsPlatformBanned: Boolean(report.targetUser?.ban),
|
||||
handledBy: report.handledBy?.personalChannel?.name ?? report.handledBy?.slack_id ?? null,
|
||||
}}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
interface ReportCasePageProps {
|
||||
params: Promise<{
|
||||
reportId: string;
|
||||
}>;
|
||||
}
|
||||
178
apps/web/src/app/(ui)/(protected)/api/admin/audit/route.ts
Normal file
178
apps/web/src/app/(ui)/(protected)/api/admin/audit/route.ts
Normal file
@@ -0,0 +1,178 @@
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { prisma } from '@hctv/db';
|
||||
import { NextRequest } from 'next/server';
|
||||
|
||||
export async function GET(request: NextRequest) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user?.isAdmin) {
|
||||
return new Response('Forbidden', { status: 403 });
|
||||
}
|
||||
|
||||
const searchParams = request.nextUrl.searchParams;
|
||||
const take = Math.min(Math.max(Number(searchParams.get('take') ?? 100), 10), 250);
|
||||
|
||||
const [adminLogs, chatLogs] = await Promise.all([
|
||||
prisma.adminAuditLog.findMany({
|
||||
orderBy: { createdAt: 'desc' },
|
||||
take,
|
||||
include: {
|
||||
actor: {
|
||||
select: {
|
||||
id: true,
|
||||
isAdmin: true,
|
||||
slack_id: true,
|
||||
personalChannel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
}),
|
||||
prisma.chatModerationEvent.findMany({
|
||||
orderBy: { createdAt: 'desc' },
|
||||
take,
|
||||
include: {
|
||||
channel: {
|
||||
select: {
|
||||
id: true,
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
moderator: {
|
||||
select: {
|
||||
id: true,
|
||||
isAdmin: true,
|
||||
slack_id: true,
|
||||
personalChannel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
targetUser: {
|
||||
select: {
|
||||
id: true,
|
||||
isAdmin: true,
|
||||
slack_id: true,
|
||||
personalChannel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
}),
|
||||
]);
|
||||
|
||||
const targetUserIds = [
|
||||
...new Set(adminLogs.map((log) => log.targetUserId).filter(Boolean)),
|
||||
] as string[];
|
||||
const targetUsers =
|
||||
targetUserIds.length > 0
|
||||
? await prisma.user.findMany({
|
||||
where: {
|
||||
id: {
|
||||
in: targetUserIds,
|
||||
},
|
||||
},
|
||||
include: {
|
||||
personalChannel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
})
|
||||
: [];
|
||||
const targetUserMap = new Map(
|
||||
targetUsers.map((targetUser) => [
|
||||
targetUser.id,
|
||||
targetUser.personalChannel?.name ?? targetUser.slack_id,
|
||||
])
|
||||
);
|
||||
const targetUserAdminMap = new Map(
|
||||
targetUsers.map((targetUser) => [targetUser.id, targetUser.isAdmin])
|
||||
);
|
||||
|
||||
const actorIds = [
|
||||
...new Set([
|
||||
...adminLogs.map((log) => log.actorId),
|
||||
...chatLogs.map((log) => log.moderatorId),
|
||||
...chatLogs.map((log) => log.targetUserId).filter(Boolean),
|
||||
...targetUserIds,
|
||||
]),
|
||||
] as string[];
|
||||
|
||||
const modRoleUsers =
|
||||
actorIds.length > 0
|
||||
? await prisma.user.findMany({
|
||||
where: {
|
||||
id: { in: actorIds },
|
||||
OR: [
|
||||
{ ownedChannels: { some: {} } },
|
||||
{ managedChannels: { some: {} } },
|
||||
{ chatModeratedChannels: { some: {} } },
|
||||
],
|
||||
},
|
||||
select: {
|
||||
id: true,
|
||||
},
|
||||
})
|
||||
: [];
|
||||
const channelModSet = new Set(modRoleUsers.map((user) => user.id));
|
||||
|
||||
const normalizedAdminLogs = adminLogs.map((log) => ({
|
||||
id: log.id,
|
||||
source: 'platform' as const,
|
||||
action: log.action,
|
||||
createdAt: log.createdAt,
|
||||
actor: log.actor.personalChannel?.name ?? log.actor.slack_id,
|
||||
target:
|
||||
log.targetChannel ??
|
||||
(log.targetUserId ? (targetUserMap.get(log.targetUserId) ?? log.targetUserId) : null),
|
||||
reason: log.reason,
|
||||
details: log.details,
|
||||
actorMeta: {
|
||||
isPlatformAdmin: log.actor.isAdmin,
|
||||
isChannelModerator: channelModSet.has(log.actorId),
|
||||
},
|
||||
targetMeta: log.targetUserId
|
||||
? {
|
||||
isPlatformAdmin: Boolean(targetUserAdminMap.get(log.targetUserId)),
|
||||
isChannelModerator: channelModSet.has(log.targetUserId),
|
||||
}
|
||||
: null,
|
||||
}));
|
||||
|
||||
const normalizedChatLogs = chatLogs.map((log) => ({
|
||||
id: log.id,
|
||||
source: 'chat' as const,
|
||||
action: log.action,
|
||||
createdAt: log.createdAt,
|
||||
actor: log.moderator.personalChannel?.name ?? log.moderator.slack_id,
|
||||
target: log.targetUser?.personalChannel?.name ?? log.channel.name,
|
||||
reason: log.reason,
|
||||
details: log.details,
|
||||
channelName: log.channel.name,
|
||||
actorMeta: {
|
||||
isPlatformAdmin: log.moderator.isAdmin,
|
||||
isChannelModerator: true,
|
||||
},
|
||||
targetMeta: log.targetUser
|
||||
? {
|
||||
isPlatformAdmin: log.targetUser.isAdmin,
|
||||
isChannelModerator: channelModSet.has(log.targetUser.id),
|
||||
}
|
||||
: null,
|
||||
}));
|
||||
|
||||
const logs = [...normalizedAdminLogs, ...normalizedChatLogs]
|
||||
.sort((a, b) => new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime())
|
||||
.slice(0, take);
|
||||
|
||||
return Response.json(logs);
|
||||
}
|
||||
142
apps/web/src/app/(ui)/(protected)/api/admin/channels/route.ts
Normal file
142
apps/web/src/app/(ui)/(protected)/api/admin/channels/route.ts
Normal file
@@ -0,0 +1,142 @@
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { AdminAuditAction, prisma } from '@hctv/db';
|
||||
import { NextRequest } from 'next/server';
|
||||
|
||||
export async function GET(request: NextRequest) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user?.isAdmin) {
|
||||
return new Response('Forbidden', { status: 403 });
|
||||
}
|
||||
|
||||
const searchParams = request.nextUrl.searchParams;
|
||||
const search = searchParams.get('search') || '';
|
||||
|
||||
const channels = await prisma.channel.findMany({
|
||||
where: search
|
||||
? {
|
||||
OR: [
|
||||
{ name: { contains: search, mode: 'insensitive' } },
|
||||
{ description: { contains: search, mode: 'insensitive' } },
|
||||
],
|
||||
}
|
||||
: undefined,
|
||||
include: {
|
||||
restriction: true,
|
||||
owner: {
|
||||
select: {
|
||||
id: true,
|
||||
slack_id: true,
|
||||
pfpUrl: true,
|
||||
personalChannel: { select: { name: true } },
|
||||
},
|
||||
},
|
||||
personalFor: {
|
||||
select: {
|
||||
id: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
take: 50,
|
||||
});
|
||||
return Response.json(channels);
|
||||
}
|
||||
|
||||
export async function POST(request: NextRequest) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user?.isAdmin) {
|
||||
return new Response('Forbidden', { status: 403 });
|
||||
}
|
||||
|
||||
let body: {
|
||||
channelId: string;
|
||||
action: 'restrict' | 'unrestrict';
|
||||
reason?: string;
|
||||
expiresAt?: string;
|
||||
};
|
||||
|
||||
try {
|
||||
body = await request.json();
|
||||
} catch {
|
||||
return new Response('Invalid JSON body', { status: 400 });
|
||||
}
|
||||
|
||||
const { channelId, action, reason, expiresAt } = body;
|
||||
|
||||
if (!channelId || !action) {
|
||||
return new Response('Missing required fields', { status: 400 });
|
||||
}
|
||||
|
||||
let expiresAtDate: Date | null = null;
|
||||
if (expiresAt !== undefined && expiresAt !== null && expiresAt !== '') {
|
||||
expiresAtDate = new Date(expiresAt);
|
||||
if (isNaN(expiresAtDate.getTime())) {
|
||||
return new Response('Invalid expiresAt date', { status: 400 });
|
||||
}
|
||||
if (expiresAtDate <= new Date()) {
|
||||
return new Response('expiresAt must be a future date', { status: 400 });
|
||||
}
|
||||
}
|
||||
|
||||
const channel = await prisma.channel.findUnique({ where: { id: channelId } });
|
||||
if (!channel) {
|
||||
return new Response('Channel not found', { status: 404 });
|
||||
}
|
||||
|
||||
if (action === 'restrict') {
|
||||
if (!reason) {
|
||||
return new Response('Reason is required for restricting', { status: 400 });
|
||||
}
|
||||
|
||||
await prisma.channelRestriction.upsert({
|
||||
where: { channelId },
|
||||
update: {
|
||||
reason,
|
||||
restrictedBy: user.id,
|
||||
expiresAt: expiresAtDate,
|
||||
},
|
||||
create: {
|
||||
channelId,
|
||||
reason,
|
||||
restrictedBy: user.id,
|
||||
expiresAt: expiresAtDate,
|
||||
},
|
||||
});
|
||||
|
||||
await prisma.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.CHANNEL_RESTRICTED,
|
||||
actorId: user.id,
|
||||
targetChannel: channel.name,
|
||||
reason,
|
||||
details: {
|
||||
channelId,
|
||||
expiresAt: expiresAtDate?.toISOString() ?? null,
|
||||
} as any,
|
||||
},
|
||||
});
|
||||
|
||||
return Response.json({ success: true, message: 'Channel restricted' });
|
||||
}
|
||||
|
||||
if (action === 'unrestrict') {
|
||||
const deleted = await prisma.channelRestriction.deleteMany({ where: { channelId } });
|
||||
if (deleted.count === 0) {
|
||||
return new Response('Channel does not have an active restriction', { status: 400 });
|
||||
}
|
||||
|
||||
await prisma.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.CHANNEL_UNRESTRICTED,
|
||||
actorId: user.id,
|
||||
targetChannel: channel.name,
|
||||
details: {
|
||||
channelId,
|
||||
} as any,
|
||||
},
|
||||
});
|
||||
|
||||
return Response.json({ success: true, message: 'Channel unrestricted' });
|
||||
}
|
||||
|
||||
return new Response('Invalid action', { status: 400 });
|
||||
}
|
||||
529
apps/web/src/app/(ui)/(protected)/api/admin/reports/route.ts
Normal file
529
apps/web/src/app/(ui)/(protected)/api/admin/reports/route.ts
Normal file
@@ -0,0 +1,529 @@
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import {
|
||||
AdminAuditAction,
|
||||
ChatModerationAction,
|
||||
ChatReportAction,
|
||||
ChatReportStatus,
|
||||
getRedisConnection,
|
||||
prisma,
|
||||
} from '@hctv/db';
|
||||
import { NextRequest } from 'next/server';
|
||||
|
||||
const redis = getRedisConnection();
|
||||
const REPORT_ALREADY_HANDLED_ERROR = 'REPORT_ALREADY_HANDLED';
|
||||
const NO_ACTIVE_CHAT_BAN_ERROR = 'NO_ACTIVE_CHAT_BAN';
|
||||
const NO_ACTIVE_PLATFORM_BAN_ERROR = 'NO_ACTIVE_PLATFORM_BAN';
|
||||
|
||||
export async function GET(request: NextRequest) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user?.isAdmin) {
|
||||
return new Response('Forbidden', { status: 403 });
|
||||
}
|
||||
|
||||
const searchParams = request.nextUrl.searchParams;
|
||||
const take = Math.min(Math.max(Number(searchParams.get('take') ?? 100), 10), 250);
|
||||
const reportId = searchParams.get('reportId')?.trim();
|
||||
|
||||
const reports = await prisma.chatUserReport.findMany({
|
||||
orderBy: { createdAt: 'desc' },
|
||||
take,
|
||||
include: {
|
||||
channel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
reporter: {
|
||||
include: {
|
||||
personalChannel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
targetUser: {
|
||||
include: {
|
||||
personalChannel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
handledBy: {
|
||||
include: {
|
||||
personalChannel: {
|
||||
select: {
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const normalizedReports = reports.map((report) => ({
|
||||
id: report.id,
|
||||
status: report.status,
|
||||
reason: report.reason,
|
||||
reportedMessage: report.reportedMessage,
|
||||
reportedMessageId: report.reportedMessageId,
|
||||
targetUsername: report.targetUsername,
|
||||
channelName: report.channel.name,
|
||||
createdAt: report.createdAt,
|
||||
handledAt: report.handledAt,
|
||||
handlingNote: report.handlingNote,
|
||||
lastAction: report.lastAction,
|
||||
reporter: report.reporter.personalChannel?.name ?? report.reporter.slack_id,
|
||||
handledBy: report.handledBy?.personalChannel?.name ?? report.handledBy?.slack_id ?? null,
|
||||
target:
|
||||
report.targetUser?.personalChannel?.name ??
|
||||
report.targetUsername ??
|
||||
report.targetUserId ??
|
||||
'unknown',
|
||||
}));
|
||||
|
||||
return Response.json({
|
||||
reports: normalizedReports,
|
||||
reportId,
|
||||
});
|
||||
}
|
||||
|
||||
export async function POST(request: NextRequest) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user?.isAdmin) {
|
||||
return new Response('Forbidden', { status: 403 });
|
||||
}
|
||||
|
||||
let body: {
|
||||
reportId?: string;
|
||||
action?:
|
||||
| 'review'
|
||||
| 'dismiss'
|
||||
| 'delete_reported_message'
|
||||
| 'timeout_10m'
|
||||
| 'timeout_1h'
|
||||
| 'ban_chat'
|
||||
| 'lift_chat_ban'
|
||||
| 'ban_platform'
|
||||
| 'unban_platform';
|
||||
note?: string;
|
||||
};
|
||||
|
||||
try {
|
||||
body = await request.json();
|
||||
} catch {
|
||||
return new Response('Invalid JSON body', { status: 400 });
|
||||
}
|
||||
|
||||
const reportId = body.reportId?.trim();
|
||||
const action = body.action;
|
||||
const note = body.note?.trim() || null;
|
||||
|
||||
if (!reportId || !action) {
|
||||
return new Response('Missing required fields', { status: 400 });
|
||||
}
|
||||
|
||||
const report = await prisma.chatUserReport.findUnique({
|
||||
where: { id: reportId },
|
||||
include: {
|
||||
channel: {
|
||||
select: {
|
||||
id: true,
|
||||
name: true,
|
||||
},
|
||||
},
|
||||
targetUser: {
|
||||
select: {
|
||||
id: true,
|
||||
isAdmin: true,
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
if (!report) {
|
||||
return new Response('Report not found', { status: 404 });
|
||||
}
|
||||
|
||||
const targetUserId = report.targetUserId ?? report.targetUser?.id ?? null;
|
||||
const isTargetAdmin = Boolean(report.targetUser?.isAdmin);
|
||||
|
||||
if (
|
||||
(action === 'ban_platform' ||
|
||||
action === 'ban_chat' ||
|
||||
action === 'timeout_10m' ||
|
||||
action === 'timeout_1h') &&
|
||||
isTargetAdmin
|
||||
) {
|
||||
return new Response('Cannot enforce this action on an admin user', { status: 400 });
|
||||
}
|
||||
|
||||
const reportPatchBase = {
|
||||
handledById: user.id,
|
||||
handledAt: new Date(),
|
||||
handlingNote: note,
|
||||
};
|
||||
|
||||
if (action === 'review') {
|
||||
try {
|
||||
await prisma.$transaction(async (tx) => {
|
||||
const claimed = await tx.chatUserReport.updateMany({
|
||||
where: { id: reportId, status: ChatReportStatus.OPEN },
|
||||
data: {
|
||||
...reportPatchBase,
|
||||
status: ChatReportStatus.REVIEWED,
|
||||
lastAction: ChatReportAction.REVIEW,
|
||||
},
|
||||
});
|
||||
if (claimed.count === 0) {
|
||||
throw new Error(REPORT_ALREADY_HANDLED_ERROR);
|
||||
}
|
||||
|
||||
await tx.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.REPORT_REVIEWED,
|
||||
actorId: user.id,
|
||||
targetUserId,
|
||||
targetChannel: report.channel.name,
|
||||
reason: note,
|
||||
details: {
|
||||
reportId,
|
||||
} as any,
|
||||
},
|
||||
});
|
||||
});
|
||||
} catch (error) {
|
||||
if (error instanceof Error && error.message === REPORT_ALREADY_HANDLED_ERROR) {
|
||||
return new Response('Report has already been handled', { status: 409 });
|
||||
}
|
||||
throw error;
|
||||
}
|
||||
|
||||
return Response.json({ success: true });
|
||||
}
|
||||
|
||||
if (action === 'dismiss') {
|
||||
try {
|
||||
await prisma.$transaction(async (tx) => {
|
||||
const claimed = await tx.chatUserReport.updateMany({
|
||||
where: { id: reportId, status: ChatReportStatus.OPEN },
|
||||
data: {
|
||||
...reportPatchBase,
|
||||
status: ChatReportStatus.DISMISSED,
|
||||
lastAction: ChatReportAction.DISMISS,
|
||||
},
|
||||
});
|
||||
if (claimed.count === 0) {
|
||||
throw new Error(REPORT_ALREADY_HANDLED_ERROR);
|
||||
}
|
||||
|
||||
await tx.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.REPORT_DISMISSED,
|
||||
actorId: user.id,
|
||||
targetUserId,
|
||||
targetChannel: report.channel.name,
|
||||
reason: note,
|
||||
details: {
|
||||
reportId,
|
||||
} as any,
|
||||
},
|
||||
});
|
||||
});
|
||||
} catch (error) {
|
||||
if (error instanceof Error && error.message === REPORT_ALREADY_HANDLED_ERROR) {
|
||||
return new Response('Report has already been handled', { status: 409 });
|
||||
}
|
||||
throw error;
|
||||
}
|
||||
|
||||
return Response.json({ success: true });
|
||||
}
|
||||
|
||||
if (action === 'delete_reported_message') {
|
||||
if (!report.reportedMessageId) {
|
||||
return new Response('No reported message id available for this report', { status: 400 });
|
||||
}
|
||||
|
||||
const channelKey = `chat:history:${report.channel.name}`;
|
||||
const history = await redis.zrange(channelKey, 0, -1);
|
||||
let deleted = false;
|
||||
|
||||
for (const entry of history) {
|
||||
try {
|
||||
const parsed = JSON.parse(entry) as { msgId?: string };
|
||||
if (parsed.msgId === report.reportedMessageId) {
|
||||
await redis.zrem(channelKey, entry);
|
||||
deleted = true;
|
||||
break;
|
||||
}
|
||||
} catch {
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
if (!deleted) {
|
||||
return new Response('Reported message was not found in chat history', { status: 404 });
|
||||
}
|
||||
|
||||
try {
|
||||
await prisma.$transaction(async (tx) => {
|
||||
const claimed = await tx.chatUserReport.updateMany({
|
||||
where: { id: reportId, status: ChatReportStatus.OPEN },
|
||||
data: {
|
||||
...reportPatchBase,
|
||||
status: ChatReportStatus.REVIEWED,
|
||||
lastAction: ChatReportAction.DELETE_REPORTED_MESSAGE,
|
||||
},
|
||||
});
|
||||
if (claimed.count === 0) {
|
||||
throw new Error(REPORT_ALREADY_HANDLED_ERROR);
|
||||
}
|
||||
|
||||
await tx.chatModerationEvent.create({
|
||||
data: {
|
||||
action: ChatModerationAction.MESSAGE_DELETED,
|
||||
channelId: report.channel.id,
|
||||
moderatorId: user.id,
|
||||
targetUserId,
|
||||
reason: note ?? 'Message deleted from report review',
|
||||
details: {
|
||||
reportId,
|
||||
msgId: report.reportedMessageId,
|
||||
} as any,
|
||||
},
|
||||
});
|
||||
|
||||
await tx.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.REPORT_ENFORCEMENT,
|
||||
actorId: user.id,
|
||||
targetUserId,
|
||||
targetChannel: report.channel.name,
|
||||
reason: note,
|
||||
details: {
|
||||
reportId,
|
||||
enforcement: 'DELETE_REPORTED_MESSAGE',
|
||||
msgId: report.reportedMessageId,
|
||||
} as any,
|
||||
},
|
||||
});
|
||||
});
|
||||
} catch (error) {
|
||||
if (error instanceof Error && error.message === REPORT_ALREADY_HANDLED_ERROR) {
|
||||
return new Response('Report has already been handled', { status: 409 });
|
||||
}
|
||||
throw error;
|
||||
}
|
||||
|
||||
return Response.json({ success: true });
|
||||
}
|
||||
|
||||
if (!targetUserId) {
|
||||
return new Response('Report target is unavailable', { status: 400 });
|
||||
}
|
||||
|
||||
if (
|
||||
action === 'timeout_10m' ||
|
||||
action === 'timeout_1h' ||
|
||||
action === 'ban_chat' ||
|
||||
action === 'lift_chat_ban'
|
||||
) {
|
||||
const timeoutSeconds = action === 'timeout_10m' ? 600 : action === 'timeout_1h' ? 3600 : null;
|
||||
try {
|
||||
await prisma.$transaction(async (tx) => {
|
||||
const claimed = await tx.chatUserReport.updateMany({
|
||||
where: { id: reportId, status: ChatReportStatus.OPEN },
|
||||
data: {
|
||||
...reportPatchBase,
|
||||
status: ChatReportStatus.REVIEWED,
|
||||
lastAction:
|
||||
action === 'timeout_10m'
|
||||
? ChatReportAction.TIMEOUT_10M
|
||||
: action === 'timeout_1h'
|
||||
? ChatReportAction.TIMEOUT_1H
|
||||
: action === 'ban_chat'
|
||||
? ChatReportAction.BAN_CHAT
|
||||
: ChatReportAction.LIFT_CHAT_BAN,
|
||||
},
|
||||
});
|
||||
if (claimed.count === 0) {
|
||||
throw new Error(REPORT_ALREADY_HANDLED_ERROR);
|
||||
}
|
||||
|
||||
if (action === 'lift_chat_ban') {
|
||||
const deleted = await tx.chatUserBan.deleteMany({
|
||||
where: {
|
||||
channelId: report.channel.id,
|
||||
userId: targetUserId,
|
||||
},
|
||||
});
|
||||
if (deleted.count === 0) {
|
||||
throw new Error(NO_ACTIVE_CHAT_BAN_ERROR);
|
||||
}
|
||||
} else {
|
||||
await tx.chatUserBan.upsert({
|
||||
where: {
|
||||
channelId_userId: {
|
||||
channelId: report.channel.id,
|
||||
userId: targetUserId,
|
||||
},
|
||||
},
|
||||
create: {
|
||||
channelId: report.channel.id,
|
||||
userId: targetUserId,
|
||||
bannedById: user.id,
|
||||
reason: note ?? report.reason,
|
||||
expiresAt: timeoutSeconds ? new Date(Date.now() + timeoutSeconds * 1000) : null,
|
||||
},
|
||||
update: {
|
||||
bannedById: user.id,
|
||||
reason: note ?? report.reason,
|
||||
expiresAt: timeoutSeconds ? new Date(Date.now() + timeoutSeconds * 1000) : null,
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
await tx.chatModerationEvent.create({
|
||||
data: {
|
||||
action:
|
||||
action === 'lift_chat_ban'
|
||||
? ChatModerationAction.USER_UNBANNED
|
||||
: action === 'ban_chat'
|
||||
? ChatModerationAction.USER_BANNED
|
||||
: ChatModerationAction.USER_TIMEOUT,
|
||||
channelId: report.channel.id,
|
||||
moderatorId: user.id,
|
||||
targetUserId,
|
||||
reason: note ?? report.reason,
|
||||
details:
|
||||
timeoutSeconds === null
|
||||
? ({ reportId } as any)
|
||||
: ({ reportId, durationSeconds: timeoutSeconds } as any),
|
||||
},
|
||||
});
|
||||
|
||||
await tx.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.REPORT_ENFORCEMENT,
|
||||
actorId: user.id,
|
||||
targetUserId,
|
||||
targetChannel: report.channel.name,
|
||||
reason: note,
|
||||
details: {
|
||||
reportId,
|
||||
enforcement:
|
||||
action === 'timeout_10m'
|
||||
? 'TIMEOUT_10M'
|
||||
: action === 'timeout_1h'
|
||||
? 'TIMEOUT_1H'
|
||||
: action === 'ban_chat'
|
||||
? 'BAN_CHAT'
|
||||
: 'LIFT_CHAT_BAN',
|
||||
} as any,
|
||||
},
|
||||
});
|
||||
});
|
||||
} catch (error) {
|
||||
if (error instanceof Error && error.message === REPORT_ALREADY_HANDLED_ERROR) {
|
||||
return new Response('Report has already been handled', { status: 409 });
|
||||
}
|
||||
if (error instanceof Error && error.message === NO_ACTIVE_CHAT_BAN_ERROR) {
|
||||
return new Response('User does not have an active chat ban for this channel', {
|
||||
status: 400,
|
||||
});
|
||||
}
|
||||
throw error;
|
||||
}
|
||||
|
||||
return Response.json({ success: true });
|
||||
}
|
||||
|
||||
if (action === 'ban_platform' || action === 'unban_platform') {
|
||||
try {
|
||||
await prisma.$transaction(async (tx) => {
|
||||
const claimed = await tx.chatUserReport.updateMany({
|
||||
where: { id: reportId, status: ChatReportStatus.OPEN },
|
||||
data: {
|
||||
...reportPatchBase,
|
||||
status: ChatReportStatus.REVIEWED,
|
||||
lastAction:
|
||||
action === 'ban_platform'
|
||||
? ChatReportAction.BAN_PLATFORM
|
||||
: ChatReportAction.UNBAN_PLATFORM,
|
||||
},
|
||||
});
|
||||
if (claimed.count === 0) {
|
||||
throw new Error(REPORT_ALREADY_HANDLED_ERROR);
|
||||
}
|
||||
|
||||
if (action === 'ban_platform') {
|
||||
await tx.userBan.upsert({
|
||||
where: { userId: targetUserId },
|
||||
update: {
|
||||
reason: note ?? report.reason,
|
||||
bannedBy: user.id,
|
||||
expiresAt: null,
|
||||
},
|
||||
create: {
|
||||
userId: targetUserId,
|
||||
reason: note ?? report.reason,
|
||||
bannedBy: user.id,
|
||||
expiresAt: null,
|
||||
},
|
||||
});
|
||||
} else {
|
||||
const deleted = await tx.userBan.deleteMany({ where: { userId: targetUserId } });
|
||||
if (deleted.count === 0) {
|
||||
throw new Error(NO_ACTIVE_PLATFORM_BAN_ERROR);
|
||||
}
|
||||
}
|
||||
|
||||
await tx.adminAuditLog.create({
|
||||
data: {
|
||||
action:
|
||||
action === 'ban_platform'
|
||||
? AdminAuditAction.USER_BANNED
|
||||
: AdminAuditAction.USER_UNBANNED,
|
||||
actorId: user.id,
|
||||
targetUserId,
|
||||
targetChannel: report.channel.name,
|
||||
reason: note,
|
||||
details: {
|
||||
reportId,
|
||||
source: 'CHAT_REPORT',
|
||||
} as any,
|
||||
},
|
||||
});
|
||||
|
||||
await tx.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.REPORT_ENFORCEMENT,
|
||||
actorId: user.id,
|
||||
targetUserId,
|
||||
targetChannel: report.channel.name,
|
||||
reason: note,
|
||||
details: {
|
||||
reportId,
|
||||
enforcement: action === 'ban_platform' ? 'BAN_PLATFORM' : 'UNBAN_PLATFORM',
|
||||
} as any,
|
||||
},
|
||||
});
|
||||
});
|
||||
} catch (error) {
|
||||
if (error instanceof Error && error.message === REPORT_ALREADY_HANDLED_ERROR) {
|
||||
return new Response('Report has already been handled', { status: 409 });
|
||||
}
|
||||
if (error instanceof Error && error.message === NO_ACTIVE_PLATFORM_BAN_ERROR) {
|
||||
return new Response('User does not have an active platform ban', { status: 400 });
|
||||
}
|
||||
throw error;
|
||||
}
|
||||
|
||||
return Response.json({ success: true });
|
||||
}
|
||||
|
||||
return new Response('Invalid action', { status: 400 });
|
||||
}
|
||||
178
apps/web/src/app/(ui)/(protected)/api/admin/users/route.ts
Normal file
178
apps/web/src/app/(ui)/(protected)/api/admin/users/route.ts
Normal file
@@ -0,0 +1,178 @@
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { AdminAuditAction, prisma } from '@hctv/db';
|
||||
import { NextRequest } from 'next/server';
|
||||
|
||||
export async function GET(request: NextRequest) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user?.isAdmin) {
|
||||
return new Response('Forbidden', { status: 403 });
|
||||
}
|
||||
|
||||
const searchParams = request.nextUrl.searchParams;
|
||||
const search = searchParams.get('search') || '';
|
||||
|
||||
const users = await prisma.user.findMany({
|
||||
where: search
|
||||
? {
|
||||
OR: [
|
||||
{ slack_id: { contains: search, mode: 'insensitive' } },
|
||||
{ email: { contains: search, mode: 'insensitive' } },
|
||||
{ personalChannel: { name: { contains: search, mode: 'insensitive' } } },
|
||||
],
|
||||
hasOnboarded: true,
|
||||
}
|
||||
: undefined,
|
||||
include: {
|
||||
ban: true,
|
||||
personalChannel: { select: { name: true } },
|
||||
},
|
||||
take: 50,
|
||||
});
|
||||
return Response.json(users);
|
||||
}
|
||||
|
||||
export async function POST(request: NextRequest) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user?.isAdmin) {
|
||||
return new Response('Forbidden', { status: 403 });
|
||||
}
|
||||
|
||||
let body: {
|
||||
userId: string;
|
||||
action: 'ban' | 'unban' | 'promote' | 'demote';
|
||||
reason?: string;
|
||||
expiresAt?: string;
|
||||
};
|
||||
|
||||
try {
|
||||
body = await request.json();
|
||||
} catch {
|
||||
return new Response('Invalid JSON body', { status: 400 });
|
||||
}
|
||||
|
||||
const { userId, action, reason, expiresAt } = body;
|
||||
|
||||
if (!userId || !action) {
|
||||
return new Response('Missing required fields', { status: 400 });
|
||||
}
|
||||
|
||||
let expiresAtDate: Date | null = null;
|
||||
if (expiresAt !== undefined && expiresAt !== null && expiresAt !== '') {
|
||||
expiresAtDate = new Date(expiresAt);
|
||||
if (isNaN(expiresAtDate.getTime())) {
|
||||
return new Response('Invalid expiresAt date', { status: 400 });
|
||||
}
|
||||
if (expiresAtDate <= new Date()) {
|
||||
return new Response('expiresAt must be a future date', { status: 400 });
|
||||
}
|
||||
}
|
||||
|
||||
const targetUser = await prisma.user.findUnique({ where: { id: userId } });
|
||||
if (!targetUser) {
|
||||
return new Response('User not found', { status: 404 });
|
||||
}
|
||||
|
||||
if (action === 'ban') {
|
||||
if (targetUser.isAdmin) {
|
||||
return new Response('Cannot ban an admin', { status: 400 });
|
||||
}
|
||||
|
||||
if (!reason) {
|
||||
return new Response('Reason is required for banning', { status: 400 });
|
||||
}
|
||||
|
||||
await prisma.userBan.upsert({
|
||||
where: { userId },
|
||||
update: {
|
||||
reason,
|
||||
bannedBy: user.id,
|
||||
expiresAt: expiresAtDate,
|
||||
},
|
||||
create: {
|
||||
userId,
|
||||
reason,
|
||||
bannedBy: user.id,
|
||||
expiresAt: expiresAtDate,
|
||||
},
|
||||
});
|
||||
|
||||
await prisma.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.USER_BANNED,
|
||||
actorId: user.id,
|
||||
targetUserId: userId,
|
||||
reason,
|
||||
details: {
|
||||
expiresAt: expiresAtDate?.toISOString() ?? null,
|
||||
} as any,
|
||||
},
|
||||
});
|
||||
|
||||
return Response.json({ success: true, message: 'User banned' });
|
||||
}
|
||||
|
||||
if (action === 'unban') {
|
||||
const deleted = await prisma.userBan.deleteMany({ where: { userId } });
|
||||
if (deleted.count === 0) {
|
||||
return new Response('User does not have an active platform ban', { status: 400 });
|
||||
}
|
||||
|
||||
await prisma.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.USER_UNBANNED,
|
||||
actorId: user.id,
|
||||
targetUserId: userId,
|
||||
},
|
||||
});
|
||||
|
||||
return Response.json({ success: true, message: 'User unbanned' });
|
||||
}
|
||||
|
||||
if (action === 'promote') {
|
||||
if (targetUser.isAdmin) {
|
||||
return new Response('User is already an admin', { status: 400 });
|
||||
}
|
||||
|
||||
await prisma.user.update({
|
||||
where: { id: userId },
|
||||
data: { isAdmin: true },
|
||||
});
|
||||
|
||||
await prisma.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.USER_PROMOTED,
|
||||
actorId: user.id,
|
||||
targetUserId: userId,
|
||||
},
|
||||
});
|
||||
|
||||
return Response.json({ success: true, message: 'User promoted to admin' });
|
||||
}
|
||||
|
||||
if (action === 'demote') {
|
||||
if (!targetUser.isAdmin) {
|
||||
return new Response('User is not an admin', { status: 400 });
|
||||
}
|
||||
|
||||
if (targetUser.id === user.id) {
|
||||
return new Response('Cannot demote yourself', { status: 400 });
|
||||
}
|
||||
|
||||
await prisma.user.update({
|
||||
where: { id: userId },
|
||||
data: { isAdmin: false },
|
||||
});
|
||||
|
||||
await prisma.adminAuditLog.create({
|
||||
data: {
|
||||
action: AdminAuditAction.USER_DEMOTED,
|
||||
actorId: user.id,
|
||||
targetUserId: userId,
|
||||
},
|
||||
});
|
||||
|
||||
return Response.json({ success: true, message: 'User demoted from admin' });
|
||||
}
|
||||
|
||||
return new Response('Invalid action', { status: 400 });
|
||||
}
|
||||
@@ -1,12 +1,17 @@
|
||||
import { prisma, getRedisConnection } from '@hctv/db';
|
||||
import { NextRequest } from 'next/server';
|
||||
import { z } from 'zod';
|
||||
import { lucia } from '@hctv/auth';
|
||||
|
||||
export async function POST(request: NextRequest) {
|
||||
const redis = getRedisConnection();
|
||||
const body = await request.json();
|
||||
|
||||
if (process.env.NODE_ENV !== 'production') {
|
||||
console.log(
|
||||
'Mediamtx publish auth request:',
|
||||
JSON.stringify(body, null, 2)
|
||||
);
|
||||
};
|
||||
const parsed = schema.safeParse(body);
|
||||
|
||||
if (!parsed.success) {
|
||||
@@ -20,11 +25,43 @@ export async function POST(request: NextRequest) {
|
||||
if (channelKey !== password) {
|
||||
return new Response('invalid stream key', { status: 403 });
|
||||
}
|
||||
|
||||
const channel = await prisma.channel.findUnique({
|
||||
where: { name: path },
|
||||
include: {
|
||||
restriction: true,
|
||||
owner: {
|
||||
include: { ban: true },
|
||||
},
|
||||
streamInfo: true,
|
||||
},
|
||||
});
|
||||
|
||||
if (channel?.restriction) {
|
||||
const isExpired = channel.restriction.expiresAt &&
|
||||
new Date(channel.restriction.expiresAt) < new Date();
|
||||
if (!isExpired) {
|
||||
return new Response('channel restricted', { status: 403 });
|
||||
}
|
||||
}
|
||||
|
||||
if (channel?.owner?.ban) {
|
||||
const isExpired = channel.owner.ban.expiresAt &&
|
||||
new Date(channel.owner.ban.expiresAt) < new Date();
|
||||
if (!isExpired) {
|
||||
return new Response('user banned', { status: 403 });
|
||||
}
|
||||
}
|
||||
|
||||
if (channel?.streamInfo[0].isLive) {
|
||||
return new Response('stream already live', { status: 403 });
|
||||
}
|
||||
|
||||
return new Response('youre in yay', { status: 200 });
|
||||
}
|
||||
} else if (action === 'read' && protocol === 'hls') {
|
||||
if (password === process.env.MEDIAMTX_PUBLISH_KEY) {
|
||||
return new Response('authorized', { status: 200 });
|
||||
return new Response('authorized (hls read key for thumbs)', { status: 200 });
|
||||
}
|
||||
const sessionExists = await redis.exists(`sessions:${password}`);
|
||||
if (!sessionExists) {
|
||||
|
||||
@@ -1,39 +0,0 @@
|
||||
import fsP from 'fs/promises';
|
||||
import fs from 'fs';
|
||||
import { getRedisConnection } from '@hctv/db';
|
||||
import { cookies } from 'next/headers';
|
||||
|
||||
export async function GET(request: Request, { params }: { params: Promise<{ path: string }> }) {
|
||||
const { path } = await params;
|
||||
const c = await cookies();
|
||||
|
||||
const sessionCookie = c.get('auth_session')?.value;
|
||||
if (!sessionCookie) {
|
||||
return new Response("Unauthorized", { status: 401 });
|
||||
}
|
||||
const sessionExists = await getRedisConnection().exists(`sessions:${sessionCookie}`);
|
||||
if (sessionExists === 0) {
|
||||
return new Response("Unauthorized", { status: 401 });
|
||||
}
|
||||
|
||||
if (path.includes('..')) {
|
||||
return new Response("nuh uh", { status: 403 });
|
||||
}
|
||||
|
||||
const basePath = '/dev/shm/hls';
|
||||
const filePath = `${basePath}/${path}`;
|
||||
const exists = fs.existsSync(filePath);
|
||||
|
||||
if (!exists) {
|
||||
return new Response("Not Found", { status: 404 });
|
||||
}
|
||||
|
||||
const file = await fsP.readFile(filePath);
|
||||
return new Response(file, {
|
||||
headers: {
|
||||
'Content-Type': 'application/octet-stream',
|
||||
'Access-Control-Allow-Origin': '*',
|
||||
'Access-Control-Allow-Methods': 'GET',
|
||||
},
|
||||
});
|
||||
}
|
||||
@@ -15,7 +15,14 @@ export async function POST(request: NextRequest) {
|
||||
key: streamKey,
|
||||
},
|
||||
include: {
|
||||
channel: true,
|
||||
channel: {
|
||||
include: {
|
||||
restriction: true,
|
||||
owner: {
|
||||
include: { ban: true },
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
@@ -24,6 +31,23 @@ export async function POST(request: NextRequest) {
|
||||
status: 403,
|
||||
});
|
||||
}
|
||||
|
||||
if (key.channel.restriction) {
|
||||
const isExpired = key.channel.restriction.expiresAt &&
|
||||
new Date(key.channel.restriction.expiresAt) < new Date();
|
||||
if (!isExpired) {
|
||||
return new Response('channel restricted', { status: 403 });
|
||||
}
|
||||
}
|
||||
|
||||
if (key.channel.owner?.ban) {
|
||||
const isExpired = key.channel.owner.ban.expiresAt &&
|
||||
new Date(key.channel.owner.ban.expiresAt) < new Date();
|
||||
if (!isExpired) {
|
||||
return new Response('user banned', { status: 403 });
|
||||
}
|
||||
}
|
||||
|
||||
return new Response('', {
|
||||
status: 302,
|
||||
headers: {
|
||||
|
||||
@@ -0,0 +1,107 @@
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { prisma, getRedisConnection } from '@hctv/db';
|
||||
import { NextRequest } from 'next/server';
|
||||
|
||||
const RATE_LIMIT_WINDOW_SECONDS = 10 * 60;
|
||||
const RATE_LIMIT_MAX_REPORTS = 5;
|
||||
|
||||
export async function POST(request: NextRequest) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user) {
|
||||
return new Response('Unauthorized', { status: 401 });
|
||||
}
|
||||
|
||||
let body: {
|
||||
channelName?: string;
|
||||
targetUserId?: string;
|
||||
targetUsername?: string;
|
||||
msgId?: string;
|
||||
message?: string;
|
||||
reason?: string;
|
||||
};
|
||||
|
||||
try {
|
||||
body = await request.json();
|
||||
} catch {
|
||||
return new Response('Invalid JSON body', { status: 400 });
|
||||
}
|
||||
|
||||
const channelName = body.channelName?.trim();
|
||||
const targetUserId = body.targetUserId?.trim();
|
||||
const reason = body.reason?.trim();
|
||||
|
||||
if (!channelName || !targetUserId || !reason) {
|
||||
return new Response('Missing required fields', { status: 400 });
|
||||
}
|
||||
|
||||
if (targetUserId === user.id) {
|
||||
return new Response('You cannot report yourself', { status: 400 });
|
||||
}
|
||||
|
||||
if (reason.length < 10 || reason.length > 1000) {
|
||||
return new Response('Reason must be between 10 and 1000 characters', { status: 400 });
|
||||
}
|
||||
|
||||
const redis = getRedisConnection();
|
||||
const rateLimitKey = `report_rl:${user.id}`;
|
||||
const currentCount = await redis.incr(rateLimitKey);
|
||||
if (currentCount === 1) {
|
||||
await redis.expire(rateLimitKey, RATE_LIMIT_WINDOW_SECONDS);
|
||||
}
|
||||
if (currentCount > RATE_LIMIT_MAX_REPORTS) {
|
||||
return new Response('Too many reports submitted. Please wait before submitting more.', {
|
||||
status: 429,
|
||||
});
|
||||
}
|
||||
|
||||
const [channel, targetUser] = await Promise.all([
|
||||
prisma.channel.findUnique({
|
||||
where: { name: channelName },
|
||||
select: { id: true },
|
||||
}),
|
||||
prisma.user.findUnique({
|
||||
where: { id: targetUserId },
|
||||
select: { id: true, personalChannel: { select: { name: true } } },
|
||||
}),
|
||||
]);
|
||||
|
||||
if (!channel) {
|
||||
return new Response('Channel not found', { status: 404 });
|
||||
}
|
||||
|
||||
if (!targetUser) {
|
||||
return new Response('Target user not found', { status: 404 });
|
||||
}
|
||||
|
||||
const msgId = body.msgId?.trim() || null;
|
||||
const duplicateCheck = await prisma.chatUserReport.findFirst({
|
||||
where: {
|
||||
channelId: channel.id,
|
||||
reporterId: user.id,
|
||||
targetUserId: targetUser.id,
|
||||
reportedMessageId: msgId,
|
||||
status: 'OPEN',
|
||||
},
|
||||
select: { id: true },
|
||||
});
|
||||
|
||||
if (duplicateCheck) {
|
||||
return new Response('You have already submitted an open report for this message.', {
|
||||
status: 409,
|
||||
});
|
||||
}
|
||||
|
||||
await prisma.chatUserReport.create({
|
||||
data: {
|
||||
channelId: channel.id,
|
||||
reporterId: user.id,
|
||||
targetUserId: targetUser.id,
|
||||
targetUsername: body.targetUsername?.trim() || targetUser.personalChannel?.name || null,
|
||||
reportedMessageId: msgId,
|
||||
reportedMessage: body.message?.trim().slice(0, 1000) || null,
|
||||
reason,
|
||||
},
|
||||
});
|
||||
|
||||
return Response.json({ success: true });
|
||||
}
|
||||
@@ -9,6 +9,7 @@ export async function GET(request: NextRequest) {
|
||||
const shouldGetOwned = searchParams.get('owned') === 'true';
|
||||
const allPersonalChannels = searchParams.get('personal') === 'true';
|
||||
const isLive = searchParams.get('live') === 'true';
|
||||
const username = searchParams.get('username');
|
||||
const { user } = await validateRequest();
|
||||
|
||||
if ((shouldGetOwned || allPersonalChannels) && !user) {
|
||||
@@ -18,6 +19,10 @@ export async function GET(request: NextRequest) {
|
||||
const where: Prisma.StreamInfoWhereInput = {};
|
||||
const channelConditions: Prisma.ChannelWhereInput[] = [];
|
||||
|
||||
if (username) {
|
||||
where.username = username;
|
||||
}
|
||||
|
||||
if (shouldGetOwned && user) {
|
||||
channelConditions.push({ ownerId: user.id });
|
||||
}
|
||||
@@ -46,6 +51,12 @@ export async function GET(request: NextRequest) {
|
||||
channel: {
|
||||
include: {
|
||||
personalFor: true,
|
||||
restriction: {
|
||||
select: {
|
||||
id: true,
|
||||
expiresAt: true,
|
||||
},
|
||||
},
|
||||
}
|
||||
},
|
||||
},
|
||||
@@ -58,6 +69,22 @@ export async function GET(request: NextRequest) {
|
||||
}
|
||||
// @ts-ignore
|
||||
delete obj.channel.obsChatGrantToken;
|
||||
|
||||
if (obj.channel.restriction) {
|
||||
const isExpired = obj.channel.restriction.expiresAt &&
|
||||
new Date(obj.channel.restriction.expiresAt) < new Date();
|
||||
if (isExpired) {
|
||||
// @ts-ignore
|
||||
obj.channel.restriction = null;
|
||||
} else {
|
||||
// @ts-ignore
|
||||
obj.channel.isRestricted = true;
|
||||
// @ts-ignore
|
||||
obj.channel.restrictionExpiresAt = obj.channel.restriction.expiresAt;
|
||||
// @ts-ignore
|
||||
delete obj.channel.restriction;
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
return Response.json(db);
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { redirect, RedirectType } from 'next/navigation';
|
||||
import { prisma } from '@hctv/db';
|
||||
|
||||
export default async function Layout({ children }: { children: React.ReactNode }) {
|
||||
const { user } = await validateRequest();
|
||||
@@ -9,5 +10,33 @@ export default async function Layout({ children }: { children: React.ReactNode }
|
||||
if (!user.hasOnboarded) {
|
||||
return redirect(`/onboarding`, RedirectType.push);
|
||||
}
|
||||
|
||||
const ban = await prisma.userBan.findUnique({
|
||||
where: { userId: user.id },
|
||||
});
|
||||
|
||||
if (ban) {
|
||||
const isExpired = ban.expiresAt && new Date(ban.expiresAt) < new Date();
|
||||
if (!isExpired) {
|
||||
return (
|
||||
<div className="flex flex-col items-center justify-center min-h-screen">
|
||||
<h1 className="text-3xl font-bold text-destructive mb-4">Account Suspended</h1>
|
||||
<p className="text-muted-foreground text-center max-w-md mb-4">
|
||||
Your account has been suspended from hackclub.tv.
|
||||
</p>
|
||||
<div className="bg-muted p-4 rounded-lg max-w-md">
|
||||
<p className="text-sm font-medium">Reason:</p>
|
||||
<p className="text-sm text-muted-foreground">{ban.reason}</p>
|
||||
</div>
|
||||
{ban.expiresAt && (
|
||||
<p className="text-sm text-muted-foreground mt-4">
|
||||
Expires: {new Date(ban.expiresAt).toLocaleDateString()}
|
||||
</p>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
return children;
|
||||
}
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
import { getBotBySlug } from '@/lib/db/resolve';
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { can } from '@/lib/auth/abac';
|
||||
import { redirect } from 'next/navigation';
|
||||
import Image from 'next/image';
|
||||
import { GeneralSettings } from '@/app/(ui)/(protected)/settings/bot/[slug]/gensettings';
|
||||
@@ -12,7 +13,7 @@ export default async function Page({ params }: { params: Promise<{ slug: string
|
||||
const { slug } = await params;
|
||||
const bot = await getBotBySlug(slug);
|
||||
|
||||
if (!bot || bot.ownerId !== user?.id) {
|
||||
if (!user || !bot || !can(user, 'update', 'bot', { bot })) {
|
||||
redirect('/settings/bot');
|
||||
}
|
||||
|
||||
|
||||
@@ -21,19 +21,36 @@ import {
|
||||
Wrench,
|
||||
Eye,
|
||||
EyeOff,
|
||||
MessageSquareWarning,
|
||||
Bot,
|
||||
} from 'lucide-react';
|
||||
import type { LucideIcon } from 'lucide-react';
|
||||
import { UniversalForm } from '@/components/app/UniversalForm/UniversalForm';
|
||||
import {
|
||||
updateChannelSettings,
|
||||
addChannelManager,
|
||||
removeChannelManager,
|
||||
addChatModerator,
|
||||
removeChatModerator,
|
||||
addChatBotModerator,
|
||||
removeChatBotModerator,
|
||||
deleteChannel,
|
||||
toggleGlobalChannelNotifs,
|
||||
editStreamInfo,
|
||||
changeUsername,
|
||||
updateChatModeration,
|
||||
} from '@/lib/form/actions';
|
||||
import { Switch } from '@/components/ui/switch';
|
||||
import { toast } from 'sonner';
|
||||
import type { Channel, User, StreamInfo, StreamKey, Follow } from '@hctv/db';
|
||||
import type {
|
||||
Channel,
|
||||
User,
|
||||
StreamInfo,
|
||||
StreamKey,
|
||||
Follow,
|
||||
ChatModerationSettings,
|
||||
BotAccount,
|
||||
} from '@hctv/db';
|
||||
import {
|
||||
Dialog,
|
||||
DialogContent,
|
||||
@@ -53,6 +70,15 @@ import { ChannelSelect } from '@/components/app/ChannelSelect/ChannelSelect';
|
||||
import { useRouter } from 'next/navigation';
|
||||
import Link from 'next/link';
|
||||
import { useConfirm } from '@omit/react-confirm-dialog';
|
||||
import {
|
||||
Select,
|
||||
SelectContent,
|
||||
SelectItem,
|
||||
SelectTrigger,
|
||||
SelectValue,
|
||||
} from '@/components/ui/select';
|
||||
import { getMediamtxClientEnvs } from '@/lib/utils/mediamtx/client';
|
||||
import type { MediaMTXRegion } from '@/lib/utils/mediamtx/regions';
|
||||
|
||||
interface ChannelSettingsClientProps {
|
||||
channel: Channel & {
|
||||
@@ -60,11 +86,17 @@ interface ChannelSettingsClientProps {
|
||||
ownerPersonalChannel: Channel | null;
|
||||
managers: User[];
|
||||
managerPersonalChannels: (Channel | null)[];
|
||||
chatModerators: User[];
|
||||
chatModeratorPersonalChannels: (Channel | null)[];
|
||||
chatModeratorBots: BotAccount[];
|
||||
teamBotAccounts: BotAccount[];
|
||||
streamInfo: StreamInfo[];
|
||||
streamKey: StreamKey | null;
|
||||
chatSettings: ChatModerationSettings | null;
|
||||
followers: (Follow & { user: { id: string; slack_id: string } })[];
|
||||
followerPersonalChannels: (Channel | null)[];
|
||||
is247: boolean;
|
||||
nameLastChanged: Date | null;
|
||||
};
|
||||
isOwner: boolean;
|
||||
currentUser: User;
|
||||
@@ -87,6 +119,7 @@ export default function ChannelSettingsClient({
|
||||
const [selTab, setSelTab] = useQueryState('tab', parseAsString.withDefault('general'));
|
||||
const [isUploading, setIsUploading] = useState(false);
|
||||
const [uploadError, setUploadError] = useState<string | null>(null);
|
||||
const [region, setRegion] = useState<MediaMTXRegion>('hq');
|
||||
const channelList = useOwnedChannels();
|
||||
const router = useRouter();
|
||||
|
||||
@@ -102,6 +135,38 @@ export default function ChannelSettingsClient({
|
||||
}
|
||||
}, []);
|
||||
|
||||
const handleModerationActionComplete = useCallback((result: any) => {
|
||||
if (result?.success) {
|
||||
toast.success('Moderation settings updated');
|
||||
}
|
||||
}, []);
|
||||
|
||||
const handleUsernameChangeComplete = useCallback(
|
||||
(result: any) => {
|
||||
if (result?.success && result?.newUsername) {
|
||||
toast.success('Username changed successfully! Redirecting...');
|
||||
router.push(`/settings/channel/${result.newUsername}?tab=${selTab}`);
|
||||
}
|
||||
},
|
||||
[router, selTab]
|
||||
);
|
||||
|
||||
const getUsernameChangeCooldownInfo = () => {
|
||||
if (!channel.nameLastChanged) {
|
||||
return { canChange: true, daysRemaining: 0 };
|
||||
}
|
||||
const daysSinceLastChange = Math.floor(
|
||||
(Date.now() - new Date(channel.nameLastChanged).getTime()) / (1000 * 60 * 60 * 24)
|
||||
);
|
||||
const cooldownDays = 30;
|
||||
if (daysSinceLastChange >= cooldownDays) {
|
||||
return { canChange: true, daysRemaining: 0 };
|
||||
}
|
||||
return { canChange: false, daysRemaining: cooldownDays - daysSinceLastChange };
|
||||
};
|
||||
|
||||
const cooldownInfo = getUsernameChangeCooldownInfo();
|
||||
|
||||
const copyStreamKey = async () => {
|
||||
if (streamKey) {
|
||||
await navigator.clipboard.writeText(streamKey);
|
||||
@@ -136,7 +201,8 @@ export default function ChannelSettingsClient({
|
||||
toast.error('Stream key not available');
|
||||
return '';
|
||||
}
|
||||
return `srt://${process.env.NEXT_PUBLIC_MEDIAMTX_INGEST_ROUTE}?streamid=publish:${channel.name}:thisusernameislongonpurposesoyoudontaccidentallyleakyourstreamkey:${streamKey}&pkt_size=1316`;
|
||||
const { ingestRoute } = getMediamtxClientEnvs(region);
|
||||
return `srt://${ingestRoute}?streamid=publish:${channel.name}:thisusernameislongonpurposesoyoudontaccidentallyleakyourstreamkey:${streamKey}&pkt_size=1316`;
|
||||
};
|
||||
|
||||
const copyStreamUrl = async () => {
|
||||
@@ -168,10 +234,10 @@ export default function ChannelSettingsClient({
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div className='flex-1' />
|
||||
<div className="flex-1" />
|
||||
<div>
|
||||
<ChannelSelect
|
||||
channelList={channelList.channels.map(c => c.channel)}
|
||||
channelList={channelList.channels.map((c) => c.channel)}
|
||||
value={channel.name}
|
||||
onSelect={(value) => {
|
||||
if (value === 'create') {
|
||||
@@ -185,7 +251,7 @@ export default function ChannelSettingsClient({
|
||||
</div>
|
||||
|
||||
<Tabs className="w-full" value={selTab} onValueChange={setSelTab}>
|
||||
<TabsList className={`grid w-full ${isPersonal ? 'grid-cols-4' : 'grid-cols-5'}`}>
|
||||
<TabsList className={`grid w-full ${isPersonal ? 'grid-cols-5' : 'grid-cols-6'}`}>
|
||||
<TabsTrigger value="general" className="flex items-center gap-2">
|
||||
<Settings className="h-4 w-4" />
|
||||
General
|
||||
@@ -204,8 +270,12 @@ export default function ChannelSettingsClient({
|
||||
<Bell className="h-4 w-4" />
|
||||
Notifications
|
||||
</TabsTrigger>
|
||||
<TabsTrigger value="moderation" className="flex items-center gap-2">
|
||||
<MessageSquareWarning className="h-4 w-4" />
|
||||
Moderation
|
||||
</TabsTrigger>
|
||||
<TabsTrigger value="utilities" className="flex items-center gap-2">
|
||||
<Wrench className='size-4' />
|
||||
<Wrench className="size-4" />
|
||||
Utilities
|
||||
</TabsTrigger>
|
||||
</TabsList>
|
||||
@@ -231,7 +301,7 @@ export default function ChannelSettingsClient({
|
||||
return (
|
||||
<div className="space-y-4">
|
||||
<input type="hidden" {...field} />
|
||||
|
||||
|
||||
{field.value && (
|
||||
<div className="flex items-center space-x-4">
|
||||
<Avatar className="h-16 w-16">
|
||||
@@ -240,7 +310,9 @@ export default function ChannelSettingsClient({
|
||||
</Avatar>
|
||||
<div className="flex-1">
|
||||
<p className="text-sm font-medium">Current profile picture</p>
|
||||
<p className="text-xs text-muted-foreground">Click "Upload new image" to replace</p>
|
||||
<p className="text-xs text-muted-foreground">
|
||||
Click "Upload new image" to replace
|
||||
</p>
|
||||
</div>
|
||||
<Button
|
||||
type="button"
|
||||
@@ -255,14 +327,14 @@ export default function ChannelSettingsClient({
|
||||
</Button>
|
||||
</div>
|
||||
)}
|
||||
|
||||
|
||||
<div>
|
||||
<UploadButton
|
||||
endpoint="pfpUpload"
|
||||
className="mt-2 ut-button:bg-mantle ut-button:text-mantle-foreground ut-allowed-content:text-muted-foreground/70"
|
||||
content={{
|
||||
button: field.value ? "Upload new image" : "Upload profile picture",
|
||||
allowedContent: "Image (1MB max)"
|
||||
button: field.value ? 'Upload new image' : 'Upload profile picture',
|
||||
allowedContent: 'Image (1MB max)',
|
||||
}}
|
||||
onUploadBegin={() => {
|
||||
setIsUploading(true);
|
||||
@@ -282,19 +354,15 @@ export default function ChannelSettingsClient({
|
||||
}}
|
||||
disabled={isUploading}
|
||||
/>
|
||||
|
||||
|
||||
{isUploading && (
|
||||
<p className="mt-2 text-sm text-primary">
|
||||
Uploading...
|
||||
</p>
|
||||
<p className="mt-2 text-sm text-primary">Uploading...</p>
|
||||
)}
|
||||
|
||||
|
||||
{uploadError && (
|
||||
<p className="mt-2 text-sm text-red-600">
|
||||
{uploadError}
|
||||
</p>
|
||||
<p className="mt-2 text-sm text-red-600">{uploadError}</p>
|
||||
)}
|
||||
|
||||
|
||||
{!field.value && !isUploading && !uploadError && (
|
||||
<p className="mt-2 text-sm text-muted-foreground">
|
||||
Upload a profile picture for your channel.
|
||||
@@ -340,7 +408,8 @@ export default function ChannelSettingsClient({
|
||||
<div>
|
||||
<label className="text-sm font-medium">24/7 Channel</label>
|
||||
<p className="text-xs text-muted-foreground">
|
||||
Mark this channel as always live. It will disable notifications on #hctv-streams.
|
||||
Mark this channel as always live. It will disable notifications on
|
||||
#hctv-streams.
|
||||
</p>
|
||||
</div>
|
||||
<Switch
|
||||
@@ -352,7 +421,7 @@ export default function ChannelSettingsClient({
|
||||
<input type="hidden" {...field} value={field.value ? 'true' : 'false'} />
|
||||
</div>
|
||||
),
|
||||
}
|
||||
},
|
||||
]}
|
||||
schemaName="updateChannelSettings"
|
||||
action={updateChannelSettings}
|
||||
@@ -360,6 +429,48 @@ export default function ChannelSettingsClient({
|
||||
onActionComplete={handleChannelSettingsActionComplete}
|
||||
/>
|
||||
|
||||
<Separator />
|
||||
|
||||
{isPersonal && isOwner && (
|
||||
<div className="space-y-4">
|
||||
<div className="flex items-center gap-2">
|
||||
<h3 className="text-lg font-semibold">Username</h3>
|
||||
</div>
|
||||
<p className="text-sm text-muted-foreground">
|
||||
Your username is how others find and mention you on hctv. You can change it once
|
||||
every 30 days.
|
||||
</p>
|
||||
{!cooldownInfo.canChange && (
|
||||
<div className="p-3 border border-accent/20 rounded-lg bg-accent/5">
|
||||
<p className="text-sm text-accent">
|
||||
You can change your username again in {cooldownInfo.daysRemaining} day
|
||||
{cooldownInfo.daysRemaining === 1 ? '' : 's'}.
|
||||
</p>
|
||||
</div>
|
||||
)}
|
||||
<UniversalForm
|
||||
fields={[
|
||||
{ name: 'channelId', type: 'hidden', value: channel.id, label: 'Channel ID' },
|
||||
{
|
||||
name: 'newUsername',
|
||||
label: 'New Username',
|
||||
type: 'text',
|
||||
value: '',
|
||||
placeholder: channel.name,
|
||||
description:
|
||||
'Only lowercase letters, numbers, underscores, and dashes. Max 20 characters.',
|
||||
inputFilter: /[^a-z0-9_-]/g,
|
||||
maxChars: 20,
|
||||
},
|
||||
]}
|
||||
schemaName="changeUsername"
|
||||
action={changeUsername}
|
||||
submitText="Change Username"
|
||||
onActionComplete={handleUsernameChangeComplete}
|
||||
/>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{isOwner && !isPersonal && (
|
||||
<>
|
||||
<Separator />
|
||||
@@ -430,7 +541,11 @@ export default function ChannelSettingsClient({
|
||||
onClick={() => setKeyVisible(!keyVisible)}
|
||||
className="absolute right-3 top-1/2 -translate-y-1/2 text-muted-foreground hover:text-foreground transition-colors"
|
||||
>
|
||||
{keyVisible ? <EyeOff className="h-4 w-4" /> : <Eye className="h-4 w-4" />}
|
||||
{keyVisible ? (
|
||||
<EyeOff className="h-4 w-4" />
|
||||
) : (
|
||||
<Eye className="h-4 w-4" />
|
||||
)}
|
||||
</button>
|
||||
</div>
|
||||
<Button onClick={regenerateStreamKey} variant="outline" size="smicon">
|
||||
@@ -442,13 +557,27 @@ export default function ChannelSettingsClient({
|
||||
onClick={copyStreamKey}
|
||||
disabled={!streamKey}
|
||||
>
|
||||
{copied.streamKey ? <Check className="h-4 w-4" /> : <Copy className="h-4 w-4" />}
|
||||
{copied.streamKey ? (
|
||||
<Check className="h-4 w-4" />
|
||||
) : (
|
||||
<Copy className="h-4 w-4" />
|
||||
)}
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="space-y-2">
|
||||
<label className="text-sm font-medium">Stream URL (for OBS)</label>
|
||||
<div className="flex items-center justify-between">
|
||||
<label className="text-sm font-medium">Stream URL (for OBS)</label>
|
||||
<Select value={region} onValueChange={(v) => setRegion(v as MediaMTXRegion)}>
|
||||
<SelectTrigger className="w-[180px] h-8">
|
||||
<SelectValue placeholder="Select region" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
<SelectItem value="hq">HQ Server A 🇺🇸</SelectItem>
|
||||
</SelectContent>
|
||||
</Select>
|
||||
</div>
|
||||
<div className="flex items-center gap-2">
|
||||
<div className="relative flex-1">
|
||||
<input
|
||||
@@ -464,7 +593,11 @@ export default function ChannelSettingsClient({
|
||||
onClick={copyStreamUrl}
|
||||
disabled={!streamKey}
|
||||
>
|
||||
{copied.streamUrl ? <Check className="h-4 w-4" /> : <Copy className="h-4 w-4" />}
|
||||
{copied.streamUrl ? (
|
||||
<Check className="h-4 w-4" />
|
||||
) : (
|
||||
<Copy className="h-4 w-4" />
|
||||
)}
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
@@ -472,7 +605,7 @@ export default function ChannelSettingsClient({
|
||||
<p className="text-xs text-muted-foreground mt-2">
|
||||
Need help getting started? Check out our{' '}
|
||||
<Link
|
||||
href="https://docs.hackclub.tv/guides/start-stream/"
|
||||
href="https://docs.hackclub.tv/guides/start-stream/"
|
||||
className="text-primary hover:underline"
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
@@ -549,7 +682,6 @@ export default function ChannelSettingsClient({
|
||||
</div>
|
||||
|
||||
<div className="space-y-3">
|
||||
{/* Owner */}
|
||||
<div className="flex items-center justify-between p-3 border rounded-lg">
|
||||
<div className="flex items-center gap-3">
|
||||
<Avatar className="h-10 w-10">
|
||||
@@ -563,13 +695,9 @@ export default function ChannelSettingsClient({
|
||||
<p className="text-sm text-mantle-foreground">Channel Owner</p>
|
||||
</div>
|
||||
</div>
|
||||
<Badge variant="default">
|
||||
<Shield className="h-3 w-3 mr-1" />
|
||||
Owner
|
||||
</Badge>
|
||||
<ChannelRoleBadges roles={['owner', 'chatModerator']} />
|
||||
</div>
|
||||
|
||||
{/* Managers */}
|
||||
{channel.managers.map((manager) => {
|
||||
const personalChannel = channel.managerPersonalChannels.find(
|
||||
(c) => c?.ownerId === manager.id
|
||||
@@ -589,24 +717,34 @@ export default function ChannelSettingsClient({
|
||||
<p className="text-sm text-mantle-foreground">Manager</p>
|
||||
</div>
|
||||
</div>
|
||||
{isOwner && (
|
||||
<Button
|
||||
variant="outline"
|
||||
size="sm"
|
||||
onClick={async () => {
|
||||
if (await confirm({
|
||||
title: 'Remove Manager',
|
||||
description: `Are you sure you want to remove ${personalChannel?.name} as a manager? They will no longer be able to stream or moderate this channel.`,
|
||||
confirmText: 'Remove',
|
||||
cancelText: 'Cancel',
|
||||
})) {
|
||||
removeChannelManager(channel.id, manager.id);
|
||||
}
|
||||
}}
|
||||
>
|
||||
<UserMinus className="h-4 w-4" />
|
||||
</Button>
|
||||
)}
|
||||
<div className="flex items-center gap-2">
|
||||
<ChannelRoleBadges
|
||||
roles={withPlatformAdmin(
|
||||
['manager', 'chatModerator'] as const,
|
||||
manager.isAdmin
|
||||
)}
|
||||
/>
|
||||
{isOwner && (
|
||||
<Button
|
||||
variant="outline"
|
||||
size="sm"
|
||||
onClick={async () => {
|
||||
if (
|
||||
await confirm({
|
||||
title: 'Remove Manager',
|
||||
description: `Are you sure you want to remove ${personalChannel?.name} as a manager? They will no longer be able to stream or moderate this channel.`,
|
||||
confirmText: 'Remove',
|
||||
cancelText: 'Cancel',
|
||||
})
|
||||
) {
|
||||
removeChannelManager(channel.id, manager.id);
|
||||
}
|
||||
}}
|
||||
>
|
||||
<UserMinus className="h-4 w-4" />
|
||||
</Button>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
@@ -694,6 +832,221 @@ export default function ChannelSettingsClient({
|
||||
</CardContent>
|
||||
</Card>
|
||||
</TabsContent>
|
||||
|
||||
<TabsContent value="moderation">
|
||||
<Card>
|
||||
<CardHeader>
|
||||
<CardTitle>Chat Moderation</CardTitle>
|
||||
<CardDescription>
|
||||
Configure rate limits, slow mode, and blocked words for this channel's live
|
||||
chat.
|
||||
</CardDescription>
|
||||
</CardHeader>
|
||||
<CardContent className="space-y-6">
|
||||
<div className="space-y-4">
|
||||
<div className="flex items-center justify-between">
|
||||
<h3 className="text-lg font-semibold">Moderators</h3>
|
||||
<div className="flex gap-2">
|
||||
<AddChatModeratorDialog
|
||||
channelId={channel.id}
|
||||
existingModerators={[
|
||||
channel.owner.id,
|
||||
...channel.managers.map((manager) => manager.id),
|
||||
...channel.chatModerators.map((moderator) => moderator.id),
|
||||
]}
|
||||
/>
|
||||
<AddChatBotModeratorDialog
|
||||
channelId={channel.id}
|
||||
teamBots={channel.teamBotAccounts}
|
||||
existingBotModerators={channel.chatModeratorBots.map((bot) => bot.id)}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="space-y-3">
|
||||
<div className="flex items-center justify-between p-3 border rounded-lg">
|
||||
<div className="flex items-center gap-3">
|
||||
<Avatar className="h-10 w-10">
|
||||
<AvatarImage src={channel.owner.pfpUrl} />
|
||||
<AvatarFallback>{channel.owner.slack_id[0]?.toUpperCase()}</AvatarFallback>
|
||||
</Avatar>
|
||||
<div>
|
||||
<p className="font-medium">{channel.ownerPersonalChannel?.name}</p>
|
||||
<p className="text-sm text-mantle-foreground">Owner</p>
|
||||
</div>
|
||||
</div>
|
||||
<ChannelRoleBadges
|
||||
roles={withPlatformAdmin(
|
||||
['owner', 'chatModerator'] as const,
|
||||
channel.owner.isAdmin
|
||||
)}
|
||||
/>
|
||||
</div>
|
||||
|
||||
{channel.managers.map((manager) => {
|
||||
const personalChannel = channel.managerPersonalChannels.find(
|
||||
(candidate) => candidate?.ownerId === manager.id
|
||||
);
|
||||
return (
|
||||
<div
|
||||
key={manager.id}
|
||||
className="flex items-center justify-between p-3 border rounded-lg"
|
||||
>
|
||||
<div className="flex items-center gap-3">
|
||||
<Avatar className="h-10 w-10">
|
||||
<AvatarImage src={manager.pfpUrl} />
|
||||
<AvatarFallback>{personalChannel?.name}</AvatarFallback>
|
||||
</Avatar>
|
||||
<div>
|
||||
<p className="font-medium">{personalChannel?.name}</p>
|
||||
<p className="text-sm text-mantle-foreground">Manager</p>
|
||||
</div>
|
||||
</div>
|
||||
<ChannelRoleBadges
|
||||
roles={withPlatformAdmin(
|
||||
['manager', 'chatModerator'] as const,
|
||||
manager.isAdmin
|
||||
)}
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
|
||||
{channel.chatModerators.map((moderator) => {
|
||||
const personalChannel = channel.chatModeratorPersonalChannels.find(
|
||||
(candidate) => candidate?.ownerId === moderator.id
|
||||
);
|
||||
return (
|
||||
<div
|
||||
key={moderator.id}
|
||||
className="flex items-center justify-between p-3 border rounded-lg"
|
||||
>
|
||||
<div className="flex items-center gap-3">
|
||||
<Avatar className="h-10 w-10">
|
||||
<AvatarImage src={moderator.pfpUrl} />
|
||||
<AvatarFallback>{personalChannel?.name}</AvatarFallback>
|
||||
</Avatar>
|
||||
<div>
|
||||
<p className="font-medium">{personalChannel?.name}</p>
|
||||
<p className="text-sm text-mantle-foreground">Chat moderator</p>
|
||||
</div>
|
||||
</div>
|
||||
<div className="flex items-center gap-2">
|
||||
<ChannelRoleBadges
|
||||
roles={withPlatformAdmin(['chatModerator'] as const, moderator.isAdmin)}
|
||||
/>
|
||||
<Button
|
||||
variant="outline"
|
||||
size="sm"
|
||||
onClick={() => {
|
||||
toast.promise(removeChatModerator(channel.id, moderator.id), {
|
||||
loading: 'Removing moderator...',
|
||||
success: 'Moderator removed',
|
||||
error: 'Failed to remove moderator',
|
||||
});
|
||||
}}
|
||||
>
|
||||
<UserMinus className="h-4 w-4" />
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
|
||||
{channel.chatModeratorBots.map((botAccount) => (
|
||||
<div
|
||||
key={botAccount.id}
|
||||
className="flex items-center justify-between p-3 border rounded-lg"
|
||||
>
|
||||
<div className="flex items-center gap-3">
|
||||
<Avatar className="h-10 w-10">
|
||||
<AvatarImage src={botAccount.pfpUrl} />
|
||||
<AvatarFallback>{botAccount.slug[0]?.toUpperCase()}</AvatarFallback>
|
||||
</Avatar>
|
||||
<div>
|
||||
<p className="font-medium">{botAccount.displayName}</p>
|
||||
<p className="text-sm text-mantle-foreground">@{botAccount.slug}</p>
|
||||
</div>
|
||||
</div>
|
||||
<div className="flex items-center gap-2">
|
||||
<ChannelRoleBadges roles={['botModerator']} />
|
||||
<Button
|
||||
variant="outline"
|
||||
size="sm"
|
||||
onClick={() => {
|
||||
toast.promise(removeChatBotModerator(channel.id, botAccount.id), {
|
||||
loading: 'Removing bot moderator...',
|
||||
success: 'Bot moderator removed',
|
||||
error: 'Failed to remove bot moderator',
|
||||
});
|
||||
}}
|
||||
>
|
||||
<UserMinus className="h-4 w-4" />
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
))}
|
||||
|
||||
{channel.chatModerators.length === 0 &&
|
||||
channel.chatModeratorBots.length === 0 && (
|
||||
<p className="text-mantle-foreground text-center py-4">
|
||||
No extra chat moderators yet.
|
||||
</p>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<Separator />
|
||||
|
||||
<UniversalForm
|
||||
fields={[
|
||||
{ name: 'channelId', type: 'hidden', value: channel.id, label: 'Channel ID' },
|
||||
{
|
||||
name: 'slowModeSeconds',
|
||||
label: 'Slow mode (seconds)',
|
||||
type: 'number',
|
||||
value: channel.chatSettings?.slowModeSeconds ?? 0,
|
||||
description: 'Users can send one message per interval. Set 0 to disable.',
|
||||
},
|
||||
{
|
||||
name: 'maxMessageLength',
|
||||
label: 'Max message length',
|
||||
type: 'number',
|
||||
value: channel.chatSettings?.maxMessageLength ?? 400,
|
||||
description: 'Maximum allowed message length in characters.',
|
||||
},
|
||||
{
|
||||
name: 'rateLimitCount',
|
||||
label: 'Messages per window',
|
||||
type: 'number',
|
||||
value: channel.chatSettings?.rateLimitCount ?? 8,
|
||||
description: 'How many messages a user can send in the rate limit window.',
|
||||
},
|
||||
{
|
||||
name: 'rateLimitWindowSeconds',
|
||||
label: 'Rate window (seconds)',
|
||||
type: 'number',
|
||||
value: channel.chatSettings?.rateLimitWindowSeconds ?? 10,
|
||||
description: 'Window size used for spam protection.',
|
||||
},
|
||||
{
|
||||
name: 'blockedTerms',
|
||||
label: 'Blocked terms',
|
||||
value: (channel.chatSettings?.blockedTerms ?? []).join('\n'),
|
||||
textArea: true,
|
||||
textAreaRows: 8,
|
||||
description:
|
||||
'One term per line (or comma-separated). Messages containing these terms are blocked.',
|
||||
},
|
||||
]}
|
||||
schemaName="updateChatModeration"
|
||||
action={updateChatModeration}
|
||||
submitText="Save moderation settings"
|
||||
onActionComplete={handleModerationActionComplete}
|
||||
/>
|
||||
</CardContent>
|
||||
</Card>
|
||||
</TabsContent>
|
||||
<TabsContent value="utilities">
|
||||
<Card>
|
||||
<CardHeader>
|
||||
@@ -705,7 +1058,7 @@ export default function ChannelSettingsClient({
|
||||
<div>
|
||||
<h3 className="text-lg font-semibold mb-2">Chat overlay</h3>
|
||||
<p className="text-sm text-mantle-foreground mb-4">
|
||||
Add a 300x600 browser source with this and enjoy!
|
||||
Add a 300x600 browser source with this and enjoy!
|
||||
</p>
|
||||
<div className="flex items-center gap-2">
|
||||
<div className="relative flex-1">
|
||||
@@ -730,6 +1083,74 @@ export default function ChannelSettingsClient({
|
||||
);
|
||||
}
|
||||
|
||||
function RoleBadge({
|
||||
icon: Icon,
|
||||
label,
|
||||
className,
|
||||
}: {
|
||||
icon: LucideIcon;
|
||||
label: string;
|
||||
className: string;
|
||||
}) {
|
||||
return (
|
||||
<Badge variant="outline" className={className}>
|
||||
<Icon className="h-3 w-3 mr-1" />
|
||||
{label}
|
||||
</Badge>
|
||||
);
|
||||
}
|
||||
|
||||
type ChannelRoleBadgeKey = 'owner' | 'manager' | 'chatModerator' | 'botModerator' | 'platformAdmin';
|
||||
|
||||
const ROLE_BADGE_META: Record<
|
||||
ChannelRoleBadgeKey,
|
||||
{ icon: LucideIcon; label: string; className: string }
|
||||
> = {
|
||||
owner: {
|
||||
icon: Shield,
|
||||
label: 'Owner',
|
||||
className: 'border-primary/30 bg-primary/10 text-primary',
|
||||
},
|
||||
manager: {
|
||||
icon: Wrench,
|
||||
label: 'Manager',
|
||||
className: 'border-sky-500/30 bg-sky-500/10 text-sky-700 dark:text-sky-300',
|
||||
},
|
||||
chatModerator: {
|
||||
icon: MessageSquareWarning,
|
||||
label: 'Chat Mod',
|
||||
className: 'border-emerald-500/30 bg-emerald-500/10 text-emerald-700 dark:text-emerald-300',
|
||||
},
|
||||
botModerator: {
|
||||
icon: Bot,
|
||||
label: 'Bot Mod',
|
||||
className: 'border-cyan-500/30 bg-cyan-500/10 text-cyan-700 dark:text-cyan-300',
|
||||
},
|
||||
platformAdmin: {
|
||||
icon: Shield,
|
||||
label: 'Platform Admin',
|
||||
className: 'border-amber-500/30 bg-amber-500/10 text-amber-700 dark:text-amber-300',
|
||||
},
|
||||
};
|
||||
|
||||
const withPlatformAdmin = (
|
||||
roles: readonly ChannelRoleBadgeKey[],
|
||||
isPlatformAdmin?: boolean
|
||||
): ChannelRoleBadgeKey[] => (isPlatformAdmin ? [...roles, 'platformAdmin'] : [...roles]);
|
||||
|
||||
function ChannelRoleBadges({ roles }: { roles: ChannelRoleBadgeKey[] }) {
|
||||
return (
|
||||
<div className="flex items-center gap-1.5 flex-wrap justify-end">
|
||||
{[...new Set(roles)].map((role) => {
|
||||
const meta = ROLE_BADGE_META[role];
|
||||
return (
|
||||
<RoleBadge key={role} icon={meta.icon} label={meta.label} className={meta.className} />
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
function AddManagerDialog({
|
||||
channelId,
|
||||
existingManagers,
|
||||
@@ -782,3 +1203,121 @@ function AddManagerDialog({
|
||||
</Dialog>
|
||||
);
|
||||
}
|
||||
|
||||
function AddChatModeratorDialog({
|
||||
channelId,
|
||||
existingModerators,
|
||||
}: {
|
||||
channelId: string;
|
||||
existingModerators: string[];
|
||||
}) {
|
||||
const [open, setOpen] = useState(false);
|
||||
const [selectedChannel, setSelectedChannel] = useState('');
|
||||
|
||||
return (
|
||||
<Dialog open={open} onOpenChange={setOpen}>
|
||||
<DialogTrigger asChild>
|
||||
<Button size="sm" variant="outline">
|
||||
<UserPlus className="h-4 w-4 mr-2" />
|
||||
Add User Moderator
|
||||
</Button>
|
||||
</DialogTrigger>
|
||||
<DialogContent>
|
||||
<DialogHeader>
|
||||
<DialogTitle>Add chat moderator</DialogTitle>
|
||||
<DialogDescription>
|
||||
Choose a user who should be able to moderate this channel's chat.
|
||||
</DialogDescription>
|
||||
</DialogHeader>
|
||||
<UserCombobox
|
||||
onValueChange={(value) => {
|
||||
setSelectedChannel(value);
|
||||
}}
|
||||
filter={existingModerators}
|
||||
value={selectedChannel}
|
||||
modal
|
||||
/>
|
||||
<DialogFooter>
|
||||
<Button
|
||||
disabled={!selectedChannel}
|
||||
onClick={() => {
|
||||
toast.promise(addChatModerator(channelId, selectedChannel), {
|
||||
loading: 'Adding moderator...',
|
||||
success: 'Moderator added',
|
||||
error: 'Failed to add moderator',
|
||||
});
|
||||
setOpen(false);
|
||||
setSelectedChannel('');
|
||||
}}
|
||||
>
|
||||
Add Moderator
|
||||
</Button>
|
||||
</DialogFooter>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
);
|
||||
}
|
||||
|
||||
function AddChatBotModeratorDialog({
|
||||
channelId,
|
||||
teamBots,
|
||||
existingBotModerators,
|
||||
}: {
|
||||
channelId: string;
|
||||
teamBots: BotAccount[];
|
||||
existingBotModerators: string[];
|
||||
}) {
|
||||
const [open, setOpen] = useState(false);
|
||||
const [selectedBotId, setSelectedBotId] = useState('');
|
||||
|
||||
const availableBots = teamBots.filter(
|
||||
(botAccount) => !existingBotModerators.includes(botAccount.id)
|
||||
);
|
||||
|
||||
return (
|
||||
<Dialog open={open} onOpenChange={setOpen}>
|
||||
<DialogTrigger asChild>
|
||||
<Button size="sm" variant="outline">
|
||||
<Bot className="h-4 w-4 mr-2" />
|
||||
Add Bot Moderator
|
||||
</Button>
|
||||
</DialogTrigger>
|
||||
<DialogContent>
|
||||
<DialogHeader>
|
||||
<DialogTitle>Add bot moderator</DialogTitle>
|
||||
<DialogDescription>
|
||||
Bots can delete messages, timeout users, and ban users in chat.
|
||||
</DialogDescription>
|
||||
</DialogHeader>
|
||||
<Select value={selectedBotId} onValueChange={setSelectedBotId}>
|
||||
<SelectTrigger>
|
||||
<SelectValue placeholder="Select bot" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
{availableBots.map((botAccount) => (
|
||||
<SelectItem key={botAccount.id} value={botAccount.id}>
|
||||
{botAccount.displayName} (@{botAccount.slug})
|
||||
</SelectItem>
|
||||
))}
|
||||
</SelectContent>
|
||||
</Select>
|
||||
<DialogFooter>
|
||||
<Button
|
||||
disabled={!selectedBotId}
|
||||
onClick={() => {
|
||||
toast.promise(addChatBotModerator(channelId, selectedBotId), {
|
||||
loading: 'Adding bot moderator...',
|
||||
success: 'Bot moderator added',
|
||||
error: 'Failed to add bot moderator',
|
||||
});
|
||||
setOpen(false);
|
||||
setSelectedBotId('');
|
||||
}}
|
||||
>
|
||||
Add Bot
|
||||
</Button>
|
||||
</DialogFooter>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -3,6 +3,7 @@ import { prisma } from '@hctv/db';
|
||||
import { redirect } from 'next/navigation';
|
||||
import ChannelSettingsClient from './page.client';
|
||||
import { resolvePersonalChannel } from '@/lib/auth/resolve';
|
||||
import { can } from '@/lib/auth/abac';
|
||||
|
||||
export default async function ChannelSettingsPage({
|
||||
params,
|
||||
@@ -21,8 +22,11 @@ export default async function ChannelSettingsPage({
|
||||
include: {
|
||||
owner: true,
|
||||
managers: true,
|
||||
chatModerators: true,
|
||||
chatModeratorBots: true,
|
||||
streamInfo: true,
|
||||
streamKey: true,
|
||||
chatSettings: true,
|
||||
followers: {
|
||||
include: {
|
||||
user: {
|
||||
@@ -42,9 +46,8 @@ export default async function ChannelSettingsPage({
|
||||
}
|
||||
|
||||
const isOwner = channel.ownerId === user.id;
|
||||
const isManager = channel.managers.some((manager) => manager.id === user.id);
|
||||
|
||||
if (!isOwner && !isManager) {
|
||||
if (!can(user, 'update', 'channel', { channel })) {
|
||||
redirect('/');
|
||||
}
|
||||
|
||||
@@ -52,17 +55,38 @@ export default async function ChannelSettingsPage({
|
||||
const managerPersonalChannels = await Promise.all(
|
||||
channel.managers.map((manager) => resolvePersonalChannel(manager.id))
|
||||
);
|
||||
const managerIds = new Set(channel.managers.map((manager) => manager.id));
|
||||
const extraChatModerators = channel.chatModerators.filter(
|
||||
(moderator) => moderator.id !== channel.ownerId && !managerIds.has(moderator.id)
|
||||
);
|
||||
const chatModeratorPersonalChannels = await Promise.all(
|
||||
extraChatModerators.map((moderator) => resolvePersonalChannel(moderator.id))
|
||||
);
|
||||
const followerPersonalChannels = await Promise.all(
|
||||
channel.followers.map((follower) => resolvePersonalChannel(follower.user.id))
|
||||
);
|
||||
const teamMemberIds = [channel.ownerId, ...channel.managers.map((manager) => manager.id)];
|
||||
const teamBotAccounts = await prisma.botAccount.findMany({
|
||||
where: {
|
||||
ownerId: {
|
||||
in: teamMemberIds,
|
||||
},
|
||||
},
|
||||
orderBy: {
|
||||
slug: 'asc',
|
||||
},
|
||||
});
|
||||
|
||||
return (
|
||||
<ChannelSettingsClient
|
||||
channel={{
|
||||
...channel,
|
||||
chatModerators: extraChatModerators,
|
||||
ownerPersonalChannel,
|
||||
managerPersonalChannels,
|
||||
chatModeratorPersonalChannels,
|
||||
followerPersonalChannels,
|
||||
teamBotAccounts,
|
||||
}}
|
||||
isOwner={isOwner}
|
||||
currentUser={user}
|
||||
|
||||
@@ -107,7 +107,9 @@ export default function OnboardingClient() {
|
||||
name: 'username',
|
||||
label: 'Channel Username',
|
||||
type: 'text',
|
||||
placeholder: 'e.g., yourname or yourname-codes'
|
||||
placeholder: 'e.g., yourname or yourname-codes',
|
||||
maxChars: 20,
|
||||
inputFilter: /[^a-z0-9_-]/g,
|
||||
},
|
||||
]}
|
||||
schemaName="onboard"
|
||||
@@ -119,7 +121,7 @@ export default function OnboardingClient() {
|
||||
<div className="mt-4 p-3 bg-muted/30 rounded-md">
|
||||
<p className="text-xs text-muted-foreground">
|
||||
<strong>Username rules:</strong> Only lowercase letters (a-z), numbers (0-9),
|
||||
underscores (_), and dashes (-) are allowed. Must be unique across the platform.
|
||||
underscores (_), and dashes (-) are allowed. Up to 20 characters. Must be unique across the platform.
|
||||
</p>
|
||||
</div>
|
||||
</CardContent>
|
||||
|
||||
@@ -4,7 +4,10 @@ import OnboardingClient from "./page.client";
|
||||
|
||||
export default async function Page() {
|
||||
const { user } = await validateRequest();
|
||||
if (user!.hasOnboarded) {
|
||||
if (!user) {
|
||||
return redirect('/');
|
||||
}
|
||||
if (user.hasOnboarded) {
|
||||
return redirect('/');
|
||||
}
|
||||
return <OnboardingClient />;
|
||||
|
||||
@@ -1,10 +1,8 @@
|
||||
import LandingPage from '@/components/app/LandingPage/LandingPage';
|
||||
import { Card, CardContent } from '@/components/ui/card';
|
||||
import StreamGrid from '@/components/app/StreamGrid/StreamGrid';
|
||||
import ConfusedDino from '@/components/ui/confuseddino';
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { prisma } from '@hctv/db';
|
||||
import { Avatar, AvatarImage, AvatarFallback } from '@radix-ui/react-avatar';
|
||||
import Image from 'next/image';
|
||||
import Link from 'next/link';
|
||||
import { redirect } from 'next/navigation';
|
||||
|
||||
@@ -13,67 +11,45 @@ export default async function Home() {
|
||||
if (user && !user?.hasOnboarded) {
|
||||
redirect('/onboarding');
|
||||
}
|
||||
const streams = await prisma.streamInfo.findMany({
|
||||
where: {
|
||||
isLive: true,
|
||||
},
|
||||
include: {
|
||||
channel: true,
|
||||
},
|
||||
});
|
||||
|
||||
const [liveStreams, offlineStreams] = await Promise.all([
|
||||
prisma.streamInfo.findMany({
|
||||
where: { isLive: true },
|
||||
include: { channel: true },
|
||||
}),
|
||||
prisma.streamInfo.findMany({
|
||||
where: { isLive: false },
|
||||
include: { channel: true },
|
||||
}),
|
||||
]);
|
||||
|
||||
if (!user) {
|
||||
return <LandingPage />;
|
||||
}
|
||||
if (!streams.length) {
|
||||
|
||||
if (!liveStreams.length && !offlineStreams.length) {
|
||||
return (
|
||||
<div className="flex justify-center items-center text-center flex-col pt-4 gap-2">
|
||||
<h2>No streams found!</h2>
|
||||
<p>...maybe start one?</p>
|
||||
<ConfusedDino className='w-40 h-40' />
|
||||
<div className="flex min-h-[60vh] flex-col items-center justify-center gap-5 px-4 text-center">
|
||||
<ConfusedDino className="h-28 w-28 opacity-80" />
|
||||
<div className="space-y-1.5">
|
||||
<h2 className="pb-0 text-2xl font-semibold tracking-tight">Nothing live right now</h2>
|
||||
<p className="text-sm text-muted-foreground">
|
||||
Nobody's streaming yet — why not be the first?
|
||||
</p>
|
||||
</div>
|
||||
<Link
|
||||
href="/settings/channel"
|
||||
className="inline-flex h-9 items-center justify-center rounded-md bg-primary px-5 text-sm font-medium text-primary-foreground shadow transition-colors hover:bg-primary/90 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring"
|
||||
>
|
||||
Start streaming
|
||||
</Link>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="p-4">
|
||||
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6">
|
||||
{streams.map((stream) => (
|
||||
<Link href={`/${stream.username}`} key={stream.id}>
|
||||
<Card className="overflow-hidden hover:shadow-lg transition-shadow">
|
||||
<CardContent className="p-0">
|
||||
<div className="relative">
|
||||
{/* eslint-disable-next-line @next/next/no-img-element */}
|
||||
<img
|
||||
src={`/api/stream/thumb/${stream.channel.name}`}
|
||||
width={512}
|
||||
height={512}
|
||||
alt={stream.title}
|
||||
className="w-full h-48 object-cover"
|
||||
/>
|
||||
<div className="absolute bottom-2 left-2 bg-red-600 text-white text-xs font-bold px-2 py-1 rounded">
|
||||
LIVE
|
||||
</div>
|
||||
<div className="absolute bottom-2 right-2 bg-black bg-opacity-70 text-white text-xs px-2 py-1 rounded">
|
||||
{stream.viewers} viewers
|
||||
</div>
|
||||
</div>
|
||||
<div className="p-4">
|
||||
<div className="flex items-start">
|
||||
<Avatar className="h-10 w-10 mr-3">
|
||||
<AvatarImage src={stream.channel.pfpUrl} />
|
||||
<AvatarFallback>{stream.channel.name}</AvatarFallback>
|
||||
</Avatar>
|
||||
<div>
|
||||
<h3 className="font-semibold line-clamp-1">{stream.title}</h3>
|
||||
<p className="text-sm text-muted-foreground">{stream.category}</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</CardContent>
|
||||
</Card>
|
||||
</Link>
|
||||
))}
|
||||
</div>
|
||||
<div className="p-3 md:p-6">
|
||||
<StreamGrid liveStreams={liveStreams} offlineStreams={offlineStreams} />
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -2,8 +2,22 @@ import { NuqsAdapter } from 'nuqs/adapters/next/app';
|
||||
import './globals.css';
|
||||
|
||||
export default function Layout({ children }: { children: React.ReactNode }) {
|
||||
const publicEnv = Object.keys(process.env).reduce((acc, key) => {
|
||||
if (key.startsWith('NEXT_PUBLIC_')) {
|
||||
acc[key] = process.env[key];
|
||||
}
|
||||
return acc;
|
||||
}, {} as Record<string, string | undefined>);
|
||||
|
||||
return (
|
||||
<html lang="en">
|
||||
<head>
|
||||
<script
|
||||
dangerouslySetInnerHTML={{
|
||||
__html: `window.__ENV = ${JSON.stringify(publicEnv)}`,
|
||||
}}
|
||||
/>
|
||||
</head>
|
||||
<NuqsAdapter>
|
||||
<body>
|
||||
<main>{children}</main>
|
||||
|
||||
@@ -9,9 +9,11 @@ import { Message } from './message';
|
||||
import { useMap } from '@uidotdev/usehooks';
|
||||
import { EmojiSearch } from './EmojiSearch';
|
||||
import { useQueryState } from 'nuqs';
|
||||
import { toast } from 'sonner';
|
||||
|
||||
export default function ChatPanel(props: Props) {
|
||||
const { username } = useParams();
|
||||
const channelName = (Array.isArray(username) ? username[0] : username) ?? '';
|
||||
const [grant, setGrant] = useQueryState('grant');
|
||||
const [message, setMessage] = useState('');
|
||||
const [chatMessages, setChatMessages] = useState<ChatMessage[]>([]);
|
||||
@@ -21,13 +23,19 @@ export default function ChatPanel(props: Props) {
|
||||
const [emojisToReq, setEmojisToReq] = useState<string[]>([]);
|
||||
const [cursorPosition, setCursorPosition] = useState(0);
|
||||
const textareaRef = useRef<HTMLTextAreaElement | null>(null);
|
||||
const [viewer, setViewer] = useState<{ id: string; username: string } | null>(null);
|
||||
const [canModerate, setCanModerate] = useState(false);
|
||||
const [chatAccess, setChatAccess] = useState<ChatAccessState>({
|
||||
canSend: true,
|
||||
restriction: null,
|
||||
});
|
||||
|
||||
useEffect(() => {
|
||||
console.log('Initializing WebSocket connection for user:', username);
|
||||
const socket = new WebSocket(
|
||||
`ws${window.location.protocol === 'https:' ? 's' : ''}://${
|
||||
window.location.host
|
||||
}/api/stream/chat/ws/${username}?grant=${grant}`
|
||||
}/api/stream/chat/ws/${channelName}?grant=${grant}`
|
||||
);
|
||||
socketRef.current = socket;
|
||||
|
||||
@@ -50,6 +58,35 @@ export default function ChatPanel(props: Props) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.type === 'session') {
|
||||
setViewer(data.viewer ?? null);
|
||||
setCanModerate(Boolean(data.permissions?.canModerate));
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.type === 'chatAccess') {
|
||||
setChatAccess({
|
||||
canSend: Boolean(data.canSend),
|
||||
restriction: data.restriction ?? null,
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.type === 'systemMsg') {
|
||||
setChatMessages((prev) => [...prev, { message: data.message, type: 'systemMsg' }]);
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.type === 'messageDeleted') {
|
||||
setChatMessages((prev) => prev.filter((message) => message.msgId !== data.msgId));
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.type === 'moderationError') {
|
||||
toast.error(data.message || 'Message blocked by moderation rules.');
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.type === 'message') {
|
||||
console.log('Adding new chat message:', data);
|
||||
setChatMessages((prev) => [...prev, data]);
|
||||
@@ -72,7 +109,7 @@ export default function ChatPanel(props: Props) {
|
||||
return () => {
|
||||
socket.close();
|
||||
};
|
||||
}, [username]);
|
||||
}, [channelName]);
|
||||
|
||||
useEffect(() => {
|
||||
if (scrollRef.current) {
|
||||
@@ -84,6 +121,14 @@ export default function ChatPanel(props: Props) {
|
||||
}, [chatMessages]);
|
||||
|
||||
const sendMessage = () => {
|
||||
if (!chatAccess.canSend) {
|
||||
toast.error(
|
||||
chatAccess.restriction?.type === 'timeout'
|
||||
? 'You are currently timed out in this chat.'
|
||||
: 'You are currently banned from this chat.'
|
||||
);
|
||||
return;
|
||||
}
|
||||
if (!message.trim()) return;
|
||||
|
||||
if (socketRef.current && socketRef.current.readyState === WebSocket.OPEN) {
|
||||
@@ -93,7 +138,7 @@ export default function ChatPanel(props: Props) {
|
||||
const socket = new WebSocket(
|
||||
`ws${window.location.protocol === 'https:' ? 's' : ''}://${
|
||||
window.location.host
|
||||
}/api/stream/chat/ws/${username}?grant=${grant}`
|
||||
}/api/stream/chat/ws/${channelName}?grant=${grant}`
|
||||
);
|
||||
socket.onopen = () => {
|
||||
socket.send(JSON.stringify({ type: 'message', message }));
|
||||
@@ -102,6 +147,15 @@ export default function ChatPanel(props: Props) {
|
||||
}
|
||||
};
|
||||
|
||||
const sendModerationCommand = (command: ChatModerationCommand) => {
|
||||
if (!socketRef.current || socketRef.current.readyState !== WebSocket.OPEN) {
|
||||
toast.error('Chat connection is offline.');
|
||||
return;
|
||||
}
|
||||
|
||||
socketRef.current.send(JSON.stringify(command));
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
const interval = setInterval(() => {
|
||||
if (socketRef.current && socketRef.current.readyState === WebSocket.OPEN) {
|
||||
@@ -146,7 +200,7 @@ export default function ChatPanel(props: Props) {
|
||||
const socket = new WebSocket(
|
||||
`ws${window.location.protocol === 'https:' ? 's' : ''}://${
|
||||
window.location.host
|
||||
}/api/stream/chat/ws/${username}?grant=${grant}`
|
||||
}/api/stream/chat/ws/${channelName}?grant=${grant}`
|
||||
);
|
||||
|
||||
socket.onopen = () => {
|
||||
@@ -209,7 +263,7 @@ export default function ChatPanel(props: Props) {
|
||||
setEmojisToReq([]);
|
||||
};
|
||||
}
|
||||
}, [emojisToReq, emojiMap, username]);
|
||||
}, [emojisToReq, emojiMap, channelName]);
|
||||
|
||||
const handleEmojiSelect = (emojiName: string) => {
|
||||
if (!textareaRef.current) return;
|
||||
@@ -245,9 +299,14 @@ export default function ChatPanel(props: Props) {
|
||||
};
|
||||
|
||||
return (
|
||||
<div className={`${props.isObsPanel ? 'w-full text-white' : 'md:border bg-mantle w-[350px] max-w-[350px]'} flex flex-col h-full`}>
|
||||
<div ref={scrollRef} className={`flex-1 p-4 ${props.isObsPanel ? 'scrollbar-hide' : ''} overflow-y-auto flex flex-col`}>
|
||||
<div className="space-y-4 flex-1">
|
||||
<div
|
||||
className={`${props.isObsPanel ? 'w-full text-white' : 'md:border-l border-border bg-mantle w-[350px] max-w-[350px]'} flex flex-col h-full`}
|
||||
>
|
||||
<div
|
||||
ref={scrollRef}
|
||||
className={`flex-1 px-4 py-2 ${props.isObsPanel ? 'scrollbar-hide' : 'scrollbar-thin scrollbar-thumb-border scrollbar-track-transparent'} overflow-y-auto overflow-x-hidden`}
|
||||
>
|
||||
<div className="space-y-1 min-h-full flex flex-col justify-end">
|
||||
{chatMessages.map((msg, i) => (
|
||||
<Message
|
||||
key={i}
|
||||
@@ -255,13 +314,25 @@ export default function ChatPanel(props: Props) {
|
||||
message={msg.message}
|
||||
type={msg.type}
|
||||
emojiMap={emojiMap}
|
||||
msgId={msg.msgId}
|
||||
canModerate={canModerate && Boolean(viewer?.id)}
|
||||
viewerId={viewer?.id}
|
||||
channelName={channelName}
|
||||
onModerationCommand={sendModerationCommand}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
{!props.isObsPanel && (
|
||||
<div className="p-4 border-t relative">
|
||||
<div className="flex space-x-2">
|
||||
<div className="p-3 border-t border-border relative">
|
||||
{!chatAccess.canSend && (
|
||||
<p className="mb-2 text-xs text-destructive">
|
||||
{chatAccess.restriction?.type === 'timeout'
|
||||
? `Timed out${chatAccess.restriction.expiresAt ? ` until ${new Date(chatAccess.restriction.expiresAt).toLocaleTimeString()}` : ''}.`
|
||||
: 'You are banned from sending messages in this chat.'}
|
||||
</p>
|
||||
)}
|
||||
<div className="flex gap-2">
|
||||
<Textarea
|
||||
ref={textareaRef}
|
||||
value={message}
|
||||
@@ -281,11 +352,17 @@ export default function ChatPanel(props: Props) {
|
||||
onClick={(e) => {
|
||||
setCursorPosition(e.currentTarget.selectionStart || 0);
|
||||
}}
|
||||
placeholder="Type a message"
|
||||
className="flex-1 bg-transparent focus-visible:ring-offset-0 min-h-[40px] max-h-[120px] resize-none py-2"
|
||||
placeholder="Send a message..."
|
||||
className="flex-1 bg-background/50 border-border focus-visible:ring-1 focus-visible:ring-primary focus-visible:ring-offset-0 min-h-[40px] max-h-[100px] resize-none py-2 text-sm"
|
||||
rows={1}
|
||||
disabled={!chatAccess.canSend}
|
||||
/>
|
||||
<Button size="icon" className="text-black transition-colors" onClick={sendMessage}>
|
||||
<Button
|
||||
size="icon"
|
||||
className="shrink-0 transition-colors"
|
||||
onClick={sendMessage}
|
||||
disabled={!message.trim() || !chatAccess.canSend}
|
||||
>
|
||||
<Send className="h-4 w-4" />
|
||||
</Button>
|
||||
</div>
|
||||
@@ -307,6 +384,32 @@ export interface ChatMessage {
|
||||
user?: User;
|
||||
message: string;
|
||||
type: 'message' | 'systemMsg';
|
||||
msgId?: string;
|
||||
}
|
||||
|
||||
export interface ChatModerationCommand {
|
||||
type:
|
||||
| 'mod:deleteMessage'
|
||||
| 'mod:timeoutUser'
|
||||
| 'mod:banUser'
|
||||
| 'mod:unbanUser'
|
||||
| 'mod:liftTimeout';
|
||||
msgId?: string;
|
||||
targetUserId?: string;
|
||||
targetUsername?: string;
|
||||
durationSeconds?: number;
|
||||
reason?: string;
|
||||
}
|
||||
|
||||
interface ChatAccessState {
|
||||
canSend: boolean;
|
||||
restriction: ChatRestriction | null;
|
||||
}
|
||||
|
||||
interface ChatRestriction {
|
||||
type: 'timeout' | 'ban';
|
||||
reason?: string;
|
||||
expiresAt?: string | null;
|
||||
}
|
||||
|
||||
export interface User {
|
||||
@@ -315,6 +418,8 @@ export interface User {
|
||||
pfpUrl: string;
|
||||
isBot: boolean;
|
||||
displayName?: string;
|
||||
isPlatformAdmin?: boolean;
|
||||
channelRole?: 'owner' | 'manager' | 'chatModerator' | 'botModerator' | null;
|
||||
}
|
||||
|
||||
interface Props {
|
||||
|
||||
@@ -28,25 +28,27 @@ export function EmojiSearch({
|
||||
useEffect(() => {
|
||||
const beforeCursor = message.substring(0, cursorPosition);
|
||||
const match = beforeCursor.match(/:[\w\-+]*$/);
|
||||
|
||||
|
||||
if (match) {
|
||||
const term = match[0].substring(1);
|
||||
setSearchTerm(term);
|
||||
|
||||
|
||||
if (term.length > 0) {
|
||||
const localResults = Array.from(emojiMap.keys())
|
||||
.filter(name => name.toLowerCase().includes(term.toLowerCase()))
|
||||
.filter((name) => name.toLowerCase().includes(term.toLowerCase()))
|
||||
.slice(0, 5);
|
||||
|
||||
|
||||
if (localResults.length > 0) {
|
||||
setSearchResults(localResults);
|
||||
}
|
||||
|
||||
|
||||
if (socket && socket.readyState === WebSocket.OPEN) {
|
||||
socket.send(JSON.stringify({
|
||||
type: 'emojiSearch',
|
||||
searchTerm: term
|
||||
}));
|
||||
socket.send(
|
||||
JSON.stringify({
|
||||
type: 'emojiSearch',
|
||||
searchTerm: term,
|
||||
})
|
||||
);
|
||||
}
|
||||
} else {
|
||||
setSearchResults([]);
|
||||
@@ -63,22 +65,22 @@ export function EmojiSearch({
|
||||
const handleEmojiSearchResponse = (event: MessageEvent) => {
|
||||
try {
|
||||
const data = JSON.parse(event.data);
|
||||
|
||||
|
||||
if (data.type === 'emojiSearchResponse') {
|
||||
const serverResults = data.results || [];
|
||||
|
||||
|
||||
const localResults = Array.from(emojiMap.keys())
|
||||
.filter(name => searchTerm && name.toLowerCase().includes(searchTerm.toLowerCase()))
|
||||
.filter((name) => searchTerm && name.toLowerCase().includes(searchTerm.toLowerCase()))
|
||||
.slice(0, 5);
|
||||
|
||||
|
||||
const combinedResults = [...serverResults];
|
||||
|
||||
localResults.forEach(name => {
|
||||
|
||||
localResults.forEach((name) => {
|
||||
if (!combinedResults.includes(name)) {
|
||||
combinedResults.push(name);
|
||||
}
|
||||
});
|
||||
|
||||
|
||||
setSearchResults(combinedResults.slice(0, 10));
|
||||
setSelectedIndex(0);
|
||||
}
|
||||
@@ -95,18 +97,18 @@ export function EmojiSearch({
|
||||
|
||||
useEffect(() => {
|
||||
if (!textareaRef.current) return;
|
||||
|
||||
|
||||
const handleKeyDown = (e: KeyboardEvent) => {
|
||||
if (!searchTerm || searchResults.length === 0) return;
|
||||
|
||||
|
||||
switch (e.key) {
|
||||
case 'ArrowDown':
|
||||
e.preventDefault();
|
||||
setSelectedIndex(prev => (prev + 1) % searchResults.length);
|
||||
setSelectedIndex((prev) => (prev + 1) % searchResults.length);
|
||||
break;
|
||||
case 'ArrowUp':
|
||||
e.preventDefault();
|
||||
setSelectedIndex(prev => (prev - 1 + searchResults.length) % searchResults.length);
|
||||
setSelectedIndex((prev) => (prev - 1 + searchResults.length) % searchResults.length);
|
||||
break;
|
||||
case 'Enter':
|
||||
if (searchResults[selectedIndex]) {
|
||||
@@ -127,10 +129,10 @@ export function EmojiSearch({
|
||||
break;
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
const textarea = textareaRef.current;
|
||||
textarea.addEventListener('keydown', handleKeyDown);
|
||||
|
||||
|
||||
return () => {
|
||||
textarea.removeEventListener('keydown', handleKeyDown);
|
||||
};
|
||||
@@ -150,25 +152,25 @@ export function EmojiSearch({
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="absolute bottom-16 left-4 bg-mantle border rounded-md shadow-lg max-h-60 overflow-y-auto z-10 min-w-[200px] max-w-[300px]">
|
||||
<div className="absolute bottom-full left-0 right-0 mb-2 mx-0 bg-mantle border border-border rounded-lg shadow-lg max-h-60 overflow-y-auto z-10">
|
||||
<div ref={resultsRef} className="py-1">
|
||||
{searchResults.map((emojiName, index) => {
|
||||
const isSelected = index === selectedIndex;
|
||||
const emojiUrl = emojiMap.get(emojiName);
|
||||
|
||||
|
||||
return (
|
||||
<div
|
||||
key={emojiName}
|
||||
className={`px-3 py-1.5 flex items-center gap-2 cursor-pointer ${
|
||||
className={`px-3 py-2 flex items-center gap-3 cursor-pointer transition-colors ${
|
||||
isSelected ? 'bg-primary/10' : 'hover:bg-primary/5'
|
||||
}`}
|
||||
onClick={() => onSelect(emojiName)}
|
||||
>
|
||||
{emojiUrl && (
|
||||
<Image src={emojiUrl} alt={emojiName} width={20} height={20} className="w-5 h-5" />
|
||||
<Image src={emojiUrl} alt={emojiName} width={24} height={24} className="w-6 h-6" />
|
||||
)}
|
||||
<span className="flex-grow text-sm">{emojiName}</span>
|
||||
{isSelected && <Check className="h-4 w-4 text-blue-500" />}
|
||||
<span className="flex-grow text-sm font-medium">{emojiName}</span>
|
||||
{isSelected && <Check className="h-4 w-4 text-primary" />}
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
|
||||
@@ -1,80 +1,373 @@
|
||||
import { User } from './ChatPanel';
|
||||
import React from 'react';
|
||||
'use client';
|
||||
|
||||
import { ChatModerationCommand, User } from './ChatPanel';
|
||||
import { useState } from 'react';
|
||||
import Image from 'next/image';
|
||||
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip';
|
||||
import { Bot } from 'lucide-react';
|
||||
import {
|
||||
Ban,
|
||||
Bot,
|
||||
Clock3,
|
||||
Crown,
|
||||
EllipsisVertical,
|
||||
Eraser,
|
||||
Flag,
|
||||
Shield,
|
||||
ShieldAlert,
|
||||
ShieldCheck,
|
||||
UserRoundCheck,
|
||||
Wrench,
|
||||
} from 'lucide-react';
|
||||
import type { LucideIcon } from 'lucide-react';
|
||||
import { Button } from '@/components/ui/button';
|
||||
import {
|
||||
DropdownMenu,
|
||||
DropdownMenuContent,
|
||||
DropdownMenuItem,
|
||||
DropdownMenuTrigger,
|
||||
} from '@/components/ui/dropdown-menu';
|
||||
import {
|
||||
Dialog,
|
||||
DialogContent,
|
||||
DialogDescription,
|
||||
DialogFooter,
|
||||
DialogHeader,
|
||||
DialogTitle,
|
||||
} from '@/components/ui/dialog';
|
||||
import { Textarea } from '@/components/ui/textarea';
|
||||
import { toast } from 'sonner';
|
||||
import { cn } from '@/lib/utils';
|
||||
|
||||
type ChannelRole = NonNullable<User['channelRole']>;
|
||||
|
||||
const ROLE_META: Record<ChannelRole, { label: string; icon: LucideIcon; className: string }> = {
|
||||
owner: { label: 'Owner', icon: Crown, className: 'text-amber-500' },
|
||||
manager: { label: 'Manager', icon: Wrench, className: 'text-violet-500' },
|
||||
chatModerator: { label: 'Chat Mod', icon: Shield, className: 'text-emerald-500' },
|
||||
botModerator: { label: 'Bot Mod', icon: ShieldCheck, className: 'text-cyan-500' },
|
||||
};
|
||||
|
||||
function TooltipIcon({
|
||||
icon: Icon,
|
||||
label,
|
||||
className,
|
||||
}: {
|
||||
icon: LucideIcon;
|
||||
label: string;
|
||||
className?: string;
|
||||
}) {
|
||||
return (
|
||||
<Tooltip delayDuration={200}>
|
||||
<TooltipTrigger asChild>
|
||||
<Icon className={cn('size-3.5 shrink-0', className)} />
|
||||
</TooltipTrigger>
|
||||
<TooltipContent side="top">{label}</TooltipContent>
|
||||
</Tooltip>
|
||||
);
|
||||
}
|
||||
|
||||
function UsernameRow({ user, displayName }: { user?: User; displayName?: string }) {
|
||||
const role = user?.channelRole ? ROLE_META[user.channelRole] : null;
|
||||
|
||||
return (
|
||||
<TooltipProvider>
|
||||
<span className="font-semibold text-primary shrink-0 flex items-center gap-1">
|
||||
{user?.isBot && <TooltipIcon icon={Bot} label="Bot" className="text-muted-foreground" />}
|
||||
{role && <TooltipIcon icon={role.icon} label={role.label} className={role.className} />}
|
||||
{user?.isPlatformAdmin && (
|
||||
<TooltipIcon icon={ShieldAlert} label="Platform Admin" className="text-destructive" />
|
||||
)}
|
||||
<span>{displayName}</span>
|
||||
<span className="font-normal text-muted-foreground select-none">:</span>
|
||||
</span>
|
||||
</TooltipProvider>
|
||||
);
|
||||
}
|
||||
|
||||
function ReportDialog({
|
||||
open,
|
||||
onOpenChange,
|
||||
displayName,
|
||||
message,
|
||||
reportReason,
|
||||
onReasonChange,
|
||||
onSubmit,
|
||||
isSubmitting,
|
||||
}: {
|
||||
open: boolean;
|
||||
onOpenChange: (open: boolean) => void;
|
||||
displayName?: string;
|
||||
message: string;
|
||||
reportReason: string;
|
||||
onReasonChange: (value: string) => void;
|
||||
onSubmit: () => void;
|
||||
isSubmitting: boolean;
|
||||
}) {
|
||||
return (
|
||||
<Dialog open={open} onOpenChange={onOpenChange}>
|
||||
<DialogContent>
|
||||
<DialogHeader>
|
||||
<DialogTitle>Report message</DialogTitle>
|
||||
<DialogDescription>
|
||||
Message against Hack Club's Code of Conduct? Let us know!
|
||||
</DialogDescription>
|
||||
</DialogHeader>
|
||||
<div className="space-y-3">
|
||||
<div className="text-sm text-muted-foreground rounded-md border p-3 bg-muted/30">
|
||||
<p className="font-medium text-foreground mb-1">Reported user</p>
|
||||
<p>{displayName}</p>
|
||||
<p className="mt-2">{message}</p>
|
||||
</div>
|
||||
<div>
|
||||
<label className="text-sm font-medium">Reason</label>
|
||||
<Textarea
|
||||
value={reportReason}
|
||||
onChange={(e) => onReasonChange(e.target.value)}
|
||||
placeholder="Describe why this should be reviewed (harassment, hate speech, spam, threats, etc)."
|
||||
rows={5}
|
||||
className="mt-2"
|
||||
/>
|
||||
<p className="text-xs text-muted-foreground mt-1">Minimum 10 characters.</p>
|
||||
</div>
|
||||
</div>
|
||||
<DialogFooter>
|
||||
<Button variant="outline" onClick={() => onOpenChange(false)} disabled={isSubmitting}>
|
||||
Cancel
|
||||
</Button>
|
||||
<Button onClick={onSubmit} disabled={isSubmitting || reportReason.trim().length < 10}>
|
||||
Submit report
|
||||
</Button>
|
||||
</DialogFooter>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
);
|
||||
}
|
||||
|
||||
export function Message({
|
||||
user,
|
||||
message,
|
||||
type,
|
||||
emojiMap,
|
||||
msgId,
|
||||
canModerate,
|
||||
viewerId,
|
||||
channelName,
|
||||
onModerationCommand,
|
||||
}: MessageProps) {
|
||||
const [reportOpen, setReportOpen] = useState(false);
|
||||
const [reportReason, setReportReason] = useState('');
|
||||
const [isSubmittingReport, setIsSubmittingReport] = useState(false);
|
||||
const displayName = user?.displayName || user?.username;
|
||||
|
||||
export function Message({ user, message, type, emojiMap }: MessageProps) {
|
||||
if (type === 'systemMsg') {
|
||||
return (
|
||||
<div className="flex items-center justify-center">
|
||||
<span className="text-xs text-muted-foreground">{message}</span>
|
||||
<div className="flex items-center justify-center py-1">
|
||||
<span className="text-xs text-muted-foreground italic">{message}</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="flex">
|
||||
<div lang="en" className="max-w-full break-all whitespace-pre-wrap hyphens-auto">
|
||||
<p className="flex flex-wrap items-center">
|
||||
<span className="font-bold mr-2 flex items-center">
|
||||
{user?.isBot && (
|
||||
<span className="text-xs text-muted-foreground flex mr-1">
|
||||
{' '}
|
||||
<Bot className="size-5" />
|
||||
</span>
|
||||
)}
|
||||
{user?.displayName || user?.username}
|
||||
</span>
|
||||
const submitReport = async () => {
|
||||
if (!user?.id || !viewerId || viewerId === user.id) return;
|
||||
|
||||
<EmojiRenderer text={message} emojiMap={emojiMap} />
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
const reason = reportReason.trim();
|
||||
if (reason.length < 10) {
|
||||
toast.error('Please include at least 10 characters explaining the report.');
|
||||
return;
|
||||
}
|
||||
|
||||
export function EmojiRenderer({ text, emojiMap }: EmojiRendererProps) {
|
||||
if (!text) return null;
|
||||
setIsSubmittingReport(true);
|
||||
try {
|
||||
const res = await fetch('/api/stream/chat/report', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({
|
||||
channelName,
|
||||
targetUserId: user.id,
|
||||
targetUsername: displayName,
|
||||
msgId,
|
||||
message,
|
||||
reason,
|
||||
}),
|
||||
});
|
||||
|
||||
const parts = text.split(/(:[\w\-+]+:)/g);
|
||||
if (!res.ok) {
|
||||
toast.error((await res.text()) || 'Failed to submit report.');
|
||||
return;
|
||||
}
|
||||
|
||||
toast.success('Report submitted. Thanks for helping keep chat safe.');
|
||||
setReportReason('');
|
||||
setReportOpen(false);
|
||||
} catch {
|
||||
toast.error('Failed to submit report.');
|
||||
} finally {
|
||||
setIsSubmittingReport(false);
|
||||
}
|
||||
};
|
||||
|
||||
const handleReportOpenChange = (open: boolean) => {
|
||||
setReportOpen(open);
|
||||
if (!open) setReportReason('');
|
||||
};
|
||||
|
||||
return (
|
||||
<>
|
||||
{parts.map((part, index) => {
|
||||
if (part.match(/^:[\w\-+]+:$/)) {
|
||||
const emojiName = part.replaceAll(':', '');
|
||||
const emojiUrl = emojiMap.get(emojiName);
|
||||
<div className="group hover:bg-primary/5 rounded px-2 py-1 -mx-2 transition-colors">
|
||||
<div className="flex items-start gap-1.5">
|
||||
<UsernameRow user={user} displayName={displayName} />
|
||||
<span
|
||||
lang="en"
|
||||
className="text-foreground min-w-0 flex-1"
|
||||
style={{ overflowWrap: 'anywhere', wordBreak: 'break-word' }}
|
||||
>
|
||||
<EmojiRenderer text={message} emojiMap={emojiMap} />
|
||||
</span>
|
||||
{type === 'message' && user?.id && (
|
||||
<MessageActionsMenu
|
||||
user={user}
|
||||
msgId={msgId}
|
||||
canModerate={canModerate}
|
||||
viewerId={viewerId}
|
||||
onModerationCommand={onModerationCommand}
|
||||
onOpenReport={() => setReportOpen(true)}
|
||||
/>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
<ReportDialog
|
||||
open={reportOpen}
|
||||
onOpenChange={handleReportOpenChange}
|
||||
displayName={displayName}
|
||||
message={message}
|
||||
reportReason={reportReason}
|
||||
onReasonChange={setReportReason}
|
||||
onSubmit={submitReport}
|
||||
isSubmitting={isSubmittingReport}
|
||||
/>
|
||||
</>
|
||||
);
|
||||
}
|
||||
|
||||
if (emojiUrl) {
|
||||
return (
|
||||
<TooltipProvider key={index}>
|
||||
<Tooltip delayDuration={250}>
|
||||
<TooltipTrigger>
|
||||
<span
|
||||
key={index}
|
||||
className="inline-block align-middle"
|
||||
style={{ height: '1.2em' }}
|
||||
>
|
||||
<Image
|
||||
src={emojiUrl}
|
||||
alt={part}
|
||||
width={20}
|
||||
height={20}
|
||||
className="inline-block"
|
||||
/>
|
||||
function MessageActionsMenu({
|
||||
user,
|
||||
msgId,
|
||||
canModerate,
|
||||
viewerId,
|
||||
onModerationCommand,
|
||||
onOpenReport,
|
||||
}: {
|
||||
user: User;
|
||||
msgId?: string;
|
||||
canModerate?: boolean;
|
||||
viewerId?: string;
|
||||
onModerationCommand?: (command: ChatModerationCommand) => void;
|
||||
onOpenReport: () => void;
|
||||
}) {
|
||||
if (!viewerId || !user.id || user.id === viewerId) return null;
|
||||
|
||||
const displayName = user.displayName || user.username;
|
||||
const canMod = Boolean(canModerate && onModerationCommand);
|
||||
|
||||
const runModeration = (command: ChatModerationCommand) => onModerationCommand?.(command);
|
||||
|
||||
const timeout = (durationSeconds: number) =>
|
||||
runModeration({
|
||||
type: 'mod:timeoutUser',
|
||||
targetUserId: user.id,
|
||||
targetUsername: displayName,
|
||||
durationSeconds,
|
||||
reason: 'Timed out by moderator',
|
||||
});
|
||||
|
||||
return (
|
||||
<DropdownMenu>
|
||||
<DropdownMenuTrigger asChild>
|
||||
<Button variant="ghost" size="icon" className="h-7 w-7 opacity-0 group-hover:opacity-100">
|
||||
<EllipsisVertical className="h-4 w-4" />
|
||||
</Button>
|
||||
</DropdownMenuTrigger>
|
||||
<DropdownMenuContent align="end" className="w-52">
|
||||
<DropdownMenuItem onClick={onOpenReport}>
|
||||
<Flag className="mr-2 h-4 w-4" />
|
||||
Report user
|
||||
</DropdownMenuItem>
|
||||
{canMod && (
|
||||
<>
|
||||
<DropdownMenuItem
|
||||
onClick={() => msgId && runModeration({ type: 'mod:deleteMessage', msgId })}
|
||||
>
|
||||
<Eraser className="mr-2 h-4 w-4" />
|
||||
Delete message
|
||||
</DropdownMenuItem>
|
||||
<DropdownMenuItem onClick={() => timeout(300)}>
|
||||
<Clock3 className="mr-2 h-4 w-4" />
|
||||
Timeout 5 min
|
||||
</DropdownMenuItem>
|
||||
<DropdownMenuItem onClick={() => timeout(3600)}>
|
||||
<Clock3 className="mr-2 h-4 w-4" />
|
||||
Timeout 1 hour
|
||||
</DropdownMenuItem>
|
||||
<DropdownMenuItem
|
||||
className="text-destructive"
|
||||
onClick={() =>
|
||||
runModeration({
|
||||
type: 'mod:banUser',
|
||||
targetUserId: user.id,
|
||||
targetUsername: displayName,
|
||||
reason: 'Banned by moderator',
|
||||
})
|
||||
}
|
||||
>
|
||||
<Ban className="mr-2 h-4 w-4" />
|
||||
Ban user
|
||||
</DropdownMenuItem>
|
||||
<DropdownMenuItem
|
||||
onClick={() =>
|
||||
runModeration({
|
||||
type: 'mod:liftTimeout',
|
||||
targetUserId: user.id,
|
||||
targetUsername: displayName,
|
||||
})
|
||||
}
|
||||
>
|
||||
<UserRoundCheck className="mr-2 h-4 w-4" />
|
||||
Lift timeout/ban
|
||||
</DropdownMenuItem>
|
||||
</>
|
||||
)}
|
||||
</DropdownMenuContent>
|
||||
</DropdownMenu>
|
||||
);
|
||||
}
|
||||
|
||||
export function EmojiRenderer({ text, emojiMap }: { text: string; emojiMap: Map<string, string> }) {
|
||||
if (!text) return null;
|
||||
|
||||
return (
|
||||
<TooltipProvider>
|
||||
<>
|
||||
{text.split(/(:[\w\-+]+:)/g).map((part, i) => {
|
||||
if (part.match(/^:[\w\-+]+:$/)) {
|
||||
const name = part.replaceAll(':', '');
|
||||
const url = emojiMap.get(name);
|
||||
if (url) {
|
||||
return (
|
||||
<Tooltip key={i} delayDuration={250}>
|
||||
<TooltipTrigger asChild>
|
||||
<span className="inline-flex items-center align-middle mx-0.5">
|
||||
<Image src={url} alt={part} width={20} height={20} className="inline-block" />
|
||||
</span>
|
||||
</TooltipTrigger>
|
||||
<TooltipContent>{part}</TooltipContent>
|
||||
</Tooltip>
|
||||
</TooltipProvider>
|
||||
);
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return <span key={index}>{part}</span>;
|
||||
})}
|
||||
</>
|
||||
return part ? <span key={i}>{part}</span> : null;
|
||||
})}
|
||||
</>
|
||||
</TooltipProvider>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -83,9 +376,9 @@ interface MessageProps {
|
||||
message: string;
|
||||
type: 'message' | 'systemMsg';
|
||||
emojiMap: Map<string, string>;
|
||||
}
|
||||
|
||||
interface EmojiRendererProps {
|
||||
text: string;
|
||||
emojiMap: Map<string, string>;
|
||||
msgId?: string;
|
||||
canModerate?: boolean;
|
||||
viewerId?: string;
|
||||
channelName: string;
|
||||
onModerationCommand?: (command: ChatModerationCommand) => void;
|
||||
}
|
||||
|
||||
@@ -8,22 +8,16 @@ export default function LandingPage() {
|
||||
return (
|
||||
<>
|
||||
<main className="flex-1">
|
||||
{/* Hero Section */}
|
||||
<section className="relative w-full py-20 md:py-32 lg:py-40 xl:py-48 overflow-hidden">
|
||||
<div className="absolute inset-0 bg-gradient-to-br from-primary/10 via-transparent to-secondary/10"></div>
|
||||
<div className="absolute inset-0"></div>
|
||||
<div className="container px-4 md:px-6 relative">
|
||||
<div className="flex flex-col items-center space-y-8 text-center">
|
||||
<Badge variant="outline" className="px-4 py-2 text-sm font-medium bg-primary/10 text-primary border-primary/20">
|
||||
<Heart className="w-4 h-4 mr-2" />
|
||||
Made with ❤️ by Hack Clubbers
|
||||
</Badge>
|
||||
|
||||
<div className="flex flex-col items-center space-y-8 text-center">
|
||||
<div className="space-y-6">
|
||||
<h1 className="text-4xl font-bold tracking-tight sm:text-5xl md:text-6xl lg:text-7xl bg-gradient-to-r from-primary via-primary/80 to-primary/60 bg-clip-text text-transparent">
|
||||
<h1 className="text-4xl font-bold tracking-tight sm:text-5xl md:text-6xl lg:text-7xl text-primary">
|
||||
hackclub.tv
|
||||
</h1>
|
||||
<p className="mx-auto max-w-[600px] text-lg text-muted-foreground md:text-xl">
|
||||
The streaming platform where Hack Clubbers share their coding journeys, workshops, and hackathon adventures with the world.
|
||||
The streaming website for Hack Clubbers, by Hack Clubbers.
|
||||
</p>
|
||||
</div>
|
||||
|
||||
@@ -42,133 +36,6 @@ export default function LandingPage() {
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
{/* Features Section */}
|
||||
<section className="w-full py-16 md:py-24 lg:py-32 bg-muted/30" id="features">
|
||||
<div className="container px-4 md:px-6">
|
||||
<div className="flex flex-col items-center text-center space-y-8 mb-16">
|
||||
<Badge variant="secondary" className="px-4 py-2">
|
||||
Platform Features
|
||||
</Badge>
|
||||
<div className="space-y-4">
|
||||
<h2 className="text-3xl font-bold tracking-tight sm:text-4xl md:text-5xl">
|
||||
Built for creators, by creators
|
||||
</h2>
|
||||
<p className="max-w-[700px] text-lg text-muted-foreground">
|
||||
Everything you need to connect, create, and grow within the Hack Club community.
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="grid gap-6 md:grid-cols-2 lg:grid-cols-3 max-w-5xl mx-auto">
|
||||
<Card className="border-0 shadow-sm hover:shadow-md transition-shadow">
|
||||
<CardHeader className="pb-4">
|
||||
<div className="w-12 h-12 bg-primary/10 rounded-lg flex items-center justify-center mb-4">
|
||||
<Zap className="w-6 h-6 text-primary" />
|
||||
</div>
|
||||
<CardTitle className="text-xl">Low-Latency Streaming</CardTitle>
|
||||
</CardHeader>
|
||||
<CardContent>
|
||||
<CardDescription className="text-base">
|
||||
Share your coding sessions with ultra-low latency. Your audience stays engaged with real-time interaction.
|
||||
</CardDescription>
|
||||
</CardContent>
|
||||
</Card>
|
||||
|
||||
<Card className="border-0 shadow-sm hover:shadow-md transition-shadow">
|
||||
<CardHeader className="pb-4">
|
||||
<div className="w-12 h-12 bg-primary/10 rounded-lg flex items-center justify-center mb-4">
|
||||
<MessageCircle className="w-6 h-6 text-primary" />
|
||||
</div>
|
||||
<CardTitle className="text-xl">Real-Time Chat</CardTitle>
|
||||
</CardHeader>
|
||||
<CardContent>
|
||||
<CardDescription className="text-base">
|
||||
Engage with your community through integrated chat. Get instant feedback and build connections.
|
||||
</CardDescription>
|
||||
</CardContent>
|
||||
</Card>
|
||||
|
||||
<Card className="border-0 shadow-sm hover:shadow-md transition-shadow">
|
||||
<CardHeader className="pb-4">
|
||||
<div className="w-12 h-12 bg-primary/10 rounded-lg flex items-center justify-center mb-4">
|
||||
<Users className="w-6 h-6 text-primary" />
|
||||
</div>
|
||||
<CardTitle className="text-xl">Community First</CardTitle>
|
||||
</CardHeader>
|
||||
<CardContent>
|
||||
<CardDescription className="text-base">
|
||||
Follow your favorite streamers, discover new creators, and be part of the vibrant Hack Club ecosystem.
|
||||
</CardDescription>
|
||||
</CardContent>
|
||||
</Card>
|
||||
|
||||
<Card className="border-0 shadow-sm hover:shadow-md transition-shadow">
|
||||
<CardHeader className="pb-4">
|
||||
<div className="w-12 h-12 bg-accent/50 rounded-lg flex items-center justify-center mb-4">
|
||||
<Code className="w-6 h-6 text-accent-foreground" />
|
||||
</div>
|
||||
<CardTitle className="text-xl">Code-Focused</CardTitle>
|
||||
</CardHeader>
|
||||
<CardContent>
|
||||
<CardDescription className="text-base">
|
||||
Built specifically for developers. Perfect for coding sessions, tutorials, and technical workshops.
|
||||
</CardDescription>
|
||||
</CardContent>
|
||||
</Card>
|
||||
|
||||
<Card className="border-0 shadow-sm hover:shadow-md transition-shadow">
|
||||
<CardHeader className="pb-4">
|
||||
<div className="w-12 h-12 bg-secondary/50 rounded-lg flex items-center justify-center mb-4">
|
||||
<Play className="w-6 h-6 text-secondary-foreground" />
|
||||
</div>
|
||||
<CardTitle className="text-xl">Easy to Use</CardTitle>
|
||||
</CardHeader>
|
||||
<CardContent>
|
||||
<CardDescription className="text-base">
|
||||
Simple, intuitive interface. Start streaming in minutes, not hours. Focus on what you love: coding.
|
||||
</CardDescription>
|
||||
</CardContent>
|
||||
</Card>
|
||||
|
||||
<Card className="border-0 shadow-sm hover:shadow-md transition-shadow">
|
||||
<CardHeader className="pb-4">
|
||||
<div className="w-12 h-12 bg-primary/10 rounded-lg flex items-center justify-center mb-4">
|
||||
<Heart className="w-6 h-6 text-primary" />
|
||||
</div>
|
||||
<CardTitle className="text-xl">Open Source</CardTitle>
|
||||
</CardHeader>
|
||||
<CardContent>
|
||||
<CardDescription className="text-base">
|
||||
Transparent, community-driven, and built in the open. Contribute, customize, and make it yours.
|
||||
</CardDescription>
|
||||
</CardContent>
|
||||
</Card>
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
{/* CTA Section */}
|
||||
<section className="w-full py-16 md:py-24">
|
||||
<div className="container px-4 md:px-6">
|
||||
<div className="flex flex-col items-center text-center space-y-8">
|
||||
<div className="space-y-4">
|
||||
<h2 className="text-3xl font-bold tracking-tight sm:text-4xl">
|
||||
Ready to share your journey?
|
||||
</h2>
|
||||
<p className="max-w-[600px] text-lg text-muted-foreground">
|
||||
Join the community of makers, builders, and dreamers. Start streaming your coding adventures today.
|
||||
</p>
|
||||
</div>
|
||||
<Link href="/login">
|
||||
<Button size="lg" className="px-8 py-3 text-lg font-semibold">
|
||||
<Play className="w-5 h-5 mr-2" />
|
||||
Get Started Now
|
||||
</Button>
|
||||
</Link>
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
</main>
|
||||
</>
|
||||
);
|
||||
|
||||
@@ -1,28 +1,77 @@
|
||||
'use client';
|
||||
|
||||
import { useEffect, useState } from 'react';
|
||||
import { format } from 'date-fns';
|
||||
import StreamPlayer from '../StreamPlayer/StreamPlayer';
|
||||
import UserInfoCard from '../UserInfoCard/UserInfoCard';
|
||||
import ChatPanel from '../ChatPanel/ChatPanel';
|
||||
import { Button } from '@/components/ui/button';
|
||||
import type { StreamInfo, Channel } from '@hctv/db';
|
||||
import { useIsMobile } from '@/lib/hooks/useMobile';
|
||||
import { useAllChannels } from '@/lib/hooks/useUserList';
|
||||
import { RefreshCw } from 'lucide-react';
|
||||
|
||||
export default function LiveStream(props: Props) {
|
||||
const isMobile = useIsMobile();
|
||||
|
||||
const { channels, refresh } = useAllChannels(5000);
|
||||
const [isRestricted, setIsRestricted] = useState(false);
|
||||
const [restrictionExpiresAt, setRestrictionExpiresAt] = useState<string | null>(null);
|
||||
const [isRefreshing, setIsRefreshing] = useState(false);
|
||||
|
||||
useEffect(() => {
|
||||
const currentStream = channels.find((s) => s.username === props.username);
|
||||
if (currentStream?.channel?.isRestricted) {
|
||||
setIsRestricted(true);
|
||||
setRestrictionExpiresAt(currentStream.channel.restrictionExpiresAt || null);
|
||||
} else if (isRestricted && currentStream && !currentStream.channel?.isRestricted) {
|
||||
setIsRestricted(false);
|
||||
setRestrictionExpiresAt(null);
|
||||
}
|
||||
}, [channels, props.username, isRestricted]);
|
||||
|
||||
const handleRefresh = async () => {
|
||||
setIsRefreshing(true);
|
||||
try {
|
||||
await refresh();
|
||||
} finally {
|
||||
setIsRefreshing(false);
|
||||
}
|
||||
};
|
||||
|
||||
if (isRestricted) {
|
||||
return (
|
||||
<div className="flex flex-col items-center justify-center h-[calc(100vh-64px)] p-4">
|
||||
<h1 className="text-2xl font-bold text-destructive mb-2">Channel Restricted</h1>
|
||||
<p className="text-muted-foreground text-center max-w-md mb-4">
|
||||
This channel has been restricted by a moderator and is no longer available for viewing.
|
||||
</p>
|
||||
{restrictionExpiresAt && (
|
||||
<p className="text-sm text-muted-foreground mb-4">
|
||||
Restriction lifts: {format(new Date(restrictionExpiresAt), 'PPP p')}
|
||||
</p>
|
||||
)}
|
||||
<Button variant="outline" onClick={handleRefresh} disabled={isRefreshing}>
|
||||
<RefreshCw className={`h-4 w-4 mr-2 ${isRefreshing ? 'animate-spin' : ''}`} />
|
||||
{isRefreshing ? 'Checking...' : 'Check again'}
|
||||
</Button>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div className={`${isMobile ? 'flex flex-col' : 'flex'} h-[calc(100vh-64px)] w-full`}>
|
||||
<div className="flex-1 flex flex-col">
|
||||
<div className="flex-1 flex flex-col min-w-0 overflow-hidden">
|
||||
<StreamPlayer />
|
||||
{isMobile && (
|
||||
<div className="h-[300px]">
|
||||
<div className="flex-1 min-h-[250px] max-h-[400px] border-t border-border">
|
||||
<ChatPanel />
|
||||
</div>
|
||||
)}
|
||||
<UserInfoCard streamInfo={props.streamInfo} />
|
||||
</div>
|
||||
|
||||
|
||||
{!isMobile && (
|
||||
<div>
|
||||
<div className="h-full shrink-0">
|
||||
<ChatPanel />
|
||||
</div>
|
||||
)}
|
||||
@@ -33,4 +82,4 @@ export default function LiveStream(props: Props) {
|
||||
interface Props {
|
||||
username: string;
|
||||
streamInfo: StreamInfo & { channel: Channel };
|
||||
}
|
||||
}
|
||||
|
||||
@@ -15,7 +15,7 @@ import { logout } from '@/lib/auth/actions';
|
||||
import { useSession } from '@/lib/providers/SessionProvider';
|
||||
import Link from 'next/link';
|
||||
import { ThemeSwitcher } from '../ThemeSwitcher/ThemeSwitcher';
|
||||
import { IdCard, Slack } from 'lucide-react';
|
||||
import { IdCard, Shield } from 'lucide-react';
|
||||
import { SidebarTrigger } from '@/components/ui/sidebar';
|
||||
|
||||
export default function Navbar(props: Props) {
|
||||
@@ -57,6 +57,17 @@ export default function Navbar(props: Props) {
|
||||
<Link href={`/settings/bot`}>
|
||||
<DropdownMenuItem className="cursor-pointer">Bot accounts</DropdownMenuItem>
|
||||
</Link>
|
||||
{user.isAdmin && (
|
||||
<>
|
||||
<DropdownMenuSeparator />
|
||||
<Link href={`/admin`}>
|
||||
<DropdownMenuItem className="cursor-pointer text-primary">
|
||||
<Shield className="w-4 h-4 mr-2" />
|
||||
Admin Panel
|
||||
</DropdownMenuItem>
|
||||
</Link>
|
||||
</>
|
||||
)}
|
||||
<DropdownMenuSeparator />
|
||||
<Link href={'https://docs.hackclub.tv'} target="_blank" rel="noreferrer">
|
||||
<DropdownMenuItem className="cursor-pointer">API Docs</DropdownMenuItem>
|
||||
|
||||
@@ -1,31 +1,25 @@
|
||||
'use client';
|
||||
|
||||
import * as React from 'react';
|
||||
import { ChevronDown, ChevronUp } from 'lucide-react';
|
||||
import { Avatar, AvatarFallback, AvatarImage } from '@/components/ui/avatar';
|
||||
import {
|
||||
Sidebar as UISidebar,
|
||||
SidebarContent,
|
||||
SidebarGroup,
|
||||
SidebarGroupContent,
|
||||
SidebarGroupLabel,
|
||||
SidebarHeader,
|
||||
SidebarMenu,
|
||||
SidebarMenuButton,
|
||||
SidebarMenuItem,
|
||||
SidebarTrigger,
|
||||
useSidebar,
|
||||
} from '@/components/ui/sidebar';
|
||||
import { StreamInfoResponse, useStreams } from '@/lib/providers/StreamInfoProvider';
|
||||
import { StreamInfoResponse } from '@/lib/providers/StreamInfoProvider';
|
||||
import { useRouter } from 'next/navigation';
|
||||
import { Skeleton } from '@/components/ui/skeleton';
|
||||
import { useAllChannels } from '@/lib/hooks/useUserList';
|
||||
import { cn } from '@/lib/utils';
|
||||
import { Separator } from '@/components/ui/separator';
|
||||
|
||||
export default function Sidebar({ ...props }: React.ComponentProps<typeof UISidebar>) {
|
||||
const { channels: stream, isLoading } = useAllChannels(5000);
|
||||
const [followedExpanded, setFollowedExpanded] = React.useState(true);
|
||||
const { state } = useSidebar();
|
||||
const isCollapsed = state === 'collapsed';
|
||||
|
||||
@@ -95,10 +89,13 @@ function StreamerItem({ streamer, isCollapsed }: { streamer: StreamInfoResponse[
|
||||
>
|
||||
<button className="flex w-full items-center gap-3">
|
||||
<div className="relative flex-shrink-0">
|
||||
<Avatar className="h-8 w-8">
|
||||
<AvatarImage src={streamer.channel.pfpUrl} alt={streamer.username} className="object-cover" />
|
||||
<AvatarFallback>{streamer.username[0]?.toUpperCase()}</AvatarFallback>
|
||||
</Avatar>
|
||||
{/* eslint-disable-next-line @next/next/no-img-element */}
|
||||
<img
|
||||
src={streamer.channel.pfpUrl}
|
||||
alt={streamer.username}
|
||||
className="h-8 w-8 rounded-full object-cover"
|
||||
loading="lazy"
|
||||
/>
|
||||
{streamer.isLive && (
|
||||
<span className="absolute -bottom-0.5 -right-0.5 flex h-3 w-3 items-center justify-center rounded-full bg-background ring-2 ring-background">
|
||||
<span className="h-2 w-2 rounded-full bg-red-500 animate-pulse" />
|
||||
|
||||
178
apps/web/src/components/app/StreamGrid/StreamGrid.tsx
Normal file
178
apps/web/src/components/app/StreamGrid/StreamGrid.tsx
Normal file
@@ -0,0 +1,178 @@
|
||||
'use client';
|
||||
|
||||
import Link from 'next/link';
|
||||
import { Avatar, AvatarFallback, AvatarImage } from '@/components/ui/avatar';
|
||||
import { Badge } from '@/components/ui/badge';
|
||||
import ConfusedDino from '@/components/ui/confuseddino';
|
||||
import {
|
||||
Carousel,
|
||||
CarouselContent,
|
||||
CarouselItem,
|
||||
CarouselNext,
|
||||
CarouselPrevious,
|
||||
} from '@/components/ui/carousel';
|
||||
import type { Channel, StreamInfo } from '@hctv/db';
|
||||
|
||||
type StreamWithChannel = StreamInfo & { channel: Channel };
|
||||
|
||||
interface StreamGridProps {
|
||||
liveStreams: StreamWithChannel[];
|
||||
offlineStreams: StreamWithChannel[];
|
||||
}
|
||||
|
||||
export default function StreamGrid({ liveStreams, offlineStreams }: StreamGridProps) {
|
||||
const sortedLiveStreams = [...liveStreams].sort((a, b) => b.viewers - a.viewers);
|
||||
|
||||
return (
|
||||
<div className="space-y-8 md:space-y-10">
|
||||
{sortedLiveStreams.length === 0 && (
|
||||
<div className="flex flex-col items-center gap-4 py-10 text-center">
|
||||
<ConfusedDino className="h-24 w-24 opacity-70" />
|
||||
<div className="space-y-1">
|
||||
<p className="font-semibold">Nobody's live right now</p>
|
||||
<p className="text-sm text-muted-foreground">Why not be the first?</p>
|
||||
</div>
|
||||
<Link
|
||||
href="/settings/channel"
|
||||
className="inline-flex h-9 items-center justify-center rounded-md bg-primary px-5 text-sm font-medium text-primary-foreground shadow-sm transition-colors hover:bg-primary/90"
|
||||
>
|
||||
Start streaming
|
||||
</Link>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{sortedLiveStreams.length > 0 && (
|
||||
<section>
|
||||
<SectionHeading label="Live now" count={sortedLiveStreams.length} />
|
||||
<div className="grid grid-cols-1 gap-3 sm:grid-cols-2 md:gap-4 lg:grid-cols-3 xl:grid-cols-4">
|
||||
{sortedLiveStreams.map((stream) => (
|
||||
<StreamCard key={stream.id} stream={stream} />
|
||||
))}
|
||||
</div>
|
||||
</section>
|
||||
)}
|
||||
|
||||
{offlineStreams.length > 0 && (
|
||||
<section>
|
||||
<SectionHeading label="Offline channels" count={offlineStreams.length} />
|
||||
<div className="relative">
|
||||
<Carousel opts={{ align: 'start', dragFree: true }}>
|
||||
<CarouselContent>
|
||||
{offlineStreams.map((stream) => (
|
||||
<CarouselItem
|
||||
key={stream.id}
|
||||
className="flex basis-[74px] justify-center sm:basis-[82px] md:basis-[90px] lg:basis-[100px]"
|
||||
>
|
||||
<OfflineCard stream={stream} />
|
||||
</CarouselItem>
|
||||
))}
|
||||
</CarouselContent>
|
||||
<CarouselPrevious className="hidden md:flex" />
|
||||
<CarouselNext className="hidden md:flex" />
|
||||
</Carousel>
|
||||
</div>
|
||||
</section>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
function StreamCard({ stream }: { stream: StreamWithChannel }) {
|
||||
return (
|
||||
<Link href={`/${stream.username}`} className="group block w-full max-w-sm">
|
||||
<div className="overflow-hidden rounded-lg border border-border bg-card shadow-sm transition-shadow duration-200 group-hover:shadow-md">
|
||||
<div className="relative aspect-video overflow-hidden bg-muted">
|
||||
<img
|
||||
src={`/api/stream/thumb/${stream.channel.name}`}
|
||||
alt={stream.title}
|
||||
className="absolute inset-0 object-cover"
|
||||
/>
|
||||
<div className="absolute inset-0 bg-gradient-to-t from-black/60 via-transparent to-transparent" />
|
||||
<div className="absolute bottom-1.5 left-1.5 md:bottom-2 md:left-2">
|
||||
<LiveBadge small />
|
||||
</div>
|
||||
<div className="absolute bottom-1.5 right-1.5 md:bottom-2 md:right-2">
|
||||
<ViewerCount count={stream.viewers} small />
|
||||
</div>
|
||||
</div>
|
||||
<div className="flex items-start gap-2 p-2 md:gap-3 md:p-3">
|
||||
<Avatar className="h-7 w-7 shrink-0 ring-1 ring-primary/20 md:h-8 md:w-8">
|
||||
<AvatarImage src={stream.channel.pfpUrl} alt={stream.channel.name} />
|
||||
<AvatarFallback className="text-[10px]">
|
||||
{stream.channel.name.slice(0, 2).toUpperCase()}
|
||||
</AvatarFallback>
|
||||
</Avatar>
|
||||
<div className="min-w-0 flex-1">
|
||||
<p className="truncate text-xs font-medium leading-snug md:text-sm">{stream.title}</p>
|
||||
<p className="truncate text-[10px] text-muted-foreground md:text-xs">
|
||||
{stream.channel.name}
|
||||
</p>
|
||||
{stream.category && (
|
||||
<Badge
|
||||
variant="secondary"
|
||||
className="mt-1 rounded-full px-1.5 py-0 text-[9px] font-medium md:mt-1.5 md:px-2 md:text-[10px]"
|
||||
>
|
||||
{stream.category}
|
||||
</Badge>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</Link>
|
||||
);
|
||||
}
|
||||
|
||||
function OfflineCard({ stream }: { stream: StreamWithChannel }) {
|
||||
return (
|
||||
<Link href={`/${stream.username}`} className="group inline-flex">
|
||||
<div className="flex w-[70px] flex-col items-center gap-1 rounded-lg p-1.5 transition-colors duration-150 hover:bg-muted/50 sm:w-[78px] md:w-[86px] md:gap-1.5 md:p-2">
|
||||
<div className="relative">
|
||||
<Avatar className="h-9 w-9 ring-2 ring-border transition-colors duration-150 group-hover:ring-border/60 sm:h-10 sm:w-10 md:h-11 md:w-11">
|
||||
<AvatarImage src={stream.channel.pfpUrl} alt={stream.channel.name} />
|
||||
<AvatarFallback className="text-xs font-semibold">
|
||||
{stream.channel.name.slice(0, 2).toUpperCase()}
|
||||
</AvatarFallback>
|
||||
</Avatar>
|
||||
<span className="absolute -bottom-0.5 -right-0.5 h-2.5 w-2.5 rounded-full border-2 border-background bg-muted-foreground/40" />
|
||||
</div>
|
||||
<p className="w-full truncate text-center text-[10px] font-medium">{stream.channel.name}</p>
|
||||
</div>
|
||||
</Link>
|
||||
);
|
||||
}
|
||||
|
||||
function LiveBadge({ small }: { small?: boolean }) {
|
||||
return (
|
||||
<span
|
||||
className={`flex items-center gap-1 rounded-full bg-red-600 font-bold uppercase tracking-wide text-white ${small ? 'px-1.5 py-0.5 text-[9px]' : 'px-2 py-0.5 text-[10px]'}`}
|
||||
>
|
||||
<span className="inline-block h-1.5 w-1.5 animate-pulse rounded-full bg-white" />
|
||||
Live
|
||||
</span>
|
||||
);
|
||||
}
|
||||
|
||||
function ViewerCount({ count, small }: { count: number; small?: boolean }) {
|
||||
return (
|
||||
<span
|
||||
className={`flex items-center gap-1 rounded-full bg-black/70 font-medium text-white backdrop-blur-sm ${small ? 'px-1.5 py-0.5 text-[9px]' : 'px-2 py-0.5 text-xs'}`}
|
||||
>
|
||||
<span className="inline-block h-1.5 w-1.5 rounded-full bg-red-400" />
|
||||
{count.toLocaleString()}
|
||||
</span>
|
||||
);
|
||||
}
|
||||
|
||||
function SectionHeading({ label, count }: { label: string; count?: number }) {
|
||||
return (
|
||||
<div className="mb-3 flex items-center gap-2">
|
||||
<h2 className="pb-0 text-sm font-semibold tracking-tight md:text-base">{label}</h2>
|
||||
{count !== undefined && (
|
||||
<span className="rounded-full bg-primary/10 px-2 py-0.5 text-xs font-medium text-primary">
|
||||
{count}
|
||||
</span>
|
||||
)}
|
||||
<div className="ml-2 h-px flex-1 bg-border" />
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -7,18 +7,20 @@ import {
|
||||
MediaLoadingIndicator,
|
||||
MediaControlBar,
|
||||
MediaPlayButton,
|
||||
MediaSeekBackwardButton,
|
||||
MediaSeekForwardButton,
|
||||
MediaMuteButton,
|
||||
MediaVolumeRange,
|
||||
MediaFullscreenButton,
|
||||
} from 'media-chrome/react';
|
||||
import HlsVideo from 'hls-video-element/react';
|
||||
import { useSession } from '@/lib/providers/SessionProvider';
|
||||
import { useUserStreamInfo } from '@/lib/hooks/useUserList';
|
||||
import { getMediamtxClientEnvs } from '@/lib/utils/mediamtx/client';
|
||||
|
||||
export default function StreamPlayer() {
|
||||
const { username } = useParams();
|
||||
const { session } = useSession();
|
||||
const { streamInfo: userInfo } = useUserStreamInfo(username!.toString());
|
||||
|
||||
const videoRef = useRef(null);
|
||||
|
||||
useEffect(() => {
|
||||
@@ -34,10 +36,15 @@ export default function StreamPlayer() {
|
||||
},
|
||||
lowLatencyMode: true,
|
||||
debug: process.env.NODE_ENV === 'development',
|
||||
backBufferLength: 90,
|
||||
enableWorker: true,
|
||||
maxLiveSyncPlaybackRate: 1.5,
|
||||
liveSyncDurationCount: 2,
|
||||
liveMaxLatencyDurationCount: 4,
|
||||
};
|
||||
|
||||
// @ts-ignore
|
||||
video.src = `${process.env.NEXT_PUBLIC_MEDIAMTX_URL}/${username}/index.m3u8`;
|
||||
video.src = `${getMediamtxClientEnvs(userInfo?.streamRegion!).publicUrl}/${username}/index.m3u8`;
|
||||
}
|
||||
|
||||
return () => {
|
||||
|
||||
@@ -21,7 +21,13 @@ import { Textarea } from '@/components/ui/textarea';
|
||||
import { cn } from '@/lib/utils';
|
||||
import {
|
||||
createBotSchema,
|
||||
createChannelSchema, editBotSchema, onboardSchema, streamInfoEditSchema, updateChannelSettingsSchema
|
||||
createChannelSchema,
|
||||
changeUsernameSchema,
|
||||
editBotSchema,
|
||||
onboardSchema,
|
||||
streamInfoEditSchema,
|
||||
updateChatModerationSchema,
|
||||
updateChannelSettingsSchema,
|
||||
} from '@/lib/form/zod';
|
||||
|
||||
export const schemaDb = [
|
||||
@@ -30,7 +36,9 @@ export const schemaDb = [
|
||||
{ name: 'createChannel', zod: createChannelSchema },
|
||||
{ name: 'updateChannelSettings', zod: updateChannelSettingsSchema },
|
||||
{ name: 'createBot', zod: createBotSchema },
|
||||
{ name: 'editBot', zod: editBotSchema }
|
||||
{ name: 'editBot', zod: editBotSchema },
|
||||
{ name: 'changeUsername', zod: changeUsernameSchema },
|
||||
{ name: 'updateChatModeration', zod: updateChatModerationSchema },
|
||||
] as const;
|
||||
|
||||
export function UniversalForm<T extends z.ZodType>({
|
||||
@@ -62,7 +70,7 @@ export function UniversalForm<T extends z.ZodType>({
|
||||
}, [fields, defaultValues]);
|
||||
|
||||
type FormData = z.infer<T>;
|
||||
|
||||
|
||||
const form = useForm<FormData>({
|
||||
resolver: zodResolver(schema as any),
|
||||
defaultValues: initialValues as FormData,
|
||||
@@ -86,8 +94,8 @@ export function UniversalForm<T extends z.ZodType>({
|
||||
control={form.control}
|
||||
name={field.name as Path<FormData>}
|
||||
render={({ field: formField }) => (
|
||||
<FormItem>
|
||||
{(field.type !== 'hidden' || field.label) && <FormLabel>{field.label}</FormLabel>}
|
||||
<FormItem className={field.type === 'hidden' ? 'hidden' : undefined}>
|
||||
{field.type !== 'hidden' && field.label && <FormLabel>{field.label}</FormLabel>}
|
||||
<FormControl>
|
||||
{field.component ? (
|
||||
field.component({ field: formField, ...field.componentProps })
|
||||
@@ -97,27 +105,37 @@ export function UniversalForm<T extends z.ZodType>({
|
||||
{...formField}
|
||||
value={formField.value ?? ''}
|
||||
rows={field.textAreaRows ?? 5}
|
||||
maxLength={field.maxChars}
|
||||
/>
|
||||
) : (
|
||||
<Input
|
||||
type={field.type || 'text'}
|
||||
placeholder={field.placeholder}
|
||||
{...formField}
|
||||
onChange={(e) => {
|
||||
if (field.inputFilter) {
|
||||
e.target.value = e.target.value.replace(field.inputFilter, '');
|
||||
}
|
||||
formField.onChange(e);
|
||||
}}
|
||||
value={formField.value ?? ''}
|
||||
maxLength={field.maxChars}
|
||||
/>
|
||||
)}
|
||||
</FormControl>
|
||||
{field.description && <FormDescription>{field.description}</FormDescription>}
|
||||
<FormMessage />
|
||||
{field.type !== 'hidden' && field.description && (
|
||||
<FormDescription>{field.description}</FormDescription>
|
||||
)}
|
||||
{field.type !== 'hidden' && <FormMessage />}
|
||||
</FormItem>
|
||||
)}
|
||||
/>
|
||||
))}
|
||||
<div className={cn("flex gap-2 py-2", submitButtonDivClassname)}>
|
||||
<div className={cn('flex gap-2 py-2', submitButtonDivClassname)}>
|
||||
{otherSubmitButton}
|
||||
<SubmitButton buttonText={submitText} className={submitClassname} />
|
||||
</div>
|
||||
</form>
|
||||
</Form>
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -12,6 +12,8 @@ export type FormFieldConfig = {
|
||||
value?: any;
|
||||
textArea?: boolean;
|
||||
textAreaRows?: number;
|
||||
maxChars?: number;
|
||||
inputFilter?: RegExp;
|
||||
component?: (props: { field: ControllerRenderProps<any, any> } & any) => React.ReactNode;
|
||||
componentProps?: Record<string, any>;
|
||||
required?: boolean;
|
||||
|
||||
@@ -2,6 +2,7 @@ import { Avatar, AvatarImage } from '@/components/ui/avatar';
|
||||
import type { StreamInfo, Channel } from '@hctv/db';
|
||||
import FollowButton from './follow';
|
||||
import FollowCountText from './followCount';
|
||||
import StreamUptime from './streamUptime';
|
||||
import ViewerCount from './viewerCount';
|
||||
import { Preview } from '@/components/ui/channel-desc-fancy-area/preview';
|
||||
|
||||
@@ -21,6 +22,7 @@ export default function UserInfoCard(props: Props) {
|
||||
</div>
|
||||
<div className="flex items-center space-x-4">
|
||||
<ViewerCount />
|
||||
<StreamUptime />
|
||||
<FollowButton channel={props.streamInfo.username} />
|
||||
</div>
|
||||
</div>
|
||||
@@ -33,4 +35,4 @@ export default function UserInfoCard(props: Props) {
|
||||
|
||||
interface Props {
|
||||
streamInfo: StreamInfo & { channel: Channel };
|
||||
}
|
||||
}
|
||||
|
||||
55
apps/web/src/components/app/UserInfoCard/streamUptime.tsx
Normal file
55
apps/web/src/components/app/UserInfoCard/streamUptime.tsx
Normal file
@@ -0,0 +1,55 @@
|
||||
'use client';
|
||||
|
||||
import { useEffect, useMemo, useState } from 'react';
|
||||
import { Clock3 } from 'lucide-react';
|
||||
import { useParams } from 'next/navigation';
|
||||
import { useStreams } from '@/lib/providers/StreamInfoProvider';
|
||||
|
||||
export default function StreamUptime() {
|
||||
const { stream, isLoading } = useStreams();
|
||||
const { username } = useParams<{ username: string }>();
|
||||
const [now, setNow] = useState(Date.now());
|
||||
|
||||
useEffect(() => {
|
||||
const timer = window.setInterval(() => {
|
||||
setNow(Date.now());
|
||||
}, 1000);
|
||||
|
||||
return () => {
|
||||
window.clearInterval(timer);
|
||||
};
|
||||
}, []);
|
||||
|
||||
const startedAt = useMemo(() => {
|
||||
if (!stream || !username) {
|
||||
return null;
|
||||
}
|
||||
|
||||
const currentStream = stream.find((entry) => entry.username === username);
|
||||
if (!currentStream?.isLive) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return new Date(currentStream.startedAt).getTime();
|
||||
}, [stream, username]);
|
||||
|
||||
if (isLoading || !startedAt) {
|
||||
return null;
|
||||
}
|
||||
|
||||
const elapsedSeconds = Math.max(0, Math.floor((now - startedAt) / 1000));
|
||||
const hours = Math.floor(elapsedSeconds / 3600)
|
||||
.toString()
|
||||
.padStart(2, '0');
|
||||
const minutes = Math.floor((elapsedSeconds % 3600) / 60)
|
||||
.toString()
|
||||
.padStart(2, '0');
|
||||
const seconds = (elapsedSeconds % 60).toString().padStart(2, '0');
|
||||
|
||||
return (
|
||||
<div className="flex items-center space-x-2 text-sm text-muted-foreground">
|
||||
<Clock3 className="h-4 w-4" />
|
||||
<span>{`${hours}:${minutes}:${seconds}`}</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
213
apps/web/src/components/ui/calendar.tsx
Normal file
213
apps/web/src/components/ui/calendar.tsx
Normal file
@@ -0,0 +1,213 @@
|
||||
"use client"
|
||||
|
||||
import * as React from "react"
|
||||
import {
|
||||
ChevronDownIcon,
|
||||
ChevronLeftIcon,
|
||||
ChevronRightIcon,
|
||||
} from "lucide-react"
|
||||
import { DayButton, DayPicker, getDefaultClassNames } from "react-day-picker"
|
||||
|
||||
import { cn } from "@/lib/utils"
|
||||
import { Button, buttonVariants } from "@/components/ui/button"
|
||||
|
||||
function Calendar({
|
||||
className,
|
||||
classNames,
|
||||
showOutsideDays = true,
|
||||
captionLayout = "label",
|
||||
buttonVariant = "ghost",
|
||||
formatters,
|
||||
components,
|
||||
...props
|
||||
}: React.ComponentProps<typeof DayPicker> & {
|
||||
buttonVariant?: React.ComponentProps<typeof Button>["variant"]
|
||||
}) {
|
||||
const defaultClassNames = getDefaultClassNames()
|
||||
|
||||
return (
|
||||
<DayPicker
|
||||
showOutsideDays={showOutsideDays}
|
||||
className={cn(
|
||||
"bg-background group/calendar p-3 [--cell-size:2rem] [[data-slot=card-content]_&]:bg-transparent [[data-slot=popover-content]_&]:bg-transparent",
|
||||
String.raw`rtl:**:[.rdp-button\_next>svg]:rotate-180`,
|
||||
String.raw`rtl:**:[.rdp-button\_previous>svg]:rotate-180`,
|
||||
className
|
||||
)}
|
||||
captionLayout={captionLayout}
|
||||
formatters={{
|
||||
formatMonthDropdown: (date) =>
|
||||
date.toLocaleString("default", { month: "short" }),
|
||||
...formatters,
|
||||
}}
|
||||
classNames={{
|
||||
root: cn("w-fit", defaultClassNames.root),
|
||||
months: cn(
|
||||
"relative flex flex-col gap-4 md:flex-row",
|
||||
defaultClassNames.months
|
||||
),
|
||||
month: cn("flex w-full flex-col gap-4", defaultClassNames.month),
|
||||
nav: cn(
|
||||
"absolute inset-x-0 top-0 flex w-full items-center justify-between gap-1",
|
||||
defaultClassNames.nav
|
||||
),
|
||||
button_previous: cn(
|
||||
buttonVariants({ variant: buttonVariant }),
|
||||
"h-[--cell-size] w-[--cell-size] select-none p-0 aria-disabled:opacity-50",
|
||||
defaultClassNames.button_previous
|
||||
),
|
||||
button_next: cn(
|
||||
buttonVariants({ variant: buttonVariant }),
|
||||
"h-[--cell-size] w-[--cell-size] select-none p-0 aria-disabled:opacity-50",
|
||||
defaultClassNames.button_next
|
||||
),
|
||||
month_caption: cn(
|
||||
"flex h-[--cell-size] w-full items-center justify-center px-[--cell-size]",
|
||||
defaultClassNames.month_caption
|
||||
),
|
||||
dropdowns: cn(
|
||||
"flex h-[--cell-size] w-full items-center justify-center gap-1.5 text-sm font-medium",
|
||||
defaultClassNames.dropdowns
|
||||
),
|
||||
dropdown_root: cn(
|
||||
"has-focus:border-ring border-input shadow-xs has-focus:ring-ring/50 has-focus:ring-[3px] relative rounded-md border",
|
||||
defaultClassNames.dropdown_root
|
||||
),
|
||||
dropdown: cn(
|
||||
"bg-popover absolute inset-0 opacity-0",
|
||||
defaultClassNames.dropdown
|
||||
),
|
||||
caption_label: cn(
|
||||
"select-none font-medium",
|
||||
captionLayout === "label"
|
||||
? "text-sm"
|
||||
: "[&>svg]:text-muted-foreground flex h-8 items-center gap-1 rounded-md pl-2 pr-1 text-sm [&>svg]:size-3.5",
|
||||
defaultClassNames.caption_label
|
||||
),
|
||||
table: "w-full border-collapse",
|
||||
weekdays: cn("flex", defaultClassNames.weekdays),
|
||||
weekday: cn(
|
||||
"text-muted-foreground flex-1 select-none rounded-md text-[0.8rem] font-normal",
|
||||
defaultClassNames.weekday
|
||||
),
|
||||
week: cn("mt-2 flex w-full", defaultClassNames.week),
|
||||
week_number_header: cn(
|
||||
"w-[--cell-size] select-none",
|
||||
defaultClassNames.week_number_header
|
||||
),
|
||||
week_number: cn(
|
||||
"text-muted-foreground select-none text-[0.8rem]",
|
||||
defaultClassNames.week_number
|
||||
),
|
||||
day: cn(
|
||||
"group/day relative aspect-square h-full w-full select-none p-0 text-center [&:first-child[data-selected=true]_button]:rounded-l-md [&:last-child[data-selected=true]_button]:rounded-r-md",
|
||||
defaultClassNames.day
|
||||
),
|
||||
range_start: cn(
|
||||
"bg-accent rounded-l-md",
|
||||
defaultClassNames.range_start
|
||||
),
|
||||
range_middle: cn("rounded-none", defaultClassNames.range_middle),
|
||||
range_end: cn("bg-accent rounded-r-md", defaultClassNames.range_end),
|
||||
today: cn(
|
||||
"bg-accent text-accent-foreground rounded-md data-[selected=true]:rounded-none",
|
||||
defaultClassNames.today
|
||||
),
|
||||
outside: cn(
|
||||
"text-muted-foreground aria-selected:text-muted-foreground",
|
||||
defaultClassNames.outside
|
||||
),
|
||||
disabled: cn(
|
||||
"text-muted-foreground opacity-50",
|
||||
defaultClassNames.disabled
|
||||
),
|
||||
hidden: cn("invisible", defaultClassNames.hidden),
|
||||
...classNames,
|
||||
}}
|
||||
components={{
|
||||
Root: ({ className, rootRef, ...props }) => {
|
||||
return (
|
||||
<div
|
||||
data-slot="calendar"
|
||||
ref={rootRef}
|
||||
className={cn(className)}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
},
|
||||
Chevron: ({ className, orientation, ...props }) => {
|
||||
if (orientation === "left") {
|
||||
return (
|
||||
<ChevronLeftIcon className={cn("size-4", className)} {...props} />
|
||||
)
|
||||
}
|
||||
|
||||
if (orientation === "right") {
|
||||
return (
|
||||
<ChevronRightIcon
|
||||
className={cn("size-4", className)}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
}
|
||||
|
||||
return (
|
||||
<ChevronDownIcon className={cn("size-4", className)} {...props} />
|
||||
)
|
||||
},
|
||||
DayButton: CalendarDayButton,
|
||||
WeekNumber: ({ children, ...props }) => {
|
||||
return (
|
||||
<td {...props}>
|
||||
<div className="flex size-[--cell-size] items-center justify-center text-center">
|
||||
{children}
|
||||
</div>
|
||||
</td>
|
||||
)
|
||||
},
|
||||
...components,
|
||||
}}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
}
|
||||
|
||||
function CalendarDayButton({
|
||||
className,
|
||||
day,
|
||||
modifiers,
|
||||
...props
|
||||
}: React.ComponentProps<typeof DayButton>) {
|
||||
const defaultClassNames = getDefaultClassNames()
|
||||
|
||||
const ref = React.useRef<HTMLButtonElement>(null)
|
||||
React.useEffect(() => {
|
||||
if (modifiers.focused) ref.current?.focus()
|
||||
}, [modifiers.focused])
|
||||
|
||||
return (
|
||||
<Button
|
||||
ref={ref}
|
||||
variant="ghost"
|
||||
size="icon"
|
||||
data-day={day.date.toLocaleDateString()}
|
||||
data-selected-single={
|
||||
modifiers.selected &&
|
||||
!modifiers.range_start &&
|
||||
!modifiers.range_end &&
|
||||
!modifiers.range_middle
|
||||
}
|
||||
data-range-start={modifiers.range_start}
|
||||
data-range-end={modifiers.range_end}
|
||||
data-range-middle={modifiers.range_middle}
|
||||
className={cn(
|
||||
"data-[selected-single=true]:bg-primary data-[selected-single=true]:text-primary-foreground data-[range-middle=true]:bg-accent data-[range-middle=true]:text-accent-foreground data-[range-start=true]:bg-primary data-[range-start=true]:text-primary-foreground data-[range-end=true]:bg-primary data-[range-end=true]:text-primary-foreground group-data-[focused=true]/day:border-ring group-data-[focused=true]/day:ring-ring/50 flex aspect-square h-auto w-full min-w-[--cell-size] flex-col gap-1 font-normal leading-none data-[range-end=true]:rounded-md data-[range-middle=true]:rounded-none data-[range-start=true]:rounded-md group-data-[focused=true]/day:relative group-data-[focused=true]/day:z-10 group-data-[focused=true]/day:ring-[3px] [&>span]:text-xs [&>span]:opacity-70",
|
||||
defaultClassNames.day,
|
||||
className
|
||||
)}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
}
|
||||
|
||||
export { Calendar, CalendarDayButton }
|
||||
262
apps/web/src/components/ui/carousel.tsx
Normal file
262
apps/web/src/components/ui/carousel.tsx
Normal file
@@ -0,0 +1,262 @@
|
||||
"use client"
|
||||
|
||||
import * as React from "react"
|
||||
import useEmblaCarousel, {
|
||||
type UseEmblaCarouselType,
|
||||
} from "embla-carousel-react"
|
||||
import { ArrowLeft, ArrowRight } from "lucide-react"
|
||||
|
||||
import { cn } from "@/lib/utils"
|
||||
import { Button } from "@/components/ui/button"
|
||||
|
||||
type CarouselApi = UseEmblaCarouselType[1]
|
||||
type UseCarouselParameters = Parameters<typeof useEmblaCarousel>
|
||||
type CarouselOptions = UseCarouselParameters[0]
|
||||
type CarouselPlugin = UseCarouselParameters[1]
|
||||
|
||||
type CarouselProps = {
|
||||
opts?: CarouselOptions
|
||||
plugins?: CarouselPlugin
|
||||
orientation?: "horizontal" | "vertical"
|
||||
setApi?: (api: CarouselApi) => void
|
||||
}
|
||||
|
||||
type CarouselContextProps = {
|
||||
carouselRef: ReturnType<typeof useEmblaCarousel>[0]
|
||||
api: ReturnType<typeof useEmblaCarousel>[1]
|
||||
scrollPrev: () => void
|
||||
scrollNext: () => void
|
||||
canScrollPrev: boolean
|
||||
canScrollNext: boolean
|
||||
} & CarouselProps
|
||||
|
||||
const CarouselContext = React.createContext<CarouselContextProps | null>(null)
|
||||
|
||||
function useCarousel() {
|
||||
const context = React.useContext(CarouselContext)
|
||||
|
||||
if (!context) {
|
||||
throw new Error("useCarousel must be used within a <Carousel />")
|
||||
}
|
||||
|
||||
return context
|
||||
}
|
||||
|
||||
const Carousel = React.forwardRef<
|
||||
HTMLDivElement,
|
||||
React.HTMLAttributes<HTMLDivElement> & CarouselProps
|
||||
>(
|
||||
(
|
||||
{
|
||||
orientation = "horizontal",
|
||||
opts,
|
||||
setApi,
|
||||
plugins,
|
||||
className,
|
||||
children,
|
||||
...props
|
||||
},
|
||||
ref
|
||||
) => {
|
||||
const [carouselRef, api] = useEmblaCarousel(
|
||||
{
|
||||
...opts,
|
||||
axis: orientation === "horizontal" ? "x" : "y",
|
||||
},
|
||||
plugins
|
||||
)
|
||||
const [canScrollPrev, setCanScrollPrev] = React.useState(false)
|
||||
const [canScrollNext, setCanScrollNext] = React.useState(false)
|
||||
|
||||
const onSelect = React.useCallback((api: CarouselApi) => {
|
||||
if (!api) {
|
||||
return
|
||||
}
|
||||
|
||||
setCanScrollPrev(api.canScrollPrev())
|
||||
setCanScrollNext(api.canScrollNext())
|
||||
}, [])
|
||||
|
||||
const scrollPrev = React.useCallback(() => {
|
||||
api?.scrollPrev()
|
||||
}, [api])
|
||||
|
||||
const scrollNext = React.useCallback(() => {
|
||||
api?.scrollNext()
|
||||
}, [api])
|
||||
|
||||
const handleKeyDown = React.useCallback(
|
||||
(event: React.KeyboardEvent<HTMLDivElement>) => {
|
||||
if (event.key === "ArrowLeft") {
|
||||
event.preventDefault()
|
||||
scrollPrev()
|
||||
} else if (event.key === "ArrowRight") {
|
||||
event.preventDefault()
|
||||
scrollNext()
|
||||
}
|
||||
},
|
||||
[scrollPrev, scrollNext]
|
||||
)
|
||||
|
||||
React.useEffect(() => {
|
||||
if (!api || !setApi) {
|
||||
return
|
||||
}
|
||||
|
||||
setApi(api)
|
||||
}, [api, setApi])
|
||||
|
||||
React.useEffect(() => {
|
||||
if (!api) {
|
||||
return
|
||||
}
|
||||
|
||||
onSelect(api)
|
||||
api.on("reInit", onSelect)
|
||||
api.on("select", onSelect)
|
||||
|
||||
return () => {
|
||||
api?.off("select", onSelect)
|
||||
}
|
||||
}, [api, onSelect])
|
||||
|
||||
return (
|
||||
<CarouselContext.Provider
|
||||
value={{
|
||||
carouselRef,
|
||||
api: api,
|
||||
opts,
|
||||
orientation:
|
||||
orientation || (opts?.axis === "y" ? "vertical" : "horizontal"),
|
||||
scrollPrev,
|
||||
scrollNext,
|
||||
canScrollPrev,
|
||||
canScrollNext,
|
||||
}}
|
||||
>
|
||||
<div
|
||||
ref={ref}
|
||||
onKeyDownCapture={handleKeyDown}
|
||||
className={cn("relative", className)}
|
||||
role="region"
|
||||
aria-roledescription="carousel"
|
||||
{...props}
|
||||
>
|
||||
{children}
|
||||
</div>
|
||||
</CarouselContext.Provider>
|
||||
)
|
||||
}
|
||||
)
|
||||
Carousel.displayName = "Carousel"
|
||||
|
||||
const CarouselContent = React.forwardRef<
|
||||
HTMLDivElement,
|
||||
React.HTMLAttributes<HTMLDivElement>
|
||||
>(({ className, ...props }, ref) => {
|
||||
const { carouselRef, orientation } = useCarousel()
|
||||
|
||||
return (
|
||||
<div ref={carouselRef} className="overflow-hidden">
|
||||
<div
|
||||
ref={ref}
|
||||
className={cn(
|
||||
"flex",
|
||||
orientation === "horizontal" ? "-ml-4" : "-mt-4 flex-col",
|
||||
className
|
||||
)}
|
||||
{...props}
|
||||
/>
|
||||
</div>
|
||||
)
|
||||
})
|
||||
CarouselContent.displayName = "CarouselContent"
|
||||
|
||||
const CarouselItem = React.forwardRef<
|
||||
HTMLDivElement,
|
||||
React.HTMLAttributes<HTMLDivElement>
|
||||
>(({ className, ...props }, ref) => {
|
||||
const { orientation } = useCarousel()
|
||||
|
||||
return (
|
||||
<div
|
||||
ref={ref}
|
||||
role="group"
|
||||
aria-roledescription="slide"
|
||||
className={cn(
|
||||
"min-w-0 shrink-0 grow-0 basis-full",
|
||||
orientation === "horizontal" ? "pl-4" : "pt-4",
|
||||
className
|
||||
)}
|
||||
{...props}
|
||||
/>
|
||||
)
|
||||
})
|
||||
CarouselItem.displayName = "CarouselItem"
|
||||
|
||||
const CarouselPrevious = React.forwardRef<
|
||||
HTMLButtonElement,
|
||||
React.ComponentProps<typeof Button>
|
||||
>(({ className, variant = "outline", size = "icon", ...props }, ref) => {
|
||||
const { orientation, scrollPrev, canScrollPrev } = useCarousel()
|
||||
|
||||
return (
|
||||
<Button
|
||||
ref={ref}
|
||||
variant={variant}
|
||||
size={size}
|
||||
className={cn(
|
||||
"absolute h-8 w-8 rounded-full",
|
||||
orientation === "horizontal"
|
||||
? "-left-12 top-1/2 -translate-y-1/2"
|
||||
: "-top-12 left-1/2 -translate-x-1/2 rotate-90",
|
||||
className
|
||||
)}
|
||||
disabled={!canScrollPrev}
|
||||
onClick={scrollPrev}
|
||||
{...props}
|
||||
>
|
||||
<ArrowLeft className="h-4 w-4" />
|
||||
<span className="sr-only">Previous slide</span>
|
||||
</Button>
|
||||
)
|
||||
})
|
||||
CarouselPrevious.displayName = "CarouselPrevious"
|
||||
|
||||
const CarouselNext = React.forwardRef<
|
||||
HTMLButtonElement,
|
||||
React.ComponentProps<typeof Button>
|
||||
>(({ className, variant = "outline", size = "icon", ...props }, ref) => {
|
||||
const { orientation, scrollNext, canScrollNext } = useCarousel()
|
||||
|
||||
return (
|
||||
<Button
|
||||
ref={ref}
|
||||
variant={variant}
|
||||
size={size}
|
||||
className={cn(
|
||||
"absolute h-8 w-8 rounded-full",
|
||||
orientation === "horizontal"
|
||||
? "-right-12 top-1/2 -translate-y-1/2"
|
||||
: "-bottom-12 left-1/2 -translate-x-1/2 rotate-90",
|
||||
className
|
||||
)}
|
||||
disabled={!canScrollNext}
|
||||
onClick={scrollNext}
|
||||
{...props}
|
||||
>
|
||||
<ArrowRight className="h-4 w-4" />
|
||||
<span className="sr-only">Next slide</span>
|
||||
</Button>
|
||||
)
|
||||
})
|
||||
CarouselNext.displayName = "CarouselNext"
|
||||
|
||||
export {
|
||||
type CarouselApi,
|
||||
Carousel,
|
||||
CarouselContent,
|
||||
CarouselItem,
|
||||
CarouselPrevious,
|
||||
CarouselNext,
|
||||
}
|
||||
121
apps/web/src/lib/auth/abac.ts
Normal file
121
apps/web/src/lib/auth/abac.ts
Normal file
@@ -0,0 +1,121 @@
|
||||
import { prisma } from '@hctv/db';
|
||||
|
||||
export type Resource = 'channel' | 'bot' | 'streamInfo';
|
||||
export type Action = 'read' | 'update' | 'delete' | 'manage';
|
||||
|
||||
type User = { id: string };
|
||||
|
||||
type ChannelWithRelations = {
|
||||
ownerId: string;
|
||||
managers?: { id: string }[];
|
||||
personalFor?: { id: string } | null;
|
||||
};
|
||||
|
||||
type BotWithRelations = {
|
||||
ownerId: string;
|
||||
};
|
||||
|
||||
type PolicyContext = {
|
||||
channel?: ChannelWithRelations;
|
||||
bot?: BotWithRelations;
|
||||
};
|
||||
|
||||
const policies: Record<Resource, Record<Action, (user: User, ctx: PolicyContext) => boolean>> = {
|
||||
channel: {
|
||||
read: () => true,
|
||||
update: (user, { channel }) => {
|
||||
if (!channel) return false;
|
||||
return channel.ownerId === user.id || (channel.managers?.some((m) => m.id === user.id) ?? false);
|
||||
},
|
||||
delete: (user, { channel }) => {
|
||||
if (!channel) return false;
|
||||
if (channel.personalFor) return false;
|
||||
return channel.ownerId === user.id;
|
||||
},
|
||||
manage: (user, { channel }) => {
|
||||
if (!channel) return false;
|
||||
return channel.ownerId === user.id;
|
||||
},
|
||||
},
|
||||
bot: {
|
||||
read: () => true,
|
||||
update: (user, { bot }) => {
|
||||
if (!bot) return false;
|
||||
return bot.ownerId === user.id;
|
||||
},
|
||||
delete: (user, { bot }) => {
|
||||
if (!bot) return false;
|
||||
return bot.ownerId === user.id;
|
||||
},
|
||||
manage: (user, { bot }) => {
|
||||
if (!bot) return false;
|
||||
return bot.ownerId === user.id;
|
||||
},
|
||||
},
|
||||
streamInfo: {
|
||||
read: () => true,
|
||||
update: (user, { channel }) => {
|
||||
if (!channel) return false;
|
||||
return channel.ownerId === user.id || (channel.managers?.some((m) => m.id === user.id) ?? false);
|
||||
},
|
||||
delete: () => false,
|
||||
manage: (user, { channel }) => {
|
||||
if (!channel) return false;
|
||||
return channel.ownerId === user.id;
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
export function can(user: User, action: Action, resource: Resource, context: PolicyContext): boolean {
|
||||
const policy = policies[resource]?.[action];
|
||||
if (!policy) return false;
|
||||
return policy(user, context);
|
||||
}
|
||||
|
||||
export async function canAccessChannel(
|
||||
user: User,
|
||||
action: Action,
|
||||
channelId: string
|
||||
): Promise<boolean> {
|
||||
const channel = await prisma.channel.findUnique({
|
||||
where: { id: channelId },
|
||||
include: { managers: { select: { id: true } }, personalFor: { select: { id: true } } },
|
||||
});
|
||||
if (!channel) return false;
|
||||
return can(user, action, 'channel', { channel });
|
||||
}
|
||||
|
||||
export async function canAccessChannelByName(
|
||||
user: User,
|
||||
action: Action,
|
||||
channelName: string
|
||||
): Promise<boolean> {
|
||||
const channel = await prisma.channel.findUnique({
|
||||
where: { name: channelName },
|
||||
include: { managers: { select: { id: true } }, personalFor: { select: { id: true } } },
|
||||
});
|
||||
if (!channel) return false;
|
||||
return can(user, action, 'channel', { channel });
|
||||
}
|
||||
|
||||
export async function canAccessBot(user: User, action: Action, botId: string): Promise<boolean> {
|
||||
const bot = await prisma.botAccount.findUnique({
|
||||
where: { id: botId },
|
||||
select: { ownerId: true },
|
||||
});
|
||||
if (!bot) return false;
|
||||
return can(user, action, 'bot', { bot });
|
||||
}
|
||||
|
||||
export async function canAccessBotBySlug(
|
||||
user: User,
|
||||
action: Action,
|
||||
slug: string
|
||||
): Promise<boolean> {
|
||||
const bot = await prisma.botAccount.findUnique({
|
||||
where: { slug },
|
||||
select: { ownerId: true },
|
||||
});
|
||||
if (!bot) return false;
|
||||
return can(user, action, 'bot', { bot });
|
||||
}
|
||||
@@ -12,5 +12,5 @@ export async function logout() {
|
||||
await lucia.invalidateSession(session!.id);
|
||||
const sessionCookie = lucia.createBlankSessionCookie();
|
||||
(await cookies()).set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes);
|
||||
return redirect('/login');
|
||||
return redirect('/');
|
||||
}
|
||||
|
||||
10
apps/web/src/lib/env.ts
Normal file
10
apps/web/src/lib/env.ts
Normal file
@@ -0,0 +1,10 @@
|
||||
export const getEnv = (key: string): string | undefined => {
|
||||
if (typeof window !== 'undefined') {
|
||||
// @ts-ignore
|
||||
return window.__ENV?.[key];
|
||||
}
|
||||
return process.env[key];
|
||||
};
|
||||
|
||||
export const MEDIAMTX_URL = getEnv('NEXT_PUBLIC_MEDIAMTX_URL');
|
||||
export const MEDIAMTX_INGEST_ROUTE = getEnv('NEXT_PUBLIC_MEDIAMTX_INGEST_ROUTE');
|
||||
@@ -2,14 +2,16 @@
|
||||
|
||||
import { revalidatePath } from 'next/cache';
|
||||
import { validateRequest } from '@/lib/auth/validate';
|
||||
import { prisma } from '@hctv/db';
|
||||
import { prisma, getRedisConnection } from '@hctv/db';
|
||||
import zodVerify from '../zodVerify';
|
||||
import {
|
||||
createBotSchema,
|
||||
createChannelSchema,
|
||||
changeUsernameSchema,
|
||||
editBotSchema,
|
||||
onboardSchema,
|
||||
streamInfoEditSchema,
|
||||
updateChatModerationSchema,
|
||||
updateChannelSettingsSchema,
|
||||
} from './zod';
|
||||
import { initializeStreamInfo } from '../instrumentation/streamInfo';
|
||||
@@ -18,6 +20,7 @@ import {
|
||||
resolveStreamInfo,
|
||||
resolveUserFromPersonalChannelName,
|
||||
} from '../auth/resolve';
|
||||
import { can } from '../auth/abac';
|
||||
import { genIdenticonUpload } from '../utils/genIdenticonUpload';
|
||||
import { generateStreamKey } from '../db/streamKey';
|
||||
|
||||
@@ -42,9 +45,7 @@ export async function editStreamInfo(prev: any, formData: FormData) {
|
||||
return { success: false, error: 'Channel not found' };
|
||||
}
|
||||
|
||||
const isBroadcaster =
|
||||
channelInfo.ownerId === user.id || channelInfo.managers.some((m) => m.id === user.id);
|
||||
if (!isBroadcaster) {
|
||||
if (!can(user, 'update', 'streamInfo', { channel: channelInfo })) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
@@ -105,8 +106,8 @@ export async function onboard(prev: any, formData: FormData) {
|
||||
|
||||
await generateStreamKey(createdChannel.id, createdChannel.name);
|
||||
|
||||
if (process.env.NODE_ENV === 'production') {
|
||||
await fetch(process.env.WELCOME_WORKFLOW_URL!, {
|
||||
if (process.env.NODE_ENV === 'production' && process.env.WELCOME_WORKFLOW_URL) {
|
||||
await fetch(process.env.WELCOME_WORKFLOW_URL, {
|
||||
method: 'POST',
|
||||
body: JSON.stringify({
|
||||
username: zod.data.username,
|
||||
@@ -202,10 +203,7 @@ export async function updateChannelSettings(prev: any, formData: FormData) {
|
||||
return { success: false, error: 'Channel not found' };
|
||||
}
|
||||
|
||||
const isOwner = channel.ownerId === user.id;
|
||||
const isManager = channel.managers.some((manager) => manager.id === user.id);
|
||||
|
||||
if (!isOwner && !isManager) {
|
||||
if (!can(user, 'update', 'channel', { channel })) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
@@ -242,7 +240,7 @@ export async function addChannelManager(channelId: string, userChannel: string)
|
||||
return { success: false, error: 'Channel not found OR is personal.' };
|
||||
}
|
||||
|
||||
if (channel.ownerId !== user.id) {
|
||||
if (!can(user, 'manage', 'channel', { channel })) {
|
||||
return { success: false, error: 'Only channel owners can add managers' };
|
||||
}
|
||||
|
||||
@@ -264,6 +262,9 @@ export async function addChannelManager(channelId: string, userChannel: string)
|
||||
managers: {
|
||||
connect: { id: userDb.id },
|
||||
},
|
||||
chatModerators: {
|
||||
connect: { id: userDb.id },
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
@@ -271,6 +272,232 @@ export async function addChannelManager(channelId: string, userChannel: string)
|
||||
return { success: true };
|
||||
}
|
||||
|
||||
export async function addChatModerator(channelId: string, userChannel: string) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
const channel = await prisma.channel.findUnique({
|
||||
where: { id: channelId },
|
||||
include: { owner: true, managers: true, chatModerators: true },
|
||||
});
|
||||
|
||||
if (!channel) {
|
||||
return { success: false, error: 'Channel not found' };
|
||||
}
|
||||
|
||||
if (!can(user, 'update', 'channel', { channel })) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
const userDb = await resolveUserFromPersonalChannelName(userChannel);
|
||||
if (!userDb) {
|
||||
return { success: false, error: 'User not found' };
|
||||
}
|
||||
|
||||
if (
|
||||
channel.ownerId === userDb.id ||
|
||||
channel.managers.some((manager) => manager.id === userDb.id)
|
||||
) {
|
||||
return { success: false, error: 'This user is already a built-in moderator' };
|
||||
}
|
||||
|
||||
if (channel.chatModerators.some((moderator) => moderator.id === userDb.id)) {
|
||||
return { success: false, error: 'User is already a chat moderator' };
|
||||
}
|
||||
|
||||
await prisma.channel.update({
|
||||
where: { id: channelId },
|
||||
data: {
|
||||
chatModerators: {
|
||||
connect: { id: userDb.id },
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
revalidatePath(`/settings/channel/${channel.name}`);
|
||||
return { success: true };
|
||||
}
|
||||
|
||||
export async function removeChatModerator(channelId: string, userId: string) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
const channel = await prisma.channel.findUnique({
|
||||
where: { id: channelId },
|
||||
include: { owner: true, managers: true },
|
||||
});
|
||||
|
||||
if (!channel) {
|
||||
return { success: false, error: 'Channel not found' };
|
||||
}
|
||||
|
||||
if (!can(user, 'update', 'channel', { channel })) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
await prisma.channel.update({
|
||||
where: { id: channelId },
|
||||
data: {
|
||||
chatModerators: {
|
||||
disconnect: { id: userId },
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
revalidatePath(`/settings/channel/${channel.name}`);
|
||||
return { success: true };
|
||||
}
|
||||
|
||||
export async function addChatBotModerator(channelId: string, botId: string) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
const channel = await prisma.channel.findUnique({
|
||||
where: { id: channelId },
|
||||
include: { owner: true, managers: true, chatModeratorBots: true },
|
||||
});
|
||||
|
||||
if (!channel) {
|
||||
return { success: false, error: 'Channel not found' };
|
||||
}
|
||||
|
||||
if (!can(user, 'update', 'channel', { channel })) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
const bot = await prisma.botAccount.findUnique({
|
||||
where: { id: botId },
|
||||
select: { id: true, ownerId: true },
|
||||
});
|
||||
|
||||
if (!bot) {
|
||||
return { success: false, error: 'Bot not found' };
|
||||
}
|
||||
|
||||
if (channel.chatModeratorBots.some((existingBot) => existingBot.id === bot.id)) {
|
||||
return { success: false, error: 'Bot is already a chat moderator' };
|
||||
}
|
||||
|
||||
const canUseBot =
|
||||
bot.ownerId === channel.ownerId ||
|
||||
channel.managers.some((manager) => manager.id === bot.ownerId);
|
||||
|
||||
if (!canUseBot) {
|
||||
return { success: false, error: 'Bot owner must be a channel manager or owner' };
|
||||
}
|
||||
|
||||
await prisma.channel.update({
|
||||
where: { id: channelId },
|
||||
data: {
|
||||
chatModeratorBots: {
|
||||
connect: { id: bot.id },
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
revalidatePath(`/settings/channel/${channel.name}`);
|
||||
return { success: true };
|
||||
}
|
||||
|
||||
export async function removeChatBotModerator(channelId: string, botId: string) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
const channel = await prisma.channel.findUnique({
|
||||
where: { id: channelId },
|
||||
include: { owner: true, managers: true },
|
||||
});
|
||||
|
||||
if (!channel) {
|
||||
return { success: false, error: 'Channel not found' };
|
||||
}
|
||||
|
||||
if (!can(user, 'update', 'channel', { channel })) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
await prisma.channel.update({
|
||||
where: { id: channelId },
|
||||
data: {
|
||||
chatModeratorBots: {
|
||||
disconnect: { id: botId },
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
revalidatePath(`/settings/channel/${channel.name}`);
|
||||
return { success: true };
|
||||
}
|
||||
|
||||
export async function updateChatModeration(prev: any, formData: FormData) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
const zod = await zodVerify(updateChatModerationSchema, formData);
|
||||
if (!zod.success) {
|
||||
return zod;
|
||||
}
|
||||
|
||||
const channel = await prisma.channel.findUnique({
|
||||
where: { id: zod.data.channelId },
|
||||
include: {
|
||||
owner: true,
|
||||
managers: true,
|
||||
},
|
||||
});
|
||||
|
||||
if (!channel) {
|
||||
return { success: false, error: 'Channel not found' };
|
||||
}
|
||||
|
||||
if (!can(user, 'update', 'channel', { channel })) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
const blockedTerms = (zod.data.blockedTerms ?? '')
|
||||
.split(/[\n,]/)
|
||||
.map((term) => term.trim().toLowerCase())
|
||||
.filter((term) => term.length >= 2)
|
||||
.slice(0, 200);
|
||||
|
||||
await prisma.chatModerationSettings.upsert({
|
||||
where: {
|
||||
channelId: channel.id,
|
||||
},
|
||||
create: {
|
||||
channelId: channel.id,
|
||||
blockedTerms,
|
||||
slowModeSeconds: zod.data.slowModeSeconds,
|
||||
maxMessageLength: zod.data.maxMessageLength,
|
||||
rateLimitCount: zod.data.rateLimitCount,
|
||||
rateLimitWindowSeconds: zod.data.rateLimitWindowSeconds,
|
||||
},
|
||||
update: {
|
||||
blockedTerms,
|
||||
slowModeSeconds: zod.data.slowModeSeconds,
|
||||
maxMessageLength: zod.data.maxMessageLength,
|
||||
rateLimitCount: zod.data.rateLimitCount,
|
||||
rateLimitWindowSeconds: zod.data.rateLimitWindowSeconds,
|
||||
},
|
||||
});
|
||||
|
||||
const redis = getRedisConnection();
|
||||
await redis.del(`chat:moderation:settings:${channel.id}`);
|
||||
|
||||
revalidatePath(`/settings/channel/${channel.name}`);
|
||||
return { success: true };
|
||||
}
|
||||
|
||||
export async function removeChannelManager(channelId: string, userId: string) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user) {
|
||||
@@ -286,7 +513,7 @@ export async function removeChannelManager(channelId: string, userId: string) {
|
||||
return { success: false, error: 'Channel not found' };
|
||||
}
|
||||
|
||||
if (channel.ownerId !== user.id) {
|
||||
if (!can(user, 'manage', 'channel', { channel })) {
|
||||
return { success: false, error: 'Only channel owners can remove managers' };
|
||||
}
|
||||
|
||||
@@ -296,6 +523,9 @@ export async function removeChannelManager(channelId: string, userId: string) {
|
||||
managers: {
|
||||
disconnect: { id: userId },
|
||||
},
|
||||
chatModerators: {
|
||||
disconnect: { id: userId },
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
@@ -355,12 +585,11 @@ export async function deleteChannel(channelId: string) {
|
||||
return { success: false, error: 'Channel not found' };
|
||||
}
|
||||
|
||||
if (channel.ownerId !== user.id) {
|
||||
return { success: false, error: 'Only channel owners can delete channels' };
|
||||
}
|
||||
|
||||
if (channel.personalFor) {
|
||||
return { success: false, error: 'Cannot delete personal channels' };
|
||||
if (!can(user, 'delete', 'channel', { channel })) {
|
||||
return {
|
||||
success: false,
|
||||
error: 'Only channel owners can delete channels (personal channels cannot be deleted)',
|
||||
};
|
||||
}
|
||||
|
||||
await prisma.channel.delete({
|
||||
@@ -416,7 +645,7 @@ export async function editBot(prev: any, formData: FormData) {
|
||||
if (!bot) {
|
||||
return { success: false, error: 'Bot not found' };
|
||||
}
|
||||
if (bot.ownerId !== user.id) {
|
||||
if (!can(user, 'update', 'bot', { bot })) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
if (bot.slug !== zod.data.slug) {
|
||||
@@ -441,3 +670,125 @@ export async function editBot(prev: any, formData: FormData) {
|
||||
|
||||
return { success: true, slug: updatedBot.slug };
|
||||
}
|
||||
|
||||
const USERNAME_CHANGE_COOLDOWN_DAYS = 30;
|
||||
|
||||
export async function changeUsername(prev: any, formData: FormData) {
|
||||
const { user } = await validateRequest();
|
||||
if (!user) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
const zod = await zodVerify(changeUsernameSchema, formData);
|
||||
if (!zod.success) {
|
||||
return zod;
|
||||
}
|
||||
|
||||
const channel = await prisma.channel.findUnique({
|
||||
where: { id: zod.data.channelId },
|
||||
include: {
|
||||
owner: true,
|
||||
managers: true,
|
||||
personalFor: true,
|
||||
streamInfo: true,
|
||||
streamKey: true,
|
||||
},
|
||||
});
|
||||
|
||||
if (!channel) {
|
||||
return { success: false, error: 'Channel not found' };
|
||||
}
|
||||
|
||||
if (!channel.personalFor || channel.personalFor.id !== user.id) {
|
||||
return { success: false, error: 'You can only change the username of your personal channel' };
|
||||
}
|
||||
|
||||
if (channel.ownerId !== user.id) {
|
||||
return { success: false, error: 'Unauthorized' };
|
||||
}
|
||||
|
||||
if (channel.nameLastChanged) {
|
||||
const daysSinceLastChange = Math.floor(
|
||||
(Date.now() - new Date(channel.nameLastChanged).getTime()) / (1000 * 60 * 60 * 24)
|
||||
);
|
||||
if (daysSinceLastChange < USERNAME_CHANGE_COOLDOWN_DAYS) {
|
||||
const daysRemaining = USERNAME_CHANGE_COOLDOWN_DAYS - daysSinceLastChange;
|
||||
return {
|
||||
success: false,
|
||||
error: `Please wait ${daysRemaining} more day${daysRemaining === 1 ? '' : 's'}.`,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
const oldName = channel.name;
|
||||
const newName = zod.data.newUsername;
|
||||
|
||||
if (oldName === newName) {
|
||||
return { success: false, error: 'New username must be different from the current one' };
|
||||
}
|
||||
|
||||
const existingChannel = await prisma.channel.findUnique({
|
||||
where: { name: newName },
|
||||
});
|
||||
if (existingChannel) {
|
||||
return { success: false, error: 'This username is already taken' };
|
||||
}
|
||||
|
||||
const redis = getRedisConnection();
|
||||
|
||||
try {
|
||||
await prisma.channel.update({
|
||||
where: { id: channel.id },
|
||||
data: {
|
||||
name: newName,
|
||||
nameLastChanged: process.env.NODE_ENV === 'production' ? new Date() : null,
|
||||
},
|
||||
});
|
||||
|
||||
if (channel.streamInfo.length > 0) {
|
||||
await prisma.streamInfo.updateMany({
|
||||
where: { channelId: channel.id },
|
||||
data: { username: newName },
|
||||
});
|
||||
}
|
||||
|
||||
if (channel.streamKey) {
|
||||
const oldStreamKey = `streamKey:${oldName}`;
|
||||
const newStreamKey = `streamKey:${newName}`;
|
||||
if (await redis.exists(oldStreamKey)) {
|
||||
await redis.rename(oldStreamKey, newStreamKey);
|
||||
}
|
||||
}
|
||||
|
||||
const oldHistoryKey = `chat:history:${oldName}`;
|
||||
const newHistoryKey = `chat:history:${newName}`;
|
||||
if (await redis.exists(oldHistoryKey)) {
|
||||
const messagesWithScores = await redis.zrange(oldHistoryKey, 0, -1, 'WITHSCORES');
|
||||
if (messagesWithScores.length > 0) {
|
||||
const args: (string | number)[] = [];
|
||||
for (let i = 0; i < messagesWithScores.length; i += 2) {
|
||||
const msgStr = messagesWithScores[i];
|
||||
const score = messagesWithScores[i + 1];
|
||||
try {
|
||||
const msg = JSON.parse(msgStr);
|
||||
msg.user.username = newName;
|
||||
args.push(score, JSON.stringify(msg));
|
||||
} catch {
|
||||
args.push(score, msgStr);
|
||||
}
|
||||
}
|
||||
await redis.zadd(newHistoryKey, ...args);
|
||||
}
|
||||
await redis.del(oldHistoryKey);
|
||||
}
|
||||
|
||||
revalidatePath(`/settings/channel/${newName}`);
|
||||
revalidatePath(`/${oldName}`);
|
||||
revalidatePath(`/${newName}`);
|
||||
|
||||
return { success: true, newUsername: newName };
|
||||
} catch (error) {
|
||||
console.error('Failed to change username:', error);
|
||||
return { success: false, error: 'Failed to change username. Please try again.' };
|
||||
}
|
||||
}
|
||||
|
||||
@@ -11,6 +11,7 @@ const disallowedUsernames = [
|
||||
const username = z
|
||||
.string()
|
||||
.min(1)
|
||||
.max(20)
|
||||
.regex(/^[a-z0-9_-]+$/, { message: 'Only characters from a-z, 0-9, underscores and dashes' })
|
||||
.refine((val) => !disallowedUsernames.includes(val.toLowerCase()), {
|
||||
message: 'This username is reserved',
|
||||
@@ -38,6 +39,15 @@ export const updateChannelSettingsSchema = z.object({
|
||||
is247: z.boolean(),
|
||||
});
|
||||
|
||||
export const updateChatModerationSchema = z.object({
|
||||
channelId: z.string().min(1),
|
||||
blockedTerms: z.string().max(5000).optional(),
|
||||
slowModeSeconds: z.coerce.number().int().min(0).max(120),
|
||||
maxMessageLength: z.coerce.number().int().min(50).max(2000),
|
||||
rateLimitCount: z.coerce.number().int().min(3).max(30),
|
||||
rateLimitWindowSeconds: z.coerce.number().int().min(5).max(60),
|
||||
});
|
||||
|
||||
export const createBotSchema = z.object({
|
||||
name: z.string().min(1, { message: 'Name is required' }),
|
||||
slug: username.refine((val) => val !== 'settings', { message: 'This slug is reserved' }),
|
||||
@@ -49,3 +59,8 @@ export const editBotSchema = createBotSchema.and(
|
||||
from: z.string().min(1),
|
||||
})
|
||||
);
|
||||
|
||||
export const changeUsernameSchema = z.object({
|
||||
channelId: z.string().min(1),
|
||||
newUsername: username,
|
||||
});
|
||||
|
||||
@@ -27,6 +27,7 @@ function createCacheKey(options: UseUserListOptions): string {
|
||||
if (options.owned) params.push('owned')
|
||||
if (options.personal) params.push('personal')
|
||||
if (options.live) params.push('live')
|
||||
if (options.username) params.push(`user-${options.username}`)
|
||||
|
||||
return params.length > 0
|
||||
? `stream-info:${params.join('-')}`
|
||||
@@ -76,6 +77,8 @@ export interface UseUserListOptions {
|
||||
personal?: boolean
|
||||
/** Only fetch live channels */
|
||||
live?: boolean
|
||||
/** Search for a specific user's streaminfo */
|
||||
username?: string
|
||||
/** Refresh interval in milliseconds */
|
||||
refreshInterval?: number
|
||||
/** Cache time to live in milliseconds (default: 5 minutes) */
|
||||
@@ -132,6 +135,7 @@ export function useUserList(options: UseUserListOptions = {}): UseUserListReturn
|
||||
owned = false,
|
||||
personal = false,
|
||||
live = false,
|
||||
username,
|
||||
refreshInterval = 30000,
|
||||
cacheTTL = 5 * 60 * 1000, // 5 minutes
|
||||
revalidateOnFocus = false,
|
||||
@@ -151,8 +155,9 @@ export function useUserList(options: UseUserListOptions = {}): UseUserListReturn
|
||||
if (owned) searchParams.set('owned', 'true')
|
||||
if (personal) searchParams.set('personal', 'true')
|
||||
if (live) searchParams.set('live', 'true')
|
||||
if (username) searchParams.set('username', username)
|
||||
return searchParams
|
||||
}, [owned, personal, live])
|
||||
}, [owned, personal, live, username])
|
||||
|
||||
const queryString = params.toString()
|
||||
const url = `/api/stream/info${queryString ? `?${queryString}` : ''}`
|
||||
@@ -325,6 +330,67 @@ export function usePersonalChannels(refreshInterval?: number): UseUserListReturn
|
||||
})
|
||||
}
|
||||
|
||||
export interface UseUserStreamInfoReturn extends Omit<UseUserListReturn, 'channels'> {
|
||||
/** The found stream info for the specific user */
|
||||
streamInfo: StreamInfoResponse[0] | null
|
||||
/** All matching channels (usually just one) */
|
||||
channels: StreamInfoResponse
|
||||
}
|
||||
|
||||
/**
|
||||
* Hook to fetch stream info for a specific user
|
||||
* Returns the first match if multiple channels exist for that user
|
||||
*/
|
||||
export function useUserStreamInfo(
|
||||
username: string | undefined,
|
||||
refresh = true,
|
||||
refreshInterval?: number,
|
||||
): UseUserStreamInfoReturn {
|
||||
const result = useUserList({
|
||||
username,
|
||||
refreshInterval: refresh ? (refreshInterval ?? 15000) : undefined,
|
||||
cacheTTL: 2 * 60 * 1000, // 2 minutes cache
|
||||
revalidateOnFocus: true,
|
||||
isPaused: !username, // Don't fetch if no username provided
|
||||
errorRetryCount: 3,
|
||||
})
|
||||
|
||||
return {
|
||||
...result,
|
||||
streamInfo: result.channels[0] || null,
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Lazy version that doesn't automatically fetch - useful for on-demand lookups
|
||||
*/
|
||||
export function useUserStreamInfoLazy(refreshInterval?: number) {
|
||||
const result = useUserList({
|
||||
refreshInterval: refreshInterval ?? 15000,
|
||||
cacheTTL: 2 * 60 * 1000,
|
||||
revalidateOnFocus: true,
|
||||
isPaused: true, // Start paused
|
||||
errorRetryCount: 3,
|
||||
})
|
||||
|
||||
const lookupUser = useCallback(async (username: string) => {
|
||||
if (!username) return null
|
||||
|
||||
try {
|
||||
const response = await enhancedFetcher(`/api/stream/info?username=${encodeURIComponent(username)}`)
|
||||
return response[0] || null
|
||||
} catch (error) {
|
||||
console.error('[useUserStreamInfoLazy] Error looking up user:', error)
|
||||
throw error
|
||||
}
|
||||
}, [])
|
||||
|
||||
return {
|
||||
...result,
|
||||
lookupUser,
|
||||
}
|
||||
}
|
||||
|
||||
// Cache management utilities with proper error handling
|
||||
export const channelCacheUtils = {
|
||||
/** Clear all channel caches */
|
||||
@@ -379,6 +445,7 @@ export const channelCacheUtils = {
|
||||
if (options.owned) params.set('owned', 'true')
|
||||
if (options.personal) params.set('personal', 'true')
|
||||
if (options.live) params.set('live', 'true')
|
||||
if (options.username) params.set('username', options.username)
|
||||
|
||||
const queryString = params.toString()
|
||||
const url = `/api/stream/info${queryString ? `?${queryString}` : ''}`
|
||||
|
||||
@@ -14,8 +14,10 @@ export default async function getLiveThumb() {
|
||||
|
||||
const thumbQueue = getThumbnailQueue();
|
||||
for (const channel of liveChannelNames) {
|
||||
const lc = liveChannels.find(c => c.channel.name === channel)!;
|
||||
await thumbQueue.add("getLiveThumb", {
|
||||
name: channel,
|
||||
server: lc.streamRegion,
|
||||
});
|
||||
}
|
||||
}
|
||||
@@ -3,6 +3,7 @@ import { HttpFlv } from '../types/liveBackendJson';
|
||||
import { getNotificationQueue } from '../workers';
|
||||
import client from '../services/slackNotifier';
|
||||
import type { paths } from '../types/mediamtx.d.ts';
|
||||
import { MEDIAMTX_SERVER_REGIONS } from '../utils/mediamtx/server';
|
||||
|
||||
export default async function runner() {
|
||||
// if there are no users it explodes so yeah
|
||||
@@ -49,37 +50,43 @@ export async function initializeStreamInfo(channelId?: string) {
|
||||
|
||||
export async function syncStream() {
|
||||
try {
|
||||
const response = await fetch(`${process.env.MEDIAMTX_API}/v3/paths/list?itemsPerPage=1000`);
|
||||
const regions = Object.keys(MEDIAMTX_SERVER_REGIONS) as Array<
|
||||
keyof typeof MEDIAMTX_SERVER_REGIONS
|
||||
>;
|
||||
|
||||
if (!response.ok) {
|
||||
console.error(`Failed to fetch stream stats: ${response.status} ${response.statusText}`);
|
||||
return;
|
||||
const allActiveStreams = new Map<string, keyof typeof MEDIAMTX_SERVER_REGIONS>();
|
||||
|
||||
for (const r of regions) {
|
||||
const region = MEDIAMTX_SERVER_REGIONS[r];
|
||||
const response = await fetch(`${region.apiUrl}/v3/paths/list?itemsPerPage=1000`);
|
||||
|
||||
if (!response.ok) {
|
||||
console.error(
|
||||
`Failed to fetch ${r} stream stats: ${response.status} ${response.statusText}`
|
||||
);
|
||||
continue;
|
||||
}
|
||||
|
||||
type ResponseType =
|
||||
paths['/v3/paths/list']['get']['responses']['200']['content']['application/json'];
|
||||
const data = (await response.json()) as ResponseType;
|
||||
|
||||
if (data?.items) {
|
||||
for (const stream of data.items) {
|
||||
if (stream.ready && stream.name) {
|
||||
allActiveStreams.set(stream.name, r);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
type ResponseType = paths['/v3/paths/list']['get']['responses']['200']['content']['application/json'];
|
||||
const data = await response.json() as ResponseType;
|
||||
|
||||
if (!data) {
|
||||
return;
|
||||
}
|
||||
|
||||
const activeStreams = data.items!;
|
||||
|
||||
// handle streams going offline
|
||||
const currentLiveStreams = await prisma.streamInfo.findMany({
|
||||
where: { isLive: true },
|
||||
});
|
||||
|
||||
const activeStreamMap = new Map();
|
||||
for (const stream of activeStreams) {
|
||||
activeStreamMap.set(stream.name, {
|
||||
isLive: stream.ready,
|
||||
});
|
||||
}
|
||||
|
||||
for (const dbStream of currentLiveStreams) {
|
||||
const streamStats = activeStreamMap.get(dbStream.username);
|
||||
|
||||
if (!streamStats || !streamStats.isLive) {
|
||||
if (!allActiveStreams.has(dbStream.username)) {
|
||||
await prisma.streamInfo.update({
|
||||
where: { username: dbStream.username },
|
||||
data: {
|
||||
@@ -91,50 +98,52 @@ export async function syncStream() {
|
||||
}
|
||||
}
|
||||
|
||||
for (const stream of activeStreams) {
|
||||
if (stream.ready) {
|
||||
const existingStream = await prisma.streamInfo.findUnique({
|
||||
where: { username: stream.name },
|
||||
include: { channel: true },
|
||||
// handle streams going online
|
||||
for (const [username, regionKey] of allActiveStreams) {
|
||||
const existingStream = await prisma.streamInfo.findUnique({
|
||||
where: { username },
|
||||
include: { channel: true },
|
||||
});
|
||||
|
||||
if (existingStream && !existingStream.isLive) {
|
||||
console.log(`Stream ${username} is now live in region ${regionKey}`);
|
||||
await prisma.streamInfo.update({
|
||||
where: { username },
|
||||
data: {
|
||||
isLive: true,
|
||||
startedAt: new Date(),
|
||||
streamRegion: regionKey,
|
||||
},
|
||||
});
|
||||
|
||||
if (existingStream && !existingStream.isLive) {
|
||||
await prisma.streamInfo.update({
|
||||
where: { username: stream.name },
|
||||
data: {
|
||||
isLive: true,
|
||||
startedAt: new Date(),
|
||||
},
|
||||
});
|
||||
const subscribedFollowers = await prisma.follow.findMany({
|
||||
where: {
|
||||
channelId: existingStream.channelId,
|
||||
notifyStream: true,
|
||||
},
|
||||
include: {
|
||||
user: true,
|
||||
},
|
||||
});
|
||||
|
||||
const subscribedFollowers = await prisma.follow.findMany({
|
||||
where: {
|
||||
channelId: existingStream.channelId,
|
||||
notifyStream: true,
|
||||
},
|
||||
include: {
|
||||
user: true,
|
||||
},
|
||||
});
|
||||
|
||||
const queue = getNotificationQueue();
|
||||
const queue = getNotificationQueue();
|
||||
|
||||
if (!existingStream.channel.is247) {
|
||||
queue.add(`streamStartChannel:${existingStream.username}`, {
|
||||
text: `${existingStream.username} is now *live*, streaming *${existingStream.title}* (${existingStream.category})!\n<https://hackclub.tv/${existingStream.username}|Go check them out>`,
|
||||
channel: process.env.NOTIFICATION_CHANNEL_ID!,
|
||||
if (!existingStream.channel.is247) {
|
||||
queue.add(`streamStartChannel:${existingStream.username}`, {
|
||||
text: `${existingStream.username} is now *live*, streaming *${existingStream.title}* (${existingStream.category})!\n<https://hackclub.tv/${existingStream.username}|Go check them out>`,
|
||||
channel: process.env.NOTIFICATION_CHANNEL_ID!,
|
||||
unfurl_links: true,
|
||||
});
|
||||
}
|
||||
|
||||
if (existingStream.enableNotifications && !existingStream.channel.is247) {
|
||||
for (const follower of subscribedFollowers) {
|
||||
queue.add(`streamStartDm:${follower.user.id}`, {
|
||||
text: `${existingStream.username} is now *live*, streaming *${existingStream.title}* (${existingStream.category})!\n<https://hackclub.tv/${existingStream.username}|Go check them out>\n_Stream notifications are enabled for this user. If you want to disable them, you can do so in \`Profile > Follows\`._`,
|
||||
channel: follower.user.slack_id,
|
||||
unfurl_links: true,
|
||||
});
|
||||
}
|
||||
if (existingStream.enableNotifications && !existingStream.channel.is247) {
|
||||
for (const follower of subscribedFollowers) {
|
||||
queue.add(`streamStartDm:${follower.user.id}`, {
|
||||
text: `${existingStream.username} is now *live*, streaming *${existingStream.title}* (${existingStream.category})!\n<https://hackclub.tv/${existingStream.username}|Go check them out>\n_Stream notifications are enabled for this user. If you want to disable them, you can do so in \`Profile > Follows\`._`,
|
||||
channel: follower.user.slack_id,
|
||||
unfurl_links: true,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -37,4 +37,9 @@ export function useStreams() {
|
||||
return context
|
||||
}
|
||||
|
||||
export type StreamInfoResponse = (StreamInfo & { channel: Channel })[]
|
||||
export type StreamInfoResponse = (StreamInfo & {
|
||||
channel: Channel & {
|
||||
isRestricted?: boolean;
|
||||
restrictionExpiresAt?: string | null;
|
||||
};
|
||||
})[]
|
||||
@@ -1,5 +1,5 @@
|
||||
import { utapi } from "../services/uploadthing/server";
|
||||
import sharp from "sharp";
|
||||
import sharp from 'sharp';
|
||||
import { utapi } from '../services/uploadthing/server';
|
||||
|
||||
export async function genIdenticonUpload(str: string, type?: string) {
|
||||
const identicon = await fetch(`https://api.dicebear.com/9.x/identicon/svg?seed=${str}&size=256`);
|
||||
@@ -7,13 +7,13 @@ export async function genIdenticonUpload(str: string, type?: string) {
|
||||
.webp({ quality: 80 })
|
||||
.toBuffer();
|
||||
|
||||
const file = new File([webpBuffer], `${str}${type ? `-${type}` : ""}.webp`, {
|
||||
type: "image/webp",
|
||||
const file = new File([new Uint8Array(webpBuffer)], `${str}${type ? `-${type}` : ''}.webp`, {
|
||||
type: 'image/webp',
|
||||
});
|
||||
const ul = await utapi.uploadFiles(file);
|
||||
if (ul.error) {
|
||||
throw new Error("Failed to upload identicon: " + ul.error);
|
||||
throw new Error('Failed to upload identicon: ' + ul.error);
|
||||
}
|
||||
|
||||
return ul.data?.ufsUrl
|
||||
}
|
||||
return ul.data?.ufsUrl;
|
||||
}
|
||||
|
||||
29
apps/web/src/lib/utils/mediamtx/client.ts
Normal file
29
apps/web/src/lib/utils/mediamtx/client.ts
Normal file
@@ -0,0 +1,29 @@
|
||||
import { MediaMTXRegion } from './regions';
|
||||
import { getEnv } from '@/lib/env';
|
||||
|
||||
export interface MediaMTXClientEnvs {
|
||||
publicUrl: string;
|
||||
ingestRoute: string;
|
||||
emoji: string;
|
||||
string: string;
|
||||
}
|
||||
|
||||
export function getMediamtxClientEnvs(region: MediaMTXRegion = 'hq'): MediaMTXClientEnvs {
|
||||
const envs: Record<MediaMTXRegion, MediaMTXClientEnvs> = {
|
||||
hq: {
|
||||
publicUrl: getEnv('NEXT_PUBLIC_MEDIAMTX_URL_HQ')!,
|
||||
ingestRoute: getEnv('NEXT_PUBLIC_MEDIAMTX_INGEST_ROUTE_HQ')!,
|
||||
emoji: '🇺🇸',
|
||||
string: 'HQ Server A',
|
||||
},
|
||||
};
|
||||
|
||||
const regionEnvs = envs[region];
|
||||
|
||||
if (!regionEnvs) {
|
||||
throw new Error(`Invalid MediaMTX region: ${region}`);
|
||||
}
|
||||
|
||||
return regionEnvs;
|
||||
}
|
||||
|
||||
1
apps/web/src/lib/utils/mediamtx/regions.ts
Normal file
1
apps/web/src/lib/utils/mediamtx/regions.ts
Normal file
@@ -0,0 +1 @@
|
||||
export type MediaMTXRegion = 'hq';
|
||||
21
apps/web/src/lib/utils/mediamtx/server.ts
Normal file
21
apps/web/src/lib/utils/mediamtx/server.ts
Normal file
@@ -0,0 +1,21 @@
|
||||
import { MediaMTXRegion } from './regions';
|
||||
|
||||
export interface MediaMTXEnvs {
|
||||
apiUrl: string;
|
||||
}
|
||||
|
||||
export const MEDIAMTX_SERVER_REGIONS: Record<MediaMTXRegion, MediaMTXEnvs> = {
|
||||
hq: {
|
||||
apiUrl: process.env.MEDIAMTX_API_HQ!,
|
||||
},
|
||||
};
|
||||
|
||||
export function getMediamtxEnvs(region: MediaMTXRegion = 'hq'): MediaMTXEnvs {
|
||||
const envs = MEDIAMTX_SERVER_REGIONS[region];
|
||||
|
||||
if (!envs) {
|
||||
throw new Error(`Invalid MediaMTX region: ${region}`);
|
||||
}
|
||||
|
||||
return envs;
|
||||
}
|
||||
@@ -3,6 +3,7 @@ import { getRedisConnection } from '@hctv/db';
|
||||
import { promisify } from 'node:util';
|
||||
import { existsSync } from 'node:fs';
|
||||
import { exec as execCallback } from 'node:child_process';
|
||||
import { getMediamtxClientEnvs } from '@/lib/utils/mediamtx/client';
|
||||
const pExec = promisify(execCallback);
|
||||
|
||||
const globalForWorker = global as unknown as {
|
||||
@@ -26,7 +27,10 @@ export async function registerThumbnailWorker(): Promise<void> {
|
||||
try {
|
||||
// this is totally unnecessary, but i'll keep it for security purposes.
|
||||
const name = job.data.name.replace(/[^a-zA-Z0-9]/g, '_');
|
||||
const m3u8location = `${process.env.NEXT_PUBLIC_MEDIAMTX_URL}/${name}/index.m3u8`;
|
||||
const server = job.data.server || 'hq';
|
||||
const srvValues = getMediamtxClientEnvs(server);
|
||||
|
||||
const m3u8location = `${srvValues.publicUrl}/${name}/index.m3u8`;
|
||||
const thumbDir = '/dev/shm/hctv-thumb';
|
||||
|
||||
if (!existsSync(thumbDir)) {
|
||||
@@ -41,11 +45,11 @@ export async function registerThumbnailWorker(): Promise<void> {
|
||||
);
|
||||
return { success: true };
|
||||
} catch (ffmpegError) {
|
||||
console.error(`FFmpeg error for ${name}:`, ffmpegError);
|
||||
console.error(`FFmpeg error for ${name} on server ${server}:`, ffmpegError);
|
||||
return { success: false, error: ffmpegError instanceof Error ? ffmpegError.message : String(ffmpegError) };
|
||||
}
|
||||
} catch (e) {
|
||||
console.error('Slack notification failed:', e);
|
||||
console.error('Thumbnail generation failed:', e);
|
||||
// @ts-ignore e is unknown
|
||||
return { success: false, error: e.message };
|
||||
}
|
||||
|
||||
65
compose.yml
Normal file
65
compose.yml
Normal file
@@ -0,0 +1,65 @@
|
||||
services:
|
||||
hctv:
|
||||
container_name: hctv
|
||||
depends_on:
|
||||
postgres:
|
||||
condition: service_healthy
|
||||
pgbouncer:
|
||||
condition: service_started
|
||||
env_file:
|
||||
- .env
|
||||
build:
|
||||
context: .
|
||||
dockerfile: apps/web/Dockerfile
|
||||
chat:
|
||||
depends_on:
|
||||
postgres:
|
||||
condition: service_healthy
|
||||
hctv:
|
||||
condition: service_started
|
||||
env_file:
|
||||
- .env
|
||||
build:
|
||||
context: .
|
||||
dockerfile: apps/chat/Dockerfile
|
||||
postgres:
|
||||
image: 'postgres:17-alpine'
|
||||
ports:
|
||||
- '6767:5432'
|
||||
environment:
|
||||
POSTGRES_USER: postgres
|
||||
POSTGRES_PASSWORD: '${PG_PASS}'
|
||||
POSTGRES_DB: hctv
|
||||
volumes:
|
||||
- 'hctv_pgdata:/var/lib/postgresql/data'
|
||||
healthcheck:
|
||||
test:
|
||||
- CMD-SHELL
|
||||
- 'pg_isready -U postgres'
|
||||
interval: 5s
|
||||
timeout: 5s
|
||||
retries: 5
|
||||
pgbouncer:
|
||||
image: 'bitnamilegacy/pgbouncer:1'
|
||||
environment:
|
||||
- POSTGRESQL_HOST=postgres
|
||||
- POSTGRESQL_PORT=5432
|
||||
- POSTGRESQL_USERNAME=postgres
|
||||
- 'POSTGRESQL_PASSWORD=${PG_PASS}'
|
||||
- PGBOUNCER_DATABASE=hctv
|
||||
- POSTGRESQL_DATABASE=hctv
|
||||
- PGBOUNCER_POOL_MODE=transaction
|
||||
- PGBOUNCER_MAX_CLIENT_CONN=100
|
||||
- PGBOUNCER_DEFAULT_POOL_SIZE=20
|
||||
depends_on:
|
||||
- postgres
|
||||
redis:
|
||||
image: 'redis:7.4-alpine'
|
||||
volumes:
|
||||
- 'hctv_redis:/data'
|
||||
mediamtx:
|
||||
image: 'bluenviron/mediamtx:latest'
|
||||
ports:
|
||||
- '8890:8890/udp'
|
||||
volumes:
|
||||
- './mediamtx.yml:/mediamtx.yml'
|
||||
@@ -1,12 +1,12 @@
|
||||
services:
|
||||
psql:
|
||||
image: postgres
|
||||
image: postgres:18-alpine
|
||||
environment:
|
||||
POSTGRES_USER: postgres
|
||||
# my condolences
|
||||
POSTGRES_PASSWORD: skbiditoilet
|
||||
volumes:
|
||||
- ./psql:/var/lib/postgresql/data
|
||||
- ./psql:/var/lib/postgresql
|
||||
ports:
|
||||
- 5555:5432
|
||||
redis:
|
||||
@@ -22,4 +22,16 @@ services:
|
||||
- 8891:8888
|
||||
- 9997:9997
|
||||
volumes:
|
||||
- ./mediamtx.yml:/mediamtx.yml
|
||||
- ./mediamtx.yml:/mediamtx.yml
|
||||
extra_hosts:
|
||||
- "host.docker.internal:host-gateway"
|
||||
# mediamtx2:
|
||||
# image: bluenviron/mediamtx:latest
|
||||
# ports:
|
||||
# - 8990:8890/udp
|
||||
# - 8991:8891
|
||||
# - 9999:9997
|
||||
# volumes:
|
||||
# - ./mediamtx.yml:/mediamtx.yml
|
||||
# extra_hosts:
|
||||
# - "host.docker.internal:host-gateway"
|
||||
@@ -8,6 +8,6 @@ srtAddress: :8890
|
||||
hls: yes
|
||||
|
||||
authMethod: http
|
||||
authHTTPAddress: http://192.168.1.47:3000/api/mediamtx/publish
|
||||
authHTTPAddress: http://host.docker.internal:3000/api/mediamtx/publish
|
||||
|
||||
api: yes
|
||||
api: yes
|
||||
|
||||
12
package.json
12
package.json
@@ -14,14 +14,16 @@
|
||||
"docker:web": "dotenvx run -f .env.docker -- docker buildx build --platform linux/amd64 -f apps/web/Dockerfile . --secret id=TURBO_TOKEN,env=TURBO_TOKEN --secret id=TURBO_TEAM,env=TURBO_TEAM --no-cache",
|
||||
"docker:chat": "dotenvx run -f .env.docker -- docker buildx build --platform linux/amd64 -f apps/chat/Dockerfile . --secret id=TURBO_TOKEN,env=TURBO_TOKEN --secret id=TURBO_TEAM,env=TURBO_TEAM --no-cache",
|
||||
"act": "act --secret-file .env.ci",
|
||||
"db:migrate": "yarn workspace @hctv/db db:migrate",
|
||||
"ui:add": "yarn workspace @hctv/web ui:add",
|
||||
"prisma": "yarn workspace @hctv/db prisma",
|
||||
"r:rtmp": "docker compose -f dev/docker-compose.yml restart nginx-rtmp -t 0"
|
||||
"db:migrate": "pnpm --filter=@hctv/db db:migrate",
|
||||
"ui:add": "pnpm --filter=@hctv/web ui:add",
|
||||
"prisma": "pnpm --filter=@hctv/db prisma",
|
||||
"r:rtmp": "docker compose -f dev/docker-compose.yml restart nginx-rtmp -t 0",
|
||||
"sdk:test": "dotenvx run -f .env.sdk -- pnpm --filter=@hctv/sdk test",
|
||||
"sdk:example": "dotenvx run -f .env.sdk -- pnpm --filter=@hctv/sdk example"
|
||||
},
|
||||
"devDependencies": {
|
||||
"prettier": "^3.6.2",
|
||||
"turbo": "^2.4.4"
|
||||
},
|
||||
"packageManager": "yarn@1.22.22+sha512.a6b2f7906b721bba3d67d4aff083df04dad64c399707841b7acf00f6b133b7ac24255f2652fa22ae3534329dc6180534e98d17432037ff6fd140556e2bb3137e"
|
||||
"packageManager": "pnpm@10.6.5"
|
||||
}
|
||||
|
||||
@@ -14,7 +14,7 @@
|
||||
"dev": "tsc --watch --preserveWatchOutput"
|
||||
},
|
||||
"dependencies": {
|
||||
"@hctv/db": "*",
|
||||
"@hctv/db": "workspace:*",
|
||||
"@lucia-auth/adapter-prisma": "^4.0.1",
|
||||
"arctic": "^3.1.1",
|
||||
"lucia": "^3.2.2"
|
||||
|
||||
@@ -10,9 +10,9 @@ export const hackClub = new OAuth2Client(
|
||||
process.env.HCID_REDIRECT_URI!
|
||||
);
|
||||
|
||||
export const HCID_AUTH_URL = "https://account.hackclub.com/oauth/authorize";
|
||||
export const HCID_TOKEN_URL = "https://account.hackclub.com/oauth/token";
|
||||
export const HCID_USER_INFO_URL = "https://account.hackclub.com/api/v1/me";
|
||||
export const HCID_AUTH_URL = "https://auth.hackclub.com/oauth/authorize";
|
||||
export const HCID_TOKEN_URL = "https://auth.hackclub.com/oauth/token";
|
||||
export const HCID_USER_INFO_URL = "https://auth.hackclub.com/api/v1/me";
|
||||
|
||||
export const lucia = new Lucia(adapter, {
|
||||
sessionCookie: {
|
||||
@@ -31,6 +31,7 @@ export const lucia = new Lucia(adapter, {
|
||||
pfpUrl: attributes.pfpUrl,
|
||||
hasOnboarded: attributes.hasOnboarded,
|
||||
personalChannelId: attributes.personalChannelId,
|
||||
isAdmin: attributes.isAdmin,
|
||||
};
|
||||
},
|
||||
});
|
||||
@@ -48,4 +49,5 @@ interface DatabaseUserAttributes {
|
||||
pfpUrl: string;
|
||||
hasOnboarded: boolean;
|
||||
personalChannelId: string | null;
|
||||
isAdmin: boolean;
|
||||
}
|
||||
|
||||
2
packages/db/.env.example
Normal file
2
packages/db/.env.example
Normal file
@@ -0,0 +1,2 @@
|
||||
DATABASE_URL=postgresql://postgres:skbiditoilet@localhost:5555/postgres
|
||||
DATABASE_DIRECT_URL=postgresql://postgres:skbiditoilet@localhost:5555/postgres
|
||||
@@ -10,9 +10,9 @@
|
||||
},
|
||||
"type": "module",
|
||||
"dependencies": {
|
||||
"@prisma/client": "^6.5.0",
|
||||
"@prisma/client": "6.5.0",
|
||||
"ioredis": "5.7.0",
|
||||
"prisma": "^6.5.0"
|
||||
"prisma": "6.5.0"
|
||||
},
|
||||
"scripts": {
|
||||
"db:generate": "prisma generate",
|
||||
|
||||
@@ -0,0 +1,2 @@
|
||||
-- AlterTable
|
||||
ALTER TABLE "User" ADD COLUMN "isAdmin" BOOLEAN NOT NULL DEFAULT false;
|
||||
@@ -0,0 +1,41 @@
|
||||
-- CreateTable
|
||||
CREATE TABLE "UserBan" (
|
||||
"id" TEXT NOT NULL,
|
||||
"userId" TEXT NOT NULL,
|
||||
"reason" TEXT NOT NULL,
|
||||
"bannedBy" TEXT NOT NULL,
|
||||
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||
"expiresAt" TIMESTAMP(3),
|
||||
|
||||
CONSTRAINT "UserBan_pkey" PRIMARY KEY ("id")
|
||||
);
|
||||
|
||||
-- CreateTable
|
||||
CREATE TABLE "ChannelRestriction" (
|
||||
"id" TEXT NOT NULL,
|
||||
"channelId" TEXT NOT NULL,
|
||||
"reason" TEXT NOT NULL,
|
||||
"restrictedBy" TEXT NOT NULL,
|
||||
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||
"expiresAt" TIMESTAMP(3),
|
||||
|
||||
CONSTRAINT "ChannelRestriction_pkey" PRIMARY KEY ("id")
|
||||
);
|
||||
|
||||
-- CreateIndex
|
||||
CREATE UNIQUE INDEX "UserBan_userId_key" ON "UserBan"("userId");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "UserBan_userId_idx" ON "UserBan"("userId");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE UNIQUE INDEX "ChannelRestriction_channelId_key" ON "ChannelRestriction"("channelId");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChannelRestriction_channelId_idx" ON "ChannelRestriction"("channelId");
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "UserBan" ADD CONSTRAINT "UserBan_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChannelRestriction" ADD CONSTRAINT "ChannelRestriction_channelId_fkey" FOREIGN KEY ("channelId") REFERENCES "Channel"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
@@ -0,0 +1,2 @@
|
||||
-- AlterTable
|
||||
ALTER TABLE "StreamInfo" ADD COLUMN "streamRegion" TEXT NOT NULL DEFAULT 'eu';
|
||||
@@ -0,0 +1,2 @@
|
||||
-- AlterTable
|
||||
ALTER TABLE "Channel" ADD COLUMN "nameLastChanged" TIMESTAMP(3);
|
||||
@@ -0,0 +1,2 @@
|
||||
-- AlterTable
|
||||
ALTER TABLE "StreamInfo" ALTER COLUMN "streamRegion" SET DEFAULT 'hq';
|
||||
@@ -0,0 +1 @@
|
||||
UPDATE "StreamInfo" SET "streamRegion" = 'hq' WHERE "streamRegion" = 'eu';
|
||||
@@ -0,0 +1,96 @@
|
||||
-- CreateEnum
|
||||
CREATE TYPE "ChatModerationAction" AS ENUM (
|
||||
'MESSAGE_BLOCKED',
|
||||
'MESSAGE_DELETED',
|
||||
'USER_TIMEOUT',
|
||||
'USER_BANNED',
|
||||
'USER_UNBANNED'
|
||||
);
|
||||
|
||||
-- CreateTable
|
||||
CREATE TABLE "ChatModerationSettings" (
|
||||
"id" TEXT NOT NULL,
|
||||
"channelId" TEXT NOT NULL,
|
||||
"blockedTerms" TEXT[] DEFAULT ARRAY[]::TEXT[],
|
||||
"slowModeSeconds" INTEGER NOT NULL DEFAULT 0,
|
||||
"maxMessageLength" INTEGER NOT NULL DEFAULT 400,
|
||||
"rateLimitCount" INTEGER NOT NULL DEFAULT 8,
|
||||
"rateLimitWindowSeconds" INTEGER NOT NULL DEFAULT 10,
|
||||
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||
"updatedAt" TIMESTAMP(3) NOT NULL,
|
||||
|
||||
CONSTRAINT "ChatModerationSettings_pkey" PRIMARY KEY ("id")
|
||||
);
|
||||
|
||||
-- CreateTable
|
||||
CREATE TABLE "ChatUserBan" (
|
||||
"id" TEXT NOT NULL,
|
||||
"channelId" TEXT NOT NULL,
|
||||
"userId" TEXT NOT NULL,
|
||||
"reason" TEXT NOT NULL,
|
||||
"bannedById" TEXT NOT NULL,
|
||||
"expiresAt" TIMESTAMP(3),
|
||||
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||
"updatedAt" TIMESTAMP(3) NOT NULL,
|
||||
|
||||
CONSTRAINT "ChatUserBan_pkey" PRIMARY KEY ("id")
|
||||
);
|
||||
|
||||
-- CreateTable
|
||||
CREATE TABLE "ChatModerationEvent" (
|
||||
"id" TEXT NOT NULL,
|
||||
"channelId" TEXT NOT NULL,
|
||||
"action" "ChatModerationAction" NOT NULL,
|
||||
"moderatorId" TEXT NOT NULL,
|
||||
"targetUserId" TEXT,
|
||||
"reason" TEXT,
|
||||
"details" JSONB,
|
||||
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||
|
||||
CONSTRAINT "ChatModerationEvent_pkey" PRIMARY KEY ("id")
|
||||
);
|
||||
|
||||
-- CreateIndex
|
||||
CREATE UNIQUE INDEX "ChatModerationSettings_channelId_key" ON "ChatModerationSettings"("channelId");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChatModerationSettings_channelId_idx" ON "ChatModerationSettings"("channelId");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE UNIQUE INDEX "ChatUserBan_channelId_userId_key" ON "ChatUserBan"("channelId", "userId");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChatUserBan_channelId_userId_idx" ON "ChatUserBan"("channelId", "userId");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChatUserBan_expiresAt_idx" ON "ChatUserBan"("expiresAt");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChatModerationEvent_channelId_createdAt_idx" ON "ChatModerationEvent"("channelId", "createdAt");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChatModerationEvent_moderatorId_idx" ON "ChatModerationEvent"("moderatorId");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChatModerationEvent_targetUserId_idx" ON "ChatModerationEvent"("targetUserId");
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatModerationSettings" ADD CONSTRAINT "ChatModerationSettings_channelId_fkey" FOREIGN KEY ("channelId") REFERENCES "Channel"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatUserBan" ADD CONSTRAINT "ChatUserBan_channelId_fkey" FOREIGN KEY ("channelId") REFERENCES "Channel"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatUserBan" ADD CONSTRAINT "ChatUserBan_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatUserBan" ADD CONSTRAINT "ChatUserBan_bannedById_fkey" FOREIGN KEY ("bannedById") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatModerationEvent" ADD CONSTRAINT "ChatModerationEvent_channelId_fkey" FOREIGN KEY ("channelId") REFERENCES "Channel"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatModerationEvent" ADD CONSTRAINT "ChatModerationEvent_moderatorId_fkey" FOREIGN KEY ("moderatorId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatModerationEvent" ADD CONSTRAINT "ChatModerationEvent_targetUserId_fkey" FOREIGN KEY ("targetUserId") REFERENCES "User"("id") ON DELETE SET NULL ON UPDATE CASCADE;
|
||||
@@ -0,0 +1,35 @@
|
||||
-- CreateEnum
|
||||
CREATE TYPE "AdminAuditAction" AS ENUM (
|
||||
'USER_BANNED',
|
||||
'USER_UNBANNED',
|
||||
'USER_PROMOTED',
|
||||
'USER_DEMOTED',
|
||||
'CHANNEL_RESTRICTED',
|
||||
'CHANNEL_UNRESTRICTED'
|
||||
);
|
||||
|
||||
-- CreateTable
|
||||
CREATE TABLE "AdminAuditLog" (
|
||||
"id" TEXT NOT NULL,
|
||||
"action" "AdminAuditAction" NOT NULL,
|
||||
"actorId" TEXT NOT NULL,
|
||||
"targetUserId" TEXT,
|
||||
"targetChannel" TEXT,
|
||||
"reason" TEXT,
|
||||
"details" JSONB,
|
||||
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||
|
||||
CONSTRAINT "AdminAuditLog_pkey" PRIMARY KEY ("id")
|
||||
);
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "AdminAuditLog_actorId_idx" ON "AdminAuditLog"("actorId");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "AdminAuditLog_createdAt_idx" ON "AdminAuditLog"("createdAt");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "AdminAuditLog_action_createdAt_idx" ON "AdminAuditLog"("action", "createdAt");
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "AdminAuditLog" ADD CONSTRAINT "AdminAuditLog_actorId_fkey" FOREIGN KEY ("actorId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
@@ -0,0 +1,37 @@
|
||||
-- CreateEnum
|
||||
CREATE TYPE "ChatReportStatus" AS ENUM ('OPEN', 'REVIEWED', 'DISMISSED');
|
||||
|
||||
-- CreateTable
|
||||
CREATE TABLE "ChatUserReport" (
|
||||
"id" TEXT NOT NULL,
|
||||
"channelId" TEXT NOT NULL,
|
||||
"reporterId" TEXT NOT NULL,
|
||||
"targetUserId" TEXT,
|
||||
"targetUsername" TEXT,
|
||||
"reportedMessage" TEXT,
|
||||
"reportedMessageId" TEXT,
|
||||
"reason" TEXT NOT NULL,
|
||||
"status" "ChatReportStatus" NOT NULL DEFAULT 'OPEN',
|
||||
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||
"updatedAt" TIMESTAMP(3) NOT NULL,
|
||||
|
||||
CONSTRAINT "ChatUserReport_pkey" PRIMARY KEY ("id")
|
||||
);
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChatUserReport_channelId_createdAt_idx" ON "ChatUserReport"("channelId", "createdAt");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChatUserReport_reporterId_createdAt_idx" ON "ChatUserReport"("reporterId", "createdAt");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChatUserReport_status_createdAt_idx" ON "ChatUserReport"("status", "createdAt");
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatUserReport" ADD CONSTRAINT "ChatUserReport_channelId_fkey" FOREIGN KEY ("channelId") REFERENCES "Channel"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatUserReport" ADD CONSTRAINT "ChatUserReport_reporterId_fkey" FOREIGN KEY ("reporterId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatUserReport" ADD CONSTRAINT "ChatUserReport_targetUserId_fkey" FOREIGN KEY ("targetUserId") REFERENCES "User"("id") ON DELETE SET NULL ON UPDATE CASCADE;
|
||||
@@ -0,0 +1,30 @@
|
||||
-- AlterEnum
|
||||
ALTER TYPE "AdminAuditAction" ADD VALUE 'REPORT_REVIEWED';
|
||||
ALTER TYPE "AdminAuditAction" ADD VALUE 'REPORT_DISMISSED';
|
||||
ALTER TYPE "AdminAuditAction" ADD VALUE 'REPORT_ENFORCEMENT';
|
||||
|
||||
-- CreateEnum
|
||||
CREATE TYPE "ChatReportAction" AS ENUM (
|
||||
'REVIEW',
|
||||
'DISMISS',
|
||||
'DELETE_REPORTED_MESSAGE',
|
||||
'TIMEOUT_10M',
|
||||
'TIMEOUT_1H',
|
||||
'BAN_CHAT',
|
||||
'LIFT_CHAT_BAN',
|
||||
'BAN_PLATFORM',
|
||||
'UNBAN_PLATFORM'
|
||||
);
|
||||
|
||||
-- AlterTable
|
||||
ALTER TABLE "ChatUserReport"
|
||||
ADD COLUMN "handledById" TEXT,
|
||||
ADD COLUMN "handledAt" TIMESTAMP(3),
|
||||
ADD COLUMN "handlingNote" TEXT,
|
||||
ADD COLUMN "lastAction" "ChatReportAction";
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "ChatUserReport_handledById_handledAt_idx" ON "ChatUserReport"("handledById", "handledAt");
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "ChatUserReport" ADD CONSTRAINT "ChatUserReport_handledById_fkey" FOREIGN KEY ("handledById") REFERENCES "User"("id") ON DELETE SET NULL ON UPDATE CASCADE;
|
||||
@@ -0,0 +1,33 @@
|
||||
-- CreateTable
|
||||
CREATE TABLE "_ChannelChatModerators" (
|
||||
"A" TEXT NOT NULL,
|
||||
"B" TEXT NOT NULL,
|
||||
|
||||
CONSTRAINT "_ChannelChatModerators_AB_pkey" PRIMARY KEY ("A","B")
|
||||
);
|
||||
|
||||
-- CreateTable
|
||||
CREATE TABLE "_ChannelChatBotModerators" (
|
||||
"A" TEXT NOT NULL,
|
||||
"B" TEXT NOT NULL,
|
||||
|
||||
CONSTRAINT "_ChannelChatBotModerators_AB_pkey" PRIMARY KEY ("A","B")
|
||||
);
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "_ChannelChatModerators_B_index" ON "_ChannelChatModerators"("B");
|
||||
|
||||
-- CreateIndex
|
||||
CREATE INDEX "_ChannelChatBotModerators_B_index" ON "_ChannelChatBotModerators"("B");
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "_ChannelChatModerators" ADD CONSTRAINT "_ChannelChatModerators_A_fkey" FOREIGN KEY ("A") REFERENCES "Channel"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "_ChannelChatModerators" ADD CONSTRAINT "_ChannelChatModerators_B_fkey" FOREIGN KEY ("B") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "_ChannelChatBotModerators" ADD CONSTRAINT "_ChannelChatBotModerators_A_fkey" FOREIGN KEY ("A") REFERENCES "BotAccount"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
|
||||
-- AddForeignKey
|
||||
ALTER TABLE "_ChannelChatBotModerators" ADD CONSTRAINT "_ChannelChatBotModerators_B_fkey" FOREIGN KEY ("B") REFERENCES "Channel"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||
@@ -1,3 +1,3 @@
|
||||
# Please do not edit this file manually
|
||||
# It should be added in your version-control system (e.g., Git)
|
||||
provider = "postgresql"
|
||||
provider = "postgresql"
|
||||
|
||||
@@ -17,21 +17,33 @@ datasource db {
|
||||
}
|
||||
|
||||
model User {
|
||||
id String @id @default(cuid())
|
||||
slack_id String
|
||||
email String?
|
||||
pfpUrl String
|
||||
id String @id @default(cuid())
|
||||
slack_id String
|
||||
email String?
|
||||
pfpUrl String
|
||||
|
||||
hasOnboarded Boolean @default(false)
|
||||
isAdmin Boolean @default(false)
|
||||
|
||||
personalChannel Channel? @relation("PersonalChannel", fields: [personalChannelId], references: [id])
|
||||
personalChannelId String? @unique
|
||||
|
||||
ownedChannels Channel[] @relation("ChannelOwner")
|
||||
managedChannels Channel[] @relation("ChannelManagers")
|
||||
chatModeratedChannels Channel[] @relation("ChannelChatModerators")
|
||||
sessions Session[]
|
||||
streams StreamInfo[]
|
||||
followers Follow[] @relation("UserFollows")
|
||||
botAccounts BotAccount[]
|
||||
ban UserBan?
|
||||
chatBans ChatUserBan[] @relation("ChatBannedUser")
|
||||
issuedChatBans ChatUserBan[] @relation("ChatBannedBy")
|
||||
chatModActions ChatModerationEvent[] @relation("ChatModerationActor")
|
||||
chatModTargets ChatModerationEvent[] @relation("ChatModerationTarget")
|
||||
adminAuditLogs AdminAuditLog[] @relation("AdminAuditActor")
|
||||
chatReportsMade ChatUserReport[] @relation("ChatReportReporter")
|
||||
chatReportsSeen ChatUserReport[] @relation("ChatReportTarget")
|
||||
chatReportsHandled ChatUserReport[] @relation("ChatReportHandler")
|
||||
|
||||
@@index([personalChannelId])
|
||||
}
|
||||
@@ -42,19 +54,27 @@ model Channel {
|
||||
description String @default("A hctv channel")
|
||||
pfpUrl String
|
||||
|
||||
createdAt DateTime @default(now())
|
||||
updatedAt DateTime @updatedAt
|
||||
createdAt DateTime @default(now())
|
||||
updatedAt DateTime @updatedAt
|
||||
nameLastChanged DateTime?
|
||||
|
||||
personalFor User? @relation("PersonalChannel")
|
||||
|
||||
owner User @relation("ChannelOwner", fields: [ownerId], references: [id])
|
||||
ownerId String
|
||||
managers User[] @relation("ChannelManagers")
|
||||
chatModerators User[] @relation("ChannelChatModerators")
|
||||
chatModeratorBots BotAccount[] @relation("ChannelChatBotModerators")
|
||||
streamInfo StreamInfo[]
|
||||
followers Follow[] @relation("ChannelFollowers")
|
||||
streamKey StreamKey?
|
||||
obsChatGrantToken String @unique @default(cuid())
|
||||
is247 Boolean @default(false)
|
||||
restriction ChannelRestriction?
|
||||
chatSettings ChatModerationSettings?
|
||||
chatBans ChatUserBan[]
|
||||
chatModEvents ChatModerationEvent[]
|
||||
chatReports ChatUserReport[]
|
||||
|
||||
@@index([ownerId])
|
||||
}
|
||||
@@ -75,6 +95,7 @@ model StreamInfo {
|
||||
category String
|
||||
startedAt DateTime
|
||||
isLive Boolean
|
||||
streamRegion String @default("hq")
|
||||
|
||||
channelId String
|
||||
channel Channel @relation(fields: [channelId], references: [id], onDelete: Cascade)
|
||||
@@ -113,13 +134,14 @@ model StreamKey {
|
||||
}
|
||||
|
||||
model BotAccount {
|
||||
id String @id @default(cuid())
|
||||
id String @id @default(cuid())
|
||||
displayName String
|
||||
slug String @unique
|
||||
description String @default("A hctv bot account")
|
||||
slug String @unique
|
||||
description String @default("A hctv bot account")
|
||||
pfpUrl String
|
||||
owner User @relation(fields: [ownerId], references: [id])
|
||||
owner User @relation(fields: [ownerId], references: [id])
|
||||
ownerId String
|
||||
moderatingChannels Channel[] @relation("ChannelChatBotModerators")
|
||||
apiKeys BotApiKey[]
|
||||
|
||||
createdAt DateTime @default(now())
|
||||
@@ -139,3 +161,159 @@ model BotApiKey {
|
||||
|
||||
@@index([botAccountId])
|
||||
}
|
||||
|
||||
model UserBan {
|
||||
id String @id @default(cuid())
|
||||
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
||||
userId String @unique
|
||||
reason String
|
||||
bannedBy String
|
||||
createdAt DateTime @default(now())
|
||||
expiresAt DateTime?
|
||||
|
||||
@@index([userId])
|
||||
}
|
||||
|
||||
model ChannelRestriction {
|
||||
id String @id @default(cuid())
|
||||
channel Channel @relation(fields: [channelId], references: [id], onDelete: Cascade)
|
||||
channelId String @unique
|
||||
reason String
|
||||
restrictedBy String
|
||||
createdAt DateTime @default(now())
|
||||
expiresAt DateTime?
|
||||
|
||||
@@index([channelId])
|
||||
}
|
||||
|
||||
model ChatModerationSettings {
|
||||
id String @id @default(cuid())
|
||||
channel Channel @relation(fields: [channelId], references: [id], onDelete: Cascade)
|
||||
channelId String @unique
|
||||
blockedTerms String[] @default([])
|
||||
slowModeSeconds Int @default(0)
|
||||
maxMessageLength Int @default(400)
|
||||
rateLimitCount Int @default(8)
|
||||
rateLimitWindowSeconds Int @default(10)
|
||||
createdAt DateTime @default(now())
|
||||
updatedAt DateTime @updatedAt
|
||||
|
||||
@@index([channelId])
|
||||
}
|
||||
|
||||
model ChatUserBan {
|
||||
id String @id @default(cuid())
|
||||
channel Channel @relation(fields: [channelId], references: [id], onDelete: Cascade)
|
||||
channelId String
|
||||
user User @relation("ChatBannedUser", fields: [userId], references: [id], onDelete: Cascade)
|
||||
userId String
|
||||
reason String
|
||||
bannedBy User @relation("ChatBannedBy", fields: [bannedById], references: [id], onDelete: Cascade)
|
||||
bannedById String
|
||||
expiresAt DateTime?
|
||||
createdAt DateTime @default(now())
|
||||
updatedAt DateTime @updatedAt
|
||||
|
||||
@@unique([channelId, userId])
|
||||
@@index([channelId, userId])
|
||||
@@index([expiresAt])
|
||||
}
|
||||
|
||||
model ChatModerationEvent {
|
||||
id String @id @default(cuid())
|
||||
channel Channel @relation(fields: [channelId], references: [id], onDelete: Cascade)
|
||||
channelId String
|
||||
action ChatModerationAction
|
||||
moderator User @relation("ChatModerationActor", fields: [moderatorId], references: [id], onDelete: Cascade)
|
||||
moderatorId String
|
||||
targetUser User? @relation("ChatModerationTarget", fields: [targetUserId], references: [id], onDelete: SetNull)
|
||||
targetUserId String?
|
||||
reason String?
|
||||
details Json?
|
||||
createdAt DateTime @default(now())
|
||||
|
||||
@@index([channelId, createdAt])
|
||||
@@index([moderatorId])
|
||||
@@index([targetUserId])
|
||||
}
|
||||
|
||||
model AdminAuditLog {
|
||||
id String @id @default(cuid())
|
||||
action AdminAuditAction
|
||||
actor User @relation("AdminAuditActor", fields: [actorId], references: [id], onDelete: Cascade)
|
||||
actorId String
|
||||
targetUserId String?
|
||||
targetChannel String?
|
||||
reason String?
|
||||
details Json?
|
||||
createdAt DateTime @default(now())
|
||||
|
||||
@@index([actorId])
|
||||
@@index([createdAt])
|
||||
@@index([action, createdAt])
|
||||
}
|
||||
|
||||
model ChatUserReport {
|
||||
id String @id @default(cuid())
|
||||
channel Channel @relation(fields: [channelId], references: [id], onDelete: Cascade)
|
||||
channelId String
|
||||
reporter User @relation("ChatReportReporter", fields: [reporterId], references: [id], onDelete: Cascade)
|
||||
reporterId String
|
||||
targetUser User? @relation("ChatReportTarget", fields: [targetUserId], references: [id], onDelete: SetNull)
|
||||
targetUserId String?
|
||||
targetUsername String?
|
||||
reportedMessage String?
|
||||
reportedMessageId String?
|
||||
reason String
|
||||
status ChatReportStatus @default(OPEN)
|
||||
handledBy User? @relation("ChatReportHandler", fields: [handledById], references: [id], onDelete: SetNull)
|
||||
handledById String?
|
||||
handledAt DateTime?
|
||||
handlingNote String?
|
||||
lastAction ChatReportAction?
|
||||
createdAt DateTime @default(now())
|
||||
updatedAt DateTime @updatedAt
|
||||
|
||||
@@index([channelId, createdAt])
|
||||
@@index([reporterId, createdAt])
|
||||
@@index([status, createdAt])
|
||||
@@index([handledById, handledAt])
|
||||
}
|
||||
|
||||
enum ChatReportStatus {
|
||||
OPEN
|
||||
REVIEWED
|
||||
DISMISSED
|
||||
}
|
||||
|
||||
enum ChatReportAction {
|
||||
REVIEW
|
||||
DISMISS
|
||||
DELETE_REPORTED_MESSAGE
|
||||
TIMEOUT_10M
|
||||
TIMEOUT_1H
|
||||
BAN_CHAT
|
||||
LIFT_CHAT_BAN
|
||||
BAN_PLATFORM
|
||||
UNBAN_PLATFORM
|
||||
}
|
||||
|
||||
enum AdminAuditAction {
|
||||
USER_BANNED
|
||||
USER_UNBANNED
|
||||
USER_PROMOTED
|
||||
USER_DEMOTED
|
||||
CHANNEL_RESTRICTED
|
||||
CHANNEL_UNRESTRICTED
|
||||
REPORT_REVIEWED
|
||||
REPORT_DISMISSED
|
||||
REPORT_ENFORCEMENT
|
||||
}
|
||||
|
||||
enum ChatModerationAction {
|
||||
MESSAGE_BLOCKED
|
||||
MESSAGE_DELETED
|
||||
USER_TIMEOUT
|
||||
USER_BANNED
|
||||
USER_UNBANNED
|
||||
}
|
||||
|
||||
@@ -27,7 +27,7 @@
|
||||
},
|
||||
"dependencies": {
|
||||
"ws": "^8.17.0",
|
||||
"@hctv/db": "*"
|
||||
"@hctv/db": "workspace:*"
|
||||
},
|
||||
"peerDependencies": {
|
||||
"@hono/node-server": "^1.11.1",
|
||||
|
||||
7
packages/sdk/.eslint.cjs
Normal file
7
packages/sdk/.eslint.cjs
Normal file
@@ -0,0 +1,7 @@
|
||||
/* eslint-env node */
|
||||
module.exports = {
|
||||
extends: ['eslint:recommended', 'plugin:@typescript-eslint/recommended'],
|
||||
parser: '@typescript-eslint/parser',
|
||||
plugins: ['@typescript-eslint'],
|
||||
root: true,
|
||||
};
|
||||
130
packages/sdk/.gitignore
vendored
Normal file
130
packages/sdk/.gitignore
vendored
Normal file
@@ -0,0 +1,130 @@
|
||||
# Logs
|
||||
logs
|
||||
*.log
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
lerna-debug.log*
|
||||
.pnpm-debug.log*
|
||||
|
||||
# Diagnostic reports (https://nodejs.org/api/report.html)
|
||||
report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json
|
||||
|
||||
# Runtime data
|
||||
pids
|
||||
*.pid
|
||||
*.seed
|
||||
*.pid.lock
|
||||
|
||||
# Directory for instrumented libs generated by jscoverage/JSCover
|
||||
lib-cov
|
||||
|
||||
# Coverage directory used by tools like istanbul
|
||||
coverage
|
||||
*.lcov
|
||||
|
||||
# nyc test coverage
|
||||
.nyc_output
|
||||
|
||||
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
|
||||
.grunt
|
||||
|
||||
# Bower dependency directory (https://bower.io/)
|
||||
bower_components
|
||||
|
||||
# node-waf configuration
|
||||
.lock-wscript
|
||||
|
||||
# Compiled binary addons (https://nodejs.org/api/addons.html)
|
||||
build/Release
|
||||
|
||||
# Dependency directories
|
||||
node_modules/
|
||||
jspm_packages/
|
||||
|
||||
# Snowpack dependency directory (https://snowpack.dev/)
|
||||
web_modules/
|
||||
|
||||
# TypeScript cache
|
||||
*.tsbuildinfo
|
||||
|
||||
# Optional npm cache directory
|
||||
.npm
|
||||
|
||||
# Optional eslint cache
|
||||
.eslintcache
|
||||
|
||||
# Optional stylelint cache
|
||||
.stylelintcache
|
||||
|
||||
# Microbundle cache
|
||||
.rpt2_cache/
|
||||
.rts2_cache_cjs/
|
||||
.rts2_cache_es/
|
||||
.rts2_cache_umd/
|
||||
|
||||
# Optional REPL history
|
||||
.node_repl_history
|
||||
|
||||
# Output of 'npm pack'
|
||||
*.tgz
|
||||
|
||||
# Yarn Integrity file
|
||||
.yarn-integrity
|
||||
|
||||
# dotenv environment variable files
|
||||
.env
|
||||
.env.development.local
|
||||
.env.test.local
|
||||
.env.production.local
|
||||
.env.local
|
||||
|
||||
# parcel-bundler cache (https://parceljs.org/)
|
||||
.cache
|
||||
.parcel-cache
|
||||
|
||||
# Next.js build output
|
||||
.next
|
||||
out
|
||||
|
||||
# Nuxt.js build / generate output
|
||||
.nuxt
|
||||
dist
|
||||
|
||||
# Gatsby files
|
||||
.cache/
|
||||
# Comment in the public line in if your project uses Gatsby and not Next.js
|
||||
# https://nextjs.org/blog/next-9-1#public-directory-support
|
||||
# public
|
||||
|
||||
# vuepress build output
|
||||
.vuepress/dist
|
||||
|
||||
# vuepress v2.x temp and cache directory
|
||||
.temp
|
||||
.cache
|
||||
|
||||
# Docusaurus cache and generated files
|
||||
.docusaurus
|
||||
|
||||
# Serverless directories
|
||||
.serverless/
|
||||
|
||||
# FuseBox cache
|
||||
.fusebox/
|
||||
|
||||
# DynamoDB Local files
|
||||
.dynamodb/
|
||||
|
||||
# TernJS port file
|
||||
.tern-port
|
||||
|
||||
# Stores VSCode versions used for testing VSCode extensions
|
||||
.vscode-test
|
||||
|
||||
# yarn v2
|
||||
.yarn/cache
|
||||
.yarn/unplugged
|
||||
.yarn/build-state.yml
|
||||
.yarn/install-state.gz
|
||||
.pnp.*
|
||||
21
packages/sdk/LICENSE
Normal file
21
packages/sdk/LICENSE
Normal file
@@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2024 SrIzan10
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
||||
10
packages/sdk/README.md
Normal file
10
packages/sdk/README.md
Normal file
@@ -0,0 +1,10 @@
|
||||
# @hctv/sdk
|
||||
|
||||
sdk for hctv, the live streaming platform for hack club.
|
||||
|
||||
check https://docs.hackclub.tv for sdk api documentation.
|
||||
check the /examples directory for example usage of the sdk.
|
||||
|
||||
## author
|
||||
|
||||
gpl3 by @srizan10 and claude
|
||||
39
packages/sdk/examples/ai.ts
Normal file
39
packages/sdk/examples/ai.ts
Normal file
@@ -0,0 +1,39 @@
|
||||
import { HctvSdk } from "../src";
|
||||
|
||||
const botToken = process.env.BOT_TOKEN;
|
||||
const aiToken = process.env.AI_TOKEN;
|
||||
if (!botToken) {
|
||||
throw new Error('BOT_TOKEN environment variable is required');
|
||||
}
|
||||
if (!aiToken) {
|
||||
throw new Error('AI_TOKEN environment variable is required');
|
||||
}
|
||||
|
||||
const sdk = new HctvSdk({ botToken })
|
||||
|
||||
await sdk.chat.connect('bot-playground')
|
||||
console.log('connected to the chat!')
|
||||
|
||||
sdk.chat.onMessage(async m => {
|
||||
if (!m.message.startsWith('/ai')) return;
|
||||
const res = await fetch('https://ai.hackclub.com/proxy/v1/chat/completions', {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Authorization': `Bearer ${aiToken}`,
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
body: JSON.stringify({
|
||||
model: 'google/gemini-3-flash-preview',
|
||||
messages: [
|
||||
{ role: 'system', content: 'You are a helpful assistant. Reply concisely and like if you were on a chat platform. Use lowercase.' },
|
||||
{ role: 'user', content: m.message.replace('/ai', '').trim() }
|
||||
],
|
||||
}),
|
||||
});
|
||||
const data = await res.json();
|
||||
const aiMessage = data.choices?.[0]?.message?.content;
|
||||
|
||||
if (aiMessage) {
|
||||
sdk.chat.sendMessage(`@${m.username} ${aiMessage}`);
|
||||
}
|
||||
})
|
||||
27
packages/sdk/examples/moderation-bot.ts
Normal file
27
packages/sdk/examples/moderation-bot.ts
Normal file
@@ -0,0 +1,27 @@
|
||||
import { HctvSdk } from '../src/index.js';
|
||||
|
||||
const botToken = process.env.BOT_TOKEN;
|
||||
const channelName = process.env.CHANNEL_NAME;
|
||||
|
||||
if (!botToken || !channelName) {
|
||||
throw new Error('Set BOT_TOKEN and CHANNEL_NAME');
|
||||
}
|
||||
|
||||
const sdk = new HctvSdk({ botToken });
|
||||
await sdk.chat.connect(channelName);
|
||||
|
||||
sdk.chat.onMessage((message) => {
|
||||
if (message.isBot || !message.userId) return;
|
||||
|
||||
if (message.message.toLowerCase().includes('badword')) {
|
||||
sdk.chat.timeoutUser(channelName, message.userId, message.username, 300, 'Used blocked word');
|
||||
//sdk.chat.sendMessage(`@${message.username} timed out for 5 minutes.`, channelName);
|
||||
}
|
||||
});
|
||||
|
||||
sdk.chat.onSystemMessage((m) => {
|
||||
console.log(`[system] ${m.type}: ${m.message}`);
|
||||
});
|
||||
sdk.chat.onModerationError((err) => {
|
||||
console.log(`[moderation] ${err.code}: ${err.message}`);
|
||||
});
|
||||
39
packages/sdk/examples/multi-channel.ts
Normal file
39
packages/sdk/examples/multi-channel.ts
Normal file
@@ -0,0 +1,39 @@
|
||||
import { HctvSdk } from '../src/index.js';
|
||||
|
||||
const sdk = new HctvSdk({
|
||||
botToken: process.env.BOT_TOKEN!,
|
||||
});
|
||||
|
||||
await sdk.chat.connect('channel1');
|
||||
await sdk.chat.connect('channel2');
|
||||
await sdk.chat.connect('channel3');
|
||||
console.log(`connected to ${sdk.chat.connectedChannels.join(', ')}`);
|
||||
|
||||
// gets messages from all channels I'm connected to
|
||||
sdk.chat.onMessage((message) => {
|
||||
console.log(`[${message.channelName}] ${message.username}: ${message.message}`);
|
||||
});
|
||||
|
||||
// specifically handle messages from channel1
|
||||
sdk.chat.onMessage((message) => {
|
||||
console.log(`ts from channel1: ${message.message}`);
|
||||
}, 'channel1');
|
||||
|
||||
sdk.chat.sendMessage('this is channel1!', 'channel1');
|
||||
sdk.chat.sendMessage('this is channel2!', 'channel2');
|
||||
|
||||
console.log(`connected to channel1? ${sdk.chat.isConnectedTo('channel1')}`);
|
||||
console.log(`connected to channel2? ${sdk.chat.isConnectedTo('channel2')}`);
|
||||
|
||||
// disconnect from channel2 after 5 seconds
|
||||
setTimeout(() => {
|
||||
console.log('disconnecting from channel2...');
|
||||
sdk.chat.disconnect('channel2');
|
||||
console.log(`still connected to: ${sdk.chat.connectedChannels.join(', ')}`);
|
||||
}, 5000);
|
||||
|
||||
// disconnect from all channels
|
||||
setTimeout(() => {
|
||||
console.log('disconnecting from all channels...');
|
||||
sdk.chat.disconnect();
|
||||
}, 10000);
|
||||
45
packages/sdk/package.json
Normal file
45
packages/sdk/package.json
Normal file
@@ -0,0 +1,45 @@
|
||||
{
|
||||
"name": "@hctv/sdk",
|
||||
"version": "0.1.0",
|
||||
"scripts": {
|
||||
"test": "vitest run",
|
||||
"build": "tsup",
|
||||
"dev": "tsup --watch",
|
||||
"example": "sh -c 'bun run examples/${0}.ts'"
|
||||
},
|
||||
"main": "./dist/index.js",
|
||||
"module": "./dist/index.mjs",
|
||||
"types": "./dist/index.d.ts",
|
||||
"exports": {
|
||||
".": {
|
||||
"require": "./dist/index.js",
|
||||
"import": "./dist/index.mjs",
|
||||
"types": "./dist/index.d.ts"
|
||||
}
|
||||
},
|
||||
"files": [
|
||||
"dist",
|
||||
"README.md",
|
||||
"LICENSE",
|
||||
"package.json"
|
||||
],
|
||||
"publishConfig": {
|
||||
"access": "public"
|
||||
},
|
||||
"repository": "https://github.com/SrIzan10/ts-lib-boilerplate.git",
|
||||
"author": "Izan Gil <npm@srizan.dev>",
|
||||
"license": "MIT",
|
||||
"dependencies": {
|
||||
"ws": "^8.18.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/node": "^25.1.0",
|
||||
"@types/ws": "^8.18.1",
|
||||
"@typescript-eslint/eslint-plugin": "^8.50.1",
|
||||
"@typescript-eslint/parser": "^8.50.1",
|
||||
"eslint": "^9.39.2",
|
||||
"tsup": "^8.5.1",
|
||||
"typescript": "^5.6.2",
|
||||
"vitest": "^4.0.16"
|
||||
}
|
||||
}
|
||||
493
packages/sdk/src/chat.ts
Normal file
493
packages/sdk/src/chat.ts
Normal file
@@ -0,0 +1,493 @@
|
||||
// most code here has been written by claude opus 4.5
|
||||
import type {
|
||||
ChatAccessHandler,
|
||||
ChatAccessState,
|
||||
ChatMessage,
|
||||
HistoryHandler,
|
||||
MessageHandler,
|
||||
ModerationCommand,
|
||||
ModerationError,
|
||||
ModerationErrorHandler,
|
||||
ModerationEvent,
|
||||
ModerationEventHandler,
|
||||
ServerChatMessage,
|
||||
SystemMessage,
|
||||
SystemMessageHandler,
|
||||
} from './types';
|
||||
|
||||
const DEFAULT_BASE_URL = 'wss://hackclub.tv/api/stream/chat/ws';
|
||||
const PING_INTERVAL = 5000; // 5 seconds (required to keep CF websocket alive)
|
||||
|
||||
interface ChannelConnection {
|
||||
ws: WebSocket;
|
||||
pingInterval: ReturnType<typeof setInterval>;
|
||||
messageHandlers: Set<MessageHandler>;
|
||||
systemMessageHandlers: Set<SystemMessageHandler>;
|
||||
historyHandlers: Set<HistoryHandler>;
|
||||
moderationErrorHandlers: Set<ModerationErrorHandler>;
|
||||
moderationEventHandlers: Set<ModerationEventHandler>;
|
||||
chatAccessHandlers: Set<ChatAccessHandler>;
|
||||
}
|
||||
|
||||
export class ChatClient {
|
||||
private botToken: string;
|
||||
private baseUrl: string;
|
||||
private connections: Map<string, ChannelConnection> = new Map();
|
||||
// Global handlers (receive messages from all channels)
|
||||
private globalMessageHandlers: Set<MessageHandler> = new Set();
|
||||
private globalSystemMessageHandlers: Set<SystemMessageHandler> = new Set();
|
||||
private globalHistoryHandlers: Set<HistoryHandler> = new Set();
|
||||
private globalModerationErrorHandlers: Set<ModerationErrorHandler> = new Set();
|
||||
private globalModerationEventHandlers: Set<ModerationEventHandler> = new Set();
|
||||
private globalChatAccessHandlers: Set<ChatAccessHandler> = new Set();
|
||||
|
||||
constructor(botToken: string, options?: ChatClientOptions) {
|
||||
this.botToken = botToken;
|
||||
this.baseUrl = options?.baseUrl ?? DEFAULT_BASE_URL;
|
||||
}
|
||||
|
||||
async connect(channelName: string): Promise<void> {
|
||||
if (this.connections.has(channelName)) {
|
||||
return Promise.reject(new Error(`already connected to channel: ${channelName}`));
|
||||
}
|
||||
|
||||
const wsUrl = `${this.baseUrl}/${channelName}?botAuth=${this.botToken}`;
|
||||
|
||||
let ws: WebSocket;
|
||||
if (typeof process !== 'undefined' && process.versions?.node) {
|
||||
const { default: WebSocket } = await import('ws');
|
||||
ws = new WebSocket(wsUrl) as any;
|
||||
} else {
|
||||
ws = new WebSocket(wsUrl);
|
||||
}
|
||||
|
||||
const connection: ChannelConnection = {
|
||||
ws,
|
||||
pingInterval: null as any,
|
||||
messageHandlers: new Set(),
|
||||
systemMessageHandlers: new Set(),
|
||||
historyHandlers: new Set(),
|
||||
moderationErrorHandlers: new Set(),
|
||||
moderationEventHandlers: new Set(),
|
||||
chatAccessHandlers: new Set(),
|
||||
};
|
||||
|
||||
this.connections.set(channelName, connection);
|
||||
|
||||
return this.setupWebSocket(channelName, connection);
|
||||
}
|
||||
|
||||
private setupWebSocket(channelName: string, connection: ChannelConnection): Promise<void> {
|
||||
return new Promise((resolve, reject) => {
|
||||
connection.ws.onopen = () => {
|
||||
const systemMsg: SystemMessage = {
|
||||
type: 'connected',
|
||||
channelName,
|
||||
message: 'Connected',
|
||||
timestamp: Date.now(),
|
||||
};
|
||||
this.emitSystem(systemMsg, connection);
|
||||
this.startPingInterval(channelName, connection);
|
||||
resolve();
|
||||
};
|
||||
|
||||
connection.ws.onmessage = (event) => {
|
||||
const data = JSON.parse(event.data.toString());
|
||||
this.handleMessage(data, channelName, connection);
|
||||
};
|
||||
|
||||
connection.ws.onerror = () => {
|
||||
const systemMsg: SystemMessage = {
|
||||
type: 'error',
|
||||
channelName,
|
||||
message: 'WebSocket error',
|
||||
timestamp: Date.now(),
|
||||
};
|
||||
this.emitSystem(systemMsg, connection);
|
||||
reject(new Error('WebSocket error'));
|
||||
};
|
||||
|
||||
connection.ws.onclose = () => {
|
||||
this.stopPingInterval(connection);
|
||||
const systemMsg: SystemMessage = {
|
||||
type: 'disconnected',
|
||||
channelName,
|
||||
message: 'Disconnected',
|
||||
timestamp: Date.now(),
|
||||
};
|
||||
this.emitSystem(systemMsg, connection);
|
||||
this.connections.delete(channelName);
|
||||
};
|
||||
});
|
||||
}
|
||||
|
||||
private handleMessage(data: any, channelName: string, connection: ChannelConnection): void {
|
||||
// Handle pong response
|
||||
if (data.type === 'pong') {
|
||||
return;
|
||||
}
|
||||
|
||||
// Handle history messages (sent on connection)
|
||||
if (data.type === 'history' && Array.isArray(data.messages)) {
|
||||
const messages: ChatMessage[] = data.messages.map((msg: ServerChatMessage) =>
|
||||
this.parseServerMessage(msg, channelName)
|
||||
);
|
||||
// Emit to channel-specific handlers
|
||||
connection.historyHandlers.forEach((handler) => handler(messages));
|
||||
// Emit to global handlers
|
||||
this.globalHistoryHandlers.forEach((handler) => handler(messages));
|
||||
return;
|
||||
}
|
||||
|
||||
// Handle regular chat messages
|
||||
// Server sends: { user: { id, username, pfpUrl, displayName?, isBot }, message }
|
||||
if (data.user && typeof data.message === 'string') {
|
||||
const chatMessage = this.parseServerMessage(data, channelName);
|
||||
this.emitMessage(chatMessage, connection);
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.type === 'chatAccess') {
|
||||
const access: ChatAccessState = {
|
||||
canSend: Boolean(data.canSend),
|
||||
restriction: data.restriction ?? null,
|
||||
};
|
||||
this.emitChatAccess(access, channelName, connection);
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.type === 'moderationError') {
|
||||
const error: ModerationError = {
|
||||
code: data.code,
|
||||
message: data.message,
|
||||
restriction: data.restriction,
|
||||
};
|
||||
this.emitModerationError(error, channelName, connection);
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.type === 'messageDeleted' && typeof data.msgId === 'string') {
|
||||
const event: ModerationEvent = {
|
||||
type: 'messageDeleted',
|
||||
msgId: data.msgId,
|
||||
channelName,
|
||||
};
|
||||
this.emitModerationEvent(event, connection);
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.type === 'systemMsg' && typeof data.message === 'string') {
|
||||
const systemMsg: SystemMessage = {
|
||||
type: 'connected',
|
||||
channelName,
|
||||
message: data.message,
|
||||
timestamp: Date.now(),
|
||||
};
|
||||
this.emitSystem(systemMsg, connection);
|
||||
return;
|
||||
}
|
||||
|
||||
// Handle emoji responses
|
||||
if (data.type === 'emojiMsgResponse' || data.type === 'emojiSearchResponse') {
|
||||
// Could add emoji handlers in the future
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
private parseServerMessage(msg: ServerChatMessage, channelName: string): ChatMessage {
|
||||
return {
|
||||
id: `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`,
|
||||
msgId: msg.msgId,
|
||||
userId: msg.user.id,
|
||||
channelName,
|
||||
username: msg.user.username,
|
||||
displayName: msg.user.displayName,
|
||||
pfpUrl: msg.user.pfpUrl,
|
||||
message: msg.message,
|
||||
timestamp: Date.now(),
|
||||
type: msg.type === 'systemMsg' ? 'systemMsg' : 'message',
|
||||
isBot: msg.user.isBot || false,
|
||||
};
|
||||
}
|
||||
|
||||
private startPingInterval(channelName: string, connection: ChannelConnection): void {
|
||||
connection.pingInterval = setInterval(() => {
|
||||
if (connection.ws.readyState === WebSocket.OPEN) {
|
||||
connection.ws.send(JSON.stringify({ type: 'ping' }));
|
||||
}
|
||||
}, PING_INTERVAL);
|
||||
}
|
||||
|
||||
private stopPingInterval(connection: ChannelConnection): void {
|
||||
if (connection.pingInterval) {
|
||||
clearInterval(connection.pingInterval);
|
||||
}
|
||||
}
|
||||
|
||||
disconnect(channelName?: string): void {
|
||||
if (channelName) {
|
||||
// Disconnect specific channel
|
||||
const connection = this.connections.get(channelName);
|
||||
if (connection) {
|
||||
this.stopPingInterval(connection);
|
||||
connection.ws.close();
|
||||
this.connections.delete(channelName);
|
||||
}
|
||||
} else {
|
||||
// Disconnect all channels
|
||||
for (const [name, connection] of this.connections) {
|
||||
this.stopPingInterval(connection);
|
||||
connection.ws.close();
|
||||
}
|
||||
this.connections.clear();
|
||||
}
|
||||
}
|
||||
|
||||
sendMessage(message: string, channelName?: string): void {
|
||||
if (channelName) {
|
||||
// Send to specific channel
|
||||
const connection = this.connections.get(channelName);
|
||||
if (!connection || connection.ws.readyState !== WebSocket.OPEN) {
|
||||
throw new Error(`Not connected to channel: ${channelName}`);
|
||||
}
|
||||
connection.ws.send(JSON.stringify({ type: 'message', message }));
|
||||
} else {
|
||||
// Send to first connected channel (backward compatibility)
|
||||
const firstConnection = Array.from(this.connections.values())[0];
|
||||
if (!firstConnection || firstConnection.ws.readyState !== WebSocket.OPEN) {
|
||||
throw new Error('Not connected to any channel');
|
||||
}
|
||||
firstConnection.ws.send(JSON.stringify({ type: 'message', message }));
|
||||
}
|
||||
}
|
||||
|
||||
sendModerationCommand(command: ModerationCommand, channelName: string): void {
|
||||
const connection = this.connections.get(channelName);
|
||||
if (!connection || connection.ws.readyState !== WebSocket.OPEN) {
|
||||
throw new Error(`Not connected to channel: ${channelName}`);
|
||||
}
|
||||
connection.ws.send(JSON.stringify(command));
|
||||
}
|
||||
|
||||
timeoutUser(
|
||||
channelName: string,
|
||||
targetUserId: string,
|
||||
targetUsername: string,
|
||||
durationSeconds = 300,
|
||||
reason?: string
|
||||
): void {
|
||||
this.sendModerationCommand(
|
||||
{
|
||||
type: 'mod:timeoutUser',
|
||||
targetUserId,
|
||||
targetUsername,
|
||||
durationSeconds,
|
||||
reason,
|
||||
},
|
||||
channelName
|
||||
);
|
||||
}
|
||||
|
||||
deleteMessage(channelName: string, msgId: string): void {
|
||||
this.sendModerationCommand(
|
||||
{
|
||||
type: 'mod:deleteMessage',
|
||||
msgId,
|
||||
},
|
||||
channelName
|
||||
);
|
||||
}
|
||||
|
||||
banUser(
|
||||
channelName: string,
|
||||
targetUserId: string,
|
||||
targetUsername: string,
|
||||
reason?: string
|
||||
): void {
|
||||
this.sendModerationCommand(
|
||||
{
|
||||
type: 'mod:banUser',
|
||||
targetUserId,
|
||||
targetUsername,
|
||||
reason,
|
||||
},
|
||||
channelName
|
||||
);
|
||||
}
|
||||
|
||||
liftTimeout(channelName: string, targetUserId: string, targetUsername: string): void {
|
||||
this.sendModerationCommand(
|
||||
{
|
||||
type: 'mod:liftTimeout',
|
||||
targetUserId,
|
||||
targetUsername,
|
||||
},
|
||||
channelName
|
||||
);
|
||||
}
|
||||
|
||||
unbanUser(channelName: string, targetUserId: string, targetUsername: string): void {
|
||||
this.sendModerationCommand(
|
||||
{
|
||||
type: 'mod:unbanUser',
|
||||
targetUserId,
|
||||
targetUsername,
|
||||
},
|
||||
channelName
|
||||
);
|
||||
}
|
||||
|
||||
onMessage(handler: MessageHandler, channelName?: string): () => void {
|
||||
if (channelName) {
|
||||
// Channel-specific handler
|
||||
const connection = this.connections.get(channelName);
|
||||
if (!connection) {
|
||||
throw new Error(`Not connected to channel: ${channelName}`);
|
||||
}
|
||||
connection.messageHandlers.add(handler);
|
||||
return () => connection.messageHandlers.delete(handler);
|
||||
} else {
|
||||
// Global handler (receives from all channels)
|
||||
this.globalMessageHandlers.add(handler);
|
||||
return () => this.globalMessageHandlers.delete(handler);
|
||||
}
|
||||
}
|
||||
|
||||
onSystemMessage(handler: SystemMessageHandler, channelName?: string): () => void {
|
||||
if (channelName) {
|
||||
// Channel-specific handler
|
||||
const connection = this.connections.get(channelName);
|
||||
if (!connection) {
|
||||
throw new Error(`Not connected to channel: ${channelName}`);
|
||||
}
|
||||
connection.systemMessageHandlers.add(handler);
|
||||
return () => connection.systemMessageHandlers.delete(handler);
|
||||
} else {
|
||||
// Global handler (receives from all channels)
|
||||
this.globalSystemMessageHandlers.add(handler);
|
||||
return () => this.globalSystemMessageHandlers.delete(handler);
|
||||
}
|
||||
}
|
||||
|
||||
onHistory(handler: HistoryHandler, channelName?: string): () => void {
|
||||
if (channelName) {
|
||||
// Channel-specific handler
|
||||
const connection = this.connections.get(channelName);
|
||||
if (!connection) {
|
||||
throw new Error(`Not connected to channel: ${channelName}`);
|
||||
}
|
||||
connection.historyHandlers.add(handler);
|
||||
return () => connection.historyHandlers.delete(handler);
|
||||
} else {
|
||||
// Global handler (receives from all channels)
|
||||
this.globalHistoryHandlers.add(handler);
|
||||
return () => this.globalHistoryHandlers.delete(handler);
|
||||
}
|
||||
}
|
||||
|
||||
onModerationError(handler: ModerationErrorHandler, channelName?: string): () => void {
|
||||
if (channelName) {
|
||||
const connection = this.connections.get(channelName);
|
||||
if (!connection) {
|
||||
throw new Error(`Not connected to channel: ${channelName}`);
|
||||
}
|
||||
connection.moderationErrorHandlers.add(handler);
|
||||
return () => connection.moderationErrorHandlers.delete(handler);
|
||||
}
|
||||
|
||||
this.globalModerationErrorHandlers.add(handler);
|
||||
return () => this.globalModerationErrorHandlers.delete(handler);
|
||||
}
|
||||
|
||||
onModerationEvent(handler: ModerationEventHandler, channelName?: string): () => void {
|
||||
if (channelName) {
|
||||
const connection = this.connections.get(channelName);
|
||||
if (!connection) {
|
||||
throw new Error(`Not connected to channel: ${channelName}`);
|
||||
}
|
||||
connection.moderationEventHandlers.add(handler);
|
||||
return () => connection.moderationEventHandlers.delete(handler);
|
||||
}
|
||||
|
||||
this.globalModerationEventHandlers.add(handler);
|
||||
return () => this.globalModerationEventHandlers.delete(handler);
|
||||
}
|
||||
|
||||
onChatAccess(handler: ChatAccessHandler, channelName?: string): () => void {
|
||||
if (channelName) {
|
||||
const connection = this.connections.get(channelName);
|
||||
if (!connection) {
|
||||
throw new Error(`Not connected to channel: ${channelName}`);
|
||||
}
|
||||
connection.chatAccessHandlers.add(handler);
|
||||
return () => connection.chatAccessHandlers.delete(handler);
|
||||
}
|
||||
|
||||
this.globalChatAccessHandlers.add(handler);
|
||||
return () => this.globalChatAccessHandlers.delete(handler);
|
||||
}
|
||||
|
||||
private emitMessage(message: ChatMessage, connection: ChannelConnection): void {
|
||||
// Emit to channel-specific handlers
|
||||
connection.messageHandlers.forEach((handler) => handler(message));
|
||||
// Emit to global handlers
|
||||
this.globalMessageHandlers.forEach((handler) => handler(message));
|
||||
}
|
||||
|
||||
private emitSystem(message: SystemMessage, connection: ChannelConnection): void {
|
||||
// Emit to channel-specific handlers
|
||||
connection.systemMessageHandlers.forEach((handler) => handler(message));
|
||||
// Emit to global handlers
|
||||
this.globalSystemMessageHandlers.forEach((handler) => handler(message));
|
||||
}
|
||||
|
||||
private emitModerationError(
|
||||
error: ModerationError,
|
||||
channelName: string,
|
||||
connection: ChannelConnection
|
||||
): void {
|
||||
connection.moderationErrorHandlers.forEach((handler) => handler(error, channelName));
|
||||
this.globalModerationErrorHandlers.forEach((handler) => handler(error, channelName));
|
||||
}
|
||||
|
||||
private emitModerationEvent(event: ModerationEvent, connection: ChannelConnection): void {
|
||||
connection.moderationEventHandlers.forEach((handler) => handler(event));
|
||||
this.globalModerationEventHandlers.forEach((handler) => handler(event));
|
||||
}
|
||||
|
||||
private emitChatAccess(
|
||||
access: ChatAccessState,
|
||||
channelName: string,
|
||||
connection: ChannelConnection
|
||||
): void {
|
||||
connection.chatAccessHandlers.forEach((handler) => handler(access, channelName));
|
||||
this.globalChatAccessHandlers.forEach((handler) => handler(access, channelName));
|
||||
}
|
||||
|
||||
isConnectedTo(channelName: string): boolean {
|
||||
const connection = this.connections.get(channelName);
|
||||
return connection ? connection.ws.readyState === WebSocket.OPEN : false;
|
||||
}
|
||||
|
||||
get connectedChannels(): string[] {
|
||||
return Array.from(this.connections.keys());
|
||||
}
|
||||
|
||||
get isConnected(): boolean {
|
||||
return (
|
||||
this.connections.size > 0 &&
|
||||
Array.from(this.connections.values()).some((c) => c.ws.readyState === WebSocket.OPEN)
|
||||
);
|
||||
}
|
||||
|
||||
get currentChannel(): string | null {
|
||||
// Return first connected channel for backward compatibility
|
||||
const channels = Array.from(this.connections.keys());
|
||||
return channels.length > 0 ? channels[0] : null;
|
||||
}
|
||||
}
|
||||
|
||||
export interface ChatClientOptions {
|
||||
/** Custom WebSocket base URL (default: wss://hackclub.tv/api/chat) */
|
||||
baseUrl?: string;
|
||||
}
|
||||
31
packages/sdk/src/index.ts
Normal file
31
packages/sdk/src/index.ts
Normal file
@@ -0,0 +1,31 @@
|
||||
import { ChatClient } from './chat.js';
|
||||
|
||||
export class HctvSdk {
|
||||
private botToken: string;
|
||||
public chat: ChatClient;
|
||||
|
||||
constructor(args: ConstructorArgs) {
|
||||
this.botToken = args.botToken;
|
||||
this.chat = new ChatClient(args.botToken, args.chatOptions);
|
||||
}
|
||||
}
|
||||
|
||||
interface ConstructorArgs {
|
||||
botToken: string;
|
||||
chatOptions?: import('./chat.js').ChatClientOptions;
|
||||
}
|
||||
export { ChatClient, type ChatClientOptions } from './chat.js';
|
||||
export type {
|
||||
ChatAccessHandler,
|
||||
ChatAccessState,
|
||||
ChatMessage,
|
||||
HistoryHandler,
|
||||
MessageHandler,
|
||||
ModerationCommand,
|
||||
ModerationError,
|
||||
ModerationErrorHandler,
|
||||
ModerationEvent,
|
||||
ModerationEventHandler,
|
||||
SystemMessage,
|
||||
SystemMessageHandler,
|
||||
} from './types.js';
|
||||
76
packages/sdk/src/types.ts
Normal file
76
packages/sdk/src/types.ts
Normal file
@@ -0,0 +1,76 @@
|
||||
export interface ChatMessage {
|
||||
id: string;
|
||||
msgId?: string;
|
||||
userId?: string;
|
||||
channelName: string;
|
||||
username: string;
|
||||
displayName?: string;
|
||||
pfpUrl?: string;
|
||||
message: string;
|
||||
timestamp: number;
|
||||
type: 'message' | 'systemMsg';
|
||||
isBot: boolean;
|
||||
}
|
||||
|
||||
export interface SystemMessage {
|
||||
type: 'connected' | 'disconnected' | 'error';
|
||||
channelName: string;
|
||||
message: string;
|
||||
timestamp: number;
|
||||
}
|
||||
|
||||
/** Message format received from the server */
|
||||
export interface ServerChatMessage {
|
||||
user: {
|
||||
id: string;
|
||||
username: string;
|
||||
pfpUrl: string;
|
||||
displayName?: string;
|
||||
isBot?: boolean;
|
||||
};
|
||||
message: string;
|
||||
msgId?: string;
|
||||
type?: 'message' | 'systemMsg';
|
||||
}
|
||||
|
||||
export interface ChatAccessState {
|
||||
canSend: boolean;
|
||||
restriction?: {
|
||||
type: 'timeout' | 'ban';
|
||||
reason?: string;
|
||||
expiresAt?: string | null;
|
||||
} | null;
|
||||
}
|
||||
|
||||
export interface ModerationError {
|
||||
code: string;
|
||||
message: string;
|
||||
restriction?: ChatAccessState['restriction'];
|
||||
}
|
||||
|
||||
export interface ModerationEvent {
|
||||
type: 'messageDeleted';
|
||||
msgId: string;
|
||||
channelName: string;
|
||||
}
|
||||
|
||||
export interface ModerationCommand {
|
||||
type:
|
||||
| 'mod:deleteMessage'
|
||||
| 'mod:timeoutUser'
|
||||
| 'mod:banUser'
|
||||
| 'mod:unbanUser'
|
||||
| 'mod:liftTimeout';
|
||||
msgId?: string;
|
||||
targetUserId?: string;
|
||||
targetUsername?: string;
|
||||
durationSeconds?: number;
|
||||
reason?: string;
|
||||
}
|
||||
|
||||
export type MessageHandler = (message: ChatMessage) => void;
|
||||
export type SystemMessageHandler = (message: SystemMessage) => void;
|
||||
export type HistoryHandler = (messages: ChatMessage[]) => void;
|
||||
export type ChatAccessHandler = (state: ChatAccessState, channelName: string) => void;
|
||||
export type ModerationErrorHandler = (error: ModerationError, channelName: string) => void;
|
||||
export type ModerationEventHandler = (event: ModerationEvent) => void;
|
||||
1041
packages/sdk/tests/chat.test.ts
Normal file
1041
packages/sdk/tests/chat.test.ts
Normal file
File diff suppressed because it is too large
Load Diff
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user