4157 lines
131 KiB
C++
4157 lines
131 KiB
C++
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1997 - 1999
|
|
// File: update.cpp
|
|
|
|
// Authors;
|
|
// Jeff Saathoff (jeffreys)
|
|
|
|
// Notes;
|
|
// SyncMgr integration
|
|
|
|
#include "pch.h"
|
|
#include "msgbox.h" // CscWin32Message
|
|
#include "folder.h"
|
|
#include <openfile.h> // OpenOfflineFile
|
|
#include "cscst.h" // PostToSystray
|
|
#include "uihooks.h" // Self-host notifications
|
|
#include "fopendlg.h" // OpenFilesWarningDialog
|
|
#include "statdlg.h" // ReconnectServers
|
|
#include "security.h"
|
|
|
|
#define RAS_CONNECT_DELAY (10 * 1000)
|
|
|
|
// Maximum length of username
|
|
#define MAX_USERNAME_CHARS 64
|
|
|
|
// SYNCTHREADDATA.dwSyncStatus flags
|
|
#define SDS_SYNC_OUT 0x00000001 // CSCMergeShare
|
|
#define SDS_SYNC_IN_QUICK 0x00000002 // CSCFillSparseFiles(FALSE)
|
|
#define SDS_SYNC_IN_FULL 0x00000004 // CSCFillSparseFiles(TRUE)
|
|
#define SDS_SYNC_FORCE_INWARD 0x00000008
|
|
#define SDS_SYNC_RAS_CONNECTED 0x00000010
|
|
#define SDS_SYNC_RESTART_MERGE 0x00000020
|
|
#define SDS_SYNC_DELETE_DELETE 0x00000040
|
|
#define SDS_SYNC_DELETE_RESTORE 0x00000080
|
|
#define SDS_SYNC_AUTOCACHE 0x00000100
|
|
#define SDS_SYNC_CONFLICT_KEEPLOCAL 0x00000200
|
|
#define SDS_SYNC_CONFLICT_KEEPNET 0x00000400
|
|
#define SDS_SYNC_CONFLICT_KEEPBOTH 0x00000800
|
|
#define SDS_SYNC_STARTED 0x00010000
|
|
#define SDS_SYNC_ERROR 0x00020000
|
|
#define SDS_SYNC_CANCELLED 0x00040000
|
|
#define SDS_SYNC_FILE_SKIPPED 0x00080000
|
|
|
|
#define SDS_SYNC_DELETE_CONFLICT_MASK (SDS_SYNC_DELETE_DELETE | SDS_SYNC_DELETE_RESTORE)
|
|
#define SDS_SYNC_FILE_CONFLICT_MASK (SDS_SYNC_CONFLICT_KEEPLOCAL | SDS_SYNC_CONFLICT_KEEPNET | SDS_SYNC_CONFLICT_KEEPBOTH)
|
|
|
|
|
|
// Sync Flags used internally by CCscUpdate
|
|
#define CSC_SYNC_OUT 0x00000001L
|
|
#define CSC_SYNC_IN_QUICK 0x00000002L
|
|
#define CSC_SYNC_IN_FULL 0x00000004L
|
|
#define CSC_SYNC_SETTINGS 0x00000008L
|
|
#define CSC_SYNC_MAYBOTHERUSER 0x00000010L
|
|
#define CSC_SYNC_NOTIFY_SYSTRAY 0x00000020L
|
|
#define CSC_SYNC_LOGOFF 0x00000040L
|
|
#define CSC_SYNC_LOGON 0x00000080L
|
|
#define CSC_SYNC_IDLE 0x00000100L
|
|
#define CSC_SYNC_NONET 0x00000200L
|
|
#define CSC_SYNC_PINFILES 0x00000400L
|
|
#define CSC_SYNC_PIN_RECURSE 0x00000800L
|
|
#define CSC_SYNC_OFWARNINGDONE 0x00001000L
|
|
#define CSC_SYNC_CANCELLED 0x00002000L
|
|
#define CSC_SYNC_SHOWUI_ALWAYS 0x00004000L
|
|
#define CSC_SYNC_IGNORE_ACCESS 0x00008000L
|
|
#define CSC_SYNC_SKIP_EFS 0x00010000L
|
|
#define CSC_SYNC_EFS_WARNING_SHOWN 0x00020000L
|
|
#define CSC_SYNC_RECONNECT 0x00040000L
|
|
|
|
#define CSC_LOCALLY_MODIFIED (FLAG_CSC_COPY_STATUS_DATA_LOCALLY_MODIFIED \
|
|
| FLAG_CSC_COPY_STATUS_LOCALLY_DELETED \
|
|
| FLAG_CSC_COPY_STATUS_LOCALLY_CREATED)
|
|
|
|
HICON g_hCscIcon = NULL;
|
|
|
|
// Used for marshalling data into the SyncMgr process
|
|
typedef struct _CSC_UPDATE_DATA
|
|
{
|
|
DWORD dwUpdateFlags;
|
|
DWORD dwFileBufferOffset;
|
|
} CSC_UPDATE_DATA, *PCSC_UPDATE_DATA;
|
|
|
|
|
|
LPTSTR GetErrorText(DWORD dwErr)
|
|
{
|
|
UINT idString = (UINT)-1;
|
|
LPTSTR pszError = NULL;
|
|
|
|
switch (dwErr)
|
|
{
|
|
case ERROR_INVALID_NAME:
|
|
// "Files of this type cannot be made available offline."
|
|
idString = IDS_CACHING_DISALLOWED;
|
|
break;
|
|
}
|
|
|
|
if ((UINT)-1 != idString)
|
|
{
|
|
LoadStringAlloc(&pszError, g_hInstance, idString);
|
|
}
|
|
else if (NOERROR != dwErr)
|
|
{
|
|
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
|
|
NULL,
|
|
dwErr,
|
|
0,
|
|
(LPTSTR)&pszError,
|
|
1,
|
|
NULL);
|
|
}
|
|
return pszError;
|
|
}
|
|
|
|
|
|
|
|
|
|
// CscRegisterHandler
|
|
|
|
// Purpose: Register/unregister CSC Update handler with SyncMgr
|
|
|
|
// Parameters: bRegister - TRUE to register, FALSE to unregister
|
|
// punkSyncMgr - (optional) instance of SyncMgr to use
|
|
|
|
// Return: HRESULT
|
|
|
|
|
|
HRESULT
|
|
CscRegisterHandler(BOOL bRegister, LPUNKNOWN punkSyncMgr)
|
|
{
|
|
HRESULT hr;
|
|
HRESULT hrComInit = E_FAIL;
|
|
ISyncMgrRegister *pSyncRegister = NULL;
|
|
const DWORD dwRegFlags = SYNCMGRREGISTERFLAG_CONNECT | SYNCMGRREGISTERFLAG_PENDINGDISCONNECT;
|
|
|
|
static BOOL s_bSyncHandlerRegistered = FALSE;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CscRegisterHandler");
|
|
|
|
if (bRegister && s_bSyncHandlerRegistered)
|
|
TraceLeaveResult(S_OK); // already registered
|
|
// Note: if bRegister and s_bSyncHandlerRegistered are both FALSE, unregister anyway.
|
|
|
|
if (punkSyncMgr)
|
|
{
|
|
hr = punkSyncMgr->QueryInterface(IID_ISyncMgrRegister, (LPVOID*)&pSyncRegister);
|
|
}
|
|
else
|
|
{
|
|
hrComInit = CoInitialize(NULL);
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_SERVER, IID_ISyncMgrRegister, (LPVOID*)&pSyncRegister);
|
|
}
|
|
FailGracefully(hr, "Unable to get ISyncMgrRegister interface");
|
|
|
|
if (bRegister)
|
|
hr = pSyncRegister->RegisterSyncMgrHandler(CLSID_CscUpdateHandler, NULL, dwRegFlags);
|
|
else
|
|
hr = pSyncRegister->UnregisterSyncMgrHandler(CLSID_CscUpdateHandler, dwRegFlags);
|
|
|
|
if (SUCCEEDED(hr))
|
|
s_bSyncHandlerRegistered = bRegister;
|
|
|
|
exit_gracefully:
|
|
|
|
DoRelease(pSyncRegister);
|
|
|
|
if (SUCCEEDED(hrComInit))
|
|
CoUninitialize();
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
|
|
|
|
// CscUpdateCache
|
|
|
|
// Purpose: Invoke SyncMgr to update the CSC cache
|
|
|
|
// Parameters: pNamelist - list of files passed to the CSC SyncMgr handler
|
|
|
|
|
|
// Return: HRESULT
|
|
|
|
|
|
HRESULT
|
|
CscUpdateCache(DWORD dwUpdateFlags, CscFilenameList *pfnl)
|
|
{
|
|
HRESULT hr;
|
|
HRESULT hrComInit = E_FAIL;
|
|
ISyncMgrSynchronizeInvoke *pSyncInvoke = NULL;
|
|
DWORD dwSyncMgrFlags = 0;
|
|
ULONG cbDataLength = sizeof(CSC_UPDATE_DATA);
|
|
PCSC_UPDATE_DATA pUpdateData = NULL;
|
|
PCSC_NAMELIST_HDR pNamelist = NULL;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CscUpdateCache");
|
|
|
|
hrComInit = CoInitialize(NULL);
|
|
hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_SERVER, IID_ISyncMgrSynchronizeInvoke, (LPVOID*)&pSyncInvoke);
|
|
FailGracefully(hr, "Unable to create SyncMgr object");
|
|
|
|
if (dwUpdateFlags & CSC_UPDATE_SELECTION)
|
|
{
|
|
if (NULL == pfnl || (0 == (CSC_UPDATE_SHOWUI_ALWAYS & dwUpdateFlags) && 0 == pfnl->GetShareCount()))
|
|
ExitGracefully(hr, E_INVALIDARG, "CSC_UPDATE_SELECTION with no selection");
|
|
|
|
pNamelist = pfnl->CreateListBuffer();
|
|
if (!pNamelist)
|
|
ExitGracefully(hr, E_OUTOFMEMORY, "Unable to create namelist buffer");
|
|
|
|
cbDataLength += pNamelist->cbSize;
|
|
}
|
|
|
|
|
|
// Alloc a buffer for the cookie data
|
|
|
|
pUpdateData = (PCSC_UPDATE_DATA)LocalAlloc(LPTR, cbDataLength);
|
|
if (!pUpdateData)
|
|
ExitGracefully(hr, E_OUTOFMEMORY, "LocalAlloc failed");
|
|
|
|
pUpdateData->dwUpdateFlags = dwUpdateFlags;
|
|
if (pNamelist)
|
|
{
|
|
pUpdateData->dwFileBufferOffset = sizeof(CSC_UPDATE_DATA);
|
|
CopyMemory(ByteOffset(pUpdateData, pUpdateData->dwFileBufferOffset),
|
|
pNamelist,
|
|
pNamelist->cbSize);
|
|
}
|
|
|
|
if (dwUpdateFlags & CSC_UPDATE_STARTNOW)
|
|
dwSyncMgrFlags |= SYNCMGRINVOKE_STARTSYNC;
|
|
|
|
|
|
// Start SyncMgr
|
|
|
|
hr = pSyncInvoke->UpdateItems(dwSyncMgrFlags,
|
|
CLSID_CscUpdateHandler,
|
|
cbDataLength,
|
|
(LPBYTE)pUpdateData);
|
|
|
|
exit_gracefully:
|
|
|
|
if (pNamelist)
|
|
CscFilenameList::FreeListBuffer(pNamelist);
|
|
|
|
if (pUpdateData)
|
|
LocalFree(pUpdateData);
|
|
|
|
DoRelease(pSyncInvoke);
|
|
|
|
if (SUCCEEDED(hrComInit))
|
|
CoUninitialize();
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
|
|
|
|
// GetNewVersionName
|
|
|
|
// Purpose: Create unique names for copies of a file
|
|
|
|
// Parameters: LPTSTR pszUNCPath - fully qualified UNC name of file
|
|
// LPTSTR pszShare - \\server\share that file lives on
|
|
// LPTSTR pszDrive - drive mapping to use for net operations
|
|
// LPTSTR *ppszNewName - filename for new version returned here (must free)
|
|
|
|
// Return: Win32 error code
|
|
|
|
|
|
DWORD
|
|
GetNewVersionName(LPCTSTR pszUNCPath,
|
|
LPCTSTR pszShare,
|
|
LPCTSTR pszDrive,
|
|
LPTSTR *ppszNewName)
|
|
{
|
|
DWORD dwErr = NOERROR;
|
|
LPTSTR pszDriveLetterPath = NULL;
|
|
LPTSTR pszPath = NULL;
|
|
LPTSTR pszFile = NULL;
|
|
LPTSTR pszExt = NULL;
|
|
LPTSTR pszWildCardName = NULL;
|
|
TCHAR szUserName[MAX_USERNAME_CHARS];
|
|
ULONG nLength;
|
|
ULONG nMaxVersion = 0;
|
|
ULONG cOlderVersions = 0;
|
|
HANDLE hFind = INVALID_HANDLE_VALUE;
|
|
WIN32_FIND_DATA fd;
|
|
LPTSTR pszT;
|
|
|
|
TraceEnter(TRACE_UPDATE, "GetNewVersionName");
|
|
TraceAssert(pszUNCPath != NULL);
|
|
TraceAssert(ppszNewName != NULL);
|
|
|
|
*ppszNewName = NULL;
|
|
|
|
// 1. Split the path into components.
|
|
// 2. Build wildcard name "X:\dir\foo (johndoe v*).txt"
|
|
// 3. Do a findfirst/findnext loop to get the min & max version #
|
|
// and count the number of old versions.
|
|
// 4. Increment the max version # and build the new filename as:
|
|
// "foo (johndoe v<max+1>).txt"
|
|
|
|
// Assume that the UNC name contains more than the share
|
|
TraceAssert(!StrCmpNI(pszUNCPath, pszShare, lstrlen(pszShare)));
|
|
TraceAssert(lstrlen(pszUNCPath) > lstrlen(pszShare));
|
|
|
|
// Copy the path (without \\server\share)
|
|
if (!LocalAllocString(&pszPath, pszUNCPath + lstrlen(pszShare)))
|
|
ExitGracefully(dwErr, ERROR_OUTOFMEMORY, "LocalAllocString failed");
|
|
|
|
// Find the file part of the name
|
|
pszT = PathFindFileName(pszPath);
|
|
if (!pszT)
|
|
ExitGracefully(dwErr, ERROR_INVALID_PARAMETER, "Incomplete path");
|
|
|
|
// Copy the filename
|
|
if (!LocalAllocString(&pszFile, pszT))
|
|
ExitGracefully(dwErr, ERROR_OUTOFMEMORY, "LocalAllocString failed");
|
|
|
|
// Look for the file extension
|
|
pszT = PathFindExtension(pszFile);
|
|
if (pszT)
|
|
{
|
|
// Copy the extension and truncate the file root at this point
|
|
LocalAllocString(&pszExt, pszT);
|
|
*pszT = TEXT('\0');
|
|
}
|
|
|
|
// Truncate the path
|
|
PathRemoveFileSpec(pszPath);
|
|
|
|
// Get the user name
|
|
nLength = ARRAYSIZE(szUserName);
|
|
if (!GetUserName(szUserName, &nLength))
|
|
LoadString(g_hInstance, IDS_UNKNOWN_USER, szUserName, ARRAYSIZE(szUserName));
|
|
|
|
// Build the wildcard path "X:\dir\foo (johndoe v*).txt"
|
|
|
|
nLength = FormatStringID(&pszWildCardName, g_hInstance, IDS_VERSION_FORMAT, pszFile, szUserName, c_szStar, pszExt);
|
|
if (!nLength)
|
|
ExitGracefully(dwErr, GetLastError(), "Unable to format string");
|
|
|
|
nLength += lstrlen(pszUNCPath) + lstrlen(szUserName);
|
|
|
|
pszDriveLetterPath = (LPTSTR)LocalAlloc(LPTR, MAX(nLength, ULONG(MAX_PATH)) * sizeof(TCHAR));
|
|
if (!pszDriveLetterPath)
|
|
ExitGracefully(dwErr, ERROR_OUTOFMEMORY, "LocalAlloc failed");
|
|
|
|
PathCombine(pszDriveLetterPath, pszDrive, pszPath);
|
|
PathAppend(pszDriveLetterPath, pszWildCardName);
|
|
nLength = (ULONG)(StrStr(pszWildCardName, c_szStar) - pszWildCardName); // remember where the '*' is
|
|
|
|
// Search for existing versions of the file with this username
|
|
hFind = FindFirstFile(pszDriveLetterPath, &fd);
|
|
|
|
if (hFind != INVALID_HANDLE_VALUE)
|
|
{
|
|
ULONG nVersion;
|
|
|
|
do
|
|
{
|
|
nVersion = StrToLong(&fd.cFileName[nLength]);
|
|
|
|
if (nVersion > nMaxVersion)
|
|
{
|
|
nMaxVersion = nVersion;
|
|
}
|
|
|
|
cOlderVersions++;
|
|
}
|
|
while (FindNextFile(hFind, &fd));
|
|
|
|
FindClose(hFind);
|
|
}
|
|
|
|
// Build the new file name to return to the caller.
|
|
// This one is version nMaxVersion+1.
|
|
ULongToString(nMaxVersion+1, pszDriveLetterPath, lstrlen(pszDriveLetterPath));
|
|
nLength = FormatStringID(ppszNewName, g_hInstance, IDS_VERSION_FORMAT, pszFile, szUserName, pszDriveLetterPath, pszExt);
|
|
if (!nLength)
|
|
ExitGracefully(dwErr, GetLastError(), "Unable to format string");
|
|
|
|
exit_gracefully:
|
|
|
|
LocalFreeString(&pszDriveLetterPath);
|
|
LocalFreeString(&pszPath);
|
|
LocalFreeString(&pszFile);
|
|
LocalFreeString(&pszExt);
|
|
LocalFreeString(&pszWildCardName);
|
|
|
|
if (NOERROR != dwErr)
|
|
{
|
|
LocalFreeString(ppszNewName);
|
|
}
|
|
|
|
TraceLeaveValue(dwErr);
|
|
}
|
|
|
|
|
|
|
|
|
|
// ConflictDlgCallback
|
|
|
|
// Purpose: Display local or remote file from conflict dialog
|
|
|
|
// Parameters: hWnd - conflict dialog handle (used as parent for UI)
|
|
// uMsg - one of RFCCM_*
|
|
// wParam - depends on uMsg (unused)
|
|
// lParam - pointer to context data (RFCDLGPARAM)
|
|
|
|
|
|
// Return: TRUE on success, FALSE otherwise
|
|
|
|
|
|
|
|
typedef struct _CONFLICT_DATA
|
|
{
|
|
LPCTSTR pszShare;
|
|
LPCTSTR pszDrive;
|
|
} CONFLICT_DATA;
|
|
|
|
BOOL
|
|
ConflictDlgCallback(HWND hWnd, UINT uMsg, WPARAM /*wParam*/, LPARAM lParam)
|
|
{
|
|
RFCDLGPARAM *pdlgParam = (RFCDLGPARAM*)lParam;
|
|
CONFLICT_DATA cd = {0};
|
|
LPTSTR pszTmpName = NULL;
|
|
ULONG cchShare = 0;
|
|
LPTSTR szFile;
|
|
DWORD dwErr = NOERROR;
|
|
|
|
TraceEnter(TRACE_UPDATE, "ConflictDlgCallback");
|
|
|
|
if (NULL == pdlgParam)
|
|
{
|
|
TraceAssert(FALSE);
|
|
TraceLeaveValue(FALSE);
|
|
}
|
|
|
|
szFile = (LPTSTR)LocalAlloc(LMEM_FIXED,
|
|
MAX(StringByteSize(pdlgParam->pszLocation)
|
|
+ StringByteSize(pdlgParam->pszFilename), MAX_PATH_BYTES));
|
|
if (!szFile)
|
|
TraceLeaveValue(FALSE);
|
|
|
|
if (pdlgParam->lCallerData)
|
|
cd = *(CONFLICT_DATA*)pdlgParam->lCallerData;
|
|
if (cd.pszShare)
|
|
cchShare = lstrlen(cd.pszShare);
|
|
|
|
switch (uMsg)
|
|
{
|
|
case RFCCM_VIEWLOCAL:
|
|
// Build UNC path and view what's in the cache
|
|
PathCombine(szFile, pdlgParam->pszLocation, pdlgParam->pszFilename);
|
|
dwErr = OpenOfflineFile(szFile);
|
|
break;
|
|
|
|
case RFCCM_VIEWNETWORK:
|
|
// Build drive letter (non-UNC) path and ShellExecute it
|
|
PathCombine(szFile, cd.pszDrive, pdlgParam->pszLocation + cchShare);
|
|
PathAppend(szFile, pdlgParam->pszFilename);
|
|
{
|
|
SHELLEXECUTEINFO si = {0};
|
|
si.cbSize = sizeof(si);
|
|
si.fMask = SEE_MASK_FLAG_NO_UI;
|
|
si.hwnd = hWnd;
|
|
si.lpFile = szFile;
|
|
si.nShow = SW_NORMAL;
|
|
|
|
Trace((TEXT("ShellExecuting \"%s\""), szFile));
|
|
if (!ShellExecuteEx(&si))
|
|
dwErr = GetLastError();
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (NOERROR != dwErr)
|
|
CscWin32Message(hWnd, dwErr, CSCUI::SEV_ERROR);
|
|
|
|
LocalFree(szFile);
|
|
TraceLeaveValue(TRUE);
|
|
}
|
|
|
|
|
|
|
|
// ShowConflictDialog
|
|
|
|
// Purpose: Invoke the conflict resolution dialog
|
|
|
|
// Parameters: hWndParent - dialog parent window
|
|
// pszUNCPath - full UNC of file that conflicts
|
|
// pszNewName - filespec to use for new copy of file (e.g. "foo (johndoe v1).txt"
|
|
// pszShare - "\\server\share"
|
|
// pszDrive - "X:" drive mapping of remote connection
|
|
// pfdLocal - Information about local file
|
|
// pfdRemote - Information about remote file
|
|
|
|
|
|
// Return: HRESULT
|
|
|
|
|
|
|
|
typedef int (WINAPI *PFNSYNCMGRRESOLVECONFLICT)(HWND hWndParent, RFCDLGPARAM *pdlgParam);
|
|
TCHAR const c_szSyncMgrDll[] = TEXT("mobsync.dll");
|
|
#ifdef UNICODE
|
|
CHAR const c_szResolveConflict[] = "SyncMgrResolveConflictW";
|
|
#else
|
|
CHAR const c_szResolveConflict[] = "SyncMgrResolveConflictA";
|
|
#endif
|
|
|
|
BOOL FileHasAssociation(LPCTSTR pszFile)
|
|
{
|
|
HRESULT hr = HRESULT_FROM_WIN32(ERROR_NO_ASSOCIATION);
|
|
if (pszFile)
|
|
{
|
|
pszFile = PathFindExtension(pszFile);
|
|
if (pszFile && *pszFile)
|
|
{
|
|
IQueryAssociations *pAssoc = NULL;
|
|
hr = AssocCreate(CLSID_QueryAssociations,
|
|
IID_IQueryAssociations,
|
|
(LPVOID*)&pAssoc);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pAssoc->Init(ASSOCF_IGNOREBASECLASS, pszFile, NULL, NULL);
|
|
pAssoc->Release();
|
|
}
|
|
}
|
|
}
|
|
return SUCCEEDED(hr);
|
|
}
|
|
|
|
int
|
|
ShowConflictDialog(HWND hWndParent,
|
|
LPCTSTR pszUNCPath,
|
|
LPCTSTR pszNewName,
|
|
LPCTSTR pszShare,
|
|
LPCTSTR pszDrive,
|
|
LPWIN32_FIND_DATA pfdLocal,
|
|
LPWIN32_FIND_DATA pfdRemote)
|
|
{
|
|
int nResult = 0;
|
|
TCHAR szUser[MAX_USERNAME_CHARS];
|
|
LPTSTR pszPath = NULL;
|
|
LPTSTR pszFile = NULL;
|
|
TCHAR szRemoteDate[MAX_PATH];
|
|
TCHAR szLocalDate[MAX_PATH];
|
|
ULONG nLength;
|
|
SYSTEMTIME st;
|
|
RFCDLGPARAM dp = {0};
|
|
CONFLICT_DATA cd;
|
|
BOOL bLocalIsDir = FALSE;
|
|
BOOL bRemoteIsDir = FALSE;
|
|
|
|
static PFNSYNCMGRRESOLVECONFLICT pfnResolveConflict = NULL;
|
|
|
|
TraceEnter(TRACE_UPDATE, "ShowConflictDialog");
|
|
TraceAssert(pszUNCPath);
|
|
|
|
if (NULL == pfnResolveConflict)
|
|
{
|
|
// The CSC Update handler is loaded by SyncMgr, so assume the SyncMgr
|
|
// dll is already loaded. We don't want to link to the LIB to keep
|
|
// SyncMgr from loading every time our context menu or icon overlay
|
|
// handler is loaded (for example).
|
|
HMODULE hSyncMgrDll = GetModuleHandle(c_szSyncMgrDll);
|
|
if (NULL != hSyncMgrDll)
|
|
pfnResolveConflict = (PFNSYNCMGRRESOLVECONFLICT)GetProcAddress(hSyncMgrDll,
|
|
c_szResolveConflict);
|
|
if (NULL == pfnResolveConflict)
|
|
return 0;
|
|
}
|
|
TraceAssert(NULL != pfnResolveConflict);
|
|
|
|
szUser[0] = TEXT('\0');
|
|
nLength = ARRAYSIZE(szUser);
|
|
GetUserName(szUser, &nLength);
|
|
|
|
szRemoteDate[0] = TEXT('\0');
|
|
if (NULL != pfdRemote)
|
|
{
|
|
DWORD dwFlags = FDTF_DEFAULT;
|
|
SHFormatDateTime(&pfdRemote->ftLastWriteTime, &dwFlags, szRemoteDate, ARRAYSIZE(szRemoteDate));
|
|
|
|
if (pfdRemote->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
bRemoteIsDir = TRUE;
|
|
}
|
|
|
|
szLocalDate[0] = TEXT('\0');
|
|
if (NULL != pfdLocal)
|
|
{
|
|
DWORD dwFlags = FDTF_DEFAULT;
|
|
SHFormatDateTime(&pfdLocal->ftLastWriteTime, &dwFlags, szLocalDate, ARRAYSIZE(szLocalDate));
|
|
|
|
if (pfdLocal->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
bLocalIsDir = TRUE;
|
|
}
|
|
|
|
if (!LocalAllocString(&pszPath, pszUNCPath))
|
|
ExitGracefully(nResult, 0, "LocalAllocString failed");
|
|
pszFile = PathFindFileName(pszUNCPath);
|
|
PathRemoveFileSpec(pszPath);
|
|
|
|
dp.dwFlags = RFCF_APPLY_ALL;
|
|
dp.pszFilename = pszFile;
|
|
dp.pszLocation = pszPath;
|
|
dp.pszNewName = pszNewName;
|
|
dp.pszNetworkModifiedBy = NULL;
|
|
dp.pszLocalModifiedBy = szUser;
|
|
dp.pszNetworkModifiedOn = szRemoteDate;
|
|
dp.pszLocalModifiedOn = szLocalDate;
|
|
dp.pfnCallBack = NULL;
|
|
dp.lCallerData = 0;
|
|
|
|
// Only turn on the View buttons (set a callback) if we're
|
|
// dealing with files that have associations.
|
|
if (!(bLocalIsDir || bRemoteIsDir) && FileHasAssociation(pszFile))
|
|
{
|
|
// Save both the share name and drive letter for building paths to view files
|
|
cd.pszShare = pszShare;
|
|
cd.pszDrive = pszDrive;
|
|
|
|
dp.pfnCallBack = ConflictDlgCallback;
|
|
dp.lCallerData = (LPARAM)&cd;
|
|
}
|
|
|
|
nResult = (*pfnResolveConflict)(hWndParent, &dp);
|
|
|
|
exit_gracefully:
|
|
|
|
LocalFreeString(&pszPath);
|
|
// No need to free pszFile
|
|
|
|
TraceLeaveValue(nResult);
|
|
}
|
|
|
|
|
|
|
|
|
|
// SyncMgr integration implementation //
|
|
|
|
|
|
|
|
CCscUpdate::CCscUpdate() : m_cRef(1), m_ShareLog(HKEY_CURRENT_USER, c_szCSCShareKey),
|
|
m_pSyncMgrCB(NULL), m_hSyncThreads(NULL),
|
|
m_pFileList(NULL), m_hSyncItems(NULL), m_hwndDlgParent(NULL),
|
|
m_hSyncInProgMutex(NULL), m_pConflictPinList(NULL),
|
|
m_pSilentFolderList(NULL), m_pSpecialFolderList(NULL)
|
|
{
|
|
DllAddRef();
|
|
InitializeCriticalSection(&m_csThreadList);
|
|
if (!g_hCscIcon)
|
|
g_hCscIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_CSCUI_ICON));
|
|
m_hSyncMutex = CreateMutex(NULL, FALSE, c_szSyncMutex);
|
|
}
|
|
|
|
|
|
CCscUpdate::~CCscUpdate()
|
|
{
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::~CCscUpdate");
|
|
|
|
SyncCompleted();
|
|
TraceAssert(NULL == m_hSyncInProgMutex);
|
|
|
|
// We should never get here while a sync thread is still running
|
|
TraceAssert(NULL == m_hSyncThreads || 0 == DPA_GetPtrCount(m_hSyncThreads));
|
|
if (NULL != m_hSyncThreads)
|
|
DPA_Destroy(m_hSyncThreads);
|
|
DeleteCriticalSection(&m_csThreadList);
|
|
|
|
if (NULL != m_hSyncItems)
|
|
DSA_Destroy(m_hSyncItems);
|
|
|
|
DoRelease(m_pSyncMgrCB);
|
|
|
|
delete m_pFileList;
|
|
delete m_pConflictPinList;
|
|
delete m_pSilentFolderList;
|
|
delete m_pSpecialFolderList;
|
|
|
|
if (NULL != m_hSyncMutex)
|
|
CloseHandle(m_hSyncMutex);
|
|
|
|
DllRelease();
|
|
TraceLeaveVoid();
|
|
}
|
|
|
|
|
|
HRESULT WINAPI
|
|
CCscUpdate::CreateInstance(REFIID riid, LPVOID *ppv)
|
|
{
|
|
HRESULT hr;
|
|
CCscUpdate *pThis;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::CreateInstance");
|
|
TraceAssert(IsCSCEnabled());
|
|
|
|
pThis = new CCscUpdate;
|
|
|
|
if (pThis)
|
|
{
|
|
hr = pThis->QueryInterface(riid, ppv);
|
|
pThis->Release();
|
|
}
|
|
else
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
|
|
|
|
// SyncMgr integration implementation (IUnknown) //
|
|
|
|
|
|
|
|
STDMETHODIMP CCscUpdate::QueryInterface(REFIID riid, void **ppv)
|
|
{
|
|
static const QITAB qit[] =
|
|
{
|
|
QITABENT(CCscUpdate, ISyncMgrSynchronize),
|
|
{ 0 },
|
|
};
|
|
return QISearch(this, qit, riid, ppv);
|
|
}
|
|
|
|
STDMETHODIMP_(ULONG) CCscUpdate::AddRef()
|
|
{
|
|
return InterlockedIncrement(&m_cRef);
|
|
}
|
|
|
|
STDMETHODIMP_(ULONG) CCscUpdate::Release()
|
|
{
|
|
if (InterlockedDecrement(&m_cRef))
|
|
return m_cRef;
|
|
|
|
delete this;
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
// Sync Manager integration implementation (ISyncMgrSynchronize) //
|
|
|
|
|
|
|
|
STDMETHODIMP
|
|
CCscUpdate::Initialize(DWORD /*dwReserved*/,
|
|
DWORD dwSyncFlags,
|
|
DWORD cbCookie,
|
|
const BYTE *pCookie)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HKEY hkCSC;
|
|
BOOL bNoNet = TRUE;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::Initialize");
|
|
TraceAssert(IsCSCEnabled());
|
|
|
|
if (!(SYNCMGRFLAG_SETTINGS & dwSyncFlags) && ::IsSyncInProgress())
|
|
{
|
|
|
|
// We need to guard against running multiple syncs at the
|
|
// same time. User notification in the UI is handled where
|
|
// the UI code calls CscUpdate(). This is so that the UI
|
|
// message contains the proper context with respect to what
|
|
// the user is doing.
|
|
|
|
TraceLeaveResult(E_FAIL);
|
|
}
|
|
|
|
m_dwSyncFlags = 0;
|
|
delete m_pFileList;
|
|
m_pFileList = NULL;
|
|
delete m_pConflictPinList;
|
|
m_pConflictPinList = NULL;
|
|
|
|
// We used to get the tray status to check for NoNet, but
|
|
// there's a timing problem at logon (the tray window may not
|
|
// be created yet). So ask RDR instead. If this call fails,
|
|
// then RDR must be dead, so bNoNet defaults to TRUE.
|
|
CSCIsServerOffline(NULL, &bNoNet);
|
|
|
|
switch (dwSyncFlags & SYNCMGRFLAG_EVENTMASK)
|
|
{
|
|
case SYNCMGRFLAG_CONNECT: // Logon
|
|
if (bNoNet)
|
|
ExitGracefully(hr, E_FAIL, "No Logon sync when no net");
|
|
m_dwSyncFlags = CSC_SYNC_OUT | CSC_SYNC_LOGON | CSC_SYNC_NOTIFY_SYSTRAY; // | CSC_SYNC_RECONNECT;
|
|
break;
|
|
|
|
case SYNCMGRFLAG_PENDINGDISCONNECT: // Logoff
|
|
if (bNoNet)
|
|
ExitGracefully(hr, E_FAIL, "No Logoff sync when no net");
|
|
m_dwSyncFlags = CSC_SYNC_LOGOFF;
|
|
if (CConfig::eSyncFull == CConfig::GetSingleton().SyncAtLogoff())
|
|
m_dwSyncFlags |= CSC_SYNC_OUT | CSC_SYNC_IN_FULL;
|
|
else
|
|
m_dwSyncFlags |= CSC_SYNC_IN_QUICK;
|
|
break;
|
|
|
|
case SYNCMGRFLAG_INVOKE: // CscUpdateCache
|
|
if (pCookie != NULL && cbCookie > 0)
|
|
{
|
|
PCSC_UPDATE_DATA pUpdateData = (PCSC_UPDATE_DATA)pCookie;
|
|
|
|
TraceAssert(cbCookie >= sizeof(CSC_UPDATE_DATA));
|
|
|
|
DWORD dwUpdateFlags = pUpdateData->dwUpdateFlags;
|
|
|
|
if (dwUpdateFlags & CSC_UPDATE_SELECTION)
|
|
{
|
|
TraceAssert(cbCookie > sizeof(CSC_UPDATE_DATA));
|
|
|
|
// Create the filelist from the selection provided
|
|
m_pFileList = new CscFilenameList((PCSC_NAMELIST_HDR)ByteOffset(pUpdateData, pUpdateData->dwFileBufferOffset),
|
|
true);
|
|
|
|
if (!m_pFileList)
|
|
ExitGracefully(hr, E_OUTOFMEMORY, "Unable to create CscFilenameList object");
|
|
|
|
if (!m_pFileList->IsValid())
|
|
ExitGracefully(hr, E_FAIL, "Unable to initialize CscFilenameList object");
|
|
|
|
if (CSC_UPDATE_SHOWUI_ALWAYS & dwUpdateFlags)
|
|
{
|
|
m_dwSyncFlags |= CSC_SYNC_SHOWUI_ALWAYS;
|
|
}
|
|
else if (0 == m_pFileList->GetShareCount())
|
|
ExitGracefully(hr, E_UNEXPECTED, "CSC_UPDATE_SELECTION with no selection");
|
|
}
|
|
|
|
if (dwUpdateFlags & CSC_UPDATE_RECONNECT)
|
|
{
|
|
m_dwSyncFlags |= CSC_SYNC_RECONNECT;
|
|
}
|
|
|
|
if (dwUpdateFlags & CSC_UPDATE_NOTIFY_DONE)
|
|
{
|
|
|
|
// Caller of CscUpdateCache want's systray notification
|
|
// when sync is complete.
|
|
|
|
m_dwSyncFlags |= CSC_SYNC_NOTIFY_SYSTRAY;
|
|
}
|
|
|
|
if (dwUpdateFlags & CSC_UPDATE_FILL_ALL)
|
|
m_dwSyncFlags |= CSC_SYNC_IN_FULL;
|
|
else if (dwUpdateFlags & CSC_UPDATE_FILL_QUICK)
|
|
m_dwSyncFlags |= CSC_SYNC_IN_QUICK;
|
|
|
|
if (dwUpdateFlags & CSC_UPDATE_REINT)
|
|
m_dwSyncFlags |= CSC_SYNC_OUT;
|
|
|
|
if (dwUpdateFlags & CSC_UPDATE_PIN_RECURSE)
|
|
m_dwSyncFlags |= CSC_SYNC_PINFILES | CSC_SYNC_PIN_RECURSE | CSC_SYNC_IN_QUICK;
|
|
else if (dwUpdateFlags & CSC_UPDATE_PINFILES)
|
|
m_dwSyncFlags |= CSC_SYNC_PINFILES | CSC_SYNC_IN_QUICK;
|
|
|
|
if (dwUpdateFlags & CSC_UPDATE_IGNORE_ACCESS)
|
|
m_dwSyncFlags |= CSC_SYNC_IGNORE_ACCESS;
|
|
}
|
|
break;
|
|
|
|
case SYNCMGRFLAG_IDLE: // Auto-sync at idle time
|
|
if (bNoNet)
|
|
ExitGracefully(hr, E_FAIL, "No idle sync when no net");
|
|
m_dwSyncFlags = CSC_SYNC_OUT | CSC_SYNC_IN_QUICK | CSC_SYNC_IDLE | CSC_SYNC_NOTIFY_SYSTRAY;
|
|
break;
|
|
|
|
case SYNCMGRFLAG_MANUAL: // Run "mobsync.exe"
|
|
m_dwSyncFlags = CSC_SYNC_OUT | CSC_SYNC_IN_FULL | CSC_SYNC_NOTIFY_SYSTRAY | CSC_SYNC_RECONNECT;
|
|
break;
|
|
|
|
case SYNCMGRFLAG_SCHEDULED: // User scheduled sync
|
|
m_dwSyncFlags = CSC_SYNC_OUT | CSC_SYNC_IN_FULL | CSC_SYNC_NOTIFY_SYSTRAY;
|
|
break;
|
|
}
|
|
|
|
if (!(m_dwSyncFlags & CSC_SYNC_PINFILES))
|
|
m_dwSyncFlags |= CSC_SYNC_SKIP_EFS; // skip EFS if not pinning
|
|
|
|
if (dwSyncFlags & SYNCMGRFLAG_SETTINGS)
|
|
m_dwSyncFlags |= CSC_SYNC_SETTINGS;
|
|
|
|
if (!m_dwSyncFlags)
|
|
ExitGracefully(hr, E_UNEXPECTED, "Nothing to do");
|
|
|
|
if (dwSyncFlags & SYNCMGRFLAG_MAYBOTHERUSER)
|
|
m_dwSyncFlags |= CSC_SYNC_MAYBOTHERUSER;
|
|
|
|
if (bNoNet)
|
|
m_dwSyncFlags |= CSC_SYNC_NONET;
|
|
|
|
GetSilentFolderList();
|
|
|
|
exit_gracefully:
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CCscUpdate::GetHandlerInfo(LPSYNCMGRHANDLERINFO *ppSyncMgrHandlerInfo)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
LPSYNCMGRHANDLERINFO pHandlerInfo;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::GetHandlerInfo");
|
|
|
|
if (NULL == ppSyncMgrHandlerInfo)
|
|
TraceLeaveResult(E_INVALIDARG);
|
|
|
|
*ppSyncMgrHandlerInfo = NULL;
|
|
|
|
pHandlerInfo = (LPSYNCMGRHANDLERINFO)CoTaskMemAlloc(sizeof(SYNCMGRHANDLERINFO));
|
|
if (NULL == pHandlerInfo)
|
|
ExitGracefully(hr, E_OUTOFMEMORY, "LocalAlloc failed");
|
|
|
|
pHandlerInfo->cbSize = sizeof(SYNCMGRHANDLERINFO);
|
|
pHandlerInfo->hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_CSCUI_ICON));
|
|
pHandlerInfo->SyncMgrHandlerFlags = SYNCMGRHANDLER_HASPROPERTIES | SYNCMGRHANDLER_MAYESTABLISHCONNECTION;
|
|
LoadStringW(g_hInstance,
|
|
IDS_APPLICATION,
|
|
pHandlerInfo->wszHandlerName,
|
|
ARRAYSIZE(pHandlerInfo->wszHandlerName));
|
|
|
|
*ppSyncMgrHandlerInfo = pHandlerInfo;
|
|
|
|
exit_gracefully:
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CCscUpdate::EnumSyncMgrItems(LPSYNCMGRENUMITEMS *ppenum)
|
|
{
|
|
HRESULT hr;
|
|
PUPDATEENUM pNewEnum;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::EnumSyncMgrItems");
|
|
|
|
*ppenum = NULL;
|
|
|
|
pNewEnum = new CUpdateEnumerator(this);
|
|
if (pNewEnum)
|
|
{
|
|
hr = pNewEnum->QueryInterface(IID_ISyncMgrEnumItems, (LPVOID*)ppenum);
|
|
pNewEnum->Release();
|
|
}
|
|
else
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CCscUpdate::GetItemObject(REFSYNCMGRITEMID /*rItemID*/, REFIID /*riid*/, LPVOID * /*ppv*/)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CCscUpdate::ShowProperties(HWND hWndParent, REFSYNCMGRITEMID rItemID)
|
|
{
|
|
CSCEntry *pShareEntry;
|
|
LPCTSTR pszShareName = TEXT("");
|
|
|
|
pShareEntry = m_ShareLog.Get(rItemID);
|
|
|
|
// We don't enumerate shares to SyncMgr unless a share entry
|
|
// exists in the registry, so m_ShareLog.Get should never fail here.
|
|
if (pShareEntry)
|
|
pszShareName = pShareEntry->Name();
|
|
|
|
COfflineFilesFolder::Open();
|
|
|
|
// Notify SyncMgr that the ShowProperties is done.
|
|
if (NULL != m_pSyncMgrCB)
|
|
m_pSyncMgrCB->ShowPropertiesCompleted(S_OK);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CCscUpdate::SetProgressCallback(LPSYNCMGRSYNCHRONIZECALLBACK pCallback)
|
|
{
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::SetProgressCallback");
|
|
|
|
DoRelease(m_pSyncMgrCB);
|
|
|
|
m_pSyncMgrCB = pCallback;
|
|
|
|
if (m_pSyncMgrCB)
|
|
m_pSyncMgrCB->AddRef();
|
|
|
|
TraceLeaveResult(S_OK);
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CCscUpdate::PrepareForSync(ULONG cNumItems,
|
|
SYNCMGRITEMID *pItemID,
|
|
HWND /*hWndParent*/,
|
|
DWORD /*dwReserved*/)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::PrepareForSync");
|
|
TraceAssert(0 != cNumItems);
|
|
TraceAssert(NULL != pItemID);
|
|
|
|
|
|
// Copy the list of item ID's
|
|
|
|
if (NULL == m_hSyncItems)
|
|
{
|
|
m_hSyncItems = DSA_Create(sizeof(SYNCMGRITEMID), 4);
|
|
if (NULL == m_hSyncItems)
|
|
ExitGracefully(hr, E_OUTOFMEMORY, "Unable to create DSA for SYNCMGRITEMID list");
|
|
}
|
|
else
|
|
DSA_DeleteAllItems(m_hSyncItems);
|
|
|
|
while (cNumItems--)
|
|
DSA_AppendItem(m_hSyncItems, pItemID++);
|
|
|
|
exit_gracefully:
|
|
|
|
// ISyncMgrSynchronize::PrepareForSync is now an asynchronous call
|
|
// so we could create another thread to do the work and return from
|
|
// this call immediately. However, since all we do is copy the list
|
|
// of Item IDs, let's do it here and call
|
|
// m_pSyncMgrCB->PrepareForSyncCompleted before returning.
|
|
|
|
if (NULL != m_pSyncMgrCB)
|
|
m_pSyncMgrCB->PrepareForSyncCompleted(hr);
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
|
|
STDMETHODIMP
|
|
CCscUpdate::Synchronize(HWND hWndParent)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
ULONG cItems = 0;
|
|
BOOL bConnectionEstablished = FALSE;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::Synchronize");
|
|
|
|
if (NULL != m_hSyncItems)
|
|
cItems = DSA_GetItemCount(m_hSyncItems);
|
|
|
|
|
|
// Don't want systray UI updates while syncing.
|
|
// Whenever the systray UI is updated, the code checks first
|
|
// for this global mutex object. If it's non-signaled, the
|
|
// systray knows there's a sync in progress and the UI isn't
|
|
// updated.
|
|
|
|
TraceAssert(NULL == m_hSyncInProgMutex);
|
|
m_hSyncInProgMutex = CreateMutex(NULL, TRUE, c_szSyncInProgMutex);
|
|
|
|
if (0 == cItems)
|
|
{
|
|
ExitGracefully(hr, E_UNEXPECTED, "Nothing to synchronize");
|
|
}
|
|
else if (1 == cItems)
|
|
{
|
|
SYNCMGRITEMID *pItemID = (SYNCMGRITEMID*)DSA_GetItemPtr(m_hSyncItems, 0);
|
|
if (NULL != pItemID && IsEqualGUID(GUID_CscNullSyncItem, *pItemID))
|
|
{
|
|
|
|
// A single item in the DSA and it's our "null sync" GUID.
|
|
// This means we really have nothing to sync but the invoker
|
|
// of the sync wants to see some SyncMgr progress UI. In
|
|
// this scenario the update item enumerator already enumerated
|
|
// the "null sync" item. Here we set this single item's progress
|
|
// UI info to 100% complete and skip any sync activity.
|
|
|
|
SYNCMGRPROGRESSITEM spi = {0};
|
|
spi.mask = SYNCMGRPROGRESSITEM_STATUSTYPE |
|
|
SYNCMGRPROGRESSITEM_STATUSTEXT |
|
|
SYNCMGRPROGRESSITEM_PROGVALUE |
|
|
SYNCMGRPROGRESSITEM_MAXVALUE;
|
|
|
|
spi.cbSize = sizeof(spi);
|
|
spi.dwStatusType = SYNCMGRSTATUS_SUCCEEDED;
|
|
spi.lpcStatusText = L" ";
|
|
spi.iProgValue = 1;
|
|
spi.iMaxValue = 1;
|
|
m_pSyncMgrCB->Progress(GUID_CscNullSyncItem, &spi);
|
|
m_pSyncMgrCB->SynchronizeCompleted(S_OK);
|
|
|
|
ExitGracefully(hr, NOERROR, "Nothing to sync. Progress UI displayed");
|
|
}
|
|
}
|
|
|
|
m_hwndDlgParent = hWndParent;
|
|
|
|
// We can pin autocached files without a net (no sync required);
|
|
// otherwise we need to establish a RAS connection to do anything.
|
|
if ((m_dwSyncFlags & CSC_SYNC_NONET) && !(m_dwSyncFlags & CSC_SYNC_PINFILES))
|
|
{
|
|
hr = m_pSyncMgrCB->EstablishConnection(NULL, 0);
|
|
FailGracefully(hr, "Unable to establish RAS connection");
|
|
|
|
bConnectionEstablished = TRUE;
|
|
}
|
|
|
|
// For each share, kick off a thread to do the work
|
|
while (cItems > 0)
|
|
{
|
|
SYNCMGRITEMID *pItemID;
|
|
CSCEntry *pShareEntry;
|
|
|
|
--cItems;
|
|
pItemID = (SYNCMGRITEMID*)DSA_GetItemPtr(m_hSyncItems, cItems);
|
|
|
|
pShareEntry = m_ShareLog.Get(*pItemID);
|
|
|
|
// We don't enumerate shares to SyncMgr unless a share entry
|
|
// exists in the registry, so m_ShareLog.Get should never fail here.
|
|
if (NULL == pShareEntry)
|
|
ExitGracefully(hr, E_UNEXPECTED, "No share entry");
|
|
|
|
hr = SynchronizeShare(pItemID, pShareEntry->Name(), bConnectionEstablished);
|
|
DSA_DeleteItem(m_hSyncItems, cItems);
|
|
FailGracefully(hr, "Unable to create sync thread");
|
|
}
|
|
|
|
TraceAssert(0 == DSA_GetItemCount(m_hSyncItems));
|
|
|
|
exit_gracefully:
|
|
|
|
if (FAILED(hr))
|
|
SetItemStatus(GUID_NULL, SYNCMGRSTATUS_STOPPED);
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CCscUpdate::SetItemStatus(REFSYNCMGRITEMID rItemID,
|
|
DWORD dwSyncMgrStatus)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
ULONG cItems;
|
|
BOOL bAllItems;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::SetItemStatus");
|
|
|
|
if (SYNCMGRSTATUS_SKIPPED != dwSyncMgrStatus && SYNCMGRSTATUS_STOPPED != dwSyncMgrStatus)
|
|
TraceLeaveResult(E_NOTIMPL);
|
|
|
|
bAllItems = FALSE;
|
|
if (SYNCMGRSTATUS_STOPPED == dwSyncMgrStatus)
|
|
{
|
|
bAllItems = TRUE;
|
|
m_dwSyncFlags |= CSC_SYNC_CANCELLED;
|
|
}
|
|
|
|
// SetItemStatus can be called between PrepareForSync and Synchronize, in
|
|
// in which case the correct thing to do is remove the item from m_hSyncItems.
|
|
if (NULL != m_hSyncItems)
|
|
{
|
|
cItems = DSA_GetItemCount(m_hSyncItems);
|
|
|
|
while (cItems > 0)
|
|
{
|
|
SYNCMGRITEMID *pItemID;
|
|
|
|
--cItems;
|
|
pItemID = (SYNCMGRITEMID*)DSA_GetItemPtr(m_hSyncItems, cItems);
|
|
|
|
if (bAllItems || IsEqualGUID(rItemID, *pItemID))
|
|
{
|
|
// Remove the item from the list of items to sync
|
|
DSA_DeleteItem(m_hSyncItems, cItems);
|
|
if (!bAllItems)
|
|
ExitGracefully(hr, S_OK, "Skipping item");
|
|
}
|
|
}
|
|
}
|
|
|
|
// Lookup the thread for the item ID and set its status
|
|
// to cause it to terminate.
|
|
hr = SetSyncThreadStatus(SyncStop, bAllItems ? GUID_NULL : rItemID);
|
|
|
|
exit_gracefully:
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CCscUpdate::ShowError(HWND /*hWndParent*/ , REFSYNCMGRERRORID /*ErrorID*/)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
|
|
HRESULT
|
|
CCscUpdate::SynchronizeShare(SYNCMGRITEMID *pItemID, LPCTSTR pszShareName, BOOL bRasConnected)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD dwThreadID;
|
|
PSYNCTHREADDATA pThreadData;
|
|
ULONG cbShareName = 0;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::SynchronizeShare");
|
|
TraceAssert(NULL != pItemID);
|
|
TraceAssert(NULL != pszShareName);
|
|
TraceAssert(*pszShareName);
|
|
|
|
EnterCriticalSection(&m_csThreadList);
|
|
if (NULL == m_hSyncThreads)
|
|
m_hSyncThreads = DPA_Create(4);
|
|
LeaveCriticalSection(&m_csThreadList);
|
|
|
|
if (NULL == m_hSyncThreads)
|
|
ExitGracefully(hr, E_OUTOFMEMORY, "Unable to create DPA for threads");
|
|
|
|
cbShareName = StringByteSize(pszShareName);
|
|
pThreadData = (PSYNCTHREADDATA)LocalAlloc(LPTR, sizeof(SYNCTHREADDATA) + cbShareName);
|
|
|
|
if (!pThreadData)
|
|
ExitGracefully(hr, E_OUTOFMEMORY, "LocalAlloc failed");
|
|
|
|
pThreadData->pThis = this;
|
|
pThreadData->ItemID = *pItemID;
|
|
pThreadData->pszShareName = (LPTSTR)(pThreadData + 1);
|
|
CopyMemory(pThreadData->pszShareName, pszShareName, cbShareName);
|
|
|
|
|
|
// If we established a RAS connection, then it will go away
|
|
// right after the sync completes, so there's no point trying
|
|
// to reconnect. That is, only check CSC_SYNC_RECONNECT and
|
|
// add the share to the reconnect list if we aren't doing RAS.
|
|
|
|
if (bRasConnected)
|
|
{
|
|
pThreadData->dwSyncStatus |= SDS_SYNC_RAS_CONNECTED;
|
|
}
|
|
else if (m_dwSyncFlags & CSC_SYNC_RECONNECT)
|
|
{
|
|
CscFilenameList::HSHARE hShare;
|
|
m_ReconnectList.AddShare(pszShareName, &hShare);
|
|
}
|
|
|
|
pThreadData->hThread = CreateThread(NULL, 0, _SyncThread, pThreadData, CREATE_SUSPENDED, &dwThreadID);
|
|
if (NULL != pThreadData->hThread)
|
|
{
|
|
EnterCriticalSection(&m_csThreadList);
|
|
DPA_AppendPtr(m_hSyncThreads, pThreadData);
|
|
LeaveCriticalSection(&m_csThreadList);
|
|
|
|
ResumeThread(pThreadData->hThread);
|
|
}
|
|
else
|
|
{
|
|
DWORD dwErr = GetLastError();
|
|
|
|
LocalFree(pThreadData);
|
|
|
|
LPTSTR pszErr = GetErrorText(GetLastError());
|
|
LogError(*pItemID,
|
|
SYNCMGRLOGLEVEL_ERROR,
|
|
IDS_FILL_SPARSE_FILES_ERROR,
|
|
pszShareName,
|
|
pszErr);
|
|
LocalFreeString(&pszErr);
|
|
hr = HRESULT_FROM_WIN32(dwErr);
|
|
}
|
|
|
|
exit_gracefully:
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
void
|
|
CCscUpdate::SetLastSyncTime(LPCTSTR pszShareName)
|
|
{
|
|
HKEY hKey = NULL;
|
|
|
|
hKey = m_ShareLog.OpenKey(pszShareName, KEY_SET_VALUE);
|
|
if (hKey)
|
|
{
|
|
FILETIME ft = {0};
|
|
GetSystemTimeAsFileTime(&ft);
|
|
RegSetValueEx(hKey, c_szLastSync, 0, REG_BINARY, (LPBYTE)&ft, sizeof(ft));
|
|
RegCloseKey(hKey);
|
|
}
|
|
}
|
|
|
|
|
|
DWORD
|
|
CCscUpdate::GetLastSyncTime(LPCTSTR pszShareName, LPFILETIME pft)
|
|
{
|
|
DWORD dwResult = ERROR_PATH_NOT_FOUND;
|
|
HKEY hKey = NULL;
|
|
|
|
hKey = m_ShareLog.OpenKey(pszShareName, KEY_QUERY_VALUE);
|
|
if (hKey)
|
|
{
|
|
DWORD dwSize = sizeof(*pft);
|
|
dwResult = RegQueryValueEx(hKey, c_szLastSync, NULL, NULL, (LPBYTE)pft, &dwSize);
|
|
RegCloseKey(hKey);
|
|
}
|
|
return dwResult;
|
|
}
|
|
|
|
|
|
void
|
|
CCscUpdate::SyncThreadCompleted(PSYNCTHREADDATA pSyncData)
|
|
{
|
|
int iThread;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::SyncThreadCompleted");
|
|
TraceAssert(NULL != pSyncData);
|
|
TraceAssert(NULL != m_hSyncThreads);
|
|
|
|
EnterCriticalSection(&m_csThreadList);
|
|
|
|
iThread = DPA_GetPtrIndex(m_hSyncThreads, pSyncData);
|
|
TraceAssert(-1 != iThread);
|
|
|
|
DPA_DeletePtr(m_hSyncThreads, iThread);
|
|
CloseHandle(pSyncData->hThread);
|
|
pSyncData->hThread = NULL;
|
|
|
|
iThread = DPA_GetPtrCount(m_hSyncThreads);
|
|
|
|
LeaveCriticalSection(&m_csThreadList);
|
|
|
|
if (0 == iThread)
|
|
{
|
|
SyncCompleted();
|
|
}
|
|
|
|
TraceLeaveVoid();
|
|
}
|
|
|
|
|
|
void
|
|
CCscUpdate::SyncCompleted(void)
|
|
{
|
|
if ((m_dwSyncFlags & CSC_SYNC_RECONNECT) &&
|
|
!(m_dwSyncFlags & CSC_SYNC_CANCELLED))
|
|
{
|
|
m_dwSyncFlags &= ~CSC_SYNC_RECONNECT;
|
|
ReconnectServers(&m_ReconnectList, FALSE, FALSE);
|
|
}
|
|
|
|
if (NULL != m_hSyncInProgMutex)
|
|
{
|
|
// We're not syncing so reset the global event
|
|
ReleaseMutex(m_hSyncInProgMutex);
|
|
CloseHandle(m_hSyncInProgMutex);
|
|
m_hSyncInProgMutex = NULL;
|
|
}
|
|
|
|
if (m_dwSyncFlags & CSC_SYNC_NOTIFY_SYSTRAY)
|
|
{
|
|
// Notify systray that we're done
|
|
PostToSystray(CSCWM_DONESYNCING, 0, 0);
|
|
m_dwSyncFlags &= ~CSC_SYNC_NOTIFY_SYSTRAY;
|
|
}
|
|
|
|
// Notify SyncMgr that the sync is done
|
|
if (NULL != m_pSyncMgrCB)
|
|
{
|
|
m_pSyncMgrCB->SynchronizeCompleted(S_OK);
|
|
}
|
|
}
|
|
|
|
|
|
UINT
|
|
GetErrorFormat(DWORD dwErr, BOOL bMerging = FALSE)
|
|
{
|
|
UINT idString = 0;
|
|
|
|
// BUGBUG these are all just initial guesses. Not sure
|
|
// which error codes we'll get from CSC.
|
|
|
|
switch (dwErr)
|
|
{
|
|
case ERROR_DISK_FULL:
|
|
// "The server disk is full."
|
|
// "The local disk is full."
|
|
idString = bMerging ? IDS_SERVER_FULL_ERROR : IDS_LOCAL_DISK_FULL_ERROR;
|
|
break;
|
|
|
|
case ERROR_LOCK_VIOLATION:
|
|
case ERROR_SHARING_VIOLATION:
|
|
case ERROR_OPEN_FILES:
|
|
case ERROR_ACTIVE_CONNECTIONS:
|
|
case ERROR_DEVICE_IN_USE:
|
|
// "'%1' is in use on %2"
|
|
idString = IDS_FILE_OPEN_ERROR;
|
|
break;
|
|
|
|
case ERROR_BAD_NETPATH:
|
|
case ERROR_DEV_NOT_EXIST:
|
|
case ERROR_NETNAME_DELETED:
|
|
case ERROR_BAD_NET_NAME:
|
|
case ERROR_SHARING_PAUSED:
|
|
case ERROR_REQ_NOT_ACCEP:
|
|
case ERROR_REDIR_PAUSED:
|
|
case ERROR_BAD_DEVICE:
|
|
case ERROR_CONNECTION_UNAVAIL:
|
|
case ERROR_NO_NET_OR_BAD_PATH:
|
|
case ERROR_NO_NETWORK:
|
|
case ERROR_CONNECTION_REFUSED:
|
|
case ERROR_GRACEFUL_DISCONNECT:
|
|
case ERROR_NETWORK_UNREACHABLE:
|
|
case ERROR_HOST_UNREACHABLE:
|
|
case ERROR_PROTOCOL_UNREACHABLE:
|
|
case ERROR_PORT_UNREACHABLE:
|
|
case ERROR_LOGON_FAILURE:
|
|
// "Unable to connect to '%1.' %2"
|
|
idString = IDS_SHARE_CONNECT_ERROR;
|
|
break;
|
|
|
|
case ERROR_OPEN_FAILED:
|
|
case ERROR_UNEXP_NET_ERR:
|
|
case ERROR_NETWORK_BUSY:
|
|
case ERROR_BAD_NET_RESP:
|
|
// "Unable to access '%1' on %2. %3"
|
|
idString = IDS_NET_ERROR;
|
|
break;
|
|
|
|
case ERROR_ACCESS_DENIED:
|
|
case ERROR_NETWORK_ACCESS_DENIED:
|
|
// "Access to '%1' is denied on %2"
|
|
idString = IDS_ACCESS_ERROR;
|
|
break;
|
|
|
|
case ERROR_BAD_FORMAT:
|
|
// "The Offline Files cache is corrupt. Restart the computer to correct the cache."
|
|
idString = IDS_CACHE_CORRUPT;
|
|
break;
|
|
|
|
default:
|
|
// "Error accessing '%1' on %2. %3"
|
|
idString = IDS_UNKNOWN_SYNC_ERROR;
|
|
break;
|
|
}
|
|
|
|
return idString;
|
|
}
|
|
|
|
|
|
HRESULT
|
|
CCscUpdate::LogError(REFSYNCMGRITEMID rItemID,
|
|
LPCTSTR pszText,
|
|
DWORD dwLogLevel,
|
|
REFSYNCMGRERRORID ErrorID)
|
|
{
|
|
HRESULT hr;
|
|
SYNCMGRLOGERRORINFO slei;
|
|
|
|
USES_CONVERSION;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::LogError");
|
|
|
|
if (NULL == m_pSyncMgrCB)
|
|
TraceLeaveResult(E_UNEXPECTED);
|
|
|
|
slei.cbSize = sizeof(slei);
|
|
slei.mask = SYNCMGRLOGERROR_ITEMID | SYNCMGRLOGERROR_ERRORID;
|
|
slei.ItemID = rItemID;
|
|
slei.ErrorID = ErrorID;
|
|
|
|
// if we have a jumptext associated with this item then
|
|
// set the enable jumptext flag
|
|
if (ErrorID != GUID_NULL)
|
|
{
|
|
slei.mask |= SYNCMGRLOGERROR_ERRORFLAGS;
|
|
slei.dwSyncMgrErrorFlags = SYNCMGRERRORFLAG_ENABLEJUMPTEXT;
|
|
}
|
|
|
|
Trace((pszText));
|
|
hr = m_pSyncMgrCB->LogError(dwLogLevel, T2CW(pszText), &slei);
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
DWORD
|
|
CCscUpdate::LogError(REFSYNCMGRITEMID rItemID,
|
|
DWORD dwLogLevel,
|
|
UINT nFormatID,
|
|
...)
|
|
{
|
|
LPTSTR pszError = NULL;
|
|
va_list args;
|
|
va_start(args, nFormatID);
|
|
if (vFormatStringID(&pszError, g_hInstance, nFormatID, &args))
|
|
{
|
|
LogError(rItemID, pszError, dwLogLevel);
|
|
LocalFree(pszError);
|
|
}
|
|
va_end(args);
|
|
return 0;
|
|
}
|
|
|
|
|
|
DWORD
|
|
CCscUpdate::LogError(REFSYNCMGRITEMID rItemID,
|
|
UINT nFormatID,
|
|
LPCTSTR pszName,
|
|
DWORD dwErr,
|
|
DWORD dwLogLevel)
|
|
{
|
|
|
|
// Break the filename into "file" and "path" components
|
|
|
|
TCHAR szPath[MAX_PATH] = TEXT("\\");
|
|
LPCTSTR pszFile = NULL;
|
|
if (pszName)
|
|
{
|
|
pszFile = PathFindFileName(pszName);
|
|
lstrcpyn(szPath, pszName, min(ARRAYSIZE(szPath),(int)(pszFile-pszName)));
|
|
}
|
|
|
|
|
|
// Get the system error text and format the error
|
|
|
|
LPTSTR pszErr = GetErrorText(dwErr);
|
|
LogError(rItemID,
|
|
dwLogLevel,
|
|
nFormatID,
|
|
pszFile,
|
|
szPath,
|
|
pszErr);
|
|
LocalFreeString(&pszErr);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
BOOL
|
|
MakeDriveLetterPath(LPCTSTR pszUNC,
|
|
LPCTSTR pszShare,
|
|
LPCTSTR pszDrive,
|
|
LPTSTR *ppszResult)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
ULONG cchShare;
|
|
|
|
if (!pszUNC || !pszShare || !ppszResult)
|
|
return FALSE;
|
|
|
|
*ppszResult = NULL;
|
|
|
|
cchShare = lstrlen(pszShare);
|
|
|
|
// If the path is on the share, use the drive letter instead
|
|
if (pszDrive && *pszDrive &&
|
|
CSTR_EQUAL == CompareString(LOCALE_USER_DEFAULT,
|
|
NORM_IGNORECASE,
|
|
pszUNC,
|
|
cchShare,
|
|
pszShare,
|
|
cchShare))
|
|
{
|
|
*ppszResult = (LPTSTR)LocalAlloc(LPTR, MAX(StringByteSize(pszUNC), MAX_PATH_BYTES));
|
|
if (*ppszResult)
|
|
{
|
|
PathCombine(*ppszResult, pszDrive, pszUNC + cchShare);
|
|
bResult = TRUE;
|
|
}
|
|
}
|
|
return bResult;
|
|
}
|
|
|
|
|
|
DWORD
|
|
CCscUpdate::CopyLocalFileWithDriveMapping(LPCTSTR pszSrc,
|
|
LPCTSTR pszDst,
|
|
LPCTSTR pszShare,
|
|
LPCTSTR pszDrive,
|
|
BOOL bDirectory)
|
|
{
|
|
DWORD dwErr = NOERROR;
|
|
LPTSTR szDst = NULL;
|
|
|
|
if (!pszSrc || !pszDst || !pszShare)
|
|
return ERROR_INVALID_PARAMETER;
|
|
|
|
// If the destination is on the share, use the drive letter instead
|
|
if (MakeDriveLetterPath(pszDst, pszShare, pszDrive, &szDst))
|
|
pszDst = szDst;
|
|
|
|
if (bDirectory)
|
|
{
|
|
// We don't need to copy the directory contents here, just create
|
|
// the tree structure on the server.
|
|
if (!CreateDirectory(pszDst, NULL))
|
|
{
|
|
dwErr = GetLastError();
|
|
if (ERROR_ALREADY_EXISTS == dwErr)
|
|
dwErr = NOERROR;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
LPTSTR pszTmpName = NULL;
|
|
|
|
if (!CSCCopyReplica(pszSrc, &pszTmpName) ||
|
|
!MoveFileEx(pszTmpName,
|
|
pszDst,
|
|
MOVEFILE_COPY_ALLOWED | MOVEFILE_REPLACE_EXISTING | MOVEFILE_WRITE_THROUGH))
|
|
{
|
|
dwErr = GetLastError();
|
|
}
|
|
|
|
if (NULL != pszTmpName)
|
|
{
|
|
DeleteFile(pszTmpName);
|
|
LocalFree(pszTmpName);
|
|
}
|
|
}
|
|
|
|
if (ERROR_PATH_NOT_FOUND == dwErr)
|
|
{
|
|
// The parent directory doesn't exist, create it now.
|
|
TCHAR szParent[MAX_PATH];
|
|
lstrcpyn(szParent, pszDst, ARRAYSIZE(szParent));
|
|
PathRemoveFileSpec(szParent);
|
|
dwErr = CopyLocalFileWithDriveMapping(pszSrc, szParent, pszShare, NULL, TRUE);
|
|
|
|
// If that worked, retry the original operation.
|
|
if (NOERROR == dwErr)
|
|
dwErr = CopyLocalFileWithDriveMapping(pszSrc, pszDst, pszShare, NULL, bDirectory);
|
|
}
|
|
|
|
LocalFreeString(&szDst);
|
|
|
|
return dwErr;
|
|
}
|
|
|
|
|
|
BOOL
|
|
HandleConflictLocally(PSYNCTHREADDATA pSyncData,
|
|
LPCTSTR pszPath,
|
|
DWORD dwCscStatus,
|
|
DWORD dwLocalAttr,
|
|
DWORD dwRemoteAttr = 0)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
LPTSTR szParent = NULL;
|
|
|
|
// If it's super-hidden or not modified locally, we can always
|
|
// handle the conflict locally.
|
|
if (!(dwCscStatus & CSC_LOCALLY_MODIFIED) || IsHiddenSystem(dwLocalAttr) || IsHiddenSystem(dwRemoteAttr))
|
|
return TRUE;
|
|
|
|
// If we're dealing with 2 folders, the worst that happens is that the
|
|
// underlying files/folders get merged.
|
|
if ((FILE_ATTRIBUTE_DIRECTORY & dwLocalAttr) && (FILE_ATTRIBUTE_DIRECTORY & dwRemoteAttr))
|
|
return TRUE;
|
|
|
|
|
|
// Next, check whether the parent path is super-hidden.
|
|
|
|
// For example, recycle bin makes super-hidden folders and puts
|
|
// metadata files in them.
|
|
|
|
// Do this on the server, since CSC has exclusive access to the database
|
|
// while merging, causing GetFileAttributes to fail with Access Denied.
|
|
|
|
if (MakeDriveLetterPath(pszPath, pSyncData->pszShareName, pSyncData->szDrive, &szParent))
|
|
{
|
|
do
|
|
{
|
|
PathRemoveFileSpec(szParent);
|
|
dwRemoteAttr = GetFileAttributes(szParent);
|
|
|
|
if ((DWORD)-1 == dwRemoteAttr)
|
|
{
|
|
// Path doesn't exist, access denied, etc.
|
|
break;
|
|
}
|
|
|
|
if (IsHiddenSystem(dwRemoteAttr))
|
|
{
|
|
bResult = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
while (!PathIsRoot(szParent));
|
|
}
|
|
|
|
LocalFreeString(&szParent);
|
|
|
|
return bResult;
|
|
}
|
|
|
|
DWORD
|
|
CCscUpdate::HandleFileConflict(PSYNCTHREADDATA pSyncData,
|
|
LPCTSTR pszName,
|
|
DWORD dwStatus,
|
|
DWORD dwHintFlags,
|
|
LPWIN32_FIND_DATA pFind32)
|
|
{
|
|
DWORD dwResult = CSCPROC_RETURN_CONTINUE;
|
|
DWORD dwErr = NOERROR;
|
|
int nErrorResolution = RFC_KEEPBOTH;
|
|
LPTSTR pszNewName = NULL;
|
|
LPTSTR szFullPath = NULL;
|
|
BOOL bApplyToAll = FALSE;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::HandleFileConflict");
|
|
Trace((TEXT("File conflict: %s"), pszName));
|
|
TraceAssert(pSyncData->dwSyncStatus & SDS_SYNC_OUT);
|
|
|
|
szFullPath = (LPTSTR)LocalAlloc(LPTR, StringByteSize(pszName) + MAX_PATH*sizeof(TCHAR));
|
|
if (!szFullPath)
|
|
{
|
|
dwErr = ERROR_OUTOFMEMORY;
|
|
ExitGracefully(dwResult, CSCPROC_RETURN_SKIP, "LocalAlloc failed");
|
|
}
|
|
|
|
HANDLE hFind;
|
|
WIN32_FIND_DATA fdRemote;
|
|
|
|
PathCombine(szFullPath, pSyncData->szDrive, pszName + lstrlen(pSyncData->pszShareName));
|
|
hFind = FindFirstFile(szFullPath, &fdRemote);
|
|
|
|
// Does the net version still exist?
|
|
if (hFind == INVALID_HANDLE_VALUE)
|
|
ExitGracefully(dwResult, HandleDeleteConflict(pSyncData, pszName, dwStatus, dwHintFlags, pFind32), "Net file deleted");
|
|
|
|
// Still exists, continue
|
|
FindClose(hFind);
|
|
|
|
// If only the attributes or file times were modified locally,
|
|
// or if the file is hidden+system, keep the server copy and
|
|
// don't bother the user. (e.g. desktop.ini)
|
|
if (HandleConflictLocally(pSyncData, pszName, dwStatus, pFind32->dwFileAttributes, fdRemote.dwFileAttributes))
|
|
{
|
|
ExitGracefully(dwResult, CSCPROC_RETURN_FORCE_INWARD, "Ignoring conflict");
|
|
}
|
|
else if (IsSilentFolder(pszName))
|
|
{
|
|
// It's in a per-user shell special folder. Last writer wins.
|
|
if (CompareFileTime(&pFind32->ftLastWriteTime, &fdRemote.ftLastWriteTime) < 0)
|
|
{
|
|
ExitGracefully(dwResult, CSCPROC_RETURN_FORCE_INWARD, "Handling special folder conflict - server copy wins");
|
|
}
|
|
else
|
|
{
|
|
ExitGracefully(dwResult, CSCPROC_RETURN_FORCE_OUTWARD, "Handling special folder conflict - local copy wins");
|
|
}
|
|
}
|
|
|
|
dwErr = GetNewVersionName(pszName,
|
|
pSyncData->pszShareName,
|
|
pSyncData->szDrive,
|
|
&pszNewName);
|
|
if (NOERROR != dwErr)
|
|
{
|
|
ExitGracefully(dwResult, CSCPROC_RETURN_SKIP, "GetNewVersionName failed");
|
|
}
|
|
|
|
switch (SDS_SYNC_FILE_CONFLICT_MASK & pSyncData->dwSyncStatus)
|
|
{
|
|
case 0:
|
|
if (CSC_SYNC_MAYBOTHERUSER & m_dwSyncFlags)
|
|
{
|
|
nErrorResolution = ShowConflictDialog(m_hwndDlgParent,
|
|
pszName,
|
|
pszNewName,
|
|
pSyncData->pszShareName,
|
|
pSyncData->szDrive,
|
|
pFind32,
|
|
&fdRemote);
|
|
if (RFC_APPLY_TO_ALL & nErrorResolution)
|
|
{
|
|
bApplyToAll = TRUE;
|
|
nErrorResolution &= ~RFC_APPLY_TO_ALL;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case SDS_SYNC_CONFLICT_KEEPLOCAL:
|
|
nErrorResolution = RFC_KEEPLOCAL;
|
|
break;
|
|
|
|
case SDS_SYNC_CONFLICT_KEEPNET:
|
|
nErrorResolution = RFC_KEEPNETWORK;
|
|
break;
|
|
|
|
case SDS_SYNC_CONFLICT_KEEPBOTH:
|
|
nErrorResolution = RFC_KEEPBOTH;
|
|
break;
|
|
}
|
|
|
|
// Self-host notification callback
|
|
CSCUI_NOTIFYHOOK((CSCH_UpdateConflict, TEXT("Update conflict: %1, resolution %2!d!"), pszName, nErrorResolution));
|
|
|
|
switch (nErrorResolution)
|
|
{
|
|
default:
|
|
case RFC_KEEPBOTH:
|
|
if (bApplyToAll)
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_CONFLICT_KEEPBOTH;
|
|
lstrcpy(szFullPath, pszName);
|
|
PathRemoveFileSpec(szFullPath);
|
|
if (FILE_ATTRIBUTE_DIRECTORY & pFind32->dwFileAttributes)
|
|
{
|
|
// Rename the local version in the cache and merge again.
|
|
lstrcpyn(pFind32->cFileName, pszNewName, ARRAYSIZE(pFind32->cFileName));
|
|
if (!CSCDoLocalRenameEx(pszName, szFullPath, pFind32, TRUE, TRUE))
|
|
{
|
|
dwErr = GetLastError();
|
|
ExitGracefully(dwResult, CSCPROC_RETURN_SKIP, "CSCDoLocalRenameEx failed");
|
|
}
|
|
// Because CSCDoLocalRenameEx and CSCMergeShare are separate operations,
|
|
// we have to abort the current merge operation and start over.
|
|
// Otherwise, the current merge operation fails due to the "left
|
|
// hand not knowing what the right hande is doing".
|
|
Trace((TEXT("Restarting merge on: %s"), pSyncData->pszShareName));
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_RESTART_MERGE;
|
|
dwResult = CSCPROC_RETURN_ABORT;
|
|
}
|
|
else
|
|
{
|
|
// Note that CSCDoLocalRenameEx would work for files also, but we
|
|
// prefer to avoid restarting CSCMergeShare so do these ourselves.
|
|
PathAppend(szFullPath, pszNewName);
|
|
dwErr = CopyLocalFileWithDriveMapping(pszName,
|
|
szFullPath,
|
|
pSyncData->pszShareName,
|
|
pSyncData->szDrive);
|
|
if (NOERROR != dwErr)
|
|
ExitGracefully(dwResult, CSCPROC_RETURN_SKIP, "CopyLocalFileWithDriveMapping failed");
|
|
|
|
// If the original file was pinned, we want to pin the copy also.
|
|
// Unfortunately, we can't reliably pin during a merge, so we have
|
|
// to remember these in a list and pin them later.
|
|
if (dwHintFlags & FLAG_CSC_HINT_PIN_USER)
|
|
{
|
|
if (!m_pConflictPinList)
|
|
m_pConflictPinList = new CscFilenameList;
|
|
if (m_pConflictPinList)
|
|
{
|
|
m_pConflictPinList->AddFile(szFullPath,
|
|
!!(pFind32->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY));
|
|
}
|
|
}
|
|
|
|
// Tell CSCMergeShare to copy the server copy to the cache
|
|
// (with the old name). This clears the dirty cache.
|
|
dwResult = CSCPROC_RETURN_FORCE_INWARD;
|
|
}
|
|
break;
|
|
|
|
case RFC_KEEPNETWORK:
|
|
// Tell CSCMergeShare to copy the server copy to the cache
|
|
dwResult = CSCPROC_RETURN_FORCE_INWARD;
|
|
if (bApplyToAll)
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_CONFLICT_KEEPNET;
|
|
break;
|
|
|
|
case RFC_KEEPLOCAL:
|
|
// Tell CSCMergeShare to push the local copy to the server
|
|
dwResult = CSCPROC_RETURN_FORCE_OUTWARD;
|
|
if (bApplyToAll)
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_CONFLICT_KEEPLOCAL;
|
|
break;
|
|
|
|
case RFC_CANCEL:
|
|
TraceMsg("HandleFileConflict: Cancelling sync - user bailed");
|
|
SetItemStatus(GUID_NULL, SYNCMGRSTATUS_STOPPED);
|
|
dwResult = CSCPROC_RETURN_ABORT;
|
|
break;
|
|
}
|
|
|
|
exit_gracefully:
|
|
|
|
if (CSCPROC_RETURN_FORCE_INWARD == dwResult)
|
|
{
|
|
// CSCMergeShare truncates (makes sparse) the
|
|
// file if we return this. We'd like to fill
|
|
// it during this sync.
|
|
pSyncData->cFilesToSync++;
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_FORCE_INWARD;
|
|
}
|
|
|
|
if (NOERROR != dwErr)
|
|
{
|
|
pszName += lstrlen(pSyncData->pszShareName);
|
|
if (*pszName == TEXT('\\'))
|
|
pszName++;
|
|
LogError(pSyncData->ItemID,
|
|
IDS_NAME_CONFLICT_ERROR,
|
|
pszName,
|
|
dwErr);
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_ERROR;
|
|
}
|
|
|
|
LocalFreeString(&szFullPath);
|
|
LocalFreeString(&pszNewName);
|
|
|
|
TraceLeaveResult(dwResult);
|
|
}
|
|
|
|
|
|
// Returns values for the Resolve Delete Conflict dialog
|
|
#define RDC_CANCEL 0x00
|
|
#define RDC_DELETE 0x01
|
|
#define RDC_RESTORE 0x02
|
|
#define RDC_APPLY_ALL 0x04
|
|
#define RDC_DELETE_ALL (RDC_APPLY_ALL | RDC_DELETE)
|
|
#define RDC_RESTORE_ALL (RDC_APPLY_ALL | RDC_RESTORE)
|
|
|
|
TCHAR const c_szDeleteSelection[] = TEXT("DeleteConflictSelection");
|
|
|
|
INT_PTR CALLBACK
|
|
DeleteConflictProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
int nResult;
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
TCHAR szShare[MAX_PATH];
|
|
LPCTSTR pszPath = (LPCTSTR)lParam;
|
|
LPTSTR pszT = NULL;
|
|
|
|
szShare[0] = TEXT('\0');
|
|
lstrcpyn(szShare, pszPath, ARRAYSIZE(szShare));
|
|
PathStripToRoot(szShare);
|
|
|
|
// Format the file name
|
|
PathSetDlgItemPath(hDlg, IDC_FILENAME, pszPath);
|
|
|
|
// Build the "Do this for all on <this share>" string
|
|
FormatStringID(&pszT, g_hInstance, IDS_FMT_DELETE_APPLY_ALL, szShare);
|
|
if (pszT)
|
|
{
|
|
SetDlgItemText(hDlg, IDC_APPLY_TO_ALL, pszT);
|
|
LocalFreeString(&pszT);
|
|
}
|
|
// else default text is OK (no share name)
|
|
|
|
// Select whatever the user chose last time, default is "restore"
|
|
DWORD dwPrevSelection = RDC_RESTORE;
|
|
DWORD dwType;
|
|
DWORD cbData = sizeof(dwPrevSelection);
|
|
SHGetValue(HKEY_CURRENT_USER,
|
|
c_szCSCKey,
|
|
c_szDeleteSelection,
|
|
&dwType,
|
|
&dwPrevSelection,
|
|
&cbData);
|
|
dwPrevSelection = (RDC_DELETE == dwPrevSelection ? IDC_DELETE_LOCAL : IDC_KEEP_LOCAL);
|
|
CheckRadioButton(hDlg, IDC_KEEP_LOCAL, IDC_DELETE_LOCAL, dwPrevSelection);
|
|
|
|
// Get the file-type icon
|
|
pszT = PathFindExtension(pszPath);
|
|
if (pszT)
|
|
{
|
|
SHFILEINFO sfi = {0};
|
|
SHGetFileInfo(pszT, 0, &sfi, sizeof(sfi), SHGFI_ICON);
|
|
if (sfi.hIcon)
|
|
{
|
|
SendDlgItemMessage(hDlg,
|
|
IDC_DLGTYPEICON,
|
|
STM_SETICON,
|
|
(WPARAM)sfi.hIcon,
|
|
0L);
|
|
}
|
|
}
|
|
}
|
|
return TRUE;
|
|
|
|
case WM_COMMAND:
|
|
nResult = -1;
|
|
switch (LOWORD(wParam))
|
|
{
|
|
case IDCANCEL:
|
|
nResult = RDC_CANCEL;
|
|
break;
|
|
|
|
case IDOK:
|
|
if (BST_CHECKED == IsDlgButtonChecked(hDlg, IDC_DELETE_LOCAL))
|
|
nResult = RDC_DELETE;
|
|
else
|
|
nResult = RDC_RESTORE;
|
|
// Remember the selection for next time
|
|
SHSetValue(HKEY_CURRENT_USER,
|
|
c_szCSCKey,
|
|
c_szDeleteSelection,
|
|
REG_DWORD,
|
|
&nResult,
|
|
sizeof(nResult));
|
|
if (BST_CHECKED == IsDlgButtonChecked(hDlg, IDC_APPLY_TO_ALL))
|
|
nResult |= RDC_APPLY_ALL;
|
|
break;
|
|
}
|
|
if (-1 != nResult)
|
|
{
|
|
EndDialog(hDlg, nResult);
|
|
return TRUE;
|
|
}
|
|
break;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
BOOL CALLBACK
|
|
ConflictPurgeCallback(LPCWSTR /*pszFile*/, LPARAM lParam)
|
|
{
|
|
PSYNCTHREADDATA pSyncData = (PSYNCTHREADDATA)lParam;
|
|
return !(SDS_SYNC_CANCELLED & pSyncData->dwSyncStatus);
|
|
}
|
|
|
|
|
|
DWORD
|
|
CCscUpdate::HandleDeleteConflict(PSYNCTHREADDATA pSyncData,
|
|
LPCTSTR pszName,
|
|
DWORD dwStatus,
|
|
DWORD dwHintFlags,
|
|
LPWIN32_FIND_DATA pFind32)
|
|
{
|
|
DWORD dwResult = CSCPROC_RETURN_CONTINUE;
|
|
int nErrorResolution = RDC_DELETE; // default action
|
|
BOOL bDirectory = (FILE_ATTRIBUTE_DIRECTORY & pFind32->dwFileAttributes);
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::HandleDeleteConflict");
|
|
Trace((TEXT("Net file deleted: %s"), pszName));
|
|
|
|
|
|
// We already know that the net file was deleted, or HandleDeleteConflict
|
|
// wouldn't be called. If the local copy was also deleted, then there
|
|
// isn't really a conflict and we can continue without prompting.
|
|
|
|
// If the file isn't pinned, handle the conflict silently if
|
|
// only attributes changed or it's super-hidden.
|
|
|
|
// Finally, if the file lives in certain special folder locations,
|
|
// such as AppData, handle the conflict silently.
|
|
|
|
// If we get past all that, ask the user what to do, but only bother
|
|
// the user as a last resort.
|
|
|
|
if ( !(dwStatus & FLAG_CSC_COPY_STATUS_LOCALLY_DELETED)
|
|
&& ((dwHintFlags & (FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_ADMIN)) ||
|
|
!HandleConflictLocally(pSyncData, pszName, dwStatus, pFind32->dwFileAttributes))
|
|
&& !IsSilentFolder(pszName)
|
|
)
|
|
{
|
|
// The file is either pinned or modified locally, so
|
|
// default action is now "restore".
|
|
nErrorResolution = RDC_RESTORE;
|
|
|
|
switch (SDS_SYNC_DELETE_CONFLICT_MASK & pSyncData->dwSyncStatus)
|
|
{
|
|
case 0:
|
|
if (CSC_SYNC_MAYBOTHERUSER & m_dwSyncFlags)
|
|
{
|
|
int idDialog = (bDirectory ? IDD_FOLDER_CONFLICT_DELETE : IDD_FILE_CONFLICT_DELETE);
|
|
nErrorResolution = (int)DialogBoxParam(g_hInstance,
|
|
MAKEINTRESOURCE(idDialog),
|
|
m_hwndDlgParent,
|
|
DeleteConflictProc,
|
|
(LPARAM)pszName);
|
|
if (RDC_DELETE_ALL == nErrorResolution)
|
|
{
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_DELETE_DELETE;
|
|
nErrorResolution = RDC_DELETE;
|
|
}
|
|
else if (RDC_RESTORE_ALL == nErrorResolution)
|
|
{
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_DELETE_RESTORE;
|
|
nErrorResolution = RDC_RESTORE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case SDS_SYNC_DELETE_DELETE:
|
|
nErrorResolution = RDC_DELETE;
|
|
break;
|
|
|
|
case SDS_SYNC_DELETE_RESTORE:
|
|
nErrorResolution = RDC_RESTORE;
|
|
break;
|
|
}
|
|
|
|
// Self-host notification callback
|
|
CSCUI_NOTIFYHOOK((CSCH_DeleteConflict, TEXT("Delete conflict: %1, resolution %2!d!"), pszName, nErrorResolution));
|
|
}
|
|
|
|
switch (nErrorResolution)
|
|
{
|
|
default:
|
|
case RDC_RESTORE:
|
|
Trace((TEXT("HandleDeleteConflict: restoring %s"), pszName));
|
|
// Tell CSCMergeShare to push the local copy to the server
|
|
dwResult = CSCPROC_RETURN_FORCE_OUTWARD;
|
|
break;
|
|
|
|
case RDC_DELETE:
|
|
Trace((TEXT("HandleDeleteConflict: deleting %s"), pszName));
|
|
if (bDirectory)
|
|
{
|
|
// Deep delete
|
|
CSCUIRemoveFolderFromCache(pszName, 0, ConflictPurgeCallback, (LPARAM)pSyncData);
|
|
}
|
|
else
|
|
{
|
|
CscDelete(pszName);
|
|
}
|
|
dwResult = CSCPROC_RETURN_SKIP;
|
|
break;
|
|
|
|
case RDC_CANCEL:
|
|
TraceMsg("HandleDeleteConflict: Cancelling sync - user bailed");
|
|
SetItemStatus(GUID_NULL, SYNCMGRSTATUS_STOPPED);
|
|
dwResult = CSCPROC_RETURN_ABORT;
|
|
break;
|
|
}
|
|
|
|
TraceLeaveResult(dwResult);
|
|
}
|
|
|
|
DWORD
|
|
CCscUpdate::CscCallback(PSYNCTHREADDATA pSyncData,
|
|
LPCTSTR pszName,
|
|
DWORD dwStatus,
|
|
DWORD dwHintFlags,
|
|
DWORD dwPinCount,
|
|
LPWIN32_FIND_DATA pFind32,
|
|
DWORD dwReason,
|
|
DWORD dwParam1,
|
|
DWORD dwParam2)
|
|
{
|
|
DWORD dwResult = CSCPROC_RETURN_CONTINUE;
|
|
SYNCMGRPROGRESSITEM spi = { sizeof(spi), 0 };
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::CscCallback");
|
|
TraceAssert(pSyncData != NULL);
|
|
TraceAssert(pSyncData->pThis == this);
|
|
|
|
// Check for Cancel
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
{
|
|
TraceMsg("Cancelling sync operation");
|
|
TraceLeaveValue(CSCPROC_RETURN_ABORT);
|
|
}
|
|
|
|
switch (dwReason)
|
|
{
|
|
case CSCPROC_REASON_BEGIN:
|
|
// First thing to do is determine if this is for the entire share
|
|
// or an individual file in the share.
|
|
if (!(pSyncData->dwSyncStatus & SDS_SYNC_STARTED))
|
|
{
|
|
// SHARE BEGIN
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_STARTED;
|
|
|
|
TraceAssert(!lstrcmpi(pszName, pSyncData->pszShareName));
|
|
Trace((TEXT("Share begin: %s"), pszName));
|
|
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_OUT)
|
|
{
|
|
// Save the drive letter to use for net operations
|
|
Trace((TEXT("Drive %s"), pFind32->cFileName));
|
|
lstrcpyn(pSyncData->szDrive, pFind32->cFileName, ARRAYSIZE(pSyncData->szDrive));
|
|
}
|
|
else
|
|
{
|
|
pSyncData->szDrive[0] = TEXT('\0');
|
|
}
|
|
|
|
// Remember whether it's an autocache share or not
|
|
switch (dwStatus & FLAG_CSC_SHARE_STATUS_CACHING_MASK)
|
|
{
|
|
case FLAG_CSC_SHARE_STATUS_AUTO_REINT:
|
|
case FLAG_CSC_SHARE_STATUS_VDO:
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_AUTOCACHE;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// FILE BEGIN
|
|
BOOL bSkipFile = FALSE;
|
|
|
|
TraceAssert(lstrlen(pszName) > lstrlen(pSyncData->pszShareName));
|
|
|
|
if (!(pFind32->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY))
|
|
{
|
|
// If we're updating a file selection and this file
|
|
// isn't part of the selection, skip it.
|
|
if (m_pFileList && !m_pFileList->FileExists(pszName, false))
|
|
{
|
|
bSkipFile = TRUE;
|
|
}
|
|
else if (!(pSyncData->dwSyncStatus & (SDS_SYNC_AUTOCACHE | SDS_SYNC_OUT)) &&
|
|
!(dwHintFlags & (FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_ADMIN)) &&
|
|
!IsSpecialFolder(pszName))
|
|
{
|
|
// Skip autocached files when filling on a
|
|
// non-autocache share. Raid #341786
|
|
bSkipFile = TRUE;
|
|
}
|
|
else if (!(pSyncData->dwSyncStatus & CSC_SYNC_IGNORE_ACCESS))
|
|
{
|
|
// dwReserved0 is the current user's access mask
|
|
// dwReserved1 is the Guest access mask
|
|
DWORD dwCurrentAccess = pFind32->dwReserved0 | pFind32->dwReserved1;
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_OUT)
|
|
{
|
|
|
|
// If the current user doesn't have sufficient access
|
|
// to merge offline changes, then don't bother trying.
|
|
// (It must be some other user's file.)
|
|
|
|
|
|
// Have the attributes changed offline?
|
|
if (FLAG_CSC_COPY_STATUS_ATTRIB_LOCALLY_MODIFIED & dwStatus)
|
|
{
|
|
// Yes. Continue if the current user has
|
|
// write-attribute access.
|
|
bSkipFile = !(dwCurrentAccess & FILE_WRITE_ATTRIBUTES);
|
|
}
|
|
|
|
// Have the contents changed offline?
|
|
if (!bSkipFile &&
|
|
((FLAG_CSC_COPY_STATUS_DATA_LOCALLY_MODIFIED
|
|
| FLAG_CSC_COPY_STATUS_LOCALLY_CREATED
|
|
| FLAG_CSC_COPY_STATUS_LOCALLY_DELETED) & dwStatus))
|
|
{
|
|
// Yes. Continue if the current user has
|
|
// write-data access.
|
|
bSkipFile = !(dwCurrentAccess & FILE_WRITE_DATA);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
|
|
// We're filling. Continue if the current user has
|
|
// read-data access, otherwise skip.
|
|
|
|
bSkipFile = !(dwCurrentAccess & FILE_READ_DATA);
|
|
}
|
|
}
|
|
}
|
|
else if (!(pSyncData->dwSyncStatus & SDS_SYNC_OUT))
|
|
{
|
|
// It's a directory and we're in CSCFillSparseFiles.
|
|
|
|
// Note that we never skip directories when merging (we may be
|
|
// interested in a file further down the tree) although we
|
|
// can skip directories when filling.
|
|
|
|
// If it's not in the file selection, skip it.
|
|
if (m_pFileList && !m_pFileList->FileExists(pszName, false))
|
|
{
|
|
bSkipFile = TRUE;
|
|
}
|
|
}
|
|
|
|
if (bSkipFile)
|
|
{
|
|
Trace((TEXT("Skipping: %s"), pszName));
|
|
dwResult = CSCPROC_RETURN_SKIP;
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_FILE_SKIPPED;
|
|
break;
|
|
}
|
|
|
|
Trace((TEXT("File begin: %s"), pszName));
|
|
|
|
|
|
// Since we sometimes don't skip directories, even when it turns
|
|
// out they have nothing that the current user is interested in,
|
|
// don't display directory names in SyncMgr.
|
|
|
|
// If we sync a file farther down the tree, we will display the
|
|
// filename and the intervening directory names will be visible
|
|
// at that time.
|
|
|
|
if (!(pFind32->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY))
|
|
{
|
|
USES_CONVERSION;
|
|
|
|
// Tell SyncMgr what we're doing
|
|
spi.mask = SYNCMGRPROGRESSITEM_STATUSTEXT;
|
|
spi.lpcStatusText = T2CW(pszName + lstrlen(pSyncData->pszShareName) + 1);
|
|
NotifySyncMgr(pSyncData, &spi);
|
|
}
|
|
|
|
// dwParam1 is non-zero when there is a conflict, i.e. both the
|
|
// local and remote versions of the file have been modified.
|
|
if (dwParam1)
|
|
{
|
|
if (dwParam2) // indicates server file deleted
|
|
{
|
|
Trace((TEXT("Delete conflict: %d"), dwParam2));
|
|
dwResult = HandleDeleteConflict(pSyncData, pszName, dwStatus, dwHintFlags, pFind32);
|
|
}
|
|
else
|
|
{
|
|
Trace((TEXT("Update conflict: %d"), dwParam1));
|
|
dwResult = HandleFileConflict(pSyncData, pszName, dwStatus, dwHintFlags, pFind32);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case CSCPROC_REASON_END:
|
|
// dwParam2 == error code (winerror.h)
|
|
if (3000 <= dwParam2 && dwParam2 <= 3200)
|
|
{
|
|
// Private error codes used in cscdll
|
|
Trace((TEXT("CSC error: %d"), dwParam2));
|
|
dwParam2 = NOERROR;
|
|
}
|
|
else if (ERROR_OPERATION_ABORTED == dwParam2)
|
|
{
|
|
// We returned CSCPROC_RETURN_ABORT for some reason.
|
|
// Whatever it was, we already reported it.
|
|
dwParam2 = NOERROR;
|
|
dwResult = CSCPROC_RETURN_ABORT;
|
|
}
|
|
if (lstrlen(pszName) == lstrlen(pSyncData->pszShareName))
|
|
{
|
|
// SHARE END
|
|
TraceAssert(!lstrcmpi(pszName, pSyncData->pszShareName));
|
|
Trace((TEXT("Share end: %s"), pszName));
|
|
|
|
pSyncData->dwSyncStatus &= ~SDS_SYNC_STARTED;
|
|
}
|
|
else
|
|
{
|
|
BOOL bUpdateProgress = FALSE;
|
|
|
|
// FILE END
|
|
if (!(pSyncData->dwSyncStatus & SDS_SYNC_FILE_SKIPPED))
|
|
{
|
|
Trace((TEXT("File end: %s"), pszName));
|
|
|
|
bUpdateProgress = TRUE;
|
|
|
|
// BUGBUG special case errors
|
|
switch (dwParam2)
|
|
{
|
|
case ERROR_ACCESS_DENIED:
|
|
if (FILE_ATTRIBUTE_DIRECTORY & pFind32->dwFileAttributes)
|
|
{
|
|
// 317751 directories are not per-user, so if a
|
|
// different user syncs, we can hit this. Don't want
|
|
// to show an error message unless we are ignoring
|
|
// access (when the user explicitly selected something
|
|
// to pin/sync).
|
|
|
|
// 394362 BrianV hit this running as an admin, so don't
|
|
// show this error for admins either.
|
|
|
|
if (!(pSyncData->dwSyncStatus & CSC_SYNC_IGNORE_ACCESS))
|
|
{
|
|
TraceMsg("Suppressing ERROR_ACCESS_DENIED on folder");
|
|
dwParam2 = NOERROR;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case ERROR_GEN_FAILURE:
|
|
TraceMsg("Received ERROR_GEN_FAILURE from cscdll");
|
|
if (dwStatus & FLAG_CSC_COPY_STATUS_FILE_IN_USE)
|
|
dwParam2 = ERROR_OPEN_FILES;
|
|
break;
|
|
|
|
case ERROR_FILE_NOT_FOUND:
|
|
case ERROR_PATH_NOT_FOUND:
|
|
// We either handle the error here or the user is
|
|
// prompted, so no need for another error message.
|
|
dwParam2 = NOERROR;
|
|
// If this is an autocache file and has not been modified
|
|
// offline, nuke it now. Otherwise, prompt for action.
|
|
if (CSCPROC_RETURN_FORCE_OUTWARD == HandleDeleteConflict(pSyncData,
|
|
pszName,
|
|
dwStatus,
|
|
dwHintFlags,
|
|
pFind32))
|
|
{
|
|
dwParam2 = CopyLocalFileWithDriveMapping(pszName,
|
|
pszName,
|
|
pSyncData->pszShareName,
|
|
pSyncData->szDrive,
|
|
(FILE_ATTRIBUTE_DIRECTORY & pFind32->dwFileAttributes));
|
|
}
|
|
break;
|
|
|
|
case ERROR_DISK_FULL:
|
|
// There's no point continuing
|
|
dwResult = CSCPROC_RETURN_ABORT;
|
|
break;
|
|
|
|
default:
|
|
// nothing
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pSyncData->dwSyncStatus &= ~SDS_SYNC_FILE_SKIPPED;
|
|
dwParam2 = NOERROR;
|
|
|
|
// If doing full sync, then we count progress for skipped
|
|
// files as well. Not true for quick fill or merge.
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_IN_FULL)
|
|
bUpdateProgress = TRUE;
|
|
}
|
|
|
|
// Update progress in SyncMgr
|
|
if (bUpdateProgress)
|
|
{
|
|
pSyncData->cFilesDone++;
|
|
|
|
spi.mask = SYNCMGRPROGRESSITEM_PROGVALUE;
|
|
spi.iProgValue = min(pSyncData->cFilesDone, pSyncData->cFilesToSync - 1);
|
|
Trace((TEXT("%d of %d files done"), spi.iProgValue, pSyncData->cFilesToSync));
|
|
NotifySyncMgr(pSyncData, &spi);
|
|
}
|
|
}
|
|
if (dwParam2 != NOERROR)
|
|
{
|
|
UINT idsError = GetErrorFormat(dwParam2, boolify(pSyncData->dwSyncStatus & SDS_SYNC_OUT));
|
|
if (IDS_SHARE_CONNECT_ERROR == idsError)
|
|
{
|
|
LPTSTR pszErr = GetErrorText(dwParam2);
|
|
|
|
// Special-case the "can't connect to share" error.
|
|
// Display only the share name in the error message
|
|
// and abort the synchronization of this share.
|
|
|
|
LogError(pSyncData->ItemID,
|
|
SYNCMGRLOGLEVEL_ERROR,
|
|
idsError,
|
|
pSyncData->pszShareName,
|
|
pszErr ? pszErr : TEXT(""));
|
|
|
|
LocalFreeString(&pszErr);
|
|
dwResult = CSCPROC_RETURN_ABORT;
|
|
}
|
|
else
|
|
{
|
|
LogError(pSyncData->ItemID,
|
|
idsError,
|
|
pszName,
|
|
dwParam2);
|
|
}
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_ERROR;
|
|
}
|
|
break;
|
|
}
|
|
|
|
// Check for Cancel
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
{
|
|
TraceMsg("Cancelling sync operation");
|
|
dwResult = CSCPROC_RETURN_ABORT;
|
|
}
|
|
|
|
TraceLeaveValue(dwResult);
|
|
}
|
|
|
|
|
|
void
|
|
CCscUpdate::NotifySyncMgr(PSYNCTHREADDATA pSyncData, LPSYNCMGRPROGRESSITEM pspi)
|
|
{
|
|
LPSYNCMGRSYNCHRONIZECALLBACK pSyncMgr = pSyncData->pThis->m_pSyncMgrCB;
|
|
|
|
if (pSyncMgr)
|
|
{
|
|
HRESULT hr = pSyncMgr->Progress(pSyncData->ItemID, pspi);
|
|
|
|
if (hr == S_SYNCMGR_CANCELITEM || hr == S_SYNCMGR_CANCELALL)
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_CANCELLED;
|
|
}
|
|
}
|
|
|
|
|
|
DWORD WINAPI
|
|
CCscUpdate::_CscCallback(LPCTSTR pszName,
|
|
DWORD dwStatus,
|
|
DWORD dwHintFlags,
|
|
DWORD dwPinCount,
|
|
LPWIN32_FIND_DATA pFind32,
|
|
DWORD dwReason,
|
|
DWORD dwParam1,
|
|
DWORD dwParam2,
|
|
DWORD_PTR dwContext)
|
|
{
|
|
DWORD dwResult = CSCPROC_RETURN_ABORT;
|
|
PSYNCTHREADDATA pSyncData = (PSYNCTHREADDATA)dwContext;
|
|
|
|
if (pSyncData != NULL && pSyncData->pThis != NULL)
|
|
dwResult = pSyncData->pThis->CscCallback(pSyncData,
|
|
pszName,
|
|
dwStatus,
|
|
dwHintFlags,
|
|
dwPinCount,
|
|
pFind32,
|
|
dwReason,
|
|
dwParam1,
|
|
dwParam2);
|
|
return dwResult;
|
|
}
|
|
|
|
|
|
BOOL
|
|
CCscUpdate::PinLinkTarget(LPCTSTR pszName, PSYNCTHREADDATA pSyncData)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
LPTSTR pszTarget = NULL;
|
|
DWORD dwAttr = 0;
|
|
|
|
TraceEnter(TRACE_SHELLEX, "PinLinkTarget");
|
|
|
|
GetLinkTarget(pszName, NULL, &pszTarget, &dwAttr);
|
|
if (pszTarget)
|
|
{
|
|
TraceAssert(!(dwAttr & FILE_ATTRIBUTE_DIRECTORY));
|
|
|
|
// Check for EFS
|
|
if ((FILE_ATTRIBUTE_ENCRYPTED & dwAttr) && SkipEFSPin(pSyncData, pszTarget))
|
|
ExitGracefully(bResult, FALSE, "Skipping EFS link target");
|
|
|
|
if (!(pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) &&
|
|
CSCPinFile(pszTarget,
|
|
FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_INHERIT_USER,
|
|
NULL,
|
|
NULL,
|
|
NULL))
|
|
{
|
|
WIN32_FIND_DATA fd = {0};
|
|
LPCTSTR pszT = PathFindFileName(pszTarget);
|
|
fd.dwFileAttributes = dwAttr;
|
|
lstrcpyn(fd.cFileName, pszT ? pszT : pszTarget, ARRAYSIZE(fd.cFileName));
|
|
|
|
ShellChangeNotify(pszTarget, &fd, FALSE);
|
|
|
|
bResult = TRUE;
|
|
|
|
if (FILE_ATTRIBUTE_ENCRYPTED & dwAttr)
|
|
{
|
|
LogError(pSyncData->ItemID,
|
|
IDS_PIN_ENCRYPT_WARNING,
|
|
pszTarget,
|
|
NOERROR,
|
|
SYNCMGRLOGLEVEL_WARNING);
|
|
}
|
|
}
|
|
}
|
|
|
|
exit_gracefully:
|
|
|
|
LocalFreeString(&pszTarget);
|
|
TraceLeaveValue(bResult);
|
|
}
|
|
|
|
|
|
|
|
|
|
DWORD WINAPI
|
|
CCscUpdate::_PinNewFilesW32Callback(LPCTSTR pszName,
|
|
ENUM_REASON eReason,
|
|
LPWIN32_FIND_DATA pFind32,
|
|
LPARAM lpContext)
|
|
{
|
|
PSYNCTHREADDATA pSyncData = (PSYNCTHREADDATA)lpContext;
|
|
DWORD dwHintFlags = 0;
|
|
DWORD dwErr = NOERROR;
|
|
|
|
// This callback is used when enumerating a pinned folder looking
|
|
// for new files on the server. Since the parent folder is pinned,
|
|
// any files in it that aren't pinned get pinned here.
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::_PinNewFilesW32Callback");
|
|
TraceAssert(pSyncData != NULL);
|
|
|
|
// Check for Cancel
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
{
|
|
TraceMsg("Cancelling sync operation");
|
|
TraceLeaveValue(CSCPROC_RETURN_ABORT);
|
|
}
|
|
|
|
// Always ignore folder_end and ignore folder_begin if we
|
|
// aren't doing a recursive pin operation.
|
|
if (eReason == ENUM_REASON_FOLDER_END ||
|
|
(eReason == ENUM_REASON_FOLDER_BEGIN && !(pSyncData->pThis->m_dwSyncFlags & CSC_SYNC_PIN_RECURSE)))
|
|
{
|
|
TraceLeaveValue(CSCPROC_RETURN_SKIP);
|
|
}
|
|
|
|
// At this point, we either have 1) a file or 2) folder_begin + recurse,
|
|
// so pin anything that isn't pinned.
|
|
|
|
// Is this file already pinned?
|
|
if (!CSCQueryFileStatus(pszName, NULL, NULL, &dwHintFlags))
|
|
dwErr = GetLastError();
|
|
|
|
if (ERROR_FILE_NOT_FOUND == dwErr ||
|
|
(NOERROR == dwErr && !(dwHintFlags & (FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_ADMIN))))
|
|
{
|
|
// Check for EFS
|
|
BOOL bIsEFSFile = (FILE_ATTRIBUTE_ENCRYPTED & pFind32->dwFileAttributes) &&
|
|
!(FILE_ATTRIBUTE_DIRECTORY & pFind32->dwFileAttributes);
|
|
|
|
if (bIsEFSFile && pSyncData->pThis->SkipEFSPin(pSyncData, pszName))
|
|
TraceLeaveResult(CSCPROC_RETURN_SKIP);
|
|
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
TraceLeaveValue(CSCPROC_RETURN_ABORT);
|
|
|
|
// Pin it now.
|
|
dwHintFlags |= FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_INHERIT_USER;
|
|
|
|
if (CSCPinFile(pszName, dwHintFlags, NULL, NULL, NULL))
|
|
{
|
|
|
|
pSyncData->cFilesToSync++;
|
|
ShellChangeNotify(pszName, pFind32, FALSE);
|
|
|
|
if (bIsEFSFile)
|
|
{
|
|
pSyncData->pThis->LogError(pSyncData->ItemID,
|
|
IDS_PIN_ENCRYPT_WARNING,
|
|
pszName,
|
|
NOERROR,
|
|
SYNCMGRLOGLEVEL_WARNING);
|
|
}
|
|
|
|
// If this is a link file, pin the target (if appropriate)
|
|
LPTSTR pszExtn = PathFindExtension(pszName);
|
|
if (pszExtn && !lstrcmpi(pszExtn, c_szLNK))
|
|
{
|
|
if (pSyncData->pThis->PinLinkTarget(pszName, pSyncData))
|
|
pSyncData->cFilesToSync++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DWORD dwError = GetLastError();
|
|
UINT idsError = GetErrorFormat(dwError);
|
|
if (IDS_SHARE_CONNECT_ERROR == idsError)
|
|
{
|
|
LPTSTR pszErr = GetErrorText(dwError);
|
|
|
|
// Special-case the "can't connect to share" error.
|
|
// Display only the share name in the error message
|
|
// and abort the pinning of this share.
|
|
|
|
pSyncData->pThis->LogError(pSyncData->ItemID,
|
|
SYNCMGRLOGLEVEL_ERROR,
|
|
idsError,
|
|
pSyncData->pszShareName,
|
|
pszErr ? pszErr : TEXT(""));
|
|
|
|
LocalFreeString(&pszErr);
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_CANCELLED;
|
|
}
|
|
else
|
|
{
|
|
pSyncData->pThis->LogError(pSyncData->ItemID,
|
|
IDS_PIN_FILE_ERROR,
|
|
pszName,
|
|
dwError);
|
|
}
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_ERROR;
|
|
}
|
|
|
|
LPTSTR pszScanMsg = NULL;
|
|
SYNCMGRPROGRESSITEM spi;
|
|
spi.cbSize = sizeof(spi);
|
|
spi.mask = SYNCMGRPROGRESSITEM_STATUSTEXT;
|
|
spi.lpcStatusText = L" ";
|
|
|
|
// Skip the share name
|
|
TraceAssert(PathIsPrefix(pSyncData->pszShareName, pszName));
|
|
pszName += lstrlen(pSyncData->pszShareName);
|
|
if (*pszName == TEXT('\\'))
|
|
pszName++;
|
|
|
|
LPCTSTR pszFile = PathFindFileName(pszName);
|
|
TCHAR szPath[MAX_PATH] = TEXT("\\");
|
|
|
|
if (*pszName)
|
|
lstrcpyn(szPath, pszName, min(ARRAYSIZE(szPath),(int)(pszFile-pszName)));
|
|
|
|
// If we still have a name, build a string like
|
|
// "scanning: dir\foo.txt" to display in SyncMgr
|
|
if (FormatStringID(&pszScanMsg, g_hInstance, IDS_NEW_SCAN, pszFile, szPath))
|
|
{
|
|
USES_CONVERSION;
|
|
spi.lpcStatusText = T2CW(pszScanMsg);
|
|
}
|
|
|
|
NotifySyncMgr(pSyncData, &spi);
|
|
|
|
LocalFreeString(&pszScanMsg);
|
|
}
|
|
else if ((dwHintFlags & FLAG_CSC_HINT_PIN_USER) &&
|
|
(pSyncData->pThis->m_dwSyncFlags & CSC_SYNC_PINFILES))
|
|
{
|
|
// FLAG_CSC_HINT_PIN_USER being set implies that CSCQueryFileStatus
|
|
// succeeded above.
|
|
|
|
// The item was already pinned. Save it in the undo exclusion list.
|
|
if (!pSyncData->pUndoExclusionList)
|
|
pSyncData->pUndoExclusionList = new CscFilenameList;
|
|
|
|
if (pSyncData->pUndoExclusionList)
|
|
pSyncData->pUndoExclusionList->AddFile(pszName);
|
|
}
|
|
|
|
TraceLeaveValue(CSCPROC_RETURN_CONTINUE);
|
|
}
|
|
|
|
|
|
DWORD WINAPI
|
|
CCscUpdate::_PinNewFilesCSCCallback(LPCTSTR pszName,
|
|
ENUM_REASON eReason,
|
|
DWORD /*dwStatus*/,
|
|
DWORD dwHintFlags,
|
|
DWORD /*dwPinCount*/,
|
|
LPWIN32_FIND_DATA /*pFind32*/,
|
|
LPARAM lpContext)
|
|
{
|
|
PSYNCTHREADDATA pSyncData = (PSYNCTHREADDATA)lpContext;
|
|
PCSCUPDATE pThis;
|
|
|
|
// This callback is used when enumerating the CSC database looking
|
|
// for pinned folders, with the intention of pinning new files
|
|
// in those folders on the server.
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::_PinNewFilesCSCCallback");
|
|
TraceAssert(pSyncData != NULL);
|
|
TraceAssert(pSyncData->pThis != NULL);
|
|
|
|
pThis = pSyncData->pThis;
|
|
|
|
// Check for Cancel
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
{
|
|
TraceMsg("Cancelling sync operation");
|
|
TraceLeaveValue(CSCPROC_RETURN_ABORT);
|
|
}
|
|
|
|
// If this isn't a directory with the user hint flag, keep looking.
|
|
if (eReason != ENUM_REASON_FOLDER_BEGIN ||
|
|
!(dwHintFlags & FLAG_CSC_HINT_PIN_USER))
|
|
{
|
|
TraceLeaveValue(CSCPROC_RETURN_CONTINUE);
|
|
}
|
|
|
|
// If we have a file list and this directory isn't in the list,
|
|
// continue without doing anything here.
|
|
if (pSyncData->pThis->m_pFileList &&
|
|
!pSyncData->pThis->m_pFileList->FileExists(pszName, false))
|
|
{
|
|
TraceLeaveValue(CSCPROC_RETURN_CONTINUE);
|
|
}
|
|
|
|
// Ok, we've found a directory with the user hint flag set. Walk
|
|
// this directory on the server, pinning any files that aren't pinned.
|
|
_Win32EnumFolder(pszName,
|
|
FALSE,
|
|
_PinNewFilesW32Callback,
|
|
(LPARAM)pSyncData);
|
|
|
|
TraceLeaveValue(CSCPROC_RETURN_CONTINUE);
|
|
}
|
|
|
|
|
|
DWORD WINAPI
|
|
CCscUpdate::_SyncThread(LPVOID pThreadData)
|
|
{
|
|
// Make sure the DLL stays loaded while this thread is running
|
|
HINSTANCE hInstThisDll = LoadLibrary(c_szDllName);
|
|
|
|
PSYNCTHREADDATA pSyncData = (PSYNCTHREADDATA)pThreadData;
|
|
PCSCUPDATE pThis;
|
|
HRESULT hrComInit = E_FAIL;
|
|
SYNCMGRPROGRESSITEM spi = {0};
|
|
DWORD dwErr = NOERROR;
|
|
CSCSHARESTATS shareStats;
|
|
CSCGETSTATSINFO si = { SSEF_NONE,
|
|
SSUF_NONE,
|
|
false, // No access info reqd (faster).
|
|
false };
|
|
ULONG cDirtyFiles = 0;
|
|
ULONG cStaleFiles = 0;
|
|
DWORD dwShareStatus = 0;
|
|
BOOL bShareOnline = FALSE;
|
|
DWORD dwConnectionSpeed = 0;
|
|
|
|
if (!pSyncData)
|
|
return ERROR_INVALID_PARAMETER;
|
|
|
|
pThis = pSyncData->pThis;
|
|
if (!pThis || !pSyncData->pszShareName || !*pSyncData->pszShareName)
|
|
{
|
|
LocalFree(pSyncData);
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
// Make sure the object stays alive as long as we need it
|
|
pThis->AddRef();
|
|
|
|
spi.cbSize = sizeof(spi);
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::_SyncThread");
|
|
|
|
hrComInit = CoInitialize(NULL);
|
|
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
ExitGracefully(dwErr, NOERROR, "Cancelling sync operation");
|
|
|
|
// Figure out how many files need updating
|
|
pSyncData->cFilesDone = 0;
|
|
pSyncData->cFilesToSync = 0;
|
|
_GetShareStatisticsForUser(pSyncData->pszShareName, &si, &shareStats);
|
|
|
|
// Get share status
|
|
CSCQueryFileStatus(pSyncData->pszShareName, &dwShareStatus, NULL, NULL);
|
|
|
|
// The root of a special folder is pinned with a pin count, but
|
|
// not the user-hint flag, so _GetShareStats doesn't count it.
|
|
// We need to count this for some of the checks below.
|
|
// (If the share is manual-cache, these look exactly like the
|
|
// Siemens scenario in 341786, but we want to sync them.)
|
|
if (shareStats.cTotal && pThis->IsSpecialFolderShare(pSyncData->pszShareName))
|
|
{
|
|
shareStats.cPinned++;
|
|
}
|
|
|
|
if (pThis->m_dwSyncFlags & CSC_SYNC_OUT)
|
|
{
|
|
cDirtyFiles = shareStats.cModified;
|
|
|
|
|
|
// Force the merge code if there are open files, so we are
|
|
// sure to do the open file warning. The danger here is that we
|
|
// don't warn because the share with open files has nothing dirty,
|
|
// but we merge changes on another share and then transition online.
|
|
// We don't want to transition online without warning about open files.
|
|
|
|
if (0 == cDirtyFiles)
|
|
{
|
|
if ((FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP & dwShareStatus) &&
|
|
(FLAG_CSC_SHARE_STATUS_FILES_OPEN & dwShareStatus))
|
|
{
|
|
cDirtyFiles++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pThis->m_dwSyncFlags & CSC_SYNC_IN_FULL)
|
|
{
|
|
// For full inward sync, always set cStaleFiles to at least 1 to force
|
|
// a call to CSCFillSparseFiles.
|
|
// Also, we get callbacks for each file and folder, even if they
|
|
// are not sparse or stale, so go with cTotal here to make
|
|
// the progress bar look right.
|
|
cStaleFiles = max(shareStats.cTotal, 1);
|
|
}
|
|
else if (pThis->m_dwSyncFlags & CSC_SYNC_IN_QUICK)
|
|
{
|
|
cStaleFiles = shareStats.cSparse;
|
|
|
|
// If we're pinning, then it's possible that nothing is sparse yet,
|
|
// but we'll need to call CSCFillSparseFiles.
|
|
if (pThis->m_dwSyncFlags & CSC_SYNC_PINFILES)
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_FORCE_INWARD;
|
|
}
|
|
|
|
// Self-host notification callback
|
|
CSCUI_NOTIFYHOOK((CSCH_SyncShare, TEXT("Sync: %1, Dirty: %2!d!, Stale: %3!d!"), pSyncData->pszShareName, cDirtyFiles, cStaleFiles));
|
|
|
|
if (dwShareStatus & FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP)
|
|
{
|
|
// Can't call CSCFillSparseFiles when disconnected (it just fails)
|
|
cStaleFiles = 0;
|
|
}
|
|
else if ((dwShareStatus & FLAG_CSC_SHARE_STATUS_CACHING_MASK) == FLAG_CSC_SHARE_STATUS_MANUAL_REINT
|
|
&& 0 == shareStats.cPinned
|
|
&& !(pThis->m_dwSyncFlags & CSC_SYNC_PINFILES))
|
|
{
|
|
// On a manual share, if nothing is pinned (and we aren't pinning)
|
|
// then we prefer not to call CSCFillSparseFiles on the share.
|
|
// Raid #341786
|
|
Trace((TEXT("Manual cache share '%s' has only autocached files"), pSyncData->pszShareName));
|
|
cStaleFiles = 0;
|
|
}
|
|
|
|
pSyncData->cFilesToSync = cDirtyFiles + cStaleFiles;
|
|
|
|
|
|
// At this point, if pSyncData->cFilesToSync is nonzero, then we are doing
|
|
// a sync, and will be calling CSCBeginSynchronization to connect to the
|
|
// share (with prompt for credentials if necessary).
|
|
|
|
// If SDS_SYNC_FORCE_INWARD is on, then we are pinning files. We will only
|
|
// call CSCFillSparseFiles is the server is in connected mode, and we will
|
|
// only call CSCBeginSynchronization if pSyncData->cFilesToSync is nonzero
|
|
// (to pin something, you must already have a connection to the share).
|
|
|
|
|
|
if (0 == pSyncData->cFilesToSync && !(pSyncData->dwSyncStatus & SDS_SYNC_FORCE_INWARD))
|
|
ExitGracefully(dwErr, NOERROR, "Nothing to synchronize");
|
|
|
|
// Tell SyncMgr how many files we're updating
|
|
spi.mask = SYNCMGRPROGRESSITEM_STATUSTYPE
|
|
| SYNCMGRPROGRESSITEM_PROGVALUE | SYNCMGRPROGRESSITEM_MAXVALUE;
|
|
spi.dwStatusType = SYNCMGRSTATUS_UPDATING;
|
|
spi.iProgValue = 0;
|
|
spi.iMaxValue = pSyncData->cFilesToSync;
|
|
Trace((TEXT("%d files to sync on %s"), spi.iMaxValue, pSyncData->pszShareName));
|
|
NotifySyncMgr(pSyncData, &spi);
|
|
|
|
#if 1
|
|
// BUGBUG RAS isn't notifying RDR before returning success, so
|
|
// do this bogus delay. Remove this when RAS is fixed.
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_RAS_CONNECTED)
|
|
{
|
|
|
|
// We've just established a RAS connection, but there's a timing
|
|
// problem with RDR. Wait until we can connect to the share, but
|
|
// don't wait longer than RAS_CONNECT_DELAY.
|
|
|
|
Trace((TEXT("RAS delay for %s"), pSyncData->pszShareName));
|
|
DWORD dwDelayTime = GetTickCount() + RAS_CONNECT_DELAY;
|
|
#ifdef DEBUG
|
|
int cAttempts = 0;
|
|
#endif
|
|
while (!bShareOnline &&
|
|
GetTickCount() < dwDelayTime &&
|
|
!(pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED))
|
|
{
|
|
WaitForSingleObject(pThis->m_hSyncMutex, INFINITE);
|
|
#ifdef DEBUG
|
|
cAttempts++;
|
|
#endif
|
|
bShareOnline = CSCBeginSynchronization(pSyncData->pszShareName,
|
|
&dwConnectionSpeed,
|
|
&pSyncData->dwCscContext);
|
|
ReleaseMutex(pThis->m_hSyncMutex);
|
|
}
|
|
Trace((TEXT("%s %s after %d attempts (%dms)"),
|
|
pSyncData->pszShareName,
|
|
(bShareOnline ? TEXT("connected") : TEXT("not reachable")),
|
|
cAttempts,
|
|
GetTickCount() - (dwDelayTime - RAS_CONNECT_DELAY)));
|
|
}
|
|
else
|
|
#endif // RAS delay
|
|
if (pSyncData->cFilesToSync)
|
|
{
|
|
|
|
// CSCBeginSynchronization makes a net connection to the share
|
|
// using the "interactive" flag. This causes a credential popup
|
|
// if the current user doesn't have access to the share.
|
|
// Use the sync mutex to avoid multiple concurrent popups.
|
|
|
|
WaitForSingleObject(pThis->m_hSyncMutex, INFINITE);
|
|
bShareOnline = CSCBeginSynchronization(pSyncData->pszShareName,
|
|
&dwConnectionSpeed,
|
|
&pSyncData->dwCscContext);
|
|
ReleaseMutex(pThis->m_hSyncMutex);
|
|
}
|
|
|
|
if (pSyncData->cFilesToSync && !bShareOnline)
|
|
{
|
|
// The share isn't reachable, so there's no point in continuing.
|
|
dwErr = GetLastError();
|
|
|
|
if (ERROR_CANCELLED == dwErr)
|
|
{
|
|
// The user cancelled the credential popup
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_CANCELLED;
|
|
ExitGracefully(dwErr, NOERROR, "User cancelled sync");
|
|
}
|
|
|
|
LPTSTR pszErr = GetErrorText(dwErr);
|
|
pThis->LogError(pSyncData->ItemID,
|
|
SYNCMGRLOGLEVEL_ERROR,
|
|
IDS_SHARE_CONNECT_ERROR,
|
|
pSyncData->pszShareName,
|
|
pszErr);
|
|
LocalFreeString(&pszErr);
|
|
ExitGracefully(dwErr, dwErr, "Share not reachable");
|
|
}
|
|
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
ExitGracefully(dwErr, NOERROR, "Cancelling sync operation");
|
|
|
|
// Note the time of this sync
|
|
pThis->SetLastSyncTime(pSyncData->pszShareName);
|
|
|
|
// Merge
|
|
if (0 != cDirtyFiles)
|
|
{
|
|
dwErr = pThis->MergeShare(pSyncData);
|
|
if (NOERROR != dwErr)
|
|
{
|
|
LPTSTR pszErr = GetErrorText(dwErr);
|
|
pThis->LogError(pSyncData->ItemID,
|
|
SYNCMGRLOGLEVEL_ERROR,
|
|
IDS_MERGE_SHARE_ERROR,
|
|
pSyncData->pszShareName,
|
|
pszErr);
|
|
LocalFreeString(&pszErr);
|
|
ExitGracefully(dwErr, dwErr, "Aborting due to merge error");
|
|
}
|
|
}
|
|
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
ExitGracefully(dwErr, NOERROR, "Cancelling sync operation");
|
|
|
|
// Fill
|
|
if (0 != cStaleFiles || (pSyncData->dwSyncStatus & SDS_SYNC_FORCE_INWARD))
|
|
{
|
|
dwErr = pThis->FillShare(pSyncData, shareStats.cPinned, dwConnectionSpeed);
|
|
}
|
|
|
|
exit_gracefully:
|
|
|
|
// If we called CSCBeginSynchronization and it succeeded,
|
|
// we need to call CSCEndSynchronization.
|
|
if (bShareOnline)
|
|
CSCEndSynchronization(pSyncData->pszShareName, pSyncData->dwCscContext);
|
|
|
|
// Tell SyncMgr that we're done (succeeded, failed, or stopped)
|
|
spi.mask = SYNCMGRPROGRESSITEM_STATUSTYPE | SYNCMGRPROGRESSITEM_STATUSTEXT
|
|
| SYNCMGRPROGRESSITEM_PROGVALUE | SYNCMGRPROGRESSITEM_MAXVALUE;
|
|
spi.dwStatusType = SYNCMGRSTATUS_SUCCEEDED; // Assume success
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
spi.dwStatusType = SYNCMGRSTATUS_STOPPED;
|
|
if (NOERROR != dwErr || (pSyncData->dwSyncStatus & SDS_SYNC_ERROR))
|
|
spi.dwStatusType = SYNCMGRSTATUS_FAILED;
|
|
spi.lpcStatusText = L" ";
|
|
spi.iProgValue = spi.iMaxValue = pSyncData->cFilesToSync; // This tells syncmgr that the item is done
|
|
|
|
NotifySyncMgr(pSyncData, &spi);
|
|
|
|
if ((pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
&& (pThis->m_dwSyncFlags & CSC_SYNC_PINFILES))
|
|
{
|
|
// We cancelled a pin operation, roll back to the previous state
|
|
CscUnpinFileList(pThis->m_pFileList,
|
|
(pThis->m_dwSyncFlags & CSC_SYNC_PIN_RECURSE),
|
|
pSyncData->pszShareName,
|
|
_UndoProgress,
|
|
(LPARAM)pSyncData);
|
|
}
|
|
|
|
// Tell the Update Handler that this thread is exiting
|
|
// This may use OLE to notify SyncMgr that the sync is done,
|
|
// so do this before CoUninitialize.
|
|
Trace((TEXT("%s finished"), pSyncData->pszShareName));
|
|
pThis->SyncThreadCompleted(pSyncData);
|
|
|
|
if (SUCCEEDED(hrComInit))
|
|
CoUninitialize();
|
|
|
|
delete pSyncData->pUndoExclusionList;
|
|
LocalFree(pSyncData);
|
|
|
|
// Release our ref on the object
|
|
// (also release our ref on the DLL)
|
|
pThis->Release();
|
|
|
|
TraceLeave();
|
|
FreeLibraryAndExitThread(hInstThisDll, dwErr);
|
|
return 0;
|
|
}
|
|
|
|
DWORD
|
|
CCscUpdate::MergeShare(PSYNCTHREADDATA pSyncData)
|
|
{
|
|
DWORD dwErr = NOERROR;
|
|
BOOL bMergeResult = TRUE;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::MergeShare");
|
|
|
|
// CSCMergeShare fails if another thread (or process) is
|
|
// currently merging. This is because CSCMergeShare uses
|
|
// a drive letter connection to the share to bypass CSC,
|
|
// and we don't want to use up all of the drive letters.
|
|
// So let's protect the call to CSCMergeShare with a mutex
|
|
// (rather than dealing with failure and retrying, etc.)
|
|
|
|
WaitForSingleObject(m_hSyncMutex, INFINITE);
|
|
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
ExitGracefully(dwErr, NOERROR, "Merge cancelled");
|
|
|
|
|
|
// It would be nice to skip the open file warning if we knew
|
|
// that the open files were on a "silent folder". The best
|
|
// we can do, though, is detect that the open files are on
|
|
// the same share as a silent folder. There's no guarantee
|
|
// that the open files are not from a different folder on
|
|
// the same share, so we have to show the warning.
|
|
|
|
//if (!IsSilentShare(pSyncData->pszShareName))
|
|
{
|
|
DWORD dwShareStatus = 0;
|
|
CSCQueryFileStatus(pSyncData->pszShareName, &dwShareStatus, NULL, NULL);
|
|
if (FLAG_CSC_SHARE_STATUS_FILES_OPEN & dwShareStatus)
|
|
{
|
|
if (CSC_SYNC_MAYBOTHERUSER & m_dwSyncFlags)
|
|
{
|
|
// Only show this warning once per sync (not per thread)
|
|
if (!(CSC_SYNC_OFWARNINGDONE & m_dwSyncFlags))
|
|
{
|
|
m_dwSyncFlags |= CSC_SYNC_OFWARNINGDONE;
|
|
if (IDOK != OpenFilesWarningDialog())
|
|
{
|
|
TraceMsg("Cancelling sync - user bailed at open file warning");
|
|
SetItemStatus(GUID_NULL, SYNCMGRSTATUS_STOPPED);
|
|
}
|
|
}
|
|
// else we already put up the warning on another thread. If the
|
|
// user cancelled, SDS_SYNC_CANCELLED will be set.
|
|
}
|
|
else
|
|
{
|
|
// Don't merge, but continue otherwise.
|
|
LogError(pSyncData->ItemID,
|
|
SYNCMGRLOGLEVEL_WARNING,
|
|
IDS_OPENFILE_MERGE_WARNING,
|
|
pSyncData->pszShareName);
|
|
ExitGracefully(dwErr, NOERROR, "Skipping merge due to open files");
|
|
}
|
|
}
|
|
}
|
|
|
|
pSyncData->dwSyncStatus = SDS_SYNC_OUT;
|
|
|
|
|
|
// Conflict resolution may require stopping and restarting CSCMergeShare,
|
|
// so do this in a loop
|
|
|
|
while (!(pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED))
|
|
{
|
|
Trace((TEXT("Calling CSCMergeShare(%s)"), pSyncData->pszShareName));
|
|
|
|
bMergeResult = CSCMergeShare(pSyncData->pszShareName,
|
|
CCscUpdate::_CscCallback,
|
|
(DWORD_PTR)pSyncData);
|
|
|
|
Trace((TEXT("CSCMergeShare(%s) returned"), pSyncData->pszShareName));
|
|
|
|
// Do we need to merge again?
|
|
if (!(SDS_SYNC_RESTART_MERGE & pSyncData->dwSyncStatus))
|
|
break;
|
|
|
|
pSyncData->dwSyncStatus &= ~SDS_SYNC_RESTART_MERGE;
|
|
}
|
|
|
|
if (!(pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) && !bMergeResult)
|
|
{
|
|
dwErr = GetLastError();
|
|
if (ERROR_OPERATION_ABORTED == dwErr)
|
|
dwErr = NOERROR;
|
|
}
|
|
|
|
exit_gracefully:
|
|
|
|
ReleaseMutex(m_hSyncMutex);
|
|
|
|
TraceLeaveValue(dwErr);
|
|
}
|
|
|
|
DWORD
|
|
CCscUpdate::FillShare(PSYNCTHREADDATA pSyncData, int cPinned, DWORD dwConnectionSpeed)
|
|
{
|
|
DWORD dwErr = NOERROR;
|
|
DWORD dwShareStatus = 0;
|
|
DWORD dwShareHints = 0;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::FillShare");
|
|
|
|
CSCQueryFileStatus(pSyncData->pszShareName, &dwShareStatus, NULL, &dwShareHints);
|
|
|
|
if (m_dwSyncFlags & CSC_SYNC_IN_FULL)
|
|
{
|
|
pSyncData->dwSyncStatus = SDS_SYNC_IN_FULL;
|
|
|
|
Trace((TEXT("Full sync at %d00 bps"), dwConnectionSpeed));
|
|
|
|
|
|
// Check the server for new files that should be pinned.
|
|
|
|
// We can't do this when disconnected. Also, this is
|
|
// time consuming, so don't do it on a slow connection.
|
|
|
|
if (!(FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP & dwShareStatus)
|
|
&& cPinned && !_PathIsSlow(dwConnectionSpeed))
|
|
{
|
|
|
|
// Look for pinned folders on this share by enumerating
|
|
// in the CSC database. Go out to the server only if/when
|
|
// we find a pinned folder.
|
|
|
|
TraceMsg("Running FrankAr code");
|
|
|
|
if (CConfig::GetSingleton().AlwaysPinSubFolders())
|
|
{
|
|
|
|
// If the "AlwaysPinSubFolders" policy is set, we
|
|
// do a recursive pin. This will cause any content
|
|
// (including folders) of a pinned folder to become pinned.
|
|
|
|
pSyncData->pThis->m_dwSyncFlags |= CSC_SYNC_PIN_RECURSE;
|
|
}
|
|
|
|
// First check the root folder
|
|
if (_PinNewFilesCSCCallback(pSyncData->pszShareName,
|
|
ENUM_REASON_FOLDER_BEGIN,
|
|
0,
|
|
dwShareHints,
|
|
0,
|
|
NULL,
|
|
(LPARAM)pSyncData) == CSCPROC_RETURN_CONTINUE)
|
|
{
|
|
_CSCEnumDatabase(pSyncData->pszShareName,
|
|
TRUE,
|
|
_PinNewFilesCSCCallback,
|
|
(LPARAM)pSyncData);
|
|
}
|
|
|
|
TraceMsg("FrankAr code complete");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pSyncData->dwSyncStatus = SDS_SYNC_IN_QUICK;
|
|
|
|
if (m_dwSyncFlags & CSC_SYNC_PINFILES)
|
|
{
|
|
|
|
// Enumerate the file list and pin everything, checking with
|
|
// SyncMgr periodically.
|
|
|
|
PinFiles(pSyncData);
|
|
}
|
|
}
|
|
|
|
if (m_pConflictPinList)
|
|
{
|
|
// Make sure that any files we created because of merge
|
|
// conflicts are pinned.
|
|
PinFiles(pSyncData, TRUE);
|
|
}
|
|
|
|
// Can't fill when disconnected
|
|
if (!(FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP & dwShareStatus))
|
|
{
|
|
// Clear the status text and update the max count in case we
|
|
// pinned somthing above
|
|
SYNCMGRPROGRESSITEM spi;
|
|
spi.cbSize = sizeof(spi);
|
|
spi.mask = SYNCMGRPROGRESSITEM_STATUSTEXT | SYNCMGRPROGRESSITEM_MAXVALUE;
|
|
spi.lpcStatusText = L" ";
|
|
spi.iMaxValue = pSyncData->cFilesToSync;
|
|
Trace((TEXT("%d files to sync on %s"), spi.iMaxValue, pSyncData->pszShareName));
|
|
NotifySyncMgr(pSyncData, &spi);
|
|
|
|
if (!(pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED))
|
|
{
|
|
Trace((TEXT("Calling CSCFillSparseFiles(%s, %s)"), pSyncData->pszShareName, (pSyncData->dwSyncStatus & SDS_SYNC_IN_FULL) ? TEXT("full") : TEXT("quick")));
|
|
if (!CSCFillSparseFiles(pSyncData->pszShareName,
|
|
!!(pSyncData->dwSyncStatus & SDS_SYNC_IN_FULL),
|
|
CCscUpdate::_CscCallback,
|
|
(DWORD_PTR)pSyncData))
|
|
{
|
|
dwErr = GetLastError();
|
|
if (ERROR_OPERATION_ABORTED == dwErr)
|
|
dwErr = NOERROR;
|
|
}
|
|
Trace((TEXT("CSCFillSparseFiles(%s) complete"), pSyncData->pszShareName));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Trace((TEXT("Skipping CSCFillSparseFiles(%s) - server is offline"), pSyncData->pszShareName));
|
|
}
|
|
|
|
TraceLeaveValue(dwErr);
|
|
}
|
|
|
|
void
|
|
CCscUpdate::PinFiles(PSYNCTHREADDATA pSyncData, BOOL bConflictPinList)
|
|
{
|
|
CscFilenameList *pfnl;
|
|
CscFilenameList::HSHARE hShare;
|
|
LPCTSTR pszFile;
|
|
BOOL bRecurse;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::PinFiles");
|
|
TraceAssert((m_dwSyncFlags & CSC_SYNC_PINFILES) || bConflictPinList);
|
|
|
|
pfnl = m_pFileList;
|
|
bRecurse = m_dwSyncFlags & CSC_SYNC_PIN_RECURSE;
|
|
|
|
if (bConflictPinList)
|
|
{
|
|
pfnl = m_pConflictPinList;
|
|
bRecurse = FALSE;
|
|
}
|
|
|
|
if (!pfnl ||
|
|
!pfnl->GetShareHandle(pSyncData->pszShareName, &hShare))
|
|
{
|
|
TraceLeaveVoid();
|
|
}
|
|
|
|
CscFilenameList::FileIter fi = pfnl->CreateFileIterator(hShare);
|
|
|
|
// Iterate over the filenames associated with the share.
|
|
while (pszFile = fi.Next())
|
|
{
|
|
TCHAR szFullPath[MAX_PATH];
|
|
TCHAR szRelativePath[MAX_PATH];
|
|
WIN32_FIND_DATA fd;
|
|
ULONG cchFile = lstrlen(pszFile) + 1; // include NULL
|
|
|
|
// Check for Cancel
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
break;
|
|
|
|
ZeroMemory(&fd, sizeof(fd));
|
|
|
|
// Directories have a trailing "\*"
|
|
if (StrChr(pszFile, TEXT('*')))
|
|
{
|
|
// It's a directory. Trim off the "\*"
|
|
cchFile -= 2;
|
|
|
|
// When pinning at the share level, pszFile points to "*"
|
|
// and cchFile is now zero.
|
|
}
|
|
|
|
szRelativePath[0] = TEXT('\0');
|
|
lstrcpyn(szRelativePath, pszFile, min(cchFile, ARRAYSIZE(szRelativePath)));
|
|
|
|
// Build the full path
|
|
PathCombine(szFullPath, pSyncData->pszShareName, szRelativePath);
|
|
|
|
// Get attributes and test for existence
|
|
fd.dwFileAttributes = GetFileAttributes(szFullPath);
|
|
if ((DWORD)-1 == fd.dwFileAttributes)
|
|
continue;
|
|
|
|
pszFile = PathFindFileName(szFullPath);
|
|
lstrcpyn(fd.cFileName, pszFile ? pszFile : szFullPath, ARRAYSIZE(fd.cFileName));
|
|
|
|
// Check for EFS
|
|
BOOL bIsEFSFile;
|
|
bIsEFSFile = (FILE_ATTRIBUTE_ENCRYPTED & fd.dwFileAttributes) &&
|
|
!(FILE_ATTRIBUTE_DIRECTORY & fd.dwFileAttributes);
|
|
|
|
if (bIsEFSFile && SkipEFSPin(pSyncData, szFullPath))
|
|
continue;
|
|
|
|
if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)
|
|
break;
|
|
|
|
// Pin it
|
|
if (CSCPinFile(szFullPath,
|
|
FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_INHERIT_USER,
|
|
NULL,
|
|
NULL,
|
|
NULL))
|
|
{
|
|
if (bConflictPinList && m_pFileList)
|
|
m_pFileList->AddFile(szFullPath, !!(fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY));
|
|
pSyncData->cFilesToSync++;
|
|
ShellChangeNotify(szFullPath, &fd, FALSE);
|
|
if (bIsEFSFile)
|
|
{
|
|
LogError(pSyncData->ItemID,
|
|
IDS_PIN_ENCRYPT_WARNING,
|
|
szFullPath,
|
|
NOERROR,
|
|
SYNCMGRLOGLEVEL_WARNING);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DWORD dwError = GetLastError();
|
|
UINT idsError = GetErrorFormat(dwError);
|
|
if (IDS_SHARE_CONNECT_ERROR == idsError)
|
|
{
|
|
LPTSTR pszErr = GetErrorText(dwError);
|
|
|
|
// Special-case the "can't connect to share" error.
|
|
// Display only the share name in the error message
|
|
// and abort the pinning of this share.
|
|
|
|
LogError(pSyncData->ItemID,
|
|
SYNCMGRLOGLEVEL_ERROR,
|
|
idsError,
|
|
pSyncData->pszShareName,
|
|
pszErr ? pszErr : TEXT(""));
|
|
|
|
LocalFreeString(&pszErr);
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_CANCELLED;
|
|
}
|
|
else
|
|
{
|
|
LogError(pSyncData->ItemID,
|
|
IDS_PIN_FILE_ERROR,
|
|
szFullPath,
|
|
dwError);
|
|
}
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_ERROR;
|
|
}
|
|
|
|
// If it's a directory, pin its contents
|
|
if (fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
{
|
|
_Win32EnumFolder(szFullPath,
|
|
bRecurse,
|
|
CCscUpdate::_PinNewFilesW32Callback,
|
|
(LPARAM)pSyncData);
|
|
}
|
|
}
|
|
|
|
// Flush the shell notify queue
|
|
ShellChangeNotify(NULL, TRUE);
|
|
TraceLeaveVoid();
|
|
}
|
|
|
|
|
|
void
|
|
CCscUpdate::NotifyUndo(PSYNCTHREADDATA pSyncData, LPCTSTR pszName)
|
|
{
|
|
LPTSTR pszMsg;
|
|
SYNCMGRPROGRESSITEM spi;
|
|
spi.cbSize = sizeof(spi);
|
|
spi.mask = SYNCMGRPROGRESSITEM_STATUSTEXT;
|
|
|
|
spi.lpcStatusText = L" ";
|
|
|
|
// Skip the share name
|
|
if (PathIsPrefix(pSyncData->pszShareName, pszName))
|
|
{
|
|
pszName += lstrlen(pSyncData->pszShareName);
|
|
if (*pszName == TEXT('\\'))
|
|
pszName++;
|
|
}
|
|
|
|
LPCTSTR pszFile = PathFindFileName(pszName);
|
|
TCHAR szPath[MAX_PATH] = TEXT("\\");
|
|
|
|
if (*pszName)
|
|
lstrcpyn(szPath, pszName, min(ARRAYSIZE(szPath),(int)(pszFile-pszName)));
|
|
|
|
// If we still have a name, build a string like
|
|
// "undo: dir\foo.txt" to display in SyncMgr
|
|
if (FormatStringID(&pszMsg, g_hInstance, IDS_UNDO_SCAN, pszFile, szPath))
|
|
{
|
|
USES_CONVERSION;
|
|
spi.lpcStatusText = T2CW(pszMsg);
|
|
}
|
|
|
|
NotifySyncMgr(pSyncData, &spi);
|
|
|
|
LocalFreeString(&pszMsg);
|
|
}
|
|
|
|
|
|
DWORD WINAPI
|
|
CCscUpdate::_UndoProgress(LPCTSTR pszItem, LPARAM lpContext)
|
|
{
|
|
PSYNCTHREADDATA pSyncData = reinterpret_cast<PSYNCTHREADDATA>(lpContext);
|
|
|
|
if (pSyncData->pUndoExclusionList &&
|
|
pSyncData->pUndoExclusionList->FileExists(pszItem))
|
|
{
|
|
return CSCPROC_RETURN_SKIP;
|
|
}
|
|
|
|
// Update SyncMgr
|
|
pSyncData->pThis->NotifyUndo(pSyncData, pszItem);
|
|
|
|
return CSCPROC_RETURN_CONTINUE;
|
|
}
|
|
|
|
#define PINEFS_SKIP 0
|
|
#define PINEFS_PIN 1
|
|
#define PINEFS_CANCEL 2
|
|
|
|
INT_PTR CALLBACK
|
|
ConfirmEFSPinProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
INT_PTR nResult = 0;
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_INITDIALOG:
|
|
{
|
|
LPTSTR pszMsg = NULL;
|
|
LPCTSTR pszFile = PathFindFileName((LPCTSTR)lParam);
|
|
FormatStringID(&pszMsg, g_hInstance, IDS_FMT_PIN_EFS_MSG, pszFile);
|
|
if (pszMsg)
|
|
{
|
|
SetDlgItemText(hDlg, IDC_EFS_MSG, pszMsg);
|
|
LocalFree(pszMsg);
|
|
}
|
|
else
|
|
EndDialog(hDlg, PINEFS_SKIP);
|
|
CheckDlgButton(hDlg, IDC_SKIP_EFS, BST_CHECKED);
|
|
}
|
|
nResult = TRUE;
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
switch (LOWORD(wParam))
|
|
{
|
|
case IDOK:
|
|
EndDialog(hDlg, BST_CHECKED == IsDlgButtonChecked(hDlg, IDC_SKIP_EFS) ? PINEFS_SKIP : PINEFS_PIN);
|
|
nResult = TRUE;
|
|
break;
|
|
|
|
case IDCANCEL:
|
|
EndDialog(hDlg, PINEFS_CANCEL);
|
|
nResult = TRUE;
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return nResult;
|
|
}
|
|
|
|
BOOL
|
|
CCscUpdate::SkipEFSPin(PSYNCTHREADDATA pSyncData, LPCTSTR pszItem)
|
|
{
|
|
BOOL bSkip = FALSE;
|
|
int iResult = PINEFS_PIN;
|
|
|
|
if ((CSC_SYNC_SKIP_EFS & m_dwSyncFlags) ||
|
|
!(CSC_SYNC_MAYBOTHERUSER & m_dwSyncFlags))
|
|
{
|
|
iResult = PINEFS_SKIP;
|
|
}
|
|
else if (!(CSC_SYNC_EFS_WARNING_SHOWN & m_dwSyncFlags))
|
|
{
|
|
EnterCriticalSection(&m_csThreadList);
|
|
|
|
// Another thread may have come through here while we were
|
|
// waiting for the critical section, so recheck any flags
|
|
// that are modified here.
|
|
|
|
// We could take the critical section at the top of the function and
|
|
// only check flags once, but this way we avoid extra blocking.
|
|
|
|
if (CSC_SYNC_SKIP_EFS & m_dwSyncFlags)
|
|
{
|
|
iResult = PINEFS_SKIP;
|
|
}
|
|
else if (!(CSC_SYNC_EFS_WARNING_SHOWN & m_dwSyncFlags))
|
|
{
|
|
m_dwSyncFlags |= CSC_SYNC_EFS_WARNING_SHOWN;
|
|
|
|
// Suspend the other sync threads
|
|
SetSyncThreadStatus(SyncPause, pSyncData->ItemID);
|
|
|
|
iResult = (int)DialogBoxParam(g_hInstance,
|
|
MAKEINTRESOURCE(IDD_CONFIRM_PIN_EFS),
|
|
m_hwndDlgParent,
|
|
ConfirmEFSPinProc,
|
|
(LPARAM)pszItem);
|
|
if (PINEFS_SKIP == iResult)
|
|
m_dwSyncFlags |= CSC_SYNC_SKIP_EFS;
|
|
|
|
// Resume syncing
|
|
SetSyncThreadStatus(SyncResume, pSyncData->ItemID);
|
|
}
|
|
|
|
LeaveCriticalSection(&m_csThreadList);
|
|
}
|
|
|
|
switch (iResult)
|
|
{
|
|
default:
|
|
case PINEFS_SKIP:
|
|
bSkip = TRUE;
|
|
break;
|
|
|
|
case PINEFS_PIN:
|
|
// continue
|
|
break;
|
|
|
|
case PINEFS_CANCEL:
|
|
// stop all threads
|
|
SetItemStatus(GUID_NULL, SYNCMGRSTATUS_STOPPED);
|
|
break;
|
|
}
|
|
|
|
return bSkip;
|
|
}
|
|
|
|
HRESULT
|
|
CCscUpdate::SetSyncThreadStatus(eSetSyncStatus status, REFGUID rItemID)
|
|
{
|
|
// Assume success here. If we don't find the thread,
|
|
// it's probably already finished.
|
|
HRESULT hr = S_OK;
|
|
BOOL bOneItem;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CCscUpdate::SetSyncThreadStatus");
|
|
|
|
bOneItem = (SyncStop == status && !IsEqualGUID(rItemID, GUID_NULL));
|
|
|
|
EnterCriticalSection(&m_csThreadList);
|
|
|
|
if (NULL != m_hSyncThreads)
|
|
{
|
|
int cItems = DPA_GetPtrCount(m_hSyncThreads);
|
|
SYNCMGRPROGRESSITEM spi = {0};
|
|
DWORD (WINAPI *pfnStartStop)(HANDLE);
|
|
|
|
pfnStartStop = ResumeThread;
|
|
|
|
spi.cbSize = sizeof(spi);
|
|
spi.mask = SYNCMGRPROGRESSITEM_STATUSTYPE | SYNCMGRPROGRESSITEM_STATUSTEXT;
|
|
spi.lpcStatusText = L" ";
|
|
spi.dwStatusType = SYNCMGRSTATUS_UPDATING;
|
|
if (SyncPause == status)
|
|
{
|
|
spi.dwStatusType = SYNCMGRSTATUS_PAUSED;
|
|
pfnStartStop = SuspendThread;
|
|
}
|
|
|
|
while (cItems > 0)
|
|
{
|
|
PSYNCTHREADDATA pSyncData;
|
|
|
|
--cItems;
|
|
pSyncData = (PSYNCTHREADDATA)DPA_FastGetPtr(m_hSyncThreads, cItems);
|
|
TraceAssert(NULL != pSyncData);
|
|
|
|
if (SyncStop == status)
|
|
{
|
|
// Tell the thread to abort
|
|
if (!bOneItem || IsEqualGUID(rItemID, pSyncData->ItemID))
|
|
{
|
|
pSyncData->dwSyncStatus |= SDS_SYNC_CANCELLED;
|
|
if (bOneItem)
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Suspend or resume the thread if it's not the current thread
|
|
if (!IsEqualGUID(rItemID, pSyncData->ItemID))
|
|
(*pfnStartStop)(pSyncData->hThread);
|
|
m_pSyncMgrCB->Progress(pSyncData->ItemID, &spi);
|
|
}
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(&m_csThreadList);
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
void
|
|
CCscUpdate::GetSilentFolderList(void)
|
|
{
|
|
delete m_pSilentFolderList;
|
|
m_pSilentFolderList = new CscFilenameList;
|
|
|
|
delete m_pSpecialFolderList;
|
|
m_pSpecialFolderList = new CscFilenameList;
|
|
|
|
BuildSilentFolderList(m_pSilentFolderList, m_pSpecialFolderList);
|
|
|
|
if (0 == m_pSilentFolderList->GetShareCount())
|
|
{
|
|
delete m_pSilentFolderList;
|
|
m_pSilentFolderList = NULL;
|
|
}
|
|
|
|
if (0 == m_pSpecialFolderList->GetShareCount())
|
|
{
|
|
delete m_pSpecialFolderList;
|
|
m_pSpecialFolderList = NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
BuildSilentFolderList(CscFilenameList *pfnlSilentFolders,
|
|
CscFilenameList *pfnlSpecialFolders)
|
|
{
|
|
|
|
// We will silently handle sync conflicts in any of the folders
|
|
// below that have a '1' after them.
|
|
|
|
// If we get complaints about conflicts in folders that we
|
|
// think we can handle silently and safely, add them.
|
|
|
|
// Note that CSIDL_PERSONAL (MyDocs) and CSIDL_MYPICTURES
|
|
// should probably never be silent, since the user
|
|
// interacts with them directly.
|
|
|
|
// This list corresponds to the list of shell folders that may
|
|
// be redirected. See also
|
|
// HKCU\Software\Microsoft\Windows\CurrentVersion\Explorer\Shell Folders
|
|
|
|
static const int s_csidlFolders[][2] =
|
|
{
|
|
{ CSIDL_PROGRAMS, 0 },
|
|
{ CSIDL_PERSONAL, 0 },
|
|
{ CSIDL_FAVORITES, 0 },
|
|
{ CSIDL_STARTUP, 0 },
|
|
{ CSIDL_RECENT, 1 },
|
|
{ CSIDL_SENDTO, 0 },
|
|
{ CSIDL_STARTMENU, 1 },
|
|
{ CSIDL_DESKTOPDIRECTORY, 0 },
|
|
{ CSIDL_NETHOOD, 0 },
|
|
{ CSIDL_TEMPLATES, 0 },
|
|
{ CSIDL_APPDATA, 1 },
|
|
{ CSIDL_PRINTHOOD, 0 },
|
|
{ CSIDL_MYPICTURES, 0 },
|
|
{ CSIDL_PROFILE, 1 },
|
|
{ CSIDL_ADMINTOOLS, 0 },
|
|
};
|
|
TCHAR szPath[MAX_PATH];
|
|
|
|
for (int i = 0; i < ARRAYSIZE(s_csidlFolders); i++)
|
|
{
|
|
if (SHGetSpecialFolderPath(NULL, szPath, s_csidlFolders[i][0] | CSIDL_FLAG_DONT_VERIFY, FALSE))
|
|
{
|
|
// We only want UNC net paths
|
|
LPTSTR pszUNC = NULL;
|
|
GetRemotePath(szPath, &pszUNC);
|
|
if (!pszUNC)
|
|
continue;
|
|
|
|
if (s_csidlFolders[i][1])
|
|
{
|
|
if (pfnlSilentFolders)
|
|
pfnlSilentFolders->AddFile(pszUNC, true);
|
|
}
|
|
else
|
|
{
|
|
if (pfnlSpecialFolders)
|
|
pfnlSpecialFolders->AddFile(pszUNC, true);
|
|
}
|
|
|
|
LocalFreeString(&pszUNC);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
// SyncMgr integration (ISyncMgrEnumItems) //
|
|
|
|
|
|
|
|
CUpdateEnumerator::CUpdateEnumerator(PCSCUPDATE pUpdate)
|
|
: m_cRef(1),
|
|
m_pUpdate(pUpdate),
|
|
m_hFind(INVALID_HANDLE_VALUE),
|
|
m_bEnumFileSelection(FALSE),
|
|
m_cCheckedItemsEnumerated(0)
|
|
{
|
|
DllAddRef();
|
|
|
|
if (m_pUpdate)
|
|
{
|
|
m_pUpdate->AddRef();
|
|
|
|
if (m_pUpdate->m_pFileList)
|
|
{
|
|
m_bEnumFileSelection = TRUE;
|
|
m_SelectionIterator = m_pUpdate->m_pFileList->CreateShareIterator();
|
|
}
|
|
}
|
|
}
|
|
|
|
CUpdateEnumerator::~CUpdateEnumerator()
|
|
{
|
|
if (m_hFind != INVALID_HANDLE_VALUE)
|
|
CSCFindClose(m_hFind);
|
|
|
|
DoRelease(m_pUpdate);
|
|
DllRelease();
|
|
}
|
|
|
|
|
|
STDMETHODIMP CUpdateEnumerator::QueryInterface(REFIID riid, void **ppv)
|
|
{
|
|
static const QITAB qit[] =
|
|
{
|
|
QITABENT(CUpdateEnumerator, ISyncMgrEnumItems),
|
|
{ 0 },
|
|
};
|
|
return QISearch(this, qit, riid, ppv);
|
|
}
|
|
|
|
STDMETHODIMP_(ULONG) CUpdateEnumerator::AddRef()
|
|
{
|
|
return InterlockedIncrement(&m_cRef);
|
|
}
|
|
|
|
STDMETHODIMP_(ULONG) CUpdateEnumerator::Release()
|
|
{
|
|
if (InterlockedDecrement(&m_cRef))
|
|
return m_cRef;
|
|
|
|
delete this;
|
|
return 0;
|
|
}
|
|
|
|
STDMETHODIMP
|
|
CUpdateEnumerator::Next(ULONG celt, LPSYNCMGRITEM rgelt, PULONG pceltFetched)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
ULONG cFetched = 0;
|
|
LPSYNCMGRITEM pItem = rgelt;
|
|
WIN32_FIND_DATA fd = {0};
|
|
DWORD dwShareStatus = 0;
|
|
DWORD dwSyncFlags;
|
|
CscFilenameList::HSHARE hShare;
|
|
LPCTSTR pszShareName = NULL;
|
|
|
|
TraceEnter(TRACE_UPDATE, "CUpdateEnumerator::Next");
|
|
TraceAssert(m_pUpdate != NULL);
|
|
|
|
if (NULL == rgelt)
|
|
TraceLeaveResult(E_INVALIDARG);
|
|
|
|
dwSyncFlags = m_pUpdate->m_dwSyncFlags;
|
|
|
|
while (cFetched < celt)
|
|
{
|
|
CSCEntry *pShareEntry;
|
|
CSCSHARESTATS shareStats;
|
|
CSCGETSTATSINFO si = { SSEF_NONE,
|
|
SSUF_TOTAL | SSUF_PINNED | SSUF_MODIFIED | SSUF_SPARSE | SSUF_DIRS,
|
|
false,
|
|
false };
|
|
|
|
if (m_bEnumFileSelection)
|
|
{
|
|
if (!m_SelectionIterator.Next(&hShare))
|
|
break;
|
|
|
|
pszShareName = m_pUpdate->m_pFileList->GetShareName(hShare);
|
|
|
|
CSCQueryFileStatus(pszShareName, &dwShareStatus, NULL, NULL);
|
|
fd.dwFileAttributes = FILE_ATTRIBUTE_DIRECTORY;
|
|
lstrcpyn(fd.cFileName, pszShareName, ARRAYSIZE(fd.cFileName));
|
|
}
|
|
else
|
|
{
|
|
if (m_hFind == INVALID_HANDLE_VALUE)
|
|
{
|
|
m_hFind = CacheFindFirst(NULL, &fd, &dwShareStatus, NULL, NULL, NULL);
|
|
|
|
if (m_hFind == INVALID_HANDLE_VALUE)
|
|
{
|
|
// The database is empty, so there's nothing to enumerate..
|
|
break;
|
|
}
|
|
|
|
pszShareName = fd.cFileName;
|
|
}
|
|
else if (CacheFindNext(m_hFind, &fd, &dwShareStatus, NULL, NULL, NULL))
|
|
{
|
|
pszShareName = fd.cFileName;
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
TraceAssert(pszShareName);
|
|
|
|
|
|
// BUGBUG This was proposed as a fix for part of 383011. However,
|
|
// that bug only applies in a multi-user scenario, and if there
|
|
// are more than 3 users using the machine, this would cause a
|
|
// user whose SID had been expelled from the CSC database to not
|
|
// be able to sync a share where they do indeed have access.
|
|
|
|
// // If the current user has no access to the share, don't enumerate it.
|
|
// if (!(CscAccessUser(dwShareStatus) || CscAccessGuest(dwShareStatus)))
|
|
// continue;
|
|
|
|
// Count the # of pinned files, sparse files, etc.
|
|
_GetShareStatisticsForUser(pszShareName, &si, &shareStats);
|
|
|
|
// The root of a special folder is pinned with a pin count, but
|
|
// not the user-hint flag, so _GetShareStats doesn't count it.
|
|
// We need to count this for some of the checks below.
|
|
// (If the share is manual-cache, these look exactly like the
|
|
// Siemens scenario in 341786, but we want to sync them.)
|
|
if (shareStats.cTotal && m_pUpdate->IsSpecialFolderShare(pszShareName))
|
|
{
|
|
shareStats.cPinned++;
|
|
}
|
|
|
|
// If we're pinning, then even if nothing is sparse now,
|
|
// there will be sparse files after we pin them.
|
|
if (dwSyncFlags & CSC_SYNC_PINFILES)
|
|
{
|
|
shareStats.cSparse++;
|
|
shareStats.cTotal++;
|
|
}
|
|
|
|
// If there's nothing cached on this share, then don't even
|
|
// enumerate it to SyncMgr. This avoids listing extra junk
|
|
// in SyncMgr.
|
|
if ((0 == shareStats.cTotal) ||
|
|
(shareStats.cTotal == shareStats.cDirs && 0 == shareStats.cPinned))
|
|
{
|
|
// Either there is nothing cached for this share, or the only
|
|
// things found were unpinned dirs (no files, no pinned dirs).
|
|
// The second case can happen if you delete files from the viewer,
|
|
// in which case you think you deleted everything but the viewer
|
|
// doesn't show directories, so they weren't deleted.
|
|
Trace((TEXT("Nothing cached on %s, not enumerating"), pszShareName));
|
|
continue;
|
|
}
|
|
|
|
if ((dwShareStatus & FLAG_CSC_SHARE_STATUS_CACHING_MASK) == FLAG_CSC_SHARE_STATUS_NO_CACHING)
|
|
{
|
|
|
|
// Don't enumerate "no-cache" shares if there's nothing to merge.
|
|
|
|
// These can exist in the cache if the share was previously
|
|
// cacheable, but has since been changed to "no caching".
|
|
|
|
// If there's something to merge, we should still sync it to
|
|
// get everything squared away.
|
|
|
|
if (!((dwSyncFlags & CSC_SYNC_OUT) && (shareStats.cModified)))
|
|
{
|
|
Trace((TEXT("Not enumerating no-cache share %s"), pszShareName));
|
|
continue;
|
|
}
|
|
Trace((TEXT("Enumerating no-cache share %s with offline changes."), pszShareName));
|
|
}
|
|
|
|
pItem->dwFlags = SYNCMGRITEM_HASPROPERTIES;
|
|
|
|
if ((dwShareStatus & FLAG_CSC_SHARE_STATUS_CACHING_MASK) == FLAG_CSC_SHARE_STATUS_MANUAL_REINT
|
|
&& 0 == shareStats.cPinned
|
|
&& !(dwSyncFlags & CSC_SYNC_PINFILES))
|
|
{
|
|
// On a manual share, if nothing is pinned (and we aren't pinning)
|
|
// then we don't want to call CSCFillSparseFiles on the share.
|
|
// Raid #341786
|
|
Trace((TEXT("Manual cache share '%s' has only autocached files"), pszShareName));
|
|
|
|
// However, if there is something to merge, then we need to sync.
|
|
if (!((dwSyncFlags & CSC_SYNC_OUT) && shareStats.cModified))
|
|
{
|
|
Trace((TEXT("Not enumerating manual-cache share %s"), pszShareName));
|
|
continue;
|
|
}
|
|
|
|
// There is something to merge, so enumerate the share but
|
|
// tell SyncMgr that it's temporary so it doesn't save state
|
|
// for this share.
|
|
pItem->dwFlags |= SYNCMGRITEM_TEMPORARY;
|
|
}
|
|
|
|
|
|
// In some circumstances, we may want to merge even if there
|
|
// are no modified files, in order to show the open files warning.
|
|
|
|
// See comments in CCscUpdate::_SyncThread
|
|
|
|
if (0 == shareStats.cModified)
|
|
{
|
|
if ((FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP & dwShareStatus) &&
|
|
(FLAG_CSC_SHARE_STATUS_FILES_OPEN & dwShareStatus))
|
|
{
|
|
shareStats.cModified++;
|
|
}
|
|
}
|
|
|
|
// Enumerate this share
|
|
cFetched++;
|
|
|
|
// Get existing share entry or create a new one
|
|
pShareEntry = m_pUpdate->m_ShareLog.Add(pszShareName);
|
|
if (!pShareEntry)
|
|
TraceLeaveResult(E_OUTOFMEMORY);
|
|
|
|
pItem->cbSize = sizeof(SYNCMGRITEM);
|
|
pItem->ItemID = pShareEntry->Guid();
|
|
pItem->hIcon = g_hCscIcon;
|
|
// BUGBUG SYNCMGRITEM_TEMPORARY causes items to not show up in
|
|
// SyncMgr's logon/logoff settings page. Raid #237288
|
|
//if (0 == shareStats.cPinned)
|
|
// pItem->dwFlags |= SYNCMGRITEM_TEMPORARY;
|
|
if (ERROR_SUCCESS == m_pUpdate->GetLastSyncTime(pszShareName, &pItem->ftLastUpdate))
|
|
pItem->dwFlags |= SYNCMGRITEM_LASTUPDATETIME;
|
|
|
|
|
|
// Determine whether this share needs syncing.
|
|
|
|
// At settings time, assume everything needs syncing (check everything)
|
|
|
|
// If outbound, shares with modified files are checked
|
|
// If inbound (full), shares with sparse or pinned files are checked
|
|
// If inbound (quick), shares with sparse files are checked
|
|
|
|
// Anything else doesn't need to be sync'ed at this time (unchecked)
|
|
|
|
pItem->dwItemState = SYNCMGRITEMSTATE_CHECKED;
|
|
if (!(dwSyncFlags & CSC_SYNC_SETTINGS) &&
|
|
!((dwSyncFlags & CSC_SYNC_OUT) && shareStats.cModified) &&
|
|
!((dwSyncFlags & CSC_SYNC_IN_FULL) && shareStats.cTotal ) &&
|
|
!((dwSyncFlags & CSC_SYNC_IN_QUICK) && shareStats.cSparse ))
|
|
{
|
|
pItem->dwItemState = SYNCMGRITEMSTATE_UNCHECKED;
|
|
}
|
|
|
|
// Get friendly share name here
|
|
LPITEMIDLIST pidl = NULL;
|
|
SHFILEINFO sfi = {0};
|
|
if (SUCCEEDED(SHSimpleIDListFromFindData(pszShareName, &fd, &pidl)))
|
|
{
|
|
SHGetFileInfo((LPCTSTR)pidl,
|
|
0,
|
|
&sfi,
|
|
sizeof(sfi),
|
|
SHGFI_PIDL | SHGFI_DISPLAYNAME);
|
|
SHFree(pidl);
|
|
}
|
|
if (TEXT('\0') != sfi.szDisplayName[0])
|
|
pszShareName = sfi.szDisplayName;
|
|
|
|
SHTCharToUnicode((LPTSTR)pszShareName, pItem->wszItemName, ARRAYSIZE(pItem->wszItemName));
|
|
if (SYNCMGRITEMSTATE_CHECKED == pItem->dwItemState)
|
|
{
|
|
m_cCheckedItemsEnumerated++;
|
|
Trace((TEXT("Enumerating %s, checked"), pszShareName));
|
|
}
|
|
else
|
|
{
|
|
Trace((TEXT("Enumerating %s, unchecked"), pszShareName));
|
|
}
|
|
|
|
pItem++;
|
|
}
|
|
|
|
|
|
if (pceltFetched)
|
|
*pceltFetched = cFetched;
|
|
|
|
if (cFetched != celt)
|
|
hr = S_FALSE;
|
|
|
|
if ((S_FALSE == hr) &&
|
|
0 == m_cCheckedItemsEnumerated &&
|
|
(CSC_SYNC_SHOWUI_ALWAYS & dwSyncFlags))
|
|
{
|
|
|
|
// Special-case where we're synching nothing but still
|
|
// want to display SyncMgr progress UI. We enumerate a
|
|
// special string rather than a share name for display in
|
|
// the status UI. Force hr == S_OK so the caller will accept
|
|
// this "dummy" item. Next() will be called once more but
|
|
// m_cCheckedItemsEnumerated will be 1 so this block won't be
|
|
// entered and we'll return S_FALSE indicating the end of the
|
|
// enumeration.
|
|
|
|
pItem->cbSize = sizeof(SYNCMGRITEM);
|
|
pItem->hIcon = g_hCscIcon;
|
|
pItem->dwFlags = SYNCMGRITEM_HASPROPERTIES;
|
|
pItem->dwItemState = SYNCMGRITEMSTATE_CHECKED;
|
|
pItem->ItemID = GUID_CscNullSyncItem;
|
|
|
|
UINT idString = IDS_NULLSYNC_ITEMNAME;
|
|
if ((CSC_SYNC_OUT & dwSyncFlags) &&
|
|
!((CSC_SYNC_IN_QUICK | CSC_SYNC_IN_FULL) & dwSyncFlags))
|
|
{
|
|
// Use different text if we are only merging
|
|
idString = IDS_NULLMERGE_ITEMNAME;
|
|
}
|
|
|
|
LoadStringW(g_hInstance,
|
|
idString,
|
|
pItem->wszItemName,
|
|
ARRAYSIZE(pItem->wszItemName));
|
|
m_cCheckedItemsEnumerated = 1;
|
|
|
|
TraceMsg("Enumerating NULL item");
|
|
hr = S_OK;
|
|
}
|
|
|
|
TraceLeaveResult(hr);
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CUpdateEnumerator::Skip(ULONG celt)
|
|
{
|
|
return Next(celt, NULL, NULL);
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CUpdateEnumerator::Reset()
|
|
{
|
|
m_cCheckedItemsEnumerated = 0;
|
|
if (m_bEnumFileSelection)
|
|
{
|
|
m_SelectionIterator.Reset();
|
|
}
|
|
else if (m_hFind != INVALID_HANDLE_VALUE)
|
|
{
|
|
CSCFindClose(m_hFind);
|
|
m_hFind = INVALID_HANDLE_VALUE;
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
STDMETHODIMP
|
|
CUpdateEnumerator::Clone(LPSYNCMGRENUMITEMS *ppenum)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|