Dans la première partie de cette série, nous avons créé une boutique en ligne Django avec htmx.
Dans cette deuxième partie, nous traiterons les commandes avec Stripe.
Nous intégrerons Stripe pour gérer les paiements en toute sécurité. C'est ce que nous voulons réaliser :
Nous devons d'abord passer à Stripe et procéder comme suit :
Commencez par créer un compte Stripe. Pour l’instant, vous n’avez pas vraiment besoin d’activer votre compte. Vous pouvez simplement travailler en mode test, ce qui vous empêchera d'effectuer de vrais paiements pendant les tests. Accédez à la page des clés API et récupérez les clés publiables et secrètes. Enregistrez-les dans les variables d'environnement de votre projet (STRIPE_PUBLISHABLE_KEY et STRIPE_SECRET_KEY). Nous utiliserons ces clés pour authentifier vos demandes Stripe.
Créez un nouveau produit sur la page produits. Remplissez les détails et définissez le type de paiement sur unique. Votre produit devrait ressembler à ceci :
Une fois que vous avez appuyé sur Ajouter un produit, vous devriez pouvoir voir votre produit dans la liste des produits. Si vous cliquez dessus et faites défiler jusqu'à la section Prix, vous pouvez trouver l'ID API pour l'élément de prix que vous avez créé — il devrait être quelque chose comme price_3ODP5…. Enregistrez-le dans une variable d'environnement (STRIPE_PRICE_ID) : vous en aurez besoin lors de la création de la session de paiement Stripe.
Nous devons créer un point de terminaison webhook que Stripe pourra appeler lorsqu'un paiement est terminé. Dans la page des webhooks, choisissez de tester dans l'environnement local. Cela vous permettra de transférer la demande vers une URL locale, comme http://127.0.0.1:8000. Commencez par télécharger la CLI Stripe. Ensuite, vous pouvez :
stripe login
stripe listen --forward-to http://127.0.0.1:8000/webhook > Ready! Your webhook signing secret is whsec_06531a7ba22363ac038f284ac547906b89e5c939f8d55dfd03a3619f9adc590a (^C to quit)
Cela garantit qu'une fois l'achat effectué, Stripe transmet les appels du webhook à votre point de terminaison local. La commande enregistrera un secret de signature de webhook, que vous devez également enregistrer en tant que variable d'environnement de projet (STRIPE_WEBHOOK_SECRET). Cela s'avérera utile pour vérifier qu'une requête provient bien de Stripe et que vous manipulez le bon webhook.
À la fin de cette section, vous devriez avoir quatre variables d'environnement Stripe. Vous pouvez maintenant les charger dans ecommerce_site/settings.py :
# ecommerce_site/settings.py import os from dotenv import load_dotenv load_dotenv() STRIPE_PUBLISHABLE_KEY = os.environ.get("STRIPE_PUBLISHABLE_KEY") STRIPE_SECRET_KEY = os.environ.get("STRIPE_SECRET_KEY") STRIPE_PRICE_ID = os.environ.get("STRIPE_PRICE_ID") STRIPE_WEBHOOK_SECRET = os.environ.get("STRIPE_WEBHOOK_SECRET")
Remarque : Nous utilisons python-dotenv pour charger les variables d'environnement.
Nous devons maintenant étendre les vues pour intégrer Stripe en créant une session de paiement, une vue d'achat réussi et une vue webhook.
Dans la vue d'achat, nous créerons une session de paiement Stripe si le formulaire d'achat est valide :
# ecommerce/views.py from django_htmx import HttpResponseClientRedirect from django.conf import settings import stripe @require_POST def purchase(request): form = OrderForm(request.POST) if form.is_valid(): quantity = form.cleaned_data["quantity"] # replace time.sleep(2) with the following code ⬇️ # 1 - set stripe api key stripe.api_key = settings.STRIPE_SECRET_KEY # 2 - create success url success_url = ( request.build_absolute_uri( reverse("purchase_success") ) + "?session_id={CHECKOUT_SESSION_ID}" ) # 3 - create cancel url cancel_url = request.build_absolute_uri(reverse("home")) # 4 - create checkout session checkout_session = stripe.checkout.Session.create( line_items=[ { "price": settings.STRIPE_PRICE_ID, "quantity": quantity, } ], mode="payment", success_url=success_url, cancel_url=cancel_url ) # 5 - redirect to checkout session url return HttpResponseClientRedirect(checkout_session.url) return render(request, "product.html", {"form": form})
Décomposons cela :
After completing the purchase, Stripe will redirect the customer to our specified success_url. Here, we can handle the post-purchase logic:
from django.shortcuts import redirect def purchase_success(request): session_id = request.GET.get("session_id") if session_id is None: return redirect("home") stripe.api_key = settings.STRIPE_SECRET_KEY try: stripe.checkout.Session.retrieve(session_id) except stripe.error.InvalidRequestError: messages.error(request, "There was a problem while buying your product. Please try again.") return redirect("home") return render(request, "purchase_success.html")
In this view, we first check if the session_id query parameter is present. If it is, we retrieve the corresponding session from Stripe using the secret key and the session_id. We then render the successful purchase template, which looks like this:
# ecommerce/templates/purchase_success.html {% extends "base.html" %} {% block content %} <section> <header> <h2>Thank you for your purchase</h2> <p> Your purchase was successful. You will receive an email with the details of your purchase soon. </p> </header> </section> {% endblock %}
You should also add it to the urlpatterns:
# ecommerce_site/urls.py # ... same imports as before urlpatterns = [ # ... same urls as before path("purchase_success", views.purchase_success, name="purchase_success"), # ⬅️ new ]
While the customer is in the purchase process, and before they are redirected to the success view, Stripe will call our webhook endpoint (remember to have the webhook listener running, as explained in the earlier 'Create the Webhook' section of this post):
from django.views.decorators.csrf import csrf_exempt from django.http import HttpResponse @csrf_exempt def webhook(request): stripe.api_key = settings.STRIPE_SECRET_KEY sig_header = request.headers.get('stripe-signature') payload = request.body event = None try: event = stripe.Webhook.construct_event( payload, sig_header, settings.STRIPE_WEBHOOK_SECRET ) except stripe.error.SignatureVerificationError: # Invalid signature return HttpResponse(status=400) # Handle the checkout.session.completed event if event.type == "checkout.session.completed": # TODO: create line orders return HttpResponse(status=200) return HttpResponse(status=400)
Let’s break this down:
Note: A Stripe event can have multiple types but we will only handle completed sessions in this post. However, you can (and should) extend a webhook by following the docs.
You should also add this view to urlpatterns:
# ecommerce_site/urls.py # ... same imports as before urlpatterns = [ # ... same urls as before path("webhook", views.webhook, name="webhook"), # ⬅️ new ]
If everything works well, once you click “buy”, you should be redirected to a Stripe payment page. Since we are in test mode, we can fill in the payment details with dummy data, like a 4242 4242 4242 4242 card:
Once you press Pay, Stripe should call the webhook view and redirect you to the purchase_success view. Congratulations, you have successfully processed a payment with Stripe!
Once a purchase is completed, we need to do a few things in the webhook view:
Let’s create a LineOrder database model in ecommerce/models.py to store some of the order information:
# ecommerce/models.py from django.db import models class LineOrder(models.Model): quantity = models.IntegerField() name = models.CharField(max_length=255, null=True, blank=True) email = models.EmailField(null=True, blank=True) shipping_details = models.TextField(null=True, blank=True) created_at = models.DateTimeField(auto_now_add=True) def __str__(self): return f"Order {self.id} - {self.quantity} units"
Remember to create and run the migrations:
python manage.py makemigrations # ⬅️ creates the migration files python manage.py migrate # ⬅️ applies the migrations in the database
We can now create a function to process the orders and call it from the webhook view:
# ecommerce/views.py @csrf_exempt def webhook(request): # ...same code as before if event.type == "checkout.session.completed": create_line_orders(event.data.object) # ⬅️ new return HttpResponse(status=200) return HttpResponse(status=400) # new ⬇️ def create_line_orders(session: stripe.checkout.Session): line_items = stripe.checkout.Session.list_line_items(session.id) for line_item in line_items.data: LineOrder.objects.create( name=session.customer_details.name, email=session.customer_details.email, shipping_details=session.shipping_details, quantity=line_item.quantity, ) mail.send_mail( "Your order has been placed", f""" Hi {session.customer_details.name}, Your order has been placed. Thank you for shopping with us! You will receive an email with tracking information shortly. Best, The one product e-commerce Team """, "from@example.com", [session.customer_details.email], ) staff_users = User.objects.filter(is_staff=True) mail.send_mail( "You have a new order!", """ Hi team! You have a new order in your shop! go to the admin page to see it. Best, The one product e-commerce Team """, "from@example.com", [user.email for user in staff_users], )
Let’s break this down:
You can now register the LineOrder model in the admin panel, so it’s accessible to staff users:
# ecommerce/admin.py from django.contrib import admin from ecommerce.models import LineOrder # Register your models here. admin.site.register(LineOrder)
When staff users log in to the admin page, they will now be able to check new orders and process them accordingly — in this case, pack and ship mugs to the customer!
Here are some tips to further improve on the store you've built:
Dans cette série en deux parties, nous avons réussi à créer un site de commerce électronique mono-produit en utilisant Django, htmx et Stripe. Ce guide vous a accompagné dans la configuration de votre projet Django, l'intégration de HTML pour des interactions utilisateur transparentes et l'intégration de paiements sécurisés avec Stripe.
Nous avons également expliqué comment gérer le traitement des commandes, notamment en enregistrant les informations de commande dans votre base de données, en informant les utilisateurs du personnel des nouveaux achats et en envoyant des e-mails de confirmation à vos clients. Grâce à ces bases, vous pouvez personnaliser et développer davantage votre site de commerce électronique en fonction de vos besoins spécifiques.
Bon codage !
P.S. Si vous souhaitez lire les articles Python dès leur sortie de presse, abonnez-vous à notre newsletter Python Wizardry et ne manquez jamais un seul article !
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!