微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

带有 JWT 身份验证实现的 Django GraphQL API 仍然允许来自 Postman 的未经身份验证的请求获取数据我该如何解决这个问题?

如何解决带有 JWT 身份验证实现的 Django GraphQL API 仍然允许来自 Postman 的未经身份验证的请求获取数据我该如何解决这个问题?

我已经构建了一个 Django API,它使用 django-graphql-authdjango-graphql-jwt 包来实现身份验证。我遵循了包的文档并让一切正常工作,一切都在我的 Angular UI 中工作。唯一的问题是,即使从 Postman 发出的没有 Authorization 标头的请求,也能够从 graphql API 获取数据。

这是我的 Django 项目的 settings.py

"""
Django settings for myproject project.

Generated by 'django-admin startproject' using Django 3.2.3.

For more information on this file,see
https://docs.djangoproject.com/en/3.2/topics/settings/

For the full list of settings and their values,see
https://docs.djangoproject.com/en/3.2/ref/settings/
"""
from pathlib import Path
import os
import sys

# Build paths inside the project like this: BASE_DIR / 'subdir'.
BASE_DIR = Path(__file__).resolve().parent.parent


# Quick-start development settings - unsuitable for production
# See https://docs.djangoproject.com/en/3.2/howto/deployment/checklist/

# Security WARNING: keep the secret key used in production secret!
SECRET_KEY = 'django-insecure-)3@2sm6lgn_p83_t(l-44hd16ou5-qbk=rso!$b1#$fu*n2^rq'

# Security WARNING: don't run with debug turned on in production!
DEBUG = True

ALLOWED_HOSTS = ["*"]

CORS_ORIGIN_ALLOW_ALL = True

# Application deFinition

INSTALLED_APPS = [
    'corsheaders','django.contrib.admin','django.contrib.auth','django.contrib.contenttypes','django.contrib.sessions','django.contrib.messages','django.contrib.staticfiles','myapp','graphene_django','graphql_jwt.refresh_token.apps.RefreshTokenConfig','graphql_auth','rest_framework','django_filters'
]

GRAPHENE = {
    'SCHEMA': 'myproject.schema.schema','MIDDLEWARE': [
        'graphql_jwt.middleware.JSONWebTokenMiddleware',],}

GRAPHENE_DJANGO_EXTRAS = {
    'DEFAULT_PAGINATION_CLASS': 'graphene_django_extras.paginations.LimitOffsetGraphqlPagination','DEFAULT_PAGE_SIZE': 20,'MAX_PAGE_SIZE': 50,'CACHE_ACTIVE': True,'CACHE_TIMEOUT': 300    # seconds
}


MIDDLEWARE = [
    'corsheaders.middleware.CorsMiddleware','django.middleware.common.CommonMiddleware','django.middleware.security.SecurityMiddleware','django.contrib.sessions.middleware.SessionMiddleware','django.contrib.auth.middleware.AuthenticationMiddleware','django.middleware.csrf.CsrfViewMiddleware','django.contrib.messages.middleware.MessageMiddleware','django.middleware.clickjacking.XFrameOptionsMiddleware','common.utils.UpdateLastActivityMiddleware'
]

AUTHENTICATION_BACKENDS = [
    'graphql_auth.backends.GraphQLAuthBackend','django.contrib.auth.backends.ModelBackend',]

GRAPHQL_JWT = {
    "JWT_ALLOW_ANY_CLASSES": [
        "graphql_auth.mutations.Register","graphql_auth.mutations.VerifyAccount","graphql_auth.mutations.ResendActivationEmail","graphql_auth.mutations.SendPasswordResetEmail","graphql_auth.mutations.PasswordReset","graphql_auth.mutations.ObtainjsONWebToken","graphql_auth.mutations.VerifyToken","graphql_auth.mutations.RefreshToken","graphql_auth.mutations.Revoketoken",'JWT_PAYLOAD_HANDLER': 'common.utils.jwt_payload',"JWT_VERIFY_EXPIRATION": True,"JWT_LONG_RUNNING_REFRESH_TOKEN": True
}

EMAIL_BACKEND = 'django.core.mail.backends.console.EmailBackend'

ROOT_URLconf = 'myproject.urls'

TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates','Dirs': [os.path.join(BASE_DIR,'templates'),'APP_Dirs': True,'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug','django.template.context_processors.request','django.contrib.auth.context_processors.auth','django.contrib.messages.context_processors.messages',},]

Wsgi_APPLICATION = 'myproject.wsgi.application'


# Database
# https://docs.djangoproject.com/en/3.2/ref/settings/#databases

# DATABASES = {
#     'default': {
#         'ENGINE': 'django.db.backends.sqlite3',#         'NAME': BASE_DIR / 'db.sqlite3',#     }
# }

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.postgresql','NAME': 'myprojectdb','USER': 'myprojectadmin','PASSWORD': 'password','HOST': 'db','PORT': '5432',}
}

