Changed code to support older Python versions

This commit is contained in:
Malasaur 2025-12-01 23:27:09 +01:00
parent eb92d2d36f
commit 582458cdd0
5027 changed files with 794942 additions and 4 deletions

View file

@ -0,0 +1,621 @@
import contextlib
import logging
import subprocess
import tempfile
import time
from enum import Enum
from itertools import cycle
from pathlib import Path
from typing import Any, Dict, List, Optional, Union
import fastar
import rignore
import typer
from httpx import Client
from pydantic import BaseModel, EmailStr, ValidationError
from rich.text import Text
from rich_toolkit import RichToolkit
from rich_toolkit.menu import Option
from typing_extensions import Annotated
from fastapi_cloud_cli.commands.login import login
from fastapi_cloud_cli.utils.api import APIClient, BuildLogError, TooManyRetriesError
from fastapi_cloud_cli.utils.apps import AppConfig, get_app_config, write_app_config
from fastapi_cloud_cli.utils.auth import is_logged_in
from fastapi_cloud_cli.utils.cli import get_rich_toolkit, handle_http_errors
from fastapi_cloud_cli.utils.pydantic_compat import (
TypeAdapter,
model_dump,
model_validate,
)
logger = logging.getLogger(__name__)
def _get_app_name(path: Path) -> str:
# TODO: use pyproject.toml to get the app name
return path.name
def _should_exclude_entry(path: Path) -> bool:
parts_to_exclude = [
".venv",
"__pycache__",
".mypy_cache",
".pytest_cache",
".gitignore",
".fastapicloudignore",
]
if any(part in path.parts for part in parts_to_exclude):
return True
if path.suffix == ".pyc":
return True
return False
def archive(path: Path, tar_path: Path) -> Path:
logger.debug("Starting archive creation for path: %s", path)
files = rignore.walk(
path,
should_exclude_entry=_should_exclude_entry,
additional_ignore_paths=[".fastapicloudignore"],
ignore_hidden=False,
)
logger.debug("Archive will be created at: %s", tar_path)
file_count = 0
with fastar.open(tar_path, "w") as tar:
for filename in files:
if filename.is_dir():
continue
arcname = filename.relative_to(path)
logger.debug("Adding %s to archive", arcname)
tar.append(filename, arcname=arcname)
file_count += 1
logger.debug("Archive created successfully with %s files", file_count)
return tar_path
class Team(BaseModel):
id: str
slug: str
name: str
def _get_teams() -> List[Team]:
with APIClient() as client:
response = client.get("/teams/")
response.raise_for_status()
data = response.json()["data"]
return [model_validate(Team, team) for team in data]
class AppResponse(BaseModel):
id: str
slug: str
def _create_app(team_id: str, app_name: str) -> AppResponse:
with APIClient() as client:
response = client.post(
"/apps/",
json={"name": app_name, "team_id": team_id},
)
response.raise_for_status()
return model_validate(AppResponse, response.json())
class DeploymentStatus(str, Enum):
waiting_upload = "waiting_upload"
ready_for_build = "ready_for_build"
building = "building"
extracting = "extracting"
extracting_failed = "extracting_failed"
building_image = "building_image"
building_image_failed = "building_image_failed"
deploying = "deploying"
deploying_failed = "deploying_failed"
verifying = "verifying"
verifying_failed = "verifying_failed"
verifying_skipped = "verifying_skipped"
success = "success"
failed = "failed"
@classmethod
def to_human_readable(cls, status: "DeploymentStatus") -> str:
return {
cls.waiting_upload: "Waiting for upload",
cls.ready_for_build: "Ready for build",
cls.building: "Building",
cls.extracting: "Extracting",
cls.extracting_failed: "Extracting failed",
cls.building_image: "Building image",
cls.building_image_failed: "Build failed",
cls.deploying: "Deploying",
cls.deploying_failed: "Deploying failed",
cls.verifying: "Verifying",
cls.verifying_failed: "Verifying failed",
cls.verifying_skipped: "Verification skipped",
cls.success: "Success",
cls.failed: "Failed",
}[status]
class CreateDeploymentResponse(BaseModel):
id: str
app_id: str
slug: str
status: DeploymentStatus
dashboard_url: str
url: str
def _create_deployment(app_id: str) -> CreateDeploymentResponse:
with APIClient() as client:
response = client.post(f"/apps/{app_id}/deployments/")
response.raise_for_status()
return model_validate(CreateDeploymentResponse, response.json())
class RequestUploadResponse(BaseModel):
url: str
fields: Dict[str, str]
def _upload_deployment(deployment_id: str, archive_path: Path) -> None:
logger.debug(
"Starting deployment upload for deployment: %s",
deployment_id,
)
logger.debug(
"Archive path: %s, size: %s bytes",
archive_path,
archive_path.stat().st_size,
)
with APIClient() as fastapi_client, Client() as client:
# Get the upload URL
logger.debug("Requesting upload URL from API")
response = fastapi_client.post(f"/deployments/{deployment_id}/upload")
response.raise_for_status()
upload_data = model_validate(RequestUploadResponse, response.json())
logger.debug("Received upload URL: %s", upload_data.url)
logger.debug("Starting file upload to S3")
with open(archive_path, "rb") as archive_file:
upload_response = client.post(
upload_data.url,
data=upload_data.fields,
files={"file": archive_file},
)
upload_response.raise_for_status()
logger.debug("File upload completed successfully")
# Notify the server that the upload is complete
logger.debug("Notifying API that upload is complete")
notify_response = fastapi_client.post(
f"/deployments/{deployment_id}/upload-complete"
)
notify_response.raise_for_status()
logger.debug("Upload notification sent successfully")
def _get_app(app_slug: str) -> Optional[AppResponse]:
with APIClient() as client:
response = client.get(f"/apps/{app_slug}")
if response.status_code == 404:
return None
response.raise_for_status()
data = response.json()
return model_validate(AppResponse, data)
def _get_apps(team_id: str) -> List[AppResponse]:
with APIClient() as client:
response = client.get("/apps/", params={"team_id": team_id})
response.raise_for_status()
data = response.json()["data"]
return [model_validate(AppResponse, app) for app in data]
WAITING_MESSAGES = [
"🚀 Preparing for liftoff! Almost there...",
"👹 Sneaking past the dependency gremlins... Don't wake them up!",
"🤏 Squishing code into a tiny digital sandwich. Nom nom nom.",
"🐱 Removing cat videos from our servers to free up space.",
"🐢 Uploading at blazing speeds of 1 byte per hour. Patience, young padawan.",
"🔌 Connecting to server... Please stand by while we argue with the firewall.",
"💥 Oops! We've angered the Python God. Sacrificing a rubber duck to appease it.",
"🧙 Sprinkling magic deployment dust. Abracadabra!",
"👀 Hoping that @tiangolo doesn't find out about this deployment.",
"🍪 Cookie monster detected on server. Deploying anti-cookie shields.",
]
LONG_WAIT_MESSAGES = [
"😅 Well, that's embarrassing. We're still waiting for the deployment to finish...",
"🤔 Maybe we should have brought snacks for this wait...",
"🥱 Yawn... Still waiting...",
"🤯 Time is relative... Especially when you're waiting for a deployment...",
]
def _configure_app(toolkit: RichToolkit, path_to_deploy: Path) -> AppConfig:
if not toolkit.confirm(f"Setup and deploy [blue]{path_to_deploy}[/]?", tag="dir"):
raise typer.Exit(0)
toolkit.print_line()
with toolkit.progress("Fetching teams...") as progress:
with handle_http_errors(
progress, message="Error fetching teams. Please try again later."
):
teams = _get_teams()
toolkit.print_line()
team = toolkit.ask(
"Select the team you want to deploy to:",
tag="team",
options=[Option({"name": team.name, "value": team}) for team in teams],
)
toolkit.print_line()
create_new_app = toolkit.confirm(
"Do you want to create a new app?", tag="app", default=True
)
toolkit.print_line()
if not create_new_app:
with toolkit.progress("Fetching apps...") as progress:
with handle_http_errors(
progress, message="Error fetching apps. Please try again later."
):
apps = _get_apps(team.id)
toolkit.print_line()
if not apps:
toolkit.print(
"No apps found in this team. You can create a new app instead.",
)
raise typer.Exit(1)
app = toolkit.ask(
"Select the app you want to deploy to:",
options=[Option({"name": app.slug, "value": app}) for app in apps],
)
else:
app_name = toolkit.input(
title="What's your app name?",
default=_get_app_name(path_to_deploy),
)
toolkit.print_line()
with toolkit.progress(title="Creating app...") as progress:
with handle_http_errors(progress):
app = _create_app(team.id, app_name)
progress.log(f"App created successfully! App slug: {app.slug}")
app_config = AppConfig(app_id=app.id, team_id=team.id)
write_app_config(path_to_deploy, app_config)
return app_config
def _wait_for_deployment(
toolkit: RichToolkit, app_id: str, deployment: CreateDeploymentResponse
) -> None:
messages = cycle(WAITING_MESSAGES)
toolkit.print(
"Checking the status of your deployment 👀",
tag="cloud",
)
toolkit.print_line()
toolkit.print(
f"You can also check the status at [link={deployment.dashboard_url}]{deployment.dashboard_url}[/link]",
)
toolkit.print_line()
time_elapsed = 0.0
started_at = time.monotonic()
last_message_changed_at = time.monotonic()
with toolkit.progress(
next(messages), inline_logs=True, lines_to_show=20
) as progress, APIClient() as client:
try:
for log in client.stream_build_logs(deployment.id):
time_elapsed = time.monotonic() - started_at
if log.type == "message":
progress.log(Text.from_ansi(log.message.rstrip()))
if log.type == "complete":
progress.log("")
progress.log(
f"🐔 Ready the chicken! Your app is ready at [link={deployment.url}]{deployment.url}[/link]"
)
progress.log("")
progress.log(
f"You can also check the app logs at [link={deployment.dashboard_url}]{deployment.dashboard_url}[/link]"
)
break
if log.type == "failed":
progress.log("")
progress.log(
f"😔 Oh no! Something went wrong. Check out the logs at [link={deployment.dashboard_url}]{deployment.dashboard_url}[/link]"
)
raise typer.Exit(1)
if time_elapsed > 30:
messages = cycle(LONG_WAIT_MESSAGES)
if (time.monotonic() - last_message_changed_at) > 2:
progress.title = next(messages)
last_message_changed_at = time.monotonic()
except (BuildLogError, TooManyRetriesError) as e:
logger.error("Build log streaming failed: %s", e)
toolkit.print_line()
toolkit.print(
f"⚠️ Unable to stream build logs. Check the dashboard for status: [link={deployment.dashboard_url}]{deployment.dashboard_url}[/link]"
)
raise typer.Exit(1) from e
class SignupToWaitingList(BaseModel):
email: EmailStr
name: Optional[str] = None
organization: Optional[str] = None
role: Optional[str] = None
team_size: Optional[str] = None
location: Optional[str] = None
use_case: Optional[str] = None
secret_code: Optional[str] = None
def _send_waitlist_form(
result: SignupToWaitingList,
toolkit: RichToolkit,
) -> None:
with toolkit.progress("Sending your request...") as progress:
with APIClient() as client:
with handle_http_errors(progress):
response = client.post("/users/waiting-list", json=model_dump(result))
response.raise_for_status()
progress.log("Let's go! Thanks for your interest in FastAPI Cloud! 🚀")
def _waitlist_form(toolkit: RichToolkit) -> None:
from rich_toolkit.form import Form
toolkit.print(
"We're currently in private beta. If you want to be notified when we launch, please fill out the form below.",
tag="waitlist",
)
toolkit.print_line()
email = toolkit.input(
"Enter your email:",
required=True,
validator=TypeAdapter(EmailStr),
)
toolkit.print_line()
result = model_validate(SignupToWaitingList, {"email": email})
if toolkit.confirm(
"Do you want to get access faster by giving us more information?",
tag="waitlist",
):
toolkit.print_line()
form = Form("Waitlist form", style=toolkit.style)
form.add_input("name", label="Name", placeholder="John Doe")
form.add_input("organization", label="Organization", placeholder="Acme Inc.")
form.add_input("team", label="Team", placeholder="Team A")
form.add_input("role", label="Role", placeholder="Developer")
form.add_input("location", label="Location", placeholder="San Francisco")
form.add_input(
"use_case",
label="How do you plan to use FastAPI Cloud?",
placeholder="I'm building a web app",
)
form.add_input("secret_code", label="Secret code", placeholder="123456")
result = form.run() # type: ignore
try:
result = model_validate(
SignupToWaitingList,
{
"email": email,
**result, # type: ignore
},
)
except ValidationError:
toolkit.print(
"[error]Invalid form data. Please try again.[/]",
)
return
toolkit.print_line()
if toolkit.confirm(
(
"Do you agree to\n"
"- Terms of Service: [link=https://fastapicloud.com/legal/terms]https://fastapicloud.com/legal/terms[/link]\n"
"- Privacy Policy: [link=https://fastapicloud.com/legal/privacy-policy]https://fastapicloud.com/legal/privacy-policy[/link]\n"
),
tag="terms",
):
toolkit.print_line()
_send_waitlist_form(
result,
toolkit,
)
with contextlib.suppress(Exception):
subprocess.run(
["open", "-g", "raycast://confetti?emojis=🐔⚡"],
stdout=subprocess.DEVNULL,
stderr=subprocess.DEVNULL,
check=False,
)
def deploy(
path: Annotated[
Union[Path, None],
typer.Argument(
help="A path to the folder containing the app you want to deploy"
),
] = None,
skip_wait: Annotated[
bool, typer.Option("--no-wait", help="Skip waiting for deployment status")
] = False,
) -> Any:
"""
Deploy a [bold]FastAPI[/bold] app to FastAPI Cloud. 🚀
"""
logger.debug("Deploy command started")
logger.debug("Deploy path: %s, skip_wait: %s", path, skip_wait)
with get_rich_toolkit() as toolkit:
if not is_logged_in():
logger.debug("User not logged in, prompting for login or waitlist")
toolkit.print_title("Welcome to FastAPI Cloud!", tag="FastAPI")
toolkit.print_line()
toolkit.print(
"You need to be logged in to deploy to FastAPI Cloud.",
tag="info",
)
toolkit.print_line()
choice = toolkit.ask(
"What would you like to do?",
tag="auth",
options=[
Option({"name": "Login to my existing account", "value": "login"}),
Option({"name": "Join the waiting list", "value": "waitlist"}),
],
)
toolkit.print_line()
if choice == "login":
login()
else:
_waitlist_form(toolkit)
raise typer.Exit(1)
toolkit.print_title("Starting deployment", tag="FastAPI")
toolkit.print_line()
path_to_deploy = path or Path.cwd()
logger.debug("Deploying from path: %s", path_to_deploy)
app_config = get_app_config(path_to_deploy)
if not app_config:
logger.debug("No app config found, configuring new app")
app_config = _configure_app(toolkit, path_to_deploy=path_to_deploy)
toolkit.print_line()
else:
logger.debug("Existing app config found, proceeding with deployment")
toolkit.print("Deploying app...")
toolkit.print_line()
with toolkit.progress("Checking app...", transient=True) as progress:
with handle_http_errors(progress):
logger.debug("Checking app with ID: %s", app_config.app_id)
app = _get_app(app_config.app_id)
if not app:
logger.debug("App not found in API")
progress.set_error(
"App not found. Make sure you're logged in the correct account."
)
if not app:
toolkit.print_line()
toolkit.print(
"If you deleted this app, you can run [bold]fastapi cloud unlink[/] to unlink the local configuration.",
tag="tip",
)
raise typer.Exit(1)
with tempfile.TemporaryDirectory() as temp_dir:
logger.debug("Creating archive for deployment")
archive_path = Path(temp_dir) / "archive.tar"
archive(path or Path.cwd(), archive_path)
with toolkit.progress(
title="Creating deployment"
) as progress, handle_http_errors(progress):
logger.debug("Creating deployment for app: %s", app.id)
deployment = _create_deployment(app.id)
progress.log(
f"Deployment created successfully! Deployment slug: {deployment.slug}"
)
progress.log("Uploading deployment...")
_upload_deployment(deployment.id, archive_path)
progress.log("Deployment uploaded successfully!")
toolkit.print_line()
if not skip_wait:
logger.debug("Waiting for deployment to complete")
_wait_for_deployment(toolkit, app.id, deployment=deployment)
else:
logger.debug("Skipping deployment wait as requested")
toolkit.print(
f"Check the status of your deployment at [link={deployment.dashboard_url}]{deployment.dashboard_url}[/link]"
)

