Files
Momento/memento-note/app/actions/brainstorm.ts
Antigravity 8c7ca69640
All checks were successful
Deploy to Production / Build and Deploy (push) Successful in 5s
fix: brainstorm infinite loop, ghost cursor, embedding ::vector cast, semantic search, billing stats, usage meter accordion
- Fix useBrainstormSocket: stable guestId via useRef, remove setState in cleanup
- Fix GhostCursor: direct DOM manipulation via refs, no useState re-renders
- Fix all SQL embedding queries: add ::vector cast on text columns
- Fix embedding truncation to 15000 chars (under 8192 token limit)
- Fix NoteEmbedding INSERT: remove non-existent updatedAt column
- Fix billing page: show all quota stats in grid instead of single metric
- Fix usage meter: accordion expand/collapse, per-feature detail
- Fix semantic search: rebuild 103 note embeddings, ::vector cast on vectorSearch
- Fix brainstorm expand/manual-idea/create: ::vector cast on embedding SQL
2026-05-16 18:50:34 +00:00

199 lines
5.2 KiB
TypeScript

'use server'
import { revalidatePath } from 'next/cache'
import prisma from '@/lib/prisma'
import { auth } from '@/auth'
export async function createBrainstormShare(
sessionId: string,
recipientEmail: string,
permission: 'editor' | 'viewer' = 'editor'
) {
const session = await auth()
if (!session?.user?.id) throw new Error('Unauthorized')
const brainstorm = await prisma.brainstormSession.findUnique({
where: { id: sessionId },
})
if (!brainstorm) throw new Error('Session not found')
if (brainstorm.userId !== session.user.id)
throw new Error('Only the owner can share')
const recipient = await prisma.user.findUnique({
where: { email: recipientEmail },
})
if (!recipient) throw new Error('No account found with this email')
if (recipient.id === session.user.id)
throw new Error('Cannot share with yourself')
const existing = await prisma.brainstormShare.findUnique({
where: { sessionId_userId: { sessionId, userId: recipient.id } },
})
if (existing) {
switch (existing.status) {
case 'accepted':
return { success: true, message: 'already_shared' }
case 'pending':
return { success: true, message: 'already_pending' }
case 'declined':
case 'removed':
await prisma.brainstormShare.update({
where: { id: existing.id },
data: {
status: 'pending',
permission,
notifiedAt: new Date(),
respondedAt: null,
},
})
return { success: true, message: 're_invited' }
}
}
await prisma.brainstormShare.create({
data: {
sessionId,
userId: recipient.id,
sharedBy: session.user.id,
status: 'pending',
permission,
notifiedAt: new Date(),
},
})
return { success: true, message: 'invited' }
}
export async function getPendingBrainstormShares() {
const session = await auth()
if (!session?.user?.id) return []
try {
return await prisma.brainstormShare.findMany({
where: { userId: session.user.id, status: 'pending' },
include: {
session: {
select: { id: true, seedIdea: true, createdAt: true },
},
sharer: {
select: { id: true, name: true, email: true, image: true },
},
},
orderBy: { createdAt: 'desc' },
})
} catch {
return []
}
}
export async function respondToBrainstormShare(
shareId: string,
action: 'accept' | 'decline'
) {
const session = await auth()
if (!session?.user?.id) throw new Error('Unauthorized')
const share = await prisma.brainstormShare.findUnique({
where: { id: shareId },
})
if (!share) throw new Error('Share not found')
if (share.userId !== session.user.id) throw new Error('Unauthorized')
if (share.status !== 'pending')
throw new Error(`Share already ${share.status}`)
const newStatus = action === 'accept' ? 'accepted' : 'declined'
await prisma.brainstormShare.update({
where: { id: shareId },
data: { status: newStatus, respondedAt: new Date() },
})
if (action === 'accept') {
await prisma.brainstormParticipant.upsert({
where: {
sessionId_userId: {
sessionId: share.sessionId,
userId: session.user.id,
},
},
create: {
sessionId: share.sessionId,
userId: session.user.id,
role: share.permission === 'viewer' ? 'viewer' : 'editor',
},
update: {
role: share.permission === 'viewer' ? 'viewer' : 'editor',
},
})
}
revalidatePath('/home')
return { success: true }
}
export async function ensureAcceptedSharesHaveParticipants() {
const session = await auth()
if (!session?.user?.id) return
const acceptedShares = await prisma.brainstormShare.findMany({
where: { userId: session.user.id, status: 'accepted' },
select: { sessionId: true, permission: true },
})
for (const share of acceptedShares) {
await prisma.brainstormParticipant.upsert({
where: {
sessionId_userId: {
sessionId: share.sessionId,
userId: session.user.id,
},
},
create: {
sessionId: share.sessionId,
userId: session.user.id,
role: share.permission === 'viewer' ? 'viewer' : 'editor',
},
update: {},
})
}
}
export async function getAcceptedBrainstormShares() {
const session = await auth()
if (!session?.user?.id) return []
try {
return await prisma.brainstormShare.findMany({
where: { userId: session.user.id, status: 'accepted' },
include: {
session: {
select: {
id: true,
seedIdea: true,
createdAt: true,
updatedAt: true,
ideas: { where: { status: 'active' }, select: { id: true } },
},
},
},
orderBy: { session: { updatedAt: 'desc' } },
})
} catch {
return []
}
}
export async function removeBrainstormShare(sessionId: string) {
const session = await auth()
if (!session?.user?.id) throw new Error('Unauthorized')
await prisma.brainstormShare.updateMany({
where: { sessionId, userId: session.user.id, status: 'accepted' },
data: { status: 'removed' },
})
revalidatePath('/home')
return { success: true }
}