2024年12月17日火曜日

メール認証付きユーザー登録の流れ

 Djangoでユーザー登録をメール承認(認証)付きにするのは多少の設定が必要ですが、Djangoが提供する機能を活用すれば、それほど大変ではありません。以下に、ステップごとに分解して説明します。


1. メール認証付きユーザー登録の流れ

  1. ユーザーがフォームで登録
    • 仮登録の状態でユーザーをデータベースに保存。
  2. メールを送信
    • ユーザーの入力したメールアドレス宛に認証用のリンクを送信。
  3. ユーザーがリンクをクリック
    • 認証リンクにアクセスした際、アカウントを有効化。
  4. 有効化されたアカウントでログイン可能に

2. 実装手順

2.1 仮登録用のモデルフィールドを追加

デフォルトのユーザーモデルには「有効/無効」を管理するフィールドがないため、is_active フィールドを利用します。

  • 新規登録時にデフォルトで is_active=False に設定し、メール認証後に True に変更します。

forms.py (変更なし)

上記で作成した CustomUserCreationForm をそのまま使用します。


2.2 認証トークン生成と検証

Djangoの signing モジュールを利用して、メール認証用のトークンを生成します。

認証トークン生成ヘルパー

utils.py を新規作成します。

python
from django.contrib.auth.tokens import default_token_generator from django.utils.http import urlsafe_base64_encode, urlsafe_base64_decode from django.utils.encoding import force_bytes, force_str from django.contrib.auth.models import User # トークン生成 def generate_email_verification_token(user): return default_token_generator.make_token(user) # トークンの検証 def verify_email_token(uidb64, token): try: uid = force_str(urlsafe_base64_decode(uidb64)) user = User.objects.get(pk=uid) except (TypeError, ValueError, OverflowError, User.DoesNotExist): return None if default_token_generator.check_token(user, token): return user return None

2.3 メール送信ビューの作成

Djangoの send_mail を使ってメールを送信します。

views.py

python
from django.shortcuts import render, redirect from django.contrib.auth import login from django.core.mail import send_mail from django.template.loader import render_to_string from django.utils.http import urlsafe_base64_encode from django.utils.encoding import force_bytes from django.contrib.sites.shortcuts import get_current_site from .forms import CustomUserCreationForm from .utils import generate_email_verification_token def register(request): if request.method == "POST": form = CustomUserCreationForm(request.POST) if form.is_valid(): user = form.save(commit=False) user.is_active = False # 仮登録 user.save() # メール送信 current_site = get_current_site(request) subject = "Activate Your Account" message = render_to_string("email_verification.html", { "user": user, "domain": current_site.domain, "uid": urlsafe_base64_encode(force_bytes(user.pk)), "token": generate_email_verification_token(user), }) send_mail(subject, message, "no-reply@example.com", [user.email]) return render(request, "email_sent.html") else: form = CustomUserCreationForm() return render(request, "register.html", {"form": form})

2.4 認証リンクのビューを作成

ユーザーがメールのリンクをクリックすると、アカウントが有効化されます。

views.py

python
from django.http import HttpResponse from django.contrib.auth.models import User from .utils import verify_email_token def activate_account(request, uidb64, token): user = verify_email_token(uidb64, token) if user is not None: user.is_active = True user.save() return HttpResponse("Your account has been activated! You can now log in.") else: return HttpResponse("Activation link is invalid!")

2.5 URLの設定

urls.py に登録用と有効化用のURLを追加します。

python
from django.urls import path from .views import register, activate_account urlpatterns = [ path('register/', register, name='register'), path('activate/<str:uidb64>/<str:token>/', activate_account, name='activate'), ]

2.6 認証メールテンプレート

email_verification.html

html
Hi {{ user.username }}, Please click the link below to verify your email address and activate your account: http://{{ domain }}{% url 'activate' uid=uid token=token %} Thank you!

2.7 メール送信設定

settings.py にメール送信設定を追加します。

python
EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend' EMAIL_HOST = 'smtp.example.com' EMAIL_PORT = 587 EMAIL_USE_TLS = True EMAIL_HOST_USER = 'your_email@example.com' EMAIL_HOST_PASSWORD = 'your_password' DEFAULT_FROM_EMAIL = 'no-reply@example.com'

3. テストとデバッグ

  1. ローカル環境でテストする場合は、console.EmailBackend を使用すると、メールがコンソールに出力されます。
    python
    EMAIL_BACKEND = 'django.core.mail.backends.console.EmailBackend'
  2. トークン生成やデコードでエラーが出た場合は、例外を確認してください。

4. メリットと応用

  • 安全性向上: メール確認を必須にすることで、無効なユーザー登録を防ぎます。
  • 通知機能と統合: この仕組みを活用して、パスワードリセットや重要な通知機能にも応用できます。