View file

@ -0,0 +1,249 @@
import logging
from pathlib import Path
from typing import Any, List, Union
import typer
from pydantic import BaseModel
from typing_extensions import Annotated
from fastapi_cloud_cli.utils.api import APIClient
from fastapi_cloud_cli.utils.apps import get_app_config
from fastapi_cloud_cli.utils.auth import is_logged_in
from fastapi_cloud_cli.utils.cli import get_rich_toolkit, handle_http_errors
from fastapi_cloud_cli.utils.env import validate_environment_variable_name
from fastapi_cloud_cli.utils.pydantic_compat import model_validate
logger = logging.getLogger(__name__)
class EnvironmentVariable(BaseModel):
name: str
value: str
class EnvironmentVariableResponse(BaseModel):
data: List[EnvironmentVariable]
def _get_environment_variables(app_id: str) -> EnvironmentVariableResponse:
with APIClient() as client:
response = client.get(f"/apps/{app_id}/environment-variables/")
response.raise_for_status()
return model_validate(EnvironmentVariableResponse, response.json())
def _delete_environment_variable(app_id: str, name: str) -> bool:
with APIClient() as client:
response = client.delete(f"/apps/{app_id}/environment-variables/{name}")
if response.status_code == 404:
return False
response.raise_for_status()
return True
def _set_environment_variable(
app_id: str, name: str, value: str, is_secret: bool = False
) -> None:
with APIClient() as client:
response = client.post(
f"/apps/{app_id}/environment-variables/",
json={"name": name, "value": value, "is_secret": is_secret},
)
response.raise_for_status()
env_app = typer.Typer()
@env_app.command()
def list(
path: Annotated[
Union[Path, None],
typer.Argument(
help="A path to the folder containing the app you want to deploy"
),
] = None,
) -> Any:
"""
List the environment variables for the app.
"""
with get_rich_toolkit(minimal=True) as toolkit:
if not is_logged_in():
toolkit.print(
"No credentials found. Use [blue]`fastapi login`[/] to login.",
tag="auth",
)
raise typer.Exit(1)
app_path = path or Path.cwd()
app_config = get_app_config(app_path)
if not app_config:
toolkit.print(
f"No app found in the folder [bold]{app_path}[/].",
)
raise typer.Exit(1)
with toolkit.progress(
"Fetching environment variables...", transient=True
) as progress:
with handle_http_errors(progress):
environment_variables = _get_environment_variables(app_config.app_id)
if not environment_variables.data:
toolkit.print("No environment variables found.")
return
toolkit.print("Environment variables:")
toolkit.print_line()
for env_var in environment_variables.data:
toolkit.print(f"[bold]{env_var.name}[/]")
@env_app.command()
def delete(
name: Union[str, None] = typer.Argument(
None,
help="The name of the environment variable to delete",
),
path: Annotated[
Union[Path, None],
typer.Argument(
help="A path to the folder containing the app you want to deploy"
),
] = None,
) -> Any:
"""
Delete an environment variable from the app.
"""
with get_rich_toolkit(minimal=True) as toolkit:
# TODO: maybe this logic can be extracted to a function
if not is_logged_in():
toolkit.print(
"No credentials found. Use [blue]`fastapi login`[/] to login.",
tag="auth",
)
raise typer.Exit(1)
path_to_deploy = path or Path.cwd()
app_config = get_app_config(path_to_deploy)
if not app_config:
toolkit.print(
f"No app found in the folder [bold]{path_to_deploy}[/].",
)
raise typer.Exit(1)
if not name:
with toolkit.progress(
"Fetching environment variables...", transient=True
) as progress:
with handle_http_errors(progress):
environment_variables = _get_environment_variables(
app_config.app_id
)
if not environment_variables.data:
toolkit.print("No environment variables found.")
return
name = toolkit.ask(
"Select the environment variable to delete:",
options=[
{"name": env_var.name, "value": env_var.name}
for env_var in environment_variables.data
],
)
assert name
else:
if not validate_environment_variable_name(name):
toolkit.print(
f"The environment variable name [bold]{name}[/] is invalid."
)
raise typer.Exit(1)
toolkit.print_line()
with toolkit.progress(
"Deleting environment variable", transient=True
) as progress:
with handle_http_errors(progress):
deleted = _delete_environment_variable(app_config.app_id, name)
if not deleted:
toolkit.print("Environment variable not found.")
raise typer.Exit(1)
toolkit.print(f"Environment variable [bold]{name}[/] deleted.")
@env_app.command()
def set(
name: Union[str, None] = typer.Argument(
None,
help="The name of the environment variable to set",
),
value: Union[str, None] = typer.Argument(
None,
help="The value of the environment variable to set",
),
path: Annotated[
Union[Path, None],
typer.Argument(
help="A path to the folder containing the app you want to deploy"
),
] = None,
) -> Any:
"""
Set an environment variable for the app.
"""
with get_rich_toolkit(minimal=True) as toolkit:
if not is_logged_in():
toolkit.print(
"No credentials found. Use [blue]`fastapi login`[/] to login.",
tag="auth",
)
raise typer.Exit(1)
path_to_deploy = path or Path.cwd()
app_config = get_app_config(path_to_deploy)
if not app_config:
toolkit.print(
f"No app found in the folder [bold]{path_to_deploy}[/].",
)
raise typer.Exit(1)
if not name:
name = toolkit.input("Enter the name of the environment variable to set:")
if not value:
value = toolkit.input(
"Enter the value of the environment variable to set:", password=True
)
with toolkit.progress(
"Setting environment variable", transient=True
) as progress:
assert name is not None
assert value is not None
with handle_http_errors(progress):
_set_environment_variable(app_config.app_id, name, value)
toolkit.print(f"Environment variable [bold]{name}[/] set.")

