Filesystem 
Overview 
Shopware 6 stores and processes a wide variety of files. This goes from product images or videos to generated documents such as invoices or delivery notes. This data should be stored securely, and backups should be generated regularly. Therefore, it is advisable to set up storage service, which scales with the size of the data, performs backups, and ensures data redundancy. In addition, for cluster setups with multiple setups, it is necessary to share the files via external storage so that each app server can access the corresponding data.
Flysystem overview 
Shopware 6 can be used with several cloud storage providers. It uses Flysystem to provide a common interface between different providers as well as the local file system. This enables your shops to read and write files through a common interface.
The file system can be divided into multiple adapters. Each adapter can handle one or more of the following directories: media, sitemaps, and more. Of course, you can also use the same configuration for:
- private files: invoices, delivery notes, plugin files, etc.
- public files: product pictures, media files, plugin files in general
- theme files
- sitemap files
- bundle assets files
Configuration 
The configuration for file storage of Shopware 6 resides in the general bundle configuration:
<project root>
└── config
   └── packages
      └── shopware.ymlTo set up a non-default filesystem for your shop, you need to add the filesystem: map to the shopware.yml. Under this key, you can separately define your storage for the public, private, theme, sitemap, and asset (bundle assets).
INFO
You can also change the URL of the file systems. This is useful if you want to use a different domain for your files. For example, you can use a CDN for your public files.
shopware:
  filesystem:
    public:
      url: "{url-to-your-public-files}"
      # The Adapter Configuration
    private:
      visibility: "private"
      # The Adapter Configuration
    theme:
      url: "{url-to-your-theme-files}"
      # The Adapter Configuration
    asset:
      url: "{url-to-your-asset-files}"
      # The Adapter Configuration
    sitemap:
      url: "{url-to-your-sitemap-files}"
      # The Adapter ConfigurationUsing YAML anchors to avoid repetition 
You can use YAML anchors to avoid repeating the same configuration for multiple filesystems. This is particularly useful when you want to use the same storage backend for public, theme, and sitemap files:
shopware:
  filesystem:
    public: &s3_config
      type: "amazon-s3"
      url: "{{S3_URL}}"
      config:
        bucket: "{{AWS_BUCKET}}"
        region: "{{AWS_REGION}}"
        endpoint: "{{AWS_ENDPOINT}}"
        use_path_style_endpoint: true
        credentials:
          key: "{{AWS_ACCESS_KEY_ID}}"
          secret: "{{AWS_SECRET_ACCESS_KEY}}"
    theme: *s3_config
    sitemap: *s3_configIn this example, the &s3_config creates an anchor that can be referenced with *s3_config in other filesystem configurations, avoiding duplication.
Fallback adapter configuration 
By default, the configuration for the theme, asset and sitemap filesystem will use the configuration from the public filesystem if they are not specifically configured. This means when you want to change the configuration used for the public filesystem, but the others should use the old configuration you have to set them explicitly.
E.g. before you had the following configuration:
shopware:
  filesystem:
    public:
      type: "local"
      url: "https://your.domain/public"
      config:
        root: "%kernel.project_dir%/public"Now you want to change the public filesystem to use an S3 adapter, but the theme, asset and sitemap filesystem should still use the local adapter. You have to set them explicitly:
shopware:
  filesystem:
    public:
      url: "{{S3_URL}}"
      type: "amazon-s3"
      config:
        bucket: "{{AWS_BUCKET}}"
        region: "{{AWS_REGION}}"
        endpoint: "{{AWS_ENDPOINT}}"
        credentials:
          key: "{{AWS_ACCESS_KEY_ID}}"
          secret: "{{AWS_SECRET_ACCESS_KEY}}"
    theme:
      type: "local"
      url: "https://your.domain/public"
      config:
        root: "%kernel.project_dir%/public"
    asset:
      type: "local"
      url: "https://your.domain/public"
      config:
        root: "%kernel.project_dir%/public"
    sitemap:
      type: "local"
      url: "https://your.domain/public"
      config:
        root: "%kernel.project_dir%/public"Additional configuration 
