Skip to content

Commit

Permalink
add local object store commands
Browse files Browse the repository at this point in the history
  • Loading branch information
rjra2611 committed Aug 22, 2023
1 parent cc35972 commit 5d0444d
Show file tree
Hide file tree
Showing 13 changed files with 257 additions and 76 deletions.
3 changes: 2 additions & 1 deletion lean/commands/cloud/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@
from lean.commands.cloud.pull import pull
from lean.commands.cloud.push import push
from lean.commands.cloud.status import status

from lean.commands.cloud.object_store import object_store

@group()
def cloud() -> None:
Expand All @@ -35,3 +35,4 @@ def cloud() -> None:
cloud.add_command(optimize)
cloud.add_command(live)
cloud.add_command(status)
cloud.add_command(object_store)
24 changes: 24 additions & 0 deletions lean/commands/cloud/object_store/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
# QUANTCONNECT.COM - Democratizing Finance, Empowering Individuals.
# Lean CLI v1.0. Copyright 2021 QuantConnect Corporation.
#
# 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.

from lean.commands.cloud.object_store.object_store import object_store
from lean.commands.cloud.object_store.get import get
from lean.commands.cloud.object_store.set import set
from lean.commands.cloud.object_store.list import list
from lean.commands.cloud.object_store.delete import delete

object_store.add_command(get)
object_store.add_command(set)
object_store.add_command(list)
object_store.add_command(delete)

29 changes: 29 additions & 0 deletions lean/commands/cloud/object_store/delete.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
# QUANTCONNECT.COM - Democratizing Finance, Empowering Individuals.
# Lean CLI v1.0. Copyright 2021 QuantConnect Corporation.
#
# 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.

from click import command, argument

from lean.click import LeanCommand
from lean.container import container


@command(cls=LeanCommand)
@argument("key", type=str)
def delete(key: str) -> str:
"""
Delete a value from the organization's cloud object store.
"""
organization_id = container.organization_manager.try_get_working_organization_id()
api_client = container.api_client
api_client.object_store.delete(key, organization_id)
53 changes: 53 additions & 0 deletions lean/commands/cloud/object_store/get.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
# QUANTCONNECT.COM - Democratizing Finance, Empowering Individuals.
# Lean CLI v1.0. Copyright 2021 QuantConnect Corporation.
#
# 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.

from click import argument
from lean.commands.cloud.object_store import object_store
from lean.click import LeanCommand
from lean.container import container


@object_store.command(cls=LeanCommand, name="get", aliases=["ls"])
@argument("key", type=str)
def get(key: str) -> str:
"""
Get a value from the organization's cloud object store.
"""
organization_id = container.organization_manager.try_get_working_organization_id()
api_client = container.api_client
logger = container.logger
data = api_client.object_store.get(key, organization_id)

try:
headers = ["size", "modified", "preview", "key"]
display_headers = ["Bytes", "Modified", "Preview", "Filename"]
row = [str(data["metadata"].get(header, "")) for header in headers]
all_rows = [display_headers] + [row]
column_widths = [max(len(row[i]) for row in all_rows) for i in range(len(all_rows[0]))]

logger.info(" ".join(header.ljust(width) for header, width in zip(display_headers, column_widths)))

bytes = str(data["metadata"].get('size', ""))
modified = data["metadata"].get('modified', "")
preview = data["metadata"].get('preview', "")[:10].rstrip()
filename = data["metadata"].get('key', "")

values = [bytes, modified, preview, filename]
logger.info(" ".join(value.ljust(width) for value, width in zip(values, column_widths)))
except KeyError as e:
logger.error(f"Key {key} not found.")
except Exception as e:
logger.error(f"Error: {e}")


43 changes: 43 additions & 0 deletions lean/commands/cloud/object_store/list.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
# QUANTCONNECT.COM - Democratizing Finance, Empowering Individuals.
# Lean CLI v1.0. Copyright 2021 QuantConnect Corporation.
#
# 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.

from click import command, argument

from lean.click import LeanCommand
from lean.container import container


