2019-07-13 20:45:48 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
# This file is part of the Calibre-Web (https://github.com/janeczku/calibre-web)
|
|
|
|
# Copyright (C) 2018-2019 OzzieIsaacs, cervinko, jkrehm, bodybybuddha, ok11,
|
|
|
|
# andy29485, idalin, Kyosfonica, wuqi, Kennyl, lemmsh,
|
|
|
|
# falgh1, grunjol, csitko, ytils, xybydy, trasba, vrabe,
|
|
|
|
# ruben-herold, marblepebble, JackED42, SiphonSquirrel,
|
|
|
|
# apetresc, nanu-c, mutschler
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>
|
|
|
|
|
|
|
|
import json
|
|
|
|
from functools import wraps
|
|
|
|
|
|
|
|
from flask import session, request, make_response, abort
|
|
|
|
from flask import Blueprint, flash, redirect, url_for
|
|
|
|
from flask_babel import gettext as _
|
|
|
|
from flask_dance.consumer import oauth_authorized, oauth_error
|
|
|
|
from flask_dance.contrib.github import make_github_blueprint, github
|
|
|
|
from flask_dance.contrib.google import make_google_blueprint, google
|
2021-04-13 19:08:02 +02:00
|
|
|
from oauthlib.oauth2 import TokenExpiredError, InvalidGrantError
|
2024-07-15 18:38:56 +02:00
|
|
|
from .cw_login import login_user, current_user
|
2019-07-13 20:45:48 +02:00
|
|
|
from sqlalchemy.orm.exc import NoResultFound
|
2024-07-15 18:38:56 +02:00
|
|
|
from .usermanagement import user_login_required
|
2019-07-13 20:45:48 +02:00
|
|
|
|
|
|
|
from . import constants, logger, config, app, ub
|
2020-12-12 11:23:17 +01:00
|
|
|
|
2021-03-14 13:28:52 +01:00
|
|
|
try:
|
|
|
|
from .oauth import OAuthBackend, backend_resultcode
|
|
|
|
except NameError:
|
|
|
|
pass
|
2019-07-13 20:45:48 +02:00
|
|
|
|
|
|
|
|
|
|
|
oauth_check = {}
|
2021-03-24 18:45:59 +01:00
|
|
|
oauthblueprints = []
|
2019-07-13 20:45:48 +02:00
|
|
|
oauth = Blueprint('oauth', __name__)
|
|
|
|
log = logger.create()
|
|
|
|
|
|
|
|
|
2019-07-20 20:01:05 +02:00
|
|
|
def oauth_required(f):
|
|
|
|
@wraps(f)
|
|
|
|
def inner(*args, **kwargs):
|
2019-07-21 08:10:23 +02:00
|
|
|
if config.config_login_type == constants.LOGIN_OAUTH:
|
2019-07-20 20:01:05 +02:00
|
|
|
return f(*args, **kwargs)
|
2020-02-08 14:39:46 +01:00
|
|
|
if request.headers.get('X-Requested-With') == 'XMLHttpRequest':
|
2019-07-20 20:01:05 +02:00
|
|
|
data = {'status': 'error', 'message': 'Not Found'}
|
|
|
|
response = make_response(json.dumps(data, ensure_ascii=False))
|
|
|
|
response.headers["Content-Type"] = "application/json; charset=utf-8"
|
|
|
|
return response, 404
|
|
|
|
abort(404)
|
|
|
|
|
2019-07-13 20:45:48 +02:00
|
|
|
return inner
|
|
|
|
|
|
|
|
|
2020-04-15 19:57:33 +02:00
|
|
|
def register_oauth_blueprint(cid, show_name):
|
|
|
|
oauth_check[cid] = show_name
|
2019-07-13 20:45:48 +02:00
|
|
|
|
|
|
|
|
|
|
|
def register_user_with_oauth(user=None):
|
|
|
|
all_oauth = {}
|
2020-04-15 19:57:33 +02:00
|
|
|
for oauth_key in oauth_check.keys():
|
|
|
|
if str(oauth_key) + '_oauth_user_id' in session and session[str(oauth_key) + '_oauth_user_id'] != '':
|
|
|
|
all_oauth[oauth_key] = oauth_check[oauth_key]
|
2019-07-13 20:45:48 +02:00
|
|
|
if len(all_oauth.keys()) == 0:
|
|
|
|
return
|
|
|
|
if user is None:
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Register with %(provider)s", provider=", ".join(list(all_oauth.values()))), category="success")
|
2019-07-13 20:45:48 +02:00
|
|
|
else:
|
2020-04-15 19:57:33 +02:00
|
|
|
for oauth_key in all_oauth.keys():
|
2019-07-13 20:45:48 +02:00
|
|
|
# Find this OAuth token in the database, or create it
|
|
|
|
query = ub.session.query(ub.OAuth).filter_by(
|
2020-04-15 19:57:33 +02:00
|
|
|
provider=oauth_key,
|
|
|
|
provider_user_id=session[str(oauth_key) + "_oauth_user_id"],
|
2019-07-13 20:45:48 +02:00
|
|
|
)
|
|
|
|
try:
|
2020-04-15 19:57:33 +02:00
|
|
|
oauth_key = query.one()
|
|
|
|
oauth_key.user_id = user.id
|
2019-07-13 20:45:48 +02:00
|
|
|
except NoResultFound:
|
|
|
|
# no found, return error
|
|
|
|
return
|
2021-03-21 18:55:02 +01:00
|
|
|
ub.session_commit("User {} with OAuth for provider {} registered".format(user.name, oauth_key))
|
2019-07-13 20:45:48 +02:00
|
|
|
|
|
|
|
|
|
|
|
def logout_oauth_user():
|
2020-04-15 19:57:33 +02:00
|
|
|
for oauth_key in oauth_check.keys():
|
|
|
|
if str(oauth_key) + '_oauth_user_id' in session:
|
|
|
|
session.pop(str(oauth_key) + '_oauth_user_id')
|
|
|
|
|
2019-07-13 20:45:48 +02:00
|
|
|
|
2021-03-15 09:55:59 +01:00
|
|
|
def oauth_update_token(provider_id, token, provider_user_id):
|
|
|
|
session[provider_id + "_oauth_user_id"] = provider_user_id
|
|
|
|
session[provider_id + "_oauth_token"] = token
|
|
|
|
|
|
|
|
# Find this OAuth token in the database, or create it
|
|
|
|
query = ub.session.query(ub.OAuth).filter_by(
|
|
|
|
provider=provider_id,
|
|
|
|
provider_user_id=provider_user_id,
|
|
|
|
)
|
|
|
|
try:
|
|
|
|
oauth_entry = query.one()
|
|
|
|
# update token
|
|
|
|
oauth_entry.token = token
|
|
|
|
except NoResultFound:
|
|
|
|
oauth_entry = ub.OAuth(
|
|
|
|
provider=provider_id,
|
|
|
|
provider_user_id=provider_user_id,
|
|
|
|
token=token,
|
|
|
|
)
|
|
|
|
ub.session.add(oauth_entry)
|
|
|
|
ub.session_commit()
|
|
|
|
|
|
|
|
# Disable Flask-Dance's default behavior for saving the OAuth token
|
|
|
|
# Value differrs depending on flask-dance version
|
|
|
|
return backend_resultcode
|
|
|
|
|
|
|
|
|
|
|
|
def bind_oauth_or_register(provider_id, provider_user_id, redirect_url, provider_name):
|
|
|
|
query = ub.session.query(ub.OAuth).filter_by(
|
|
|
|
provider=provider_id,
|
|
|
|
provider_user_id=provider_user_id,
|
|
|
|
)
|
|
|
|
try:
|
|
|
|
oauth_entry = query.first()
|
|
|
|
# already bind with user, just login
|
|
|
|
if oauth_entry.user:
|
|
|
|
login_user(oauth_entry.user)
|
2023-01-21 15:19:59 +01:00
|
|
|
log.debug("You are now logged in as: '%s'", oauth_entry.user.name)
|
2024-06-18 20:13:26 +02:00
|
|
|
flash(_("Success! You are now logged in as: %(nickname)s", nickname=oauth_entry.user.name),
|
2021-03-15 09:55:59 +01:00
|
|
|
category="success")
|
|
|
|
return redirect(url_for('web.index'))
|
|
|
|
else:
|
|
|
|
# bind to current user
|
|
|
|
if current_user and current_user.is_authenticated:
|
|
|
|
oauth_entry.user = current_user
|
|
|
|
try:
|
|
|
|
ub.session.add(oauth_entry)
|
|
|
|
ub.session.commit()
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Link to %(oauth)s Succeeded", oauth=provider_name), category="success")
|
2021-04-13 19:08:02 +02:00
|
|
|
log.info("Link to {} Succeeded".format(provider_name))
|
2021-03-15 09:55:59 +01:00
|
|
|
return redirect(url_for('web.profile'))
|
2021-04-04 19:40:34 +02:00
|
|
|
except Exception as ex:
|
2022-03-12 17:14:54 +01:00
|
|
|
log.error_or_exception(ex)
|
2021-03-15 09:55:59 +01:00
|
|
|
ub.session.rollback()
|
|
|
|
else:
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Login failed, No User Linked With OAuth Account"), category="error")
|
2021-03-15 09:55:59 +01:00
|
|
|
log.info('Login failed, No User Linked With OAuth Account')
|
|
|
|
return redirect(url_for('web.login'))
|
|
|
|
# return redirect(url_for('web.login'))
|
|
|
|
# if config.config_public_reg:
|
|
|
|
# return redirect(url_for('web.register'))
|
|
|
|
# else:
|
2023-01-21 15:19:59 +01:00
|
|
|
# flash(_("Public registration is not enabled"), category="error")
|
2021-03-15 09:55:59 +01:00
|
|
|
# return redirect(url_for(redirect_url))
|
|
|
|
except (NoResultFound, AttributeError):
|
|
|
|
return redirect(url_for(redirect_url))
|
|
|
|
|
|
|
|
|
|
|
|
def get_oauth_status():
|
|
|
|
status = []
|
|
|
|
query = ub.session.query(ub.OAuth).filter_by(
|
|
|
|
user_id=current_user.id,
|
|
|
|
)
|
|
|
|
try:
|
|
|
|
oauths = query.all()
|
|
|
|
for oauth_entry in oauths:
|
|
|
|
status.append(int(oauth_entry.provider))
|
|
|
|
return status
|
|
|
|
except NoResultFound:
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
def unlink_oauth(provider):
|
|
|
|
if request.host_url + 'me' != request.referrer:
|
|
|
|
pass
|
|
|
|
query = ub.session.query(ub.OAuth).filter_by(
|
|
|
|
provider=provider,
|
|
|
|
user_id=current_user.id,
|
|
|
|
)
|
|
|
|
try:
|
|
|
|
oauth_entry = query.one()
|
|
|
|
if current_user and current_user.is_authenticated:
|
|
|
|
oauth_entry.user = current_user
|
|
|
|
try:
|
|
|
|
ub.session.delete(oauth_entry)
|
|
|
|
ub.session.commit()
|
|
|
|
logout_oauth_user()
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Unlink to %(oauth)s Succeeded", oauth=oauth_check[provider]), category="success")
|
2021-04-13 19:08:02 +02:00
|
|
|
log.info("Unlink to {} Succeeded".format(oauth_check[provider]))
|
2021-04-04 19:40:34 +02:00
|
|
|
except Exception as ex:
|
2022-03-12 17:14:54 +01:00
|
|
|
log.error_or_exception(ex)
|
2021-03-15 09:55:59 +01:00
|
|
|
ub.session.rollback()
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Unlink to %(oauth)s Failed", oauth=oauth_check[provider]), category="error")
|
2021-03-15 09:55:59 +01:00
|
|
|
except NoResultFound:
|
|
|
|
log.warning("oauth %s for user %d not found", provider, current_user.id)
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Not Linked to %(oauth)s", oauth=provider), category="error")
|
2021-03-15 09:55:59 +01:00
|
|
|
return redirect(url_for('web.profile'))
|
|
|
|
|
2024-06-18 20:13:26 +02:00
|
|
|
|
2021-03-15 09:55:59 +01:00
|
|
|
def generate_oauth_blueprints():
|
2019-07-20 20:01:05 +02:00
|
|
|
if not ub.session.query(ub.OAuthProvider).count():
|
2021-01-03 09:53:34 +01:00
|
|
|
for provider in ("github", "google"):
|
|
|
|
oauthProvider = ub.OAuthProvider()
|
|
|
|
oauthProvider.provider_name = provider
|
|
|
|
oauthProvider.active = False
|
|
|
|
ub.session.add(oauthProvider)
|
|
|
|
ub.session_commit("{} Blueprint Created".format(provider))
|
2019-07-20 20:01:05 +02:00
|
|
|
|
2019-07-21 08:10:23 +02:00
|
|
|
oauth_ids = ub.session.query(ub.OAuthProvider).all()
|
2020-04-15 19:57:33 +02:00
|
|
|
ele1 = dict(provider_name='github',
|
|
|
|
id=oauth_ids[0].id,
|
|
|
|
active=oauth_ids[0].active,
|
|
|
|
oauth_client_id=oauth_ids[0].oauth_client_id,
|
|
|
|
scope=None,
|
|
|
|
oauth_client_secret=oauth_ids[0].oauth_client_secret,
|
|
|
|
obtain_link='https://github.com/settings/developers')
|
|
|
|
ele2 = dict(provider_name='google',
|
|
|
|
id=oauth_ids[1].id,
|
|
|
|
active=oauth_ids[1].active,
|
2020-08-26 08:56:56 -07:00
|
|
|
scope=["https://www.googleapis.com/auth/userinfo.email"],
|
2020-04-15 19:57:33 +02:00
|
|
|
oauth_client_id=oauth_ids[1].oauth_client_id,
|
|
|
|
oauth_client_secret=oauth_ids[1].oauth_client_secret,
|
2020-08-26 03:15:39 +01:00
|
|
|
obtain_link='https://console.developers.google.com/apis/credentials')
|
2019-07-20 20:01:05 +02:00
|
|
|
oauthblueprints.append(ele1)
|
|
|
|
oauthblueprints.append(ele2)
|
|
|
|
|
|
|
|
for element in oauthblueprints:
|
2019-07-21 08:10:23 +02:00
|
|
|
if element['provider_name'] == 'github':
|
2019-07-20 20:01:05 +02:00
|
|
|
blueprint_func = make_github_blueprint
|
|
|
|
else:
|
|
|
|
blueprint_func = make_google_blueprint
|
|
|
|
blueprint = blueprint_func(
|
|
|
|
client_id=element['oauth_client_id'],
|
|
|
|
client_secret=element['oauth_client_secret'],
|
|
|
|
redirect_to="oauth."+element['provider_name']+"_login",
|
2020-04-15 19:57:33 +02:00
|
|
|
scope=element['scope']
|
2019-07-20 20:01:05 +02:00
|
|
|
)
|
2020-04-15 19:57:33 +02:00
|
|
|
element['blueprint'] = blueprint
|
2019-07-21 08:10:23 +02:00
|
|
|
element['blueprint'].backend = OAuthBackend(ub.OAuth, ub.session, str(element['id']),
|
|
|
|
user=current_user, user_required=True)
|
2020-04-16 17:58:42 +02:00
|
|
|
app.register_blueprint(blueprint, url_prefix="/login")
|
2019-07-20 20:01:05 +02:00
|
|
|
if element['active']:
|
2019-07-21 08:10:23 +02:00
|
|
|
register_oauth_blueprint(element['id'], element['provider_name'])
|
2021-03-15 09:55:59 +01:00
|
|
|
return oauthblueprints
|
|
|
|
|
2019-07-13 20:45:48 +02:00
|
|
|
|
2021-03-15 09:55:59 +01:00
|
|
|
if ub.oauth_support:
|
|
|
|
oauthblueprints = generate_oauth_blueprints()
|
2019-07-13 20:45:48 +02:00
|
|
|
|
2019-07-20 20:01:05 +02:00
|
|
|
@oauth_authorized.connect_via(oauthblueprints[0]['blueprint'])
|
2019-07-13 20:45:48 +02:00
|
|
|
def github_logged_in(blueprint, token):
|
|
|
|
if not token:
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Failed to log in with GitHub."), category="error")
|
2021-04-13 19:08:02 +02:00
|
|
|
log.error("Failed to log in with GitHub")
|
2019-07-13 20:45:48 +02:00
|
|
|
return False
|
|
|
|
|
|
|
|
resp = blueprint.session.get("/user")
|
|
|
|
if not resp.ok:
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Failed to fetch user info from GitHub."), category="error")
|
2021-04-13 19:08:02 +02:00
|
|
|
log.error("Failed to fetch user info from GitHub")
|
2019-07-13 20:45:48 +02:00
|
|
|
return False
|
|
|
|
|
|
|
|
github_info = resp.json()
|
|
|
|
github_user_id = str(github_info["id"])
|
2019-07-21 08:10:23 +02:00
|
|
|
return oauth_update_token(str(oauthblueprints[0]['id']), token, github_user_id)
|
2019-07-13 20:45:48 +02:00
|
|
|
|
|
|
|
|
2019-07-20 20:01:05 +02:00
|
|
|
@oauth_authorized.connect_via(oauthblueprints[1]['blueprint'])
|
2019-07-13 20:45:48 +02:00
|
|
|
def google_logged_in(blueprint, token):
|
|
|
|
if not token:
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Failed to log in with Google."), category="error")
|
2021-04-13 19:08:02 +02:00
|
|
|
log.error("Failed to log in with Google")
|
2019-07-13 20:45:48 +02:00
|
|
|
return False
|
|
|
|
|
|
|
|
resp = blueprint.session.get("/oauth2/v2/userinfo")
|
|
|
|
if not resp.ok:
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Failed to fetch user info from Google."), category="error")
|
2021-04-13 19:08:02 +02:00
|
|
|
log.error("Failed to fetch user info from Google")
|
2019-07-13 20:45:48 +02:00
|
|
|
return False
|
|
|
|
|
|
|
|
google_info = resp.json()
|
|
|
|
google_user_id = str(google_info["id"])
|
2019-07-21 08:10:23 +02:00
|
|
|
return oauth_update_token(str(oauthblueprints[1]['id']), token, google_user_id)
|
2019-07-13 20:45:48 +02:00
|
|
|
|
|
|
|
|
2024-06-18 20:13:26 +02:00
|
|
|
|
2019-07-13 20:45:48 +02:00
|
|
|
# notify on OAuth provider error
|
2019-07-20 20:01:05 +02:00
|
|
|
@oauth_error.connect_via(oauthblueprints[0]['blueprint'])
|
2019-07-13 20:45:48 +02:00
|
|
|
def github_error(blueprint, error, error_description=None, error_uri=None):
|
|
|
|
msg = (
|
2023-01-21 15:23:18 +01:00
|
|
|
"OAuth error from {name}! "
|
|
|
|
"error={error} description={description} uri={uri}"
|
2019-07-13 20:45:48 +02:00
|
|
|
).format(
|
|
|
|
name=blueprint.name,
|
|
|
|
error=error,
|
|
|
|
description=error_description,
|
|
|
|
uri=error_uri,
|
2020-04-15 19:57:33 +02:00
|
|
|
) # ToDo: Translate
|
2019-07-13 20:45:48 +02:00
|
|
|
flash(msg, category="error")
|
|
|
|
|
2019-07-20 20:01:05 +02:00
|
|
|
@oauth_error.connect_via(oauthblueprints[1]['blueprint'])
|
2019-07-13 20:45:48 +02:00
|
|
|
def google_error(blueprint, error, error_description=None, error_uri=None):
|
|
|
|
msg = (
|
2023-01-21 15:23:18 +01:00
|
|
|
"OAuth error from {name}! "
|
|
|
|
"error={error} description={description} uri={uri}"
|
2019-07-13 20:45:48 +02:00
|
|
|
).format(
|
|
|
|
name=blueprint.name,
|
|
|
|
error=error,
|
|
|
|
description=error_description,
|
|
|
|
uri=error_uri,
|
2020-04-15 19:57:33 +02:00
|
|
|
) # ToDo: Translate
|
2019-07-13 20:45:48 +02:00
|
|
|
flash(msg, category="error")
|
|
|
|
|
|
|
|
|
2021-03-21 08:19:54 +01:00
|
|
|
@oauth.route('/link/github')
|
|
|
|
@oauth_required
|
|
|
|
def github_login():
|
|
|
|
if not github.authorized:
|
|
|
|
return redirect(url_for('github.login'))
|
2021-04-13 19:08:02 +02:00
|
|
|
try:
|
|
|
|
account_info = github.get('/user')
|
|
|
|
if account_info.ok:
|
|
|
|
account_info_json = account_info.json()
|
|
|
|
return bind_oauth_or_register(oauthblueprints[0]['id'], account_info_json['id'], 'github.login', 'github')
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("GitHub Oauth error, please retry later."), category="error")
|
2021-04-13 19:08:02 +02:00
|
|
|
log.error("GitHub Oauth error, please retry later")
|
|
|
|
except (InvalidGrantError, TokenExpiredError) as e:
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("GitHub Oauth error: {}").format(e), category="error")
|
2021-04-13 19:08:02 +02:00
|
|
|
log.error(e)
|
2021-03-21 08:19:54 +01:00
|
|
|
return redirect(url_for('web.login'))
|
|
|
|
|
|
|
|
|
|
|
|
@oauth.route('/unlink/github', methods=["GET"])
|
2024-07-14 16:24:07 +02:00
|
|
|
@user_login_required
|
2021-03-21 08:19:54 +01:00
|
|
|
def github_login_unlink():
|
|
|
|
return unlink_oauth(oauthblueprints[0]['id'])
|
|
|
|
|
|
|
|
|
|
|
|
@oauth.route('/link/google')
|
|
|
|
@oauth_required
|
|
|
|
def google_login():
|
|
|
|
if not google.authorized:
|
|
|
|
return redirect(url_for("google.login"))
|
2021-04-13 19:08:02 +02:00
|
|
|
try:
|
|
|
|
resp = google.get("/oauth2/v2/userinfo")
|
|
|
|
if resp.ok:
|
|
|
|
account_info_json = resp.json()
|
|
|
|
return bind_oauth_or_register(oauthblueprints[1]['id'], account_info_json['id'], 'google.login', 'google')
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Google Oauth error, please retry later."), category="error")
|
2021-04-13 19:08:02 +02:00
|
|
|
log.error("Google Oauth error, please retry later")
|
|
|
|
except (InvalidGrantError, TokenExpiredError) as e:
|
2023-01-21 15:19:59 +01:00
|
|
|
flash(_("Google Oauth error: {}").format(e), category="error")
|
2021-04-13 19:08:02 +02:00
|
|
|
log.error(e)
|
2021-03-21 08:19:54 +01:00
|
|
|
return redirect(url_for('web.login'))
|
|
|
|
|
|
|
|
|
|
|
|
@oauth.route('/unlink/google', methods=["GET"])
|
2024-07-14 16:24:07 +02:00
|
|
|
@user_login_required
|
2021-03-21 08:19:54 +01:00
|
|
|
def google_login_unlink():
|
|
|
|
return unlink_oauth(oauthblueprints[1]['id'])
|