Notepad3/grepWinNP3/sktoolslib_mod/TempFile.cpp
2020-09-19 02:24:33 +02:00

146 lines
4.4 KiB
C++

// sktoolslib - common files for SK tools
// Copyright (C) 2014, 2017, 2020 - Stefan Kueng
// 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,
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
//
#include "stdafx.h"
#include "TempFile.h"
#include "StringUtils.h"
#include "PathUtils.h"
#include "DirFileEnum.h"
#include "SmartHandle.h"
CTempFiles::CTempFiles(void)
{
}
CTempFiles::~CTempFiles(void)
{
for (const auto& f : m_TempFileList)
DeleteFile(f.c_str());
}
CTempFiles& CTempFiles::Instance()
{
static CTempFiles instance;
return instance;
}
std::wstring CTempFiles::ConstructTempPath(const std::wstring& path) const
{
DWORD len = ::GetTempPath(0, nullptr);
auto temppath = std::make_unique<wchar_t[]>(len + 1);
auto tempF = std::make_unique<wchar_t[]>(len + 50);
::GetTempPath(len + 1, temppath.get());
std::wstring tempfile;
std::wstring possibletempfile;
if (path.empty())
{
::GetTempFileName(temppath.get(), L"skt", 0, tempF.get());
tempfile = std::wstring(tempF.get());
}
else
{
int i = 0;
do
{
std::wstring filename = path;
// the inner loop assures that the resulting path is < MAX_PATH
// if that's not possible without reducing the 'filename' to less than 5 chars, use a path
// that's longer than MAX_PATH (in that case, we can't really do much to avoid longer paths)
std::wstring sExt = CPathUtils::GetFileExtension(path);
do
{
possibletempfile = CStringUtils::Format(L"%s%s.svn%3.3x.tmp.%s", temppath.get(), (LPCWSTR)filename.c_str(), i, (LPCWSTR)sExt.c_str());
tempfile = possibletempfile;
filename = filename.substr(0, filename.size() - 1);
} while ((filename.size() > 4) && (tempfile.size() >= MAX_PATH));
i++;
} while (PathFileExists(tempfile.c_str()));
}
// caller has to actually grab the file path
return tempfile;
}
std::wstring CTempFiles::CreateTempPath(bool bRemoveAtEnd, const std::wstring& path, bool directory)
{
bool succeeded = false;
for (int retryCount = 0; retryCount < MAX_RETRIES; ++retryCount)
{
std::wstring tempfile = ConstructTempPath(path);
// now create the temp file / directory, so that subsequent calls to GetTempFile() return
// different filenames.
// Handle races, i.e. name collisions.
if (directory)
{
DeleteFile(tempfile.c_str());
if (CreateDirectory(tempfile.c_str(), nullptr) == FALSE)
{
if (GetLastError() != ERROR_ALREADY_EXISTS)
return std::wstring();
}
else
succeeded = true;
}
else
{
CAutoFile hFile = CreateFile(tempfile.c_str(), GENERIC_READ, FILE_SHARE_READ, nullptr, CREATE_ALWAYS, FILE_ATTRIBUTE_TEMPORARY, nullptr);
if (!hFile)
{
if (GetLastError() != ERROR_ALREADY_EXISTS)
return std::wstring();
}
else
{
succeeded = true;
}
}
// done?
if (succeeded)
{
if (bRemoveAtEnd)
m_TempFileList.insert(tempfile);
return tempfile;
}
}
// give up
return std::wstring();
}
std::wstring CTempFiles::GetTempFilePath(bool bRemoveAtEnd, const std::wstring& path /*= std::wstring()*/)
{
return CreateTempPath(bRemoveAtEnd, path, false);
}
std::wstring CTempFiles::GetTempFilePathString()
{
return CreateTempPath(true, std::wstring(), false);
}
std::wstring CTempFiles::GetTempDirPath(bool bRemoveAtEnd, const std::wstring& path /* = CTSVNPath() */)
{
return CreateTempPath(bRemoveAtEnd, path, true);
}