Django выдает ошибку при наличии нескольких форм в наборах форм при выполнении запроса POST

#django #django-forms

#django #django-forms

Вопрос:

У меня есть проект django с панелью мониторинга и формами с наборами форм, встроенными в django admin. Я пытаюсь отправить запрос POST из форм в API. У меня есть два набора форм, которые должны быть конкретными. Когда у меня есть более одной формы в любом из наборов форм, я получаю приведенную ниже ошибку. Однако выполняется одна форма. Ниже приведен мой код и ошибка,

 Internal Server Error: /admin/campaign/add
Traceback (most recent call last):
  File "/usr/local/lib/python3.7/site-packages/django/core/handlers/exception.py", line 34, in inner
    response = get_response(request)
  File "/usr/local/lib/python3.7/site-packages/django/core/handlers/base.py", line 126, in _get_response
    response = self.process_exception_by_middleware(e, request)
  File "/usr/local/lib/python3.7/site-packages/django/core/handlers/base.py", line 124, in _get_response
    response = wrapped_callback(request, *callback_args, **callback_kwargs)
  File "/usr/local/lib/python3.7/site-packages/django/utils/decorators.py", line 142, in _wrapped_view
    response = view_func(request, *args, **kwargs)
  File "/usr/local/lib/python3.7/site-packages/django/views/decorators/cache.py", line 45, in _wrapped_view_func
    add_never_cache_headers(response)
  File "/usr/local/lib/python3.7/site-packages/django/utils/cache.py", line 252, in add_never_cache_headers
    patch_response_headers(response, cache_timeout=-1)
  File "/usr/local/lib/python3.7/site-packages/django/utils/cache.py", line 243, in patch_response_headers
    if not response.has_header('Expires'):
AttributeError: 'NoneType' object has no attribute 'has_header'
  

forms.py

 class CampaignForm(forms.Form):
    consumer = forms.CharField(label="Consumer", max_length=200)
    startDate = forms.DateTimeField(label="Start Date",
        input_formats=['%d/%m/%Y %H:%M'])
    endDate = forms.DateTimeField(label="End Date",
        input_formats=['%d/%m/%Y %H:%M'])
    referreeCredits = forms.IntegerField(label="Referree Credits")
    referrerCredits = forms.IntegerField(label="Referrer Credits")
    maxReferreeCredits = forms.IntegerField(label="Max Referree Credits")
    maxReferrerCredits = forms.IntegerField(label="Max Referrer Credits")
    message = forms.CharField(label="Message", max_length=200)
    kramerTemplateId = forms.CharField(label="Kramer Template ID", max_length=200)
    paymentMode = forms.ChoiceField(label="Payment Mode", choices=[("PAYTM","PAYTM")])

class RuleForm(forms.Form):
    eventName = forms.CharField(label="Event Name", max_length=200)
    operator = forms.ChoiceField(label="Operator", choices=[("EQUAL","EQUAL"), ("EVERY","EVERY")])
    value = forms.IntegerField(label="Value")

class MilestoneRulesForm(forms.Form):
    operator = forms.ChoiceField(label="Operator", choices=[("EQUAL","EQUAL"), ("EVERY","EVERY")])
    value = forms.IntegerField(label="Value")
    referrerCredits = forms.IntegerField(label="Referrer Credits")
  

