Files
openmonetis/src/features/transactions/attachment-copy.ts
Felipe Coutinho b14f487824 feat(transactions): edição cooperativa e visibilidade de divisões
Adiciona splitGroupId para vincular as duas shares de um lançamento
dividido (schema + índice + migration 0026). Habilita:

- Edição de par dividido com escolha de escopo (apenas este lado ou
  ambos) via novo SplitPairDialog e updateTransactionSplitPairAction
- Filtro "Somente divididos" (isDivided) na tabela de lançamentos
- Visibilidade de anexos para pessoas com acesso compartilhado via
  payerShares; upload e detach em massa expandem para shares irmãs
- Cópia independente de anexos no fluxo "Importar para Minha Conta"
  (novo fileKey, novo userId, S3 CopyObject) com seção read-only
  "Anexos que serão copiados" no dialog de importação
- Ícone de clipe na tabela de lançamentos da página da pessoa via
  EXISTS em fetchPagadorLancamentos

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-04-25 14:45:35 +00:00

108 lines
2.8 KiB
TypeScript

import { randomUUID } from "node:crypto";
import { CopyObjectCommand } from "@aws-sdk/client-s3";
import { eq } from "drizzle-orm";
import { attachments, transactionAttachments, transactions } from "@/db/schema";
import { db } from "@/shared/lib/db";
import { getPayerAccess } from "@/shared/lib/payers/access";
import { deleteS3Object } from "@/shared/lib/storage/presign";
import { S3_BUCKET, s3 } from "@/shared/lib/storage/s3-client";
const SAFE_EXTENSION = /^[a-z0-9]{1,10}$/i;
function sanitizeExtension(fileKey: string): string {
const ext = fileKey.split(".").pop() ?? "";
return SAFE_EXTENSION.test(ext) ? ext.toLowerCase() : "bin";
}
export async function copyAttachmentsForImport({
sourceTransactionId,
targetTransactionIds,
targetUserId,
}: {
sourceTransactionId: string;
targetTransactionIds: string[];
targetUserId: string;
}): Promise<void> {
if (targetTransactionIds.length === 0) return;
const [source] = await db
.select({
id: transactions.id,
userId: transactions.userId,
payerId: transactions.payerId,
})
.from(transactions)
.where(eq(transactions.id, sourceTransactionId));
if (!source) return;
if (source.userId !== targetUserId) {
if (!source.payerId) return;
const access = await getPayerAccess(targetUserId, source.payerId);
if (!access) return;
}
const sourceAttachments = await db
.select({
fileKey: attachments.fileKey,
fileName: attachments.fileName,
fileSize: attachments.fileSize,
mimeType: attachments.mimeType,
})
.from(transactionAttachments)
.innerJoin(
attachments,
eq(transactionAttachments.attachmentId, attachments.id),
)
.where(eq(transactionAttachments.transactionId, sourceTransactionId));
if (sourceAttachments.length === 0) return;
for (const src of sourceAttachments) {
const newFileKey = `${targetUserId}/${randomUUID()}.${sanitizeExtension(src.fileKey)}`;
try {
await s3.send(
new CopyObjectCommand({
Bucket: S3_BUCKET,
CopySource: `${S3_BUCKET}/${src.fileKey}`,
Key: newFileKey,
ContentType: src.mimeType,
MetadataDirective: "COPY",
}),
);
} catch (error) {
console.error("Falha ao copiar anexo no S3:", error);
continue;
}
try {
const [newAttachment] = await db
.insert(attachments)
.values({
userId: targetUserId,
fileKey: newFileKey,
fileName: src.fileName,
fileSize: src.fileSize,
mimeType: src.mimeType,
})
.returning({ id: attachments.id });
if (!newAttachment) {
await deleteS3Object(newFileKey);
continue;
}
await db.insert(transactionAttachments).values(
targetTransactionIds.map((tid) => ({
transactionId: tid,
attachmentId: newAttachment.id,
})),
);
} catch (error) {
console.error("Falha ao registrar anexo copiado:", error);
await deleteS3Object(newFileKey).catch(() => {});
}
}
}