diff --git a/Classes/Driver/AmazonS3Driver.php b/Classes/Driver/AmazonS3Driver.php
index cbb3cd84..3493845c 100644
--- a/Classes/Driver/AmazonS3Driver.php
+++ b/Classes/Driver/AmazonS3Driver.php
@@ -80,6 +80,14 @@ class AmazonS3Driver extends AbstractHierarchicalFilesystemDriver implements Str
*/
protected $baseUrl = '';
+ /**
+ * Folder that is used as root folder.
+ * Must be empty or have a trailing slash.
+ *
+ * @var string
+ */
+ protected $baseFolder = '';
+
/**
* Stream wrapper protocol: Will be set in the constructor
*
@@ -219,6 +227,7 @@ public function processConfiguration()
public function initialize()
{
$this->initializeBaseUrl()
+ ->initializeBaseFolder()
->initializeSettings()
->initializeClient();
$this->resetRequestCache();
@@ -239,7 +248,7 @@ public function getPublicUrl($identifier)
{
$uriParts = GeneralUtility::trimExplode('/', ltrim($identifier, '/'), true);
$uriParts = array_map('rawurlencode', $uriParts);
- return $this->baseUrl . '/' . implode('/', $uriParts);
+ return $this->baseUrl . '/' . $this->addBaseFolder(implode('/', $uriParts));
}
/**
@@ -395,14 +404,14 @@ public function addFile($localFilePath, $targetFolderIdentifier, $newFileName =
if (filesize($localFilePath) === 0) { // Multipart uploader would fail to upload empty files
$this->s3Client->upload(
$this->configuration['bucket'],
- $targetIdentifier,
+ $this->addBaseFolder($targetIdentifier),
''
);
} else {
$multipartUploadAdapter = GeneralUtility::makeInstance(MultipartUploaderAdapter::class, $this->s3Client);
$multipartUploadAdapter->upload(
$localFilePath,
- $targetIdentifier,
+ $this->addBaseFolder($targetIdentifier),
$this->configuration['bucket'],
$this->getCacheControl($targetIdentifier)
);
@@ -525,7 +534,7 @@ public function getFileForLocalProcessing($fileIdentifier, $writable = true)
$temporaryPath = $this->getTemporaryPathForFile($fileIdentifier);
$this->s3Client->getObject([
'Bucket' => $this->configuration['bucket'],
- 'Key' => $fileIdentifier,
+ 'Key' => $this->addBaseFolder($fileIdentifier),
'SaveAs' => $temporaryPath,
]);
if (!is_file($temporaryPath)) {
@@ -600,7 +609,7 @@ public function getFileContents($fileIdentifier)
{
$result = $this->s3Client->getObject([
'Bucket' => $this->configuration['bucket'],
- 'Key' => $fileIdentifier
+ 'Key' => $this->addBaseFolder($fileIdentifier)
]);
return (string)$result['Body'];
}
@@ -1084,6 +1093,19 @@ protected function initializeBaseUrl()
return $this;
}
+ /**
+ * Set the $baseFolder variable from configuration
+ */
+ protected function initializeBaseFolder(): self
+ {
+ $baseFolder = $this->configuration['baseFolder'] ?? '';
+ if ($baseFolder != '') {
+ $baseFolder = rtrim($baseFolder, '/') . '/';
+ }
+ $this->baseFolder = $baseFolder;
+ return $this;
+ }
+
/**
* initializeSettings
*
@@ -1184,6 +1206,30 @@ protected function testConnection()
}
}
+ /**
+ * Prefix the given file/path identifier with the base folder.
+ * Used by internal functions that directly speak with S3Client
+ */
+ protected function addBaseFolder(string $identifier): string
+ {
+ if ($this->baseFolder) {
+ $identifier = str_replace('//', '/', $this->baseFolder . $identifier);
+ }
+ return $identifier;
+ }
+
+ /**
+ * Remove base folder prefix from a given S3 path.
+ * Used when returning information from S3.
+ */
+ protected function removeBaseFolder(string $identifier): string
+ {
+ if ($this->baseFolder) {
+ $identifier = substr($identifier, strlen($this->baseFolder));
+ }
+ return $identifier;
+ }
+
/**
* @return \TYPO3\CMS\Core\Messaging\FlashMessageQueue
*/
@@ -1238,7 +1284,7 @@ protected function getMetaInfo($identifier)
try {
$metadata = $this->s3Client->headObject([
'Bucket' => $this->configuration['bucket'],
- 'Key' => $identifier
+ 'Key' => $this->addBaseFolder($identifier)
])->toArray();
$metaInfoDownloadAdapter = GeneralUtility::makeInstance(MetaInfoDownloadAdapter::class);
$metaInfo = $metaInfoDownloadAdapter->getMetaInfoFromResponse($this, $identifier, $metadata);
@@ -1325,7 +1371,7 @@ protected function getObjectPermissions($identifier)
try {
$response = $this->s3Client->getObjectAcl([
'Bucket' => $this->configuration['bucket'],
- 'Key' => $identifier
+ 'Key' => $this->addBaseFolder($identifier)
])->toArray();
// Until the SDK provides any useful information about folder permissions, we take full access for granted as long as one user with full access exists.
@@ -1363,10 +1409,10 @@ protected function getObjectPermissions($identifier)
*/
protected function deleteObject(string $identifier): bool
{
- $this->s3Client->deleteObject(['Bucket' => $this->configuration['bucket'], 'Key' => $identifier]);
+ $this->s3Client->deleteObject(['Bucket' => $this->configuration['bucket'], 'Key' => $this->addBaseFolder($identifier)]);
$this->flushMetaInfoCache($identifier);
$this->resetRequestCache();
- return !$this->s3Client->doesObjectExist($this->configuration['bucket'], $identifier);
+ return !$this->s3Client->doesObjectExist($this->configuration['bucket'], $this->addBaseFolder($identifier));
}
/**
@@ -1394,7 +1440,7 @@ protected function createObject($identifier, $body = '', $overrideArgs = [])
$this->normalizeIdentifier($identifier);
$args = [
'Bucket' => $this->configuration['bucket'],
- 'Key' => $identifier,
+ 'Key' => $this->addBaseFolder($identifier),
'Body' => $body
];
$this->s3Client->putObject(array_merge_recursive($args, $overrideArgs));
@@ -1453,7 +1499,7 @@ protected function getStreamWrapperPath($file)
throw new \RuntimeException('Type "' . gettype($file) . '" is not supported.', 1325191178);
}
$this->normalizeIdentifier($identifier);
- return $basePath . $identifier;
+ return $basePath . $this->addBaseFolder($identifier);
}
/**
@@ -1517,7 +1563,7 @@ protected function getListObjects($identifier, $overrideArgs = [])
{
$args = [
'Bucket' => $this->configuration['bucket'] ?? '',
- 'Prefix' => $identifier,
+ 'Prefix' => $this->addBaseFolder($identifier),
];
$result = $this->getCachedResponse('listObjectsV2', array_merge_recursive($args, $overrideArgs));
// Cache the given meta info
@@ -1525,14 +1571,31 @@ protected function getListObjects($identifier, $overrideArgs = [])
// with many files we come to the recursion which lessens the home of a cache hit, so we do not create the cache here
if (isset($result['Contents']) && is_array($result['Contents'])) {
- foreach ($result['Contents'] as $content) {
- $fileIdentifier = $content['Key'];
+ $baseFolderSelfKey = null;
+ foreach ($result['Contents'] as $key => &$content) {
+ $content['Key'] = $this->removeBaseFolder($content['Key']);
+ if ($content['Key'] === '') {
+ $baseFolderSelfKey = $key;
+ continue;
+ }
+ $fileIdentifier = $identifier . $content['Key'];
$this->normalizeIdentifier($fileIdentifier);
$cacheIdentifier = md5($fileIdentifier);
if (!$this->metaInfoCache->has($cacheIdentifier)) {
$this->metaInfoCache->set($cacheIdentifier, $metaInfoDownloadAdapter->getMetaInfoFromResponse($this, $fileIdentifier, $content));
}
}
+ unset($content);
+ if ($baseFolderSelfKey !== null) {
+ unset($result['Contents'][$baseFolderSelfKey]);
+ }
+ }
+
+ if (isset($result['CommonPrefixes'])) {
+ foreach ($result['CommonPrefixes'] as &$prefix) {
+ $prefix['Prefix'] = $this->removeBaseFolder($prefix['Prefix']);
+ }
+ unset($prefix);
}
if (isset($overrideArgs['MaxKeys']) && $overrideArgs['MaxKeys'] <= 1000) {
@@ -1586,8 +1649,8 @@ protected function copyObject($identifier, $targetIdentifier)
{
$this->s3Client->copyObject([
'Bucket' => $this->configuration['bucket'],
- 'CopySource' => $this->configuration['bucket'] . '/' . $identifier,
- 'Key' => $targetIdentifier,
+ 'CopySource' => $this->configuration['bucket'] . '/' . $this->addBaseFolder($identifier),
+ 'Key' => $this->addBaseFolder($targetIdentifier),
'CacheControl' => $this->getCacheControl($targetIdentifier)
]);
$this->flushMetaInfoCache($targetIdentifier);
diff --git a/Configuration/FlexForm/AmazonS3DriverFlexForm.xml b/Configuration/FlexForm/AmazonS3DriverFlexForm.xml
index 4c2f0211..0846f54f 100644
--- a/Configuration/FlexForm/AmazonS3DriverFlexForm.xml
+++ b/Configuration/FlexForm/AmazonS3DriverFlexForm.xml
@@ -187,6 +187,16 @@
+
+
+
+ LLL:EXT:aus_driver_amazon_s3/Resources/Private/Language/locallang_flexform.xlf:driverConfiguration.baseFolder-description
+
+ input
+ 30
+
+
+
diff --git a/Resources/Private/Language/locallang_flexform.xlf b/Resources/Private/Language/locallang_flexform.xlf
index 9799c7a4..5c1dbd5b 100644
--- a/Resources/Private/Language/locallang_flexform.xlf
+++ b/Resources/Private/Language/locallang_flexform.xlf
@@ -33,6 +33,12 @@
+
+
+
+
+
+