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:
12
packages/shared/src/validation/auth.ts
Normal file
12
packages/shared/src/validation/auth.ts
Normal 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>;
|
||||
14
packages/shared/src/validation/comment.ts
Normal file
14
packages/shared/src/validation/comment.ts
Normal 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>;
|
||||
4
packages/shared/src/validation/index.ts
Normal file
4
packages/shared/src/validation/index.ts
Normal file
@@ -0,0 +1,4 @@
|
||||
export * from './auth.js';
|
||||
export * from './project.js';
|
||||
export * from './track.js';
|
||||
export * from './comment.js';
|
||||
30
packages/shared/src/validation/project.ts
Normal file
30
packages/shared/src/validation/project.ts
Normal 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>;
|
||||
32
packages/shared/src/validation/track.ts
Normal file
32
packages/shared/src/validation/track.ts
Normal 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>;
|
||||
Reference in New Issue
Block a user