使用 Laravel 5.0 存储门面将元数据、标头(Expires、CacheControl)添加到上传到 Amazon S3 的文件中

Posted

技术标签:

【中文标题】使用 Laravel 5.0 存储门面将元数据、标头(Expires、CacheControl)添加到上传到 Amazon S3 的文件中【英文标题】:Add Metadata, headers (Expires, CacheControl) to a file uploaded to Amazon S3 using the Laravel 5.0 Storage facade 【发布时间】:2015-05-13 10:03:13 【问题描述】:

我正在尝试了解如何将元数据或标头(Expires、CacheControl 等)添加到使用 Laravel 5.0 存储门面上传的文件中。我已将此处的页面用作参考。

http://laravel.com/docs/5.0/filesystem

以下代码可以正常工作:

Storage::disk('s3')->put('/test.txt', 'test');

经过挖掘,我还发现有一个“可见性”参数将 ACL 设置为“公共读取”,因此以下内容也可以正常工作。

Storage::disk('s3')->put('/test.txt', 'test', 'public');

但我希望能够为文件的标题设置一些其他值。我尝试了以下方法:

Storage::disk('s3')->put('/index4.txt', 'test', 'public', array('Expires'=>'Expires, Fri, 30 Oct 1998 14:19:41 GMT'));

哪个不行,我也试过了:

Storage::disk('s3')->put('/index4.txt', 'test', array('ACL'=>'public-read'));

但这会产生一个错误,即“可见性”参数无法从字符串转换为数组。我检查了 AwsS3Adapter 的来源,似乎有选项代码,但我似乎看不到如何正确传递它们。我认为它需要以下内容:

protected static $metaOptions = [
    'CacheControl',
    'Expires',
    'StorageClass',
    'ServerSideEncryption',
    'Metadata',
    'ACL',
    'ContentType',
    'ContentDisposition',
    'ContentLanguage',
    'ContentEncoding',
];

任何有关如何完成此操作的帮助将不胜感激。

【问题讨论】:

【参考方案1】:

首先,您需要致电getDriver,以便发送一系列选项。然后您需要将选项作为数组发送。

所以对于你的例子:

Storage::disk('s3')->getDriver()->put('/index4.txt', 'test', [ 'visibility' => 'public', 'Expires' => 'Expires, Fri, 30 Oct 1998 14:19:41 GMT']);

请注意,如果您要设置 Cache-Control,则必须将其作为 CacheControl 传递。对于具有非字母数字字符的其他键,这很可能是正确的。

【讨论】:

如果您想在配置中设置全局默认值,请查看我的回答 (***.com/a/46145866/7377984) 谢谢你,在我需要设置“ContentType”元数据之前,没有 getDriver() 一切正常。 感谢您指出 Cache-Control => CacheControl 问题。节省了我很多时间【参考方案2】:

如果你想在标题中使用全局默认值,这在 Laravel 5.4 中有效。像这样更改您的 config/filesystems.php 文件:

s3' => [
    'driver' => 's3',
    'key' => env('AWS_KEY'),
    'secret' => env('AWS_SECRET'),
    'region' => env('AWS_REGION'),
    'bucket' => env('AWS_BUCKET'),
    'options' => ['CacheControl' => 'max-age=315360000, no-transform, public', 
                  'ContentEncoding' => 'gzip']
],

【讨论】:

为什么投反对票?此解决方案有效,如果您不相信它有效,想知道您为什么这么认为? 哇,这应该是最好的答案,但有人无缘无故地否决了它 我也不确定为什么这也被否决了,它在 Laravel 5.4 中对我来说非常有效。 即使在 Laravel 5.8 中,这也是一个可行的解决方案。很好,谢谢! 感谢您,我认为它可能已被否决,因为这是一个全局解决方案,而不是每个文件/上传解决方案。您可能不想在全局级别上进行设置。当我问这个问题时我没有。【参考方案3】:

