phlib/flysystem-pdo

A Flysystem adapter for storing files in a database using PDO

2.1.0 2022-09-01 06:29 UTC

This package is auto-updated.

Last update: 2025-01-03 09:38:51 UTC


README

Code Checks Codecov Latest Stable Version Total Downloads Licence

This is a PDO Adapter for the League's Flysystem.

  • Uses multiple tables.
  • Stores files in chunks.
  • Option to compress the file when stored.

This implementation is optimised for use with large files when using the streams. It avoids loading the complete file into memory, preferring to store files during operation on the local file system.

Usage

use Phlib\Flysystem\Pdo\PdoAdapter;
use League\Flysystem\Filesystem;

$pdo        = new \PDO('mysql:host=hostname;dbname=database_name', 'username', 'password');
$adapter    = new PdoAdapter($pdo);
$filesystem = new Filesystem($adapter);

Configuration on write and writeStream

use League\Flysystem\Config;

$config = new Config([
    'enable_compression' => false,
    'visibility'         => AdapterInterface::VISIBILITY_PUBLIC
]); 
$adapter->writeStream('/path/to/file.zip', $handle, $config);

Adapter Configuration

Example

use League\Flysystem\Config;

$config = new Config([
    'table_prefix'            => 'flysystem',
    'enable_compression'      => true,
    'chunk_size'              => 1048576,
    'temp_dir'                => '/var/tmp',
    'disable_mysql_buffering' => true
]);
$adapter = new PdoAdapter($pdo, $config);

File Configuration

The following optional file configurations supplement the standard behaviour. The sample schemas include the columns, but they are optional and may be omitted.

Expiry

By specifying 'expiry' as a configuration parameter when writing or updating a file the PdoAdatper will store the value in a column called 'expiry'. When the information about the file is selected out, if the expiry exists and can be parsed by strtotime, then the expiry time will be evaluated. False is returned if the file doesn't exist or has expired.

Example

$config = new Config(['expiry' => date('Y-m-d H:i:s', strtotime('+2 days'))]);
$adapter->write($path, $content, $config);

The expiry is now part of the file description.

$data = $adapter->getMetadata($path);
[
    'path' => '...',
    '...',
    'expiry' => ''
]

Additional Metadata

It's possible to store additional meta data about a file or directory. This could include owner, permissions or groups for example. The information is stored as a JSON encoded string in whatever form you provide. One the item is retrieved from the Filesystem the additional meta information is provided in the same format it was originally provided.

Example

$config = new Config(['meta' => ['owner' => 'John Smith', 'permissions' => 600]]);
$adapter->write($path, $content, $config);

Those details are now part of the file description.

$data = $adapter->getMetadata($path);
[
    'path' => '...',
    '...',
    'meta' => [
        'owner' => 'John Smith',
        'permissions' => 600
    ]
]

Schema

Schemas can be found in the schema directory. Specific types can be changed based on requirements. All field names should remain the same. Notes about the DB specific definitions are below.

MySQL Notes

  • The path column is set to allow up to 255 characters.
  • The size column has been set to a unsigned INT type to allow for convenient searching. This allows up to 4G files to be recorded. This can be changed to a VARCHAR if searching is not required.
  • The size of chunks, allows for up to 16M per chunk.

Memory Usage (and gotchas)

Any use of read, write or update with large files will cause problems with memory usage. The associated stream methods have been optimised to use as little memory as possible. The adapter first saves the file to the local filesystem before transferring it to the database.

Buffering

On MySQL, the default behaviour is to buffer all query results. When reading a file back from the database this could cause memory problems. There is a configuration option which disables the buffering. This has the side effect that the pdo connection specified in the constructor is altered to set this attribute.

Compression

Compression is especially useful when storing text based files. The compression option defaults to on. The side effect of this is that when reading files back some files may cause larger than expected memory usage. As an example, a very large file filled with a single letter 'a', can be compressed to a tiny size. When that file is read, the tiny chunk is expanded and will fill the memory.

When a file is stored, the setting for compression is stored with it. This can not be changed.

Chunking

Chunking has been implemented to aid where systems have been set up for replication. Packet sizes are a consideration here.

License

This package is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.

This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.

You should have received a copy of the GNU Lesser General Public License along with this program. If not, see http://www.gnu.org/licenses/.