]> git.treefish.org Git - phys/latlib.git/blob - configcache.cpp
...
[phys/latlib.git] / configcache.cpp
1 #include "configcache.h"
2
3 #include <stdlib.h>
4 #include <iostream>
5 #include <time.h>
6 #include <dirent.h>
7
8 configcache::configcache(const string& cacheid, const int& nequi, const int& nskip, const string& datadir, char **configmem, const int& configMemSize, const int& cachemode,
9                          writeout *out_a){
10   out = out_a;
11
12   NEQUI = nequi;
13   NSKIP = nskip;
14   DATADIR = datadir;
15   CACHEID = cacheid;
16
17   configMem = (char*)malloc(configMemSize);
18   tmpConfig = (char*)malloc(configMemSize);
19
20   *configmem = configMem;
21   configSize = configMemSize;
22
23   outBuffer = NULL;
24   inBuffer = NULL;
25
26   MODE = cachemode;
27
28   refetchDataFiles = false;
29 }
30
31 string configcache::getFileId(const bool& shortid)
32 {
33   stringstream fileid;
34
35   if(!shortid) fileid << CACHEID << "_" << NEQUI << "_" << NSKIP;
36   for(int ipara=0; ipara<Paras.size(); ipara++)
37     fileid << "_" << Paras[ipara].id << Paras[ipara].val;
38
39   return fileid.str();
40 }
41
42 void configcache::fetchDataFiles()
43 {
44   struct dirent *de=NULL;
45   DIR *d=NULL;
46   
47   d=opendir(DATADIR.c_str());
48   if(d != NULL){
49     while(de = readdir(d)){
50       string filename = de->d_name;
51       if(isValidInFile(filename)) 
52         {
53           inFiles.push_back(filename);
54         }
55     }
56   }
57 }
58
59 bool configcache::isValidInFile(const string& infile)
60 {
61   char *inchar, *inParts;
62   string truncIn, truncOut;
63
64   if( infile.size() < 4 ) return false;
65
66   if( infile.substr(infile.size()-3) != "dat" ) return false;
67
68   inchar = new char [infile.size()+1];
69   strcpy (inchar, infile.c_str());
70   
71   inParts = strtok( inchar, "_" );
72   for(int iPart=0; inParts!=NULL; iPart++)
73     {
74       if( iPart>3 ) { truncIn += "_"; truncIn += inParts; }
75
76       switch(iPart)
77         {
78         case 1: if(inParts != CACHEID) return false; break;
79         case 2: if(atoi(inParts) != NEQUI) return false; break;
80         case 3: if(atoi(inParts) != NSKIP) return false; break;
81         }
82       inParts = strtok( NULL, "_");
83     }
84   truncIn = truncIn.substr(0, truncIn.size()-4);
85
86   delete[] inchar;
87
88   if( truncIn.find( getFileId(true) + "_" ) == string::npos ) return false;
89
90   return true;
91 }
92
93 bool configcache::readConfig()
94 {
95   if(DATADIR == "" || MODE == CACHE_MODE_DISABLED) return false;
96
97   if(refetchDataFiles){
98     refetchDataFiles = false;
99     fetchDataFiles();
100   }
101
102   while(true)
103     {
104       if( (!inFile.is_open()) && inFiles.size() == 0 ) return false;
105
106       while( (!inFile.is_open()) && inFiles.size() > 0 )
107         {
108           if(out) *out->log << "CCACHE: Opening dat-file: " << inFiles.back() << endl << flush;
109
110           inFile.open( (DATADIR + "/" + inFiles.back()).c_str(), std::ios::binary );
111           inFiles.pop_back();
112
113           if( !inFile.is_open() ) continue;
114
115           std::streampos fsize = inFile.tellg();
116           inFile.seekg( 0, std::ios::end );
117           fsize = inFile.tellg() - fsize;
118           if( fsize == 0 ) {
119             *out->log << "CCACHE: dat-file has zero length! Skipping." << endl << flush;
120             inFile.close();
121             continue;
122           }
123           inFile.seekg(0);
124
125           inBuffer = new boost::iostreams::filtering_istreambuf;
126           inBuffer->push( boost::iostreams::bzip2_decompressor() );
127           inBuffer->push(inFile);
128         }
129
130       if( inFile.is_open() ) 
131         {
132           int readturn = -1;
133           try { readturn = boost::iostreams::read(*inBuffer, tmpConfig, configSize); }
134           catch (std::exception const& ex) {
135             if(out) *out->log << "CCACHE: Caught exception: " << ex.what() << endl << flush;
136             inFile.close();
137           }
138           
139           if( readturn == configSize && inFile.is_open() )
140             {
141               memcpy(configMem, tmpConfig, configSize);
142               return true;
143             }
144           else inFile.close(); 
145         }
146     }
147 }
148
149 void configcache::writeConfig()
150 {
151   if( DATADIR == "" || MODE < 2 ) return;
152
153   if(!outFile.is_open()){
154     time_t secstamp = time(NULL);
155
156     outFileName.str("");
157     outFileName << DATADIR << "/" << secstamp << "_" << getFileId() << "_.dat.tmp";    
158     outFile.open( outFileName.str().c_str(), std::ios::binary );
159
160     outBuffer = new boost::iostreams::filtering_ostreambuf;
161     outBuffer->push(boost::iostreams::bzip2_compressor());
162     outBuffer->push(outFile);
163   }
164
165   boost::iostreams::write(*outBuffer, configMem, configSize);
166 }
167
168 void configcache::addPara(const string& parid, const double& val){
169   parameter newPara;
170   newPara.id = parid;
171   newPara.val = val;
172   Paras.push_back(newPara);
173 }
174
175 int configcache::getParIndex(const string& parid){
176   for(int ipara=0; ipara<Paras.size(); ipara++)
177     if(Paras[ipara].id == parid) return ipara;
178 }
179
180 void configcache::setPara(const string& parid, const double& value){
181   Paras[getParIndex(parid)].val = value;
182   finishOutFile();
183   if(inBuffer != NULL) { delete inBuffer; inBuffer=NULL; } 
184   inFile.close();
185   inFiles.clear();
186
187   refetchDataFiles = true;
188 }
189
190 configcache::~configcache()
191 {
192   finishOutFile();
193   delete inBuffer;
194   inBuffer = NULL;
195 }
196
197 void configcache::finishOutFile()
198 {
199   if( outBuffer != NULL )
200     {
201       delete outBuffer;
202       outBuffer = NULL;
203     }
204
205   if( outFile.is_open() )
206     {
207       outFile.close();
208       rename( outFileName.str().c_str(), outFileName.str().substr(0, outFileName.str().size()-4).c_str() );
209     }
210 }