Source code for camel.storages.object_storages.azure_blob
# ========= Copyright 2023-2024 @ CAMEL-AI.org. All Rights Reserved. =========
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
# ========= Copyright 2023-2024 @ CAMEL-AI.org. All Rights Reserved. =========
import os
from pathlib import Path, PurePath
from typing import Optional, Tuple
from warnings import warn
from camel.loaders import File, create_file_from_raw_bytes
from camel.storages.object_storages.base import BaseObjectStorage
[docs]
class AzureBlobStorage(BaseObjectStorage):
r"""A class to connect to Azure Blob Storage. It will connect to one
container in the storage account.
Args:
storage_account_name (str): The name of the storage account.
container_name (str): The name of the container.
access_key (Optional[str], optional): The access key of the storage
account. Defaults to None.
References:
https://azure.microsoft.com/en-us/products/storage/blobs
"""
def __init__(
self,
storage_account_name: str,
container_name: str,
create_if_not_exists: bool = True,
access_key: Optional[str] = None,
) -> None:
access_key = access_key or os.getenv("AZURE_ACCESS_KEY")
self._create_if_not_exists = create_if_not_exists
if not access_key:
warn("AZURE_ACCESS_KEY not provided.")
# Make all the empty values None
access_key = None
from azure.storage.blob import ContainerClient
self._client = ContainerClient(
account_url="https://"
f"{storage_account_name}.blob.core.windows.net",
credential=access_key,
container_name=container_name,
)
self._prepare_and_check()
def _prepare_and_check(self) -> None:
r"""Check privileges and existence of the container."""
from azure.core.exceptions import ClientAuthenticationError
try:
exists = self._client.exists()
if not exists and self._create_if_not_exists:
self._client.create_container()
warn(
f"Container {self._client.container_name} not found. "
f"Automatically created."
)
elif not exists:
raise FileNotFoundError(
f"Failed to access container {self._client.container_name}"
f": Not found."
)
except ClientAuthenticationError:
raise PermissionError(
f"Failed to access container {self._client.container_name}: "
f"No permission."
)
[docs]
@staticmethod
def canonicalize_path(file_path: PurePath) -> Tuple[str, str]:
r"""Canonicalize file path for Azure Blob Storage.
Args:
file_path (PurePath): The path to be canonicalized.
Returns:
Tuple[str, str]: The canonicalized file key and file name.
"""
# for Azure, both slash and backslash will be treated as separator
filename = file_path.name
if "\\" in filename:
raise ValueError(
"Azure Blob Storage does not support backslash in filename."
)
return file_path.as_posix(), filename
def _put_file(self, file_key: str, file: File) -> None:
r"""Put a file to the Azure Blob Storage container.
Args:
file_key (str): The path to the object in the container.
file (File): The file to be uploaded.
"""
self._client.upload_blob(
name=file_key, data=file.raw_bytes, overwrite=True
)
def _get_file(self, file_key: str, filename: str) -> File:
r"""Get a file from the Azure Blob Storage container.
Args:
file_key (str): The path to the object in the container.
filename (str): The name of the file.
Returns:
File: The object from the container.
"""
raw_bytes = self._client.download_blob(file_key).readall()
file = create_file_from_raw_bytes(raw_bytes, filename)
return file
def _upload_file(
self, local_file_path: Path, remote_file_key: str
) -> None:
r"""Upload a local file to the Azure Blob Storage container.
Args:
local_file_path (Path): The path to the local file to be uploaded.
remote_file_key (str): The path to the object in the container.
"""
with open(local_file_path, "rb") as f:
self._client.upload_blob(
name=remote_file_key, data=f, overwrite=True
)
def _download_file(
self, local_file_path: Path, remote_file_key: str
) -> None:
r"""Download a file from the Azure Blob Storage container to the local
system.
Args:
local_file_path (Path): The path to the local file to be saved.
remote_file_key (str): The key of the object in the container.
"""
with open(local_file_path, "wb") as f:
f.write(self._client.download_blob(remote_file_key).readall())
def _object_exists(self, file_key: str) -> bool:
r"""
Check if the object exists in the Azure Blob Storage container.
Args:
file_key: The key of the object in the container.
Returns:
bool: Whether the object exists in the container.
"""
return self._client.get_blob_client(file_key).exists()