Google App Engine, определите класс предварительной обработки - PullRequest
2 голосов
/ 20 декабря 2010


Я пытаюсь определить класс base request handling, чтобы страницы веб-приложения могли наследовать некоторые базовые методы и переменные, которые в противном случае должны были бы неоднократно задаваться для каждой страницы приложения. Этакий подобный функционал как django preprocessors. Это мой базовый класс, от которого наследуются другие страницы:

class BasePage(webapp.RequestHandler):
    def __init__(self):    
        self.user = users.get_current_user()    
        self.template_values = {
                'user': self.user,       
                'environ': self,   #I don't like the idea of passing the whole environ object to a template

                ##The below three functions cannot be executed during _init_ because of absence of self.request 
                #'openid_providers': self.openid_providers(),  
                #'logout_url': self.get_logout_url(),
                #'request': self.get_request(),
            }           

    ##A sort of similar functionality like render_to_response in django
    def render_template(self, template_name, values = None, *args, **kwargs):
        #PATH is the directory containing the templates
        if values: 
            for value in values:  self.template_values[value] = values[value]
        self.response.out.write(template.render(PATH+template_name, self.template_values, *args, **kwargs))                 

    ##Returns request as the name suggests
    def logout_url(self):
        return users.create_logout_url(self.request.url)

    ##Returns request as the name suggests
    def request(self):
        return request

    ##Returns openid login urls        
    def openid_providers(self):
        #OPENID_POVIDERS  is a list of dictionary 
        for p in OPENID_PROVIDERS:
             p['login_url'] = users.create_login_url(self.request.get('next', '/') , p['name'], p['url'])
        return OPENID_PROVIDERS  

Все работает нормально, за исключением того, что я не могу передать некоторые переменные во время инициализации, поскольку self.request недоступен. Поэтому для обхода проблемы я передал всю переменную self как переменную шаблона.

Есть ли какой-нибудь другой способ предоставления шаблонных переменных (request, logout_url и т. Д.) Шаблонам?

Ответы [ 2 ]

5 голосов
/ 20 декабря 2010

Гораздо более простое решение, чем в bgporter, - выполнить общую настройку в методе initialize webapp.RequestHandler. Вот пример из работы, где мы хотели добавить Django-подобный is_ajax метод к объекту запроса:

class BaseHandler(webapp.RequestHandler):
    def initialize(self, request, response):
        super(BaseHandler, self).initialize(request, response)
        # Add a Django-like is_ajax() method to the request object
        request.is_ajax = lambda: \
            request.environ.get('HTTP_X_REQUESTED_WITH') == 'XMLHttpRequest'

Этот метод вызывается для инициализации каждого обработчика запроса текущими объектами запроса и ответа перед вызовом соответствующих get или post (или любых других) методов.

2 голосов
/ 20 декабря 2010

Я решил эту проблему в своем коде AppEngine с помощью Pattern Method Pattern

В основном базовый класс выглядит так:

class MyBasePage(webapp.RequestHandler):
    def __init__(self):
        # common setup/init stuff here, 
        # omitted for this discussion

    def Setup(self):
        # request handling setup code needed in both GET/POST methods, like
        # checking for user login, getting session cookies, etc.
        # omitted for this discussion

    def get(self, *args):
        self.Setup()
        # call the derived class' 'DoGet' method that actually has 
        # the logic inside it
        self.DoGet(*args)

    def post(self, *args):
        self.Setup()
        # call the derived class' 'DoPost' method 
        self.DoPost(*args)

    def DoGet(self, *args):
        ''' derived classes override this method and 
            put all of their GET logic inside. Base class does nothing.'''
        pass

    def DoPost(self, *args):
        ''' derived classes override this method and 
            put all of their POST logic inside. Base class does nothing.'''
        pass

... тогда вашим производным классам в основном нужно беспокоиться о смелости тех методов DoGet() и DoPost().

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...