Django 并在不同视图中向外部 API 发布请求

dld*_*ian 3 javascript python api django django-rest-framework

所以我想创建一个 Django 应用程序,我的用户可以在其中通过表单发布数据并向外部 API 发出发布请求,但在同一页面/视图中获得响应

例如,我有我的看法

class Home(TemplateView):
    template_name: 'home/index.html'
Run Code Online (Sandbox Code Playgroud)

我有我的 index.html:

<form id="formdata" >
    <select id="options">
        <option id="sku">Option 1</option>
        <option id="sku2">Option 2</option>
    </select>    

    <input name="number" type="text"  id="number">

    <select id="price">
        <option id="price1">$5</option>
        <option id="price2">%10</option>
    </select>    

    <button type="button" data-loading-text="enviando..." onclick="submitInfo()">Send</button>
</form>
Run Code Online (Sandbox Code Playgroud)

让我们忽略 HTML 可能是错误的事实,它是具有选择和输入字段的表单的基本结构,但请注意,我需要在 post 请求中传递“产品”、“数字”和“价格”作为参数。

问题是,当用户点击提交按钮时,他们向外部 api 发出发布请求,我知道我可以使用 fetch 使用 JavaScript 执行 int,但问题是我需要在正文中传递我的个人令牌密钥params,我也想隐藏真正的api url,用我的网站的url来隐藏它,例如:www.myurl.com/my-api-call

所以我正在考虑创建一个带有 post 请求的“external_api_view”,如下所示:

import requests
import time
from rest_framework import status
from rest_framework.response import Response

def external_api_view(request):
    if request.method == "POST":
        attempt_num = 0  # keep track of how many times we've retried
        while attempt_num < MAX_RETRIES:
            url = 'www.apiexternal.com/endpoint'
            payload = {'Token':'My_Secret_Token','product':'product_select_in_form','price':'price_selected_in_form'}
            response = requests.post(url, data = payload)
            if r.status_code == 200:
                data = r.json()
                return Response(data, status=status.HTTP_200_OK)
            else:
                attempt_num += 1
                # You can probably use a logger to log the error here
                time.sleep(5)  # Wait for 5 seconds before re-trying
        return Response({"error": "Request failed"}, status=r.status_code)
    else:
        return Response({"error": "Method not allowed"}, status=status.HTTP_400_BAD_REQUEST)
Run Code Online (Sandbox Code Playgroud)

但不是我有问题,我不知道如何将表单的输入传递到该视图中,所以我可以发出发布请求并通过我的 index.html 文件上的 javascript 获取响应(显然添加了需要的 javascript)

我什至不知道这是否可能,我正在考虑用休息框架做类似的事情,但我也不知道如何

任何帮助将非常感激 :)

小智 6

首先向您的表单添加一个操作属性和方法属性。然后添加一个 csrf 令牌以确保安全。还要向 select 元素添加名称属性。

<form method="post" action="/external" id="formdata" >
  {% csrf_token %}
<select name="options" id="options">
    <option id="sku">Option 1</option>
    <option id="sku2">Option 2</option>
</select>    

<input name="number" type="text"  id="number">

<select name="price" id="price">
    <option id="price1">$5</option>
    <option id="price2">%10</option>
</select>    

<button type="button" data-loading-text="enviando..." onclick="submitInfo()">Send</button>
Run Code Online (Sandbox Code Playgroud)

接下来为您添加的操作添加 url 路径。网址.py:

from django.urls import path
from . import views

app_name = "main"   

urlpatterns = [
    ...
    path("external", views.external_api_view, name="home")

]
Run Code Online (Sandbox Code Playgroud)

然后在views.py中获取输入值

import requests
import time
from rest_framework import status
from rest_framework.response import Response

def external_api_view(request):
    if request.method == "POST":
        attempt_num = 0  # keep track of how many times we've retried
        while attempt_num < MAX_RETRIES:
            url = 'www.apiexternal.com/endpoint'
            payload = {'Token':'My_Secret_Token','product':request.POST.get("options"),'price':request.POST.get("price")}
            r = requests.post(url, data = payload)
            if r.status_code == 200:
                data = r.json()
                return Response(data, status=status.HTTP_200_OK)
            else:
                attempt_num += 1
                # You can probably use a logger to log the error here
                time.sleep(5)  # Wait for 5 seconds before re-trying
        return Response({"error": "Request failed"}, status=r.status_code)
    else:
        return Response({"error": "Method not allowed"}, status=status.HTTP_400_BAD_REQUEST)

Run Code Online (Sandbox Code Playgroud)