/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set et sw=2 ts=2: */
/***************************************************************************
 *            session.cc
 *
 *  Tue Dec 15 13:36:49 CET 2009
 *  Copyright 2009 Bent Bisballe Nyeng
 *  deva@aasimon.org
 ****************************************************************************/

/*
 *  This file is part of Pracro.
 *
 *  Pracro 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.
 *
 *  Pracro 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 Pracro; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA.
 */
#include "session.h"

#include <stdlib.h>
#include <stdio.h>

// for stat
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <errno.h>

#include "journal.h"
#include "journal_uploadserver.h"

#include "database.h"
#include "configuration.h"
#include "connectionpool.h"
#include "sessionserialiser.h"
#include "environment.h"

Session::Session(Environment *e,
                 std::string sid, std::string pid, std::string t)
 : env(e)
{
  _journal = NULL;

  sessionid = sid;
  patientid = pid;
  templ = t;
  
  isreadonly = true;
}

Session::~Session()
{
  if(_journal) delete _journal;
}

std::string Session::id()
{
  if(sessionid == "") {
    AutoBorrower<Database*> borrower(env->dbpool);
    Database *db = borrower.get();
    sessionid = db->newSessionId();
  }

  return sessionid;
}

void Session::lock()
{
  mutex.lock();
}

void Session::unlock()
{
  mutex.unlock();
}

bool Session::active()
{
  if(isreadonly) return true;

  {
    AutoBorrower<Database*> borrower(env->dbpool);
    Database *db = borrower.get();
    return db->active(id());
  }
}

void Session::setActive(bool a)
{
  if(isreadonly == false) {
    AutoBorrower<Database*> borrower(env->dbpool);
    Database *db = borrower.get();
    return db->setActive(id(), a);
  }
}

void Session::commit()
{
  if(_journal != NULL) {
    _journal->commit();
    delete _journal;
    _journal = NULL;
  }
  if(isreadonly == false) {
    AutoBorrower<Database*> borrower(env->dbpool);
    Database *db = borrower.get();
    db->commit(id());
  }
}

void Session::nocommit()
{
  if(isreadonly == false) {
    AutoBorrower<Database*> borrower(env->dbpool);
    Database *db = borrower.get();
    db->nocommit(id());
  }
}

void Session::discard()
{
  if(_journal) {
    delete _journal;
    _journal = NULL;
  }
  if(isreadonly == false) {
    AutoBorrower<Database*> borrower(env->dbpool);
    Database *db = borrower.get();
    db->discard(id());
  }
}

Journal *Session::journal()
{
  if(_journal == NULL) {
   _journal =
      new JournalUploadServer(Conf::journal_commit_addr,
                              Conf::journal_commit_port);
  }
  return _journal;
}
/*
Database *Session::database()
{
  if(_database == NULL) {
   _database =
      new Database(Conf::database_backend, Conf::database_addr, "",
                   Conf::database_user, Conf::database_passwd, "");
  }
  return _database;
}
*/
Sessions::Sessions(Environment *e) : env(e)
{
}

static bool fexists(const std::string &f)
{
  bool ret;

  FILE *fp = fopen(f.c_str(), "r");
  ret = fp != NULL;
  if(fp) fclose(fp);

  return ret;
}

Session *Sessions::newSession(std::string patientid, std::string templ)
  throw(SessionAlreadyActive)
{
  std::map<std::string, Session *>::iterator i = sessions.begin();
  while(i != sessions.end()) {
    if(i->second->patientid == patientid &&
       i->second->templ == templ) {
      Session *session = i->second;
      if(session->active()) throw SessionAlreadyActive(session->id());
      return session;
    }

    i++;
  }

  { // Look up patientid / template tupple in session files.
    SessionSerialiser ser(env, Conf::session_path);
    Session *session = ser.findFromTupple(patientid, templ);
    if(session) {
      sessions[session->id()] = session;
      if(session->active()) throw SessionAlreadyActive(session->id());
      return session;
    }
  }

  Session *session = new Session(env, "", patientid, templ);
  sessions[session->id()] = session;
  return session;
}

Session *Sessions::session(std::string sessionid)
{
  if(sessions.find(sessionid) != sessions.end())
    return sessions[sessionid];

  std::string filename = getSessionFilename(Conf::session_path, sessionid);
  if(fexists(filename)) {
    SessionSerialiser ser(env, Conf::session_path);
    Session *s = ser.load(sessionid);
    sessions[s->id()] = s;
    return s;
  }

  return NULL;
}

Session *Sessions::takeSession(std::string sessionid)
{
  DEBUG(session,"%s\n", sessionid.c_str());

  Session *s = NULL;
  if(sessions.find(sessionid) != sessions.end()) {
    s = sessions[sessionid];
  }

  if(s) {
    sessions.erase(sessionid);
  }
  else DEBUG(session, "No such session!\n");

  return s;
}

void Sessions::deleteSession(std::string sessionid)
{
  Session *s = takeSession(sessionid);
  if(s) delete s;
}

size_t Sessions::size()
{
  return sessions.size();
}

void Sessions::store()
{
  std::map<std::string, Session*>::iterator i = sessions.begin();
  while(i != sessions.end()) {
    SessionSerialiser ser(env, Conf::session_path);
    ser.save(i->second);
    delete i->second;
    sessions.erase(i);
    i++;
  }
  sessions.clear();
}

SessionAutolock::SessionAutolock(Session &s)
  : session(s)
{
  session.lock();
}

SessionAutolock::~SessionAutolock()
{
  session.unlock();
}

#ifdef TEST_SESSION
//deps: configuration.cc journal.cc journal_commit.cc mutex.cc debug.cc sessionserialiser.cc sessionparser.cc saxparser.cc
//cflags: -I.. $(PTHREAD_CFLAGS) $(EXPAT_CFLAGS)
//libs: $(PTHREAD_LIBS) $(EXPAT_LIBS)
#include <test.h>

TEST_BEGIN;
Sessions sessions;

Conf::session_path = "/tmp";

srand(0); // force seed
Session *s1 = sessions.newSession();
srand(0); // force seed
Session *s2 = sessions.newSession();

TEST_NOTEQUAL(s1->id(), s2->id(), "Testing if IDs are unique.");

TEST_EQUAL(sessions.size(), 2, "Testing if size match.");

std::string sessionid = s1->id();
SessionSerialiser ser(Conf::session_path, s1);
ser.save();

sessions.deleteSession(sessionid);
TEST_EQUAL(sessions.size(), 1, "Testing if size match.");

s1 = sessions.session(sessionid);
TEST_NOTEQUAL(s1, NULL, "Did we reload the session from disk?");

sessions.store();
TEST_EQUAL(sessions.size(), 0, "Testing if size match.");

s1 = sessions.session(sessionid);
TEST_NOTEQUAL(s1, NULL, "Did we reload the session from disk?");

TEST_END;

#endif/*TEST_SESSION*/