/*
   net/netclient.cc
   This file is part of the Osirion project and is distributed under
   the terms of the GNU General Public License version 2
*/

#include <string.h>
#include <zlib.h>

#include <iostream>
#include <sstream>

#include "sys/sys.h"
#include "core/net.h"
#include "core/application.h"
#include "core/stats.h"

namespace core
{

NetClient::NetClient(std::string host, int port, int fd) :
		client_host(host)
{
	client_error = true;
	client_state = Connecting;
	client_fd = fd;
	client_host = host;
	client_port = port;

	client_player = new NetPlayer(this);

	if (!fd) {
		con_warn << "Network invalid client file descriptor!" << std::endl;
		abort();
		return;
	}
	con_print << host  << ":" << port << " connected." << std::endl;

	client_addr.sin_family = AF_INET;
	client_addr.sin_port = htons(port);
	client_addr.sin_addr.s_addr = inet_addr(host.c_str());
	memset(client_addr.sin_zero, '\0', sizeof(client_addr.sin_zero));

	if (client_addr.sin_addr.s_addr == INADDR_NONE) {
		con_warn << "Network invalid client address " << host << "!" << std::endl;
		abort();
		return;
	}

	sendq.clear();
	messageblock.clear();

	client_keepalive = application()->time();
	client_timeout = application()->time();

	client_error = false;
}

NetClient::~NetClient()
{
	con_print << host()  << ":" << port() << " disconnected." << std::endl;
	delete client_player;
}

void NetClient::abort()
{
	client_error = true;
}

std::string NetClient::host() const
{
	return client_host;
}

int NetClient::port() const
{
	return client_port;
}

Player *NetClient::player()
{
	return client_player;
}

bool NetClient::has_messages() const
{
	return (recvq.size() > 0);
}

void NetClient::retreive(std::string & message)
{
	if (recvq.size() > 0) {
		message.assign(recvq.front());
		recvq.pop_front();
	} else {
		message.clear();
	}
}

// receive data and decode it into lines
void NetClient::receive(char *data)
{
	const char *c = data;

	while (*c) {
		if ((*c == '\n') || (*c == '\r')) {
			if (messageblock.size() > 0) {
				recvq.push_back(messageblock);
				messageblock.clear();
			}
		} else {
			if (messageblock.size() < FRAMESIZE) {
				messageblock += *c;
			} else {
				con_warn << "Incoming message exceeds " << FRAMESIZE << " bytes!\n";
				messageblock.clear();
			}
		}
		c++;
	}

	client_timeout = application()->time();
}

void NetClient::send_raw(std::string const &msg)
{
	if (error())
		return;

	if ((sendq.size()) && (sendq.size() + msg.size() >= BLOCKSIZE - 16)) {
		transmit();
	}

	sendq.append(msg);
}

void NetClient::transmit()
{
	if (!sendq.size()) {
		if (client_keepalive + NETTIMEOUT / 2 < application()->time()) {
			sendq.assign("ping\n");
		} else {
			return;
		}
	} else if (sendq.size() >= BLOCKSIZE - 16) {
		con_warn << host() << ":" << port() << " outgoing data exceeds " << BLOCKSIZE - 16 << " bytes!\n";
		sendq.clear();
		return;
	}

	unsigned char	zbuf[BLOCKSIZE];
	const char	*data = 0;
	uLong		compressed_size = (uLong) BLOCKSIZE - 5;
	uLong 		uncompressed_size = (uLong) sendq.size();
	size_t		total_size = 0;

	memset(zbuf, 0, sizeof(zbuf));

	Stats::network_uncompressed_bytes_sent += sendq.size();

	// zlib compress
	int status = compress((Bytef*)(zbuf + 4), &compressed_size, (Bytef*)sendq.c_str(), uncompressed_size);

	if ((status == Z_OK) && (compressed_size + 4 < sendq.size())) {
		// add a header to the compress packet
		data = (char *) zbuf;
		total_size = (size_t)(compressed_size + 4);
		zbuf[0] = '\xff';
		zbuf[1] = '\xff';
		zbuf[2] = compressed_size % 256;
		zbuf[3] = compressed_size >> 8;
	} else {
		data = sendq.c_str();
		total_size = sendq.size();
	}

	size_t		total_sent = 0;

	while (total_sent < total_size && !error()) {
		ssize_t bytes_sent = ::sendto(fd(), data, total_size - total_sent, 0,
					      (struct sockaddr *) & client_addr, sizeof(client_addr));

		if (bytes_sent < 0) {
			abort();
			return;
		}

		total_sent += bytes_sent;
		data += bytes_sent;
		Stats::network_bytes_sent += bytes_sent;
	}

	sendq.clear();
	client_keepalive = application()->time();
}

}