Source code for oauth_dropins.handlers

"""Based flow request handlers. Clients should use the individual site modules.

Example usage:

application = webapp2.WSGIApplication([
  ('/oauth_start', facebook.StartHandler.to('/oauth_callback')),
  ('/oauth_callback', facebook.CallbackHandler.to('/done')),
  ('/done', AuthenticatedHandler),
  ...
  ]
"""
import logging
import urllib.parse
import webapp2

from .webutil import handlers
from .webutil import util


[docs]class BaseHandler(webapp2.RequestHandler): """Base request handler class. Provides the to() factory method. Attributes (some may be overridden by subclasses): DEFAULT_SCOPE: string, default OAuth scope(s) to request SCOPE_SEPARATOR: string, used to separate multiple scopes LABEL: string, human-readable label, eg 'Blogger' NAME: string module name; usually same as `__name__.split('.')[-1]` """ DEFAULT_SCOPE = '' SCOPE_SEPARATOR = ',' LABEL = None NAME = None handle_exception = handlers.handle_exception to_path = None @classmethod def to(cls, path, scopes=None): class ToHandler(cls): to_path = path scope = cls.make_scope_str(scopes) return ToHandler
[docs] @classmethod def make_scope_str(cls, extra): """Returns an OAuth scopes query parameter value. Combines DEFAULT_SCOPE and extra. Args: extra: string, sequence of strings, or None """ if not extra: return cls.DEFAULT_SCOPE if not isinstance(extra, str): extra = cls.SCOPE_SEPARATOR.join(extra) return cls.SCOPE_SEPARATOR.join(util.trim_nulls((cls.DEFAULT_SCOPE, extra)))
[docs] def to_url(self, state=None): """Returns a fully qualified callback URL based on to_path. Includes scheme, host, and optional state. """ url = self.request.host_url + self.to_path if state: # unquote first or state will be double-quoted state = urllib.parse.unquote_plus(state) url = util.add_query_params(url, [('state', state)]) return url
[docs] def request_url_with_state(self): """Returns the current request URL, with the state query param if provided. """ state = self.request.get('state') if state: return util.add_query_params(self.request.path_url, [('state', state)]) else: return self.request.path_url
[docs]class StartHandler(BaseHandler): """Base class for starting an OAuth flow. Users should use the to() class method when using this request handler in a WSGI application. See the file docstring for details. If the 'state' query parameter is provided in the request data, it will be returned to the client in the OAuth callback handler. If the 'scope' query parameter is provided, it will be added to the existing OAuth scopes. Alternatively, clients may call redirect_url() and HTTP 302 redirect to it manually, which will start the same OAuth flow. """ def __init__(self, *args, **kwargs): assert self.to_path, 'No `to` URL. Did you forget to use the to() class method in your request handler mapping?' super(StartHandler, self).__init__(*args, **kwargs) def get(self): self.post() def post(self): scopes = set(self.request.params.getall('scope')) if self.scope: scopes.add(self.scope) self.scope = self.SCOPE_SEPARATOR.join(util.trim_nulls(scopes)) # str() is since WSGI middleware chokes on unicode redirect URLs :/ eg: # InvalidResponseError: header values must be str, got 'unicode' (u'...') for 'Location' # https://console.cloud.google.com/errors/CPafw-Gq18CrnwE url = str(self.redirect_url(state=self.request.get('state'))) logging.info('Starting OAuth flow: redirecting to %s', url) self.redirect(url)
[docs] def redirect_url(self, state=None): """Returns the local URL for the OAuth service to redirect back to. oauth-dropin subclasses must implement this. Args: state: string, user-provided value to be returned as a query parameter in the return redirect """ raise NotImplementedError()
[docs] @classmethod def button_html(cls, to_path, form_classes='', form_method='post', form_extra='', image_prefix='', image_file=None, input_style='', scopes='', outer_classes=''): """Returns an HTML string with a login form and button for this site. Args: to_path: string, path or URL for the form to POST to form_classes: string, optional, HTML classes to add to the <form> form_classes: string, optional, HTML classes to add to the outer <div> form_method: string, optional, form action ie HTTP method, eg 'get'; defaults to 'post' form_extra: string, optional, extra HTML to insert inside the <form> before the button scopes: string, optional, OAuth scopes to override site's default(s) image_prefix: string, optional, prefix to add to the beginning of image URL path, eg '/oauth_dropins/' image_file: string, optional, image filename. defaults to [cls.NAME].png input_style: string, optional, inline style to apply to the button <input> Returns: string """ if image_file is None: image_file = '%s_2x.png' % cls.NAME vars = locals() vars.update({ 'label': cls.LABEL, 'image': urllib.parse.urljoin(image_prefix, image_file), }) html = """\ <form method="%(form_method)s" action="%(to_path)s" class="%(form_classes)s"> <nobr> %(form_extra)s <input type="image" height="50" title="%(label)s" class="shadow" src="%(image)s" style="%(input_style)s" /> <input name="scope" type="hidden" value="%(scopes)s"> </nobr> </form> """ % vars if outer_classes: html = '<div class="%s">%s</div>' % (outer_classes, html) return html
[docs]class CallbackHandler(BaseHandler): """Base OAuth callback request handler. Users can use the to() class method when using this request handler in a WSGI application to make it redirect to a given URL path on completion. See the file docstring for details. Alternatively, you can subclass it and implement finish(), which will be called in the OAuth callback request directly, after the user has been authenticated. The auth entity and optional state parameter provided to StartHandler will be passed to finish() or as query parameters to the redirect URL. """
[docs] def finish(self, auth_entity, state=None): """Called when the OAuth flow is complete. Clients may override. Args: auth_entity: a site-specific subclass of models.BaseAuth, or None if the user declined the site's OAuth authorization request. state: the string passed to StartHandler.redirect_url() """ assert self.to_path, 'No `to` URL. Did you forget to use the to() class method in your request handler mapping?' if auth_entity is None: params = [('declined', True)] else: params = [('auth_entity', auth_entity.key.urlsafe().decode()), ('state', state)] try: token = auth_entity.access_token() if isinstance(token, str): params.append(('access_token', token)) elif token: params += [('access_token_key', token[0]), ('access_token_secret', token[1])] except NotImplementedError: logging.info('access_token() not implemented') try: token = auth_entity.refresh_token params.append(('refresh_token', token)) except AttributeError: logging.info('refresh_token not included') url = util.add_query_params(self.to_path, params) logging.info('Finishing OAuth flow: redirecting to %s', url) self.redirect(url)