Как динамически выбирать подкласс из унаследованного класса при определении глобальных методов

Это мой первый настоящий набег на классы Python, поэтому прошу прощения за любое неправильное использование терминов.

Я пытаюсь использовать подход Factory Pattern для динамического выбора подкласса на основе URL, предоставленного пользователем. Вот настройки, которые у меня есть:

import requests
from bs4 import BeautifulSoup as BS

class Templates(object):
    def __init__(self, url):
        self.headers = {'User-Agent': 'Mozilla/5.0 (Windows NT 6.1; WOW64; rv:54.0) Gecko/20100101 Firefox/54.0'}
        self.url = url

    def url(self):
        return self.url

    def response(self):
        return self.response

    def text(self):
        return self.text

    def dom(self):
        return self.dom

    @staticmethod
    def get_template(url):
        headers = {'User-Agent': 'Mozilla/5.0 (Windows NT 6.1; WOW64; rv:54.0) Gecko/20100101 Firefox/54.0'}
        response = requests.get(url, headers=headers)
        text = response.text
        if 'Website by Dealer.com' in text:
            return Dealer(url)
        else:
            return None

class Dealer(Templates):
    def __init__(self, url):
        Templates.__init__(self, url)
        self.new_vehicle_url = "{}/new-inventory/index.htm".format(self.url.rstrip('/'))
        self.used_vehicle_url = "{}/used-inventory/index.htm".format(self.url.rstrip('/'))

Что я хочу сделать, это позвонить Templates.get_template(url) и пусть он вернет соответствующий подкласс. Этот вопрос выглядел очень просто, но я что-то упустил в процессе, потому что я также хочу определить общие методы в моем Templates класс, который требует от меня пройти url в Template,

Как Pythonic настраивает это так, чтобы следующий код работал?

template = Templates().get_template(url) # assume returns Dealer()
template.url
# 'http://www.some-website.com'
template.response.status_code
# 200

1 ответ

Решение

В вашем коде вы используете Templates() но на самом деле пытается создать экземпляр Templates - который терпит неудачу, потому что отсутствует url параметр для в Templates.__init__,

Но вам не нужен экземпляр для вызова статического метода:

template = Templates.get_template(url) # assume returns Dealer()
template.url
# 'http://www.some-website.com'
template.response.status_code
# 200

Я только что удалил () после Templates там.

Другие вопросы по тегам