1 # -*- coding: utf-8 -*-
2 # Copyright 2013 Google Inc. All Rights Reserved.
4 # Licensed under the Apache License, Version 2.0 (the "License");
5 # you may not use this file except in compliance with the License.
6 # You may obtain a copy of the License at
8 # http://www.apache.org/licenses/LICENSE-2.0
10 # Unless required by applicable law or agreed to in writing, software
11 # distributed under the License is distributed on an "AS IS" BASIS,
12 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 # See the License for the specific language governing permissions and
14 # limitations under the License.
15 """Gsutil API delegator for interacting with cloud storage providers."""
17 from __future__
import absolute_import
20 from boto
import config
21 from gslib
.cloud_api
import ArgumentException
22 from gslib
.cloud_api
import CloudApi
23 from gslib
.cs_api_map
import ApiMapConstants
24 from gslib
.cs_api_map
import ApiSelector
27 class CloudApiDelegator(CloudApi
):
28 """Class that handles delegating requests to gsutil Cloud API implementations.
30 This class is responsible for determining at runtime which gsutil Cloud API
31 implementation should service the request based on the Cloud storage provider,
32 command-level API support, and configuration file override.
34 During initialization it takes as an argument a gsutil_api_map which maps
35 providers to their default and supported gsutil Cloud API implementations
36 (see comments in cs_api_map for details).
38 Instantiation of multiple delegators per-thread is required for multiprocess
39 and/or multithreaded operations. Calling methods on the same delegator in
40 multiple threads is unsafe.
43 def __init__(self
, bucket_storage_uri_class
, gsutil_api_map
, logger
,
44 provider
=None, debug
=0):
45 """Performs necessary setup for delegating cloud storage requests.
47 This function has different arguments than the gsutil Cloud API __init__
48 function because of the delegation responsibilties of this class.
51 bucket_storage_uri_class: boto storage_uri class, used by APIs that
52 provide boto translation or mocking.
53 gsutil_api_map: Map of providers and API selector tuples to api classes
54 which can be used to communicate with those providers.
55 logger: logging.logger for outputting log messages.
56 provider: Default provider prefix describing cloud storage provider to
58 debug: Debug level for the API implementation (0..3).
60 super(CloudApiDelegator
, self
).__init
__(bucket_storage_uri_class
, logger
,
61 provider
=provider
, debug
=debug
)
62 self
.api_map
= gsutil_api_map
63 self
.prefer_api
= boto
.config
.get('GSUtil', 'prefer_api', '').upper()
66 if not self
.api_map
[ApiMapConstants
.API_MAP
]:
67 raise ArgumentException('No apiclass supplied for gsutil Cloud API map.')
69 def _GetApi(self
, provider
):
70 """Returns a valid CloudApi for use by the caller.
72 This function lazy-loads connection and credentials using the API map
73 and credential store provided during class initialization.
76 provider: Provider to load API for. If None, class-wide default is used.
79 ArgumentException if there is no matching API available in the API map.
82 Valid API instance that can be used to communicate with the Cloud
85 provider
= provider
or self
.provider
87 raise ArgumentException('No provider selected for _GetApi')
89 provider
= str(provider
)
90 if provider
not in self
.loaded_apis
:
91 self
.loaded_apis
[provider
] = {}
93 api_selector
= self
.GetApiSelector(provider
)
94 if api_selector
not in self
.loaded_apis
[provider
]:
95 # Need to load the API.
96 self
._LoadApi
(provider
, api_selector
)
98 return self
.loaded_apis
[provider
][api_selector
]
100 def _LoadApi(self
, provider
, api_selector
):
101 """Loads a CloudApi into the loaded_apis map for this class.
104 provider: Provider to load the API for.
105 api_selector: cs_api_map.ApiSelector defining the API type.
107 if provider
not in self
.api_map
[ApiMapConstants
.API_MAP
]:
108 raise ArgumentException(
109 'gsutil Cloud API map contains no entry for provider %s.' % provider
)
110 if api_selector
not in self
.api_map
[ApiMapConstants
.API_MAP
][provider
]:
111 raise ArgumentException(
112 'gsutil Cloud API map does not support API %s for provider %s.' %
113 (api_selector
, provider
))
114 self
.loaded_apis
[provider
][api_selector
] = (
115 self
.api_map
[ApiMapConstants
.API_MAP
][provider
][api_selector
](
116 self
.bucket_storage_uri_class
,
121 def GetApiSelector(self
, provider
=None):
122 """Returns a cs_api_map.ApiSelector based on input and configuration.
125 provider: Provider to return the ApiSelector for. If None, class-wide
129 cs_api_map.ApiSelector that will be used for calls to the delegator
132 selected_provider
= provider
or self
.provider
133 if not selected_provider
:
134 raise ArgumentException('No provider selected for CloudApi')
136 if (selected_provider
not in self
.api_map
[ApiMapConstants
.DEFAULT_MAP
] or
137 self
.api_map
[ApiMapConstants
.DEFAULT_MAP
][selected_provider
] not in
138 self
.api_map
[ApiMapConstants
.API_MAP
][selected_provider
]):
139 raise ArgumentException('No default api available for provider %s' %
142 if selected_provider
not in self
.api_map
[ApiMapConstants
.SUPPORT_MAP
]:
143 raise ArgumentException('No supported apis available for provider %s' %
146 api
= self
.api_map
[ApiMapConstants
.DEFAULT_MAP
][selected_provider
]
148 # If we have only HMAC credentials for Google Cloud Storage, we must use
149 # the XML API as the JSON API does not support HMAC.
151 # Technically if we have only HMAC credentials, we should still be able to
152 # access public read resources via the JSON API, but the XML API can do
153 # that just as well. It is better to use it than inspect the credentials on
155 if (provider
== 'gs' and
156 not config
.has_option('Credentials', 'gs_oauth2_refresh_token') and
157 not (config
.has_option('Credentials', 'gs_service_client_id')
158 and config
.has_option('Credentials', 'gs_service_key_file')) and
159 (config
.has_option('Credentials', 'gs_access_key_id')
160 and config
.has_option('Credentials', 'gs_secret_access_key'))):
161 api
= ApiSelector
.XML
162 # Try to force the user's preference to a supported API.
163 elif self
.prefer_api
in (self
.api_map
[ApiMapConstants
.SUPPORT_MAP
]
164 [selected_provider
]):
165 api
= self
.prefer_api
168 # For function docstrings, see CloudApi class.
169 def GetBucket(self
, bucket_name
, provider
=None, fields
=None):
170 return self
._GetApi
(provider
).GetBucket(bucket_name
, fields
=fields
)
172 def ListBuckets(self
, project_id
=None, provider
=None, fields
=None):
173 return self
._GetApi
(provider
).ListBuckets(project_id
=project_id
,
176 def PatchBucket(self
, bucket_name
, metadata
, canned_acl
=None,
177 canned_def_acl
=None, preconditions
=None, provider
=None,
179 return self
._GetApi
(provider
).PatchBucket(
180 bucket_name
, metadata
, canned_acl
=canned_acl
,
181 canned_def_acl
=canned_def_acl
, preconditions
=preconditions
,
184 def CreateBucket(self
, bucket_name
, project_id
=None, metadata
=None,
185 provider
=None, fields
=None):
186 return self
._GetApi
(provider
).CreateBucket(
187 bucket_name
, project_id
=project_id
, metadata
=metadata
, fields
=fields
)
189 def DeleteBucket(self
, bucket_name
, preconditions
=None, provider
=None):
190 return self
._GetApi
(provider
).DeleteBucket(bucket_name
,
191 preconditions
=preconditions
)
193 def ListObjects(self
, bucket_name
, prefix
=None, delimiter
=None,
194 all_versions
=None, provider
=None, fields
=None):
195 return self
._GetApi
(provider
).ListObjects(
196 bucket_name
, prefix
=prefix
, delimiter
=delimiter
,
197 all_versions
=all_versions
, fields
=fields
)
199 def GetObjectMetadata(self
, bucket_name
, object_name
, generation
=None,
200 provider
=None, fields
=None):
201 return self
._GetApi
(provider
).GetObjectMetadata(
202 bucket_name
, object_name
, generation
=generation
, fields
=fields
)
204 def PatchObjectMetadata(self
, bucket_name
, object_name
, metadata
,
205 canned_acl
=None, generation
=None, preconditions
=None,
206 provider
=None, fields
=None):
207 return self
._GetApi
(provider
).PatchObjectMetadata(
208 bucket_name
, object_name
, metadata
, canned_acl
=canned_acl
,
209 generation
=generation
, preconditions
=preconditions
, fields
=fields
)
212 self
, bucket_name
, object_name
, download_stream
, provider
=None,
213 generation
=None, object_size
=None,
214 download_strategy
=CloudApi
.DownloadStrategy
.ONE_SHOT
,
215 start_byte
=0, end_byte
=None, progress_callback
=None,
216 serialization_data
=None, digesters
=None):
217 return self
._GetApi
(provider
).GetObjectMedia(
218 bucket_name
, object_name
, download_stream
,
219 download_strategy
=download_strategy
, start_byte
=start_byte
,
220 end_byte
=end_byte
, generation
=generation
, object_size
=object_size
,
221 progress_callback
=progress_callback
,
222 serialization_data
=serialization_data
, digesters
=digesters
)
224 def UploadObject(self
, upload_stream
, object_metadata
, size
=None,
225 canned_acl
=None, preconditions
=None, progress_callback
=None,
226 provider
=None, fields
=None):
227 return self
._GetApi
(provider
).UploadObject(
228 upload_stream
, object_metadata
, size
=size
, canned_acl
=canned_acl
,
229 preconditions
=preconditions
, progress_callback
=progress_callback
,
232 def UploadObjectStreaming(self
, upload_stream
, object_metadata
,
233 canned_acl
=None, preconditions
=None,
234 progress_callback
=None, provider
=None, fields
=None):
235 return self
._GetApi
(provider
).UploadObjectStreaming(
236 upload_stream
, object_metadata
, canned_acl
=canned_acl
,
237 preconditions
=preconditions
, progress_callback
=progress_callback
,
240 def UploadObjectResumable(
241 self
, upload_stream
, object_metadata
, canned_acl
=None, preconditions
=None,
242 provider
=None, fields
=None, size
=None, serialization_data
=None,
243 tracker_callback
=None, progress_callback
=None):
244 return self
._GetApi
(provider
).UploadObjectResumable(
245 upload_stream
, object_metadata
, canned_acl
=canned_acl
,
246 preconditions
=preconditions
, size
=size
, fields
=fields
,
247 serialization_data
=serialization_data
,
248 tracker_callback
=tracker_callback
, progress_callback
=progress_callback
)
250 def CopyObject(self
, src_obj_metadata
, dst_obj_metadata
, src_generation
=None,
251 canned_acl
=None, preconditions
=None, progress_callback
=None,
252 max_bytes_per_call
=None, provider
=None, fields
=None):
253 return self
._GetApi
(provider
).CopyObject(
254 src_obj_metadata
, dst_obj_metadata
, src_generation
=src_generation
,
255 canned_acl
=canned_acl
, preconditions
=preconditions
,
256 progress_callback
=progress_callback
,
257 max_bytes_per_call
=max_bytes_per_call
, fields
=fields
)
259 def ComposeObject(self
, src_objs_metadata
, dst_obj_metadata
,
260 preconditions
=None, provider
=None, fields
=None):
261 return self
._GetApi
(provider
).ComposeObject(
262 src_objs_metadata
, dst_obj_metadata
, preconditions
=preconditions
,
265 def DeleteObject(self
, bucket_name
, object_name
, preconditions
=None,
266 generation
=None, provider
=None):
267 return self
._GetApi
(provider
).DeleteObject(
268 bucket_name
, object_name
, preconditions
=preconditions
,
269 generation
=generation
)
271 def WatchBucket(self
, bucket_name
, address
, channel_id
, token
=None,
272 provider
=None, fields
=None):
273 return self
._GetApi
(provider
).WatchBucket(
274 bucket_name
, address
, channel_id
, token
=token
, fields
=fields
)
276 def StopChannel(self
, channel_id
, resource_id
, provider
=None):
277 return self
._GetApi
(provider
).StopChannel(channel_id
, resource_id
)
279 def XmlPassThroughGetAcl(self
, storage_url
, def_obj_acl
=False, provider
=None):
280 """XML compatibility function for getting ACLs.
283 storage_url: StorageUrl object.
284 def_obj_acl: If true, get the default object ACL on a bucket.
285 provider: Cloud storage provider to connect to. If not present,
286 class-wide default is used.
289 ArgumentException for errors during input validation.
290 ServiceException for errors interacting with cloud storage providers.
293 ACL XML for the resource specified by storage_url.
295 return self
._GetApi
(provider
).XmlPassThroughGetAcl(storage_url
,
296 def_obj_acl
=def_obj_acl
)
298 def XmlPassThroughSetAcl(self
, acl_text
, storage_url
, canned
=True,
299 def_obj_acl
=False, provider
=None):
300 """XML compatibility function for setting ACLs.
303 acl_text: XML ACL or canned ACL string.
304 storage_url: StorageUrl object.
305 canned: If true, acl_text is treated as a canned ACL string.
306 def_obj_acl: If true, set the default object ACL on a bucket.
307 provider: Cloud storage provider to connect to. If not present,
308 class-wide default is used.
311 ArgumentException for errors during input validation.
312 ServiceException for errors interacting with cloud storage providers.
317 self
._GetApi
(provider
).XmlPassThroughSetAcl(
318 acl_text
, storage_url
, canned
=canned
, def_obj_acl
=def_obj_acl
)
320 def XmlPassThroughGetCors(self
, storage_url
, provider
=None):
321 """XML compatibility function for getting CORS configuration on a bucket.
324 storage_url: StorageUrl object.
325 provider: Cloud storage provider to connect to. If not present,
326 class-wide default is used.
329 ArgumentException for errors during input validation.
330 ServiceException for errors interacting with cloud storage providers.
333 CORS configuration XML for the bucket specified by storage_url.
335 return self
._GetApi
(provider
).XmlPassThroughGetCors(storage_url
)
337 def XmlPassThroughSetCors(self
, cors_text
, storage_url
, provider
=None):
338 """XML compatibility function for setting CORS configuration on a bucket.
341 cors_text: Raw CORS XML string.
342 storage_url: StorageUrl object.
343 provider: Cloud storage provider to connect to. If not present,
344 class-wide default is used.
347 ArgumentException for errors during input validation.
348 ServiceException for errors interacting with cloud storage providers.
353 self
._GetApi
(provider
).XmlPassThroughSetCors(cors_text
, storage_url
)
355 def XmlPassThroughGetLifecycle(self
, storage_url
, provider
=None):
356 """XML compatibility function for getting lifecycle config on a bucket.
359 storage_url: StorageUrl object.
360 provider: Cloud storage provider to connect to. If not present,
361 class-wide default is used.
364 ArgumentException for errors during input validation.
365 ServiceException for errors interacting with cloud storage providers.
368 Lifecycle configuration XML for the bucket specified by storage_url.
370 return self
._GetApi
(provider
).XmlPassThroughGetLifecycle(storage_url
)
372 def XmlPassThroughSetLifecycle(self
, lifecycle_text
, storage_url
,
374 """XML compatibility function for setting CORS configuration on a bucket.
377 lifecycle_text: Raw lifecycle configuration XML string.
378 storage_url: StorageUrl object.
379 provider: Cloud storage provider to connect to. If not present,
380 class-wide default is used.
383 ArgumentException for errors during input validation.
384 ServiceException for errors interacting with cloud storage providers.
389 self
._GetApi
(provider
).XmlPassThroughSetLifecycle(lifecycle_text
,
392 def XmlPassThroughGetLogging(self
, storage_url
, provider
=None):
393 """XML compatibility function for getting logging configuration on a bucket.
396 storage_url: StorageUrl object.
397 provider: Cloud storage provider to connect to. If not present,
398 class-wide default is used.
401 ArgumentException for errors during input validation.
402 ServiceException for errors interacting with cloud storage providers.
405 Logging configuration XML for the bucket specified by storage_url.
407 return self
._GetApi
(provider
).XmlPassThroughGetLogging(storage_url
)
409 def XmlPassThroughGetWebsite(self
, storage_url
, provider
=None):
410 """XML compatibility function for getting website configuration on a bucket.
413 storage_url: StorageUrl object.
414 provider: Cloud storage provider to connect to. If not present,
415 class-wide default is used.
418 ArgumentException for errors during input validation.
419 ServiceException for errors interacting with cloud storage providers.
422 Website configuration XML for the bucket specified by storage_url.
424 return self
._GetApi
(provider
).XmlPassThroughGetWebsite(storage_url
)