//***********************************************************
// Copyright © 2008 Valentin Pavlyuchenko
//
// This file is part of Boltun.
//
// Boltun 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.
//
// Boltun 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 Boltun. If not, see .
//
//***********************************************************
#include "..\stdafx.h"
using namespace std;
typedef vector string_vec;
typedef multimap string_mmap;
Mind::Mind()
{
data = new MindData();
data->referenceCount = 1;
data->maxSmileLen = 0;
}
Mind::~Mind()
{
if (--data->referenceCount == 0)
delete data;
}
Mind::Mind(const Mind& mind)
{
mind.data->referenceCount++;
data = mind.data;
}
const MindData *Mind::GetData() const
{
return data;
}
Mind& Mind::operator= (const Mind& mind)
{
if (--data->referenceCount == 0)
delete data;
mind.data->referenceCount++;
data = mind.data;
return *this;
}
inline void format(tstring& s)
{
int pos = (int)s.length() - 1;
if (s[pos] == _T('\r'))
s.resize(pos);
}
void toLowerStr(TCHAR* ch)
{
CharLower(ch);
}
vector Mind::Parse(tstring s)
{
int len = (int)s.length() - 1;
vector res;
while (len != -1 && _istspace(s[len]))
len--;
if (len < 0)
return res;
s.resize(len);
int it = 0;
while (it != len) {
while (it != len && _istspace(s[it]))
it++;
if (it == len)
break;
int start = it;
while (it != len && !_istspace(s[it]))
it++;
res.push_back(s.substr(start, it - start));
}
return res;
}
void Mind::Load(tstring filename)
{
basic_ifstream > file;
locale ulocale(locale(), new MyCodeCvt);
file.imbue(ulocale);
file.open(filename.c_str(), ios_base::in | ios_base::binary);
tstring s1, st;
TCHAR *c, *co = NULL;
size_t count;
int error = 0;
int line = 1;
bool start = true;
try {
while (file.good()) {
getline(file, st);
if (st.empty())
break;
line++;
if (start) {
if (st[0] == 65279) {
st.erase(0, 1);
fileTypeMark = true;
}
else
fileTypeMark = false;
start = false;
}
format(st);
count = st.length();
c = co = new TCHAR[count + 1];
mir_tstrcpy(c, st.c_str());
size_t pos = 0;
while (pos < count && _istspace(*c)) {
++pos;
++c;
}
count -= pos;
if (count > 2) {
switch (*c) {
case '(':
if (c[count - 1] != ')')
abort();
if (file.eof())
throw error;
getline(file, s1);
line++;
format(s1);
++c;
count -= 2;
c[count] = '\0';
toLowerStr(c);
{
WordsList l(c);
if (!l.IsEmpty()) {
if (l.IsQuestion())
data->qkeywords.insert(make_pair(l, s1));
else
data->keywords.insert(make_pair(l, s1));
}
}
break;
case '{':
if (c[count - 1] != '}')
abort();
if (file.eof())
throw error;
getline(file, s1);
line++;
format(s1);
++c;
count -= 2;
c[count] = '\0';
toLowerStr(c);
{
WordsList l(c);
if (!l.IsEmpty())
if (l.IsQuestion())
data->qspecialEscapes.insert(make_pair(l, s1));
else
data->specialEscapes.insert(make_pair(l, s1));
}
break;
case '[':
if (c[count - 1] != ']')
throw error;
if (file.eof())
throw error;
getline(file, s1);
line++;
format(s1);
++c;
count -= 2;
c[count] = '\0';
toLowerStr(c);
data->widelyUsed.insert(make_pair(c, s1));
break;
case '<':
if (c[count - 1] != '>')
throw error;
if (file.eof())
throw error;
getline(file, s1);
line++;
format(s1);
++c;
count -= 2;
c[count] = '\0';
if (mir_tstrcmp(c, _T("QUESTION")) == 0) {
toLowerStr(c);
data->question.insert(s1);
}
else if (mir_tstrcmp(c, _T("IGNORED")) == 0) {
toLowerStr(c);
data->special.insert(s1);
}
else if (mir_tstrcmp(c, _T("ESCAPE")) == 0) {
data->escape.push_back(s1);
}
else if (mir_tstrcmp(c, _T("FAILURE")) == 0) {
data->failure.push_back(s1);
}
else if (mir_tstrcmp(c, _T("REPEAT")) == 0) {
data->repeats.push_back(s1);
}
else {
if (mir_tstrcmp(c, _T("INITIAL")) != 0)
throw error;
data->initial.push_back(s1);
}
break;
case '@':
if (file.eof())
throw error;
getline(file, s1);
line++;
format(s1);
++c;
count -= 1;
toLowerStr(c);
{
std::transform(s1.begin(), s1.end(), s1.begin(), ::tolower);
vector strs = Parse(s1);
data->raliases.insert(make_pair(s1, strs));
for (vector::const_iterator it = strs.begin(); it != strs.end(); ++it)
data->aliases.insert(make_pair(*it, s1));
}
break;
default:
if (file.eof())
throw error;
getline(file, s1);
line++;
format(s1);
toLowerStr(c);
data->study.insert(make_pair(c, s1));
}
}
else if (count) {
if (file.eof())
throw error;
getline(file, s1);
line++;
format(s1);
data->study.insert(make_pair(c, s1));
}
}
if (!file.eof())
throw error;
delete co;
}
catch (...) {
throw CorruptedMind(line);
delete co;
}
}
void Mind::Save(tstring filename) const
{
basic_ofstream > file;
locale ulocale(locale(), new MyCodeCvt);
file.imbue(ulocale);
file.open(filename.c_str(), ios_base::out | ios_base::binary);
if (fileTypeMark)
file << TCHAR(65279);
for (string_mmap::iterator it = data->study.begin(); it != data->study.end(); ++it) {
file << (*it).first << _T('\r') << endl;
file << (*it).second << _T('\r') << endl;
}
for (multimap::iterator it = data->keywords.begin(); it != data->keywords.end(); ++it) {
file << _T(" (") << (tstring)(*it).first << _T(")") << _T('\r') << endl;
file << (*it).second << _T('\r') << endl;
}
for (multimap::iterator it = data->qkeywords.begin(); it != data->qkeywords.end(); ++it) {
file << _T(" (") << (tstring)(*it).first << _T(")") << _T('\r') << endl;
file << (*it).second << _T('\r') << endl;
}
for (multimap::iterator it = data->specialEscapes.begin(); it != data->specialEscapes.end(); ++it) {
file << _T(" {") << (tstring)(*it).first << _T("}") << _T('\r') << endl;
file << (*it).second << _T('\r') << endl;
}
for (multimap::iterator it = data->qspecialEscapes.begin(); it != data->qspecialEscapes.end(); ++it) {
file << _T(" {") << (tstring)(*it).first << _T("}") << _T('\r') << endl;
file << (*it).second << _T('\r') << endl;
}
for (string_mmap::iterator it = data->widelyUsed.begin(); it != data->widelyUsed.end(); ++it) {
file << _T(" [") << (*it).first << _T("]") << _T('\r') << endl;
file << (*it).second << _T('\r') << endl;
}
for (set::iterator it = data->question.begin(); it != data->question.end(); ++it) {
file << _T(" ") << _T('\r') << endl;
file << (*it) << _T('\r') << endl;
}
for (set::iterator it = data->special.begin(); it != data->special.end(); ++it) {
file << _T(" ") << _T('\r') << endl;
file << (*it) << _T('\r') << endl;
}
for (string_vec::iterator it = data->escape.begin(); it != data->escape.end(); ++it) {
file << _T(" ") << _T('\r') << endl;
file << (*it) << _T('\r') << endl;
}
for (string_vec::iterator it = data->initial.begin(); it != data->initial.end(); ++it) {
file << _T(" ") << _T('\r') << endl;
file << (*it) << _T('\r') << endl;
}
for (string_vec::iterator it = data->failure.begin(); it != data->failure.end(); ++it) {
file << _T(" ") << _T('\r') << endl;
file << (*it) << _T('\r') << endl;
}
for (string_vec::iterator it = data->repeats.begin(); it != data->repeats.end(); ++it) {
file << _T(" ") << _T('\r') << endl;
file << (*it) << _T('\r') << endl;
}
for (map>::const_iterator it = data->raliases.begin(); it != data->raliases.end(); ++it) {
tstring s;
const vector& v = (*it).second;
bool first = true;
for (vector::const_iterator it1 = v.begin(); it1 != v.end(); ++it1) {
if (first) {
first = false;
s = *it1;
}
else {
s += _T(" ") + *it1;
}
}
file << _T('@') << (*it).first << _T('\r') << endl;
file << s << _T('\r') << endl;
}
}
void Mind::LoadSmiles(tstring filename)
{
basic_ifstream > file;
file.open(filename.c_str());
data->smiles.clear();
tstring s;
unsigned int l = 0;
while (!file.eof()) {
getline(file, s);
if (s.length() > l)
l = (int)s.length();
data->smiles.insert(s);
}
data->maxSmileLen = l;
}
void Mind::LoadSmiles(void *smiles, size_t size)
{
data->smiles.clear();
TCHAR* buf = (TCHAR*)smiles;
unsigned l = 0;
TCHAR* end = buf + size;
while (buf != end) {
TCHAR *lend = buf;
while (lend != end && *lend != _T('\r'))
lend++;
tstring s(buf, lend - buf);
if ((unsigned)(lend - buf) > l)
l = (int)s.length();
data->smiles.insert(s);
if (lend == end)
break;
buf = lend + 2;
}
data->maxSmileLen = l;
}