/* * RetroShare File lists IO methods. * * file_sharing/filelist_io.h * * Copyright 2016 Mr.Alice * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License Version 2 as published by the Free Software Foundation. * * This library 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 Library General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 * USA. * * Please report all bugs and problems to "retroshare.project@gmail.com". * */ #include "retroshare/rsids.h" #include "pqi/authssl.h" #include "util/rsdir.h" #include "serialiser/rsbaseserial.h" #include "filelist_io.h" template<> bool FileListIO::serialise(unsigned char *buff,uint32_t size,uint32_t& offset,const uint32_t & val) { return setRawUInt32(buff,size,&offset,val) ; } template<> bool FileListIO::serialise(unsigned char *buff,uint32_t size,uint32_t& offset,const size_t & val) { return setRawUInt64(buff,size,&offset,val) ; } template<> bool FileListIO::serialise(unsigned char *buff,uint32_t size,uint32_t& offset,const std::string & val) { return setRawString(buff,size,&offset,val) ; } template<> bool FileListIO::serialise(unsigned char *buff,uint32_t size,uint32_t& offset,const Sha1CheckSum & val) { return val.serialise(buff,size,offset) ; } template<> bool FileListIO::deserialise(const unsigned char *buff,uint32_t size,uint32_t& offset,uint32_t & val) { return getRawUInt32(const_cast(buff),size,&offset,&val) ; } template<> bool FileListIO::deserialise(const unsigned char *buff,uint32_t size,uint32_t& offset,size_t & val) { return getRawUInt64(const_cast(buff),size,&offset,&val) ; } template<> bool FileListIO::deserialise(const unsigned char *buff,uint32_t size,uint32_t& offset,std::string & val) { return getRawString(const_cast(buff),size,&offset,val) ; } template<> bool FileListIO::deserialise(const unsigned char *buff,uint32_t size,uint32_t& offset,Sha1CheckSum & val) { return val.deserialise(const_cast(buff),size,offset) ; } template<> uint32_t FileListIO::serial_size(const uint32_t & ) { return 4 ; } template<> uint32_t FileListIO::serial_size(const size_t & ) { return 8 ; } template<> uint32_t FileListIO::serial_size(const std::string & val) { return getRawStringSize(val) ; } template<> uint32_t FileListIO::serial_size(const Sha1CheckSum & ) { return Sha1CheckSum::serial_size(); } bool FileListIO::writeField( unsigned char*&buff,uint32_t& buff_size,uint32_t& offset,uint8_t section_tag,const unsigned char * val,uint32_t size) { if(!checkSectionSize(buff,buff_size,offset,size)) return false; if(!writeSectionHeader(buff,buff_size,offset,section_tag,size)) return false; memcpy(&buff[offset],val,size) ; offset += size ; return true; } bool FileListIO::readField (const unsigned char *buff,uint32_t buff_size,uint32_t& offset,uint8_t check_section_tag, unsigned char *& val,uint32_t& size) { if(!readSectionHeader(buff,buff_size,offset,check_section_tag,size)) return false; val = (unsigned char *)rs_malloc(size) ; if(!val) return false; memcpy(val,&buff[offset],size); offset += size ; return true ; } bool FileListIO::write125Size(unsigned char *data,uint32_t data_size,uint32_t& offset,uint32_t S) { if(S < 192) { if(offset+1 > data_size) return false; data[offset++] = (uint8_t)S ; return true; } else if(S < 8384) { if(offset+2 > data_size) return false; data[offset+1] = (S - 192) & 0xff ; data[offset ] = ((S - 192 - data[offset+1]) >> 8) + 192; offset += 2 ; return true; } else { if(offset+5 > data_size) return false; data[offset++] = 0xff ; data[offset++] = (uint8_t)((S >> 24) & 255) ; data[offset++] = (uint8_t)((S >> 16) & 255) ; data[offset++] = (uint8_t)((S >> 8) & 255) ; data[offset++] = (uint8_t)((S ) & 255) ; return true ; } } bool FileListIO::read125Size(const unsigned char *data,uint32_t data_size,uint32_t& offset,uint32_t& S) { if(offset + 1 >= data_size) return false; uint8_t b1 = data[offset++] ; if(b1 < 192) { S = b1; return true ; } if(offset + 1 >= data_size) return false; uint8_t b2 = data[offset++] ; if(b1 < 224) { S = ((b1-192) << 8) + b2 + 192 ; return true; } if(b1 != 0xff) return false; if(offset + 3 >= data_size) return false; uint8_t b3 = data[offset++]; uint8_t b4 = data[offset++]; uint8_t b5 = data[offset++]; S = (b2 << 24) | (b3 << 16) | (b4 << 8) | b5 ; return true; } bool FileListIO::saveEncryptedDataToFile(const std::string& fname,const unsigned char *data,uint32_t total_size) { void *encryptedData = NULL ; int encDataLen = 0 ; if(!AuthSSL::getAuthSSL()->encrypt( encryptedData, encDataLen, data,total_size, AuthSSL::getAuthSSL()->OwnId())) { std::cerr << "Cannot encrypt hash cache. Something's wrong." << std::endl; return false; } FILE *F = fopen( (fname+".tmp").c_str(),"wb" ) ; if(!F) { std::cerr << "Cannot open encrypted file cache for writing: " << fname+".tmp" << std::endl; free(encryptedData); return false; } if(fwrite(encryptedData,1,encDataLen,F) != (uint32_t)encDataLen) { std::cerr << "Could not write entire encrypted hash cache file. Out of disc space??" << std::endl; fclose(F) ; free(encryptedData); return false; } fclose(F) ; RsDirUtil::renameFile(fname+".tmp",fname) ; #ifdef FIM_DEBUG std::cerr << "done." << std::endl ; #endif free(encryptedData); return true; } bool FileListIO::loadEncryptedDataFromFile(const std::string& fname,unsigned char *& data,uint32_t& total_size) { uint64_t file_size ; if(!RsDirUtil::checkFile( fname,file_size,false ) ) { std::cerr << "Encrypted hash cache file not present." << std::endl; return false; } // read the binary stream into memory. // RsTemporaryMemory buffer(file_size) ; if(buffer == NULL) return false; FILE *F = fopen( fname.c_str(),"rb") ; if (!F) { std::cerr << "Cannot open file for reading encrypted file cache, filename " << fname << std::endl; return false; } if(fread(buffer,1,file_size,F) != file_size) { std::cerr << "Cannot read from file " + fname << ": something's wrong." << std::endl; fclose(F) ; return false; } fclose(F) ; // now decrypt void *decrypted_data =NULL; int decrypted_data_size =0; if(!AuthSSL::getAuthSSL()->decrypt(decrypted_data, decrypted_data_size, buffer, file_size)) { std::cerr << "Cannot decrypt encrypted file cache. Something's wrong." << std::endl; return false; } data = (unsigned char*)decrypted_data ; total_size = decrypted_data_size ; return true; }