如何通过电子邮件强制执行 Django 用户注册单步(而不是两步)过程?

2022-01-25 00:00:00 python django django-admin django-views

问题描述

一旦管理员将新用户添加到管理站点,我希望 Django 向用户电子邮件地址发送一封包含登录详细信息的电子邮件.所以我尝试使用 Django 信号,但只是因为 django 用户注册是一个两步过程,信号会得到通知仅在第一步中并在没有电子邮件地址的情况下调用电子邮件功能(在第二步中出现).我的信号码:

I want Django to send an email to user email-address with Login details once admin adds a new user to admin site.So I tried using Django signals for that but just becoz django user registration is a two step process signals get notified in first step only and called email function without email address(which comes in second step). My signal code:

def email_new_user(sender, **kwargs):
    if kwargs["created"]:  # only for new users
        new_user = kwargs["instance"]
       send_mail('Subject here', 'Here is the message.', 'from@example.com',['to@example.com'], fail_silently=False)


post_save.connect(email_new_user, sender=User)

所以我试图克服这个问题.我在 admin.py 中使用此代码

So what i tried to overcome this problem.I use this code in admin.py

class UserAdmin(admin.ModelAdmin):
    list_display = ('username', 'email', 'first_name', 'last_name', 'date_joined', 'last_login')
    search_fields = ['username', 'email']
    filter_horizontal = ('user_permissions',)

admin.site.unregister(User)
admin.site.register(User, UserAdmin)

这使得所有注册过程成为一个单一的步骤过程,我的信号开始工作并在添加新用户时向 user_id 发送邮件.但问题出现在此之后:

This makes all registration process a single step process and my signals start working and sending mail to user_id on new user addition.But the problem came after this were:

1.用户密码没有转换成hash,在进入表单时可见,导致用户无法登录管理站点.

2.表单中的电子邮件字段不是强制性的,我希望是强制性的.

请帮帮我:(

I tried your code But I m still at same place where i was before posting this question.
the code i used in my admin.py is:
from django.contrib import admin
from mysite.naturefarms.models import *
from django.contrib.auth.models import User,Group
from django.contrib.auth.admin import UserAdmin
from django.contrib.auth.forms import UserCreationForm, UserChangeForm
from django import forms
from django.contrib.admin.views.main import *

class MyUserCreationForm(UserCreationForm):
    class Meta:
        model = User
        fields = ('username', 'email',)
class UserAdmin(admin.ModelAdmin):
    add_form = MyUserCreationForm

admin.site.unregister(User)

class MyUserAdmin(UserAdmin):
    add_form = MyUserCreationForm
    add_fieldsets = (
        (None, {
            'classes': ('wide',),
            'fields': ('username', 'email', 'password1', 'password2')}
        ),
    )




admin.site.register(User, MyUserAdmin)


解决方案

如果你查看 django.contrib.auth admin.py,你会看到 UserAdmin 类将 add_form 指定为 UserCreationForm.

If you look in django.contrib.auth admin.py, you'll see that the UserAdmin class specifies the add_form as UserCreationForm.

UserCreationForm 仅包含来自 User 模型的用户名"字段.

UserCreationForm only includes the 'username' field from the User model.

由于您提供自己的 UserAdmin,因此您只需将 add_form 覆盖为自定义 UserCreationForm,其中包含使信号正常工作所需的字段.

Since you're providing your own UserAdmin, you can just override the add_form to a custom UserCreationForm that includes the fields you need to make your signal work properly.

希望对你有所帮助.

这是来自 contrib.auth forms.py 的 UserCreationForm:

Here's the UserCreationForm from contrib.auth forms.py:

class UserCreationForm(forms.ModelForm):
    """
    A form that creates a user, with no privileges, from the given username and password.
    """
    username = forms.RegexField(label=_("Username"), max_length=30, regex=r'^[w.@+-]+$',
        help_text = _("Required. 30 characters or fewer. Letters, digits and @/./+/-/_ only."),
        error_messages = {'invalid': _("This value may contain only letters, numbers and @/./+/-/_ characters.")})
    password1 = forms.CharField(label=_("Password"), widget=forms.PasswordInput)
    password2 = forms.CharField(label=_("Password confirmation"), widget=forms.PasswordInput,
        help_text = _("Enter the same password as above, for verification."))

    class Meta:
        model = User
        fields = ("username",)

    def clean_username(self):
        username = self.cleaned_data["username"]
        try:
            User.objects.get(username=username)
        except User.DoesNotExist:
            return username
        raise forms.ValidationError(_("A user with that username already exists."))

    def clean_password2(self):
        password1 = self.cleaned_data.get("password1", "")
        password2 = self.cleaned_data["password2"]
        if password1 != password2:
            raise forms.ValidationError(_("The two password fields didn't match."))
        return password2

    def save(self, commit=True):
        user = super(UserCreationForm, self).save(commit=False)
        user.set_password(self.cleaned_data["password1"])
        if commit:
            user.save()
        return user

注意 fields = ("username",) 元组,它排除了 User 模型上的所有其他字段.你需要这样的东西:

Notice the fields = ("username",) tuple which excludes all other fields on the User model. You need something like:

class MyUserCreationForm(UserCreationForm):
    class Meta:
        model = User
        fields = ('username', 'email',)

然后您可以将其用作自定义 UserAdmin 中的 add_form:

then you can use that as the add_form in your custom UserAdmin:

class UserAdmin(admin.ModelAdmin):
    add_form = MyUserCreationForm

在我所在的地区已经很晚了,但我会看看我明天能否为你弄到一个工作样本.

It's pretty late in my part of the world, but I'll see if I can get a working sample for you tomorrow.

好的,这是您需要进行的必要更改才能完成这项工作.我已经使用 Django 1.3 对其进行了测试:

Ok, here's the necessary changes you'll need to make to make this work. I've tested it using Django 1.3:

from django.contrib.auth.forms import UserCreationForm
from django.contrib.auth.admin import UserAdmin
from django.contrib.auth.models import User
from django import forms

admin.site.unregister(User)

class MyUserAdmin(UserAdmin):
    add_form = MyUserCreationForm
    add_fieldsets = (
        (None, {
            'classes': ('wide',),
            'fields': ('username', 'email', 'password1', 'password2')}
        ),
    )

admin.site.register(User, MyUserAdmin)

我没有看到 UserAdmin 最初有 add_fieldset 属性.这就是添加表单中没有显示电子邮件字段的原因.

I didn't see that the UserAdmin had an add_fieldset property initially. That's why the email field wasn't displaying in the add form.

相关文章