我用Django和React Redux TypeScript制作了一个原创应用程序(TwitterAPI),第一部分

首先,让我介绍一下自己。

初次见面。

这是我在Qiita的首次投稿。

我叫Kensho。
自2019年开始认真学习编程,并且有网站制作自由职业者的经验,之后在IT公司实习期间一直自学网页工程师的知识。
预计从2021年4月开始作为一名上市公司的新毕业生WEB工程师工作。

我开始使用Qiita有三个原因:
1. 我希望通过分享自己的困难经历来帮助其他学习者。
2. 我可以将学习成果进行输出。
3. 我希望提升自己的影响力。

由于这是我第一次投稿,而且我还是一个不太熟练的工程师,可能有些地方表达不够清楚,或者知识不足,但我希望这些信息对尽可能多的编程学习者有所帮助。那么,让我们开始吧!

这篇关于原创应用开发的文章

我使用Django、React Redux和TypeScript制作了一个原创应用(TwitterAPI)第1部分。

我使用Django、React Redux和TypeScript创建了一个原创应用程序(TwitterAPI)第二部分。

我使用Django、React Redux和TypeScript来制作了一个原创应用程序(Twitter API)第三部分。

我使用Django、React Redux和TypeScript来制作了一个原创应用程序(Twitter API),这是第4部分。

原始应用程序简介

git 下载链接:
https://github.com/kenshow-blog/twitter

スクリーンショット 2021-02-21 16.05.10.png
スクリーンショット 2021-02-21 16.10.12.png
スクリーンショット 2021-02-21 16.10.48.png

