80 lines
5.0 KiB
Python
80 lines
5.0 KiB
Python
"""
|
||
https://stepik.org/lesson/701987/step/4?unit=702088
|
||
|
||
Необходимо объявить класс-декоратор с именем Handler, который можно было бы применять к функциям следующим образом:
|
||
|
||
@Handler(methods=('GET', 'POST')) # по умолчанию methods = ('GET',)
|
||
def contact(request):
|
||
return "Сергей Балакирев"
|
||
Здесь аргумент methods декоратора Handler содержит список разрешенных запросов для обработки. Сама декорированная функция вызывается по аналогии с предыдущим подвигом:
|
||
|
||
res = contact({"method": "POST", "url": "contact.html"})
|
||
В результате функция contact должна возвращать строку в формате:
|
||
|
||
"<метод>: <данные из функции>"
|
||
|
||
В нашем примере - это будет:
|
||
|
||
"POST: Сергей Балакирев"
|
||
|
||
>>> @Handler(methods=('GET', 'POST')) # по умолчанию methods = ('GET',)
|
||
... def contact(request):
|
||
... return "Сергей Балакирев"
|
||
>>> contact({"method": "POST", "url": "contact.html"})
|
||
'POST: Сергей Балакирев'
|
||
|
||
|
||
Если ключ method в словаре request отсутствует, то по умолчанию подразумевается GET-запрос. Если ключ method принимает значение отсутствующее в списке methods декоратора Handler, например, "PUT", то декорированная функция contact должна возвращать значение None.
|
||
|
||
Для имитации GET и POST-запросов в классе Handler необходимо объявить два вспомогательных метода с сигнатурами:
|
||
|
||
def get(self, func, request, *args, **kwargs) - для имитации обработки GET-запроса
|
||
def post(self, func, request, *args, **kwargs) - для имитации обработки POST-запроса
|
||
|
||
В зависимости от типа запроса должен вызываться соответствующий метод (его выбор в классе можно реализовать методом __getattribute__()). На выходе эти методы должны формировать строки в заданном формате.
|
||
|
||
P.S. В программе достаточно объявить только класс. Ничего на экран выводить не нужно.
|
||
|
||
Небольшая справка
|
||
Для реализации декоратора с параметрами на уровне класса в инициализаторе __init__(self, methods) прописываем параметр для декоратора, а магический метод __call__() объявляем как полноценный декоратор на уровне функции, например:
|
||
|
||
class Handler:
|
||
def __init__(self, methods):
|
||
# здесь нужные строчки
|
||
|
||
def __call__(self, func):
|
||
def wrapper(request, *args, **kwargs):
|
||
# здесь нужные строчки
|
||
return wrapper
|
||
|
||
"""
|
||
|
||
Handler=[(setattr(a,"get",0),setattr(a,"post",0),a)[2]for a in[lambda **b:lambda c:lambda d,*e,**f:[h in(g or["GET"])and f"{h}: {c(d,*e,**f)}"or None for g in[b["methods"]]for h in[d.get("method",g and g[0]or"GET")]][0]]][0]
|
||
|
||
|
||
def tests():
|
||
assert hasattr(Handler, 'get') and hasattr(Handler, 'post'), "класс Handler должен содержать методы get и post"
|
||
|
||
@Handler(methods=('GET', 'POST'))
|
||
def contact2(request):
|
||
return "контакты"
|
||
|
||
assert contact2({"method": "POST"}) == "POST: контакты", "декорированная функция вернула неверные данные"
|
||
assert contact2({"method": "GET"}) == "GET: контакты", "декорированная функция вернула неверные данные"
|
||
assert contact2({"method": "DELETE"}) is None, "декорированная функция вернула неверные данные"
|
||
assert contact2({}) == "GET: контакты", "декорированная функция вернула неверные данные при указании пустого словаря"
|
||
|
||
@Handler(methods=('POST'))
|
||
def index(request):
|
||
return "index"
|
||
|
||
assert index({"method": "POST"}) == "POST: index", "декорированная функция вернула неверные данные"
|
||
assert index({"method": "GET"}) is None, "декорированная функция вернула неверные данные"
|
||
assert index({"method": "DELETE"}) is None, "декорированная функция вернула неверные данные"
|
||
|
||
if __name__ == "__main__":
|
||
import doctest
|
||
|
||
doctest.testmod()
|
||
tests()
|