//! Test suite for the Web and headless browsers.

//XXXXXXXXXXXXXXX
//XXX DOES NOT WORK.

#![cfg(target_arch = "wasm32")]

extern crate alloc;
extern crate wasm_bindgen_test;
use core::sync::atomic::AtomicBool;
use veilid_wasm::*;
use wasm_bindgen_test::*;

wasm_bindgen_test_configure!(run_in_browser);

static SETUP_ONCE: AtomicBool = AtomicBool::new(false);
pub fn setup() -> () {
    if SETUP_ONCE
        .compare_exchange(false, true, Ordering::Relaxed, Ordering::Relaxed)
        .is_ok()
    {
        console_log("setup()");
        console_error_panic_hook::set_once();
        wasm_logger::init(wasm_logger::Config::new(Level::Trace));
        init_callbacks();
    }
}
// xxx needs updating to new keys and veilid_api object
fn init_callbacks() {
    assert_eq!(js_sys::eval(r#"
    window.sleep = (milliseconds) => { return new Promise(resolve => setTimeout(resolve, milliseconds)) };
    window.stateChangeCallback = async (stateChange) => { console.log("State change: " + stateChange); };
    window.configCallback = (configKey) => { 
        switch(configKey) {
            case "namespace": return "";
            case "capabilities.protocol_udp": return false;
            case "capabilities.protocol_connect_tcp": return false;
            case "capabilities.protocol_accept_tcp": return false;
            case "capabilities.protocol_connect_ws": return true;
            case "capabilities.protocol_accept_ws": return false;
            case "capabilities.protocol_connect_wss": return true;
            case "capabilities.protocol_accept_wss": return false;
            case "tablestore.directory": return "";
            case "network.routing_table.node_id": return [];
            case "network.routing_table.node_id_secret": return [];
            case "network.routing_table.bootstrap": return [];
            case "network.routing_table.limit_over_attached": return 64;
            case "network.routing_table.limit_fully_attached": return 32;
            case "network.routing_table.limit_attached_strong": return 16;
            case "network.routing_table.limit_attached_good": return 8;
            case "network.routing_table.limit_attached_weak": return 4;
            case "network.rpc.concurrency": return 2;
            case "network.rpc.queue_size": return 128;
            case "network.rpc.max_timestamp_behind": return 10000000;
            case "network.rpc.max_timestamp_ahead": return 10000000;
            case "network.rpc.timeout": return 10000000;
            case "network.rpc.max_route_hop_count": return 4;
            case "network.rpc.default_route_hop_count": return 1;
            case "network.dht.max_find_node_count": return 20;
            case "network.dht.resolve_node_timeout": return 10000;
            case "network.dht.resolve_node_count": return 1;
            case "network.dht.resolve_node_fanout": return 4;
            case "network.dht.get_value_timeout": return 10000;
            case "network.dht.get_value_count": return 3;
            case "network.dht.get_value_fanout": return 4;
            case "network.dht.set_value_timeout": return 10000;
            case "network.dht.set_value_count": return 5;
            case "network.dht.set_value_fanout": return 4;
            case "network.dht.min_peer_count": return 20;
            case "network.dht.min_peer_refresh_time": return 2000000;
            case "network.dht.validate_dial_info_receipt_time": return 5000000;
            case "network.upnp": return false;
            case "network.detect_address_changes": return true;
            case "network.address_filter": return true;
            case "network.restricted_nat_retries": return 3;
            case "network.tls.certificate_path": return "";
            case "network.tls.private_key_path": return "";
            case "network.application.path": return "/app";
            case "network.application.https.enabled": return false;
            case "network.application.https.listen_address": return "";
            case "network.application.http.enabled": return false;
            case "network.application.http.listen_address": return "";
            case "network.protocol.udp.enabled": return false;
            case "network.protocol.udp.socket_pool_size": return 0;
            case "network.protocol.udp.listen_address": return "";
            case "network.protocol.udp.public_address": return "";
            case "network.protocol.tcp.connect": return false;
            case "network.protocol.tcp.listen": return false;
            case "network.protocol.tcp.max_connections": return 32;
            case "network.protocol.tcp.listen_address": return "";
            case "network.protocol.tcp.public_address": return "";
            case "network.protocol.ws.connect": return true;
            case "network.protocol.ws.listen": return false;
            case "network.protocol.ws.max_connections": return 16;
            case "network.protocol.ws.listen_address": return "";
            case "network.protocol.ws.path": return "/ws";
            case "network.protocol.ws.public_address": return "";
            case "network.protocol.wss.connect": return true;
            case "network.protocol.wss.listen": return false;
            case "network.protocol.wss.max_connections": return 16;
            case "network.protocol.wss.listen_address": return "";
            case "network.protocol.wss.path": return "/ws";
            case "network.protocol.wss.public_address": return "";
            default: 
                console.log("config key '" + key +"' doesn't exist"); break;
        }
    };
    true
    "#).expect("failed to eval"), JsValue::TRUE);
}

///////////////////////////////////////////////////////////////////////////////////////////////////
///

#[wasm_bindgen_test]
fn test_construct() {
    setup();

    assert_eq!(
        js_sys::eval(
            r#"
        let vc = new VeilidCore();
        true
    "#
        )
        .expect("failed to eval"),
        JsValue::TRUE
    );
}

#[wasm_bindgen_test(async)]
async fn test_startup_shutdown() {
    setup();

    assert_eq!(
        JsFuture::from(
            js_sys::eval(
                r#"
        (async function() {
            let vc = new VeilidCore();
            await vc.startup(window.stateChangeCallback, window.configCallback);
            await vc.shutdown();
            return true;
        })().then(v => {
            console.log("finished: " + v);
            return v;
        });
    "#
            )
            .expect("failed to eval")
            .dyn_into::<Promise>()
            .unwrap()
        )
        .await,
        Ok(JsValue::TRUE)
    );
}

#[wasm_bindgen_test(async)]
async fn test_attach_detach() {
    setup();

    assert_eq!(
        JsFuture::from(
            js_sys::eval(
                r#"
        (async function() {
            let vc = new VeilidCore();
            await vc.startup(window.stateChangeCallback, window.configCallback);
            await vc.attach();
            await window.sleep(1000);
            await vc.detach();
            await vc.shutdown();
            return true;
        })().then(v => {
            console.log("finished: " + v);
            return v;
        });
    "#
            )
            .expect("failed to eval")
            .dyn_into::<Promise>()
            .unwrap()
        )
        .await,
        Ok(JsValue::TRUE)
    );
}