Skip to content

Commit

Permalink
adding file upload (#423)
Browse files Browse the repository at this point in the history
* adding file upload

* init drivers internally

* add docs

* add docs
  • Loading branch information
kaplanelad authored Feb 8, 2024
1 parent 5d1ab37 commit a15d354
Show file tree
Hide file tree
Showing 32 changed files with 3,142 additions and 61 deletions.
18 changes: 12 additions & 6 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -23,13 +23,16 @@ rust-version.workspace = true

[features]
default = ["auth_jwt", "cli", "with-db"]

auth_jwt = ["dep:jsonwebtoken"]
cli = ["dep:clap"]
testing = ["dep:axum-test"]
with-db = ["dep:sea-orm", "dep:sea-orm-migration"]
channels = ["dep:socketioxide"]

# Storage features
all_storage = ["storage_aws_s3", "storage_azure", "storage_gcp"]
storage_aws_s3 = ["object_store/aws"]
storage_azure = ["object_store/azure"]
storage_gcp = ["object_store/gcp"]

[dependencies]

Expand Down Expand Up @@ -102,12 +105,11 @@ bcrypt = { version = "0.15.0", optional = true }
validator = { version = "0.16.1", features = ["derive"] }
futures-util = "0.3"
tower = "0.4"
hyper = "1"
hyper = "1.1"
mime = "0.3"
bytes = "1.1"

axum-test = { version = "14.0.0-rc.1", optional = true }

axum-test = { version = "14.3.0", optional = true }

# gen
rrgen = "0.5.3"
Expand All @@ -119,8 +121,13 @@ cfg-if = "1"
uuid = { version = "1.6", features = ["v4"] }
requestty = "0.5.0"

# A socket.io server implementation
socketioxide = { version = "0.10.0", features = ["state"], optional = true }


# File Upload
object_store = { version = "0.9.0", default-features = false }

[workspace.dependencies]
async-trait = { version = "0.1.74" }
axum = { version = "0.7.1", features = ["macros"] }
Expand All @@ -137,7 +144,6 @@ features = [
"sqlx-sqlite",
]


[dev-dependencies]
rstest = "0.18.2"
insta = { version = "1.34.0", features = ["redactions", "yaml", "filters"] }
41 changes: 41 additions & 0 deletions docs-site/content/docs/testing/storage.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
+++
title = "Storage"
description = ""
date = 2021-05-01T18:20:00+00:00
updated = 2021-05-01T18:20:00+00:00
draft = false
weight = 23
sort_by = "weight"
template = "docs/page.html"

[extra]
lead = ""
toc = true
top = false
+++

By testing file storage in your controller you can follow this example:
```rust
#[tokio::test]
#[serial]
async fn can_register() {
testing::request::<App, _, _>(|request, ctx| async move {
let file_content = "loco file upload";
let file_part = Part::bytes(file_content.as_bytes()).file_name("loco.txt");

let multipart_form = MultipartForm::new().add_part("file", file_part);

let response = request.post("/upload/file").multipart(multipart_form).await;

response.assert_status_ok();

let res: views::upload::Response = serde_json::from_str(&response.text()).unwrap();

let stored_file: String = ctx.storage.unwrap().download(&res.path).await.unwrap();

assert_eq!(stored_file, file_content);
})
.await;
}
```

193 changes: 193 additions & 0 deletions docs-site/content/docs/the-app/storage.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,193 @@
+++
title = "Storage"
description = ""
date = 2024-02-07T08:00:00+00:00
updated = 2024-02-07T08:00:00+00:00
draft = false
weight = 21
sort_by = "weight"
template = "docs/page.html"

[extra]
lead = ""
toc = true
top = false
+++

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`

## Setup

Add the `storage` function as a Hook in the `app.rs` file and import the `storage` module from `loco_rs`.

```rust
use loco_rs::storage;

impl Hooks for App {
async fn storage(_config: &Config, environment: &Environment) -> Result<Option<storage::Storage>> {
return Ok(None);
}
}
```

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
| | |
| - | - |
| `driver` | Trait implementation for diverse storage operations. |
| `Storage`| Abstraction implementation for managing one or more storage operations. |
| `Strategy`| Trait implementing various strategies for operations, 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.

```rust
use loco_rs::storage;
async fn storage(
_config: &Config,
environment: &Environment,
) -> Result<Option<storage::Storage>> {
let storage = Storage::single(storage::drivers::mem::new());
return Ok(Some(storage));
}
```

### 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:

```rust
use crate::storage::{drivers, Storage};

let aws_1 = drivers::aws::new("users");
let azure = drivers::azure::new("users");
let aws_2 = drivers::aws::new("users-mirror");
```

#### 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.

#### Behiver

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:
```rust

// Define the mirror strategy by setting the primary store and secondary stores by names.
let strategy = Box::new(MirrorStrategy::new(
"store_1",
Some(vec!["store_2".to_string(), "store_3".to_string()]),
FailureMode::MirrorAll,
)) as Box<dyn StorageStrategyTrait>;

// Create the storage with the store mapping and the strategy.
let storage = Storage::new(
BTreeMap::from([
("store_1".to_string(), aws_1.clone()),
("store_2".to_string(), azure.clone()),
("store_3".to_string(), aws_2.clone()),
]),
strategy.into(),
);
```

### 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:
```rust

// Define the backup strategy by setting the primary store and secondary stores by names.
let strategy: Box<dyn StorageStrategyTrait> = Box::new(BackupStrategy::new(
"store_1",
Some(vec!["store_2".to_string(), "store_3".to_string()]),
FailureMode::AllowBackupFailure,
)) as Box<dyn StorageStrategyTrait>;

let storage = Storage::new(
BTreeMap::from([
("store_1".to_string(), store_1.clone()),
("store_2".to_string(), store_2.clone()),
("store_3".to_string(), store_3.clone()),
]),
strategy.into(),
);
```

## Create Your Own Strategy

In case you have a specific strategy, you can easily create it by implementing the StorageStrategyTrait and implementing all store functionality.

## Usage In Controller

Follow this example, make sure you enable `multipart` feature in axum crate.

```rust
async fn upload_file(
State(ctx): State<AppContext>,
mut multipart: Multipart,
) -> Result<Json<views::upload::Response>> {
let mut file = None;
while let Some(field) = multipart.next_field().await.map_err(|err| {
tracing::error!(error = ?err,"could not readd multipart");
Error::BadRequest("could not readd multipart".into())
})? {
let file_name = match field.file_name() {
Some(file_name) => file_name.to_string(),
_ => return Err(Error::BadRequest("file name not found".into())),
};

let content = field.bytes().await.map_err(|err| {
tracing::error!(error = ?err,"could not readd bytes");
Error::BadRequest("could not readd bytes".into())
})?;

let path = PathBuf::from("folder").join(file_name);
ctx.storage
.as_ref()
.unwrap()
.upload(path.as_path(), &content)
.await?;

file = Some(path);
}

file.map_or_else(not_found, |path| {
format::json(views::upload::Response::new(path.as_path()))
})
}
```
## Testing

Test storage in controller implementation, refer to the [documentation here](@/docs/testing/storage.md)
Loading

0 comments on commit a15d354

Please sign in to comment.