Initial commit: Music Hub collaboration platform

Full-stack music production collaboration tool with:
- SvelteKit frontend with Design System (CSS vars, 8 shared components)
- Hono API with auth, projects, tracks, versions, comments
- PostgreSQL + Drizzle ORM (8 tables, roles, permissions)
- S3-compatible storage with presigned upload URLs
- wavesurfer.js audio player with waveform visualization
- A/B version comparison with synchronized playback
- Timestamped comments with threading and resolve workflow
- Magic Link authentication with Resend email integration
- Background audio processing (ffmpeg transcode + waveform peaks)
- Role-based access control (Owner, Engineers, Artist, Label, Management, Viewer)
- Toast notifications, skeleton loading, responsive layout
- Docker deployment setup (API + Web + Postgres)

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
Robin Choice
2026-04-02 13:23:10 +02:00
commit e420ed198b
88 changed files with 7306 additions and 0 deletions

View File

@@ -0,0 +1,12 @@
import { z } from 'zod';
export const magicLinkSchema = z.object({
email: z.string().email(),
});
export const verifyTokenSchema = z.object({
token: z.string().min(1),
});
export type MagicLinkInput = z.infer<typeof magicLinkSchema>;
export type VerifyTokenInput = z.infer<typeof verifyTokenSchema>;

View File

@@ -0,0 +1,14 @@
import { z } from 'zod';
export const createCommentSchema = z.object({
body: z.string().min(1).max(5000),
timestampSeconds: z.number().nonnegative().optional(),
parentId: z.string().uuid().optional(),
});
export const updateCommentSchema = z.object({
body: z.string().min(1).max(5000),
});
export type CreateCommentInput = z.infer<typeof createCommentSchema>;
export type UpdateCommentInput = z.infer<typeof updateCommentSchema>;

View File

@@ -0,0 +1,4 @@
export * from './auth.js';
export * from './project.js';
export * from './track.js';
export * from './comment.js';

View File

@@ -0,0 +1,30 @@
import { z } from 'zod';
import { PROJECT_ROLES } from '../constants/roles.js';
export const createProjectSchema = z.object({
name: z.string().min(1).max(255),
description: z.string().max(2000).optional(),
});
export const updateProjectSchema = z.object({
name: z.string().min(1).max(255).optional(),
description: z.string().max(2000).optional(),
});
export const inviteMemberSchema = z.object({
email: z.string().email(),
role: z.enum(PROJECT_ROLES).refine((r) => r !== 'owner', {
message: 'Cannot invite as owner',
}),
});
export const updateMemberSchema = z.object({
role: z.enum(PROJECT_ROLES).refine((r) => r !== 'owner', {
message: 'Cannot change role to owner',
}),
});
export type CreateProjectInput = z.infer<typeof createProjectSchema>;
export type UpdateProjectInput = z.infer<typeof updateProjectSchema>;
export type InviteMemberInput = z.infer<typeof inviteMemberSchema>;
export type UpdateMemberInput = z.infer<typeof updateMemberSchema>;

View File

@@ -0,0 +1,32 @@
import { z } from 'zod';
import { SUPPORTED_AUDIO_FORMATS, MAX_FILE_SIZE } from '../constants/audio.js';
export const createTrackSchema = z.object({
name: z.string().min(1).max(255),
description: z.string().max(2000).optional(),
});
export const updateTrackSchema = z.object({
name: z.string().min(1).max(255).optional(),
description: z.string().max(2000).optional(),
});
export const requestUploadUrlSchema = z.object({
fileName: z.string().min(1),
mimeType: z.enum(SUPPORTED_AUDIO_FORMATS),
fileSize: z.number().int().positive().max(MAX_FILE_SIZE),
});
export const createVersionSchema = z.object({
fileKey: z.string().min(1),
label: z.string().max(100).optional(),
notes: z.string().max(2000).optional(),
originalFileName: z.string().min(1),
mimeType: z.string().min(1),
fileSize: z.number().int().positive(),
});
export type CreateTrackInput = z.infer<typeof createTrackSchema>;
export type UpdateTrackInput = z.infer<typeof updateTrackSchema>;
export type RequestUploadUrlInput = z.infer<typeof requestUploadUrlSchema>;
export type CreateVersionInput = z.infer<typeof createVersionSchema>;