これで、Djangoでメール認証付きユーザー登録を実現できます!不明点や追加のカスタマイズがあれば、教えてください。

2024年12月9日月曜日

Python基礎:エラー処理 (Try Except)

 今回はエラー処理について解説します

エラーは以下の2種類に分けられます。

・構文エラー
・例外エラー


構文エラー

Pythonの文法違反によるエラーで、プログラムの実行ができません。

例: 行末のコロン忘れやインデントの不正。発生すると即停止します。


例外エラー

実行時に発生するエラーで、tryとexceptを使って制御可能です。

適切に処理することでエラーの影響を回避できます。


try exceptの書き方

try: 
    処理1
    処理2
    ~
    
except エラー名:
    エラー発生時の処理
try:
    num = input("数字を入力して下さい")
    
    intData = int( num )    # 整数に変換
    data = 100 / intData    #  割り算します 
    print("結果",data)
 
except ZeroDivisionError:
    print("0で割ることはできません")
    
except ValueError:
    print("値のチェックが必要です(数字でない)")
 
except Exception as e:
    print("その他のエラー", e)


スクリプトの説明


スクリプトを詳しく見てみましょう! 「数字を入力して下さい」を表示して入力要求します、
     整数に変換、
     100で割る、
     最後に結果を表示します。

スクリプトは、入力された数字の割り算を行うというものですが、エラーをつかまえてエラー処理をおこないます。

エラーは、0割りのエラー、数字以外のエラー、その他エラーをprint文で表示しています。

つかまえるにはエラーは、

    except ZeroDivisionError
    eexcept ValueError
    eexcept Exception as e

を使います、最後のException as eの「e」変数にはエラー情報が入っています、文字列型ではないので注意して下さい。(文字列型にするには、str(e)のように文字列型にキャストします)

このコードはPythonでエラー処理を行う方法を示す例です。
コードの概要
tryブロック
    実行したい処理を記述します。ここでは、ユーザー入力を受け取り、
    それを整数に変換して100を割る計算を試みています。
exceptブロック
    特定のエラーが発生した場合に実行される処理を記述します。

処理の流れ
・ユーザーが数字を入力しない場合(例えば文字列を入力)、ValueError が発生し、
    対応するメッセージが表示されます。
・ユーザーが 0 を入力した場合、ZeroDivisionError が発生し、割り算のエラーとしてメッセージを表示します。
・それ以外の予期しないエラーが発生した場合は、Exception によってキャッチされ、
    エラー内容が表示されます。
出力例
    1. 数字を正しく入力した場合
            数字を入力して下さい: 10
            結果 10.0
    2. 0 を入力した場合
            数字を入力して下さい: 0
            0で割ることはできません
    3. 数字以外の文字列を入力した場合
           数字を入力して下さい: abc
            値のチェックが必要です(数字でない)
4. 他のエラーが発生した場合(ほとんどの場合発生しない)
            その他のエラー <エラー内容>
ポイント
    ・エラーを明確に分類して処理することで、ユーザーに適切なフィードバックを提供できます。
    ・Exception as e は、予測できないエラーの詳細を取得するのに役立ちます。



関数(消費税を求める) 関数の書き方と呼び出し方まとめ

関数(消費税を求める)


次のような関数を作成してみます。

「金額と税率を引数として渡し、支払い額と税額をタプルで返却する消費税計算関数を作成します。関数呼び出し後、結果を print 文で表示します。」

コードは次のようになります。

def get_tax( kingaku, rate=0.1 ):
 
  tax = kingaku * rate
  total = kingaku + tax
 
  return total,tax

消費税を計算する関数 get_tax を定義しています。

コード説明

引数
    kingaku: 元の金額(消費税の計算対象)
    rate: 税率(デフォルト値は 0.1 = 10%)

処理
    tax に税額を計算します: kingaku * rate
    total に支払い総額を計算します: kingaku + tax

返却値
    total(支払い総額)と tax(税額)をタプルとして返します










関数の書き方と呼び出し方まとめ

1. 関数の呼び出し方

    関数名を書き、渡したい値(引数)を括弧で囲んで渡します。

        関数名(引数1, 引数2)


2. 関数の作り方

    関数内で実行する機能を決める。

    関数を以下の形式で定義する
     (デフォルト値が設定されている引数は、省略可能)

        def 関数名(引数1, 引数2=デフォルト値, ...):
            # 処理内容
            return 戻り値


3. 関数から値を返す方法
    return を使って複数の値を返すことが可能。
    引数としてリスト型や辞書型を指定する。
    必要に応じてグローバル変数を使用する。

サンプルコード

def 計算関数(数値1, 数値2=10):
    合計 = 数値1 + 数値2
    差 = 数値1 - 数値2
    return 合計, 差

# 呼び出し例
結果 = 計算関数(20)
print(結果)  # 出力: (30, 10)