はじめに
42Tokyoの学生です。
42の課題でDjangoを触る機会があったので、個人的に役に立った機能をまとめてみました。
本記事はあくまでも紹介レベルなので、詳細を知りたい方は別の記事や公式ドキュメントを参照してください。
ミドルウエア
ミドルウエアという言葉で調べると、たいていOSとアプリケーションの中間のソフトという意味で出てきますが、Djangoにおいてはクライアント側の送信したデータに対して、受信時および送信時に実行できる処理を表します。
つまり、Viewで行う処理の前後に、リクエストデータ・レスポンスデータに対して何らかの処理をすることができます。
今回の課題ではデフォルトの状態から以下のものを追加しました。
LoginRequiredMiddleware
Djangoで管理するすべてのページに対して、ログインしていなければアクセスできないようにします。これを利用しなければ、次のように@login_requiredでデコレートしなければなりません。
@login_required
def view_test(request):
pass
#class viewの時は下記のようにする
@method_decorator(login_required, name="dispatch")
class ViewTest(TemplateView):
pass
すべてのviewに対して書かなければならないので面倒ですし書き忘れも発生します。
しかし、LoginRequiredMiddlewareを使えば上記のようなに@login_requiredを書く必要がなく、デフォルトでユーザーがログインしていない限りアクセスできなくなります。
注意点として、login画面もlogin指定な限りアクセスできなくなるので、そういう場合は次のように@login_not_requiredでデコレートします
@login_not_required
def login_test(request):
pass
#class viewの時は下記のようにする
@method_decorator(login_not_required, name="dispatch")
class LoginTest(TemplateView):
pass
LocaleMiddleware
多言語機能のために利用しています。
言語の切り替えを簡単に実装できます。
ミドルウエアのカスタマイズ
あまり推奨されるやり方ではないのかもしれませんが、既存のミドルウエアをカスタマイズして差し替えることもできます。差し替えは、settings.pyを変更することで実施できます。
SessionMiddlewareのカスタマイズ
#既存のものをコメントアウトし、カスタマイズしたものに差し替える
MIDDLEWARE = [
# "django.contrib.sessions.middleware.SessionMiddleware"
"accounts.middleware.CustomSessionMiddleware"
]
こちらを実施した理由はJWTをセッションIDに差し替えるためです。
詳細はこちらを参照してください。
https://qiita.com/hsano43/items/509544025882f852aa62
UserActionLoggingMiddleware
こちらは新規に作成したミドルウエアです。
詳細は語りませんが、ユーザーからのリクエストデータをすべて保持するための作成しました。
テンプレート
ざっくりいうとサーバー側でHTMLを作成する機能です。
ただHTMLを書くだけでなく、View側の処理を反映させることができます。
例えば次のようなHTMLコードがあるとします。
<div>{{tournament.name}}</div>
<div>{{tournament.organizer}}</div>
View側の実装は次のようにします。
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
context["tournament"] = {
"name": _("テストトーナメント"),
"organizer": _("主催者1"),
}
return context
すると、クライアント側では次のようなHTMLデータを受け取ることができます
<div>テストトーナメント</div>
<div>主催者1</div>
ほかにも、includeを利用して関数のようにhtmlを使いまわすこともできます。
(card.htmlはここでは省略していますがBootstrapのCardを利用しています)
{% for tournament in tournaments %}
{% include 'tournament/card.html' with tournament=tournament attr=attr %}
{% endfor %}
ここでやっていることは、tournamentsはView側で定義したListで、その個数分だけcard.htmlを作成していることになります。もちろん、リストの内部データはすべて異なるので、異なるカードが表示されることになります。
ほかにもif~elseで処理を分岐したり、多言語に対応させたりできます。
認証機能
デフォルトのままだと課題要件がクリアできなかったのでいろいろと変更しました。
ドキュメントにも書いてありますが、デフォルトでも十分に使えます。
カスタマイズの詳細は、以前に書きました。
https://qiita.com/hsano43/items/afc6dfcfae06ca8f112c
Django以外の機能
Djangoの機能ではないですが、RedisとCeleryは大変役に立ちました。
Redis
有効期限を設定すれば自動で消去してくれるので一時的なデータ保持に有効でした。
redisがなければ定期的に削除するか、あるいは一時データの保存サイズが大きくなったら自動で削除するなど、面倒な処理をしなければなりませんでした。
Celery
非同期タスクを簡単に扱うことができるようになりました。
なぜ必要なのかというと、クライアントからリクエストが来た際、サーバーは当然レスポンスを返します。この時、ある処理を5分後に実行したいとすると、同期処理の場合は5分待たなければレスポンスを返すことができません。これを防ぐために、5分待つ処理を非同期にする必要があります。
Celeryはその非同期処理を実施してくれるライブラリになります。
また、cronのように定期実行もすることができるので、こちらも役に立ちました。
まとめ
以上がDjangoどころかWeb開発も初めての自分が、42の課題に対して活用したDjangoの機能になります。
ミドルウエアは最初何のためのものかよくわかりませんでしたが、レスポンスデータをViewで処理する前後に様々な処理を加えることができて便利でしたし、テンプレートも同様に最初はただのHTMLを書いていただけでしたが、一度利点に気づくと活用方法がいくつも出てきて役に立ちました。
自分が知らない役立つ機能はおそらくまだまだあると思うので、また機会があれば探してみたいと思います