/* Copyright (C) 2019 Mr Goldberg
   This file is part of the Goldberg Emulator

   The Goldberg Emulator is free software; you can redistribute it and/or
   modify it under the terms of the GNU Lesser General Public
   License as published by the Free Software Foundation; either
   version 3 of the License, or (at your option) any later version.

   The Goldberg Emulator is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
   Lesser General Public License for more details.

   You should have received a copy of the GNU Lesser General Public
   License along with the Goldberg Emulator; if not, see
   <http://www.gnu.org/licenses/>.  */

#include "base.h"
#include <queue>

class Steam_Game_Coordinator :
public ISteamGameCoordinator
{
    class Settings *settings;
    class Networking *network;
    class SteamCallResults *callback_results;
    class SteamCallBacks *callbacks;
    class RunEveryRunCB *run_every_runcb;

    static const uint32 protobuf_mask = 0x80000000;
    std::queue<std::string> outgoing_messages;

void push_incoming(std::string message)
{
    outgoing_messages.push(message);

    struct GCMessageAvailable_t data;
    data.m_nMessageSize = message.size();
    callbacks->addCBResult(data.k_iCallback, &data, sizeof(data));
}

public:
static void steam_callback(void *object, Common_Message *msg)
{
    PRINT_DEBUG("steam_gamecoordinator_callback\n");

    Steam_Game_Coordinator *steam_gamecoordinator = (Steam_Game_Coordinator *)object;
    steam_gamecoordinator->Callback(msg);
}

static void steam_run_every_runcb(void *object)
{
    PRINT_DEBUG("steam_gamecoordinator_run_every_runcb\n");

    Steam_Game_Coordinator *steam_gamecoordinator = (Steam_Game_Coordinator *)object;
    steam_gamecoordinator->RunCallbacks();
}

Steam_Game_Coordinator(class Settings *settings, class Networking *network, class SteamCallResults *callback_results, class SteamCallBacks *callbacks, class RunEveryRunCB *run_every_runcb)
{
    this->settings = settings;
    this->network = network;
    this->run_every_runcb = run_every_runcb;
    //this->network->setCallback(CALLBACK_ID_USER_STATUS, settings->get_local_steam_id(), &Steam_Game_Coordinator::steam_callback, this);
    this->run_every_runcb->add(&Steam_Game_Coordinator::steam_run_every_runcb, this);

    this->callback_results = callback_results;
    this->callbacks = callbacks;
}

~Steam_Game_Coordinator()
{
    //TODO rm network callbacks
    this->run_every_runcb->remove(&Steam_Game_Coordinator::steam_run_every_runcb, this);
}

// sends a message to the Game Coordinator
EGCResults SendMessage_( uint32 unMsgType, const void *pubData, uint32 cubData )
{
    PRINT_DEBUG("Steam_Game_Coordinator::SendMessage %X %u len %u\n", unMsgType, (~protobuf_mask) & unMsgType, cubData);
    if (protobuf_mask & unMsgType) {
        uint32 message_type = (~protobuf_mask) & unMsgType;
        if (message_type == 4006) { //client hello
            std::string message("\xA4\x0F\x00\x80\x00\x00\x00\x00\x08\x00", 10);
            push_incoming(message);
        } else
        if (message_type == 4007) { //server hello
            std::string message("\xA5\x0F\x00\x80\x00\x00\x00\x00\x08\x00", 10);
            push_incoming(message);
        }
    }

    return k_EGCResultOK;
}

// returns true if there is a message waiting from the game coordinator
bool IsMessageAvailable( uint32 *pcubMsgSize )
{
    PRINT_DEBUG("Steam_Game_Coordinator::IsMessageAvailable\n");
    if (outgoing_messages.size()) {
        if (pcubMsgSize) *pcubMsgSize = outgoing_messages.front().size();
        return true;
    } else {
        return false;
    }
}

// fills the provided buffer with the first message in the queue and returns k_EGCResultOK or 
// returns k_EGCResultNoMessage if there is no message waiting. pcubMsgSize is filled with the message size.
// If the provided buffer is not large enough to fit the entire message, k_EGCResultBufferTooSmall is returned
// and the message remains at the head of the queue.
EGCResults RetrieveMessage( uint32 *punMsgType, void *pubDest, uint32 cubDest, uint32 *pcubMsgSize )
{
    PRINT_DEBUG("Steam_Game_Coordinator::RetrieveMessage\n");
    if (outgoing_messages.size()) {
        if (outgoing_messages.front().size() > cubDest) {
            return k_EGCResultBufferTooSmall;
        }

        outgoing_messages.front().copy((char *)pubDest, cubDest);
        if (pcubMsgSize) *pcubMsgSize = outgoing_messages.front().size();
        if (punMsgType && outgoing_messages.front().size() >= sizeof(uint32)) {
            outgoing_messages.front().copy((char *)punMsgType, sizeof(uint32));
            *punMsgType = ntohl(*punMsgType);
        }

        outgoing_messages.pop();
        return k_EGCResultOK;
    } else {
        return k_EGCResultNoMessage;
    }
}

void RunCallbacks()
{
}

void Callback(Common_Message *msg)
{
    if (msg->has_low_level()) {
        if (msg->low_level().type() == Low_Level::CONNECT) {
            
        }

        if (msg->low_level().type() == Low_Level::DISCONNECT) {

        }
    }

    if (msg->has_networking_sockets()) {

    }
}

};