1 #include "../../include/freenet/siteinserter.h"
\r
2 #include "../../include/global.h"
\r
4 #include <Poco/DateTime.h>
\r
5 #include <Poco/Timespan.h>
\r
6 #include <Poco/DateTimeFormatter.h>
\r
12 SiteInserter::SiteInserter()
\r
17 SiteInserter::SiteInserter(FCPv2::Connection *fcp):IIndexInserter<long>(fcp)
\r
22 void SiteInserter::CheckForNeededInsert()
\r
24 // only do 1 insert at a time
\r
25 if(m_inserting.size()==0)
\r
27 Poco::DateTime date;
\r
28 date.assign(date.year(),date.month(),date.day(),0,0,0);
\r
30 SQLite3DB::Statement st=m_db->Prepare("SELECT LocalIdentityID FROM tblLocalIdentity WHERE PublishFreesite='true' AND (LastInsertedFreesite IS NULL OR LastInsertedFreesite<?);");
\r
31 st.Bind(0,Poco::DateTimeFormatter::format(date,"%Y-%m-%d"));
\r
34 if(st.RowReturned())
\r
36 int localidentityid=0;
\r
37 st.ResultInt(0,localidentityid);
\r
38 StartInsert(localidentityid);
\r
43 std::string SiteInserter::GenerateIndex(const std::string &htmltemplate, const long localidentityid, const std::string &name)
\r
45 std::string content="";
\r
47 SQLite3DB::Statement boardst=m_db->Prepare("SELECT tblBoard.BoardName FROM tblBoard INNER JOIN tblMessageBoard ON tblBoard.BoardID=tblMessageBoard.BoardID WHERE tblMessageBoard.MessageID=? ORDER BY tblBoard.BoardName COLLATE NOCASE;");
\r
48 SQLite3DB::Statement st=m_db->Prepare("SELECT tblMessage.Body, tblMessage.Subject, tblMessage.MessageID FROM tblMessage INNER JOIN tblIdentity ON tblMessage.IdentityID=tblIdentity.IdentityID INNER JOIN tblLocalIdentity ON tblIdentity.PublicKey=tblLocalIdentity.PublicKey WHERE tblLocalIdentity.LocalIdentityID=? ORDER BY tblMessage.MessageDate DESC, tblMessage.MessageTime DESC LIMIT 0,10;");
\r
49 st.Bind(0,localidentityid);
\r
52 while(st.RowReturned())
\r
54 std::string post="";
\r
55 std::string subject="";
\r
56 std::string boards="";
\r
59 st.ResultText(0,post);
\r
60 st.ResultText(1,subject);
\r
61 st.ResultInt(2,messageid);
\r
63 boardst.Bind(0,messageid);
\r
65 while(boardst.RowReturned())
\r
67 std::string board="";
\r
68 boardst.ResultText(0,board);
\r
78 content+="<div class=\"post\">";
\r
79 content+="<div class=\"postboards\">";
\r
80 content+=SanitizeOutput(boards);
\r
82 content+="<div class=\"postsubject\">";
\r
83 content+=SanitizeOutput(subject);
\r
85 content+="<div class=\"postbody\">";
\r
86 content+=SanitizeOutput(post);
\r
87 //post=SanitizeOutput(post);
\r
88 //StringFunctions::Replace(post,"\r\n","<br>");
\r
96 return StringFunctions::Replace(htmltemplate,"[CONTENT]",content);
\r
100 std::string SiteInserter::GenerateLinks(const bool publishtrustlist, const bool publishboardlist)
\r
102 std::string links="";
\r
104 links+="<li><a href=\"index.htm\">Home</a></li>";
\r
105 if(publishtrustlist)
\r
107 links+="<li><a href=\"trustlist.htm\">Trust List</a></li>";
\r
109 if(publishboardlist)
\r
111 // links+="<li><a href=\"boardlist.htm\">Board List</a></li>";
\r
117 void SiteInserter::GeneratePages(const long localidentityid, std::string &uskkey, std::map<std::string,std::string> &pages)
\r
119 SQLite3DB::Statement st=m_db->Prepare("SELECT Name, PrivateKey, PublishTrustList, PublishBoardList, FreesiteEdition FROM tblLocalIdentity WHERE LocalIdentityID=?;");
\r
120 st.Bind(0,localidentityid);
\r
123 if(st.RowReturned())
\r
125 std::string htmltemplate="";
\r
126 std::string filename="";
\r
127 std::string name="";
\r
128 std::string key="";
\r
129 std::string publishtrustliststr="";
\r
130 std::string publishboardliststr="";
\r
131 bool publishtrustlist=false;
\r
132 bool publishboardlist=false;
\r
133 std::string editionstr="";
\r
135 st.ResultText(0,name);
\r
136 st.ResultText(1,key);
\r
137 st.ResultText(2,publishtrustliststr);
\r
138 st.ResultText(3,publishboardliststr);
\r
139 st.ResultText(4,editionstr);
\r
141 publishtrustliststr=="true" ? publishtrustlist=true : publishtrustlist=false;
\r
142 publishboardliststr=="true" ? publishboardlist=true : publishboardlist=false;
\r
143 // no edition exists - start at 0
\r
148 // previous edition exists - add 1
\r
152 StringFunctions::Convert(editionstr,edition);
\r
154 StringFunctions::Convert(edition,editionstr);
\r
158 if(key.find("SSK@")==0)
\r
163 key+=m_messagebase+"/"+editionstr+"/";
\r
166 filename=name+"-template.htm";
\r
167 FILE *infile=fopen(filename.c_str(),"rb");
\r
170 infile=fopen("site-template.htm","rb");
\r
174 fseek(infile,0,SEEK_END);
\r
175 long len=ftell(infile);
\r
176 fseek(infile,0,SEEK_SET);
\r
178 std::vector<unsigned char> data;
\r
180 fread(&data[0],1,data.size(),infile);
\r
183 htmltemplate.append(data.begin(),data.end());
\r
185 htmltemplate=StringFunctions::Replace(htmltemplate,"[LINKS]",GenerateLinks(publishtrustlist,publishboardlist));
\r
186 htmltemplate=StringFunctions::Replace(htmltemplate,"[IDENTITYNAME]",SanitizeOutput(name));
\r
188 pages["index.htm"]=GenerateIndex(htmltemplate,localidentityid,name);
\r
189 if(publishtrustlist)
\r
191 pages["trustlist.htm"]=GenerateTrustList(htmltemplate,localidentityid,name);
\r
193 if(publishboardlist)
\r
195 // pages["boardlist.htm"]=GenerateBoardList(htmltemplate,localidentityid,name);
\r
201 m_log->error("SiteInserter::GeneratePages unable to open "+filename+" or site-template.htm.");
\r
204 // get extra files that the user wants to add to the Freesite
\r
205 filename=name+"-files.txt";
\r
206 infile=fopen(filename.c_str(),"rb");
\r
209 std::vector<std::string> files;
\r
211 fseek(infile,0,SEEK_END);
\r
212 long len=ftell(infile);
\r
213 fseek(infile,0,SEEK_SET);
\r
215 std::vector<unsigned char> data;
\r
217 fread(&data[0],1,data.size(),infile);
\r
220 // split on \r and \n - on systems with \r\n line endings there will be blank entries, but we'll just skip those
\r
221 std::string filecontent(data.begin(),data.end());
\r
222 StringFunctions::SplitMultiple(filecontent,"\r\n",files);
\r
224 for(std::vector<std::string>::iterator i=files.begin(); i!=files.end(); i++)
\r
226 if((*i)!="" && (*i).find("index.htm")==std::string::npos && (*i).find("trustlist.htm")==std::string::npos && (*i).find("files.htm")==std::string::npos)
\r
229 infile=fopen(filename.c_str(),"rb");
\r
232 fseek(infile,0,SEEK_END);
\r
234 fseek(infile,0,SEEK_SET);
\r
237 fread(&data[0],1,data.size(),infile);
\r
241 filecontent.append(data.begin(),data.end());
\r
243 // strip off path from filename
\r
244 while(filename.find_first_of("/")!=std::string::npos)
\r
246 filename.erase(0,filename.find_first_of("/")+1);
\r
249 if(filecontent.size()>0)
\r
251 pages[filename]=filecontent;
\r
257 m_log->error("SiteInserter::GeneratePages could not include user file "+(*i));
\r
267 std::string SiteInserter::GenerateTrustList(const std::string &htmltemplate, const long localidentityid, const std::string &name)
\r
269 std::string content="";
\r
270 Poco::DateTime date;
\r
272 date-=Poco::Timespan(20,0,0,0,0);
\r
273 SQLite3DB::Statement st=m_db->Prepare("SELECT Name,PublicKey,tblIdentityTrust.LocalMessageTrust,tblIdentityTrust.LocalTrustListTrust,tblIdentity.IdentityID,tblIdentityTrust.MessageTrustComment,tblIdentityTrust.TrustListTrustComment,tblIdentity.FreesiteEdition FROM tblIdentity LEFT JOIN (SELECT IdentityID,LocalMessageTrust,LocalTrustListTrust,MessageTrustComment,TrustListTrustComment FROM tblIdentityTrust WHERE LocalIdentityID=?) AS 'tblIdentityTrust' ON tblIdentity.IdentityID=tblIdentityTrust.IdentityID WHERE PublicKey IS NOT NULL AND LastSeen IS NOT NULL AND LastSeen>=? ORDER BY Name COLLATE NOCASE;");
\r
274 st.Bind(0,localidentityid);
\r
275 st.Bind(1,Poco::DateTimeFormatter::format(date,"%Y-%m-%d %H:%M:%S"));
\r
278 content+="<table class=\"trustlist\">";
\r
279 content+="<tr class=\"title\"><thcolspan=\"5\">";
\r
280 content+="Trust List of "+SanitizeOutput(name);
\r
281 content+="</th></tr>";
\r
282 content+="<tr class=\"headings\"><th></th><th>Message Trust</th><th>Message Comment</th><th>Trust List Trust</th><th>Trust Comment</th></tr>";
\r
283 while(st.RowReturned())
\r
285 std::string idname="";
\r
286 std::string thisid="";
\r
287 std::string messagetrustcomment="";
\r
288 std::string trustlisttrustcomment="";
\r
289 std::string messagetrust="";
\r
290 std::string trustlisttrust="";
\r
291 std::string publickey="";
\r
292 std::string uskkey="";
\r
293 std::string freesiteedition="";
\r
295 st.ResultText(0,idname);
\r
296 st.ResultText(1,publickey);
\r
297 st.ResultText(2,messagetrust);
\r
298 st.ResultText(3,trustlisttrust);
\r
299 st.ResultText(4,thisid);
\r
300 st.ResultText(5,messagetrustcomment);
\r
301 st.ResultText(6,trustlisttrustcomment);
\r
302 st.ResultText(7,freesiteedition);
\r
304 if(freesiteedition!="")
\r
306 if(publickey.find("SSK@")==0)
\r
310 uskkey="USK"+uskkey;
\r
311 uskkey+=m_messagebase+"/"+freesiteedition+"/";
\r
316 if(freesiteedition!="")
\r
318 content+="<td><div><a href=\""+uskkey+"\">"+SanitizeOutput(CreateShortIdentityName(idname,publickey))+"</a></div></td>";
\r
322 content+="<td><div>"+SanitizeOutput(CreateShortIdentityName(idname,publickey))+"</div></td>";
\r
324 content+="<td "+GetClassString(messagetrust)+">"+messagetrust+"</td>";
\r
325 content+="<td>"+SanitizeOutput(messagetrustcomment)+"</td>";
\r
326 content+="<td "+GetClassString(trustlisttrust)+">"+trustlisttrust+"</td>";
\r
327 content+="<td>"+SanitizeOutput(trustlisttrustcomment)+"</td>";
\r
328 content+="</tr>\r\n";
\r
332 content+="</table>";
\r
334 return StringFunctions::Replace(htmltemplate,"[CONTENT]",content);
\r
338 const std::string SiteInserter::GetClassString(const std::string &trustlevel)
\r
341 std::string tempstr;
\r
343 StringFunctions::Convert(trustlevel,tempint);
\r
345 StringFunctions::Convert(tempint,tempstr);
\r
349 return "class=\"trust"+tempstr+"\"";
\r
357 const bool SiteInserter::HandlePutFailed(FCPv2::Message &message)
\r
359 std::vector<std::string> idparts;
\r
360 long localidentityid;
\r
362 StringFunctions::Split(message["Identifier"],"|",idparts);
\r
363 StringFunctions::Convert(idparts[1],localidentityid);
\r
365 RemoveFromInsertList(localidentityid);
\r
367 m_log->error("SiteInserter::HandlePutFailed failed to insert Freesite, Freenet error code : "+message["Code"]);
\r
372 const bool SiteInserter::HandlePutSuccessful(FCPv2::Message &message)
\r
374 std::vector<std::string> idparts;
\r
375 std::vector<std::string> uriparts;
\r
376 long localidentityid;
\r
378 Poco::DateTime now;
\r
380 StringFunctions::Split(message["Identifier"],"|",idparts);
\r
381 StringFunctions::Convert(idparts[1],localidentityid);
\r
383 // edition is very last part of uri
\r
384 StringFunctions::Split(message["URI"],"/",uriparts);
\r
385 if(uriparts.size()>0)
\r
387 StringFunctions::Convert(uriparts[uriparts.size()-1],edition);
\r
390 SQLite3DB::Statement st=m_db->Prepare("UPDATE tblLocalIdentity SET LastInsertedFreesite=?, FreesiteEdition=? WHERE LocalIdentityID=?;");
\r
391 st.Bind(0,Poco::DateTimeFormatter::format(now,"%Y-%m-%d %H:%M:%S"));
\r
392 st.Bind(1,edition);
\r
393 st.Bind(2,localidentityid);
\r
396 m_log->information("SiteInserter::HandlePutSuccessful successfully inserted Freesite.");
\r
398 RemoveFromInsertList(localidentityid);
\r
403 void SiteInserter::Initialize()
\r
405 m_fcpuniquename="SiteInserter";
\r
408 const std::string SiteInserter::SanitizeOutput(const std::string &input)
\r
410 // must do & first because all other elements have & in them!
\r
411 std::string output=StringFunctions::Replace(input,"&","&");
\r
412 output=StringFunctions::Replace(output,"<","<");
\r
413 output=StringFunctions::Replace(output,">",">");
\r
414 output=StringFunctions::Replace(output,"\"",""");
\r
415 output=StringFunctions::Replace(output," "," ");
\r
419 const bool SiteInserter::StartInsert(const long &localidentityid)
\r
421 FCPv2::Message message;
\r
422 std::string localidentityidstr="";
\r
423 std::string sizestr="";
\r
424 std::string uskkey="";
\r
425 std::map<std::string,std::string> pages;
\r
428 StringFunctions::Convert(localidentityid,localidentityidstr);
\r
430 GeneratePages(localidentityid,uskkey,pages);
\r
432 message.SetName("ClientPutComplexDir");
\r
433 message["URI"]=uskkey;
\r
434 message["Identifier"]=m_fcpuniquename+"|"+localidentityidstr+"|"+message["URI"];
\r
435 message["DefaultName"]="index.htm";
\r
437 // add each page to the message
\r
438 for(std::map<std::string,std::string>::iterator pagei=pages.begin(); pagei!=pages.end(); pagei++)
\r
440 std::string filenumstr;
\r
441 StringFunctions::Convert(filenum,filenumstr);
\r
444 StringFunctions::Convert((*pagei).second.size(),sizestr);
\r
446 message["Files."+filenumstr+".Name"]=(*pagei).first;
\r
447 message["Files."+filenumstr+".UploadFrom"]="direct";
\r
448 message["Files."+filenumstr+".DataLength"]=sizestr;
\r
453 m_fcp->Send(message);
\r
455 // send data of each page
\r
456 for(std::map<std::string,std::string>::iterator pagei=pages.begin(); pagei!=pages.end(); pagei++)
\r
458 m_fcp->Send(std::vector<char>((*pagei).second.begin(),(*pagei).second.end()));
\r
461 m_inserting.push_back(localidentityid);
\r