Browse Source

Better POO using interfaces

For chapter 9
master
Dimitri Merejkowsky 5 years ago
parent
commit
2c16141fde
1 changed files with 133 additions and 0 deletions
  1. +133
    -0
      sources/marvel/marvel_06.py

+ 133
- 0
sources/marvel/marvel_06.py View File

@@ -0,0 +1,133 @@
import abc
import hashlib
import requests
import shutil
import time
import textwrap
import sys


class Authorization:
def __init__(self, filename):
with open(filename, "r") as file:
lines = file.readlines()

if len(lines) != 2:
sys.exit("Incorrect api-keys file")

self.public_key = lines[0].strip()
self.private_key = lines[1].strip()

def generate_params(self):
params = dict()
params["apikey"] = self.public_key
ts = str(time.time())
to_hash = ts + self.private_key + self.public_key
hasher = hashlib.md5()
hasher.update(to_hash.encode())
digest = hasher.hexdigest()
params["ts"] = ts
params["hash"] = digest
return params


class Client():
base_url = "http://gateway.marvel.com/v1/public"

def __init__(self, auth):
self.auth = auth

def make_request(self, query):
params = self.auth.generate_params()
params.update(query.params())
url = Client.base_url + query.path
response = requests.get(url, params=params)
status_code = response.status_code
if status_code != 200:
sys.exit("got status: " + str(status_code))
body = response.json()
result = query.extract(body)
return (query.to_text(result), body["attributionText"])


class Query(metaclass=abc.ABCMeta):
@abc.abstractmethod
def params(self):
pass

@abc.abstractmethod
def path(self):
pass

@abc.abstractmethod
def extract(self, body):
pass

def to_text(self, result):
return result


class CharacterDescription(Query):
path = "/characters"

def __init__(self, name):
self.name = name

def params(self):
return { "name": self.name }

def extract(self, body):
return body["data"]["results"][0]["description"]


class CreatorNumberOfSeries(Query):
path = "/creators"

def __init__(self, first_name, last_name):
self.first_name = first_name
self.last_name = last_name

def params(self):
return { "firstName": self.first_name, "lastName": self.last_name }

def extract(self, body):
return body["data"]["results"][0]["series"]["available"]

def to_text(self, result):
return f"{self.first_name} {self.last_name} worked on {result} series"


class Display:
def __init__(self, width=80):
self.width = width

def display(self, text):
print("\n".join(textwrap.wrap(text, width=self.width)))


def main():

query_type = sys.argv[1]
if query_type == "character-description":
name = sys.argv[2]
query = CharacterDescription(name)
elif query_type == "creator-number-of-series":
first_name = sys.argv[2]
last_name = sys.argv[3]
query = CreatorNumberOfSeries(first_name, last_name)

auth = Authorization("api-keys.txt")
client = Client(auth)
text, attribution = client.make_request(query)

terminal_size = shutil.get_terminal_size()
columns = terminal_size.columns
terminal = Display(width=columns)

terminal.display(text)
terminal.display("---")
terminal.display(attribution)


if __name__ == "__main__":
main()