mirror of
https://github.com/felipegcoutinho/openmonetis.git
synced 2026-05-09 11:01:45 +00:00
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>
108 lines
2.8 KiB
TypeScript
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(() => {});
|
|
}
|
|
}
|
|
}
|