Skip to content

Commit

Permalink
Add ability to make changes to a gist through the object
Browse files Browse the repository at this point in the history
by changing attributes and then calling the edit method with no args
  • Loading branch information
WitherredAway committed Sep 15, 2022
1 parent 5364d46 commit 89dd834
Show file tree
Hide file tree
Showing 3 changed files with 47 additions and 20 deletions.
2 changes: 1 addition & 1 deletion gists/client.py
Original file line number Diff line number Diff line change
Expand Up @@ -160,8 +160,8 @@ async def edit_gist(
self,
gist_id: str,
*,
description: Optional[str] = None,
files: Optional[typing.List[File]] = None,
description: str = None,
) -> typing.Dict:
"""Edit the gist associated with the provided gist id, and return the edited data"""

Expand Down
6 changes: 6 additions & 0 deletions gists/exceptions.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,12 @@ class ClientException(GistsException):
pass


class GistException(GistsException):
"""Exception that is raised when an operation in the Client class fails"""

pass


class HTTPException(ClientException):
"""Exception that is raised for HTTP request related failures"""

Expand Down
59 changes: 40 additions & 19 deletions gists/gist.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,10 @@
import typing
from typing import Optional
import datetime
from functools import cached_property

from .file import File
from .exceptions import GistException
from .constants import TIME_FORMAT


Expand All @@ -23,8 +25,10 @@ class Gist:
"comments_url",
"commits_url",
"_created_at",
"_description",
"description",
"_files",
"files",
"forks",
"forks_url",
"git_pull_url",
Expand Down Expand Up @@ -73,7 +77,7 @@ def _update_attrs(self, data: typing.Dict):
self.comments_url: str = data.get("comments_url", None)
self.commits_url: str = data.get("commits_url", None)
self._created_at: str = data.get("created_at", None)
self.description: str = data.get("description", None)
self._description: str = data.get("description", None)
self._files: typing.Dict = data.get("files", None)
self.forks: typing.List = data.get("forks", None) # TODO Fork object
self.forks_url: str = data.get("forks_url", None)
Expand All @@ -90,7 +94,11 @@ def _update_attrs(self, data: typing.Dict):
self.api_url: str = data.get("url", None)
self.user: None = data.get("user", None)

def _get_dt_obj(self, time: str) -> datetime.datetime:
self.description: str = self._description
self.files: typing.List[File] = File.from_dict(self._files)

@staticmethod
def _get_dt_obj(time: str) -> datetime.datetime:
"""Internal method to convert string datetime format to datetime object"""
time = time + " +0000" # Tells datetime that the timezone is UTC
dt_obj: datetime.datetime = datetime.datetime.strptime(time, TIME_FORMAT)
Expand All @@ -104,19 +112,6 @@ def created_at(self) -> datetime.datetime:
def created_at(self, value: str):
self._created_at = value

@property
def files(self) -> typing.List[File]:
file_objs: typing.List[File] = File.from_dict(self._files)
return file_objs

@files.setter
def files(self, files: typing.List[File]):
files_dict = {}
for file in files:
files_dict.update(file.to_dict())

self._files = files_dict

@property
def updated_at(self) -> datetime.datetime:
return self._get_dt_obj(self._updated_at)
Expand All @@ -132,13 +127,39 @@ async def update(self):
self._update_attrs(updated_gist_data)

async def edit(
self, *, files: Optional[typing.List[File]] = None, description: str = None
self,
*,
description: Optional[str] = None,
files: Optional[typing.List[File]] = None
):
"""Edit the gist associated with the Gist object, then update the Gist object"""

kwargs = {"description": description}
if files:
kwargs["files"] = files
# Since both files and description are optional arguments,
# you can edit a file through Gist.files (e.g. Gist.files[0].content = "Edited content")
# or the description through Gist.description (e.g. Gist.description = "Edited description")
# and then call Gist.edit() without passing any args to sync the gist with the changed values.
if not description:
description = self.description

if not files:
files = self.files

# If there are no changes, the gist will not edit and the Gist object will raise an error
if all(
(
self.description == self._description,
all(
(
file.name == self._files[file.name]["filename"]
and file.content == self._files[file.name]["content"]
for file in self.files
)
),
)
):
raise GistException("No changes found to edit.")

kwargs = {"description": description, "files": files}

edited_gist_data = await self.client.edit_gist(self.id, **kwargs)
self._update_attrs(edited_gist_data)
Expand Down

0 comments on commit 89dd834

Please sign in to comment.