opendal/types/
capability.rs

1// Licensed to the Apache Software Foundation (ASF) under one
2// or more contributor license agreements.  See the NOTICE file
3// distributed with this work for additional information
4// regarding copyright ownership.  The ASF licenses this file
5// to you under the Apache License, Version 2.0 (the
6// "License"); you may not use this file except in compliance
7// with the License.  You may obtain a copy of the License at
8//
9//   http://www.apache.org/licenses/LICENSE-2.0
10//
11// Unless required by applicable law or agreed to in writing,
12// software distributed under the License is distributed on an
13// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
14// KIND, either express or implied.  See the License for the
15// specific language governing permissions and limitations
16// under the License.
17
18use std::fmt::Debug;
19
20/// Capability defines the supported operations and their constraints for a storage Operator.
21///
22/// # Overview
23///
24/// This structure provides a comprehensive description of an Operator's capabilities,
25/// including:
26///
27/// - Basic operations support (read, write, delete, etc.)
28/// - Advanced operation variants (conditional operations, metadata handling)
29/// - Operational constraints (size limits, batch limitations)
30///
31/// # Capability Types
32///
33/// Every operator maintains two capability sets:
34///
35/// 1. [`OperatorInfo::native_capability`][crate::OperatorInfo::native_capability]:
36///    Represents operations natively supported by the storage backend.
37///
38/// 2. [`OperatorInfo::full_capability`][crate::OperatorInfo::full_capability]:
39///    Represents all available operations, including those implemented through
40///    alternative mechanisms.
41///
42/// # Implementation Details
43///
44/// Some operations might be available even when not natively supported by the
45/// backend. For example:
46///
47/// - Blocking operations are provided through the BlockingLayer
48///
49/// Developers should:
50/// - Use `full_capability` to determine available operations
51/// - Use `native_capability` to identify optimized operations
52///
53/// # Field Naming Conventions
54///
55/// Fields follow these naming patterns:
56///
57/// - Basic operations: Simple lowercase (e.g., `read`, `write`)
58/// - Compound operations: Underscore-separated (e.g., `presign_read`)
59/// - Variants: Capability description (e.g., `write_can_empty`)
60/// - Parameterized operations: With-style (e.g., `read_with_if_match`)
61/// - Limitations: Constraint description (e.g., `write_multi_max_size`)
62/// - Metadata Results: Returning metadata capabilities (e.g., `stat_has_content_length`)
63///
64/// All capability fields are public and can be accessed directly.
65#[derive(Copy, Clone, Default)]
66pub struct Capability {
67    /// Indicates if the operator supports metadata retrieval operations.
68    pub stat: bool,
69    /// Indicates if conditional stat operations using If-Match are supported.
70    pub stat_with_if_match: bool,
71    /// Indicates if conditional stat operations using If-None-Match are supported.
72    pub stat_with_if_none_match: bool,
73    /// Indicates if conditional stat operations using If-Modified-Since are supported.
74    pub stat_with_if_modified_since: bool,
75    /// Indicates if conditional stat operations using If-Unmodified-Since are supported.
76    pub stat_with_if_unmodified_since: bool,
77    /// Indicates if Cache-Control header override is supported during stat operations.
78    pub stat_with_override_cache_control: bool,
79    /// Indicates if Content-Disposition header override is supported during stat operations.
80    pub stat_with_override_content_disposition: bool,
81    /// Indicates if Content-Type header override is supported during stat operations.
82    pub stat_with_override_content_type: bool,
83    /// Indicates if versions stat operations are supported.
84    pub stat_with_version: bool,
85    /// Indicates whether cache control information is available in stat response
86    pub stat_has_cache_control: bool,
87    /// Indicates whether content disposition information is available in stat response
88    pub stat_has_content_disposition: bool,
89    /// Indicates whether content length information is available in stat response
90    pub stat_has_content_length: bool,
91    /// Indicates whether content MD5 checksum is available in stat response
92    pub stat_has_content_md5: bool,
93    /// Indicates whether content range information is available in stat response
94    pub stat_has_content_range: bool,
95    /// Indicates whether content type information is available in stat response
96    pub stat_has_content_type: bool,
97    /// Indicates whether content encoding information is available in stat response
98    pub stat_has_content_encoding: bool,
99    /// Indicates whether entity tag is available in stat response
100    pub stat_has_etag: bool,
101    /// Indicates whether last modified timestamp is available in stat response
102    pub stat_has_last_modified: bool,
103    /// Indicates whether version information is available in stat response
104    pub stat_has_version: bool,
105    /// Indicates whether user-defined metadata is available in stat response
106    pub stat_has_user_metadata: bool,
107
108    /// Indicates if the operator supports read operations.
109    pub read: bool,
110    /// Indicates if conditional read operations using If-Match are supported.
111    pub read_with_if_match: bool,
112    /// Indicates if conditional read operations using If-None-Match are supported.
113    pub read_with_if_none_match: bool,
114    /// Indicates if conditional read operations using If-Modified-Since are supported.
115    pub read_with_if_modified_since: bool,
116    /// Indicates if conditional read operations using If-Unmodified-Since are supported.
117    pub read_with_if_unmodified_since: bool,
118    /// Indicates if Cache-Control header override is supported during read operations.
119    pub read_with_override_cache_control: bool,
120    /// Indicates if Content-Disposition header override is supported during read operations.
121    pub read_with_override_content_disposition: bool,
122    /// Indicates if Content-Type header override is supported during read operations.
123    pub read_with_override_content_type: bool,
124    /// Indicates if versions read operations are supported.
125    pub read_with_version: bool,
126
127    /// Indicates if the operator supports write operations.
128    pub write: bool,
129    /// Indicates if multiple write operations can be performed on the same object.
130    pub write_can_multi: bool,
131    /// Indicates if writing empty content is supported.
132    pub write_can_empty: bool,
133    /// Indicates if append operations are supported.
134    pub write_can_append: bool,
135    /// Indicates if Content-Type can be specified during write operations.
136    pub write_with_content_type: bool,
137    /// Indicates if Content-Disposition can be specified during write operations.
138    pub write_with_content_disposition: bool,
139    /// Indicates if Content-Encoding can be specified during write operations.
140    pub write_with_content_encoding: bool,
141    /// Indicates if Cache-Control can be specified during write operations.
142    pub write_with_cache_control: bool,
143    /// Indicates if conditional write operations using If-Match are supported.
144    pub write_with_if_match: bool,
145    /// Indicates if conditional write operations using If-None-Match are supported.
146    pub write_with_if_none_match: bool,
147    /// Indicates if write operations can be conditional on object non-existence.
148    pub write_with_if_not_exists: bool,
149    /// Indicates if custom user metadata can be attached during write operations.
150    pub write_with_user_metadata: bool,
151    /// Maximum size supported for multipart uploads.
152    /// For example, AWS S3 supports up to 5GiB per part in multipart uploads.
153    pub write_multi_max_size: Option<usize>,
154    /// Minimum size required for multipart uploads (except for the last part).
155    /// For example, AWS S3 requires at least 5MiB per part.
156    pub write_multi_min_size: Option<usize>,
157    /// Maximum total size supported for write operations.
158    /// For example, Cloudflare D1 has a 1MB total size limit.
159    pub write_total_max_size: Option<usize>,
160
161    /// Indicates if directory creation is supported.
162    pub create_dir: bool,
163
164    /// Indicates if delete operations are supported.
165    pub delete: bool,
166    /// Indicates if versions delete operations are supported.
167    pub delete_with_version: bool,
168    /// Maximum size supported for single delete operations.
169    pub delete_max_size: Option<usize>,
170
171    /// Indicates if copy operations are supported.
172    pub copy: bool,
173
174    /// Indicates if rename operations are supported.
175    pub rename: bool,
176
177    /// Indicates if list operations are supported.
178    pub list: bool,
179    /// Indicates if list operations support result limiting.
180    pub list_with_limit: bool,
181    /// Indicates if list operations support continuation from a specific point.
182    pub list_with_start_after: bool,
183    /// Indicates if recursive listing is supported.
184    pub list_with_recursive: bool,
185    /// Indicates if versions listing is supported.
186    #[deprecated(since = "0.51.1", note = "use with_versions instead")]
187    pub list_with_version: bool,
188    /// Indicates if listing with versions included is supported.
189    pub list_with_versions: bool,
190    /// Indicates if listing with deleted files included is supported.
191    pub list_with_deleted: bool,
192    /// Indicates whether cache control information is available in list response
193    pub list_has_cache_control: bool,
194    /// Indicates whether content disposition information is available in list response
195    pub list_has_content_disposition: bool,
196    /// Indicates whether content length information is available in list response
197    pub list_has_content_length: bool,
198    /// Indicates whether content MD5 checksum is available in list response
199    pub list_has_content_md5: bool,
200    /// Indicates whether content range information is available in list response
201    pub list_has_content_range: bool,
202    /// Indicates whether content type information is available in list response
203    pub list_has_content_type: bool,
204    /// Indicates whether entity tag is available in list response
205    pub list_has_etag: bool,
206    /// Indicates whether last modified timestamp is available in list response
207    pub list_has_last_modified: bool,
208    /// Indicates whether version information is available in list response
209    pub list_has_version: bool,
210    /// Indicates whether user-defined metadata is available in list response
211    pub list_has_user_metadata: bool,
212
213    /// Indicates if presigned URL generation is supported.
214    pub presign: bool,
215    /// Indicates if presigned URLs for read operations are supported.
216    pub presign_read: bool,
217    /// Indicates if presigned URLs for stat operations are supported.
218    pub presign_stat: bool,
219    /// Indicates if presigned URLs for write operations are supported.
220    pub presign_write: bool,
221    /// Indicates if presigned URLs for delete operations are supported.
222    pub presign_delete: bool,
223
224    /// Indicate if the operator supports shared access.
225    pub shared: bool,
226
227    /// Indicates if blocking operations are supported.
228    pub blocking: bool,
229}
230
231impl Debug for Capability {
232    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
233        // NOTE: All services in opendal are readable.
234        if self.read {
235            f.write_str("Read")?;
236        }
237        if self.write {
238            f.write_str("| Write")?;
239        }
240        if self.list {
241            f.write_str("| List")?;
242        }
243        if self.presign {
244            f.write_str("| Presign")?;
245        }
246        if self.shared {
247            f.write_str("| Shared")?;
248        }
249        if self.blocking {
250            f.write_str("| Blocking")?;
251        }
252        Ok(())
253    }
254}