"use strict"; const {strict: assert} = require("assert"); const _ = require("lodash"); const {set_global, zrequire} = require("../zjsunit/namespace"); const {run_test} = require("../zjsunit/test"); const reload = set_global("reload", {}); zrequire("reload_state"); const channel = zrequire("channel"); const default_stub_xhr = "default-stub-xhr"; const $ = set_global("$", {}); function test_with_mock_ajax(test_params) { const {xhr = default_stub_xhr, run_code, check_ajax_options} = test_params; let ajax_called; let ajax_options; $.ajax = (options) => { $.ajax = undefined; ajax_called = true; ajax_options = options; options.simulate_success = (data, text_status) => { options.success(data, text_status, xhr); }; options.simulate_error = () => { options.error(xhr); }; return xhr; }; run_code(); assert(ajax_called); check_ajax_options(ajax_options); } run_test("post", () => { test_with_mock_ajax({ run_code() { channel.post({}); }, check_ajax_options(options) { assert.equal(options.type, "POST"); assert.equal(options.dataType, "json"); // Just make sure these don't explode. options.simulate_success(); options.simulate_error(); }, }); }); run_test("patch", () => { test_with_mock_ajax({ run_code() { channel.patch({}); }, check_ajax_options(options) { assert.equal(options.type, "POST"); assert.equal(options.data.method, "PATCH"); assert.equal(options.dataType, "json"); // Just make sure these don't explode. options.simulate_success(); options.simulate_error(); }, }); }); run_test("put", () => { test_with_mock_ajax({ run_code() { channel.put({}); }, check_ajax_options(options) { assert.equal(options.type, "PUT"); assert.equal(options.dataType, "json"); // Just make sure these don't explode. options.simulate_success(); options.simulate_error(); }, }); }); run_test("delete", () => { test_with_mock_ajax({ run_code() { channel.del({}); }, check_ajax_options(options) { assert.equal(options.type, "DELETE"); assert.equal(options.dataType, "json"); // Just make sure these don't explode. options.simulate_success(); options.simulate_error(); }, }); }); run_test("get", () => { test_with_mock_ajax({ run_code() { channel.get({}); }, check_ajax_options(options) { assert.equal(options.type, "GET"); assert.equal(options.dataType, "json"); // Just make sure these don't explode. options.simulate_success(); options.simulate_error(); }, }); }); run_test("normal_post", () => { const data = { s: "some_string", num: 7, lst: [1, 2, 4, 8], }; let orig_success_called; let orig_error_called; const stub_xhr = "stub-xhr-normal-post"; test_with_mock_ajax({ xhr: stub_xhr, run_code() { channel.post({ data, url: "/json/endpoint", success(data, text_status, xhr) { orig_success_called = true; assert.equal(data, "response data"); assert.equal(text_status, "success"); assert.equal(xhr, stub_xhr); }, error() { orig_error_called = true; }, }); }, check_ajax_options(options) { assert.equal(options.type, "POST"); assert.equal(options.dataType, "json"); assert.deepEqual(options.data, data); assert.equal(options.url, "/json/endpoint"); options.simulate_success("response data", "success"); assert(orig_success_called); options.simulate_error(); assert(orig_error_called); }, }); }); run_test("patch_with_form_data", () => { let appended; const data = { append(k, v) { assert.equal(k, "method"); assert.equal(v, "PATCH"); appended = true; }, }; test_with_mock_ajax({ run_code() { channel.patch({ data, processData: false, }); assert(appended); }, check_ajax_options(options) { assert.equal(options.type, "POST"); assert.equal(options.dataType, "json"); // Just make sure these don't explode. options.simulate_success(); options.simulate_error(); }, }); }); run_test("reload_on_403_error", () => { test_with_mock_ajax({ xhr: { status: 403, responseText: '{"msg": "CSRF Fehler: etwas", "code": "CSRF_FAILED"}', }, run_code() { channel.post({}); }, check_ajax_options(options) { let reload_initiated; reload.initiate = (options) => { reload_initiated = true; assert.deepEqual(options, { immediate: true, save_pointer: true, save_narrow: true, save_compose: true, }); }; options.simulate_error(); assert(reload_initiated); }, }); }); run_test("unexpected_403_response", () => { test_with_mock_ajax({ xhr: { status: 403, responseText: "unexpected", }, run_code() { channel.post({}); }, check_ajax_options(options) { blueslip.expect("error", "Unexpected 403 response from server"); options.simulate_error(); }, }); }); run_test("retry", () => { test_with_mock_ajax({ run_code() { channel.post({ idempotent: true, data: 42, }); }, check_ajax_options(options) { set_global("setTimeout", (f, delay) => { assert.equal(delay, 0); f(); }); blueslip.expect("log", "Retrying idempotent[object Object]"); test_with_mock_ajax({ run_code() { options.simulate_success(); }, check_ajax_options(options) { assert.equal(options.data, 42); }, }); }, }); }); run_test("too_many_pending", () => { $.ajax = () => { const xhr = "stub"; return xhr; }; blueslip.expect( "warn", "The length of pending_requests is over 50. " + "Most likely they are not being correctly removed.", ); _.times(50, () => { channel.post({}); }); }); run_test("xhr_error_message", () => { let xhr = { status: "200", responseText: "does not matter", }; let msg = "data added"; assert.equal(channel.xhr_error_message(msg, xhr), "data added"); xhr = { status: "404", responseText: '{"msg": "file not found"}', }; msg = "some message"; assert.equal(channel.xhr_error_message(msg, xhr), "some message: file not found"); });