📄 archiverecovery.cpp
字号:
//this file is part of eMule
//Copyright (C)2002 Merkur ( devs@emule-project.net / http://www.emule-project.net )
//
//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 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., 675 Mass Ave, Cambridge, MA 02139, USA.
#include "stdafx.h"
#include "emule.h"
#include "ArchiveRecovery.h"
#include "OtherFunctions.h"
#include "Preferences.h"
#include "PartFile.h"
#include <zlib/zlib.h>
#include "Log.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
// At some point it may be worth displaying messages to alert the user if there were errors, or where to find the file.
void CArchiveRecovery::recover(CPartFile *partFile, bool preview, bool bCreatePartFileCopy)
{
if (partFile->m_bPreviewing || partFile->m_bRecoveringArchive)
return;
partFile->m_bRecoveringArchive = true;
AddLogLine(true, _T("%s \"%s\""), GetResString(IDS_ATTEMPTING_RECOVERY), partFile->GetFileName());
// Get the current filled list for this file
CTypedPtrList<CPtrList, Gap_Struct*> *filled = new CTypedPtrList<CPtrList, Gap_Struct*>;
partFile->GetFilledList(filled);
#ifdef _DEBUG
{
int i = 0;
TRACE("%s: filled\n", __FUNCTION__);
POSITION pos = filled->GetHeadPosition();
while (pos){
Gap_Struct* gap = filled->GetNext(pos);
TRACE("%3u: %10u %10u (%u)\n", i++, gap->start, gap->end, gap->end - gap->start + 1);
}
}
#endif
// The rest of the work can be safely done in a new thread
ThreadParam *tp = new ThreadParam;
tp->partFile = partFile;
tp->filled = filled;
tp->preview = preview;
tp->bCreatePartFileCopy = bCreatePartFileCopy;
// - do NOT use Windows API 'CreateThread' to create a thread which uses MFC/CRT -> lot of mem leaks!
if (!AfxBeginThread(run, (LPVOID)tp)){
partFile->m_bRecoveringArchive = false;
LogError(LOG_STATUSBAR, _T("%s \"%s\""), GetResString(IDS_RECOVERY_FAILED), partFile->GetFileName());
// Need to delete the memory here as won't be done in thread
DeleteMemory(tp);
}
}
UINT AFX_CDECL CArchiveRecovery::run(LPVOID lpParam)
{
ThreadParam *tp = (ThreadParam *)lpParam;
DbgSetThreadName("ArchiveRecovery");
InitThreadLocale();
if (!performRecovery(tp->partFile, tp->filled, tp->preview, tp->bCreatePartFileCopy))
theApp.QueueLogLine(true, GetResString(IDS_RECOVERY_FAILED));
tp->partFile->m_bRecoveringArchive = false;
// Delete memory used by copied gap list
DeleteMemory(tp);
return 0;
}
bool CArchiveRecovery::performRecovery(CPartFile *partFile, CTypedPtrList<CPtrList, Gap_Struct*> *filled,
bool preview, bool bCreatePartFileCopy)
{
bool success = false;
try
{
CFile temp;
CString tempFileName;
if (bCreatePartFileCopy)
{
// Copy the file
tempFileName = CString(thePrefs.GetTempDir()) + _T("\\") + partFile->GetFileName().Mid(0, 5) + _T("-rec.tmp");
if (!CopyFile(partFile, filled, tempFileName))
return false;
// Open temp file for reading
if (!temp.Open(tempFileName, CFile::modeRead|CFile::shareDenyWrite))
return false;
}
else
{
if (!temp.Open(partFile->GetFilePath(), CFile::modeRead | CFile::shareDenyNone))
return false;
}
// Open the output file
CString ext = partFile->GetFileName().Right(4);
CString outputFileName = CString(thePrefs.GetTempDir()) + _T("\\") + partFile->GetFileName().Mid(0, 5) + _T("-rec") + ext;
CFile output;
ULONGLONG ulTempFileSize = 0;
if (output.Open(outputFileName, CFile::modeWrite | CFile::shareDenyWrite | CFile::modeCreate))
{
// Process the output file
if (ext.CompareNoCase(_T(".zip")) == 0)
success = recoverZip(&temp, &output, filled, (temp.GetLength() == partFile->GetFileSize()));
else if (ext.CompareNoCase(_T(".rar")) == 0)
success = recoverRar(&temp, &output, filled);
ulTempFileSize = output.GetLength();
// Close output
output.Close();
}
// Close temp file
temp.Close();
// Remove temp file
if (!tempFileName.IsEmpty())
CFile::Remove(tempFileName);
// Report success
if (success)
{
theApp.QueueLogLine(true, _T("%s \"%s\""), GetResString(IDS_RECOVERY_SUCCESSFUL), partFile->GetFileName());
theApp.QueueDebugLogLine(false, _T("Part file size: %s, temp. archive file size: %s (%.1f%%)"), CastItoXBytes(partFile->GetFileSize()), CastItoXBytes(ulTempFileSize), partFile->GetFileSize() ? (ulTempFileSize * 100.0 / partFile->GetFileSize()) : 0.0);
// Preview file if required
if (preview)
{
SHELLEXECUTEINFO SE;
memset(&SE,0,sizeof(SE));
SE.fMask = SEE_MASK_NOCLOSEPROCESS ;
SE.lpVerb = _T("open");
SE.lpFile = outputFileName.GetBuffer();
SE.nShow = SW_SHOW;
SE.cbSize = sizeof(SE);
ShellExecuteEx(&SE);
if (SE.hProcess)
{
WaitForSingleObject(SE.hProcess, INFINITE);
CloseHandle(SE.hProcess);
}
CFile::Remove(outputFileName);
}
}
}
catch (CFileException* error){
error->Delete();
}
#ifndef _DEBUG
catch (...){
ASSERT(0);
}
#endif
return success;
}
bool CArchiveRecovery::recoverZip(CFile *zipInput, CFile *zipOutput, CTypedPtrList<CPtrList, Gap_Struct*> *filled, bool fullSize)
{
bool retVal = false;
long fileCount = 0;
try
{
CTypedPtrList<CPtrList, ZIP_CentralDirectory*> centralDirectoryEntries;
Gap_Struct *fill;
// If the central directory is intact this is simple
if (fullSize && readZipCentralDirectory(zipInput, ¢ralDirectoryEntries, filled))
{
if (centralDirectoryEntries.GetCount() == 0)
return false;
ZIP_CentralDirectory *cdEntry;
POSITION pos = centralDirectoryEntries.GetHeadPosition();
bool deleteCD;
for (int i=centralDirectoryEntries.GetCount(); i>0; i--)
{
deleteCD = false;
cdEntry = centralDirectoryEntries.GetAt(pos);
uint32 lenEntry = sizeof(ZIP_Entry) + cdEntry->lenFilename + cdEntry->lenExtraField + cdEntry->lenCompressed;
if (IsFilled(cdEntry->relativeOffsetOfLocalHeader, cdEntry->relativeOffsetOfLocalHeader + lenEntry, filled))
{
zipInput->Seek(cdEntry->relativeOffsetOfLocalHeader, CFile::begin);
// Update offset
cdEntry->relativeOffsetOfLocalHeader = zipOutput->GetPosition();
if (!processZipEntry(zipInput, zipOutput, lenEntry, NULL))
deleteCD = true;
}
else
deleteCD = true;
if (deleteCD)
{
delete [] cdEntry->filename;
if (cdEntry->lenExtraField > 0)
delete [] cdEntry->extraField;
if (cdEntry->lenComment > 0)
delete [] cdEntry->comment;
delete cdEntry;
POSITION del = pos;
centralDirectoryEntries.GetNext(pos);
centralDirectoryEntries.RemoveAt(del);
}
else
centralDirectoryEntries.GetNext(pos);
}
}
else // Have to scan the file the hard way
{
// Loop through filled areas of the file looking for entries
POSITION pos = filled->GetHeadPosition();
while (pos != NULL)
{
fill = filled->GetNext(pos);
uint32 filePos = zipInput->GetPosition();
// The file may have been positioned to the next entry in ScanForMarker() or processZipEntry()
if (filePos > fill->end)
continue;
if (filePos < fill->start)
zipInput->Seek(fill->start, CFile::begin);
// If there is any problem, then don't bother checking the rest of this part
for (;;)
{
// Scan for entry marker within this filled area
if (!scanForZipMarker(zipInput, (uint32)ZIP_LOCAL_HEADER_MAGIC, (fill->end - zipInput->GetPosition() + 1)))
break;
if (zipInput->GetPosition() > fill->end)
break;
if (!processZipEntry(zipInput, zipOutput, (fill->end - zipInput->GetPosition() + 1), ¢ralDirectoryEntries))
break;
}
}
}
// Remember offset before CD entries
uint32 startOffset = zipOutput->GetPosition();
// Write all central directory entries
fileCount = centralDirectoryEntries.GetCount();
if (fileCount > 0)
{
ZIP_CentralDirectory *cdEntry;
POSITION pos = centralDirectoryEntries.GetHeadPosition();
while (pos != NULL)
{
cdEntry = centralDirectoryEntries.GetNext(pos);
writeUInt32(zipOutput, ZIP_CD_MAGIC);
writeUInt16(zipOutput, cdEntry->versionMadeBy);
writeUInt16(zipOutput, cdEntry->versionToExtract);
writeUInt16(zipOutput, cdEntry->generalPurposeFlag);
writeUInt16(zipOutput, cdEntry->compressionMethod);
writeUInt16(zipOutput, cdEntry->lastModFileTime);
writeUInt16(zipOutput, cdEntry->lastModFileDate);
writeUInt32(zipOutput, cdEntry->crc32);
writeUInt32(zipOutput, cdEntry->lenCompressed);
writeUInt32(zipOutput, cdEntry->lenUnompressed);
writeUInt16(zipOutput, cdEntry->lenFilename);
writeUInt16(zipOutput, cdEntry->lenExtraField);
writeUInt16(zipOutput, cdEntry->lenComment);
writeUInt16(zipOutput, 0); // Disk number start
writeUInt16(zipOutput, cdEntry->internalFileAttributes);
writeUInt32(zipOutput, cdEntry->externalFileAttributes);
writeUInt32(zipOutput, cdEntry->relativeOffsetOfLocalHeader);
zipOutput->Write(cdEntry->filename, cdEntry->lenFilename);
if (cdEntry->lenExtraField > 0)
zipOutput->Write(cdEntry->extraField, cdEntry->lenExtraField);
if (cdEntry->lenComment > 0)
zipOutput->Write(cdEntry->comment, cdEntry->lenComment);
delete [] cdEntry->filename;
if (cdEntry->lenExtraField > 0)
delete [] cdEntry->extraField;
if (cdEntry->lenComment > 0)
delete [] cdEntry->comment;
delete cdEntry;
}
// Remember offset before CD entries
uint32 endOffset = zipOutput->GetPosition();
// Write end of central directory
writeUInt32(zipOutput, ZIP_END_CD_MAGIC);
writeUInt16(zipOutput, 0); // Number of this disk
writeUInt16(zipOutput, 0); // Number of the disk with the start of the central directory
writeUInt16(zipOutput, fileCount);
writeUInt16(zipOutput, fileCount);
writeUInt32(zipOutput, endOffset - startOffset);
writeUInt32(zipOutput, startOffset);
writeUInt16(zipOutput, strlen(ZIP_COMMENT));
zipOutput->Write(ZIP_COMMENT, strlen(ZIP_COMMENT));
centralDirectoryEntries.RemoveAll();
}
retVal = true;
}
catch (CFileException* error){
error->Delete();
}
#ifndef _DEBUG
catch (...){
ASSERT(0);
}
#endif
// Tell the user how many files were recovered
CString msg;
if (fileCount == 1)
msg = GetResString(IDS_RECOVER_SINGLE);
else
msg.Format(GetResString(IDS_RECOVER_MULTIPLE), fileCount);
theApp.QueueLogLine(true, _T("%s"), msg);
return retVal;
}
bool CArchiveRecovery::readZipCentralDirectory(CFile *zipInput, CTypedPtrList<CPtrList, ZIP_CentralDirectory*> *centralDirectoryEntries, CTypedPtrList<CPtrList, Gap_Struct*> *filled)
{
bool retVal = false;
try
{
// Ideally this zip file will not have a comment and the End-CD will be easy to find
zipInput->Seek(-22, CFile::end);
if (!(readUInt32(zipInput) == ZIP_END_CD_MAGIC))
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -