it-roy-ru.com

AJAX Сбой запроса при отправке FormData (), включая ввод пустого файла в Safari 10.13.4

Я запускаю веб-приложение на основе Symfony 2.8, которое отправляет некоторые данные формы обратно в контроллер с помощью Ajax.

Пока все работало нормально, но с момента последнего обновления macOS до версии 10.13.4 пользователи начинают сообщать, что отправка формы больше не работает в Safari. Другие версии MacOS и другие браузеры на 10.13.4 по-прежнему работают нормально, поэтому в Safari это кажется проблемой. Конечно, я подал отчет об ошибке в Apple, но я не думаю, что когда-нибудь получу обратную связь оттуда ...

Мне удалось изолировать источник проблемы: не удается отправить данные, которые содержат пустой файл:

// safri_bug.html
<html>
    <head>
        <script src="https://ajax.googleapis.com/ajax/libs/jquery/3.3.1/jquery.min.js"></script>
    </head>
<body>
    <form name="app_booking" method="post" action="/test/submit.php">
        <div><input type="text" id="someValue" name="value"></div>
        <div><input id="thefile" type="file" name="file"></div>
    </form>

    <button id="bSubmit" type="button">Submit</button>

    <script>    
        $(document).ready(function() {              
            $('#bSubmit').click(function() {
                var form = $('form');
                var data = new FormData(form[0]);

                $.ajax({
                    url : '/submit.php',
                    type : 'POST',
                    data : data,
                    contentType: false,
                    processData: false,
                    context : this,
                    success : function(response) {
                            alert('success: ' + response);
                    },
                    error: function (xhr, ajaxOptions, thrownError) {
                            alert('error: ' + xhr.responseText + ' - ' + thrownError);
                    }
                });
            });
        });
    </script>
</body>
</html>


// submit.php
<?php 
    echo "OK";

Результат

  • Отправка формы отлично работает на всех протестированных браузерах и платформах, но в Safari в macOS 10.13.4
  • В Safari на macOS 10.13.4:
    • Если не выбран файл: Ajax-запрос выполняется в течение примерно 20 секунд (время ожидания?) И затем возвращается с пустым ответом об успешном завершении. submit.php вызываетНЕ
    • Если файл был выбран: все работает нормально ...

Итак, это, кажется, ошибка в последнем обновлении Safari? Или что-то не так с моим кодом?

Есть идеи, как предотвратить эту ошибку?

13
Andrei Herford

Тем временем я нашел это быстрое и грязное решение. Но на самом деле я ищу реальный обходной путь. Есть идеи?

// Filter out empty file just before the Ajax request
// Use try/catch since Safari < 10.13.4 does not support FormData.entries()
try {
   for (var pair of data.entries()) {
      if (pair[1] instanceof File && pair[1].name == '' && pair[1].size == 0)
         data.delete(pair[0]);  
   }
} catch(e) {}
1
Andrei Herford

Решение Андрея Херфорда приведет к сбою в других браузерах, которые не поддерживают метод entry () FormData - при использовании try/catch будут обнаружены только ошибки выполнения, а не синтаксические ошибки.

Нашим решением было использовать обычный JavaScript для удаления пустого элемента ввода файла перед / создания объекта FormData, таким образом:

for (i = 0; i < form.elements.length; i++) {
  if (form.elements[i].type == 'file') {
    if (form.elements[i].value == '') {
      form.elements[i].parentNode.removeChild(form.elements[i]);
    }
  }
}
6
lemonrock

Я использую FormData на своем сайте и могу убедиться, что это проблема последней версии Safari. Удаление пустого файла устраняет проблему. Вот код, который работал для меня:

  var form = $('#formID');
  var data = new FormData(form[0])

  //hack to fix safari bug where upload fails if file input is empty
  if (document.getElementById("fileID").files.length == 0 ) { //if the file is empty
      data.delete('fileID'); //remove it from the upload data
  }
6
Leopold Mc

Я использовал это решение и работает для меня.

var $form = $('#website_settings_form');
var $inputs = $('input[type="file"]:not([disabled])', $form); //select input files
    $inputs.each(function(_, input) {
        if (input.files.length > 0) return 
        $(input).prop('disabled', true) //if the input doesn't have uploaded files will be disable
    })
    var formData = new FormData($form[0]);// create the form data
    $inputs.prop('disabled', false);//enable fields again.
3
Antonio Reyes

Я использовал предложение Андрея, которое работало на сафари, но сломало IE.

Единственное решение, которое я смог найти, которое бы работало в обоих браузерах, - это использование eval ().

Так как это похоже на ошибку, влияющую только на Safari 11, я также добавил проверку версии браузера.

if(dataObj instanceof FormData && navigator.userAgent.match(/version\/11((\.[0-9]*)*)? .*safari/i)) {
    try {
        eval('for (var pair of dataObj.entries()) {\
            if (pair[1] instanceof File && pair[1].name === \'\' && pair[1].size === 0) {\
                dataObj.delete(pair[0]);\
            }\
        }');
    } catch(e) {}
}
0
Andrew Primeau