mirror of
https://github.com/marcogll/TaxHacker_s23.git
synced 2026-01-13 13:25:18 +00:00
214 lines
6.9 KiB
TypeScript
214 lines
6.9 KiB
TypeScript
"use server"
|
|
|
|
import { transactionFormSchema } from "@/forms/transactions"
|
|
import { ActionState } from "@/lib/actions"
|
|
import { getCurrentUser } from "@/lib/auth"
|
|
import { getDirectorySize, getTransactionFileUploadPath, getUserUploadsDirectory } from "@/lib/files"
|
|
import { updateField } from "@/models/fields"
|
|
import { createFile, deleteFile } from "@/models/files"
|
|
import {
|
|
bulkDeleteTransactions,
|
|
createTransaction,
|
|
deleteTransaction,
|
|
getTransactionById,
|
|
updateTransaction,
|
|
updateTransactionFiles,
|
|
} from "@/models/transactions"
|
|
import { updateUser } from "@/models/users"
|
|
import { Transaction } from "@prisma/client"
|
|
import { randomUUID } from "crypto"
|
|
import { mkdir, writeFile } from "fs/promises"
|
|
import { revalidatePath } from "next/cache"
|
|
import path from "path"
|
|
|
|
export async function createTransactionAction(
|
|
_prevState: ActionState<Transaction> | null,
|
|
formData: FormData
|
|
): Promise<ActionState<Transaction>> {
|
|
try {
|
|
const user = await getCurrentUser()
|
|
const validatedForm = transactionFormSchema.safeParse(Object.fromEntries(formData.entries()))
|
|
|
|
if (!validatedForm.success) {
|
|
return { success: false, error: validatedForm.error.message }
|
|
}
|
|
|
|
const transaction = await createTransaction(user.id, validatedForm.data)
|
|
|
|
revalidatePath("/transactions")
|
|
return { success: true, data: transaction }
|
|
} catch (error) {
|
|
console.error("Failed to create transaction:", error)
|
|
return { success: false, error: "Failed to create transaction" }
|
|
}
|
|
}
|
|
|
|
export async function saveTransactionAction(
|
|
_prevState: ActionState<Transaction> | null,
|
|
formData: FormData
|
|
): Promise<ActionState<Transaction>> {
|
|
try {
|
|
const user = await getCurrentUser()
|
|
const transactionId = formData.get("transactionId") as string
|
|
const validatedForm = transactionFormSchema.safeParse(Object.fromEntries(formData.entries()))
|
|
|
|
if (!validatedForm.success) {
|
|
return { success: false, error: validatedForm.error.message }
|
|
}
|
|
|
|
const transaction = await updateTransaction(transactionId, user.id, validatedForm.data)
|
|
|
|
revalidatePath("/transactions")
|
|
return { success: true, data: transaction }
|
|
} catch (error) {
|
|
console.error("Failed to update transaction:", error)
|
|
return { success: false, error: "Failed to save transaction" }
|
|
}
|
|
}
|
|
|
|
export async function deleteTransactionAction(
|
|
_prevState: ActionState<Transaction> | null,
|
|
transactionId: string
|
|
): Promise<ActionState<Transaction>> {
|
|
try {
|
|
const user = await getCurrentUser()
|
|
const transaction = await getTransactionById(transactionId, user.id)
|
|
if (!transaction) throw new Error("Transaction not found")
|
|
|
|
await deleteTransaction(transaction.id, user.id)
|
|
|
|
revalidatePath("/transactions")
|
|
|
|
return { success: true, data: transaction }
|
|
} catch (error) {
|
|
console.error("Failed to delete transaction:", error)
|
|
return { success: false, error: "Failed to delete transaction" }
|
|
}
|
|
}
|
|
|
|
export async function deleteTransactionFileAction(
|
|
transactionId: string,
|
|
fileId: string
|
|
): Promise<ActionState<Transaction>> {
|
|
if (!fileId || !transactionId) {
|
|
return { success: false, error: "File ID and transaction ID are required" }
|
|
}
|
|
|
|
const user = await getCurrentUser()
|
|
const transaction = await getTransactionById(transactionId, user.id)
|
|
if (!transaction) {
|
|
return { success: false, error: "Transaction not found" }
|
|
}
|
|
|
|
await updateTransactionFiles(
|
|
transactionId,
|
|
user.id,
|
|
transaction.files ? (transaction.files as string[]).filter((id) => id !== fileId) : []
|
|
)
|
|
|
|
await deleteFile(fileId, user.id)
|
|
|
|
// Update user storage used
|
|
const storageUsed = await getDirectorySize(await getUserUploadsDirectory(user))
|
|
await updateUser(user.id, { storageUsed })
|
|
|
|
revalidatePath(`/transactions/${transactionId}`)
|
|
return { success: true, data: transaction }
|
|
}
|
|
|
|
export async function uploadTransactionFilesAction(formData: FormData): Promise<ActionState<Transaction>> {
|
|
try {
|
|
const transactionId = formData.get("transactionId") as string
|
|
const files = formData.getAll("files") as File[]
|
|
|
|
if (!files || !transactionId) {
|
|
return { success: false, error: "No files or transaction ID provided" }
|
|
}
|
|
|
|
const user = await getCurrentUser()
|
|
const transaction = await getTransactionById(transactionId, user.id)
|
|
if (!transaction) {
|
|
return { success: false, error: "Transaction not found" }
|
|
}
|
|
|
|
const userUploadsDirectory = await getUserUploadsDirectory(user)
|
|
|
|
const fileRecords = await Promise.all(
|
|
files.map(async (file) => {
|
|
const fileUuid = randomUUID()
|
|
const relativeFilePath = await getTransactionFileUploadPath(fileUuid, file.name, transaction)
|
|
const arrayBuffer = await file.arrayBuffer()
|
|
const buffer = Buffer.from(arrayBuffer)
|
|
|
|
const fullFilePath = path.join(userUploadsDirectory, relativeFilePath)
|
|
await mkdir(path.dirname(fullFilePath), { recursive: true })
|
|
|
|
console.log("userUploadsDirectory", userUploadsDirectory)
|
|
console.log("relativeFilePath", relativeFilePath)
|
|
console.log("fullFilePath", fullFilePath)
|
|
|
|
await writeFile(fullFilePath, buffer)
|
|
|
|
// Create file record in database
|
|
const fileRecord = await createFile(user.id, {
|
|
id: fileUuid,
|
|
filename: file.name,
|
|
path: relativeFilePath,
|
|
mimetype: file.type,
|
|
isReviewed: true,
|
|
metadata: {
|
|
size: file.size,
|
|
lastModified: file.lastModified,
|
|
},
|
|
})
|
|
|
|
return fileRecord
|
|
})
|
|
)
|
|
|
|
// Update invoice with the new file ID
|
|
await updateTransactionFiles(
|
|
transactionId,
|
|
user.id,
|
|
transaction.files
|
|
? [...(transaction.files as string[]), ...fileRecords.map((file) => file.id)]
|
|
: fileRecords.map((file) => file.id)
|
|
)
|
|
|
|
// Update user storage used
|
|
const storageUsed = await getDirectorySize(await getUserUploadsDirectory(user))
|
|
await updateUser(user.id, { storageUsed })
|
|
|
|
revalidatePath(`/transactions/${transactionId}`)
|
|
return { success: true }
|
|
} catch (error) {
|
|
console.error("Upload error:", error)
|
|
return { success: false, error: `File upload failed: ${error}` }
|
|
}
|
|
}
|
|
|
|
export async function bulkDeleteTransactionsAction(transactionIds: string[]) {
|
|
try {
|
|
const user = await getCurrentUser()
|
|
await bulkDeleteTransactions(transactionIds, user.id)
|
|
revalidatePath("/transactions")
|
|
return { success: true }
|
|
} catch (error) {
|
|
console.error("Failed to delete transactions:", error)
|
|
return { success: false, error: "Failed to delete transactions" }
|
|
}
|
|
}
|
|
|
|
export async function updateFieldVisibilityAction(fieldCode: string, isVisible: boolean) {
|
|
try {
|
|
const user = await getCurrentUser()
|
|
await updateField(user.id, fieldCode, {
|
|
isVisibleInList: isVisible,
|
|
})
|
|
return { success: true }
|
|
} catch (error) {
|
|
console.error("Failed to update field visibility:", error)
|
|
return { success: false, error: "Failed to update field visibility" }
|
|
}
|
|
}
|