@command(cls=LeanCommand)
@argument("root-key", type=str)
def list(root_key: str) -> str:
"""
List all values for the given root key in the organization's cloud object store.
"""
organization_id = container.organization_manager.try_get_working_organization_id()
api_client = container.api_client
logger = container.logger
data = api_client.object_store.list(root_key, organization_id)

try:
headers = ["key", "size", "folder", "name"]
display_headers = ["Key", "Bytes", "Folder", "Filename"]
rows = [[str(obj.get(header, "")) for header in headers] for obj in data['objects']]
all_rows = [display_headers] + rows
column_widths = [max(len(row[i]) for row in all_rows) for i in range(len(all_rows[0]))]
for row in all_rows:
logger.info(" ".join(value.ljust(width) for value, width in zip(row, column_widths)))
except KeyError as e:
logger.error(f"Key {root_key} not found.")
except Exception as e:
logger.error(f"Error: {e}")
23 changes: 23 additions & 0 deletions lean/commands/cloud/object_store/object_store.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
# QUANTCONNECT.COM - Democratizing Finance, Empowering Individuals.
# Lean CLI v1.0. Copyright 2021 QuantConnect Corporation.
#
# 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.

from click import group
from lean.components.util.click_aliased_command_group import AliasedCommandGroup

@group(cls=AliasedCommandGroup, invoke_without_command=True)
def object_store() -> None:
"""Interact with the Organization's Cloud Object Store."""
# This method is intentionally empty
# It is used as the command group for all `lean object-store <command>` commands
pass

34 changes: 34 additions & 0 deletions lean/commands/cloud/object_store/set.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
# QUANTCONNECT.COM - Democratizing Finance, Empowering Individuals.
# Lean CLI v1.0. Copyright 2021 QuantConnect Corporation.
#
# 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.

from click import command, argument
from pathlib import Path
from lean.click import LeanCommand, PathParameter
from lean.container import container


@command(cls=LeanCommand)
@argument("key", type=str)
@argument("path", type= PathParameter(exists=True, file_okay=True, dir_okay=False))
def set(key: str, path: Path) -> None:
"""Sets the data to the given key in the organization's cloud object store.
:param key: The key to set the data to.
:param path: Path to the file containing the object data.
"""
organization_id = container.organization_manager.try_get_working_organization_id()
container.logger.info(f"Setting object {key} in organization {organization_id}")
api_client = container.api_client
with open(path, 'r', encoding='utf-8') as f:
objectData = f.read()
api_client.object_store.set(key, objectData, organization_id)
13 changes: 5 additions & 8 deletions lean/commands/object_store/delete.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,19 +11,16 @@
# See the License for the specific language governing permissions and
# limitations under the License.

from click import command, argument

from click import command
from lean.click import LeanCommand
from lean.container import container
from lean.components.util.object_store_helper import open_storage_directory_in_explorer


@command(cls=LeanCommand)
@argument("key", type=str)
def delete(key: str) -> str:
def delete() -> str:
"""
Delete a value from the organization's object store.
Opens the local storage directory in the file explorer.
"""
organization_id = container.organization_manager.try_get_working_organization_id()
api_client = container.api_client
api_client.object_store.delete(key, organization_id)
open_storage_directory_in_explorer(container.lean_config_manager)
34 changes: 5 additions & 29 deletions lean/commands/object_store/get.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,43 +11,19 @@
# See the License for the specific language governing permissions and
# limitations under the License.

from click import argument
from lean.commands.object_store import object_store
from lean.click import LeanCommand
from lean.container import container
from lean.components.util.object_store_helper import open_storage_directory_in_explorer
from lean.commands.object_store import object_store


@object_store.command(cls=LeanCommand, name="get", aliases=["ls"])
@argument("key", type=str)
def get(key: str) -> str:
def get() -> str:
"""
Get a value from the organization's object store.
Opens the local storage directory in the file explorer.
"""
organization_id = container.organization_manager.try_get_working_organization_id()
api_client = container.api_client
logger = container.logger
data = api_client.object_store.get(key, organization_id)

