Storage
In Loco Storage, we facilitate working with files through multiple operations. Storage can be in-memory, on disk, or use cloud services such as AWS S3, GCP, and Azure.
Loco supports simple storage operations and advanced features like mirroring data or backup strategies with different failure modes.
By default, in-memory and disk storage come out of the box. To work with cloud providers, you should specify the following features:
storage_aws_s3
storage_azure
storage_gcp
all_storage
By default loco initialize a Null
provider, meaning any work with the storage will return an error.
Setup
Add the after_context
function as a Hook in the app.rs
file and import the storage
module from loco_rs
.
use storage;
async
This hook returns a Storage instance that holds all storage configurations, covered in the next sections. This Storage instance is stored as part of the application context and is available in controllers, endpoints, task workers, and more.
Glossary
StorageDriver | Trait implementation something that does storage |
Storage | Abstraction implementation for managing one or more storage drivers. |
Strategy | Trait implementing various strategies for Storage, such as mirror or backup. |
FailureMode | Implemented within each Strategy, determining how to handle operations in case of failures. |
Initialize Storage
Storage can be configured with a single driver or multiple drivers.
Single Store
In this example, we initialize the in-memory driver and create a new storage with the single function.
use storage;
async
Multiple Drivers
For advanced usage, you can set up multiple drivers and apply smart strategies that come out of the box. Each strategy has its own set of failure modes that you can decide how to handle.
Creating multiple drivers:
use crate;
let aws_1 = new;
let azure = new;
let aws_2 = new;
Mirror Strategy:
You can keep multiple services in sync by defining a mirror service. A mirror service replicates uploads, deletes, rename and copy across two or more subordinate services. The download behavior redundantly retrieves data, meaning if the file retrieval fails from the primary, the first file found in the secondaries is returned.
Behaviour
After creating the three store instances, we need to create the mirror strategy instance and define the failure mode. The mirror strategy expects the primary store and a list of secondary stores, along with failure mode options:
MirrorAll
: All secondary storages must succeed. If one fails, the operation continues to the rest but returns an error.AllowMirrorFailure
: The operation does not return an error when one or more mirror operations fail.
The failure mode is relevant for upload, delete, move, and copy.
Example:
// Define the mirror strategy by setting the primary store and secondary stores by names.
let strategy = Box newas ;
// Create the storage with the store mapping and the strategy.
let storage = new;
Backup Strategy:
You can back up your operations across multiple storages and control the failure mode policy.
After creating the three store instances, we need to create the backup strategy instance and define the failure mode. The backup strategy expects the primary store and a list of secondary stores, along with failure mode options:
BackupAll
: All secondary storages must succeed. If one fails, the operation continues to the rest but returns an error.AllowBackupFailure
: The operation does not return an error when one or more backup operations fail.AtLeastOneFailure
: At least one operation should pass.CountFailure
: The given number of backups should pass.
The failure mode is relevant for upload, delete, move, and copy. The download always retrieves the file from the primary.
Example:
// Define the backup strategy by setting the primary store and secondary stores by names.
let strategy: = Box newas ;
let storage = new;
Create Your Own Strategy
In case you have a specific strategy, you can easily create it by implementing the StorageStrategy and implementing all store functionality.
Usage In Controller
Follow this example, make sure you enable multipart
feature in axum crate.
async
Testing
By testing file storage in your controller you can follow this example:
use *;
async