Как подделать ответ jquery.ajax ()? - PullRequest
37 голосов
/ 11 марта 2011

Я пишу несколько тестов QUnit для JavaScript, который делает вызовы AJAX.

Для изоляции я перезаписываю $.ajax, чтобы записать массив параметров вызова AJAX в переменную.Это работает для проверки того, как методы используют функции AJAX, но у меня возникают трудности при тестировании обработчика успеха $.load()

Из документации на http://api.jquery.com/load/:

Когда обнаружен успешный ответ (т. Е. Когда textStatus имеет значение «success» или «notmodified»), .load () устанавливает содержимое HTML сопоставленного элемента в возвращаемые данные.

Итак, я попыталсявернуть объект, содержащий объекты с тем же именем, что и переменные для обработчика успеха:

    //Mock ajax function
    $.ajax = function (param) {
        _mockAjaxOptions = param;
        var fakeAjaxSuccess = { responseText: "success", textStatus: "success", XMLHttpRequest: "success" };
        return fakeAjaxSuccess;
    };

Но этот подход не сработал.

Как мне воспроизвести поведение успешного AJAX?звоните

Ответы [ 6 ]

23 голосов
/ 16 апреля 2015

Этот вопрос имеет несколько лет, и для новых версий jQuery немного изменился.

Чтобы сделать это с Jasmin, вы можете попробовать подход Майкла Фалаги

Раствор

  function ajax_response(response) {
    var deferred = $.Deferred().resolve(response);
    return deferred.promise();
  }

С жасмином

  describe("Test test", function() {
    beforeEach(function() {
      spyOn($, 'ajax').and.returnValue(
        ajax_response([1, 2, 3])
      );
    });
    it("is it [1, 2, 3]", function() {
      var response;
      $.ajax('GET', 'some/url/i/fancy').done(function(data) {
        response = data;
      });
      expect(response).toEqual([1, 2, 3]);
    });
  });

Без жасмина

  $.ajax = ajax_response([1, 2, 3]);
  $.ajax('GET', 'some/url/i/fancy').done(function(data) {
     console.log(data); // [1, 2, 3]
  });
14 голосов
/ 11 марта 2011

После чтения, вдохновленного @Robusto и @Val, я нашел метод, который работает:

//Mock ajax function
$.ajax = function (param) {
    _mockAjaxOptions = param;
    //call success handler
    param.complete("data", "textStatus", "jqXHR");
};

Вместо того, чтобы вызывать событие из любого реального $ .ajax-кода или вызывать какие-либо события, у меня есть мойфальшивый объект ajax вызывает функцию (которая передается в качестве параметра $.ajax()) как часть моей фальшивой функции.

11 голосов
/ 24 сентября 2015

Используйте закрытие, чтобы переопределить $.ajax с фиктивным ответом

После попытки принятого ответа и ответа, отправленного пользователем1634074 , я разработал этот простой игибкое сочетание двух.

В его наиболее простой форме…

function ajax_response(response) {
  return function (params) {
    params.success(response);
  };
}
$.ajax = ajax_response('{ "title": "My dummy JSON" }');

В приведенном выше примере определите функцию ajax_response(), которая принимает некоторую строку JSON какаргумент (или любое количество пользовательских аргументов, полезных для имитации ответа) и возвращает анонимную функцию закрытия, которая будет назначена на $.ajax в качестве переопределения для модульного тестирования.

Анонимная функция принимает paramsаргумент, который будет содержать объект настроек, переданный в функцию $.ajax.И он использует аргументы, переданные внешней функции, для имитации ответа от сервера.В этом примере он всегда симулирует успешный ответ от сервера, просто вызывая обратный вызов success и предоставляя ему фиктивный JSON.

Его легко перенастроить…

function ajax_response(response, success) {
  return function (params) {
    if (success) {
      params.success(response);
    } else {
      params.error(response);
    }
  };
}

// Simulate success
$.ajax = ajax_response('{ "title": "My dummy JSON." }', true); 
doAsyncThing(); // Function that calls $.ajax

// Simulate error
$.ajax = ajax_response('{ "error": "Who is the dummy now?" }', false); 
doAsyncThing(); // Function that calls $.ajax

Ниже мы видим это в действии…

/* FUNCTION THAT MAKES AJAX REQUEST */
function doAsyncThing() {
  $.ajax({
    type: "POST",
    url: "somefile.php",
    // data: {…},
    success: function (results) {
      var json = $.parseJSON(results),
          html = $('#ids').html();
      $('#ids').html(html + '<br />' + json.id);
    }
  });
}

/* BEGIN MOCK TEST */
// CREATE CLOSURE TO RETURN DUMMY FUNCTION AND FAKE RESPONSE
function ajax_response(response) {
  return function (params) {
    params.success(response);
  };
}

var n = prompt("Number of AJAX calls to make", 10);

for (var i = 1; i <= n; ++i) {
  
  // OVERRIDE $.ajax WITH DUMMY FUNCTION AND FAKE RESPONSE
  $.ajax = ajax_response('{ "id": ' + i + ' }');
  doAsyncThing();
}
/* END MOCK TEST */
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<p id="ids">IDs:</p>
4 голосов
/ 11 января 2016

Копируйте $ .ajax по мере необходимости, не мешая jQuery

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

Объект API использует $ .ajax, поэтому вы можете вызвать метод API следующим образом:

api.products({ price: { $lt: 150, tags: ['nike', 'shoes'] } })
.done(function(json) {
  // do something with the data
})
.error(function(err) {
  // handle error
});

Этот метод делает свое дело:

function mockAjax(options) {
  var that = {
    done: function done(callback) {
      if (options.success)
        setTimeout(callback, options.timeout, options.response);
      return that;
    },
    error: function error(callback) {
      if (!options.success)
        setTimeout(callback, options.timeout, options.response);
      return that;
    }
  };
  return that;
}

Затем переопределите один вызов API, не касаясь $ .ajax:

api.products = function() {
  return mockAjax({
    success: true,
    timeout: 500,
    response: {
      results: [
        { upc: '123123', name: 'Jordans' },
        { upc: '4345345', name: 'Wind Walkers' }
      ]
    }
  });
};

https://jsfiddle.net/Lsf3ezaz/2/

1 голос
/ 11 марта 2011

Я думаю, что ссылка ниже должна помочь.Что касается параметра, я не уверен, но это может быть.

$.fn.ajax.success =  function (){
  ///the rest goest here
}

Переопределить функцию jQuery .val ()?

1 голос
/ 11 марта 2011

Посмотрите документацию по jQuery: вы увидите, что в настройке Ajax есть ряд других условий, которые проверяются. Если вы заставите их всех указать на ваш fakeAjaxSuccess, вы можете достичь своей цели.

В качестве альтернативы, оберните ваш вызов $.ajax в его собственную функцию и используйте любые вызовы, которые он вызывает, просто вызовите ваш обработчик событий с объектом fakeAjaxSuccess.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...