我用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



请在此输入框中输入您在Twitter上的屏幕名称。
这次我们将输入“tokimeki_cafe”用户的屏幕名称。
(https://twitter.com/tokimeki_cafe)
当加载开始后,请在后台开始收集该账户的图像。


功能细节
请您输入您的邮件地址和密码进行登录。
注册功能… 输入昵称、电子邮件、密码进行注册。
在中国家庭屏幕上,我使用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的序列化器和视图的构建!非常感谢您阅读到这里!?♂️