mirror of
https://github.com/simstudioai/sim.git
synced 2026-04-28 03:00:29 -04:00
* improvement(collab): do not refetch active workflow id * progress on files * more integrations * separate server and client logic * consolidate more code * fix integrations * fix types * consolidate more code * fix tests * fix more bugbot comments * fix type check * fix circular impport * address more bugbot comments * fix ocr integrations * fix typing * remove leftover type * address bugbot comment * fix file block adv mode * fix * normalize file input * fix v2 blocmks for ocr * fix for v2 versions * fix more v2 blocks * update single file blocks * make interface simpler * cleanup fireflies * remove file only annotation * accept all types * added wand to ssh block * user files should be passed through * improve docs * fix slack to include successful execs * fix dropbox upload file * fix sendgrid * fix dropbox * fix * fix * update skills * fix uploaded file --------- Co-authored-by: waleed <walif6@gmail.com>
301 lines
8.9 KiB
TypeScript
301 lines
8.9 KiB
TypeScript
import { createLogger } from '@sim/logger'
|
|
import { type NextRequest, NextResponse } from 'next/server'
|
|
import { z } from 'zod'
|
|
import { checkInternalAuth } from '@/lib/auth/hybrid'
|
|
import { generateRequestId } from '@/lib/core/utils/request'
|
|
import { RawFileInputSchema } from '@/lib/uploads/utils/file-schemas'
|
|
import { processSingleFileToUserFile } from '@/lib/uploads/utils/file-utils'
|
|
import { downloadFileFromStorage } from '@/lib/uploads/utils/file-utils.server'
|
|
import {
|
|
GOOGLE_WORKSPACE_MIME_TYPES,
|
|
handleSheetsFormat,
|
|
SOURCE_MIME_TYPES,
|
|
} from '@/tools/google_drive/utils'
|
|
|
|
export const dynamic = 'force-dynamic'
|
|
|
|
const logger = createLogger('GoogleDriveUploadAPI')
|
|
|
|
const GOOGLE_DRIVE_API_BASE = 'https://www.googleapis.com/upload/drive/v3/files'
|
|
|
|
const GoogleDriveUploadSchema = z.object({
|
|
accessToken: z.string().min(1, 'Access token is required'),
|
|
fileName: z.string().min(1, 'File name is required'),
|
|
file: RawFileInputSchema.optional().nullable(),
|
|
mimeType: z.string().optional().nullable(),
|
|
folderId: z.string().optional().nullable(),
|
|
})
|
|
|
|
/**
|
|
* Build multipart upload body for Google Drive API
|
|
*/
|
|
function buildMultipartBody(
|
|
metadata: Record<string, any>,
|
|
fileBuffer: Buffer,
|
|
mimeType: string,
|
|
boundary: string
|
|
): string {
|
|
const parts: string[] = []
|
|
|
|
parts.push(`--${boundary}`)
|
|
parts.push('Content-Type: application/json; charset=UTF-8')
|
|
parts.push('')
|
|
parts.push(JSON.stringify(metadata))
|
|
|
|
parts.push(`--${boundary}`)
|
|
parts.push(`Content-Type: ${mimeType}`)
|
|
parts.push('Content-Transfer-Encoding: base64')
|
|
parts.push('')
|
|
parts.push(fileBuffer.toString('base64'))
|
|
|
|
parts.push(`--${boundary}--`)
|
|
|
|
return parts.join('\r\n')
|
|
}
|
|
|
|
export async function POST(request: NextRequest) {
|
|
const requestId = generateRequestId()
|
|
|
|
try {
|
|
const authResult = await checkInternalAuth(request, { requireWorkflowId: false })
|
|
|
|
if (!authResult.success) {
|
|
logger.warn(`[${requestId}] Unauthorized Google Drive upload attempt: ${authResult.error}`)
|
|
return NextResponse.json(
|
|
{
|
|
success: false,
|
|
error: authResult.error || 'Authentication required',
|
|
},
|
|
{ status: 401 }
|
|
)
|
|
}
|
|
|
|
logger.info(
|
|
`[${requestId}] Authenticated Google Drive upload request via ${authResult.authType}`,
|
|
{
|
|
userId: authResult.userId,
|
|
}
|
|
)
|
|
|
|
const body = await request.json()
|
|
const validatedData = GoogleDriveUploadSchema.parse(body)
|
|
|
|
logger.info(`[${requestId}] Uploading file to Google Drive`, {
|
|
fileName: validatedData.fileName,
|
|
mimeType: validatedData.mimeType,
|
|
folderId: validatedData.folderId,
|
|
hasFile: !!validatedData.file,
|
|
})
|
|
|
|
if (!validatedData.file) {
|
|
return NextResponse.json(
|
|
{
|
|
success: false,
|
|
error: 'No file provided. Use the text content field for text-only uploads.',
|
|
},
|
|
{ status: 400 }
|
|
)
|
|
}
|
|
|
|
// Process file - convert to UserFile format if needed
|
|
const fileData = validatedData.file
|
|
|
|
let userFile
|
|
try {
|
|
userFile = processSingleFileToUserFile(fileData, requestId, logger)
|
|
} catch (error) {
|
|
return NextResponse.json(
|
|
{
|
|
success: false,
|
|
error: error instanceof Error ? error.message : 'Failed to process file',
|
|
},
|
|
{ status: 400 }
|
|
)
|
|
}
|
|
|
|
logger.info(`[${requestId}] Downloading file from storage`, {
|
|
fileName: userFile.name,
|
|
key: userFile.key,
|
|
size: userFile.size,
|
|
})
|
|
|
|
let fileBuffer: Buffer
|
|
|
|
try {
|
|
fileBuffer = await downloadFileFromStorage(userFile, requestId, logger)
|
|
} catch (error) {
|
|
logger.error(`[${requestId}] Failed to download file:`, error)
|
|
return NextResponse.json(
|
|
{
|
|
success: false,
|
|
error: `Failed to download file: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
},
|
|
{ status: 500 }
|
|
)
|
|
}
|
|
|
|
let uploadMimeType = validatedData.mimeType || userFile.type || 'application/octet-stream'
|
|
const requestedMimeType = validatedData.mimeType || userFile.type || 'application/octet-stream'
|
|
|
|
if (GOOGLE_WORKSPACE_MIME_TYPES.includes(requestedMimeType)) {
|
|
uploadMimeType = SOURCE_MIME_TYPES[requestedMimeType] || 'text/plain'
|
|
logger.info(`[${requestId}] Converting to Google Workspace type`, {
|
|
requestedMimeType,
|
|
uploadMimeType,
|
|
})
|
|
}
|
|
|
|
if (requestedMimeType === 'application/vnd.google-apps.spreadsheet') {
|
|
try {
|
|
const textContent = fileBuffer.toString('utf-8')
|
|
const { csv } = handleSheetsFormat(textContent)
|
|
if (csv !== undefined) {
|
|
fileBuffer = Buffer.from(csv, 'utf-8')
|
|
uploadMimeType = 'text/csv'
|
|
logger.info(`[${requestId}] Converted to CSV for Google Sheets upload`)
|
|
}
|
|
} catch (error) {
|
|
logger.warn(`[${requestId}] Could not convert to CSV, uploading as-is:`, error)
|
|
}
|
|
}
|
|
|
|
const metadata: {
|
|
name: string
|
|
mimeType: string
|
|
parents?: string[]
|
|
} = {
|
|
name: validatedData.fileName,
|
|
mimeType: requestedMimeType,
|
|
}
|
|
|
|
if (validatedData.folderId && validatedData.folderId.trim() !== '') {
|
|
metadata.parents = [validatedData.folderId.trim()]
|
|
}
|
|
|
|
const boundary = `boundary_${Date.now()}_${Math.random().toString(36).substring(7)}`
|
|
|
|
const multipartBody = buildMultipartBody(metadata, fileBuffer, uploadMimeType, boundary)
|
|
|
|
logger.info(`[${requestId}] Uploading to Google Drive via multipart upload`, {
|
|
fileName: validatedData.fileName,
|
|
size: fileBuffer.length,
|
|
uploadMimeType,
|
|
requestedMimeType,
|
|
})
|
|
|
|
const uploadResponse = await fetch(
|
|
`${GOOGLE_DRIVE_API_BASE}?uploadType=multipart&supportsAllDrives=true`,
|
|
{
|
|
method: 'POST',
|
|
headers: {
|
|
Authorization: `Bearer ${validatedData.accessToken}`,
|
|
'Content-Type': `multipart/related; boundary=${boundary}`,
|
|
'Content-Length': Buffer.byteLength(multipartBody, 'utf-8').toString(),
|
|
},
|
|
body: multipartBody,
|
|
}
|
|
)
|
|
|
|
if (!uploadResponse.ok) {
|
|
const errorText = await uploadResponse.text()
|
|
logger.error(`[${requestId}] Google Drive API error:`, {
|
|
status: uploadResponse.status,
|
|
statusText: uploadResponse.statusText,
|
|
error: errorText,
|
|
})
|
|
return NextResponse.json(
|
|
{
|
|
success: false,
|
|
error: `Google Drive API error: ${uploadResponse.statusText}`,
|
|
},
|
|
{ status: uploadResponse.status }
|
|
)
|
|
}
|
|
|
|
const uploadData = await uploadResponse.json()
|
|
const fileId = uploadData.id
|
|
|
|
logger.info(`[${requestId}] File uploaded successfully`, { fileId })
|
|
|
|
if (GOOGLE_WORKSPACE_MIME_TYPES.includes(requestedMimeType)) {
|
|
logger.info(`[${requestId}] Updating file name to ensure it persists after conversion`)
|
|
|
|
const updateNameResponse = await fetch(
|
|
`https://www.googleapis.com/drive/v3/files/${fileId}?supportsAllDrives=true`,
|
|
{
|
|
method: 'PATCH',
|
|
headers: {
|
|
Authorization: `Bearer ${validatedData.accessToken}`,
|
|
'Content-Type': 'application/json',
|
|
},
|
|
body: JSON.stringify({
|
|
name: validatedData.fileName,
|
|
}),
|
|
}
|
|
)
|
|
|
|
if (!updateNameResponse.ok) {
|
|
logger.warn(
|
|
`[${requestId}] Failed to update filename after conversion, but content was uploaded`
|
|
)
|
|
}
|
|
}
|
|
|
|
const finalFileResponse = await fetch(
|
|
`https://www.googleapis.com/drive/v3/files/${fileId}?supportsAllDrives=true&fields=id,name,mimeType,webViewLink,webContentLink,size,createdTime,modifiedTime,parents`,
|
|
{
|
|
headers: {
|
|
Authorization: `Bearer ${validatedData.accessToken}`,
|
|
},
|
|
}
|
|
)
|
|
|
|
const finalFile = await finalFileResponse.json()
|
|
|
|
logger.info(`[${requestId}] Upload complete`, {
|
|
fileId: finalFile.id,
|
|
fileName: finalFile.name,
|
|
webViewLink: finalFile.webViewLink,
|
|
})
|
|
|
|
return NextResponse.json({
|
|
success: true,
|
|
output: {
|
|
file: {
|
|
id: finalFile.id,
|
|
name: finalFile.name,
|
|
mimeType: finalFile.mimeType,
|
|
webViewLink: finalFile.webViewLink,
|
|
webContentLink: finalFile.webContentLink,
|
|
size: finalFile.size,
|
|
createdTime: finalFile.createdTime,
|
|
modifiedTime: finalFile.modifiedTime,
|
|
parents: finalFile.parents,
|
|
},
|
|
},
|
|
})
|
|
} catch (error) {
|
|
if (error instanceof z.ZodError) {
|
|
logger.warn(`[${requestId}] Invalid request data`, { errors: error.errors })
|
|
return NextResponse.json(
|
|
{
|
|
success: false,
|
|
error: 'Invalid request data',
|
|
details: error.errors,
|
|
},
|
|
{ status: 400 }
|
|
)
|
|
}
|
|
|
|
logger.error(`[${requestId}] Error uploading file to Google Drive:`, error)
|
|
|
|
return NextResponse.json(
|
|
{
|
|
success: false,
|
|
error: error instanceof Error ? error.message : 'Internal server error',
|
|
},
|
|
{ status: 500 }
|
|
)
|
|
}
|
|
}
|