Windows2000/private/inet/wininet/urlcache/unxcache.cxx
2020-09-30 17:12:32 +02:00

335 lines
6.7 KiB
C++

#include <cache.hxx>
#include <conmgr.hxx>
#include <time.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <dirent.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>
extern int errno;
/* Code swiped from cachecfg.cxx */
static BOOL _NormalisePath(LPCTSTR pszPath, LPCTSTR pszEnvVar,
LPTSTR pszResult, UINT cbResult)
{
TCHAR szEnvVar[MAX_PATH];
// don't count the NULL
ExpandEnvironmentStrings(pszEnvVar, szEnvVar, sizeof(szEnvVar)-1);
DWORD dwEnvVar = lstrlen(szEnvVar);
if (CompareString(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, szEnvVar,
dwEnvVar, pszPath, dwEnvVar) == 2)
{
if (lstrlen(pszPath) + dwEnvVar < cbResult)
{
strncpy(pszResult, pszEnvVar, MAX_PATH);
strncat(pszResult, pszPath + dwEnvVar, MAX_PATH);
return TRUE;
}
}
return FALSE;
}
void UnixNormalisePath(LPTSTR pszOrigPath, LPCTSTR pszEnvVar)
{
TCHAR szScratch[MAX_PATH];
if (_NormalisePath(pszOrigPath,pszEnvVar,szScratch,sizeof(szScratch)))
strncpy(pszOrigPath,szScratch,MAX_PATH);
}
void UnixNormaliseIfCachePath(LPTSTR pszOrigPath, LPCTSTR pszEnvVar,
LPCTSTR pszKeyName)
{
if (!strncmp(pszKeyName,CACHE_PATH_VALUE,lstrlen(CACHE_PATH_VALUE)))
UnixNormalisePath(pszOrigPath,pszEnvVar);
}
int UnixPathExists(LPCTSTR pszPath)
{
struct stat statbuf;
if (stat(pszPath, &statbuf) < 0)
{
/* If path does not exist */
if (errno == ENOENT)
return 0;
else
return -1;
}
/* TODO */
/* Make sure path points to a directory */
return 1;
}
void UnixGetValidParentPath(LPTSTR szDevice)
{
TCHAR szDeviceExists[MAX_PATH];
PTSTR pszDeviceExists = NULL;
PTSTR pszEnd = NULL;
if (!szDevice)
return;
lstrcpy(szDeviceExists, szDevice);
pszDeviceExists = szDeviceExists;
pszEnd = szDeviceExists + lstrlen(szDeviceExists);
for(;;)
{
int fPathExists;
if (pszEnd == pszDeviceExists)
break;
fPathExists = UnixPathExists(pszDeviceExists);
if (fPathExists == -1)
{
/* Error */
break;
}
else
if (fPathExists == 0)
{
/* Path does not exist */
while (*pszEnd != DIR_SEPARATOR_CHAR &&
pszEnd != pszDeviceExists)
pszEnd--;
*pszEnd = '\0';
continue;
}
else
{
/* Path exists */
lstrcpy(szDevice, pszDeviceExists);
break;
}
}
}
/* CopyDir */
static int DoCopy();
static int UnixCopyCacheFile(const char* file_src,
const char* file_dest,
mode_t fmode);
static int UnixCreateCacheFolder( const char* dir_dest, mode_t fmode);
#ifndef BUFSIZ
#define BUFSIZ 4096
#endif /* BUFSIZ */
#define CUR_DIR "."
#define PREV_DIR ".."
static char* pathdir1 = NULL;
static char* pathdir2 = NULL;
int CopyDir(const char* dirname1, const char* dirname2)
{
int Error = 0;
struct stat statdir1, statdir2;
if (!dirname1 || !dirname2)
{
goto Cleanup;
}
/* We are assuming that dirname1 and dirname2 are absolute paths */
if (stat(dirname1, &statdir1) < 0)
{
Error = errno;
goto Cleanup;
}
else
if (!S_ISDIR(statdir1.st_mode))
{
Error = -1; /* source is not directory */
goto Cleanup;
}
if (stat(dirname2, &statdir2) < 0)
{
if (errno != ENOENT)
{
Error = errno;
goto Cleanup;
}
/* It is fine if the destination dir does not exist
* provided all directories above the leaf dir exist
*/
}
else
if (!S_ISDIR(statdir2.st_mode))
{
Error = -1; /* destination is not directory */
goto Cleanup;
}
pathdir1 = (char*)malloc((MAX_PATH+1)*sizeof(char));
pathdir2 = (char*)malloc((MAX_PATH+1)*sizeof(char));
lstrcpy(pathdir1, dirname1);
lstrcpy(pathdir2, dirname2);
Error = DoCopy();
Cleanup:
if (pathdir1)
free(pathdir1);
if (pathdir2)
free(pathdir2);
pathdir1 = pathdir2 = NULL;
return Error;
}
int DoCopy()
{
struct stat statbuf;
struct dirent *dirp;
DIR *dp;
int Error;
char *ptr1, *ptr2;
if (stat(pathdir1, &statbuf) < 0)
{
Error = errno;
goto Cleanup;
}
/* Check if this is a regular file */
if ((statbuf.st_mode & S_IFMT) == S_IFREG)
{
Error = UnixCopyCacheFile(pathdir1, pathdir2, statbuf.st_mode);
goto Cleanup;
}
/* Now, we are dealing with a directory */
if ((Error = UnixCreateCacheFolder(pathdir2, statbuf.st_mode)))
goto Cleanup;
ptr1 = pathdir1 + lstrlen(pathdir1);
*ptr1++ = '/';
*ptr1 = 0;
ptr2 = pathdir2 + lstrlen(pathdir2);
*ptr2++ = '/';
*ptr2 = 0;
if ((dp = opendir(pathdir1)) == NULL)
{
Error = errno;
goto Cleanup;
}
while ((dirp = readdir(dp)) != NULL)
{
if (!lstrcmp(dirp->d_name, CUR_DIR) ||
!lstrcmp(dirp->d_name, PREV_DIR))
continue;
lstrcpy(ptr1, dirp->d_name);
lstrcpy(ptr2, dirp->d_name);
if ((Error = DoCopy()))
break;
}
ptr1[-1] = 0;
ptr2[-1] = 0;
/* If this fails, ignore this error */
closedir(dp);
Cleanup:
return Error;
}
static int UnixCreateCacheFolder( const char* path_dest, mode_t mode_src)
{
int Error = 0;
struct stat statbuf2;
if (stat(path_dest, &statbuf2) < 0)
{
if (errno == ENOENT)
{
if (mkdir(path_dest, mode_src) < 0)
{
Error = errno;
goto Cleanup;
}
}
else
{
Error = errno;
goto Cleanup;
}
}
else
if (!S_ISDIR(statbuf2.st_mode))
Error = -1; /* we are expecting a directory */
Cleanup:
return Error;
}
int UnixCopyCacheFile(const char* file_src, const char* file_dest, mode_t fmode)
{
int Error = 0;
int fd1, fd2;
char buf[BUFSIZ];
int nread, nwrite;
if ((fd1 = open(file_src, O_RDONLY)) < 0)
{
Error = errno;
goto Cleanup;
}
if ((fd2 = open(file_dest, O_CREAT|O_TRUNC|O_WRONLY, fmode)) < 0)
{
Error = errno;
goto Cleanup;
}
while((nread = read(fd1, buf, BUFSIZ)) > 0)
{
if ((nwrite = write(fd2, buf, nread)) != nread)
{
Error = errno;
goto Cleanup;
}
}
Error = 0;
Cleanup:
if (fd1 > 0)
close(fd1);
if (fd2 > 0)
close(fd2);
return Error;
}