#include "ConnectRequest.h"
#include "Error.h"
#include "ConnectResponse.h"
#include <panda/encode/base64.h>
#include <panda/log.h>

namespace panda { namespace protocol { namespace websocket {

void ConnectRequest::process_headers () {
    bool ok;

    if (_method != Method::Get) {
        _error = errc::method_mustbe_get;
        return;
    }

    if (http_version != 11) {
        _error = errc::http_1_1_required;
        return;
    }

    if (!body.empty()) {
        _error = errc::body_prohibited;
        return;
    }

    auto it = headers.find("Connection");
    if (it == headers.end() || !string_contains_ci(it->value, "upgrade")) {
        _error = errc::connection_mustbe_upgrade;
        return;
    }

    it = headers.find("Upgrade");
    if (it == headers.end() || !string_contains_ci(it->value, "websocket")) {
        _error = errc::upgrade_mustbe_websocket;
        return;
    }

    ok = false;
    it = headers.find("Sec-WebSocket-Key");
    if (it != headers.end()) {
        _ws_key = it->value;
        auto decoded = panda::encode::decode_base64(_ws_key);
        if (decoded.length() == 16) ok = true;
    }
    if (!ok) {
        _error = errc::sec_accept_missing;
        return;
    }

    _ws_version_supported = false;
    it = headers.find("Sec-WebSocket-Version");
    if (it != headers.end()) {
        it->value.to_number(_ws_version);
        for (int v : supported_ws_versions) {
            if (_ws_version != v) continue;
            _ws_version_supported = true;
            break;
        }
    }
    if (!_ws_version_supported) {
        _error = errc::unsupported_version;
        return;
    }

    auto ext_range = headers.get_multi("Sec-WebSocket-Extensions");
    for (auto& val : ext_range) {
        parse_header_value(val, _ws_extensions);
    }

    _ws_protocol = headers.get("Sec-WebSocket-Protocol");
}

void ConnectRequest::add_deflate(const DeflateExt::Config& cfg) {
    DeflateExt::request(_ws_extensions, cfg);
}

string ConnectRequest::to_string() {
    if (!uri || !uri->host()) {
        throw Error("HTTPRequest[to_string] uri with net location must be defined");
    }
    if (uri && uri->scheme() && uri->scheme() != "ws" && uri->scheme() != "wss") {
        throw Error("ConnectRequest[to_string] uri scheme must be 'ws' or 'wss'");
    }
    if (body.length()) {
        throw Error("ConnectRequest[to_string] http body is not allowed for websocket handshake request");
    }

    _method = Request::Method::Get;

    if (!_ws_key) {
        int32_t keybuf[] = {std::rand(), std::rand(), std::rand(), std::rand()};
        _ws_key = panda::encode::encode_base64(string_view((const char*)keybuf, sizeof(keybuf)), false, true);
    }
    headers.set("Sec-WebSocket-Key", _ws_key);

    if (_ws_protocol) headers.set("Sec-WebSocket-Protocol", _ws_protocol);

    if (!_ws_version) _ws_version = 13;
    headers.set("Sec-WebSocket-Version", string::from_number(_ws_version));

    if (_ws_extensions.size()) headers.set("Sec-WebSocket-Extensions", compile_header_value(_ws_extensions));

    headers.set("Connection", "Upgrade");
    headers.set("Upgrade", "websocket");

    if (!headers.has("User-Agent")) headers.add("User-Agent", "Panda-WebSocket");
    if (!headers.has("Host"))       headers.add("Host", uri->host());

    return http::Request::to_string();
}

http::ResponseSP ConnectRequest::new_response() const{
    return new ConnectResponse();
}


}}}