【问题标题】:django rest-auth social connect to Vkontakte (vk)django rest-auth 社交连接到 Vkontakte (vk)
【发布时间】:2018-09-15 23:01:32
【问题描述】:

请帮助我为提供商 Vkontakte 制作 REST 社交连接。 我正在使用 django-allauth 库 (https://github.com/pennersr/django-allauth)。 还有 django-rest-auth 用于 REST 身份验证 (https://github.com/Tivix/django-rest-auth)。

已经有 VKOAuth2Serializer 可以正常登录并注册。 但是连接不起作用。 这是网址:

url(r'^rest-auth/vk/connect/$', views.VkConnect.as_view(), name='vk_connect'),
url(r'^rest-auth/vk/', views.VkLogin.as_view()),

查看次数:

from allauth.socialaccount.providers.oauth2.client import OAuth2Client
from allauth.socialaccount.providers.vk.views import VKOAuth2Adapter

class VkLogin(CustomSocialLoginView):
    adapter_class = VKOAuth2Adapter
    serializer_class = VKOAuth2Serializer
    client_class = OAuth2Client
    callback_url = 'http://localhost:3000'

class VkConnect(SocialConnectView): 
    adapter_class = VKOAuth2Adapter
    #May be here should be something else?

序列化器:

from allauth.socialaccount.helpers import complete_social_login
from rest_auth.registration.serializers import SocialLoginSerializer
from django.utils.translation import ugettext_lazy as _
from requests.exceptions import HTTPError
from rest_framework import serializers

class VKOAuth2Serializer(SocialLoginSerializer):
    email = serializers.CharField(required=False, allow_blank=True)
    user_id = serializers.CharField(required=False, allow_blank=True)

    def validate(self, attrs):
        view = self.context.get('view')
        request = self._get_request()

        if not view:
            raise serializers.ValidationError(_("View is not defined, pass it as a context variable"))

        adapter_class = getattr(view, 'adapter_class', None)
        if not adapter_class:
            raise serializers.ValidationError(_("Define adapter_class in view"))

        adapter = adapter_class(request)
        app = adapter.get_provider().get_app(request)

        # Case 1: We received the access_token
        if attrs.get('access_token'):
            if not attrs.get('user_id') or not attrs.get('email'):
                raise serializers.ValidationError(_("Incorrect input. email and user_id is required with access_token."))

            access_data = {
                'access_token': attrs.get('access_token'),
                'user_id': attrs.get('user_id'),
                'email': attrs.get('email'),
            }

        # Case 2: We received the authorization code
        elif attrs.get('code'):
            self.callback_url = getattr(view, 'callback_url', None)
            self.client_class = getattr(view, 'client_class', None)

            if not self.callback_url:
                raise serializers.ValidationError(_("Define callback_url in view"))
            if not self.client_class:
                raise serializers.ValidationError(_("Define client_class in view"))

            code = attrs.get('code')

            provider = adapter.get_provider()
            scope = provider.get_scope(request)
            client = self.client_class(
                request,
                app.client_id,
                app.secret,
                adapter.access_token_method,
                adapter.access_token_url,
                self.callback_url,
                scope
            )
            access_data = client.get_access_token(code)
            if attrs.get('email'):
                access_data['email'] = attrs.get('email')
            if not access_data.get('email'):
                raise serializers.ValidationError(_("Incorrect input. Social account must have email, otherwise send it in email field."))
        else:
            raise serializers.ValidationError(_("Incorrect input. access_token or code is required."))

        social_token = adapter.parse_token({'access_token': access_data['access_token']})
        social_token.app = app

        try:
            login = self.get_social_login(adapter, app, social_token, access_data)
            complete_social_login(request, login)
        except HTTPError:
            raise serializers.ValidationError(_('Incorrect value'))

        if not login.is_existing:
            login.lookup()
            login.save(request, connect=True)
        attrs['user'] = login.account.user

        return attrs

这里是注册/登录和正常工作的 url 的视图:

请帮我为 Vkontakte 建立 REST 连接。

项目在这里: https://github.com/taime/imetrics

尝试连接 VK 时出现错误:

Environment:


Request Method: POST
Request URL: http://localhost:8000/rest-auth/vk/connect/

Django Version: 2.0.3
Python Version: 3.5.2
Installed Applications:
['django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
'django_filters',
'rest_framework',
'rest_framework.authtoken',
'rest_auth',
'django.contrib.sites',
'allauth',
'allauth.account',
'rest_auth.registration',
'allauth.socialaccount',
'allauth.socialaccount.providers.facebook',
'allauth.socialaccount.providers.vk',
'allauth.socialaccount.providers.twitter',
'storages',
'core',
'api']
Installed Middleware:
['django.middleware.security.SecurityMiddleware',
'django.contrib.sessions.middleware.SessionMiddleware',
'django.middleware.common.CommonMiddleware',
'django.middleware.csrf.CsrfViewMiddleware',
'django.contrib.auth.middleware.AuthenticationMiddleware',
'django.contrib.messages.middleware.MessageMiddleware',
'django.middleware.clickjacking.XFrameOptionsMiddleware']



Traceback:

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/django/core/handlers/exception.py" in inner
35.             response = get_response(request)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/django/core/handlers/base.py" in _get_response
128.                 response = self.process_exception_by_middleware(e, request)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/django/core/handlers/base.py" in _get_response
126.                 response = wrapped_callback(request, *callback_args, **callback_kwargs)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/django/views/decorators/csrf.py" in wrapped_view
54.         return view_func(*args, **kwargs)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/django/views/generic/base.py" in view
69.             return self.dispatch(request, *args, **kwargs)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/django/utils/decorators.py" in _wrapper
62.             return bound_func(*args, **kwargs)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/django/views/decorators/debug.py" in sensitive_post_parameters_wrapper
76.             return view(request, *args, **kwargs)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/django/utils/decorators.py" in bound_func
58.                 return func.__get__(self, type(self))(*args2, **kwargs2)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/rest_auth/views.py" in dispatch
49.         return super(LoginView, self).dispatch(*args, **kwargs)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/rest_framework/views.py" in dispatch
494.             response = self.handle_exception(exc)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/rest_framework/views.py" in handle_exception
454.             self.raise_uncaught_exception(exc)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/rest_framework/views.py" in dispatch
491.             response = handler(request, *args, **kwargs)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/rest_auth/views.py" in post
93.         self.serializer.is_valid(raise_exception=True)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/rest_framework/serializers.py" in is_valid
236.                 self._validated_data = self.run_validation(self.initial_data)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/rest_framework/serializers.py" in run_validation
438.             value = self.validate(value)

File "/Users/tema/Sites/instametrics/code/core/serializers.py" in validate
73.             login = self.get_social_login(adapter, app, social_token, access_data)

File "/Users/tema/Sites/instametrics/.venv/lib/python3.5/site-packages/rest_auth/registration/serializers.py" in get_social_login
58.         social_login = adapter.complete_login(request, app, token, response=response)

File "/Users/tema/Sites/instametrics/code/allauth/socialaccount/providers/vk/views.py" in complete_login
54.         extra_data = resp.json()['response'][0]

Exception Type: KeyError at /rest-auth/vk/connect/
Exception Value: 'response'

【问题讨论】:

  • 能否请您发布您从连接视图中得到的错误?
  • KeyError at /rest-auth/vk/connect/ 'response' 请求方法:POST 请求 URL:instametrics.gyds.ru/rest-auth/vk/connect Django 版本:2.0.3 异常类型:KeyError 异常值:'response'异常位置: /code/allauth/socialaccount/providers/vk/views.py in complete_login, line 54
  • 我已经检查过了。当我尝试连接 VK 时,它会创建新帐户而不是将其连接到当前...
  • 您是否为两个帐户使用相同的电子邮件?
  • 我使用不同的电子邮件...但它可以连接 twitter 和 facebook(它也有不同的电子邮件)。只有 VK 创建新帐户而不是连接。是否有可能使 vk 的工作方式与 twitter 和 facebook 一样?

标签: django django-allauth vk django-rest-auth


【解决方案1】:

由于 VK 需要自定义提供程序逻辑,您需要使用序列化程序,其逻辑在 VKOAuth2Serializer 中定义,但社交登录状态进程设置为 connect。您可以通过使用SocialConnectMixin 创建VKOAuth2Serializer 的子类来实现它。

因此,您的新序列化程序和连接视图将如下所示:

序列化器:

from rest_auth.registration.serializers import SocialConnectMixin


class VKOAuth2ConnectSerializer(SocialConnectMixin, VKOAuth2Serializer):
    pass

查看:

class VkConnect(SocialConnectView): 
    adapter_class = VKOAuth2Adapter
    serializer_class = VKOAuth2ConnectSerializer

【讨论】:

  • 谢谢,我会检查它,稍后再回来
【解决方案2】:

现在需要 VK API 版本字段。您可以从 github repo 重写 VKOAuth2Adapter 类 https://github.com/pennersr/django-allauth/blob/master/allauth/socialaccount/providers/vk/views.py

【讨论】:

  • 谢谢,我知道了,我已经有了 vk api 版本字段。正如我所说,REST 登录工作正常。我也需要 REST 连接才能正常工作。普通连接(不是 REST)也适用于 vk。现在需要通过 REST 使其工作......
猜你喜欢
  • 1970-01-01
  • 2021-10-17
  • 2019-12-30
  • 2016-03-30
  • 2019-05-30
  • 2014-05-13
  • 1970-01-01
  • 2019-02-14
  • 2020-03-05
相关资源
最近更新 更多