Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/rg3915/drf-example
Django REST framework examples and tests
https://github.com/rg3915/drf-example
django django-rest-framework
Last synced: 3 months ago
JSON representation
Django REST framework examples and tests
- Host: GitHub
- URL: https://github.com/rg3915/drf-example
- Owner: rg3915
- Created: 2021-08-31T05:57:22.000Z (over 3 years ago)
- Default Branch: main
- Last Pushed: 2022-08-28T20:55:44.000Z (over 2 years ago)
- Last Synced: 2024-04-16T01:27:10.802Z (9 months ago)
- Topics: django, django-rest-framework
- Language: Python
- Homepage:
- Size: 437 KB
- Stars: 15
- Watchers: 3
- Forks: 3
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# drf-example
Django REST framework exemplos e testes.
## Este projeto foi feito com:
* [Python 3.9.6](https://www.python.org/)
* [Django 3.2.7](https://www.djangoproject.com/)
* [Django Rest Framework 3.12.4](https://www.django-rest-framework.org/)## Como rodar o projeto?
* Clone esse repositório.
* Crie um virtualenv com Python 3.
* Ative o virtualenv.
* Instale as dependências.
* Rode as migrações.```
git clone https://github.com/rg3915/drf-example.git
cd drf-example
python -m venv .venv
source .venv/bin/activate
pip install -r requirements.txt
python contrib/env_gen.py
python manage.py migrate
python manage.py createsuperuser --username="admin" --email=""
```## Sumário
* [dr-scaffold](#dr-scaffold)
* [drf-yasg - Yet another Swagger generator](#drf-yasg---yet-another-swagger-generator)
* [djoser](#djoser)
* [Reset de Senha com djoser](#reset-de-senha-com-djoser)
* [Autenticação via JWT com djoser](#autenticação-via-jwt-com-djoser)
* [Django CORS headers](#django-cors-headers)
* [Paginação](#paginação)
* [django-filter](#django-filter)
* [Criando subrota com action](#criando-subrota-com-action)## dr-scaffold
https://github.com/Abdenasser/dr_scaffold
https://www.abdenasser.com/scaffold-django-apis
dr-scaffold é uma lib para criar models e uma API simples em Django REST framework.
### Experimentando e criando um projeto do zero
```
python -m venv .venv
source .venv/bin/activate
pip install dr-scaffold djangorestframework django-extensions python-decouple
```Crie um arquivo `.env`
```
cat << EOF > .env
SECRET_KEY=my-super-secret-key-dev-only
EOF
```Crie um novo projeto.
```
django-admin startproject backend .
```Edite `settings.py`
```python
# settings.pyfrom decouple import config
SECRET_KEY = config('SECRET_KEY')
INSTALLED_APPS = [
...
'rest_framework',
'dr_scaffold',
]
```#### Exemplo 1
Rodando o comando `dr_scaffold`
```
python manage.py dr_scaffold blog Author name:charfieldpython manage.py dr_scaffold blog Post body:textfield author:foreignkey:Author
```Edite `settings.py`
```python
# settings.py
INSTALLED_APPS = [
...
'rest_framework',
'dr_scaffold',
'blog', # <--
]
```Edite `urls.py`
```python
# urls.py
from django.contrib import admin
from django.urls import include, pathurlpatterns = [
path('blog/', include('blog.urls')),
path('admin/', admin.site.urls),
]
``````
python manage.py makemigrations
python manage.py migrate
```#### Exemplo 2
```
python manage.py dr_scaffold product Product title:charfield price:decimalfieldpython manage.py dr_scaffold ecommerce Order nf:charfield
python manage.py dr_scaffold ecommerce OrderItems \
order:foreignkey:Order \
product:foreignkey:Product \
quantity:integerfield \
price:decimalfield
```Edite `settings.py`
```python
INSTALLED_APPS = [
...
'rest_framework',
'dr_scaffold',
'blog', # <--
'product', # <--
'ecommerce', # <--
]```
Edite `urls.py`
```python
...
path('product/', include('product.urls')),
path('ecommerce/', include('ecommerce.urls')),
...
```Não se esqueça de editar `ecommerce/models.py`
```python
from product.models import Product```
```
python manage.py makemigrations
python manage.py migrate
```## drf-yasg - Yet another Swagger generator
https://github.com/axnsan12/drf-yasg/
[drf-yasg](https://github.com/axnsan12/drf-yasg/) é uma outra biblioteca para gerar a documentação com Swagger e reDoc.
```
pip install -U drf-yasgpip freeze | grep drf-yasg >> requirements.txt
```Edite `settings.py`
```python
INSTALLED_APPS = [
...
'django.contrib.staticfiles', # required for serving swagger ui's css/js files
'drf_yasg',
...
]
```Edite `urls.py`
```python
from django.conf.urls import url
from django.contrib import admin
from django.urls import include, path
from drf_yasg import openapi
from drf_yasg.views import get_schema_view
from rest_framework import permissionsschema_view = get_schema_view(
openapi.Info(
title="Snippets API",
default_version='v1',
description="Test description",
terms_of_service="https://www.google.com/policies/terms/",
contact=openapi.Contact(email="[email protected]"),
license=openapi.License(name="BSD License"),
),
public=True,
permission_classes=[permissions.AllowAny],
)urlpatterns = [
path('blog/', include('blog.urls')),
path('product/', include('product.urls')),
path('ecommerce/', include('ecommerce.urls')),
path('admin/', admin.site.urls),
]# swagger
urlpatterns += [
url(r'^swagger(?P\.json|\.yaml)$', schema_view.without_ui(cache_timeout=0), name='schema-json'), # noqa E501
url(r'^swagger/$', schema_view.with_ui('swagger', cache_timeout=0), name='schema-swagger-ui'), # noqa E501
url(r'^redoc/$', schema_view.with_ui('redoc', cache_timeout=0), name='schema-redoc'), # noqa E501
]
```## djoser
https://djoser.readthedocs.io/en/latest/
```
pip install -U djoser
pip freeze | grep djoser >> requirements.txt
```Configure `INSTALLED_APPS`
```python
INSTALLED_APPS = (
'django.contrib.auth',
(...),
'rest_framework',
'rest_framework.authtoken', # <-- rode ./manage.py migrate
'djoser', # <--
(...),
)REST_FRAMEWORK = {
'DEFAULT_AUTHENTICATION_CLASSES': (
'rest_framework.authentication.BasicAuthentication',
'rest_framework.authentication.TokenAuthentication',
),
'DEFAULT_PERMISSION_CLASSES': (
'rest_framework.permissions.IsAuthenticated',
)
}```
Configure `urls.py`
```python
# djoser
urlpatterns += [
path('api/v1/', include('djoser.urls')),
path('api/v1/auth/', include('djoser.urls.authtoken')),
]
``````
python manage.py migrate
python manage.py drf_create_token admintoken d7643a4710c7e19915df7d5e3d82f70cb07998ba # o seu será um novo
```Veja no video como rodar no **Postman** e no **Swagger**.
Exemplos com curl
```
# Cria novo usuário
curl -X POST http://127.0.0.1:8000/api/v1/users/ --data 'username=djoser&password=api127rg'# Login
curl -X POST http://127.0.0.1:8000/api/v1/auth/token/login/ --data 'username=djoser&password=api127rg'# Informações do usuário
curl -X GET http://127.0.0.1:8000/api/v1/users/me/ -H 'Authorization: Token d7643a4710c7e19915df7d5e3d82f70cb07998ba' # o seu será um novo# Logout
curl -X GET http://127.0.0.1:8000/api/v1/auth/token/logout/ -H 'Authorization: Token d7643a4710c7e19915df7d5e3d82f70cb07998ba' # o seu será um novo
```Quando faz o logout ele apaga o token, e só gera um novo quando você fizer login novamente.
## Reset de Senha com djoser
### MailHog
Rode o [MailHog](https://github.com/mailhog/MailHog) usando Docker.
`docker run -d -p 1025:1025 -p 8025:8025 mailhog/mailhog`
Endpoints
```
/api/v1/auth/token/login/
/api/v1/users/reset_password/
/api/v1/users/reset_password_confirm/
```![reset_password](reset_password.png)
Edite `settings.py`
```python
DJOSER = {
'PASSWORD_RESET_CONFIRM_URL': 'password/reset/confirm/{uid}/{token}',
}EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend'
DEFAULT_FROM_EMAIL = config('DEFAULT_FROM_EMAIL', 'webmaster@localhost')
EMAIL_HOST = config('EMAIL_HOST', '0.0.0.0') # localhost
EMAIL_PORT = config('EMAIL_PORT', 1025, cast=int)
EMAIL_HOST_USER = config('EMAIL_HOST_USER', '')
EMAIL_HOST_PASSWORD = config('EMAIL_HOST_PASSWORD', '')
EMAIL_USE_TLS = config('EMAIL_USE_TLS', default=False, cast=bool)
```### Página em pt-br
Edite `settings.py`
```python
LANGUAGE_CODE = 'pt-br'
```### Página com template personalizado
Edite `settings.py`
```python
DJOSER = {
'PASSWORD_RESET_CONFIRM_URL': 'password/reset/confirm/{uid}/{token}',
'EMAIL': {
'password_reset': 'accounts.email.PasswordResetEmail'
}
}TEMPLATES = [
{
'BACKEND': 'django.template.backends.django.DjangoTemplates',
'DIRS': [
BASE_DIR,
BASE_DIR.joinpath('templates')
],
...
}
]
```Crie uma nova app
```
python manage.py startapp accounts
rm -f accounts/{admin,models,tests,views}.py
```Crie um arquivo `email.py`
```python
cat << EOF > accounts/email.py
from djoser import emailclass PasswordResetEmail(email.PasswordResetEmail):
template_name = 'accounts/email/password_reset.html'EOF
```Crie o template de e-mail
```
mkdir -p accounts/templates/accounts/email
touch accounts/templates/accounts/email/password_reset.html
```https://github.com/sunscrapers/djoser/blob/master/djoser/templates/email/password_reset.html
Edite `password_reset.html`
```html
{% block text_body %}
Olá {{ user.first_name }},
...{% block html_body %}
Olá {{ user.first_name }},
...
```### Postman
#### Login
POST: http://localhost:8000/api/v1/auth/token/login/
```
{
"username": "huguinho",
"password": "d"
}
```#### Reset
POST: http://localhost:8000/api/v1/users/reset_password/
```
{
"email": "[email protected]"
}
```* Não precisa de Token Authorization.
#### Reset Password Confirm
POST: http://localhost:8000/api/v1/users/reset_password_confirm/
```
{
"uid": "MQ",
"token": "at61wx-d98ea2d93ae43ba571252177750c4de8",
"new_password": "my_super_new_password123"
}
```* Não precisa de Token Authorization.
Se em settings você definir `PASSWORD_RESET_CONFIRM_RETYPE=True` então você precisa passar `re_new_password`.
```
{
"uid": "MQ",
"token": "at61wx-d98ea2d93ae43ba571252177750c4de8",
"new_password": "my_super_new_password123"
"re_new_password": "my_super_new_password123"
}
```## Autenticação via JWT com djoser
https://djoser.readthedocs.io/en/latest/jwt_endpoints.html
`pip install djoser djangorestframework-simplejwt`
![jwt.png](jwt.png)
```python
# settings.py
from datetime import timedeltaREST_FRAMEWORK = {
'DEFAULT_AUTHENTICATION_CLASSES': (
'rest_framework_simplejwt.authentication.JWTAuthentication', # <--
'rest_framework.authentication.BasicAuthentication',
'rest_framework.authentication.TokenAuthentication',
),
'DEFAULT_PERMISSION_CLASSES': (
'rest_framework.permissions.IsAuthenticated',
)
}SIMPLE_JWT = {
"AUTH_HEADER_TYPES": ("Bearer",), # na doc está JWT mas pode mudar pra Bearer.
"ACCESS_TOKEN_LIFETIME": timedelta(minutes=60),
"REFRESH_TOKEN_LIFETIME": timedelta(days=1),
# "AUTH_TOKEN_CLASSES": ("rest_framework_simplejwt.tokens.AccessToken",),
}
``````python
# urls.py
urlpatterns += [
path('api/v1/', include('djoser.urls')),
path('api/v1/auth/', include('djoser.urls.authtoken')),
path('api/v1/auth/', include('djoser.urls.jwt')), # <--
]
``````python
# views.py
from rest_framework import viewsets
from rest_framework.permissions import IsAuthenticatedfrom product.models import Product
from product.serializers import ProductSerializerclass ProductViewSet(viewsets.ModelViewSet):
queryset = Product.objects.all()
serializer_class = ProductSerializer
permission_classes = (IsAuthenticated,) # <--
``````
curl -X POST http://127.0.0.1:8000/api/v1/auth/token/login/ --data 'username=admin&password=d'curl -X POST http://127.0.0.1:8000/api/v1/auth/jwt/create/ --data 'username=admin&password=d'
```Pegar access
```
# atualizar
curl -X POST \
http://127.0.0.1:8000/api/v1/auth/jwt/refresh/ \
-H 'Content-Type: application/json' \
--data '{"refresh": ""}'curl -X GET \
http://127.0.0.1:8000/product/products/ \
-H 'Authorization: Bearer eyJ0eXAiOiJKV1QiLC...'
```### Consumindo a API com Python
```python
'''
Usage:python consumer.py -u usuario -p senha
'''
from pprint import pprint
from typing import Dictimport click
import requests
from requests.auth import HTTPBasicAuthBASE_URL = 'http://localhost:8000/api/v1/'
def fetch_token(session, endpoint, username, password):
'''
Faz autenticação do usuário.
'''
# headers = {'Content-type': 'application/json'} # Não precisou
data = {
'username': username,
'password': password,
}
with session.post(
endpoint,
auth=HTTPBasicAuth(username, password),
# headers=headers, # Não precisou
data=data
) as response:
return response.json()def get_token(username: str, password: str) -> Dict[str, str]:
'''
Pega o access_token do usuário logado.
'''
with requests.Session() as session:
endpoint = f'{BASE_URL}auth/jwt/create/'
response = fetch_token(session, endpoint, username, password)
data = {
'access_token': response['access'],
}
return datadef fetch(session, endpoint, access_token):
'''
Faz a autenticação usando JWT.
'''
headers = {'Authorization': f'Bearer {access_token}'}
with session.get(endpoint, headers=headers) as response:
return response.json()def post_product(session, endpoint, access_token, title, price):
'''
Salva o produto.
'''
headers = {'Authorization': f'Bearer {access_token}'}
data = {
'title': title,
'price': price,
}
with session.post(endpoint, headers=headers, data=data) as response:
print(response)
pprint(response.json())@click.command()
@click.option('--username', '-u', prompt='username', help='Type the username.')
@click.option('--password', '-p', prompt='password', help='Type the password.')
@click.option('--title', '-t', help='Type the title.')
@click.option('--price', '-pr', help='Type the price.')
def main(username, password, title=None, price=None):
'''
Consumindo a lista de produtos.
'''
token = get_token(username, password)
access_token = token['access_token']
with requests.Session() as session:
endpoint = 'http://127.0.0.1:8000/product/products/'
response = fetch(session, endpoint, access_token)
pprint(response)if title and price:
print(f'Salvando produto: {title}')
post_product(session, endpoint, access_token, title, price)if __name__ == '__main__':
print('Produtos')
main()```
## Django CORS headers
https://pt.wikipedia.org/wiki/Cross-origin_resource_sharing
https://github.com/adamchainz/django-cors-headers
```
python -m pip install django-cors-headers
pip freeze | grep django-cors-headers >> requirements.txt
```Edite `settings.py`
```python
CORS_ALLOWED_ORIGINS = [
'http://localhost:8080',
]INSTALLED_APPS = [
...,
'corsheaders',
...,
]MIDDLEWARE = [
...,
# corsheaders
'corsheaders.middleware.CorsMiddleware',
'django.middleware.common.CommonMiddleware',
...,
]```
```
# Se necessário
python manage.py drf_create_token huguinho
```### Frontend
```
npm install -g @vue/cli
vue create frontend# adicione Router, Vuex e sass-loader (CSS Pre-processors) na instalação
Tire o Linter / Formatter
Marque também Sass/SCSS (with node-sass)cd frontend
npm install axios bulma bulma-toast
npm audit fix
``````js
// main.js
import { createApp } from 'vue'
import App from './App.vue'
import router from './router'
import store from './store'
import axios from 'axios'axios.defaults.baseURL = 'http://127.0.0.1:8000'
createApp(App).use(store).use(router, axios).mount('#app')
``````js
// App.vue
// ...
Login
// ...@import '../node_modules/bulma';
// ...```
Edite router/index.js
```
import Login from '../views/Login.vue'{
path: '/login',
name: 'Login',
component: Login
}
```touch src/views/Login.vue
```js
Login
Usuário
Senha
{{ error }}
Entrar
import axios from 'axios'
export default {
name: 'Login',
data() {
return {
username: '',
password: '',
errors: []
}
},
methods: {
async submitForm() {
axios.defaults.headers.common['Authorization'] = ''
localStorage.removeItem('token')
const formData = {
username: this.username,
password: this.password
}
await axios
.post('/api/v1/auth/token/login/', formData)
.then(response => {
const token = response.data.auth_token
axios.defaults.headers.common['Authorization'] = 'Token ' + token
localStorage.setItem('token', token)
})
.catch(error => {
if (error.response) {
for (const property in error.response.data) {
this.errors.push(`${property}: ${error.response.data[property]}`)
}
} else if (error.message) {
this.errors.push('Algo deu errado. Por favor tente novamente!')
}
})
}
}
}```
## Paginação
Doc: https://www.django-rest-framework.org/api-guide/pagination
Vamos precisar do [django-seed](https://www.dicas-de-django.com.br/41-django-seed)
```
pip install django-seed
```Editar `settings.py`
```python
INSTALLED_APPS = [
...
'django_seed',
```Rodar o comando
```
python manage.py seed blog --number=250
```Editar `blog/models.py`
```python
from django.db import modelsclass Author(models.Model):
name = models.CharField(max_length=255)class Meta:
verbose_name_plural = "Authors"def __str__(self):
return self.nameclass Post(models.Model):
body = models.TextField()
author = models.ForeignKey(Author, on_delete=models.CASCADE, null=True)
created = models.DateTimeField(auto_now_add=True)class Meta:
verbose_name_plural = "Posts"def __str__(self):
return self.body
```Editar `blog/views.py`
```python
from rest_framework.permissions import AllowAnyclass AuthorViewSet(viewsets.ModelViewSet):
...
permission_classes = (AllowAny,)class PostViewSet(viewsets.ModelViewSet):
...
permission_classes = (AllowAny,)
```Editar `blog/serializers.py`
```python
class AuthorSerializer(serializers.ModelSerializer):
...class PostSerializer(serializers.ModelSerializer):
...
```### LimitOffsetPagination
Editar `settings.py`
```python
REST_FRAMEWORK = {
...
'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.LimitOffsetPagination',
'PAGE_SIZE': 5
}
```### PageNumberPagination
Editar `settings.py`
```python
REST_FRAMEWORK = {
...
'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.PageNumberPagination',
'PAGE_SIZE': 5
}
```### Paginação personalizada global (Custom Pagination)
> Criar app core
```
rm -f core/{admin,models,tests,views}.py
rm -rf core/migrations
touch core/pagination.py
```Editar `core/pagination.py`
```python
from rest_framework.pagination import PageNumberPaginationclass StandardResultsSetPagination(PageNumberPagination):
page_size = 10
page_size_query_param = 'page_size'
max_page_size = 100
```Editar `settings.py`
```python
REST_FRAMEWORK = {
...
'DEFAULT_PAGINATION_CLASS': 'core.pagination.StandardResultsSetPagination',
'PAGE_SIZE': 5
}
```### Paginação personalizada para o blog
Editar `blog/pagination.py`
```python
from rest_framework.pagination import PageNumberPaginationclass CustomBlogResultsSetPagination(PageNumberPagination):
page_size = 7
page_size_query_param = 'page_size'
max_page_size = 70
```Editar `blog/views.py`
```python
from blog.pagination import CustomBlogResultsSetPaginationclass PostViewSet(viewsets.ModelViewSet):
...
pagination_class = CustomBlogResultsSetPagination
```### Cursor Pagination
Editar `settings.py`
```python
REST_FRAMEWORK = {
...
'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.CursorPagination',
'PAGE_SIZE': 5
}
```> Requer um campo com o nome `created` no seu modelo.
## django-filter
### Filtrando a queryset
Editar `blog/models.py`
```python
# blog/models.py
from django.contrib.auth.models import User
from django.db import modelsclass Author(models.Model):
first_name = models.CharField(max_length=30)
last_name = models.CharField(max_length=30, null=True)class Meta:
verbose_name_plural = "Authors"@property
def full_name(self):
return f'{self.first_name} {self.last_name or ""}'.strip()def __str__(self):
return self.full_nameclass Post(models.Model):
title = models.CharField(max_length=30)
body = models.TextField()
author = models.ForeignKey(Author, on_delete=models.CASCADE, null=True)
created_by = models.ForeignKey(
User,
on_delete=models.CASCADE,
verbose_name='criado por',
null=True
)
created = models.DateTimeField(auto_now_add=True)class Meta:
verbose_name_plural = "Posts"def __str__(self):
return self.title
```Editar `blog/admin.py`
```python
# blog/admin.py
from django.contrib import adminfrom blog.models import Author, Post
@admin.register(Author)
class AuthorAdmin(admin.ModelAdmin):
list_display = ('__str__',)@admin.register(Post)
class PostAdmin(admin.ModelAdmin):
list_display = ('title', 'author', 'created_by')
```Editar `blog/views.py`
```python
# blog/views.py
from rest_framework.permissions import AllowAny, IsAuthenticatedclass PostViewSet(viewsets.ModelViewSet):
# queryset = Post.objects.all()
queryset = Post.objects.filter(created_by__username='regis')
serializer_class = PostSerializer
# permission_classes = (AllowAny,)
permission_classes = (IsAuthenticated,)
# pagination_class = CustomBlogResultsSetPagination
```### Filtrando pelo usuário logado
Editar `blog/views.py`
```python
# blog/views.py
class PostViewSet(viewsets.ModelViewSet):
# queryset = Post.objects.all()
# queryset = Post.objects.filter(created_by__username='regis')
serializer_class = PostSerializer
# permission_classes = (AllowAny,)
permission_classes = (IsAuthenticated,)
# pagination_class = CustomBlogResultsSetPaginationdef get_queryset(self):
user = self.request.user
return Post.objects.filter(created_by=user)
```Erro:
```
AssertionError: `basename` argument not specified, and could not automatically determine the name from the viewset, as it does not have a `.queryset` attribute.
```Editar `blog/urls.py`
```python
# blog/urls.py
...
router.register(r'authors', AuthorViewSet, basename='Author')
router.register(r'posts', PostViewSet, basename='Post')
```### Filtrando a partir de query parameters
Editar `settings.py`
```python
# settings.py
REST_FRAMEWORK = {
'DEFAULT_AUTHENTICATION_CLASSES': (
# 'rest_framework_simplejwt.authentication.JWTAuthentication',
'rest_framework.authentication.BasicAuthentication',
'rest_framework.authentication.TokenAuthentication',
),
```Editar `blog/views.py`
```python
# blog/views.py
class PostViewSet(viewsets.ModelViewSet):
serializer_class = PostSerializer
permission_classes = (IsAuthenticated,)def get_queryset(self):
queryset = Post.objects.all()
username = self.request.query_params.get('username')if username is not None:
queryset = queryset.filter(created_by__username=username)title = self.request.query_params.get('title')
if title is not None:
queryset = queryset.filter(title__icontains=title)return queryset
```### Filtro Genérico django-filter
https://django-filter.readthedocs.io/en/stable/guide/rest_framework.html#integration-with-drf
```
pip install django-filterpip freeze | grep django-filter >> requirements.txt
```Editar `settings.py`
```python
# settings.py
INSTALLED_APPS = [
...
# 3rd apps
'django_filters',
...
]REST_FRAMEWORK = {
...
'DEFAULT_FILTER_BACKENDS': ['django_filters.rest_framework.DjangoFilterBackend']
}
```Editar `blog/views.py`
```python
# blog/views.py
class PostViewSet(viewsets.ModelViewSet):
queryset = Post.objects.all()
serializer_class = PostSerializer
permission_classes = (IsAuthenticated,)
filterset_fields = ('title', 'body')# comentar def get_queryset(self)
```Filtra pelo texto completo.
### Adicionando filtro específico com filterset_class
Editar `blog/filters.py`
```python
# blog/filters.py
from django_filters import rest_framework as filtersfrom blog.models import Post
class PostFilter(filters.FilterSet):
title = filters.CharFilter(field_name="title", lookup_expr='icontains')
body = filters.CharFilter(field_name="body", lookup_expr='icontains')class Meta:
model = Post
fields = ('title', 'body')
```Editar `blog/views.py`
```python
# blog/views.py
from blog.filters import PostFilterclass PostViewSet(viewsets.ModelViewSet):
queryset = Post.objects.all()
serializer_class = PostSerializer
permission_classes = (IsAuthenticated,)
# filterset_fields = ('title', 'body')
filterset_class = PostFilter
```### Campo de busca
Editar `blog/views.py`
```python
# blog/views.py
from rest_framework.filters import SearchFilterclass AuthorViewSet(viewsets.ModelViewSet):
...
filter_backends = (SearchFilter,)
search_fields = ('first_name', 'last_name')
```## Criando subrota com action
Doc: [https://www.django-rest-framework.org/api-guide/viewsets/#marking-extra-actions-for-routing](https://www.django-rest-framework.org/api-guide/viewsets/#marking-extra-actions-for-routing)
Doc: [https://www.django-rest-framework.org/api-guide/routers/#routing-for-extra-actions](https://www.django-rest-framework.org/api-guide/routers/#routing-for-extra-actions)
Editar `blog/models.py`
```python
class Post(models.Model):
...
like = models.BooleanField(null=True)
```Editar `blog/views.py`
```python
from rest_framework.decorators import action
from rest_framework.response import Responseclass PostViewSet(viewsets.ModelViewSet):
queryset = Post.objects.all()
serializer_class = PostSerializer
permission_classes = (IsAuthenticated,)@action(detail=True, methods=['put'])
def like(self, request, pk=None):
'''
Marca Like = True
'''
post_obj = self.get_object()
post_obj.like = True
post_obj.save()
serializer = self.get_serializer(post_obj)
return Response(serializer.data)@action(detail=True, methods=['put'])
def unlike(self, request, pk=None):
'''
Marca Like = False
'''
post_obj = self.get_object()
post_obj.like = False
post_obj.save()
serializer = self.get_serializer(post_obj)
return Response(serializer.data)@action(detail=False, methods=['get'])
def my_posts(self, request, pk=None):
'''
Retorna somente os meus posts.
'''
user = self.request.user
# posts = Post.objects.filter(created_by=user)
posts = self.get_queryset().filter(created_by=user)page = self.paginate_queryset(posts)
if page is not None:
serializer = self.get_serializer(page, many=True)
return self.get_paginated_response(serializer.data)serializer = self.get_serializer(posts, many=True)
return Response(serializer.data)
```Editar `blog/admin.py`
```python
class PostAdmin(admin.ModelAdmin):
list_display = ('title', 'author', 'created_by', 'like')
list_filter = ('like',)
```As novas rotas são:
```
/blog/posts//like/
/blog/posts//unlike/
/blog/posts/my_posts/
```