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,11 @@
{
"name": "@music-hub/shared",
"version": "0.0.1",
"type": "module",
"exports": {
".": "./src/index.ts"
},
"dependencies": {
"zod": "^3.24"
}
}

View File

@@ -0,0 +1,24 @@
export const SUPPORTED_AUDIO_FORMATS = [
'audio/wav',
'audio/x-wav',
'audio/mp3',
'audio/mpeg',
'audio/flac',
'audio/x-flac',
'audio/aiff',
'audio/x-aiff',
] as const;
export const SUPPORTED_EXTENSIONS = ['.wav', '.mp3', '.flac', '.aiff', '.aif'] as const;
export const MAX_FILE_SIZE = 500 * 1024 * 1024; // 500 MB
export const VERSION_STATUSES = [
'uploaded',
'processing',
'ready',
'approved',
'rejected',
] as const;
export type VersionStatus = (typeof VERSION_STATUSES)[number];

View File

@@ -0,0 +1,3 @@
export * from './roles.js';
export * from './permissions.js';
export * from './audio.js';

View File

@@ -0,0 +1,69 @@
import type { ProjectRole } from './roles.js';
export type Permission =
| 'project.edit'
| 'project.invite'
| 'track.upload'
| 'track.listen'
| 'track.download'
| 'version.comment'
| 'version.approve';
const PERMISSIONS: Record<ProjectRole, Permission[]> = {
owner: [
'project.edit',
'project.invite',
'track.upload',
'track.listen',
'track.download',
'version.comment',
'version.approve',
],
recording_engineer: [
'track.upload',
'track.listen',
'track.download',
'version.comment',
],
mixing_engineer: [
'track.upload',
'track.listen',
'track.download',
'version.comment',
],
mastering_engineer: [
'track.upload',
'track.listen',
'track.download',
'version.comment',
],
artist: [
'track.listen',
'track.download',
'version.comment',
'version.approve',
],
label: [
'track.listen',
'version.comment',
'version.approve',
],
management: [
'project.invite',
'track.listen',
'track.download',
'version.comment',
'version.approve',
],
viewer: [
'track.listen',
],
};
export function hasPermission(role: ProjectRole, permission: Permission): boolean {
return PERMISSIONS[role].includes(permission);
}
export function getPermissions(role: ProjectRole): Permission[] {
return PERMISSIONS[role];
}

View File

@@ -0,0 +1,29 @@
export const PROJECT_ROLES = [
'owner',
'recording_engineer',
'mixing_engineer',
'mastering_engineer',
'artist',
'label',
'management',
'viewer',
] as const;
export type ProjectRole = (typeof PROJECT_ROLES)[number];
export const ROLE_LABELS: Record<ProjectRole, string> = {
owner: 'Owner',
recording_engineer: 'Recording Engineer',
mixing_engineer: 'Mixing Engineer',
mastering_engineer: 'Mastering Engineer',
artist: 'Artist',
label: 'Label',
management: 'Management',
viewer: 'Viewer',
};
export const ENGINEER_ROLES: ProjectRole[] = [
'recording_engineer',
'mixing_engineer',
'mastering_engineer',
];

View File

@@ -0,0 +1,2 @@
export * from './constants/index.js';
export * from './validation/index.js';

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>;

View File

@@ -0,0 +1,13 @@
{
"compilerOptions": {
"target": "ESNext",
"module": "ESNext",
"moduleResolution": "bundler",
"strict": true,
"esModuleInterop": true,
"skipLibCheck": true,
"declaration": true,
"outDir": "dist"
},
"include": ["src"]
}