2020-09-30 16:53:55 +02:00

609 lines
19 KiB
C++

//+------------------------------------------------------------------
//
// Copyright (C) 1993, Microsoft Corporation.
//
// File: FileEnum.cxx
//
// Contents: class encapsulating file enumeration, including a deep option
//
// Classes: CFileEnumeration
//
// History: Nov-93 DaveMont Created.
// Feb-98 BrunoSc CFileEnumerate::Init changed to avoid problems with
// files with german Umlaute.
//
//-------------------------------------------------------------------
#include "pch.h"
#include "t2.hxx"
#include "FileEnum.hxx"
#include <locale.h>
#include <wchar.h>
#if DBG
extern ULONG Debug;
#endif
//+---------------------------------------------------------------------------
//
// Member: CFileEnumerate::CFileEnumerate, public
//
// Synopsis: initializes data members, constructor will not throw
//
// Arguments: IN [fdeep] - TRUE = go into sub-directories
//
//----------------------------------------------------------------------------
CFileEnumerate::CFileEnumerate(BOOL fdeep)
: _fdeep(fdeep),
_findeep(FALSE),
_froot(FALSE),
_fcannotaccess(FALSE),
_pcfe(NULL),
_pwfileposition(NULL),
_handle(INVALID_HANDLE_VALUE)
{
ENUMERATE_RETURNS((stderr, L"CFileEnumerate ctor\n"))
}
//+---------------------------------------------------------------------------
//
// Member: Dtor, public
//
// Synopsis: closes handles
//
// Arguments: none
//
//----------------------------------------------------------------------------
CFileEnumerate::~CFileEnumerate()
{
if (_handle != INVALID_HANDLE_VALUE)
FindClose(_handle);
ENUMERATE_RETURNS((stderr, L"CFileEnumerate dtor (%ws)\n", _wpath))
}
//+---------------------------------------------------------------------------
//
// Member: CFileEnumerate::Init, public
//
// Synopsis: Init must be called before any other methods - this
// is not enforced. converts a ASCII file/path to a UNICODE
// file/path, and gets the first file in the enumeration
//
// Arguments: IN [filename] - the path/file to enumerate
// OUT [wfilename] - first file in the enumeration
// OUT [fdir] - TRUE = returned file is a directory
//
//----------------------------------------------------------------------------
/*
ULONG CFileEnumerate::Init(LPCWSTR filename, LPWSTR *wfilename, BOOL *fdir)
{
// Initialize the file name
int iError = 0;
if (filename && (wcslen(filename) < MAX_PATH))
{
// make it wchar
WCHAR winfilename[MAX_PATH];
iError = MultiByteToWideChar( CP_ACP,
MB_PRECOMPOSED,
(LPTSTR) filename,
-1,
winfilename,
wcslen( filename ) * 2
);
if (!iError)
{ // error has occured during translation
return( iError = GetLastError());
}
_pwfileposition = NULL;
// finish initialization
return(_ialize(winfilename, wfilename, fdir));
}
ENUMERATE_FAIL((stderr, "Init bad file name: %ld\n",ERROR_INVALID_NAME))
return(ERROR_INVALID_NAME);
}
*/
//+---------------------------------------------------------------------------
//
// Member: CFileEnumerate::Init, public
//
// Synopsis: Same as previous, except takes UNICODE file/path as input
//
// Arguments: IN [filename] - the path/file to enumerate
// OUT [wfilename] - first file in the enumeration
// OUT [fdir] - TRUE = returned file is a directory
//
//----------------------------------------------------------------------------
ULONG CFileEnumerate::Init(LPCWSTR filename, LPWSTR *wfilename, BOOL *fdir)
{
// Initialize the file name
if (filename && (wcslen(filename) < MAX_PATH))
{
return(_ialize(filename, wfilename, fdir));
}
ENUMERATE_FAIL((stderr, L"Init bad file name: %ld\n",ERROR_INVALID_NAME))
return(ERROR_INVALID_NAME);
}
//+---------------------------------------------------------------------------
//
// Member: CFileEnumerate::_ialize, private
//
// Synopsis: finishes initialization and starts search for first file in
// the enumeration
//
// Arguments: OUT [wfilename] - first file in the enumeration
// OUT [fdir] - TRUE = returned file is a directory
//
//----------------------------------------------------------------------------
ULONG CFileEnumerate::_ialize(LPCWSTR winfilename, LPWSTR *wfilename, BOOL *fdir)
{
ENUMERATE_RETURNS((stderr, L"Init start, path = %ws\n", winfilename))
ULONG ret = ERROR_SUCCESS;
ENUMERATE_STAT((stderr, L"start path = %ws\n",winfilename))
// save the location of the filename or wildcards
ULONG cwcharcount;
if (!(cwcharcount = GetFullPathName((LPCTSTR)winfilename,
MAX_PATH,
(LPTSTR)_wpath,
(LPTSTR*)&_pwfileposition)))
{
return(ERROR_INVALID_NAME);
}
ENUMERATE_STAT((stderr, L"got full path name = %ws, filename = (%ws), total chars = %d\n",_wpath, _pwfileposition, cwcharcount))
// if the filepart (_pwfileposition) is NULL, then the name must end in a slash.
// add a *
if (NULL == _pwfileposition)
{
_pwfileposition = (LPWSTR) Add2Ptr(_wpath,wcslen(_wpath)*sizeof(WCHAR));
}
// save the filename/wildcards
wcscpy(_wwildcards, _pwfileposition);
ENUMERATE_EXTRA((stderr, L"wild cards = %ws\n",_wwildcards))
// if we are at a root (path ends in :\)
if ( (_wpath[wcslen(_wpath) - 1] == L'\\') &&
(wcslen(_wpath) > 1) &&
(_wpath[wcslen(_wpath) - 2] == L':') )
{
_wfd.dwFileAttributes = FILE_ATTRIBUTE_DIRECTORY;
_wfd.cFileName[0] = L'\0';
*wfilename = _wpath;
*fdir = TRUE;
_froot = TRUE;
} else
{
// check to see if we can iterate through files
if ( (INVALID_HANDLE_VALUE == ( _handle = FindFirstFile((LPCTSTR)_wpath, &_wfd ) ) ) )
{
ret = GetLastError();
_fcannotaccess = (ERROR_ACCESS_DENIED == ret);
ENUMERATE_FAIL((stderr, L"find first returned: %ld\n",ret))
}
if (ERROR_SUCCESS == ret)
{ // reject . & .. filenames (go on to next file )
if ( (0 == _wcsicmp(_wfd.cFileName, L".")) ||
(0 == _wcsicmp(_wfd.cFileName, L"..")) )
{
ret = _NextLocal(wfilename,fdir);
} else
{
// return the current directory
if (_wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
*fdir = TRUE;
else
*fdir = FALSE;
// add the filename to the path so the whole thing is returned
//_pwfileposition is pointer to postion of file in _wpath. _wpath is
//MAX_PATH long. MAX_PATH - (_pwfileposition - _wpath) is the length of
//buffer _pwfileposition
//
HRESULT hr = StringCchCopy((LPWSTR)_pwfileposition,MAX_PATH - (_pwfileposition- _wpath), (LPCWSTR )_wfd.cFileName);
if(FAILED(hr))
{
return HRESULT_CODE(hr);
}
*wfilename = _wpath;
}
}
ENUMERATE_STAT((stderr, L"next filename = %ws\n", *wfilename))
}
// if we are going deep and we did not find a file yet:
if ( _fdeep && ( ( ERROR_NO_MORE_FILES == ret ) ||
( ERROR_FILE_NOT_FOUND == ret ) ) )
{
if (_handle != INVALID_HANDLE_VALUE)
{
FindClose(_handle);
_handle = INVALID_HANDLE_VALUE;
}
ret = _InitDir(wfilename, fdir);
}
ENUMERATE_RETURNS((stderr, L"Init returning = %ws(%ld)\n\n", *wfilename, ret))
return(ret);
}
//+---------------------------------------------------------------------------
//
// Member: CFileEnumerate::Next, public
//
// Synopsis: finds the next file in the enumeration
//
// Arguments: OUT [wfilename] - first file in the enumeration
// OUT [fdir] - TRUE = returned file is a directory
//
//----------------------------------------------------------------------------
ULONG CFileEnumerate::Next(LPWSTR *wfilename, BOOL *fdir)
{
ENUMERATE_RETURNS((stderr, L"Next start, path = %ws\n", _wpath))
ULONG ret = ERROR_NO_MORE_FILES;
// if we failed to initialize with an ERROR_ACCESS_DENIED, then exit
if (_fcannotaccess)
return(ERROR_NO_MORE_FILES);
// if we are not in deep
if (!_findeep)
{
if (!_froot)
ret = _NextLocal(wfilename, fdir);
// if we ran out of files and we are going deep:
if ( _fdeep &&
( ( ERROR_NO_MORE_FILES == ret ) ||
( ERROR_FILE_NOT_FOUND == ret ) || _froot ) )
{
if (_handle != INVALID_HANDLE_VALUE)
{
FindClose(_handle);
_handle = INVALID_HANDLE_VALUE;
}
ret = _InitDir(wfilename, fdir);
_froot = FALSE; // (we are past the root now)
}
} else
{
// if we are already down a directory (and in deep)
if (_pcfe)
{
if (ERROR_SUCCESS != (ret = _pcfe->Next(wfilename, fdir)))
{
if (ERROR_ACCESS_DENIED != ret)
{
delete _pcfe;
_pcfe = NULL;
}
}
}
// we need to go to the next directory in the current dir
if (ERROR_NO_MORE_FILES == ret)
{
ret = _NextDir(wfilename, fdir);
}
}
ENUMERATE_RETURNS((stderr, L"Next returning = %ws(%ld)\n\n", *wfilename, ret))
return(ret);
}
//+---------------------------------------------------------------------------
//
// Member: CFileEnumerate::_NextLocal, private
//
// Synopsis: searchs for the next file in the current directory
//
// Arguments: OUT [wfilename] - first file in the enumeration
// OUT [fdir] - TRUE = returned file is a directory
//
//----------------------------------------------------------------------------
ULONG CFileEnumerate::_NextLocal(LPWSTR *wfilename, BOOL *fdir)
{
ENUMERATE_RETURNS((stderr, L"_NextLocal start, path = %ws\n", _wpath))
ULONG ret = ERROR_SUCCESS;
// ensure that we have a valid handle for a findnextfile
if (INVALID_HANDLE_VALUE == _handle)
{
ret = ERROR_INVALID_HANDLE;
} else
{
do
{
if (!FindNextFile(_handle, &_wfd))
{
ret = GetLastError();
ENUMERATE_FAIL((stderr, L"find next returned: %ld\n",ret))
} else
ret = ERROR_SUCCESS;
}
while ( (ERROR_SUCCESS == ret) &&
( (0 == _wcsicmp(_wfd.cFileName, L".")) ||
(0 == _wcsicmp(_wfd.cFileName, L"..")) ) );
// if we found a file
if (ERROR_SUCCESS == ret)
{
// return the directory attrib.
if (_wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
*fdir = TRUE;
else
*fdir = FALSE;
//_pwfileposition is pointer to postion of file in _wpath. _wpath is
//MAX_PATH long. MAX_PATH - (_pwfileposition - _wpath) is the length of
//buffer _pwfileposition
//
HRESULT hr = StringCchCopy((LPWSTR)_pwfileposition,
MAX_PATH - (_pwfileposition- _wpath),
(const wchar_t*)_wfd.cFileName);
if(FAILED(hr))
{
return HRESULT_CODE(hr);
}
*wfilename = _wpath;
ENUMERATE_STAT((stderr, L"next filename = %ws\n", *wfilename))
}
}
ENUMERATE_RETURNS((stderr, L"_NextLocal returning = %ws(%ld)\n", *wfilename, ret))
return(ret);
}
//+---------------------------------------------------------------------------
//
// Member: CFileEnumerate::_InitDir, private
//
// Synopsis: (only called if going deep)
// goes down a directory (and thus causing a new CFileEnumerator
// to be created, or re-initializies
//
// Arguments: OUT [wfilename] - first file in the enumeration
// OUT [fdir] - TRUE = returned file is a directory
//
//----------------------------------------------------------------------------
ULONG CFileEnumerate::_InitDir(LPWSTR *wfilename, BOOL *fdir)
{
ENUMERATE_RETURNS((stderr, L"_InitDir start, path = %ws\n", _wpath))
ULONG ret = ERROR_SUCCESS;
// check and see if a directory was entered as the filename
if ( (0 == _wcsicmp( _wwildcards, _wfd.cFileName)) &&
(_wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) )
{
ENUMERATE_EXTRA((stderr, L"first file matched directory = %ws\n", _wpath))
_pwfileposition += wcslen((LPWSTR)_wfd.cFileName);
//_pwfileposition is pointer to postion of file in _wpath. _wpath is
//MAX_PATH long. MAX_PATH - (_pwfileposition - _wpath) is the length of
//buffer _pwfileposition
//
HRESULT hr = StringCchCopy((LPWSTR)_pwfileposition,
MAX_PATH - (_pwfileposition- _wpath),
L"\\*.*");
if(FAILED(hr))
{
return HRESULT_CODE(hr);
}
_pwfileposition++;
wcscpy(_wwildcards, L"*.*");
ENUMERATE_EXTRA((stderr, L" path = %ws\n",_wpath))
ENUMERATE_EXTRA((stderr, L"wild cards = %ws\n",_wwildcards))
WCHAR winfilename[MAX_PATH] = L"";
hr = StringCchCopy((LPWSTR)winfilename,
MAX_PATH,
_wpath);
if(FAILED(hr))
{
return HRESULT_CODE(hr);
}
ret = _ialize(winfilename, wfilename, fdir);
} else
{
// we are in deep
_findeep = TRUE;
// search thru all directories
//_pwfileposition is pointer to postion of file in _wpath. _wpath is
//MAX_PATH long. MAX_PATH - (_pwfileposition - _wpath) is the length of
//buffer _pwfileposition
//
HRESULT hr = StringCchCopy((LPWSTR)_pwfileposition,
MAX_PATH - (_pwfileposition- _wpath),
L"*.*");
if(FAILED(hr))
{
return HRESULT_CODE(hr);
}
if (INVALID_HANDLE_VALUE == ( _handle = FindFirstFile((LPCTSTR)_wpath, &_wfd) ))
{
ret = GetLastError();
ENUMERATE_FAIL((stderr, L"find first (dir) returned: %ld\n",ret))
} else
{
if ( !(_wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) ||
(0 == wcscmp(( LPCWSTR)_wfd.cFileName, L".")) ||
(0 == wcscmp(( LPCWSTR)_wfd.cFileName, L"..")) )
{
ret = _NextDir(wfilename, fdir);
} else
{
// if we have a sub directory, go down it
ret = _DownDir(wfilename, fdir);
// if we found nothing in that first sub directory, go the the next one
if ( (ERROR_NO_MORE_FILES == ret ) ||
(ERROR_FILE_NOT_FOUND == ret ) )
{
ret = _NextDir(wfilename, fdir);
}
}
}
}
ENUMERATE_RETURNS((stderr, L"_InitDir returning = %ws(%ld)\n", *wfilename, ret))
return(ret);
}
//+---------------------------------------------------------------------------
//
// Member: CFileEnumerate::_NextDir, private
//
// Synopsis: (only called if going deep)
// finds the next sub-directory from the current directory,
// and then goes down into that directory
//
// Arguments: OUT [wfilename] - first file in the enumeration
// OUT [fdir] - TRUE = returned file is a directory
//
//----------------------------------------------------------------------------
ULONG CFileEnumerate::_NextDir(LPWSTR *wfilename, BOOL *fdir)
{
ENUMERATE_RETURNS((stderr, L"_NextDir start, path = %ws\n", _wpath))
ULONG ret = ERROR_SUCCESS;
// skip the . & .. & files we cannot access
if (INVALID_HANDLE_VALUE == _handle)
{
ret = ERROR_INVALID_HANDLE;
} else
{
do
{
do
{
if (!FindNextFile(_handle, &_wfd))
{
ret = GetLastError();
ENUMERATE_FAIL((stderr, L"find next returned: %ld\n",ret))
} else
ret = ERROR_SUCCESS;
}
while ( (ERROR_SUCCESS == ret) &&
( !(_wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) ||
(0 == wcscmp(( LPCWSTR )_wfd.cFileName, L".")) ||
(0 == wcscmp(( LPCWSTR )_wfd.cFileName, L"..")) ) );
// if we found a directory
if (ERROR_SUCCESS == ret)
{
ret = _DownDir(wfilename, fdir);
} else
{
// out of subdirectories to search, break out of the loop
break;
}
}
while (( ERROR_NO_MORE_FILES == ret) || (ERROR_FILE_NOT_FOUND == ret));
}
ENUMERATE_RETURNS((stderr, L"_NextDir returning = %ws(%ld)\n", *wfilename, ret))
return(ret);
}
//+---------------------------------------------------------------------------
//
// Member: CFileEnumerate::_DownDir, private
//
// Synopsis: (only called if going deep)
// creates a new CFileEnumerator for a sub-directory
//
// Arguments: OUT [wfilename] - first file in the enumeration
// OUT [fdir] - TRUE = returned file is a directory
//
//----------------------------------------------------------------------------
ULONG CFileEnumerate::_DownDir(LPWSTR *wfilename, BOOL *fdir)
{
ENUMERATE_RETURNS((stderr, L"_DownDir start, path = %ws\n", _wpath))
ULONG ret;
// make a new file enumerator class (this one) We should only go down
// 8 directories at most.
_pcfe = new CFileEnumerate(_fdeep);
if ( NULL == _pcfe )
{
ret = GetLastError () ;
ENUMERATE_RETURNS((stderr, L"_DownDir returning = %ws(%ld)\n", *wfilename, ret ))
return(ret);
}
// add the wildcards to the end of the directory we are going down
//_pwfileposition is pointer to postion of file in _wpath. _wpath is
//MAX_PATH long. MAX_PATH - (_pwfileposition - _wpath) is the length of
//buffer _pwfileposition
//
HRESULT hr = StringCchCopy((LPWSTR)_pwfileposition,
MAX_PATH - (_pwfileposition- _wpath),
( const wchar_t *)_wfd.cFileName);
if(FAILED(hr))
{
return HRESULT_CODE(hr);
}
hr = StringCchCat(_wpath,MAX_PATH,L"\\");
if(FAILED(hr))
{
return HRESULT_CODE(hr);
}
hr = StringCchCat(_wpath,MAX_PATH, _wwildcards);
if(FAILED(hr))
{
return HRESULT_CODE(hr);
}
// start it up and see if we find a match
if (ERROR_SUCCESS != (ret = _pcfe->Init(_wpath, wfilename, fdir)))
{
if (ERROR_ACCESS_DENIED != ret)
{
delete _pcfe;
_pcfe = NULL;
}
}
ENUMERATE_RETURNS((stderr, L"_DownDir returning = %ws(%ld)\n", *wfilename, ret))
return(ret);
}