You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1069 lines
36 KiB
1069 lines
36 KiB
/* /////////////////////////////////////////////////////////////////////////
|
|
* File: inetstl/filesystem/filesystem_traits.hpp (originally MInetEnm.h)
|
|
*
|
|
* Purpose: Contains the filesystem_traits template class, and ANSI and
|
|
* Unicode specialisations thereof.
|
|
*
|
|
* Created: 30th April 1999
|
|
* Updated: 16th June 2010
|
|
*
|
|
* Home: http://stlsoft.org/
|
|
*
|
|
* Copyright (c) 1999-2010, Matthew Wilson and Synesis Software
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* - Redistributions of source code must retain the above copyright notice, this
|
|
* list of conditions and the following disclaimer.
|
|
* - Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
* - Neither the name(s) of Matthew Wilson and Synesis Software nor the names of
|
|
* any contributors may be used to endorse or promote products derived from
|
|
* this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* ////////////////////////////////////////////////////////////////////// */
|
|
|
|
|
|
/** \file inetstl/filesystem/filesystem_traits.hpp
|
|
*
|
|
* \brief [C++ only] Definition of the inetstl::filesystem_traits
|
|
* traits class template
|
|
* (\ref group__library__filesystem "File System" Library).
|
|
*/
|
|
|
|
#ifndef INETSTL_INCL_INETSTL_FILESYSTEM_HPP_FILESYSTEM_TRAITS
|
|
#define INETSTL_INCL_INETSTL_FILESYSTEM_HPP_FILESYSTEM_TRAITS
|
|
|
|
#ifndef STLSOFT_DOCUMENTATION_SKIP_SECTION
|
|
# define INETSTL_VER_INETSTL_FILESYSTEM_HPP_FILESYSTEM_TRAITS_MAJOR 4
|
|
# define INETSTL_VER_INETSTL_FILESYSTEM_HPP_FILESYSTEM_TRAITS_MINOR 2
|
|
# define INETSTL_VER_INETSTL_FILESYSTEM_HPP_FILESYSTEM_TRAITS_REVISION 2
|
|
# define INETSTL_VER_INETSTL_FILESYSTEM_HPP_FILESYSTEM_TRAITS_EDIT 76
|
|
#endif /* !STLSOFT_DOCUMENTATION_SKIP_SECTION */
|
|
|
|
/* /////////////////////////////////////////////////////////////////////////
|
|
* Includes
|
|
*/
|
|
|
|
#ifndef INETSTL_INCL_INETSTL_H_INETSTL
|
|
# include <inetstl/inetstl.h>
|
|
#endif /* !INETSTL_INCL_INETSTL_H_INETSTL */
|
|
#ifndef INETSTL_OS_IS_WINDOWS
|
|
# error This file is currently compatible only with the Win32/Win64 API
|
|
#endif /* !INETSTL_OS_IS_WINDOWS */
|
|
#ifndef STLSOFT_INCL_STLSOFT_CONVERSION_HPP_SAP_CAST
|
|
# include <stlsoft/conversion/sap_cast.hpp>
|
|
#endif /* !STLSOFT_INCL_STLSOFT_CONVERSION_HPP_SAP_CAST */
|
|
#ifndef STLSOFT_INCL_STLSOFT_CONVERSION_HPP_ANY_CAST
|
|
# include <stlsoft/conversion/any_cast.hpp>
|
|
#endif /* !STLSOFT_INCL_STLSOFT_CONVERSION_HPP_ANY_CAST */
|
|
#ifndef STLSOFT_INCL_STLSOFT_INTERNAL_H_SAFESTR
|
|
# include <stlsoft/internal/safestr.h>
|
|
#endif /* !STLSOFT_INCL_STLSOFT_INTERNAL_H_SAFESTR */
|
|
|
|
#ifndef STLSOFT_INCL_H_STRING
|
|
# define STLSOFT_INCL_H_STRING
|
|
# include <string.h>
|
|
#endif /* !STLSOFT_INCL_H_STRING */
|
|
#ifndef STLSOFT_INCL_H_WCHAR
|
|
# define STLSOFT_INCL_H_WCHAR
|
|
# include <wchar.h>
|
|
#endif /* !STLSOFT_INCL_H_WCHAR */
|
|
|
|
/* /////////////////////////////////////////////////////////////////////////
|
|
* Namespace
|
|
*/
|
|
|
|
#ifndef _INETSTL_NO_NAMESPACE
|
|
# if defined(_STLSOFT_NO_NAMESPACE) || \
|
|
defined(STLSOFT_DOCUMENTATION_SKIP_SECTION)
|
|
/* There is no stlsoft namespace, so must define ::inetstl */
|
|
namespace inetstl
|
|
{
|
|
# else
|
|
/* Define stlsoft::inetstl_project */
|
|
|
|
namespace stlsoft
|
|
{
|
|
|
|
namespace inetstl_project
|
|
{
|
|
|
|
# endif /* _STLSOFT_NO_NAMESPACE */
|
|
#endif /* !_INETSTL_NO_NAMESPACE */
|
|
|
|
/* /////////////////////////////////////////////////////////////////////////
|
|
* Classes
|
|
*/
|
|
|
|
#ifdef STLSOFT_DOCUMENTATION_SKIP_SECTION
|
|
/** \brief Traits for accessing the correct file-system functions for a given character type
|
|
*
|
|
* \ingroup group__library__filesystem
|
|
*
|
|
* filesystem_traits is a traits class for determining the correct file-system
|
|
* structures and functions for a given character type.
|
|
*
|
|
* \param C The character type
|
|
*/
|
|
template <ss_typename_param_k C>
|
|
struct filesystem_traits
|
|
{
|
|
/// \name Member Types
|
|
/// @{
|
|
public:
|
|
/// The character type
|
|
typedef C char_type;
|
|
/// The size type
|
|
typedef is_size_t size_type;
|
|
/// The difference type
|
|
typedef is_ptrdiff_t difference_type;
|
|
/// The find data type
|
|
typedef WIN32_FIND_DATA find_data_type;
|
|
/// The stat data type
|
|
typedef WIN32_FIND_DATA stat_data_type;
|
|
/// The current instantion of the type
|
|
typedef filesystem_traits<C> class_type;
|
|
/// The (signed) integer type
|
|
typedef is_int_t int_type;
|
|
/// The Boolean type
|
|
typedef is_bool_t bool_type;
|
|
/// The type of system error codes
|
|
typedef DWORD error_type;
|
|
/// @}
|
|
|
|
/// \name General string handling
|
|
/// @{
|
|
public:
|
|
/// Copies a specific number of characters from the source to the destination
|
|
static char_type* char_copy(char_type* dest, char_type const* src, size_type n);
|
|
#if !defined(STLSOFT_USING_SAFE_STR_FUNCTIONS) || \
|
|
defined(_CRT_SECURE_NO_DEPRECATE)
|
|
/// Copies the contents of \c src to \c dest
|
|
static char_type* str_copy(char_type* dest, char_type const* src);
|
|
/// Copies the contents of \c src to \c dest, up to cch \c characters
|
|
static char_type* str_n_copy(char_type* dest, char_type const* src, is_size_t cch);
|
|
/// Appends the contents of \c src to \c dest
|
|
static char_type* str_cat(char_type* dest, char_type const* src);
|
|
#endif /* !STLSOFT_USING_SAFE_STR_FUNCTIONS || _CRT_SECURE_NO_DEPRECATE */
|
|
/// Compares the contents of \c src and \c dest
|
|
static int_type str_compare(char_type const* s1, char_type const* s2);
|
|
/// Compares the contents of \c src and \c dest in a case-insensitive fashion
|
|
static int_type str_compare_no_case(char_type const* s1, char_type const* s2);
|
|
/// Compares the contents of \c src and \c dest up to \c cch characters
|
|
static int_type str_n_compare(char_type const* s1, char_type const* s2, size_type cch);
|
|
/// Evaluates the length of \c src
|
|
static size_type str_len(char_type const* src);
|
|
/// Finds the given character \c ch in \c s
|
|
static char_type* str_chr(char_type const* s, char_type ch);
|
|
/// Finds the rightmost instance \c ch in \c s
|
|
static char_type* str_rchr(char_type const* s, char_type ch);
|
|
/// Finds the given substring \c sub in \c s
|
|
static char_type* str_str(char_type const* s, char_type const* sub);
|
|
/// @}
|
|
|
|
/// \name File-system entry names
|
|
/// @{
|
|
public:
|
|
/// Appends a path name separator to \c dir if one does not exist
|
|
///
|
|
/// \see \link #path_name_separator path_name_separator() \endlink
|
|
static char_type* ensure_dir_end(char_type* dir);
|
|
/// Removes the path name separator from the end of \c dir, if it has it
|
|
///
|
|
/// \see \link #path_name_separator path_name_separator() \endlink
|
|
static char_type* remove_dir_end(char_type* dir);
|
|
/// Returns \c true if \c dir has trailing path name separator
|
|
///
|
|
/// \see \link #path_name_separator path_name_separator() \endlink
|
|
static bool_type has_dir_end(char_type const* dir);
|
|
|
|
/// Returns \c true if dir is \c "." or \c ".."
|
|
static bool_type is_dots(char_type const* dir);
|
|
/// Returns \c true if path is rooted
|
|
static bool_type is_path_rooted(char_type const* path);
|
|
/// Returns \c true if path is an absolute path
|
|
static bool_type is_path_absolute(char_type const* path);
|
|
|
|
/// \brief Returns \c true if the character is a path-name separator
|
|
///
|
|
/// \note Both \c / and \c \\ are interpreted as a path name separator
|
|
static bool_type is_path_name_separator(char_type ch);
|
|
|
|
/// Returns the path separator
|
|
///
|
|
/// This is the separator that is used to separate multiple paths on the operating system. On UNIX it is ':'
|
|
static char_type path_separator();
|
|
/// Returns the path name separator
|
|
///
|
|
/// This is the separator that is used to separate parts of a path on the operating system. On UNIX it is '/'
|
|
static char_type path_name_separator();
|
|
/// Returns the wildcard pattern that represents all possible matches
|
|
///
|
|
/// \note It is '*'
|
|
static char_type const* pattern_all();
|
|
/// Gets the full path name into the given buffer, returning a pointer to the file-part
|
|
static size_type get_full_path_name(HINTERNET hconn, char_type const* fileName, size_type cchBuffer, char_type* buffer, char_type** ppFile);
|
|
/// Gets the full path name into the given buffer
|
|
static size_type get_full_path_name(HINTERNET hconn, char_type const* fileName, size_type cchBuffer, char_type* buffer);
|
|
/// @}
|
|
|
|
/// \name Internet connectivity
|
|
/// @{
|
|
public:
|
|
/// Opens a WinInet session
|
|
static HINTERNET internet_open(char_type const* agent, is_dword_t accessType, char_type const* proxy, char_type const* proxyBypass, is_dword_t flags);
|
|
/// Makes a connection to a FTP or HTTP site
|
|
static HINTERNET internet_connect(HINTERNET hsess, char_type const* server, INTERNET_PORT port, char_type const* userName, char_type const* password, is_dword_t service, is_dword_t flags, is_dword_t context);
|
|
/// Closes the connection to the FTP or HTTP site
|
|
static void close_connection(HINTERNET hconn);
|
|
/// @}
|
|
|
|
/// \name File-system enumeration
|
|
/// @{
|
|
public:
|
|
/// Initiate a file-system search
|
|
static HINTERNET find_first_file(HINTERNET hconn, char_type const* spec, find_data_type *findData, is_dword_t flags = 0, is_dword_t context = 0);
|
|
/// Advance a given file-system search
|
|
static bool_type find_next_file(HANDLE h, find_data_type *findData);
|
|
/// Closes the file-search
|
|
static void find_close(HINTERNET hfind);
|
|
/// @}
|
|
|
|
/// \name File-system state
|
|
/// @{
|
|
public:
|
|
/// Sets the current directory to \c dir
|
|
static bool_type set_current_directory(HINTERNET hconn, char_type const* dir);
|
|
/// Retrieves the name of the current directory into \c buffer up to a maximum of \c cchBuffer characters
|
|
static bool_type get_current_directory(HINTERNET hconn, is_size_t &cchBuffer, char_type* buffer);
|
|
|
|
/// Returns whether a file exists or not
|
|
static bool_type file_exists(HINTERNET hconn, char_type const* fileName);
|
|
/// Returns whether the given path represents a file
|
|
static bool_type is_file(HINTERNET hconn, char_type const* path);
|
|
/// Returns whether the given path represents a directory
|
|
static bool_type is_directory(HINTERNET hconn, char_type const* path);
|
|
/// Gets the information for a particular file system entry
|
|
static bool_type stat(HINTERNET hconn, char_type const* path, stat_data_type *stat_data);
|
|
|
|
/// Returns whether the given stat info represents a file
|
|
static bool_type is_file(stat_data_type const* stat_data);
|
|
/// Returns whether the given stat info represents a directory
|
|
static bool_type is_directory(stat_data_type const* stat_data);
|
|
/// Returns whether the given stat info represents a link
|
|
static bool_type is_link(stat_data_type const* stat_data);
|
|
/// Returns whether the given stat info represents a read-only entry
|
|
static bool_type is_readonly(stat_data_type const* stat_data);
|
|
/// @}
|
|
|
|
/// \name File-system control
|
|
/// @{
|
|
public:
|
|
/// Creates a directory
|
|
static bool_type create_directory(HINTERNET hconn, char_type const* dir);
|
|
/// Deletes a directory
|
|
static bool_type remove_directory(HINTERNET hconn, char_type const* dir);
|
|
|
|
/// Delete a file
|
|
static bool_type delete_file(HINTERNET hconn, char_type const* file);
|
|
/// Rename a file
|
|
static bool_type rename_file(HINTERNET hconn, char_type const* currentName, char_type const* newName);
|
|
/// @}
|
|
};
|
|
|
|
#else /* ? STLSOFT_DOCUMENTATION_SKIP_SECTION */
|
|
|
|
template <ss_typename_param_k C>
|
|
struct filesystem_traits;
|
|
|
|
STLSOFT_TEMPLATE_SPECIALISATION
|
|
struct filesystem_traits<is_char_a_t>
|
|
{
|
|
public:
|
|
typedef is_char_a_t char_type;
|
|
typedef is_size_t size_type;
|
|
typedef is_ptrdiff_t difference_type;
|
|
typedef WIN32_FIND_DATAA find_data_type;
|
|
typedef WIN32_FIND_DATAA stat_data_type;
|
|
typedef filesystem_traits<is_char_a_t> class_type;
|
|
typedef is_int_t int_type;
|
|
typedef is_bool_t bool_type;
|
|
typedef DWORD error_type;
|
|
|
|
public:
|
|
static char_type* char_copy(char_type* dest, char_type const* src, size_type n)
|
|
{
|
|
INETSTL_ASSERT(NULL != dest);
|
|
INETSTL_ASSERT(0 == n || NULL != src);
|
|
|
|
return static_cast<char_type*>(::memcpy(dest, src, sizeof(char_type) * n));
|
|
}
|
|
|
|
// General string handling
|
|
#if !defined(STLSOFT_USING_SAFE_STR_FUNCTIONS) || \
|
|
defined(_CRT_SECURE_NO_DEPRECATE)
|
|
static char_type* str_copy(char_type* dest, char_type const* src)
|
|
{
|
|
INETSTL_ASSERT(NULL != dest);
|
|
INETSTL_ASSERT(NULL != src);
|
|
|
|
# ifdef STLSOFT_MIN_CRT
|
|
return ::lstrcpyA(dest, src);
|
|
# else /*? STLSOFT_MIN_CRT */
|
|
return ::strcpy(dest, src);
|
|
# endif /* STLSOFT_MIN_CRT */
|
|
}
|
|
|
|
static char_type* str_n_copy(char_type* dest, char_type const* src, is_size_t cch)
|
|
{
|
|
INETSTL_ASSERT(NULL != dest);
|
|
INETSTL_ASSERT(0 == cch || NULL != src);
|
|
|
|
return ::strncpy(dest, src, cch);
|
|
}
|
|
|
|
static char_type* str_cat(char_type* dest, char_type const* src)
|
|
{
|
|
INETSTL_ASSERT(NULL != dest);
|
|
INETSTL_ASSERT(NULL != src);
|
|
|
|
# ifdef STLSOFT_MIN_CRT
|
|
return ::lstrcatA(dest, src);
|
|
# else /*? STLSOFT_MIN_CRT */
|
|
return ::strcat(dest, src);
|
|
# endif /* STLSOFT_MIN_CRT */
|
|
}
|
|
#endif /* !STLSOFT_USING_SAFE_STR_FUNCTIONS || _CRT_SECURE_NO_DEPRECATE */
|
|
|
|
static int_type str_compare(char_type const* s1, char_type const* s2)
|
|
{
|
|
INETSTL_ASSERT(NULL != s1);
|
|
INETSTL_ASSERT(NULL != s2);
|
|
|
|
#ifdef STLSOFT_MIN_CRT
|
|
return ::lstrcmpA(s1, s2);
|
|
#else /*? STLSOFT_MIN_CRT */
|
|
return ::strcmp(s1, s2);
|
|
#endif /* STLSOFT_MIN_CRT */
|
|
}
|
|
|
|
static int_type str_compare_no_case(char_type const* s1, char_type const* s2)
|
|
{
|
|
INETSTL_ASSERT(NULL != s1);
|
|
INETSTL_ASSERT(NULL != s2);
|
|
|
|
return ::lstrcmpiA(s1, s2);
|
|
}
|
|
|
|
static int_type str_n_compare(char_type const* s1, char_type const* s2, size_type cch)
|
|
{
|
|
INETSTL_ASSERT(NULL != s1);
|
|
INETSTL_ASSERT(NULL != s2);
|
|
|
|
return ::strncmp(s1, s2, cch);
|
|
}
|
|
|
|
static size_type str_len(char_type const* src)
|
|
{
|
|
#ifdef STLSOFT_MIN_CRT
|
|
return static_cast<size_type>(::lstrlenA(src));
|
|
#else /*? STLSOFT_MIN_CRT */
|
|
return ::strlen(src);
|
|
#endif /* STLSOFT_MIN_CRT */
|
|
}
|
|
|
|
static char_type* str_chr(char_type const* s, char_type ch)
|
|
{
|
|
return const_cast<char_type*>(::strchr(s, ch));
|
|
}
|
|
|
|
static char_type* str_rchr(char_type const* s, char_type ch)
|
|
{
|
|
return const_cast<char_type*>(::strrchr(s, ch));
|
|
}
|
|
|
|
static char_type* str_str(char_type const* s, char_type const* sub)
|
|
{
|
|
return const_cast<char_type*>(::strstr(s, sub));
|
|
}
|
|
|
|
// File-system entry names
|
|
static char_type* ensure_dir_end(char_type* dir)
|
|
{
|
|
char_type *end;
|
|
char_type const separator = (NULL == str_chr(dir, '/') && NULL != str_chr(dir, '\\')) ? '\\' : '/';
|
|
|
|
for(end = dir; *end != '\0'; ++end)
|
|
{}
|
|
|
|
if( dir < end &&
|
|
*(end - 1) != separator)
|
|
{
|
|
*end = separator;
|
|
*(end + 1) = '\0';
|
|
}
|
|
|
|
return dir;
|
|
}
|
|
|
|
static char_type* remove_dir_end(char_type* dir)
|
|
{
|
|
char_type *end;
|
|
|
|
for(end = dir; *end != '\0'; ++end)
|
|
{}
|
|
|
|
if( dir < end &&
|
|
*(end - 1) == path_name_separator())
|
|
{
|
|
*(end - 1) = '\0';
|
|
}
|
|
|
|
return dir;
|
|
}
|
|
|
|
static bool_type has_dir_end(char_type const* dir)
|
|
{
|
|
is_size_t len = str_len(dir);
|
|
|
|
return (0 < len) && path_name_separator() == dir[len - 1];
|
|
}
|
|
|
|
static bool_type is_dots(char_type const* dir)
|
|
{
|
|
return dir != 0 &&
|
|
dir[0] == '.' &&
|
|
( dir[1] == '\0' ||
|
|
( dir[1] == '.' &&
|
|
dir[2] == '\0'));
|
|
}
|
|
|
|
static bool_type is_path_rooted(char_type const* path)
|
|
{
|
|
INETSTL_ASSERT(NULL != path);
|
|
|
|
return '/' == *path;
|
|
}
|
|
|
|
static bool_type is_path_absolute(char_type const* path)
|
|
{
|
|
return is_path_rooted(path);
|
|
}
|
|
|
|
static bool_type is_path_name_separator(char_type ch)
|
|
{
|
|
return '/' == ch;
|
|
}
|
|
|
|
static char_type path_separator()
|
|
{
|
|
return ';';
|
|
}
|
|
|
|
static char_type path_name_separator()
|
|
{
|
|
return '/';
|
|
}
|
|
|
|
static char_type const* pattern_all()
|
|
{
|
|
return "*";
|
|
}
|
|
|
|
static size_type get_full_path_name(HINTERNET hconn, char_type const* fileName, size_type cchBuffer, char_type* buffer, char_type** ppFile)
|
|
{
|
|
INETSTL_ASSERT(0 == cchBuffer || NULL != buffer);
|
|
INETSTL_ASSERT(NULL == buffer || 0 != cchBuffer);
|
|
INETSTL_ASSERT(NULL != fileName);
|
|
|
|
// Deduce the separator
|
|
char_type const separator = (NULL == str_chr(fileName, '/') && NULL != str_chr(fileName, '\\')) ? '\\' : '/';
|
|
char_type fullPath[1 + _MAX_PATH];
|
|
size_type len = str_len(fileName);
|
|
|
|
// If we're not rooted, then get the current directory and concatenate
|
|
if(separator != *fileName)
|
|
{
|
|
is_size_t cchBuffer = STLSOFT_NUM_ELEMENTS(fullPath);
|
|
const int isDot = '.' == 0[fileName] && '\0' == 0[fileName];
|
|
|
|
#ifdef __SYNSOFT_DBS_COMPILER_SUPPORTS_PRAGMA_MESSAGE
|
|
# pragma message(_sscomp_fileline_message("This looks a bit dodgy. Better to use an auto_buffer, and cycle the size, testing the return value from get_current_directory"))
|
|
#endif /* __SYNSOFT_DBS_COMPILER_SUPPORTS_PRAGMA_MESSAGE */
|
|
|
|
if(!get_current_directory(hconn, cchBuffer, fullPath))
|
|
{
|
|
fullPath[0] = '\0';
|
|
}
|
|
size_type cchFullPath = str_len(fullPath);
|
|
|
|
if(!isDot)
|
|
{
|
|
if(!has_dir_end(fullPath))
|
|
{
|
|
ensure_dir_end(fullPath);
|
|
++cchFullPath;
|
|
}
|
|
char_copy(&fullPath[0] + cchFullPath, fileName, 1 + len);
|
|
}
|
|
|
|
fileName = fullPath;
|
|
len += cchFullPath;
|
|
}
|
|
|
|
if(NULL != buffer)
|
|
{
|
|
if(cchBuffer < len)
|
|
{
|
|
len = cchBuffer;
|
|
}
|
|
|
|
char_copy(buffer, fileName, cchBuffer);
|
|
|
|
if(NULL != ppFile)
|
|
{
|
|
char_type const* pRSlash = str_rchr(buffer, '/');
|
|
char_type const* pRBackSlash = str_rchr(buffer, '\\');
|
|
|
|
if(pRSlash < pRBackSlash)
|
|
{
|
|
pRSlash = pRBackSlash;
|
|
}
|
|
|
|
if(NULL == pRSlash)
|
|
{
|
|
*ppFile = NULL;
|
|
}
|
|
else
|
|
{
|
|
*ppFile = const_cast<char_type*>(pRSlash) + 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
return len;
|
|
}
|
|
|
|
static size_type get_full_path_name(HINTERNET hconn, char_type const* fileName, size_type cchBuffer, char_type* buffer)
|
|
{
|
|
char_type* pFile;
|
|
|
|
return get_full_path_name(hconn, fileName, cchBuffer, buffer, &pFile);
|
|
}
|
|
|
|
// Internet connectivity
|
|
static HINTERNET internet_open(char_type const* agent, is_dword_t accessType, char_type const* proxy, char_type const* proxyBypass, is_dword_t flags)
|
|
{
|
|
return ::InternetOpenA(agent, accessType, proxy, proxyBypass, flags);
|
|
}
|
|
|
|
static HINTERNET internet_connect(HINTERNET hsess, char_type const* server, INTERNET_PORT port, char_type const* userName, char_type const* password, is_dword_t service, is_dword_t flags, is_dword_t context)
|
|
{
|
|
return ::InternetConnectA(hsess, server, port, userName, password, service, flags, context);
|
|
}
|
|
|
|
static void close_connection(HINTERNET hconn)
|
|
{
|
|
INETSTL_ASSERT(NULL != hconn);
|
|
|
|
::InternetCloseHandle(hconn);
|
|
}
|
|
|
|
// FindFile() API
|
|
static HINTERNET find_first_file(HINTERNET hconn, char_type const* spec, find_data_type *findData, is_dword_t flags = 0, is_dword_t context = 0)
|
|
{
|
|
HINTERNET hfind = ::FtpFindFirstFileA(hconn, spec, stlsoft_ns_qual(any_caster)<find_data_type*, LPWIN32_FIND_DATAA, LPWIN32_FIND_DATAW>(findData), flags, context);
|
|
|
|
#if 0
|
|
if(NULL == hfind)
|
|
{
|
|
findData->cFileName[0] = '\0';
|
|
}
|
|
|
|
printf("find_first_file(0x%08x, %s => %s)\n", hfind, spec, findData->cFileName);
|
|
#endif /* 0 */
|
|
|
|
return hfind;
|
|
}
|
|
|
|
static bool_type find_next_file(HANDLE h, find_data_type *findData)
|
|
{
|
|
return FALSE != ::InternetFindNextFileA(h, findData);
|
|
}
|
|
|
|
static void find_close(HINTERNET hfind)
|
|
{
|
|
INETSTL_ASSERT(NULL != hfind);
|
|
|
|
::InternetCloseHandle(hfind);
|
|
}
|
|
|
|
// File system state
|
|
static bool_type set_current_directory(HINTERNET hconn, char_type const* dir)
|
|
{
|
|
return ::FtpSetCurrentDirectoryA(hconn, dir) != FALSE;
|
|
}
|
|
|
|
static bool_type get_current_directory(HINTERNET hconn, is_size_t &cchBuffer, char_type* buffer)
|
|
{
|
|
return FALSE != ::FtpGetCurrentDirectoryA(hconn, buffer, sap_cast<unsigned long*>(&cchBuffer));
|
|
}
|
|
|
|
static bool_type file_exists(HINTERNET hconn, char_type const* fileName)
|
|
{
|
|
find_data_type data;
|
|
HINTERNET hfind = find_first_file(hconn, fileName, &data);
|
|
|
|
return (NULL == hfind) ? false : (find_close(hfind), true);
|
|
}
|
|
|
|
#if 0
|
|
static bool_type is_file(HINTERNET hconn, char_type const* path);
|
|
static bool_type is_directory(HINTERNET hconn, char_type const* path);
|
|
static bool_type stat(HINTERNET hconn, char_type const* path, stat_data_type *stat_data);
|
|
#endif /* 0 */
|
|
|
|
static bool_type is_file(stat_data_type const* stat_data)
|
|
{
|
|
return FILE_ATTRIBUTE_DIRECTORY != (stat_data->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY);
|
|
}
|
|
static bool_type is_directory(stat_data_type const* stat_data)
|
|
{
|
|
return FILE_ATTRIBUTE_DIRECTORY == (stat_data->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY);
|
|
}
|
|
static bool_type is_link(stat_data_type const* stat_data);
|
|
static bool_type is_readonly(stat_data_type const* stat_data)
|
|
{
|
|
return FILE_ATTRIBUTE_READONLY == (stat_data->dwFileAttributes & FILE_ATTRIBUTE_READONLY);
|
|
}
|
|
|
|
// File system control
|
|
static bool_type create_directory(HINTERNET hconn, char_type const* dir)
|
|
{
|
|
return FALSE != ::FtpCreateDirectoryA(hconn, dir);
|
|
}
|
|
|
|
static bool_type remove_directory(HINTERNET hconn, char_type const* dir)
|
|
{
|
|
return FALSE != ::FtpRemoveDirectoryA(hconn, dir);
|
|
}
|
|
|
|
static bool_type delete_file(HINTERNET hconn, char_type const* file)
|
|
{
|
|
return FALSE != ::FtpDeleteFileA(hconn, file);
|
|
}
|
|
|
|
static bool_type rename_file(HINTERNET hconn, char_type const* currentName, char_type const* newName)
|
|
{
|
|
return FALSE != ::FtpRenameFileA(hconn, currentName, newName);
|
|
}
|
|
};
|
|
|
|
STLSOFT_TEMPLATE_SPECIALISATION
|
|
struct filesystem_traits<is_char_w_t>
|
|
{
|
|
public:
|
|
typedef is_char_w_t char_type;
|
|
typedef is_size_t size_type;
|
|
typedef is_ptrdiff_t difference_type;
|
|
typedef WIN32_FIND_DATAW find_data_type;
|
|
typedef WIN32_FIND_DATAW stat_data_type;
|
|
typedef filesystem_traits<is_char_w_t> class_type;
|
|
typedef is_int_t int_type;
|
|
typedef is_bool_t bool_type;
|
|
typedef DWORD error_type;
|
|
|
|
public:
|
|
static char_type* char_copy(char_type* dest, char_type const* src, size_type n)
|
|
{
|
|
INETSTL_ASSERT(NULL != dest);
|
|
INETSTL_ASSERT(0 == n || NULL != src);
|
|
|
|
return static_cast<char_type*>(::memcpy(dest, src, sizeof(char_type) * n));
|
|
}
|
|
|
|
// General string handling
|
|
#if !defined(STLSOFT_USING_SAFE_STR_FUNCTIONS) || \
|
|
defined(_CRT_SECURE_NO_DEPRECATE)
|
|
static char_type* str_copy(char_type* dest, char_type const* src)
|
|
{
|
|
INETSTL_ASSERT(NULL != dest);
|
|
INETSTL_ASSERT(NULL != src);
|
|
|
|
# ifdef STLSOFT_MIN_CRT
|
|
return ::lstrcpyW(dest, src);
|
|
# else /*? STLSOFT_MIN_CRT */
|
|
return ::wcscpy(dest, src);
|
|
# endif /* STLSOFT_MIN_CRT */
|
|
}
|
|
|
|
static char_type* str_n_copy(char_type* dest, char_type const* src, is_size_t cch)
|
|
{
|
|
INETSTL_ASSERT(NULL != dest);
|
|
INETSTL_ASSERT(0 == cch || NULL != src);
|
|
|
|
return ::wcsncpy(dest, src, cch);
|
|
}
|
|
|
|
static char_type* str_cat(char_type* dest, char_type const* src)
|
|
{
|
|
INETSTL_ASSERT(NULL != dest);
|
|
INETSTL_ASSERT(NULL != src);
|
|
|
|
# ifdef STLSOFT_MIN_CRT
|
|
return ::lstrcatW(dest, src);
|
|
# else /*? STLSOFT_MIN_CRT */
|
|
return ::wcscat(dest, src);
|
|
# endif /* STLSOFT_MIN_CRT */
|
|
}
|
|
#endif /* !STLSOFT_USING_SAFE_STR_FUNCTIONS || _CRT_SECURE_NO_DEPRECATE */
|
|
|
|
static int_type str_compare(char_type const* s1, char_type const* s2)
|
|
{
|
|
INETSTL_ASSERT(NULL != s1);
|
|
INETSTL_ASSERT(NULL != s2);
|
|
|
|
#ifdef STLSOFT_MIN_CRT
|
|
return ::lstrcmpW(s1, s2);
|
|
#else /*? STLSOFT_MIN_CRT */
|
|
return ::wcscmp(s1, s2);
|
|
#endif /* STLSOFT_MIN_CRT */
|
|
}
|
|
|
|
static int_type str_compare_no_case(char_type const* s1, char_type const* s2)
|
|
{
|
|
INETSTL_ASSERT(NULL != s1);
|
|
INETSTL_ASSERT(NULL != s2);
|
|
|
|
return ::lstrcmpiW(s1, s2);
|
|
}
|
|
|
|
static int_type str_n_compare(char_type const* s1, char_type const* s2, size_type cch)
|
|
{
|
|
INETSTL_ASSERT(NULL != s1);
|
|
INETSTL_ASSERT(NULL != s2);
|
|
|
|
return ::wcsncmp(s1, s2, cch);
|
|
}
|
|
|
|
static size_type str_len(char_type const* src)
|
|
{
|
|
#ifdef STLSOFT_MIN_CRT
|
|
return static_cast<size_type>(::lstrlenW(src));
|
|
#else /*? STLSOFT_MIN_CRT */
|
|
return ::wcslen(src);
|
|
#endif /* STLSOFT_MIN_CRT */
|
|
}
|
|
|
|
static char_type* str_chr(char_type const* s, char_type ch)
|
|
{
|
|
return const_cast<char_type*>(::wcschr(s, ch));
|
|
}
|
|
|
|
static char_type* str_rchr(char_type const* s, char_type ch)
|
|
{
|
|
return const_cast<char_type*>(::wcsrchr(s, ch));
|
|
}
|
|
|
|
static char_type* str_str(char_type const* s, char_type const* sub)
|
|
{
|
|
return const_cast<char_type*>(::wcsstr(s, sub));
|
|
}
|
|
|
|
// File-system entry names
|
|
static char_type* ensure_dir_end(char_type* dir)
|
|
{
|
|
char_type *end;
|
|
char_type const separator = (NULL == str_chr(dir, L'/') && NULL != str_chr(dir, L'\\')) ? L'\\' : L'/';
|
|
|
|
for(end = dir; *end != L'\0'; ++end)
|
|
{}
|
|
|
|
if( dir < end &&
|
|
*(end - 1) != separator)
|
|
{
|
|
*end = separator;
|
|
*(end + 1) = L'\0';
|
|
}
|
|
|
|
return dir;
|
|
}
|
|
|
|
static char_type* remove_dir_end(char_type* dir)
|
|
{
|
|
char_type *end;
|
|
|
|
for(end = dir; *end != '\0'; ++end)
|
|
{}
|
|
|
|
if( dir < end &&
|
|
*(end - 1) == path_name_separator())
|
|
{
|
|
*(end - 1) = '\0';
|
|
}
|
|
|
|
return dir;
|
|
}
|
|
|
|
static bool_type has_dir_end(char_type const* dir)
|
|
{
|
|
is_size_t len = str_len(dir);
|
|
|
|
return (0 < len) && path_name_separator() == dir[len - 1];
|
|
}
|
|
|
|
static bool_type is_dots(char_type const* dir)
|
|
{
|
|
return dir != 0 &&
|
|
dir[0] == '.' &&
|
|
( dir[1] == L'\0' ||
|
|
( dir[1] == L'.' &&
|
|
dir[2] == L'\0'));
|
|
}
|
|
|
|
static bool_type is_path_rooted(char_type const* path)
|
|
{
|
|
INETSTL_ASSERT(NULL != path);
|
|
|
|
return L'/' == *path;
|
|
}
|
|
|
|
static bool_type is_path_absolute(char_type const* path)
|
|
{
|
|
return is_path_rooted(path);
|
|
}
|
|
|
|
static bool_type is_path_name_separator(char_type ch)
|
|
{
|
|
return L'/' == ch;
|
|
}
|
|
|
|
static char_type path_separator()
|
|
{
|
|
return L';';
|
|
}
|
|
|
|
static char_type path_name_separator()
|
|
{
|
|
return L'/';
|
|
}
|
|
|
|
static char_type const* pattern_all()
|
|
{
|
|
return L"*";
|
|
}
|
|
|
|
static size_type get_full_path_name(HINTERNET hconn, char_type const* fileName, size_type cchBuffer, char_type* buffer, char_type** ppFile)
|
|
{
|
|
INETSTL_ASSERT(0 == cchBuffer || NULL != buffer);
|
|
INETSTL_ASSERT(NULL == buffer || 0 != cchBuffer);
|
|
INETSTL_ASSERT(NULL != fileName);
|
|
|
|
// Deduce the separator
|
|
char_type const separator = (NULL == str_chr(fileName, L'/') && NULL != str_chr(fileName, L'\\')) ? L'\\' : L'/';
|
|
char_type fullPath[1 + _MAX_PATH];
|
|
size_type len = str_len(fileName);
|
|
|
|
// If we're not rooted, then get the current directory and concatenate
|
|
if(separator != *fileName)
|
|
{
|
|
is_size_t cchBuffer = STLSOFT_NUM_ELEMENTS(fullPath);
|
|
const int isDot = L'.' == 0[fileName] && L'\0' == 0[fileName];
|
|
|
|
#ifdef __SYNSOFT_DBS_COMPILER_SUPPORTS_PRAGMA_MESSAGE
|
|
# pragma message(_sscomp_fileline_message("This looks a bit dodgy. Better to use an auto_buffer, and cycle the size, testing the return value from get_current_directory"))
|
|
#endif /* __SYNSOFT_DBS_COMPILER_SUPPORTS_PRAGMA_MESSAGE */
|
|
|
|
if(!get_current_directory(hconn, cchBuffer, fullPath))
|
|
{
|
|
fullPath[0] = L'\0';
|
|
}
|
|
size_type cchFullPath = str_len(fullPath);
|
|
|
|
if(!isDot)
|
|
{
|
|
if(!has_dir_end(fullPath))
|
|
{
|
|
ensure_dir_end(fullPath);
|
|
++cchFullPath;
|
|
}
|
|
char_copy(&fullPath[0] + cchFullPath, fileName, 1 + len);
|
|
}
|
|
|
|
fileName = fullPath;
|
|
len += cchFullPath;
|
|
}
|
|
|
|
if(NULL != buffer)
|
|
{
|
|
if(cchBuffer < len)
|
|
{
|
|
len = cchBuffer;
|
|
}
|
|
|
|
char_copy(buffer, fileName, cchBuffer);
|
|
|
|
if(NULL != ppFile)
|
|
{
|
|
char_type const* pRSlash = str_rchr(buffer, L'/');
|
|
char_type const* pRBackSlash = str_rchr(buffer, L'\\');
|
|
|
|
if(pRSlash < pRBackSlash)
|
|
{
|
|
pRSlash = pRBackSlash;
|
|
}
|
|
|
|
if(NULL == pRSlash)
|
|
{
|
|
*ppFile = NULL;
|
|
}
|
|
else
|
|
{
|
|
*ppFile = const_cast<char_type*>(pRSlash) + 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
return len;
|
|
}
|
|
|
|
static size_type get_full_path_name(HINTERNET hconn, char_type const* fileName, size_type cchBuffer, char_type* buffer)
|
|
{
|
|
char_type* pFile;
|
|
|
|
return get_full_path_name(hconn, fileName, cchBuffer, buffer, &pFile);
|
|
}
|
|
|
|
// Internet connectivity
|
|
static HINTERNET internet_open(char_type const* agent, is_dword_t accessType, char_type const* proxy, char_type const* proxyBypass, is_dword_t flags)
|
|
{
|
|
return ::InternetOpenW(agent, accessType, proxy, proxyBypass, flags);
|
|
}
|
|
|
|
static HINTERNET internet_connect(HINTERNET hsess, char_type const* server, INTERNET_PORT port, char_type const* userName, char_type const* password, is_dword_t service, is_dword_t flags, is_dword_t context)
|
|
{
|
|
return ::InternetConnectW(hsess, server, port, userName, password, service, flags, context);
|
|
}
|
|
|
|
static void close_connection(HINTERNET hconn)
|
|
{
|
|
INETSTL_ASSERT(NULL != hconn);
|
|
|
|
::InternetCloseHandle(hconn);
|
|
}
|
|
|
|
// FindFile() API
|
|
static HINTERNET find_first_file(HINTERNET hconn, char_type const* spec, find_data_type *findData, is_dword_t flags = 0, is_dword_t context = 0)
|
|
{
|
|
return ::FtpFindFirstFileW(hconn, spec, stlsoft_ns_qual(any_caster)<find_data_type*, LPWIN32_FIND_DATAA, LPWIN32_FIND_DATAW>(findData), flags, context);
|
|
}
|
|
|
|
static bool_type find_next_file(HANDLE h, find_data_type *findData)
|
|
{
|
|
return FALSE != ::InternetFindNextFileW(h, findData);
|
|
}
|
|
|
|
static void find_close(HINTERNET hfind)
|
|
{
|
|
INETSTL_ASSERT(NULL != hfind);
|
|
|
|
::InternetCloseHandle(hfind);
|
|
}
|
|
|
|
// File system state
|
|
static bool_type set_current_directory(HINTERNET hconn, char_type const* dir)
|
|
{
|
|
return ::FtpSetCurrentDirectoryW(hconn, dir) != FALSE;
|
|
}
|
|
|
|
static bool_type get_current_directory(HINTERNET hconn, is_size_t &cchBuffer, char_type* buffer)
|
|
{
|
|
return FALSE != ::FtpGetCurrentDirectoryW(hconn, buffer, sap_cast<unsigned long*>(&cchBuffer));
|
|
}
|
|
|
|
/// Returns whether a file exists or not
|
|
static bool_type file_exists(HINTERNET hconn, char_type const* fileName)
|
|
{
|
|
find_data_type data;
|
|
HINTERNET hfind = find_first_file(hconn, fileName, &data);
|
|
|
|
return (NULL == hfind) ? false : (find_close(hfind), true);
|
|
}
|
|
|
|
#if 0
|
|
static bool_type is_file(HINTERNET hconn, char_type const* path);
|
|
static bool_type is_directory(HINTERNET hconn, char_type const* path);
|
|
static bool_type stat(HINTERNET hconn, char_type const* path, stat_data_type *stat_data);
|
|
#endif /* 0 */
|
|
|
|
static bool_type is_file(stat_data_type const* stat_data)
|
|
{
|
|
return FILE_ATTRIBUTE_DIRECTORY != (stat_data->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY);
|
|
}
|
|
static bool_type is_directory(stat_data_type const* stat_data)
|
|
{
|
|
return FILE_ATTRIBUTE_DIRECTORY == (stat_data->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY);
|
|
}
|
|
static bool_type is_link(stat_data_type const* stat_data);
|
|
static bool_type is_readonly(stat_data_type const* stat_data)
|
|
{
|
|
return FILE_ATTRIBUTE_READONLY == (stat_data->dwFileAttributes & FILE_ATTRIBUTE_READONLY);
|
|
}
|
|
|
|
// File system control
|
|
static bool_type create_directory(HINTERNET hconn, char_type const* dir)
|
|
{
|
|
return FALSE != ::FtpCreateDirectoryW(hconn, dir);
|
|
}
|
|
|
|
static bool_type remove_directory(HINTERNET hconn, char_type const* dir)
|
|
{
|
|
return FALSE != ::FtpRemoveDirectoryW(hconn, dir);
|
|
}
|
|
|
|
static bool_type delete_file(HINTERNET hconn, char_type const* file)
|
|
{
|
|
return FALSE != ::FtpDeleteFileW(hconn, file);
|
|
}
|
|
|
|
static bool_type rename_file(HINTERNET hconn, char_type const* currentName, char_type const* newName)
|
|
{
|
|
return FALSE != ::FtpRenameFileW(hconn, currentName, newName);
|
|
}
|
|
};
|
|
|
|
#endif /* STLSOFT_DOCUMENTATION_SKIP_SECTION */
|
|
|
|
/* ////////////////////////////////////////////////////////////////////// */
|
|
|
|
#ifndef _INETSTL_NO_NAMESPACE
|
|
# if defined(_STLSOFT_NO_NAMESPACE) || \
|
|
defined(STLSOFT_DOCUMENTATION_SKIP_SECTION)
|
|
} // namespace inetstl
|
|
# else
|
|
} // namespace inetstl_project
|
|
} // namespace stlsoft
|
|
# endif /* _STLSOFT_NO_NAMESPACE */
|
|
#endif /* !_INETSTL_NO_NAMESPACE */
|
|
|
|
/* ////////////////////////////////////////////////////////////////////// */
|
|
|
|
#endif /* INETSTL_INCL_INETSTL_FILESYSTEM_HPP_FILESYSTEM_TRAITS */
|
|
|
|
/* ///////////////////////////// end of file //////////////////////////// */
|