# Password validation
# https://docs.djangoproject.com/en/3.2/ref/settings/#auth-password-validators

AUTH_PASSWORD_VALIDATORS = [
    {
        'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator',{
        'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator',{
        'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator',{
        'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator',]


# Internationalization
# https://docs.djangoproject.com/en/3.2/topics/i18n/

LANGUAGE_CODE = 'en-us'

TIME_ZONE = 'UTC'

USE_I18N = True

USE_L10N = True

USE_TZ = True


# Static files (CSS,JavaScript,Images)
# https://docs.djangoproject.com/en/3.2/howto/static-files/

STATIC_URL = '/static/'

STATICFILES_Dirs = (
    BASE_DIR / "static",'/var/www/static/',)

# Media files
MEDIA_URL = '/media/'
MEDIA_ROOT = os.path.join(BASE_DIR,"media")

# Default primary key field type
# https://docs.djangoproject.com/en/3.2/ref/settings/#default-auto-field

DEFAULT_AUTO_FIELD = 'django.db.models.BigAutoField'

# This is here because we are using a custom User model
# https://docs.djangoproject.com/en/2.2/topics/auth/customizing/#substituting-a-custom-user-model
AUTH_USER_MODEL = "myapp.User"

urls.py

from django.contrib import admin
from django.urls import include,path
from django.views.decorators.csrf import csrf_exempt
from graphene_django.views import GraphQLView
from django.conf import settings
from django.conf.urls.static import static

urlpatterns = [
    path('',include('myapp.urls')),path('admin/',admin.site.urls),path('graphql/',csrf_exempt(GraphQLView.as_view(graphiql=True))),] + static(settings.MEDIA_URL,document_root=settings.MEDIA_ROOT)

示例查询:-

query users {
  users {
    id
    nickName
    lastLogin
  }
}

如您所见,我已按照所有必要的步骤来适应这两个软件包中的说明。如何防止未经授权的请求访问我的数据?

更新:-

我使用 Django-graphql-extras 进行分页,过滤我的 Graphql api。所以查询都使用该包中的内置方法

然而,这些突变是手动的。

查询文件:-

from graphene_django.types import ObjectType
from .gqTypes import InstitutionType,UserType,GroupType
from graphene_django_extras import DjangoObjectField,DjangoFilterPaginateListField,LimitOffsetGraphqlPagination


class Query(ObjectType):
    institution = DjangoObjectField(
        InstitutionType,description='Single User query')
    user = DjangoObjectField(UserType,description='Single User query')
    group = DjangoObjectField(GroupType,description='Single User query')
    institutions = DjangoFilterPaginateListField(
        InstitutionType,pagination=LimitOffsetGraphqlPagination())
    users = DjangoFilterPaginateListField(
        UserType,pagination=LimitOffsetGraphqlPagination())
    groups = DjangoFilterPaginateListField(
        GroupType,pagination=LimitOffsetGraphqlPagination())

变异代码示例:-

class createuser(graphene.Mutation):
    class Meta:
        description = "Mutation to create a new User"

    class Arguments:
        input = UserInput(required=True)

    ok = graphene.Boolean()
    user = graphene.Field(UserType)

    @staticmethod
    def mutate(root,info,input=None):
        ok = True
        error = ""
        if input.name is None:
            error += "Name is a required field<br />"
        if len(error) > 0:
            raise GraphQLError(error)
        searchField = input.name
        searchField += input.title if input.title is not None else ""
        searchField += input.bio if input.bio is not None else ""
        searchField = searchField.lower()

        user_instance = User(user_id=input.user_id,title=input.title,bio=input.bio,institution_id=input.institution_id,searchField=searchField)
        user_instance.save()
        return createuser(ok=ok,user=user_instance)

解决方法

您应该将 login_required 装饰器添加到您的查询和突变解析器中。像这样:

from graphql_jwt.decorators import login_required


class Query(graphene.ObjectType):
    viewer = graphene.Field(UserType)

    @login_required
    def resolve_viewer(self,info,**kwargs):
        return info.context.user

在你的情况下,把它放在 staticmethod 装饰器之后,就像这样:

@staticmethod
@login_required
def mutate():
    pass

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?