/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/***************************************************************************
 *            miav_config.cc
 *
 *  Sat Feb 19 14:13:19 CET 2005
 *  Copyright  2005 Bent Bisballe
 *  deva@aasimon.org
 ****************************************************************************/

/*
 *  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 Library 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 */

#include <config.h>
#include "miav_config.h"

MiavConfig *config;

MiavConfig::MiavConfig(char *file, Error* err)
{
  error = err;
  configs = NULL;
  
  filename = string(file);

  // Read config file
  FILE* fp = fopen(file, "r");
  
  if(!fp) {
    char errbuf[256];
    sprintf(errbuf, "Error reading configuration file %s\n", file);
    if(error) error->pushError(errbuf);
    return;
  }
  fseek(fp, 0, SEEK_END);
  int fsz = ftell(fp);
  fseek(fp, 0, SEEK_SET);
  
  char *raw = (char*)calloc(fsz, 1);
  fread(raw, 1, fsz, fp);

  fclose(fp);

  parse(raw);
  free(raw);
}

MiavConfig::~MiavConfig()
{
  _cfg *die = NULL;
  _cfg *cfg = configs;

  while(cfg) {
    if(die) free(die);
    die = cfg;
    cfg = cfg->next;
  }
  if(die) free(die);
}

/**
 * Prints a reasonable error message when a parse error occurres.
 */
_cfg *MiavConfig::parseError(char* msg, char* line)
{
  char errbuf[512];
  sprintf(errbuf, "Error parsing file %s at line:\n\t%s\n\t%s\n", filename.c_str(), line, msg);
  if(error) error->pushError(errbuf);
  return NULL;
}

/** 
 * Adds one configuration entry, from a single zero terminated line.
 */
_cfg *MiavConfig::addConfig(_cfg *parent, char* conf)
{
  // Check for wellformed input:
  // Check for =
  if(strstr(conf, "=") == 0) return parseError("Missing '='", conf);
  /*
  if(strstr(conf, "\"")) {
    if(strstr(conf, "=") > strstr(conf, "\""))
    return parseError("Missing '=', first occurrence inside string", conf);
  }
  */

  // Check for nonempty left side
  if(strstr(conf, "=") == conf) return parseError("Empty left side", conf);

  // Check for nonempty right side
  if(strstr(conf, "=") == conf + strlen(conf) - 1) return parseError("Empty right side.", conf);

  // Parse this wellformed input.
  _cfg *cfg;

  cfg = (_cfg*) malloc(sizeof(_cfg));
  if(!parent) configs = cfg;

  int namelen = strchr(conf, '=') - conf;
  char* name = (char*)calloc(namelen + 1, 1);
  strncpy(name, conf, namelen);

  int vallen = conf + strlen(conf) - (strchr(conf, '=') + 1);
  char* val = (char*)calloc(vallen + 1, 1);
  strncpy(val, conf + strlen(conf) - vallen, vallen);

  // TODO: Check valid rightside (true, false, number or "..")

  cfg->name = new string((const char*)name);
  free(name);

  cfg->stringval = new string((const char*)val);
  cfg->intval = atoi(val);
  cfg->floatval = atof(val);
  cfg->boolval = atoi(val) != 0;
  free(val);

  cfg->next = NULL;

  if(parent) parent->next = cfg;
  return cfg;
}

/**
 * Main parse function, iterates the lines of the file.
 */
int MiavConfig::parse(char* raw)
{
  // Strip the string
  char *conf = strip(raw);
  char *conf_end = conf + strlen(conf);
  char *start = conf;
  char *p;

  _cfg *cfg = NULL;

  // Iterate the lines in the string
  for(p = conf; p < conf_end; p++) {
    if(*p == '\n') {
      *p = '\0';
      if(!(cfg = addConfig(cfg, start))) return 1;
      start = p+1;
    }
  }
  // Allocated in strip
  free(conf);
  return 0;
}

/**
 * Strip all unwanted data from the string, initial to parsing.
 */
char* MiavConfig::strip(char* conf)
{
  // Freed in parse!!!
  char *stripped = (char*)calloc(strlen(conf) + 2, 1);
  char *r;
  char *w = stripped;

  bool instring = false;
  bool incomment = false;

  // Iterate over the characters in the input string.
  for(r = conf; r < conf + strlen(conf); r++) {
    if(strchr("#", *r)) incomment = true;
    if(strchr("\n", *r)) incomment = false;

    if(!incomment) {
      if(instring) {
        // When in a string, accept anything, except ".
        if(*r != '\"') {
          *w = *r;
          w++;
        }
      } else {
        // Only copy valid characters
        if(strchr("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz1234567890-_,.=", *r)) {
          // Change comma into dot
          if(*r == ',') *r = '.';
          *w = *r;
          w++;
        }
        // We don't want double newlines and initial newline!
        if((*r == '\n') && (*(w-1) != '\n') && (w != stripped)) {
          *w = *r;
          w++;
        }
      }
    }

    if(strchr("\"", *r)) instring = !instring;
  }

  // If we are not ending on a newline, we better append one
  if(*(w-1) != '\n') {
    *w = '\n';
    w++;
  }

  // Make sure we are nullterminated.
  *w = '\0';

  return stripped;
}

int MiavConfig::readInt(char *node)
{
  _cfg* n = findNode(node);
  if(n) return n->intval;
  else return 0;
}

bool MiavConfig::readBool(char *node)
{
  _cfg* n = findNode(node);
  if(n) return n->boolval;
  else return false;
}

string *MiavConfig::readString(char *node)
{
  _cfg* n = findNode(node);
  if(n) return n->stringval;
  else return &emptyString;
}

float MiavConfig::readFloat(char *node)
{
  _cfg* n = findNode(node);
  if(n) return n->floatval;
  else return 0.0f;
}

_cfg *MiavConfig::findNode(char* node)
{
  _cfg *cfg = configs;

  while(cfg) {
    if(!strcmp(node, cfg->name->c_str())) return cfg;
    cfg = cfg->next;
  }
  char errbuf[256];
  sprintf(errbuf, "Request for nonexisting node \"%s\"!\n", node);
  if(error) error->pushError(errbuf);
  return NULL;
}

Error* MiavConfig::getErrorObj()
{
  return error;
}