views.py

 def campaign_add(self, request):
        form = CampaignForm()
        RuleFormSet = formset_factory(RuleForm)
        MilestoneFormSet = formset_factory(MilestoneRulesForm)
        if request.method == 'POST':
            # import ipdb; ipdb.set_trace()
            form = CampaignForm(request.POST)
            rule_formset = RuleFormSet(request.POST, prefix='rules')
            milestone_formset = MilestoneFormSet(request.POST, prefix='milestones')
            if form.is_valid() and rule_formset.is_valid() and milestone_formset.is_valid():
                    import ipdb; ipdb.set_trace()
                    dat = {}
                    dat["consumer"] = form.cleaned_data["consumer"]
                    dat["startDate"] = self.datetime_to_epoch(form.cleaned_data["startDate"])
                    dat["endDate"] = self.datetime_to_epoch(form.cleaned_data["endDate"])
                    dat["referreeCredits"] = form.cleaned_data["referreeCredits"]
                    dat["referrerCredits"] = form.cleaned_data["referrerCredits"]
                    dat["maxReferreeCredits"] = form.cleaned_data["maxReferreeCredits"]
                    dat["maxReferrerCredits"] = form.cleaned_data["maxReferrerCredits"]
                    dat["message"] = form.cleaned_data["message"]
                    dat["kramerTemplateId"] = form.cleaned_data["kramerTemplateId"]
                    dat["paymentMode"] = form.cleaned_data["paymentMode"]
                    arrRules = rule_formset.cleaned_data
                    arrMilestoneRules = milestone_formset.cleaned_data
                    dat["eventRules"] = arrRules
                    dat["milestoneRules"] = arrMilestoneRules
                    print(rule_formset.cleaned_data)
                    print(milestone_formset.cleaned_data)
                    res = requests.post("https://example.com", data=json.dumps(dat), headers={'content-type': 'application/json'})
                    if res.status_code == 201 or res.status_code == 200:
                        messages.success(request, 'Success!')
                        return redirect("admin:campaign-pg1")
                    else:
                        messages.error(request, 'Submission Failed.')
        else:
            rule_formset = RuleFormSet(prefix="rules")
            milestone_formset = MilestoneFormSet(prefix="milestones")
            return TemplateResponse(request, "admin/campaign_add.html", {"form":form, "rule_formset": rule_formset, "milestone_formset": milestone_formset})
  

Html шаблон:

 {% block content %}
    <div class="form_div" style="width:60%;">
        <h2>Campaign Form</h2>
        <form method="POST">
            {% csrf_token %}
            {% for field in form.visible_fields %}
                <div class="form-group">
                    {{ field.label_tag }}
                    {% render_field field class="form-control" %}
                    {% if field.help_text %}
                        <small class="form-text text-muted">{{ field.help_text }}</small>
                    {% endif %}
                </div>
            {% endfor %}            
            <div style="display: flex; justify-content: space-between; width: 280px;">
                <h3>Rules</h3>
                <button type="button" id="generate-rule" class="btn btn-primary"> </button>
            </div>
            <div class="parent">
            <div>
            {{ rule_formset.management_form }}
            {% for ruleForm in rule_formset %}
                {{ ruleForm }}
                <input type="button"  class="btn btn-warn" style="background: red;" value="-">
            {% endfor %}
            </div>
            </div>
             <div style="display: flex; justify-content: space-between; width: 280px;">
                <h3>Milestone Rules</h3>
                <button  type="button" id="generate-milestone-rule" class="btn btn-primary"> </button>
            </div>
            <div class="milestone-parent">
            <div>
            {{ milestone_formset.management_form }}
            {% for milestoneForm in milestone_formset %}
                {{ milestoneForm }}
                <input type="button"  class="btn btn-warn" style="background: red;" value="-">
            {% endfor %}
            </div>
            </div>
            <div>
            <input type="submit" value="Post">
            </div>
        </form>
    </div>

    <script>
        $(function () {
            $("#id_startDate").datetimepicker({
            format: 'd/m/Y H:i',
            });
            $("#id_endDate").datetimepicker({
            format: 'd/m/Y H:i',
            });
        });
    </script>

    <script type='text/javascript'>

        const ruleform = `
        <div>
            <label for="id_rules-{n}-eventName">Event Name:</label>
            <input type="text" name="rules-{n}-eventName" maxlength="200" id="id_rules-{n}-eventName">

            <label for="id_rules-{n}-operator">Operator:</label>
            <select name="rules-{n}-operator" id="id_rules-{n}-operator">
                <option value="EQUAL">EQUAL</option>
                <option value="EVERY">EVERY</option>
            </select>

            <label for="id_rules-{n}-value">Value:</label>
            <input type="number" name="rules-{n}-value" id="id_rules-{n}-value">
            <input type="button" class="btn btn-warn" style="background: red;" value="-">
        </div>
        `
        const milestoneform = `
        <div>
            <label for="id_milestones-{n}-operator">Operator:</label>
            <select name="milestones-{n}-operator" id="id_milestones-{n}-operator">
                <option value="EQUAL">EQUAL</option>
                <option value="EVERY">EVERY</option>
            </select>

            <label for="id_milestones-{n}-value">Value:</label>
            <input type="number" name="milestones-{n}-value" id="id_milestones-{n}-value">

            <label for="id_milestones-{n}-mreferrerCredits">Referrer Credits:</label>
            <input type="number" name="milestones-{n}-mReferrerCredits" maxlength="200" id="id_milestones-{n}-mReferrerCredits">

            <input type="button" class="btn btn-warn" style="background: red;" value="-">
        </div>
        `
        $("#generate-rule").on('click', function(e){
            var n = $("#id_rules-TOTAL_FORMS").val(); // current number of forms;
            var new_form = ruleform.replace(/{n}/g, n); // replace all {n} by the new form number
            $(".parent").append(new_form);
            $("#id_rules-TOTAL_FORMS").val(parseInt(n)   1); // update the total forms number
            console.log($("#id_rules-TOTAL_FORMS").val());
        })
        $("#generate-milestone-rule").on('click', function(e){
             var n = $("#id_milestones-TOTAL_FORMS").val(); // current number of forms;
            var new_form = milestoneform.replace(/{n}/g, n); // replace all {n} by the new form number
            $(".milestone-parent").append(new_form);
            $("#id_milestones-TOTAL_FORMS").val(parseInt(n)   1); // update the total forms number
            console.log($("#id_milestones-TOTAL_FORMS").val());
        })
        $(".parent").on("click", (e) => {
            //n = $("#id_rules-TOTAL_FORMS").val();
            //$("#id_rules-TOTAL_FORMS").val(parseInt(n)-1);
            //console.log($("#id_rules-TOTAL_FORMS").val());
            return e.target.classList.contains('btn-warn') amp;amp; e.target.parentNode.remove()
        })

        $(".milestone-parent").on("click", (e) => {
            //n = $("#id_milestones-TOTAL_FORMS").val();
            //$("#id_milestones-TOTAL_FORMS").val(parseInt(n)-1);
            //console.log($("#id_milestones-TOTAL_FORMS").val());
            return e.target.classList.contains('btn-warn') amp;amp; e.target.parentNode.remove()
        })
    </script>
{% endblock %}
  

