python - 电子邮件字段中的 django UniqueConstraint 错误(sendgrid 验证电子邮件)
问题描述
我正在尝试对我的 django 项目实施 sendgrid 电子邮件验证。我一直在关注https://www.twilio.com/blog/send-dynamic-emails-python-twilio-sendgrid教程。我有相应的模型和视图设置但是,我在尝试使用相同的电子邮件地址两次时遇到了 UniqueConstraint 错误(尽管我的模型将电子邮件字段设置为唯一 = True)。而不是得到 UniqueConstraint 错误,我希望该应用程序只是为了防止用户两次添加相同的电子邮件。我尝试使用cleaned_data 作为表单,但无论如何我都无法通过这个UniqueConstraint 错误。
我的错误:异常类型:IntegrityError 异常值:
唯一约束失败:newsletter_newsletteruser.email
我的模型.py:
from django.db import models
class NewsletterUser (models.Model):
email= models.EmailField(unique=True)
date_added = models.DateTimeField(auto_now_add=True)
conf_num = models.CharField(max_length=15)
confirmed = models.BooleanField(default=False)
def __str__(self):
return self.email + " (" + ("not " if not self.confirmed else "") + "confirmed)"
我的管理员.py
from .models import NewsletterUser, MailMessage
# Register your models here.
class NewsletterAdmin(admin.ModelAdmin):
list_display = ('email','date_added','conf_num', 'confirmed')
admin.site.register(NewsletterUser, NewsletterAdmin)
admin.site.register(MailMessage)
我的观点.py
from django.db.models.fields import EmailField
from django.shortcuts import render, redirect
from sendgrid.helpers.mail.bcc_email import Bcc
from django.http import HttpResponse
from .forms import NewsletterUserForm, MailMessageForm
from django.contrib import messages
from django.contrib.auth.decorators import login_required
import os
from sendgrid import SendGridAPIClient
from sendgrid.helpers.mail import Mail, Email, Content, To
from .models import NewsletterUser
from django import forms
import pandas as pd
from django.shortcuts import render
from django.http import HttpResponse
from django.conf import settings
from django.views.decorators.csrf import csrf_exempt
import random
from sendgrid import SendGridAPIClient
# Create your views here.
#!REGISTER CURRENT HOST!
host = 'http://127.0.0.1:8000/'
# Helper Functions
def random_digits():
return "%0.12d" % random.randint(0, 999999999999)
@csrf_exempt
def newsletters(request):
if request.method == 'POST':
sub = NewsletterUser(email=request.POST['email'], conf_num=random_digits())
sub.save()
message = Mail(
from_email='pasiekaradosc@gmail.com',
to_emails=sub.email,
subject='Potwierdzenie Adresu Email w Newsletterze Pasieka Radość',
html_content= "<h1>Dziękujemy za zapisanie się do newslettera Pasieka Radość!<h1> \
<h2>Aby potwierdzić rejestrację newslettera</h2> \
<h2><a href='{}confirm/?email={}&conf_num={}'> Kliknij w poniższy link</a>.<h2>".format(request.build_absolute_uri(host),
sub.email,
sub.conf_num))
sg = SendGridAPIClient(os.environ.get('SENDGRID_API_KEY'))
response = sg.send(message)
return render(request, 'newsletter/newsletter.html', {'email': sub.email, 'action': 'wysłano', 'form': NewsletterUserForm()})
else:
return render(request, 'newsletter/newsletter.html', {'form': NewsletterUserForm()})
def confirm(request):
sub = NewsletterUser.objects.get(email=request.GET['email'])
if sub.conf_num == request.GET['conf_num']:
sub.confirmed = True
sub.save()
messages.success(request, f'Dziękujemy za zapisanie adresu email do naszego newslettera!')
return render(request, 'pages/home.html' , {'email': sub.email, 'action': 'confirmed'})
我的表格.py
from django import forms
from .models import NewsletterUser, MailMessage
class NewsletterUserForm (forms.ModelForm):
email = forms.EmailField(label='Twój email',
max_length=100,
widget=forms.EmailInput(attrs={'class': 'form-control'}))
class Meta:
model = NewsletterUser
fields = ['email']
我的 html 模板 (newsletter.html)
{% extends "base.html" %}
{% load crispy_forms_tags %}
{% block content%}
<h1> {{title}} </h1>
<h4>Zapisz się do newslettera i odbieraj najświeższe informacje dotyczące naszej pasieki i oferty.</h4>
<div class="col-12">
{% if email %}
<p class="alert alert-info">Na {{ email }} {{ action }} wiadomość weryfikacyjną. Otwórz wiadomość i kliknij w link aby dodać adres do newslettera. </p>
{% endif %}
<form method="POST" autocomplete="off">
{% csrf_token %}
<div class="col-sm-4">
{{form|crispy}}
</div>
<br>
<div class="col-sm-4">
<input class="btn btn-info" type='submit' value='Zapisz Się'>
</div>
</form>
<br>
<div class="row">
<div class="col-sm-4">
<p>Spokojnie, nie będziemy wysyłać Ci tony niepotrzebnych wiadomości. Zapisz się do newslettera i dowiedz się pierwszy/pierwsza, kiedy w naszej pasiecie pojawi się nowa dostawa miodów lub innych interesujących produktów.</p>
</div>
</div>
</div>
{% endblock content %}
我对 django 和 sendgrid 很陌生,所以这让我想知道是否需要以某种方式完全重建我的代码,或者我可以采取一些步骤让它像上面那样工作。如果有任何提示,请分享。谢谢
解决方案
Twilio SendGrid 开发人员布道者在这里。
我相信 Django 在这里表现得和预期一样。从Django 文档:
如果您尝试在唯一字段中保存具有重复值的模型,模型的方法
django.db.IntegrityError
将引发a。save()
您需要做的是捕获它IntegrityError
,然后将错误返回给用户。
@csrf_exempt
def newsletters(request):
if request.method == 'POST':
sub = NewsletterUser(email=request.POST['email'], conf_num=random_digits())
try:
sub.save()
message = Mail(
from_email='pasiekaradosc@gmail.com',
to_emails=sub.email,
subject='Potwierdzenie Adresu Email w Newsletterze Pasieka Radość',
html_content= "<h1>Dziękujemy za zapisanie się do newslettera Pasieka Radość!<h1> \
<h2>Aby potwierdzić rejestrację newslettera</h2> \
<h2><a href='{}confirm/?email={}&conf_num={}'> Kliknij w poniższy link</a>.<h2>".format(request.build_absolute_uri(host),
sub.email,
sub.conf_num))
sg = SendGridAPIClient(os.environ.get('SENDGRID_API_KEY'))
response = sg.send(message)
return render(request, 'newsletter/newsletter.html', {'email': sub.email, 'action': 'wysłano', 'form': NewsletterUserForm()})
except IntegrityError as ex:
# Return the error to your user
else:
return render(request, 'newsletter/newsletter.html', {'form': NewsletterUserForm()})
我已经在上面的代码中添加了,当电子邮件不是唯一的时try/except
,您需要在块中填写要做什么。except
推荐阅读
- javascript - 无法在我的页面上处理 ajax 调用,因为 Django request.is_ajax 在页面初始加载时返回 True
- kotlin - 在匿名 lambda 中返回 false
- r - 使用数字数据和字符创建数组
- microsoft-teams - 如何使用 Graph API 向 MS Team 的频道发送消息
- asp.net-core - 如何设置 WASM IdentityServer 与用户/通过身份验证和设备身份验证与来自控制台应用程序的客户端/秘密
- version - 在 NuxtJS 应用程序上显示 package.json 版本
- java - java.lang.NoClassDefFoundError: jdk/nashorn/api/scripting/NashornScriptEngineFactory
- r - 带有年份的条形图滑块
- node.js - 您如何在 Vercel node.js 应用程序中解析多部分表单?
- python - 第二个对话框窗口上的按钮不起作用