Source code for zipline.client

"""
Copyright 2023-present fretgfr

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
"""

from __future__ import annotations

import datetime
from types import TracebackType
from typing import TYPE_CHECKING, List, Literal, Optional, Type, Union

import aiohttp

from .enums import NameFormat
from .http import HTTPClient, Route
from .models import File, FileData, Folder, Invite, PartialInvite, ServerVersionInfo, ShortenedURL, UploadResponse, User
from .utils import to_iso_format, utcnow

if TYPE_CHECKING:
    from typing_extensions import Self

__all__ = ("Client",)


[docs] class Client: """A Zipline Client. .. container:: operations .. describe:: async with x: Returns the Client itself. Used to gracefully close the client on exit. .. code-block:: python3 async with zipline.Client(server_url, token) as client: ... """ __slots__ = ("server_url", "http") def __init__(self, server_url: str, token: str) -> None: """Creates a new Client. Parameters ---------- server_url: :class:`str` The URL of the Zipline server. token: :class:`str` Your Zipline token. """ self.server_url = server_url self.http = HTTPClient(server_url, token)
[docs] async def get_version(self) -> ServerVersionInfo: """|coro| Gets the Zipline server version information Returns ------- :class:`~zipline.models.ServerVersionInfo` The version information for the server. """ r = Route("GET", "/api/version") js = await self.http.request(r) return ServerVersionInfo._from_data(js)
[docs] async def create_user(self, *, username: str, password: str, administrator: bool = False) -> User: """|coro| Creates a User. Parameters ---------- username: :class:`str` The username of the user to create. password: :class:`str` The password of the user to create. administrator: Optional[:class:`bool`] Whether this user should be an administrator, by default False Returns ------- User The created User. Raises ------ BadRequest Something went wrong handling the request. Forbidden You are not an administrator and cannot use this method. """ json = {"username": username, "password": password, "administrator": administrator} r = Route("POST", "/api/auth/register") data = await self.http.request(r, json=json) return User._from_data(data, http=self.http)
[docs] async def get_password_protected_image(self, *, id: int, password: str) -> bytes: """|coro| Retrieves the content of a password protected File. Parameters ---------- id: :class:`int` The id of the File to get. password: :class:`str` The password of the File to get. Returns ------- :class:`bytes` The File's content. Raises ------ BadRequest Something went wrong handling the request. NotFound The File could not be found on the server. """ query_params = {"id": id, "password": password} r = Route("GET", "/api/auth/image") return await self.http.request(r, params=query_params)
[docs] async def get_all_invites(self) -> List[Invite]: """|coro| Retrieves all Invites. Returns ------- List[:class:`~zipline.models.Invite`] The invites on the server. Raises ------ BadRequest Something went wrong handling this request. Forbidden You are not an administrator and do not have permission to access this resource. """ r = Route("GET", "/api/auth/invite") js = await self.http.request(r) return [Invite._from_data(data, http=self.http) for data in js]
[docs] async def create_invites(self, *, count: int = 1, expires_at: Optional[datetime.datetime] = None) -> List[PartialInvite]: """|coro| Creates user invites. Parameters ---------- count: :class:`int` The number of invites to create, by default 1 expires_at: Optional[:class:`datetime.datetime`] When the created invite(s) should expire. Defaults to 24 hours from creation. .. versionchanged:: 0.17.0 Added default expiration of 24 hours. Returns ------- List[:class:`~zipline.models.PartialInvite`] The created invites. Raises ------ ZiplineError The server returned the invites in an unexpected format. BadRequest The server could not process the request. Forbidden You are not an administrator and cannot use this method. """ expires_at = expires_at or (utcnow() + datetime.timedelta(hours=24)) data = {"count": count, "expiresAt": f"date={to_iso_format(expires_at)}"} r = Route("POST", "/api/auth/invite") js = await self.http.request(r, json=data) return [PartialInvite._from_data(data) for data in js]
[docs] async def delete_invite(self, code: str, /) -> Invite: """|coro| Deletes an Invite with given code. Parameters ---------- code: :class:`str` The code of the Invite to delete. Returns ------- :class:`~zipline.models.Invite` The deleted Invite Raises ------ Forbidden You are not an administrator and cannot use this method. NotFound No Invite was found with the provided code. """ query_params = {"code": code} r = Route("DELETE", "/api/auth/invite") js = await self.http.request(r, params=query_params) return Invite._from_data(js, http=self.http)
[docs] async def get_all_folders(self, *, with_files: bool = False) -> List[Folder]: """|coro| Returns all Folders Parameters ---------- with_files: Optional[:class:`bool`] Whether the retrieved Folder should contain File information, by default False Returns ------- List[:class:`~zipline.models.Folder`] The retrieved Folders """ query_params = {} if with_files: query_params["files"] = int(with_files) r = Route("GET", "/api/user/folders") js = await self.http.request(r, params=query_params) return [Folder._from_data(data, http=self.http) for data in js]
[docs] async def create_folder(self, name: str, /, *, files: Optional[List[File]] = None) -> Folder: """|coro| Creates a Folder. Parameters ---------- name: :class:`str` The name of the folder to create. files: Optional[List[:class:`~zipline.models.File`]] Files that should be added to the created folder, by default None Returns ------- :class:`~zipline.models.Folder` The created Folder Raises ------ BadRequest The server could not process the request. """ data = {"name": name, "add": [file.id for file in files] if files is not None else None} r = Route("POST", "/api/user/folders/") js = await self.http.request(r, json=data) return Folder._from_data(js, http=self.http)
[docs] async def get_folder(self, id: int, /, *, with_files: bool = False) -> Folder: """|coro| Gets a folder with a given id. Parameters ---------- id: :class:`int` The id of the folder to get. with_files: Optional[:class:`bool`] Whether File information should be retrieved, by default False Returns ------- :class:`~zipline.models.Folder` The requested Folder Raises ------ Forbidden You do not have access to the Folder requested. NotFound A folder with that id could not be found. """ query_params = {} if with_files: query_params["files"] = int(with_files) r = Route("GET", f"/api/user/folders/{id}") js = await self.http.request(r, params=query_params) return Folder._from_data(js, http=self.http)
[docs] async def get_user(self, id: int, /) -> User: """|coro| Returns a User with the given id. Parameters ---------- id: :class:`int` The id of the User to get. Returns ------- :class:`~zipline.models.User` The retrieved User Raises ------ Forbidden You are not an administrator and cannot use this method. NotFound A user with that id could not be found """ r = Route("GET", f"/api/user/{id}") js = await self.http.request(r) return User._from_data(js, http=self.http)
# TODO methods for /api/user/export
[docs] async def get_all_files(self) -> List[File]: """|coro| Gets all Files belonging to your user. Returns ------- List[:class:`~zipline.models.File`] The returned Files """ r = Route("GET", "/api/user/files") js = await self.http.request(r) return [File._from_data(data, http=self.http) for data in js]
[docs] async def delete_all_files(self) -> int: """|coro| Deletes all of your Files Returns ------- :class:`int` The number of removed :class:`~zipline.models.File`'s """ data = {"all": True} r = Route("DELETE", "/api/user/files") js = await self.http.request(r, json=data) return js["count"]
[docs] async def get_recent_files(self, *, amount: int = 4, filter: Literal["all", "media"] = "all") -> List[File]: """|coro| Gets recent files uploaded by you. Parameters ---------- amount: Optional[:class:`int`] The number of results to return. Must be in 1 <= amount <= 50, by default 4 filter: Optional[Literal["all", "media"]] What files to get. "all" to get all Files, "media" to get images/videos/etc., by default "all" Returns ------- List[:class:`~zipline.models.File`] The requested Files. Raises ------ ValueError Amount was not within the specified bounds. """ if amount < 1 or amount > 50: raise ValueError("Amount must be within 1 <= amount <= 50") query_params = {"take": amount, "filter": filter} r = Route("GET", "/api/user/recent") js = await self.http.request(r, params=query_params) return [File._from_data(data, http=self.http) for data in js]
[docs] async def get_all_shortened_urls(self) -> List[ShortenedURL]: """|coro| Retrieves all shortened urls for your user. Returns ------- List[:class:`~zipline.models.ShortenedURL`] The requested shortened urls. """ r = Route("GET", "/api/user/urls") js = await self.http.request(r) return [ShortenedURL._from_data(data, http=self.http) for data in js]
[docs] async def shorten_url( self, original_url: str, *, vanity: Optional[str] = None, max_views: Optional[int] = None, zero_width_space: bool = False, ) -> str: """|coro| Shortens a url Parameters ---------- original_url: :class:`str` The url to shorten vanity: Optional[:class:`str`] A vanity name to use. None to shorten normally, by default None max_views: Optional[:class:`int`] The number of times the url can be used before being deleted. None for unlimited uses, by default None zero_width_space: Optional[:class:`bool`] Whether to incude zero width spaces in the returned url, by default False Returns ------- :class:`str` The shortened url Raises ------ ValueError Invalid value for max views passed. BadRequest The server could not process your request. NotAuthenticated An incorrect authorization header was passed """ if max_views is not None and max_views < 0: raise ValueError("max_views must be greater than or equal to 0") headers = { "Zws": "true" if zero_width_space else "", "Max-Views": str(max_views) if max_views is not None else "", } data = {"url": original_url, "vanity": vanity} r = Route("POST", "/api/shorten") js = await self.http.request(r, headers=headers, json=data) return js["url"]
[docs] async def get_all_users(self) -> List[User]: """|coro| Gets all users. Returns ------- List[:class:`~zipline.models.User`] The retrieved users Raises ------ Forbidden You are not an administrator and cannot use this method. """ r = Route("GET", "/api/users") js = await self.http.request(r) return [User._from_data(data, http=self.http) for data in js]
# TODO /api/stats methods # TODO /api/exif methods
[docs] async def upload_file( self, payload: FileData, *, format: NameFormat = NameFormat.uuid, compression_percent: int = 0, expiry: Optional[datetime.datetime] = None, password: Optional[str] = None, zero_width_space: bool = False, embed: bool = False, max_views: Optional[int] = None, text: bool = False, override_name: Optional[str] = None, original_name: Optional[str] = None, folder: Optional[Union[Folder, int]] = None, ) -> UploadResponse: """|coro| Uploads a File to Zipline Parameters ---------- payload: :class:`~zipline.models.FileData` The file to upload. format: Optional[:class:`~zipline.enums.NameFormat`] The format of the name to assign to the uploaded File, by default :attr:`~zipline.enums.NameFormat`'s `uuid`. compression_percent: Optional[:class:`int`] How compressed should the uploaded File be, by default 0 expiry: Optional[:class:`datetime.datetime`] When the uploaded File should expire, by default None password: Optional[:class:`str`] The password required to view the uploaded File, by default None zero_width_space: Optional[:class:`bool`] Whether to include zero width spaces in the name of the uploaded File, by default False embed: Optional[:class:`bool`] Whether to include embed data for the uploaded File, typically used on Discord, by default False max_views: Optional[:class:`int`] The number of times the uploaded File can be viewed before it is deleted, by default None text: Optional[:class:`bool`] Whether the File is a text file, by default False override_name: Optional[:class:`str`] A name to give the uploaded file. If provided this will override the server generated name, by default None original_name: Optional[:class:`str`] The original_name of the file. None to not preserve this data, by default None folder: Optional[Union[:class:`~zipline.models.Folder`, :class:`int`]] The Folder (or it's ID) to place this upload into automatically .. versionadded:: 0.15.0 Returns ------- :class:`~zipline.models.UploadResponse` The uploaded File Raises ------ ValueError compression_percent was not in 0 <= compression_percent <= 100 ValueError max_views passed was less than 0 ValueError type passed for folder was incorrect BadRequest Server could not process the request ServerError The server responded with a 5xx error code. """ if compression_percent < 0 or compression_percent > 100: raise ValueError("compression_percent must be between 0 and 100") if max_views and max_views < 0: raise ValueError("max_views must be greater than 0") headers = { "Format": format.value, "Image-Compression-Percent": str(compression_percent), "Expires-At": f"date={expiry.isoformat()}" if expiry is not None else "", "Password": password if password is not None else "", "Zws": "true" if zero_width_space else "", "Embed": "true" if embed else "", "Max-Views": str(max_views) if max_views is not None else "", "UploadText": "true" if text else "", "X-Zipline-Filename": override_name if override_name is not None else "", "Original-Name": original_name if original_name is not None else "", } if folder is not None: if not isinstance(folder, (Folder, int)): raise ValueError("folder argument must be a Folder or integer") headers["X-Zipline-Folder"] = str(folder.id) if isinstance(folder, Folder) else str(folder) formdata = aiohttp.FormData() formdata.add_field("file", payload.data, filename=payload.filename, content_type=payload.mimetype) r = Route("POST", "/api/upload") js = await self.http.request(r, headers=headers, data=formdata) return UploadResponse._from_data(js)
[docs] async def close(self) -> None: """Gracefully close the client.""" await self.http.close()
async def __aenter__(self) -> Self: return self async def __aexit__( self, exc_type: Optional[Type[BaseException]], exc: Optional[BaseException], tb: Optional[TracebackType], ) -> None: await self.close()