mirror of
https://github.com/OpenHands/OpenHands.git
synced 2025-12-26 13:52:43 +08:00
81 lines
2.7 KiB
Python
81 lines
2.7 KiB
Python
import os
|
|
|
|
from google.api_core.exceptions import NotFound
|
|
from google.cloud import storage
|
|
from google.cloud.storage.blob import Blob
|
|
from google.cloud.storage.bucket import Bucket
|
|
from google.cloud.storage.client import Client
|
|
|
|
from openhands.storage.files import FileStore
|
|
|
|
|
|
class GoogleCloudFileStore(FileStore):
|
|
def __init__(self, bucket_name: str | None = None) -> None:
|
|
"""Create a new FileStore.
|
|
|
|
If GOOGLE_APPLICATION_CREDENTIALS is defined in the environment it will be used
|
|
for authentication. Otherwise access will be anonymous.
|
|
"""
|
|
if bucket_name is None:
|
|
bucket_name = os.environ['GOOGLE_CLOUD_BUCKET_NAME']
|
|
self.storage_client: Client = storage.Client()
|
|
self.bucket: Bucket = self.storage_client.bucket(bucket_name)
|
|
|
|
def write(self, path: str, contents: str | bytes) -> None:
|
|
blob: Blob = self.bucket.blob(path)
|
|
mode = 'wb' if isinstance(contents, bytes) else 'w'
|
|
with blob.open(mode) as f:
|
|
f.write(contents)
|
|
|
|
def read(self, path: str) -> str:
|
|
blob: Blob = self.bucket.blob(path)
|
|
try:
|
|
with blob.open('r') as f:
|
|
return str(f.read())
|
|
except NotFound as err:
|
|
raise FileNotFoundError(err)
|
|
|
|
def list(self, path: str) -> list[str]:
|
|
if not path or path == '/':
|
|
path = ''
|
|
elif not path.endswith('/'):
|
|
path += '/'
|
|
# The delimiter logic screens out directories, so we can't use it. :(
|
|
# For example, given a structure:
|
|
# foo/bar/zap.txt
|
|
# foo/bar/bang.txt
|
|
# ping.txt
|
|
# prefix=None, delimiter="/" yields ["ping.txt"] # :(
|
|
# prefix="foo", delimiter="/" yields [] # :(
|
|
blobs: set[str] = set()
|
|
prefix_len = len(path)
|
|
for blob in self.bucket.list_blobs(prefix=path):
|
|
name: str = blob.name
|
|
if name == path:
|
|
continue
|
|
try:
|
|
index = name.index('/', prefix_len + 1)
|
|
if index != prefix_len:
|
|
blobs.add(name[: index + 1])
|
|
except ValueError:
|
|
blobs.add(name)
|
|
return list(blobs)
|
|
|
|
def delete(self, path: str) -> None:
|
|
# Sanitize path
|
|
if not path or path == '/':
|
|
path = ''
|
|
if path.endswith('/'):
|
|
path = path[:-1]
|
|
|
|
# Try to delete any child resources (Assume the path is a directory)
|
|
for blob in self.bucket.list_blobs(prefix=f'{path}/'):
|
|
blob.delete()
|
|
|
|
# Next try to delete item as a file
|
|
try:
|
|
file_blob: Blob = self.bucket.blob(path)
|
|
file_blob.delete()
|
|
except NotFound:
|
|
pass
|