From 2f18c2522e4d04881b466995b6c126ef8327dcb6 Mon Sep 17 00:00:00 2001 From: Carlos Santos <4a.santos@gmail.com> Date: Tue, 15 Oct 2024 13:22:18 +0200 Subject: [PATCH] backend: Fixed limited response listing recordings from S3 --- .../src/controllers/recording.controller.ts | 5 +- backend/src/services/recording.service.ts | 10 +- backend/src/services/s3.service.ts | 105 ++++++++---------- 3 files changed, 56 insertions(+), 64 deletions(-) diff --git a/backend/src/controllers/recording.controller.ts b/backend/src/controllers/recording.controller.ts index 5fffbf95..746be08c 100644 --- a/backend/src/controllers/recording.controller.ts +++ b/backend/src/controllers/recording.controller.ts @@ -52,12 +52,13 @@ export const stopRecording = async (req: Request, res: Response) => { /** * Endpoint only available for the admin user + * !WARNING: This will be removed in future versions */ export const getAllRecordings = async (req: Request, res: Response) => { try { logger.info('Getting all recordings'); - const continuationToken = req.query.continuationToken as string; - const response = await recordingService.getAllRecordings(continuationToken); + // const continuationToken = req.query.continuationToken as string; + const response = await recordingService.getAllRecordings(); return res .status(200) .json({ recordings: response.recordingInfo, continuationToken: response.continuationToken }); diff --git a/backend/src/services/recording.service.ts b/backend/src/services/recording.service.ts index a0f70882..067266e1 100644 --- a/backend/src/services/recording.service.ts +++ b/backend/src/services/recording.service.ts @@ -133,14 +133,12 @@ export class RecordingService { } /** - * Retrieves the list of recordings. + * Retrieves the list of all recordings. * @returns A promise that resolves to an array of RecordingInfo objects. */ - async getAllRecordings( - continuationToken?: string - ): Promise<{ recordingInfo: RecordingInfo[]; continuationToken?: string }> { + async getAllRecordings(): Promise<{ recordingInfo: RecordingInfo[]; continuationToken?: string }> { try { - const allEgress = await this.s3Service.listObjects('.metadata', '.json', 'openvidu', 20, continuationToken); + const allEgress = await this.s3Service.listObjects('.metadata', '.json'); const promises: Promise[] = []; allEgress.Contents?.forEach((item) => { @@ -149,7 +147,7 @@ export class RecordingService { } }); - return { recordingInfo: await Promise.all(promises), continuationToken: allEgress.NextContinuationToken }; + return { recordingInfo: await Promise.all(promises), continuationToken: undefined }; } catch (error) { this.logger.error(`Error getting recordings: ${error}`); throw error; diff --git a/backend/src/services/s3.service.ts b/backend/src/services/s3.service.ts index 14cb5e21..ea328bbb 100644 --- a/backend/src/services/s3.service.ts +++ b/backend/src/services/s3.service.ts @@ -1,4 +1,5 @@ import { + _Object, DeleteObjectCommand, DeleteObjectCommandOutput, DeleteObjectsCommand, @@ -123,76 +124,68 @@ export class S3Service { } } - // async deleteFolder(folderName: string, bucket: string = CALL_S3_BUCKET) { - // try { - // const listParams = { - // Bucket: bucket, - // Prefix: folderName.endsWith('/') ? folderName : `${folderName}/` - // }; - // // Get all objects in the folder - // const listedObjects: ListObjectsV2CommandOutput = await this.run(new ListObjectsV2Command(listParams)); - // const deleteParams = { - // Bucket: bucket, - // Delete: { - // Objects: listedObjects?.Contents?.map(({ Key }) => ({ Key })) - // } - // }; - - // // Skip if no objects found - // if (!deleteParams.Delete.Objects || deleteParams.Delete.Objects.length === 0){ - // this.logger.error(`No objects found in folder ${folderName}. Nothing to delete`); - // return; - // } - - // this.logger.info(`Deleting objects in S3: ${deleteParams.Delete.Objects}`); - // await this.run(new DeleteObjectsCommand(deleteParams)); - - // if (listedObjects.IsTruncated) { - // this.logger.verbose(`Folder ${folderName} is truncated, deleting next batch`); - // await this.deleteFolder(bucket, folderName); - // } - // } catch (error) { - // this.logger.error(`Error deleting folder in S3: ${error}`); - // throw internalError(error); - // } - // } - /** - * Lists objects in an S3 bucket. + * Lists all objects in an S3 bucket with optional subbucket and search pattern filtering. * - * @param subbucket - The subbucket within the bucket to list objects from. - * @param searchPattern - The search pattern to filter the objects by. - * @param bucket - The name of the S3 bucket. - * @param maxObjects - The maximum number of objects to retrieve. - * @returns A promise that resolves to the list of objects. - * @throws Throws an error if there was an error listing the objects. + * @param {string} [subbucket=''] - The subbucket within the main bucket to list objects from. + * @param {string} [searchPattern=''] - A regex pattern to filter the objects by their keys. + * @param {string} [bucket=CALL_S3_BUCKET] - The name of the S3 bucket. Defaults to CALL_S3_BUCKET. + * @param {number} [maxObjects=1000] - The maximum number of objects to retrieve in one request. Defaults to 1000. + * @returns {Promise} - A promise that resolves to the output of the ListObjectsV2Command. + * @throws {Error} - Throws an error if there is an issue listing the objects. */ async listObjects( subbucket = '', searchPattern = '', bucket: string = CALL_S3_BUCKET, - maxObjects = 20, - continuationToken?: string + maxObjects = 1000 ): Promise { const prefix = subbucket ? `${subbucket}/` : ''; - const command = new ListObjectsV2Command({ - Bucket: bucket, - Prefix: prefix, - MaxKeys: maxObjects, - ContinuationToken: continuationToken - }); + let allContents: _Object[] = []; + let continuationToken: string | undefined = undefined; + let isTruncated = true; + let fullResponse: ListObjectsV2CommandOutput | undefined = undefined; try { - const response: ListObjectsV2CommandOutput = await this.run(command); - - if (searchPattern) { - const regex = new RegExp(searchPattern); - response.Contents = response.Contents?.filter((object) => { - return object.Key && regex.test(object.Key); + while (isTruncated) { + const command = new ListObjectsV2Command({ + Bucket: bucket, + Prefix: prefix, + MaxKeys: maxObjects, + ContinuationToken: continuationToken }); + + const response: ListObjectsV2CommandOutput = await this.run(command); + + if (!fullResponse) { + fullResponse = response; + } + + // Filter the objects by the search pattern if it is provided + let objects = response.Contents || []; + + if (searchPattern) { + const regex = new RegExp(searchPattern); + objects = objects.filter((object) => object.Key && regex.test(object.Key)); + } + + // Add the objects to the list of all objects + allContents = allContents.concat(objects); + + // Update the loop control variables + isTruncated = response.IsTruncated ?? false; + continuationToken = response.NextContinuationToken; + } + + if (fullResponse) { + fullResponse.Contents = allContents; + fullResponse.IsTruncated = false; + fullResponse.NextContinuationToken = undefined; + fullResponse.MaxKeys = allContents.length; + fullResponse.KeyCount = allContents.length; } - return response; + return fullResponse!; } catch (error) { this.logger.error(`Error listing objects: ${error}`);