View file

@ -0,0 +1,125 @@
import logging
import time
from typing import Any
import httpx
import typer
from pydantic import BaseModel
from fastapi_cloud_cli.config import Settings
from fastapi_cloud_cli.utils.api import APIClient
from fastapi_cloud_cli.utils.auth import (
AuthConfig,
get_auth_token,
is_logged_in,
is_token_expired,
write_auth_config,
)
from fastapi_cloud_cli.utils.cli import get_rich_toolkit, handle_http_errors
from fastapi_cloud_cli.utils.pydantic_compat import model_validate_json
logger = logging.getLogger(__name__)
class AuthorizationData(BaseModel):
user_code: str
device_code: str
verification_uri: str
verification_uri_complete: str
interval: int = 5
class TokenResponse(BaseModel):
access_token: str
def _start_device_authorization(
client: httpx.Client,
) -> AuthorizationData:
settings = Settings.get()
response = client.post(
"/login/device/authorization", data={"client_id": settings.client_id}
)
response.raise_for_status()
return model_validate_json(AuthorizationData, response.text)
def _fetch_access_token(client: httpx.Client, device_code: str, interval: int) -> str:
settings = Settings.get()
while True:
response = client.post(
"/login/device/token",
data={
"device_code": device_code,
"client_id": settings.client_id,
"grant_type": "urn:ietf:params:oauth:grant-type:device_code",
},
)
if response.status_code not in (200, 400):
response.raise_for_status()
if response.status_code == 400:
data = response.json()
if data.get("error") != "authorization_pending":
response.raise_for_status()
if response.status_code == 200:
break
time.sleep(interval)
response_data = model_validate_json(TokenResponse, response.text)
return response_data.access_token
def login() -> Any:
"""
Login to FastAPI Cloud. 🚀
"""
token = get_auth_token()
if token is not None and is_token_expired(token):
with get_rich_toolkit(minimal=True) as toolkit:
toolkit.print("Your session has expired. Logging in again...")
toolkit.print_line()
if is_logged_in():
with get_rich_toolkit(minimal=True) as toolkit:
toolkit.print("You are already logged in.")
toolkit.print(
"Run [bold]fastapi cloud logout[/bold] first if you want to switch accounts."
)
return
with get_rich_toolkit() as toolkit, APIClient() as client:
toolkit.print_title("Login to FastAPI Cloud", tag="FastAPI")
toolkit.print_line()
with toolkit.progress("Starting authorization") as progress:
with handle_http_errors(progress):
authorization_data = _start_device_authorization(client)
url = authorization_data.verification_uri_complete
progress.log(f"Opening [link={url}]{url}[/link]")
toolkit.print_line()
with toolkit.progress("Waiting for user to authorize...") as progress:
typer.launch(url)
with handle_http_errors(progress):
access_token = _fetch_access_token(
client, authorization_data.device_code, authorization_data.interval
)
write_auth_config(AuthConfig(access_token=access_token))
progress.log("Now you are logged in! 🚀")

