clan-core/pkgs/clan-cli/clan_cli/clan_uri.py

170 lines
5.9 KiB
Python
Raw Normal View History

2023-12-05 17:08:27 +00:00
# Import the urllib.parse, enum and dataclasses modules
2023-12-05 17:16:51 +00:00
import dataclasses
2023-12-05 15:17:15 +00:00
import urllib.parse
2023-12-19 17:02:06 +00:00
import urllib.request
2023-12-05 17:08:27 +00:00
from dataclasses import dataclass
2023-12-05 17:16:51 +00:00
from enum import Enum, member
2023-12-05 17:08:27 +00:00
from pathlib import Path
2024-01-16 16:11:26 +00:00
from typing import Any, Self
2023-12-05 17:16:51 +00:00
2023-12-05 15:17:15 +00:00
from .errors import ClanError
2023-12-05 17:16:51 +00:00
2023-12-05 15:17:15 +00:00
2023-12-05 17:08:27 +00:00
# Define an enum with different members that have different values
2023-12-05 15:17:15 +00:00
class ClanScheme(Enum):
2023-12-05 17:08:27 +00:00
# Use the dataclass decorator to add fields and methods to the members
@member
@dataclass
class REMOTE:
2023-12-05 17:16:51 +00:00
url: str # The url field holds the HTTP URL
def __str__(self) -> str:
return f"REMOTE({self.url})" # The __str__ method returns a custom string representation
2023-12-05 17:16:51 +00:00
2023-12-05 17:08:27 +00:00
@member
@dataclass
class LOCAL:
2023-12-05 17:16:51 +00:00
path: Path # The path field holds the local path
def __str__(self) -> str:
return f"LOCAL({self.path})" # The __str__ method returns a custom string representation
2023-12-05 17:08:27 +00:00
2023-12-05 15:17:15 +00:00
2023-12-05 17:08:27 +00:00
# Parameters defined here will be DELETED from the nested uri
# so make sure there are no conflicts with other webservices
@dataclass
2023-12-05 17:16:51 +00:00
class ClanParameters:
flake_attr: str = "defaultVM"
2023-12-05 15:17:15 +00:00
2023-12-05 17:16:51 +00:00
2023-12-05 15:17:15 +00:00
# Define the ClanURI class
class ClanURI:
# Initialize the class with a clan:// URI
def __init__(self, uri: str) -> None:
2024-01-04 15:30:26 +00:00
# users might copy whitespace along with the uri
uri = uri.strip()
2023-12-12 20:31:47 +00:00
self._full_uri = uri
2024-01-16 16:11:26 +00:00
2023-12-05 17:08:27 +00:00
# Check if the URI starts with clan://
2024-01-16 16:11:26 +00:00
# If it does, remove the clan:// prefix
2023-12-05 15:17:15 +00:00
if uri.startswith("clan://"):
2023-12-05 17:08:27 +00:00
self._nested_uri = uri[7:]
2023-12-05 15:17:15 +00:00
else:
2023-12-06 16:13:32 +00:00
raise ClanError(f"Invalid scheme: expected clan://, got {uri}")
2023-12-05 15:17:15 +00:00
# Parse the URI into components
2023-12-05 17:08:27 +00:00
# scheme://netloc/path;parameters?query#fragment
self._components = urllib.parse.urlparse(self._nested_uri)
# Parse the query string into a dictionary
query = urllib.parse.parse_qs(self._components.query)
2023-12-05 17:08:27 +00:00
2024-01-16 16:11:26 +00:00
# Create a new dictionary with only the parameters we want
# example: https://example.com?flake_attr=myVM&password=1234
# becomes: https://example.com?password=1234
# clan_params = {"flake_attr": "myVM"}
# query = {"password": ["1234"]}
clan_params: dict[str, str] = {}
2023-12-05 17:08:27 +00:00
for field in dataclasses.fields(ClanParameters):
if field.name in query:
values = query[field.name]
if len(values) > 1:
2023-12-06 16:13:32 +00:00
raise ClanError(f"Multiple values for parameter: {field.name}")
2024-01-16 16:11:26 +00:00
clan_params[field.name] = values[0]
2023-12-05 17:08:27 +00:00
# Remove the field from the query dictionary
# clan uri and nested uri share one namespace for query parameters
# we need to make sure there are no conflicts
del query[field.name]
2024-01-16 16:11:26 +00:00
# Reencode the query dictionary into a query string
real_query = urllib.parse.urlencode(query, doseq=True)
2023-12-05 17:08:27 +00:00
2024-01-16 16:11:26 +00:00
# If the fragment contains a #, use the part after the # as the flake_attr
# on multiple #, use the first one
if self._components.fragment != "":
clan_params["flake_attr"] = self._components.fragment.split("#")[0]
# Replace the query string in the components with the new query string
2024-01-30 06:28:12 +00:00
self._components = self._components._replace(query=real_query, fragment="")
2024-01-16 16:11:26 +00:00
# Create a ClanParameters object from the clan_params dictionary
self.params = ClanParameters(**clan_params)
2023-12-05 17:08:27 +00:00
comb = (
self._components.scheme,
self._components.netloc,
self._components.path,
self._components.params,
self._components.query,
self._components.fragment,
)
match comb:
case ("file", "", path, "", "", "") | ("", "", path, "", "", _): # type: ignore
self.scheme = ClanScheme.LOCAL.value(Path(path).expanduser().resolve()) # type: ignore
2023-12-05 17:08:27 +00:00
case _:
self.scheme = ClanScheme.REMOTE.value(self._components.geturl()) # type: ignore
def get_internal(self) -> str:
match self.scheme:
case ClanScheme.LOCAL.value(path):
2023-12-19 17:02:06 +00:00
return str(path)
case ClanScheme.REMOTE.value(url):
2023-12-19 17:02:06 +00:00
return url
case _:
raise ClanError(f"Unsupported uri components: {self.scheme}")
2023-12-12 20:31:47 +00:00
def get_full_uri(self) -> str:
return self._full_uri
2024-01-20 12:15:25 +00:00
def get_id(self) -> str:
2024-02-07 09:08:48 +00:00
return f"{self.get_internal()}#{self.params.flake_attr}"
2024-01-20 12:15:25 +00:00
2023-12-12 20:31:47 +00:00
@classmethod
2024-01-16 16:11:26 +00:00
def from_path(
cls, # noqa
path: Path,
flake_attr: str | None = None,
params: dict[str, Any] | ClanParameters | None = None,
) -> Self:
return cls.from_str(str(path), flake_attr=flake_attr, params=params)
2023-12-12 20:31:47 +00:00
@classmethod
2024-01-16 16:11:26 +00:00
def from_str(
cls, # noqa
url: str,
flake_attr: str | None = None,
params: dict[str, Any] | ClanParameters | None = None,
) -> Self:
if flake_attr is not None and params is not None:
raise ClanError("flake_attr and params are mutually exclusive")
prefix = "clan://"
if url.startswith(prefix):
url = url[len(prefix) :]
2024-01-16 16:11:26 +00:00
if params is None and flake_attr is None:
2023-12-12 20:31:47 +00:00
return cls(f"clan://{url}")
comp = urllib.parse.urlparse(url)
query = urllib.parse.parse_qs(comp.query)
2024-01-16 16:11:26 +00:00
if isinstance(params, dict):
query.update(params)
elif isinstance(params, ClanParameters):
query.update(params.__dict__)
elif flake_attr is not None:
query["flake_attr"] = [flake_attr]
else:
raise ClanError(f"Unsupported params type: {type(params)}")
2023-12-12 20:31:47 +00:00
new_query = urllib.parse.urlencode(query, doseq=True)
comp = comp._replace(query=new_query)
new_url = urllib.parse.urlunparse(comp)
return cls(f"clan://{new_url}")
2023-12-08 11:18:55 +00:00
def __str__(self) -> str:
return self.get_full_uri()
def __repr__(self) -> str:
return f"ClanURI({self.get_full_uri()})"