/*
    sockpool.* - socket connection pool
    Copyright (C) 1999-2002  Matthew Mueller <donut AT dakotacom.net>

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

    This program 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 General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with this program; if not, write to the Free Software
    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include "sockpool.h"
#include <errno.h>

int Connection::putline(int echo,const char * str,...){
	va_list ap;
	va_start(ap,str);
	int i=doputline(echo,str,ap);
	va_end(ap);
	return i;
}
int Connection::doputline(int echo,const char * str,va_list ap){
	char *fpbuf;
	int i;

	i=vasprintf(&fpbuf,str,ap);
	if (!(fpbuf=(char*)realloc(fpbuf,i+3))){
		free(fpbuf);
		close();
		throw TransportExError(Ex_INIT,"nntp_putline:realloc(%p,%i) %s(%i)",fpbuf,i+3,strerror(errno),errno);
	}
	if (echo)
		printf("%s << %s\n", server->shortname.c_str(), fpbuf);
	fpbuf[i]='\r';fpbuf[i+1]='\n';
	try {
		i=sock.write(fpbuf,i+2);
	} catch (FileEx &e) {
		free(fpbuf);
		close(1);
		throw TransportExError(Ex_INIT,"nntp_putline: %s:%i: %s",e.getExFile(), e.getExLine(), e.getExStr());
	}
	free(fpbuf);
	return i;
}

int Connection::getline(int echo){
	int i;
	try {
		i=sock.bgets();
	} catch (FileEx &e) {
		close(1);
		throw TransportExError(Ex_INIT,"nntp_getline: %s:%i: %s",e.getExFile(), e.getExLine(), e.getExStr());
	}
	if (i<=0){
		close(1);
		throw TransportExError(Ex_INIT,"nntp_getline: connection closed unexpectedly");
	}else {
		if (echo)
			printf("%s >> %s\n", server->shortname.c_str(), sock.rbufp());
	}
	return i;
}


void SockPool::connection_erase(t_connection_map::iterator i) {
	try {
		i->second->close();
	} catch (FileEx &e) {//ignore transport errors while closing
		print_ex_with_message(e, "ignored error");
	}
	delete i->second;
	connections.erase(i);
}

Connection* SockPool::connect(const c_server::ptr &server){
	Connection *c;

	//use existing connection when possible
	t_connection_map::iterator i = connections.find(server);
	if (i!=connections.end()){
		c = i->second;	
		if (c->isopen()) {
			try {
				while (c->sock.datawaiting()) {
					c->getline(1);
				}
			} catch (baseCommEx &e) {//ignore transport errors (probably server timeout)
				print_ex_with_message(e, "ignored error");
			}
		}
		if (c->isopen()) {
			c->touch();
			return c;
		} else {
			connection_erase(i);
		}
	}
	
	//check penalization before expire_old_connection
	nconfig.check_penalized(server); // (throws exception if penalized)
	
	//if we have to create a new one, don't go over max
	if (nconfig.maxconnections > 0 && (signed)connections.size() >= nconfig.maxconnections)
		expire_old_connection();
	
	//create new connection
	try {
		c = new Connection(server);
	}  catch (FileEx &e) {
		nconfig.penalize(server);
		throw TransportExError(Ex_INIT,"Connection: %s",e.getExStr());
	}
	connections.insert(t_connection_map::value_type(server, c));
	c->touch();
	return c;
}

void SockPool::release(Connection *connection) {
	assert(connection);
	bool keepopen = connection->isopen();
	if (connection->server_ok) {
		nconfig.unpenalize(connection->server);
		connection->server_ok=false; //reset so that problems later on with this connection can still be caught.
	} else
		if (nconfig.penalize(connection->server))
			keepopen=false;
	if (keepopen)
		connection->touch();
	else
		connection_erase(connections.find(connection->server));
}

void SockPool::expire_old_connection(void) {
	t_connection_map::iterator i, oldest=connections.end();
	for (i=connections.begin(); i!=connections.end(); ++i) {
		Connection *c = i->second;
		if (!c->isopen()) {//if connection is closed, we can get rid of it right now.
			oldest = i;
			break;
		}
		if (oldest==connections.end() || c->age()>oldest->second->age())
			oldest = i;
	}
	assert(oldest!=connections.end());//no old connections to kill??
	connection_erase(oldest);
}

void SockPool::expire_connections(bool closeall) {
	t_connection_map::iterator i, p;
	i=connections.begin();
	while (i!=connections.end()) {
		Connection *c = i->second;
		p=i;
		++i;
		if (closeall || !c->isopen() || (c->age() >= c->server->idletimeout)) {
			connection_erase(p);
		}
	}
}


syntax highlighted by Code2HTML, v. 0.9.1