@aws-amplify/storage
Version:
Storage category of aws-amplify
180 lines (166 loc) • 4.79 kB
text/typescript
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
// SPDX-License-Identifier: Apache-2.0
import {
KeyValueStorageInterface,
StorageAccessLevel,
} from '@aws-amplify/core';
import {
ContentDisposition,
ResolvedS3Config,
UploadDataChecksumAlgorithm,
} from '../../../../types/options';
import { StorageUploadDataPayload } from '../../../../../../types';
import { Part, createMultipartUpload } from '../../../../utils/client/s3data';
import { logger } from '../../../../../../utils';
import { constructContentDisposition } from '../../../../utils/constructContentDisposition';
import { CHECKSUM_ALGORITHM_CRC32 } from '../../../../utils/constants';
import { calculateContentCRC32 } from '../../../../utils/crc32';
import {
cacheMultipartUpload,
findCachedUploadPartsAndEvictExpired,
getUploadsCacheKey,
} from './uploadCache';
interface LoadOrCreateMultipartUploadOptions {
s3Config: ResolvedS3Config;
data: StorageUploadDataPayload;
bucket: string;
size: number;
accessLevel?: StorageAccessLevel;
keyPrefix?: string;
key: string;
contentType?: string;
contentDisposition?: string | ContentDisposition;
contentEncoding?: string;
metadata?: Record<string, string>;
abortSignal?: AbortSignal;
checksumAlgorithm?: UploadDataChecksumAlgorithm;
optionsHash: string;
resumableUploadsCache?: KeyValueStorageInterface;
expectedBucketOwner?: string;
}
interface LoadOrCreateMultipartUploadResult {
uploadId: string;
cachedParts: Part[];
finalCrc32?: string;
}
/**
* Load the in-progress multipart upload from local storage or async storage(RN) if it exists, or create a new multipart
* upload.
*
* @internal
*/
export const loadOrCreateMultipartUpload = async ({
s3Config,
data,
size,
contentType,
bucket,
accessLevel,
keyPrefix,
key,
contentDisposition,
contentEncoding,
metadata,
abortSignal,
checksumAlgorithm,
optionsHash,
resumableUploadsCache,
expectedBucketOwner,
}: LoadOrCreateMultipartUploadOptions): Promise<LoadOrCreateMultipartUploadResult> => {
const finalKey = keyPrefix !== undefined ? keyPrefix + key : key;
let cachedUpload:
| {
parts: Part[];
uploadId: string;
uploadCacheKey: string;
finalCrc32?: string;
}
| undefined;
if (!resumableUploadsCache) {
logger.debug(
'uploaded cache instance cannot be determined, skipping cache.',
);
cachedUpload = undefined;
} else {
const uploadCacheKey = getUploadsCacheKey({
size,
contentType,
file: data instanceof File ? data : undefined,
bucket,
accessLevel,
key,
optionsHash,
});
const cachedUploadParts = await findCachedUploadPartsAndEvictExpired({
s3Config,
cacheKey: uploadCacheKey,
bucket,
finalKey,
resumableUploadsCache,
});
cachedUpload = cachedUploadParts
? { ...cachedUploadParts, uploadCacheKey }
: undefined;
}
if (cachedUpload) {
return {
uploadId: cachedUpload.uploadId,
cachedParts: cachedUpload.parts,
finalCrc32: cachedUpload.finalCrc32,
};
} else {
/**
* Note: This step reads the uploading file from beginning to end to calculate the CRC32 checksum of the entire
* object before sending the 1st byte over the wire. This is a performance bottleneck when uploading large files.
* The rationale to do this is S3 team wants to reduce the possibility of a file getting corrupted(on disk or in
* memory). So we calculate the full-object checksum as soon as possible in the upload flow.
*
* Going forward we should re-evaluate this decision with S3 team. The alternative is calling calculateContentCRC32()
* as we upload each part sequentially with seeds from already uploaded parts, ideally inside the data chunker.
*/
const finalCrc32 =
checksumAlgorithm === CHECKSUM_ALGORITHM_CRC32
? await calculateContentCRC32(data)
: undefined;
const { UploadId } = await createMultipartUpload(
{
...s3Config,
abortSignal,
},
{
Bucket: bucket,
Key: finalKey,
ContentType: contentType,
ContentDisposition: constructContentDisposition(contentDisposition),
ContentEncoding: contentEncoding,
Metadata: metadata,
ChecksumAlgorithm: finalCrc32 ? 'CRC32' : undefined,
ChecksumType: finalCrc32 ? 'FULL_OBJECT' : undefined,
ExpectedBucketOwner: expectedBucketOwner,
},
);
if (resumableUploadsCache) {
const uploadCacheKey = getUploadsCacheKey({
size,
contentType,
file: data instanceof File ? data : undefined,
bucket,
accessLevel,
key,
optionsHash,
});
await cacheMultipartUpload(resumableUploadsCache, uploadCacheKey, {
uploadId: UploadId!,
bucket,
key,
finalCrc32,
fileName: data instanceof File ? data.name : '',
});
}
return {
uploadId: UploadId!,
cachedParts: [],
finalCrc32,
};
}
};