View file

@ -0,0 +1,12 @@
from fastapi_cloud_cli.utils.auth import delete_auth_config
from fastapi_cloud_cli.utils.cli import get_rich_toolkit
def logout() -> None:
"""
Logout from FastAPI Cloud. 🚀
"""
with get_rich_toolkit(minimal=True) as toolkit:
delete_auth_config()
toolkit.print("You are now logged out! 🚀")

View file

@ -0,0 +1,29 @@
import logging
import shutil
from pathlib import Path
from typing import Any
import typer
from fastapi_cloud_cli.utils.cli import get_rich_toolkit
logger = logging.getLogger(__name__)
def unlink() -> Any:
"""
Unlink by deleting the `.fastapicloud` directory.
"""
with get_rich_toolkit(minimal=True) as toolkit:
config_dir = Path.cwd() / ".fastapicloud"
if not config_dir.exists():
toolkit.print(
"No FastAPI Cloud configuration found in the current directory."
)
logger.debug(f"Configuration directory not found: {config_dir}")
raise typer.Exit(1)
shutil.rmtree(config_dir)
toolkit.print("FastAPI Cloud configuration has been unlinked successfully! 🚀")
logger.debug(f"Deleted configuration directory: {config_dir}")

View file

@ -0,0 +1,27 @@
import logging
from typing import Any
from rich import print
from rich_toolkit.progress import Progress
from fastapi_cloud_cli.utils.api import APIClient
from fastapi_cloud_cli.utils.auth import is_logged_in
from fastapi_cloud_cli.utils.cli import handle_http_errors
logger = logging.getLogger(__name__)
def whoami() -> Any:
if not is_logged_in():
print("No credentials found. Use [blue]`fastapi login`[/] to login.")
return
with APIClient() as client:
with Progress(title="⚡ Fetching profile", transient=True) as progress:
with handle_http_errors(progress, message=""):
response = client.get("/users/me")
response.raise_for_status()
data = response.json()
print(f"⚡ [bold]{data['email']}[/bold]")