chore: migrate to object_store
Move to the more modern and maintained object_store library to access generic buckets. We should be able also to do streaming of files now, and proceed by implementing paginated results for listings if we want. Fixes #1.
This commit is contained in:
parent
996be0f6df
commit
32e2a5ea4a
11 changed files with 765 additions and 558 deletions
932
Cargo.lock
generated
932
Cargo.lock
generated
File diff suppressed because it is too large
Load diff
13
Cargo.toml
13
Cargo.toml
|
@ -3,7 +3,7 @@
|
||||||
|
|
||||||
[package]
|
[package]
|
||||||
name = "serves3"
|
name = "serves3"
|
||||||
version = "1.1.2"
|
version = "1.2.0"
|
||||||
|
|
||||||
authors = ["Matteo Settenvini <matteo.settenvini@montecristosoftware.eu>"]
|
authors = ["Matteo Settenvini <matteo.settenvini@montecristosoftware.eu>"]
|
||||||
description = "A very simple proxy to browse files from private S3 buckets"
|
description = "A very simple proxy to browse files from private S3 buckets"
|
||||||
|
@ -21,14 +21,14 @@ edition = "2024"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
anyhow = "1.0"
|
anyhow = "1.0"
|
||||||
|
bytes = "1.10"
|
||||||
|
futures = "0.3"
|
||||||
human-size = "0.4"
|
human-size = "0.4"
|
||||||
lazy_static = "1.4"
|
lazy_static = "1.4"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
rocket = "0.5"
|
rocket = "0.5"
|
||||||
rocket_dyn_templates = { version = "0.2.0", features = ["tera"] }
|
rocket_dyn_templates = { version = "0.2.0", features = ["tera"] }
|
||||||
rust-s3 = { version = "0.35", default-features = false, features = [
|
object_store = { version = "0.12", features = ["aws"] }
|
||||||
"tokio-rustls-tls",
|
|
||||||
] }
|
|
||||||
serde = "1.0"
|
serde = "1.0"
|
||||||
tempfile = "3.20"
|
tempfile = "3.20"
|
||||||
|
|
||||||
|
@ -36,11 +36,12 @@ tempfile = "3.20"
|
||||||
delegate = "0.13"
|
delegate = "0.13"
|
||||||
futures = "0.3"
|
futures = "0.3"
|
||||||
libc = "0.2"
|
libc = "0.2"
|
||||||
|
minio = "0.3"
|
||||||
regex = "1.11"
|
regex = "1.11"
|
||||||
reqwest = "0.12"
|
reqwest = "0.12"
|
||||||
rstest = "0.25"
|
rstest = "0.26"
|
||||||
scraper = "0.23"
|
scraper = "0.23"
|
||||||
test-log = "0.2"
|
test-log = "0.2"
|
||||||
testcontainers = "0.24"
|
testcontainers = "0.24"
|
||||||
testcontainers-modules = { version = "0.12", features = ["minio"] }
|
testcontainers-modules = { version = "0.12", features = ["minio"] }
|
||||||
tokio = { version = "1", features = ["process"] }
|
tokio = { version = "1", features = ["process", "rt-multi-thread"] }
|
||||||
|
|
21
Containerfile
Normal file
21
Containerfile
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
# SPDX-FileCopyrightText: © Matteo Settenvini <matteo.settenvini@montecristosoftware.eu>
|
||||||
|
# SPDX-License-Identifier: EUPL-1.2
|
||||||
|
|
||||||
|
FROM docker.io/rust:1.88-alpine3.20
|
||||||
|
|
||||||
|
RUN apk --no-cache add musl-dev && \
|
||||||
|
cargo install --path .
|
||||||
|
|
||||||
|
FROM docker.io/alpine:3.20
|
||||||
|
|
||||||
|
COPY --from=rust /usr/local/cargo/bin/serves3 /usr/bin/serves3
|
||||||
|
|
||||||
|
EXPOSE 8000
|
||||||
|
|
||||||
|
ENV ROCKET_PROFILE=production
|
||||||
|
|
||||||
|
# This env variable should be in the format:
|
||||||
|
# {endpoint=<endpoint>,region=<region>,access_key_id=<access_key_id>,secret_access_key=<secret_access_key>}
|
||||||
|
ENV ROCKET_S3_BUCKET={endpoint=,region=,access_key_id=,secret_access_key=}
|
||||||
|
|
||||||
|
CMD ["serves3"]
|
|
@ -109,6 +109,12 @@ serves3
|
||||||
|
|
||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## 1.2.0
|
||||||
|
|
||||||
|
* Migrate to [object_store](https://crates.io/crates/object_store). This should
|
||||||
|
allow streaming of bigger files.
|
||||||
|
* Bump dependencies
|
||||||
|
|
||||||
## 1.1.2
|
## 1.1.2
|
||||||
|
|
||||||
* Bump dependencies, adopt Rust 2024 edition
|
* Bump dependencies, adopt Rust 2024 edition
|
||||||
|
|
|
@ -77,6 +77,7 @@ ignore = [
|
||||||
#{ id = "RUSTSEC-0000-0000", reason = "you can specify a reason the advisory is ignored" },
|
#{ id = "RUSTSEC-0000-0000", reason = "you can specify a reason the advisory is ignored" },
|
||||||
#"a-crate-that-is-yanked@0.1.1", # you can also ignore yanked crate versions if you wish
|
#"a-crate-that-is-yanked@0.1.1", # you can also ignore yanked crate versions if you wish
|
||||||
#{ crate = "a-crate-that-is-yanked@0.1.1", reason = "you can specify why you are ignoring the yanked crate" },
|
#{ crate = "a-crate-that-is-yanked@0.1.1", reason = "you can specify why you are ignoring the yanked crate" },
|
||||||
|
{ id = "RUSTSEC-2024-0388", reason = "derivative is used indirectly through minio, which is used only for testing" },
|
||||||
]
|
]
|
||||||
# If this is true, then cargo deny will use the git executable to fetch advisory database.
|
# If this is true, then cargo deny will use the git executable to fetch advisory database.
|
||||||
# If this is false, then it uses a built-in git library.
|
# If this is false, then it uses a built-in git library.
|
||||||
|
|
187
src/main.rs
187
src/main.rs
|
@ -6,16 +6,19 @@ mod sizes;
|
||||||
|
|
||||||
use {
|
use {
|
||||||
anyhow::Result,
|
anyhow::Result,
|
||||||
|
bytes::Bytes,
|
||||||
|
futures::{StreamExt, stream::BoxStream},
|
||||||
lazy_static::lazy_static,
|
lazy_static::lazy_static,
|
||||||
|
object_store::path::Path as ObjectStorePath,
|
||||||
rocket::{
|
rocket::{
|
||||||
State,
|
Request, State,
|
||||||
fairing::AdHoc,
|
fairing::AdHoc,
|
||||||
figment::{
|
figment::{
|
||||||
Profile,
|
Profile,
|
||||||
providers::{Env, Format as _, Toml},
|
providers::{Env, Format as _, Toml},
|
||||||
},
|
},
|
||||||
http::uri::Origin,
|
http::ContentType,
|
||||||
response::{Redirect, Responder},
|
response::{self, Responder, stream::ByteStream},
|
||||||
serde::Serialize,
|
serde::Serialize,
|
||||||
},
|
},
|
||||||
rocket_dyn_templates::{Template, context},
|
rocket_dyn_templates::{Template, context},
|
||||||
|
@ -23,15 +26,21 @@ use {
|
||||||
std::path::PathBuf,
|
std::path::PathBuf,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Responder)]
|
|
||||||
enum FileView {
|
enum FileView {
|
||||||
#[response(content_type = "text/html")]
|
|
||||||
Folder(Template),
|
Folder(Template),
|
||||||
|
File(ByteStream<BoxStream<'static, Bytes>>),
|
||||||
|
}
|
||||||
|
|
||||||
#[response(content_type = "application/octet-stream")]
|
impl<'r> Responder<'r, 'r> for FileView {
|
||||||
File(Vec<u8>),
|
fn respond_to(self, req: &'r Request<'_>) -> response::Result<'r> {
|
||||||
|
match self {
|
||||||
Redirect(Redirect),
|
Self::Folder(template) => template.respond_to(req).map(|mut r| {
|
||||||
|
r.set_header(ContentType::HTML);
|
||||||
|
r
|
||||||
|
}),
|
||||||
|
Self::File(stream) => stream.respond_to(req),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Serialize)]
|
#[derive(Serialize)]
|
||||||
|
@ -47,20 +56,33 @@ enum Error {
|
||||||
#[response(status = 404)]
|
#[response(status = 404)]
|
||||||
NotFound(String),
|
NotFound(String),
|
||||||
|
|
||||||
|
#[response(status = 400)]
|
||||||
|
InvalidRequest(String),
|
||||||
|
|
||||||
#[response(status = 500)]
|
#[response(status = 500)]
|
||||||
UnknownError(String),
|
UnknownError(String),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[rocket::get("/")]
|
||||||
|
async fn index_root(state: &State<Settings>) -> Result<FileView, Error> {
|
||||||
|
index(None, state).await
|
||||||
|
}
|
||||||
|
|
||||||
#[rocket::get("/<path..>")]
|
#[rocket::get("/<path..>")]
|
||||||
async fn index(
|
async fn index(path: Option<PathBuf>, state: &State<Settings>) -> Result<FileView, Error> {
|
||||||
path: PathBuf,
|
let object_path = if let Some(url_path) = path.as_ref() {
|
||||||
uri: &Origin<'_>,
|
let s = url_path.to_str().ok_or(Error::InvalidRequest(
|
||||||
state: &State<Settings>,
|
"Path cannot be converted to UTF-8".into(),
|
||||||
) -> Result<FileView, Error> {
|
))?;
|
||||||
|
|
||||||
|
Some(ObjectStorePath::from(s))
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
The way things work in S3, the following holds for us:
|
The way things work in S3, the following holds for us:
|
||||||
- we need to use a slash as separator
|
- we need to use a slash as separator
|
||||||
- folders need to be queried ending with a slash
|
|
||||||
- getting the bucket address (empty prefix) will
|
- getting the bucket address (empty prefix) will
|
||||||
return an XML file with all properties; we don't
|
return an XML file with all properties; we don't
|
||||||
want that.
|
want that.
|
||||||
|
@ -68,113 +90,108 @@ async fn index(
|
||||||
We try first to retrieve list an object as a file. If we fail,
|
We try first to retrieve list an object as a file. If we fail,
|
||||||
we fallback to retrieving the equivalent folder.
|
we fallback to retrieving the equivalent folder.
|
||||||
*/
|
*/
|
||||||
if let Ok(result) = s3_serve_file(&path, &state).await {
|
|
||||||
Ok(result)
|
|
||||||
} else {
|
|
||||||
// We need to redirect to a path ending with a slash as
|
|
||||||
// per comment above if we know this is not a file.
|
|
||||||
let mut uri = uri.to_string();
|
|
||||||
if !uri.ends_with('/') {
|
|
||||||
uri.push('/');
|
|
||||||
return Ok(FileView::Redirect(Redirect::permanent(uri)));
|
|
||||||
}
|
|
||||||
|
|
||||||
let objects = s3_fileview(&path, &state).await?;
|
if let Some(path) = &object_path
|
||||||
|
&& object_exists(path, &state).await?
|
||||||
|
{
|
||||||
|
serve_object(&path, &state).await
|
||||||
|
} else {
|
||||||
|
let objects = file_view(object_path, &state).await?;
|
||||||
|
|
||||||
let rendered = Template::render(
|
let rendered = Template::render(
|
||||||
"index",
|
"index",
|
||||||
context! {
|
context! {
|
||||||
path: format!("{}/", path.display()),
|
path: format!("{}/", path.unwrap_or("".into()).display()),
|
||||||
objects
|
objects
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
Ok(FileView::Folder(rendered))
|
Ok(FileView::Folder(rendered))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn s3_serve_file(path: &PathBuf, settings: &Settings) -> Result<FileView, Error> {
|
async fn object_exists(s3_path: &ObjectStorePath, settings: &Settings) -> Result<bool, Error> {
|
||||||
let is_root_prefix = path.as_os_str().is_empty();
|
match settings.s3_bucket.head(s3_path).await {
|
||||||
if is_root_prefix {
|
Ok(_metadata) => Ok(true),
|
||||||
return Err(Error::NotFound("Root prefix is not a file".into()));
|
Err(object_store::Error::NotFound { path: _, source: _ }) => Ok(false),
|
||||||
}
|
Err(e) => Err(Error::UnknownError(e.to_string())),
|
||||||
|
|
||||||
// FIXME: this can be big, we should use streaming,
|
|
||||||
// not loading in memory!
|
|
||||||
let response = settings
|
|
||||||
.s3_bucket
|
|
||||||
.get_object(format!("{}", path.display()))
|
|
||||||
.await
|
|
||||||
.map_err(|_| Error::UnknownError("Unable to connect to S3 bucket".into()))?;
|
|
||||||
|
|
||||||
match response.status_code() {
|
|
||||||
200 | 204 => {
|
|
||||||
let bytes = response.bytes().to_vec();
|
|
||||||
Ok(FileView::File(bytes))
|
|
||||||
}
|
|
||||||
404 => Err(Error::NotFound("Object not found".into())),
|
|
||||||
_ => Err(Error::UnknownError("Unknown S3 error".into())),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn s3_fileview(path: &PathBuf, settings: &Settings) -> Result<Vec<FileViewItem>, Error> {
|
async fn serve_object(s3_path: &ObjectStorePath, settings: &Settings) -> Result<FileView, Error> {
|
||||||
|
let object_stream = settings
|
||||||
|
.s3_bucket
|
||||||
|
.get(&s3_path)
|
||||||
|
.await
|
||||||
|
.map_err(|e| match e {
|
||||||
|
object_store::Error::NotFound { path: _, source: _ } => Error::NotFound(e.to_string()),
|
||||||
|
_ => Error::UnknownError(e.to_string()),
|
||||||
|
})?
|
||||||
|
.into_stream();
|
||||||
|
|
||||||
|
let s3_path = s3_path.clone();
|
||||||
|
let stream = object_stream
|
||||||
|
.map(move |chunk| match chunk {
|
||||||
|
Ok(bytes) => bytes,
|
||||||
|
Err(err) => {
|
||||||
|
log::error!("connection error while reading {}: {}", s3_path, err);
|
||||||
|
Bytes::new() // Forces end of stream
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.boxed();
|
||||||
|
|
||||||
|
// TODO: unfortunately Rocket does not have a ByteStream with a Result per chunk,
|
||||||
|
// meaning that if there is a failure in the middle of the stream, the best we can do is...
|
||||||
|
// nothing? Panic? All options are bad.
|
||||||
|
|
||||||
|
Ok(FileView::File(ByteStream::from(stream)))
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn file_view(
|
||||||
|
s3_folder_path: Option<ObjectStorePath>,
|
||||||
|
settings: &Settings,
|
||||||
|
) -> Result<Vec<FileViewItem>, Error> {
|
||||||
/*
|
/*
|
||||||
if listing a folder:
|
if listing a folder:
|
||||||
- folders will be under 'common_prefixes'
|
- folders will be under 'common_prefixes'
|
||||||
- files will be under the 'contents' property
|
- files will be under the 'contents' property
|
||||||
*/
|
*/
|
||||||
|
|
||||||
let parent = path.parent();
|
|
||||||
let s3_folder_path = match parent {
|
|
||||||
Some(_) => format!("{}/", path.display()),
|
|
||||||
None => "".into(),
|
|
||||||
};
|
|
||||||
|
|
||||||
let s3_objects = settings
|
let s3_objects = settings
|
||||||
.s3_bucket
|
.s3_bucket
|
||||||
.list(s3_folder_path, Some("/".into()))
|
.list_with_delimiter(s3_folder_path.as_ref())
|
||||||
.await
|
.await
|
||||||
.map_err(|_| Error::NotFound("Object not found".into()))?;
|
.map_err(|err| match err {
|
||||||
|
object_store::Error::NotFound { path: _, source: _ } => {
|
||||||
|
Error::NotFound("object not found".into())
|
||||||
|
}
|
||||||
|
err => Error::UnknownError(err.to_string()),
|
||||||
|
})?;
|
||||||
|
|
||||||
let objects = s3_objects
|
let folders = s3_objects.common_prefixes.into_iter().map(|dir| {
|
||||||
.iter()
|
let dirname = dir.parts().last().unwrap();
|
||||||
.flat_map(|list| -> Vec<Option<FileViewItem>> {
|
FileViewItem {
|
||||||
let prefix = if let Some(p) = &list.prefix {
|
path: dirname.as_ref().into(),
|
||||||
p.as_str()
|
|
||||||
} else {
|
|
||||||
""
|
|
||||||
};
|
|
||||||
|
|
||||||
let folders = list.common_prefixes.iter().flatten().map(|dir| {
|
|
||||||
let path = dir.prefix.strip_prefix(&prefix);
|
|
||||||
path.map(|path| FileViewItem {
|
|
||||||
path: path.to_owned(),
|
|
||||||
size_bytes: 0,
|
size_bytes: 0,
|
||||||
size: "[DIR]".to_owned(),
|
size: "[DIR]".to_owned(),
|
||||||
last_modification: String::default(),
|
last_modification: String::default(),
|
||||||
})
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
let files = list.contents.iter().map(|obj| {
|
let files = s3_objects.objects.into_iter().map(|obj| FileViewItem {
|
||||||
let path = obj.key.strip_prefix(&prefix);
|
path: obj.location.filename().unwrap().into(),
|
||||||
path.map(|path| FileViewItem {
|
|
||||||
path: path.to_owned(),
|
|
||||||
size_bytes: obj.size,
|
size_bytes: obj.size,
|
||||||
size: sizes::bytes_to_human(obj.size),
|
size: sizes::bytes_to_human(obj.size),
|
||||||
last_modification: obj.last_modified.clone(),
|
last_modification: obj.last_modified.to_rfc3339(),
|
||||||
})
|
|
||||||
});
|
});
|
||||||
|
|
||||||
folders.chain(files).collect()
|
Ok(folders.chain(files).collect())
|
||||||
})
|
|
||||||
.flatten()
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
Ok(objects)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
lazy_static! {
|
lazy_static! {
|
||||||
// Workaround for https://github.com/SergioBenitez/Rocket/issues/1792
|
// Workaround for https://github.com/SergioBenitez/Rocket/issues/1792
|
||||||
static ref EMPTY_DIR: tempfile::TempDir = tempfile::tempdir()
|
static ref EMPTY_DIR: tempfile::TempDir = tempfile::tempdir()
|
||||||
.expect("Unable to create an empty temporary folder, is the whole FS read-only?");
|
.expect("unable to create an empty temporary folder, is the whole FS read-only?");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[rocket::launch]
|
#[rocket::launch]
|
||||||
|
@ -186,7 +203,7 @@ fn rocket() -> _ {
|
||||||
.select(Profile::from_env_or("SERVES3_PROFILE", "default"));
|
.select(Profile::from_env_or("SERVES3_PROFILE", "default"));
|
||||||
|
|
||||||
rocket::custom(config_figment)
|
rocket::custom(config_figment)
|
||||||
.mount("/", rocket::routes![index])
|
.mount("/", rocket::routes![index_root, index])
|
||||||
.attach(AdHoc::config::<Settings>())
|
.attach(AdHoc::config::<Settings>())
|
||||||
.attach(Template::custom(|engines| {
|
.attach(Template::custom(|engines| {
|
||||||
engines
|
engines
|
||||||
|
|
|
@ -1,16 +1,20 @@
|
||||||
// SPDX-FileCopyrightText: © Matteo Settenvini <matteo.settenvini@montecristosoftware.eu>
|
// SPDX-FileCopyrightText: © Matteo Settenvini <matteo.settenvini@montecristosoftware.eu>
|
||||||
// SPDX-License-Identifier: EUPL-1.2
|
// SPDX-License-Identifier: EUPL-1.2
|
||||||
|
|
||||||
use {anyhow::anyhow, rocket::serde::Deserialize, serde::de::Error};
|
use {
|
||||||
|
object_store::{ObjectStore, aws},
|
||||||
|
rocket::serde::Deserialize,
|
||||||
|
serde::de::Error,
|
||||||
|
};
|
||||||
|
|
||||||
#[derive(Deserialize)]
|
#[derive(Deserialize)]
|
||||||
#[serde(crate = "rocket::serde")]
|
#[serde(crate = "rocket::serde")]
|
||||||
pub struct Settings {
|
pub struct Settings {
|
||||||
#[serde(deserialize_with = "deserialize_s3_bucket")]
|
#[serde(deserialize_with = "deserialize_s3_bucket")]
|
||||||
pub s3_bucket: Box<s3::Bucket>,
|
pub s3_bucket: Box<dyn ObjectStore>,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn deserialize_s3_bucket<'de, D>(deserializer: D) -> Result<Box<s3::Bucket>, D::Error>
|
fn deserialize_s3_bucket<'de, D>(deserializer: D) -> Result<Box<dyn ObjectStore>, D::Error>
|
||||||
where
|
where
|
||||||
D: serde::Deserializer<'de>,
|
D: serde::Deserializer<'de>,
|
||||||
{
|
{
|
||||||
|
@ -31,37 +35,27 @@ pub struct S3Config {
|
||||||
pub secret_access_key: String,
|
pub secret_access_key: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TryInto<Box<s3::Bucket>> for S3Config {
|
impl TryInto<Box<dyn ObjectStore>> for S3Config {
|
||||||
type Error = anyhow::Error;
|
type Error = anyhow::Error;
|
||||||
|
|
||||||
fn try_into(self) -> Result<Box<s3::Bucket>, Self::Error> {
|
fn try_into(self) -> Result<Box<dyn ObjectStore>, Self::Error> {
|
||||||
let region = s3::Region::Custom {
|
// TODO: support object stores other than than AWS
|
||||||
region: self.region,
|
let object_store = aws::AmazonS3Builder::new()
|
||||||
endpoint: self.endpoint,
|
.with_region(self.region)
|
||||||
};
|
.with_endpoint(&self.endpoint)
|
||||||
|
.with_bucket_name(&self.name)
|
||||||
let credentials = s3::creds::Credentials::new(
|
.with_access_key_id(self.access_key_id)
|
||||||
Some(&self.access_key_id),
|
.with_secret_access_key(self.secret_access_key)
|
||||||
Some(&self.secret_access_key),
|
.with_virtual_hosted_style_request(!self.path_style)
|
||||||
None,
|
.with_allow_http(true)
|
||||||
None,
|
.build()?;
|
||||||
None,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
log::info!(
|
log::info!(
|
||||||
"Serving contents from bucket {} at {}",
|
"Serving contents from bucket {} at {}",
|
||||||
&self.name,
|
self.endpoint,
|
||||||
region.endpoint()
|
self.name,
|
||||||
);
|
);
|
||||||
|
|
||||||
let bucket = s3::Bucket::new(&self.name, region, credentials).map_err(|e| anyhow!(e));
|
Ok(Box::new(object_store))
|
||||||
if self.path_style {
|
|
||||||
bucket.map(|mut b| {
|
|
||||||
b.set_path_style();
|
|
||||||
b
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
bucket
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
-->
|
-->
|
||||||
<html>
|
<html>
|
||||||
<head>
|
<head>
|
||||||
|
<link rel="icon" href="">
|
||||||
<style>
|
<style>
|
||||||
body {
|
body {
|
||||||
font-family: monospace;
|
font-family: monospace;
|
||||||
|
|
|
@ -11,7 +11,7 @@ use {
|
||||||
testcontainers_modules::minio,
|
testcontainers_modules::minio,
|
||||||
};
|
};
|
||||||
|
|
||||||
const MINIO_IMAGE_TAG: &'static str = "RELEASE.2025-06-13T11-33-47Z";
|
const MINIO_IMAGE_TAG: &'static str = "RELEASE.2025-07-23T15-54-02Z";
|
||||||
|
|
||||||
pub struct MinIO {
|
pub struct MinIO {
|
||||||
inner: minio::MinIO,
|
inner: minio::MinIO,
|
||||||
|
|
|
@ -4,7 +4,11 @@
|
||||||
mod minio;
|
mod minio;
|
||||||
|
|
||||||
use {
|
use {
|
||||||
|
::minio::s3::{
|
||||||
|
client::Client as MinIOClient, creds::StaticProvider, http::BaseUrl, types::S3Api,
|
||||||
|
},
|
||||||
anyhow::{Result, anyhow},
|
anyhow::{Result, anyhow},
|
||||||
|
object_store::{ObjectStore, aws::AmazonS3Builder},
|
||||||
reqwest::Url,
|
reqwest::Url,
|
||||||
std::{ptr::null_mut, str::FromStr},
|
std::{ptr::null_mut, str::FromStr},
|
||||||
testcontainers::{ContainerAsync, runners::AsyncRunner},
|
testcontainers::{ContainerAsync, runners::AsyncRunner},
|
||||||
|
@ -13,7 +17,7 @@ use {
|
||||||
|
|
||||||
pub struct Test {
|
pub struct Test {
|
||||||
pub base_url: Url,
|
pub base_url: Url,
|
||||||
pub bucket: Box<s3::Bucket>,
|
pub bucket: Box<dyn ObjectStore>,
|
||||||
pub serves3: tokio::process::Child,
|
pub serves3: tokio::process::Child,
|
||||||
pub _minio: ContainerAsync<minio::MinIO>,
|
pub _minio: ContainerAsync<minio::MinIO>,
|
||||||
}
|
}
|
||||||
|
@ -41,24 +45,22 @@ impl Test {
|
||||||
port = container.get_host_port_ipv4(9000).await?
|
port = container.get_host_port_ipv4(9000).await?
|
||||||
);
|
);
|
||||||
|
|
||||||
let credentials = s3::creds::Credentials::new(
|
// We need to create the bucket
|
||||||
Some(&ACCESS_KEY),
|
let minio_client = MinIOClient::new(
|
||||||
Some(&SECRET_KEY),
|
BaseUrl::from_str(&endpoint).unwrap(),
|
||||||
|
Some(Box::new(StaticProvider::new(ACCESS_KEY, SECRET_KEY, None))),
|
||||||
None,
|
None,
|
||||||
None,
|
None,
|
||||||
Some("test"),
|
|
||||||
)?;
|
)?;
|
||||||
let bucket = s3::Bucket::create_with_path_style(
|
minio_client.create_bucket(BUCKET_NAME).send().await?;
|
||||||
&BUCKET_NAME,
|
|
||||||
s3::Region::Custom {
|
let bucket = AmazonS3Builder::new()
|
||||||
region: REGION.into(),
|
.with_endpoint(&endpoint)
|
||||||
endpoint: endpoint.clone(),
|
.with_access_key_id(ACCESS_KEY)
|
||||||
},
|
.with_secret_access_key(SECRET_KEY)
|
||||||
credentials,
|
.with_bucket_name(BUCKET_NAME)
|
||||||
s3::BucketConfiguration::private(),
|
.with_allow_http(true)
|
||||||
)
|
.build()?;
|
||||||
.await?
|
|
||||||
.bucket;
|
|
||||||
|
|
||||||
let bin = std::env!("CARGO_BIN_EXE_serves3");
|
let bin = std::env!("CARGO_BIN_EXE_serves3");
|
||||||
let mut child = tokio::process::Command::new(bin)
|
let mut child = tokio::process::Command::new(bin)
|
||||||
|
@ -104,7 +106,7 @@ impl Test {
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
base_url,
|
base_url,
|
||||||
bucket,
|
bucket: Box::new(bucket),
|
||||||
serves3: child,
|
serves3: child,
|
||||||
_minio: container,
|
_minio: container,
|
||||||
})
|
})
|
||||||
|
|
|
@ -3,17 +3,27 @@
|
||||||
|
|
||||||
mod common;
|
mod common;
|
||||||
|
|
||||||
use scraper::{Html, Selector};
|
use {
|
||||||
|
object_store::{PutPayload, path::Path as ObjectStorePath},
|
||||||
|
scraper::{Html, Selector},
|
||||||
|
};
|
||||||
|
|
||||||
#[test_log::test(tokio::test)]
|
#[test_log::test(tokio::test(flavor = "multi_thread"))]
|
||||||
async fn serves_files() -> anyhow::Result<()> {
|
async fn serves_files() -> anyhow::Result<()> {
|
||||||
let test = common::Test::new().await?;
|
let test = common::Test::new().await?;
|
||||||
|
|
||||||
test.bucket
|
test.bucket
|
||||||
.put_object("file.txt", "I am a file".as_bytes())
|
.put(
|
||||||
|
&ObjectStorePath::from("file.txt"),
|
||||||
|
PutPayload::from_static("I am a file".as_bytes()),
|
||||||
|
)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
test.bucket
|
test.bucket
|
||||||
.put_object("folder/file.txt", "I am a file in a folder".as_bytes())
|
.put(
|
||||||
|
&ObjectStorePath::from("folder/file.txt"),
|
||||||
|
PutPayload::from_static("I am a file in a folder".as_bytes()),
|
||||||
|
)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
let resp = reqwest::get(test.base_url.join("file.txt")?).await?;
|
let resp = reqwest::get(test.base_url.join("file.txt")?).await?;
|
||||||
|
@ -25,15 +35,21 @@ async fn serves_files() -> anyhow::Result<()> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test_log::test(tokio::test)]
|
#[test_log::test(tokio::test(flavor = "multi_thread"))]
|
||||||
async fn serves_top_level_folder() -> anyhow::Result<()> {
|
async fn serves_top_level_folder() -> anyhow::Result<()> {
|
||||||
let test = common::Test::new().await?;
|
let test = common::Test::new().await?;
|
||||||
|
|
||||||
test.bucket
|
test.bucket
|
||||||
.put_object("file.txt", "I am a file".as_bytes())
|
.put(
|
||||||
|
&ObjectStorePath::from("file.txt"),
|
||||||
|
PutPayload::from_static("I am a file".as_bytes()),
|
||||||
|
)
|
||||||
.await?;
|
.await?;
|
||||||
test.bucket
|
test.bucket
|
||||||
.put_object("folder/file.txt", "I am a file in a folder".as_bytes())
|
.put(
|
||||||
|
&ObjectStorePath::from("folder/file.txt"),
|
||||||
|
PutPayload::from_static("I am a file in a folder".as_bytes()),
|
||||||
|
)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
// Check that a file in the toplevel is listed:
|
// Check that a file in the toplevel is listed:
|
||||||
|
@ -43,6 +59,7 @@ async fn serves_top_level_folder() -> anyhow::Result<()> {
|
||||||
"Request failed with {}",
|
"Request failed with {}",
|
||||||
resp.status()
|
resp.status()
|
||||||
);
|
);
|
||||||
|
|
||||||
let text = resp.text().await?;
|
let text = resp.text().await?;
|
||||||
println!("{}", &text);
|
println!("{}", &text);
|
||||||
let document = Html::parse_document(&text);
|
let document = Html::parse_document(&text);
|
||||||
|
@ -55,8 +72,8 @@ async fn serves_top_level_folder() -> anyhow::Result<()> {
|
||||||
let selector =
|
let selector =
|
||||||
Selector::parse(r#"table > tbody > tr:nth-child(1) > td:first-child > a"#).unwrap();
|
Selector::parse(r#"table > tbody > tr:nth-child(1) > td:first-child > a"#).unwrap();
|
||||||
for item in document.select(&selector) {
|
for item in document.select(&selector) {
|
||||||
assert_eq!(item.attr("href"), Some("folder/"));
|
assert_eq!(item.attr("href"), Some("folder"));
|
||||||
assert_eq!(item.text().next(), Some("folder/"));
|
assert_eq!(item.text().next(), Some("folder"));
|
||||||
}
|
}
|
||||||
|
|
||||||
let selector =
|
let selector =
|
||||||
|
@ -69,15 +86,22 @@ async fn serves_top_level_folder() -> anyhow::Result<()> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test_log::test(tokio::test)]
|
#[test_log::test(tokio::test(flavor = "multi_thread"))]
|
||||||
async fn serves_second_level_folder() -> anyhow::Result<()> {
|
async fn serves_second_level_folder() -> anyhow::Result<()> {
|
||||||
let test = common::Test::new().await?;
|
let test = common::Test::new().await?;
|
||||||
|
|
||||||
test.bucket
|
test.bucket
|
||||||
.put_object("file.txt", "I am a file".as_bytes())
|
.put(
|
||||||
|
&ObjectStorePath::from("file.txt"),
|
||||||
|
PutPayload::from_static("I am a file".as_bytes()),
|
||||||
|
)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
test.bucket
|
test.bucket
|
||||||
.put_object("folder/file.txt", "I am a file in a folder".as_bytes())
|
.put(
|
||||||
|
&ObjectStorePath::from("folder/file.txt"),
|
||||||
|
PutPayload::from_static("I am a file in a folder".as_bytes()),
|
||||||
|
)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
// Check that a file in the second level is listed:
|
// Check that a file in the second level is listed:
|
||||||
|
@ -113,15 +137,21 @@ async fn serves_second_level_folder() -> anyhow::Result<()> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test_log::test(tokio::test)]
|
#[test_log::test(tokio::test(flavor = "multi_thread"))]
|
||||||
async fn serves_second_level_folder_without_ending_slash() -> anyhow::Result<()> {
|
async fn serves_second_level_folder_without_ending_slash() -> anyhow::Result<()> {
|
||||||
let test = common::Test::new().await?;
|
let test = common::Test::new().await?;
|
||||||
|
|
||||||
test.bucket
|
test.bucket
|
||||||
.put_object("file.txt", "I am a file".as_bytes())
|
.put(
|
||||||
|
&ObjectStorePath::from("file.txt"),
|
||||||
|
PutPayload::from_static("I am a file".as_bytes()),
|
||||||
|
)
|
||||||
.await?;
|
.await?;
|
||||||
test.bucket
|
test.bucket
|
||||||
.put_object("folder/file.txt", "I am a file in a folder".as_bytes())
|
.put(
|
||||||
|
&ObjectStorePath::from("folder/file.txt"),
|
||||||
|
PutPayload::from_static("I am a file in a folder".as_bytes()),
|
||||||
|
)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
// Check that a file in the second level is listed even without an ending slash:
|
// Check that a file in the second level is listed even without an ending slash:
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue