[pbs-devel] [PATCH proxmox v10 3/3] s3 client: Add missing S3 object key max length check
Christian Ebner
c.ebner at proxmox.com
Mon Jul 21 18:44:21 CEST 2025
S3 object keys are limited to 1024 bytes, including the path
components and separating slashes in addition to the filename.
Check the length when creating the key from a string.
Signed-off-by: Christian Ebner <c.ebner at proxmox.com>
---
changes since version 9:
- not present in previous version
proxmox-s3-client/examples/s3_client.rs | 4 ++--
proxmox-s3-client/src/object_key.rs | 26 ++++++++++++++++++-------
2 files changed, 21 insertions(+), 9 deletions(-)
diff --git a/proxmox-s3-client/examples/s3_client.rs b/proxmox-s3-client/examples/s3_client.rs
index c65ceb83..1cbb3939 100644
--- a/proxmox-s3-client/examples/s3_client.rs
+++ b/proxmox-s3-client/examples/s3_client.rs
@@ -46,7 +46,7 @@ async fn run() -> Result<(), anyhow::Error> {
// Check if the bucket can be accessed
s3_client.head_bucket().await?;
- let rel_object_key = S3ObjectKey::from("object.txt");
+ let rel_object_key = S3ObjectKey::try_from("object.txt")?;
let body = proxmox_http::Body::empty();
let replace_existing_key = true;
let _response = s3_client
@@ -63,7 +63,7 @@ async fn run() -> Result<(), anyhow::Error> {
.await?;
// Delete a single object
- let rel_object_key = S3ObjectKey::from("object.txt");
+ let rel_object_key = S3ObjectKey::try_from("object.txt")?;
let _response = s3_client.delete_object(rel_object_key).await?;
Ok(())
}
diff --git a/proxmox-s3-client/src/object_key.rs b/proxmox-s3-client/src/object_key.rs
index 49959b6e..327e8ac7 100644
--- a/proxmox-s3-client/src/object_key.rs
+++ b/proxmox-s3-client/src/object_key.rs
@@ -1,4 +1,8 @@
-use anyhow::Error;
+use anyhow::{bail, Error};
+
+/// Byte limit for s3 object keys.
+/// See https://docs.aws.amazon.com/AmazonS3/latest/userguide/object-keys.html
+const S3_OBJECT_KEY_MAX_LENGTH: usize = 1024;
#[derive(Clone, Debug)]
/// S3 Object Key
@@ -9,13 +13,21 @@ pub enum S3ObjectKey {
Relative(String),
}
-impl core::convert::From<&str> for S3ObjectKey {
- fn from(s: &str) -> Self {
- if let Some(s) = s.strip_prefix("/") {
- Self::Full(s.to_string())
+impl core::convert::TryFrom<&str> for S3ObjectKey {
+ type Error = Error;
+
+ fn try_from(s: &str) -> Result<Self, Error> {
+ let (key, key_byte_length) = if let Some(s) = s.strip_prefix("/") {
+ (Self::Full(s.to_string()), s.as_bytes().len())
} else {
- Self::Relative(s.to_string())
+ (Self::Relative(s.to_string()), s.as_bytes().len())
+ };
+ if key_byte_length > S3_OBJECT_KEY_MAX_LENGTH {
+ bail!(
+ "Object key length of {key_byte_length} exceeds limit of {S3_OBJECT_KEY_MAX_LENGTH}",
+ );
}
+ Ok(key)
}
}
impl S3ObjectKey {
@@ -56,7 +68,7 @@ impl std::str::FromStr for S3ObjectKey {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
- Ok(Self::from(s))
+ Self::try_from(s)
}
}
--
2.47.2
More information about the pbs-devel
mailing list