在尝试了上述答案并且无法添加客户用户元数据之后,事实证明,在挖掘 SDK 代码之后,它比我想象的要容易一些(假设 $path 是图像文件的路径) .我似乎也不需要调用 getDriver() 方法,不太确定这是否与当前版本的 AWS 开发工具包有任何区别。

Storage::put(
    'image.jpg',
    file_get_contents($path),
    [
        'visibility' => 'public',
        'Metadata' => [
            'thumb' => '320-180',
        ],
    ]
);

所以现在如果您在 S3 中查看新上传的文件,您将看到自定义元数据:

希望这对某人有所帮助。

【讨论】:

【参考方案4】:

@Paras 的回答很好。但是有一点会让新手感到困惑:

'options'     => [
    'Expires' => gmdate('D, d M Y H:i:s GMT', strtotime('+1 month')),
    >>> WRONG visibility' => 'public', WRONG <<<
]

如果您想为 HEADERS 定义全局选项,选项数组是正确的方法。但是,如果您还想定义可见性,则不能将其混淆。可见性必须在选项数组之外定义。

?

'visibility'  => 'public',
'options'     => ['Expires' => gmdate('D, d M Y H:i:s GMT', strtotime('+1 month'))]

【讨论】:

【参考方案5】:

这是一个从 Laravel 5.8 开始如何将文件上传到 S3 的示例,其中包含过期和缓存控制标头,例如:

Storage::put($directory . '/' . $imageName, 
            $image, [
              'visibility' => 'public',
              'Expires' => gmdate('D, d M Y H:i:s \G\M\T', time() + (60 * 60 * 24 * 7)),
              'CacheControl' => 'max-age=315360000, no-transform, public',
        ]);

如果您正在测试,请不要忘记取消选中 Chrome 中的“禁用缓存”复选框,但它似乎永远无法正常工作,这让我在一个小时内感到很糟糕,即使我最终得到了浏览器也不会缓存内容S3 中的标题。

【讨论】:

【参考方案6】:

嘿,我解决了这个问题,你需要创建一个自定义 S3 文件系统

首先,新建一个文件 CustomS3Filesystem.php 并保存到 app/providers,这个自定义 S3 文件系统使用 S3 Adapter,但是你可以添加元数据和头文件。

<?php namespace App\Providers;
use Storage;
use League\Flysystem\Filesystem;
use Aws\S3\S3Client;
use League\Flysystem\AwsS3v2\AwsS3Adapter as S3Adapter;
use Illuminate\Support\ServiceProvider;

class CustomS3Filesystem extends ServiceProvider 

public function boot()

    Storage::extend('s3_custom', function($app, $config)
    
        $s3Config = array_only($config, ['key', 'region', 'secret', 'signature', 'base_url']);
        $flysystemConfig = ['mimetype' => 'text/xml'];
        $metadata['cache_control']='max-age=0, no-cache, no-store, must-revalidate';
        return new Filesystem(new S3Adapter(S3Client::factory($s3Config), $config['bucket'], null, ['mimetype' => 'text/xml', 'Metadata' => $metadata]), $flysystemConfig);
    );

public function register()

    //


将提供者添加到 config/app.php 的提供者列表中

'App\Providers\CustomS3Filesystem',

在 config/filesystems 中创建新的文件系统名称

's3-new' => [
            'driver' => 's3_custom',
            'key'    => 'XXX',
            'secret' => 'XXX',
            'bucket' => 'XXX',
        ],

使用新创建的自定义 s3 适配器

Storage::disk('s3-new')->put(filename, file_get_contents($file), public);

我使用 laravel 文档自定义 s3 适配器 http://laravel.com/docs/5.0/filesystem#custom-filesystems

希望对你有帮助。

【讨论】:

【参考方案7】:

我正在使用 Laravel 4.2,但我认为我的解决方案也可能对 Laravel 5.0 有所帮助(不能肯定地说,因为我还没有尝试升级)。您需要为您正在使用的 Flysystem 驱动程序更新配置中的元选项。在我的例子中,我创建了一个名为 s3static 的连接来访问我存储不会更改的图像的存储桶。

我的配置文件:

's3static' => [
            'driver'     => 'awss3',
            'key'        => 'my-key',
            'secret'     => 'my-secret',
            'bucket'     => 'my-bucket',
            // 'region'     => 'your-region',
            // 'base_url'   => 'your-url',
            'options'    => array(
                'CacheControl' => 'max_age=2592000'
            ),
            // 'prefix'     => 'your-prefix',
            // 'visibility' => 'public',
            // 'eventable'  => true,
            // 'cache'      => 'foo'
        ],

现在,当我使用此连接将任何文件放到 S3 上时,它们具有 Cache-Control 元数据集。

【讨论】:

这根本没有记录。【参考方案8】:

为了扩展@sergiodebcn 的答案,这里是同样适用于 S3 v3 和最新 Laravel 的 CustomS3Filesystem 类。注意我已经删除了 XML mimetype 并设置了 5 天的缓存时间:

namespace App\Providers;

use Illuminate\Support\Arr;
use Storage;
use League\Flysystem\Filesystem;
use Aws\S3\S3Client;
use League\Flysystem\AwsS3v3\AwsS3Adapter as S3Adapter;
use Illuminate\Support\ServiceProvider;

class CustomS3Filesystem extends ServiceProvider


    /**
     * Format the given S3 configuration with the default options.
     *
     * @param  array  $config
     * @return array
     */
    protected function formatS3Config(array $config)
    
        $config += ['version' => 'latest'];

        if ($config['key'] && $config['secret']) 
            $config['credentials'] = Arr::only($config, ['key', 'secret']);
        

        return $config;
    

    /**
     * Bootstrap a custom filesystem
     *
     * @return void
     */
    public function boot()
    
        Storage::extend('s3_custom', function($app, $config)
        
            $s3Config = $this->formatS3Config($config);
            return new Filesystem(
                new S3Adapter(
                    new S3Client($s3Config),
                    $config['bucket'],
                    null,
                    [
                        'CacheControl'  => 'max-age=432000'
                    ]
                )
            );
        );
    

    public function register()
    
        //
    

【讨论】:

【参考方案9】:

对于 Laravel 9 用户来说,这变得更加容易。您不再需要致电-&gt;getDriver()。您可以直接将选项传递给put 命令。

Storage::disk('s3')->put('/index.txt', 'file content', [
    // S3 Object ACL
    'visibility' => 'public', // or 'private',

    // HTTP Headers
    'CacheControl' => 'public,max-age=315360000',
    'ContentDisposition' => 'attachment; filename="index.txt"',
    'Expires' => 'Thu, 12 Feb 2032 08:24:43 GMT',

    // Metadata or other S3 options
    'MetadataDirective' => 'REPLACE'
    'Metadata' => [
        'Custom-Key' => 'test',
    ],
])

如果您需要其他头文件或选项,请查看 flysystem 源代码以获取所有可用的头文件和选项。

https://github.com/thephpleague/flysystem-aws-s3-v3/blob/master/src/AwsS3Adapter.php#L38

public const AVAILABLE_OPTIONS = [
    'ACL',
    'CacheControl',
    'ContentDisposition',
    'ContentEncoding',
    'ContentLength',
    'ContentType',
    'Expires',
    'GrantFullControl',
    'GrantRead',
    'GrantReadACP',
    'GrantWriteACP',
    'Metadata',
    'MetadataDirective',
    'RequestPayer',
    'SSECustomerAlgorithm',
    'SSECustomerKey',
    'SSECustomerKeyMD5',
    'SSEKMSKeyId',
    'ServerSideEncryption',
    'StorageClass',
    'Tagging',
    'WebsiteRedirectLocation',
];

【讨论】:

以上是关于使用 Laravel 5.0 存储门面将元数据、标头(Expires、CacheControl)添加到上传到 Amazon S3 的文件中的主要内容,如果未能解决你的问题,请参考以下文章

Laravel密码经纪人门面

laravel门面和服务提供者使用

将元标记添加到 laravel 页面

Laravel 控制反转和门面模式概念详解

将元数据存储到 Jackrabbit 存储库中

laravel 门面