try:
headers = ["size", "modified", "preview", "key"]
display_headers = ["Bytes", "Modified", "Preview", "Filename"]
row = [str(data["metadata"].get(header, "")) for header in headers]
all_rows = [display_headers] + [row]
column_widths = [max(len(row[i]) for row in all_rows) for i in range(len(all_rows[0]))]

logger.info(" ".join(header.ljust(width) for header, width in zip(display_headers, column_widths)))
open_storage_directory_in_explorer(container.lean_config_manager)

bytes = str(data["metadata"].get('size', ""))
modified = data["metadata"].get('modified', "")
preview = data["metadata"].get('preview', "")[:10].rstrip()
filename = data["metadata"].get('key', "")

values = [bytes, modified, preview, filename]
logger.info(" ".join(value.ljust(width) for value, width in zip(values, column_widths)))
except KeyError as e:
logger.error(f"Key {key} not found.")
except Exception as e:
logger.error(f"Error: {e}")


27 changes: 5 additions & 22 deletions lean/commands/object_store/list.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,33 +11,16 @@
# See the License for the specific language governing permissions and
# limitations under the License.

from click import command, argument

from click import command
from lean.click import LeanCommand
from lean.container import container
from lean.components.util.object_store_helper import open_storage_directory_in_explorer


@command(cls=LeanCommand)
@argument("root-key", type=str)
def list(root_key: str) -> str:
def list() -> str:
"""
List all values for the given root key in the organization's object store.
Opens the local storage directory in the file explorer.
"""
organization_id = container.organization_manager.try_get_working_organization_id()
api_client = container.api_client
logger = container.logger
data = api_client.object_store.list(root_key, organization_id)

try:
headers = ["key", "size", "folder", "name"]
display_headers = ["Key", "Bytes", "Folder", "Filename"]
rows = [[str(obj.get(header, "")) for header in headers] for obj in data['objects']]
all_rows = [display_headers] + rows
column_widths = [max(len(row[i]) for row in all_rows) for i in range(len(all_rows[0]))]
for row in all_rows:
logger.info(" ".join(value.ljust(width) for value, width in zip(row, column_widths)))
except KeyError as e:
logger.error(f"Key {root_key} not found.")
except Exception as e:
logger.error(f"Error: {e}")
open_storage_directory_in_explorer(container.lean_config_manager)
2 changes: 1 addition & 1 deletion lean/commands/object_store/object_store.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@

@group(cls=AliasedCommandGroup, invoke_without_command=True)
def object_store() -> None:
"""Interact with the Organization's Object Store."""
"""Interact with the Organization's Local Object Store."""
# This method is intentionally empty
# It is used as the command group for all `lean object-store <command>` commands
pass
Expand Down
22 changes: 7 additions & 15 deletions lean/commands/object_store/set.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,24 +11,16 @@
# See the License for the specific language governing permissions and
# limitations under the License.

from click import command, argument
from pathlib import Path
from lean.click import LeanCommand, PathParameter
from click import command
from lean.click import LeanCommand
from lean.container import container
from lean.components.util.object_store_helper import open_storage_directory_in_explorer


@command(cls=LeanCommand)
@argument("key", type=str)
@argument("path", type= PathParameter(exists=True, file_okay=True, dir_okay=False))
def set(key: str, path: Path) -> None:
"""Sets the data to the given key in the organization's object store.
def set() -> None:
"""
Opens the local storage directory in the file explorer.
:param key: The key to set the data to.
:param path: Path to the file containing the object data.
"""
organization_id = container.organization_manager.try_get_working_organization_id()
container.logger.info(f"Setting object {key} in organization {organization_id}")
api_client = container.api_client
with open(path, 'r', encoding='utf-8') as f:
objectData = f.read()
api_client.object_store.set(key, objectData, organization_id)
open_storage_directory_in_explorer(container.lean_config_manager)
Loading

0 comments on commit 5d0444d

Please sign in to comment.