请在此输入框中输入您在Twitter上的屏幕名称。
这次我们将输入“tokimeki_cafe”用户的屏幕名称。
(https://twitter.com/tokimeki_cafe)

当加载开始后,请在后台开始收集该账户的图像。

スクリーンショット 2021-02-21 16.16.42.png
スクリーンショット 2021-02-21 16.17.06.png

功能细节

请您输入您的邮件地址和密码进行登录。

注册功能… 输入昵称、电子邮件、密码进行注册。

在中国家庭屏幕上,我使用React、Redux Toolkit和TypeScript进行了实现。未来计划创建除了“Media”按钮外的其他按钮,并使用TwitterAPI实现各种功能的应用程序。

在媒体画面中,主要使用material-ui进行开发,包括输入框、按钮等。此外,当按下收集屏幕名称旁边的”DEL”按钮时,会弹出一个对话框显示”是否确定要删除?”,如果点击”Yes”,则所有已收集的图像都将被删除.

媒体个别画面…我们也使用material-ui的Grid实现了一个横3列的美观图片列表页面。按下”DEL”按钮可以删除指定的图片。

使用语言、库

python3
django(django-rest-frame-work)
react
redux-toolkit
TypeScript

致使开发的经过

有三个原因:
– 因为想通过采用现代的开发方式制作原创应用来提高自己作为Web工程师的技能水平。
– 因为想尝试将API集成到应用开发中。
– 因为我对机器学习等方面很感兴趣,特别是对图像识别技术的学习,所以我认为在这里收集的图像会对我有帮助。

应用程序的结构

Django的后端目录结构图。

twitter
├── auth_api #ユーザーのアカウントを管理するところ
│   ├── __init__.py
│   ├── __pycache__
│   ├── admin.py
│   ├── apps.py
│   ├── migrations
│   ├── models.py
│   ├── serializers.py
│   ├── tests.py
│   ├── urls.py
│   └── views.py
├── db.sqlite3
├── manage.py
├── media
│   └── media
│       ├── 収集した画像が入る
│
├── media_api #TwitterAPIを叩いて画像を収集し、保存をしてくれるところ
│   ├── __init__.py
│   ├── __pycache__
│   ├── admin.py
│   ├── apps.py
│   ├── config.py #APIキーを格納しておくファイル
│   ├── migrations
│   ├── models.py 
│   ├── serializers.py
│   ├── tests.py
│   ├── urls.py
│   └── views.py
└── twitter
    ├── __init__.py
    ├── __pycache__
    ├── asgi.py
    ├── settings.py
    ├── urls.py
    └── wsgi.py

模特儿 (Mó tè ér)

用户模型
– 用户名
– 电子邮件
– 密码

形象发表模型

为了将用户和屏幕名称关联起来而创建
– scrName(屏幕名称)
– userPost(外键。与用户模型关联)
– created_on(创建日期)

图像模型
保存收集到的图像,并创建与图像发布模型的关联
– userImg(外键。与用户模型关联)
– imgs(收集到的图像)
– imgPost(外键。与收集到的图像和图像发布关联)
– created_on(创建日期)

请参考以下文章:https://qiita.com/xKxAxKx/items/60e8fb93d6bbeebcf065

用户模型代码↓


from django.db import models
from django.contrib.auth.models import BaseUserManager, AbstractBaseUser, _user_has_perm
from django.core import validators
from django.utils.translation import ugettext_lazy as _
from django.utils import timezone


class AccountManager(BaseUserManager):
    def create_user(self, request_data, **kwargs):
        now = timezone.now()
        if not request_data['email']:
            raise ValueError('Users must have an email address.')


        user = self.model(
            username=request_data['username'],
            email=self.normalize_email(request_data['email']),
            #normalize_emailで受け取ったデータがemailであるかをチェックしている
            is_active=True,
            last_login=now,
            date_joined=now,
        )

        user.set_password(request_data['password'])
        user.save(using=self._db)
        return user

    def create_superuser(self, username, email, password, **extra_fields):
        request_data = {
            'username': username,
            'email': email,
            'password': password
        }

        user = self.create_user(request_data)
        user.is_active = True
        user.is_staff = True
        user.is_admin = True
        user.save(using=self._db)
        return user


class Account(AbstractBaseUser):
    username = models.CharField(_('username'), max_length=30, unique=True)
    first_name = models.CharField(
        _('first name'), max_length=30, blank=True)
    last_name = models.CharField(_('last name'), max_length=30, blank=True)
    email = models.EmailField(
        verbose_name='email address', max_length=255, unique=True)
    is_active = models.BooleanField(default=True)
    is_staff = models.BooleanField(default=False)
    is_admin = models.BooleanField(default=False)
    date_joined = models.DateTimeField(
        _('date joined'), default=timezone.now)

    objects = AccountManager()

    USERNAME_FIELD = 'email'
    REQUIRED_FIELDS = ['username']

    def user_has_perm(user, perm, obj):
        return _user_has_perm(user, perm, obj)

    def has_perm(self, perm, obj=None):
        return _user_has_perm(self, perm, obj=obj)

    def has_module_perms(self, app_label):
        return self.is_admin

    def get_short_name(self):
        return self.first_name

    @property
    def is_superuser(self):
        return self.is_admin

    class Meta:
        db_table = 'auth_user'
        swappable = 'AUTH_USER_MODEL'



我们可以通过将”USERNAME_FIELD”更改为”email”来自定义登录的方式,使其变成通过”email”和”密码”进行登录,而不是使用通常的”用户名”和”密码”。

图像发布,图像模型代码↓

from django.db import models
from django.contrib.auth.models import AbstractBaseUser, BaseUserManager, PermissionsMixin
from django.conf import settings

#下記は収集した画像の名前を変更して保存するための関数
def upload_post_path(instance, filename):
    ext = filename.split('.')[-1]
    return '/'.join(['media', str(instance.userImg)+'_'+str(instance.imgPost.scrName)+'_'+str(instance.id)+str(".")+str(ext)])

class ImagePost(models.Model):
    scrName = models.CharField(max_length=100)
    userPost = models.ForeignKey(
        settings.AUTH_USER_MODEL, related_name='userPost',
        on_delete=models.CASCADE
    )
    created_on = models.DateTimeField(auto_now_add=True)

    def __str__(self):
        return self.scrName



class Images(models.Model):
    id = models.AutoField(primary_key=True)
    userImg = models.ForeignKey(
        settings.AUTH_USER_MODEL, related_name='userImg',
        on_delete=models.CASCADE
    )

    imgs = models.ImageField(blank=True, null=True,upload_to=upload_post_path)
    imgPost = models.ForeignKey(
        ImagePost, on_delete=models.CASCADE
    )


    def __str__(self):
        return str(self.userImg)+'_'+str(self.imgPost)+'_'+str(self.id)


关于图像帖子和Images模型的思考

我预计到了想要获取的用户屏幕名称,由于需要收集的图像是多个,所以会出现一对多的关系。因此,在编写代码阶段,我认为只需创建两个模型,一个用于用户的屏幕名称,另一个用于收集图像。

设置

关于认证,我们选择JWT作为认证的方式。

选择原因如下:
1. 安全性:JWT中包含了数字签名,即使被篡改也可以进行校验。
2. 实现的便捷性:可以轻松实现安全的令牌生成。
3. 管理的便捷性:JWT由可包含在URL中的字符构成,便于在HTTP请求中处理。

我之前听说这些往往也在实际工作中被采用,所以我决定这次试试看。
参考文章↓
https://qiita.com/Syoitu/items/bc32b5e1c2fa891c2f96


import django
from pathlib import Path
import os
# 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.1/howto/deployment/checklist/

# SECURITY WARNING: keep the secret key used in production secret!
SECRET_KEY = 'シークレットキー'

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

ALLOWED_HOSTS = []


# Application definition

INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
     #追記したもの↓
    'corsheaders',
    'rest_framework',
    'media_api.apps.MediaApiConfig',#media_api
    'auth_api.apps.AuthApiConfig',#auth_api
    'djoser',
]