If you want to regulate the uploaded file types, then you could add the keys allowed_extensionsfor the public filesystem or private_local_download_strategy for the private filesystem. With the private_local_download_strategy key you could choose the download strategy for private files (e.g., the downloadable products):
shopware:
  filesystem:
    public:
      # The Adapter Configuration
    private:
      # The Adapter Configuration
    allowed_extensions: # Array with allowed file extensions for public filesystem
    private_allowed_extensions: # Array with allowed file extensions for private filesystem
    private_local_download_strategy: # Name of the download strategy: php, x-sendfile or x-accelThe following download strategies are valid:
- php(default): A streamed response of content type- application/octet-streamwith binary data
- x-sendfile(Apache only): X-Sendfile allows you to use PHP to instruct the server to send a file to a user, without having to load that file into PHP. You must have the- mod_xsendfileApache module installed.
- x-accel(Nginx only): X-accel allows for internal redirection to a location determined by a header returned from a backend. See the example configuration.
CDN configuration 
If your public files are available on a CDN, you can use the following config to serve images and other assets via that CDN.
# <project root>/config/packages/prod/shopware.yml
shopware:
  filesystem:
    public:
      url: "YOUR_CDN_URL"
      type: "local"
      config:
        root: "%kernel.project_dir%/public"INFO
Be aware of the prod in the config path. CDNs are typically for production environments, but you can also set them for all environments in config/packages/shopware.yml.
Supported adapter configurations 
Local 
shopware:
    filesystem:
      {ADAPTER_NAME}:
        type: "local"
        config:
          root: "%kernel.project_dir%/public"Amazon S3 
In order to use the S3 adapter you need to install the league/flysystem-async-aws-s3 package.
composer require league/flysystem-async-aws-s3Example configuration:
shopware:
    filesystem:
      {ADAPTER_NAME}:
        type: "amazon-s3"
        url: "https://your-cloudfront-url"
        visibility: "private" # Default is "public", can be set only on shopware.filesystem.private
        config:
            bucket: "{your-public-bucket-name}"
            region: "{your-bucket-region}"
            endpoint: "{your-s3-provider-endpoint}"
            root: "{your-root-folder}"
            # Optional, otherwise will be automatically discovered with AWS content discovery
            credentials:
              key: '{your-access-key}'
              secret: '{your-secret-key}'If your S3 provider does not use buckets as subdomain like Minio in default configuration, you need to set use_path_style_endpoint to true inside config.
Google Cloud Platform 
In order to use the Google Cloud Platform adapter you need to install the league/flysystem-google-cloud-storage package.
composer require league/flysystem-google-cloud-storageExample configuration:
shopware:
    filesystem:
      {ADAPTER_NAME}:
        type: "google-storage"
        url: "https://storage.googleapis.com/{your-public-bucket-name}"
        visibility: "private" # Default is "public", can be set only on shopware.filesystem.private
        config:
            bucket: "{your-public-bucket-name}"
            projectId: "{your-project-id}"
            keyFilePath: "{path-to-your-keyfile}"The bucket needs to use the "Fine-grained" ACL mode. This is required so that Shopware can manage the ACL of the objects.
Add your own adapter 
To create your own adapter, check out the official Flysystem guide.
To make your adapter available in Shopware, you will need to create an AdapterFactory for your Flysystem provided adapter. An example of that could look like this:
<?php
use Shopware\Core\Framework\Adapter\Filesystem\Adapter\AdapterFactoryInterface;
use League\Flysystem\AdapterInterface;
class MyFlysystemAdapterFactory implements AdapterFactoryInterface
{
    public function getType(): string
    {
        return 'my-adapter-prefix'; // This must match with the type in the yaml file
    }
    public function create(array $config): AdapterInterface
    {
        // $config contains the given config from the yaml
        return new MyFlysystemAdapter($config);
    }
}This new class needs to be registered in the DI with the tag shopware.filesystem.factory to be usable.