Как мне устранить эту ошибку, чтобы я мог выполнить запрос POST с любым количеством форм в моих наборах форм?

Редактировать: вот как выглядит мой набор форм. Когда у меня более одной формы, я получаю указанную выше ошибку. мои наборы форм

Ответ №1:

может ли быть так, что вы достигаете условия, когда код состояния вашего запроса недействителен (не 200 или 201)? В этом случае вы не возвращаете никакого ответа, который вы просто вызываете:

 else:
    messages.error(request, 'Submission Failed.')
  

Трассировка стека вашего исключения предполагает, что нет объекта ответа (вы не возвращаете никакого ответа из представления)

 File "/usr/local/lib/python3.7/site-packages/django/utils/cache.py", line 243, in patch_response_headers
    if not response.has_header('Expires'):
AttributeError: 'NoneType' object has no attribute 'has_header'
  

Проверьте, почему не удается выполнить вызов API, и решите проблему, после чего вы сможете отправить свою форму.

Комментарии:

1. Привет. Но почему я получаю ошибку, когда у меня есть более одной формы в правилах и правилах milestone, тогда как запрос выполняется, когда у меня есть только одна форма в том же.

2. Похоже, что большинство полей вашей формы являются обязательными полями. Возможно, если у вас более одной формы, вы отправляете пустые формы, которые не пройдут проверку. Единственный способ выяснить это — посмотреть на все ваши атрибуты formsets [‘errors’] (с pdb или с помощью инструкции печати), а также на ваш атрибут form [‘errors’]

Ответ №2:

По умолчанию formset_factory() определяет одну дополнительную форму. Количество отображаемых пустых форм контролируется дополнительным параметром.

 MilestoneFormSet = formset_factory(MilestoneForm, extra=2)
RuleFormSet = formset_factory(RuleForm, extra=2)