MIDDLEWARE = [
    'corsheaders.middleware.CorsMiddleware',#←追記
    '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',
]

#フロントのreactとのやりとりを可能にするため↓
CORS_ORIGIN_WHITELIST = [
    'http://localhost:3000'
]

#JWT認証
JWT_AUTH = {
    'JWT_VERIFY_EXPIRATION':  False,
    'JWT_AUTH_HEADER_PREFIX': 'JWT',
}


ROOT_URLCONF = 'twitter.urls'

TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [],
        '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',
            ],
        },
    },
]


REST_FRAMEWORK = {
    'DEFAULT_PERMISSION_CLASSES': [
        'rest_framework.permissions.IsAuthenticated',
    ],
    'DEFAULT_AUTHENTICATION_CLASSES': [
        'rest_framework_jwt.authentication.JSONWebTokenAuthentication',
    ],
    'NON_FIELD_ERROS_KEY': 'detail',
    'TEST_REQUEST_DEFAULT_FORMAT': 'json'
}

WSGI_APPLICATION = 'twitter.wsgi.application'


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

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


# Password validation
# https://docs.djangoproject.com/en/3.1/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.1/topics/i18n/

LANGUAGE_CODE = 'en-us'

TIME_ZONE = 'Asia/Tokyo'#←これにすると日本時間をデフォルトにしてくれる

USE_I18N = True

USE_L10N = True

USE_TZ = True


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

AUTH_USER_MODEL = 'auth_api.Account'

STATIC_URL = '/static/'
#↓下記を記述してtwitter/media/media/に画像を保存できるようにしている
MEDIA_ROOT = os.path.join(BASE_DIR, 'media')
MEDIA_URL = '/media/'

通过使用AUTH_USER_MODEL,我们可以让Django识别我们自定义的用户模型,从而引用我们创建的用户API。
参考文章:
https://djangobrothers.com/blogs/referencing_the_user_model/

将项目的URL传递。

from django.conf.urls import url
from django.conf.urls.static import static
from django.conf import settings
from django.urls import path, include
from django.contrib import admin

import django
import os
import sys

sys.path.append('twitterまでの絶対path')
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'twitter.settings')
django.setup()

from rest_framework_jwt.views import obtain_jwt_token
urlpatterns = [
    path('admin/', admin.site.urls),
    path('media_api/', include('media_api.urls')),
    path('login/', obtain_jwt_token),
    path('api/', include('auth_api.urls'))

]

urlpatterns += static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)
from rest_framework_jwt.views import obtain_jwt_token

当我描述了这段代码时,出现了一个奇怪的错误,它没有引用这个库。所以我在导入这个库之前追加了以下内容,问题就解决了。

import django
import os
import sys

sys.path.append('twitterまでのpath')
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'twitter.settings')
django.setup()

原因尚未理解,但明白后会在另一篇文章中发布?
(如果有明白的人,请评论告诉我?男)

我对到目前为止的印象。

构建用户模型真的很费劲哈哈。因为我坚持要将电子邮件和密码作为登录方式,所以不得不自己定制django的用户模型… ?

下次我们将介绍auth_api的序列化器和视图,以及media_api的序列化器和视图的构建!非常感谢您阅读到这里!?‍♂️

bannerAds