mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2877 lines
80 KiB
2877 lines
80 KiB
//+---------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1996 - 2002.
|
|
//
|
|
// File: docstore.cxx
|
|
//
|
|
// Contents: Deals with the client side document store implementation.
|
|
//
|
|
// Classes: CClientDocStore
|
|
//
|
|
// History: 12-03-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
#include <pch.cxx>
|
|
#pragma hdrstop
|
|
|
|
// for definition of CRequestQueue
|
|
#include <query.hxx>
|
|
#include <srequest.hxx>
|
|
#include <regacc.hxx>
|
|
|
|
#include <docstore.hxx>
|
|
#include <docname.hxx>
|
|
#include <qsession.hxx>
|
|
#include <cicat.hxx>
|
|
#include <cinulcat.hxx>
|
|
#include <seccache.hxx>
|
|
#include <dmnstart.hxx>
|
|
#include <propmap.hxx>
|
|
#include <notifyev.hxx>
|
|
#include <catalog.hxx>
|
|
#include <catarray.hxx>
|
|
#include <glbconst.hxx>
|
|
#include <cisvcex.hxx>
|
|
#include <cisvcfrm.hxx>
|
|
#include <cistore.hxx>
|
|
#include <enumstr.hxx>
|
|
#include <filterob.hxx>
|
|
#include <catadmin.hxx>
|
|
|
|
#include <fsci.hxx>
|
|
|
|
extern CCatArray Catalogs;
|
|
|
|
CRequestQueue * g_pFSCIRequestQueue = 0;
|
|
|
|
extern "C" const GUID clsidStorageFilterObject =
|
|
{ 0xaa205a4d, 0x681f, 0x11d0, { 0xa2, 0x43, 0x8, 0x0, 0x2b, 0x36, 0xfc, 0xa4 } };
|
|
|
|
extern "C" const GUID guidStorageDocStoreLocatorObject;
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::QueryInterface
|
|
//
|
|
// Synopsis: Supports IID_IUnknown
|
|
// IID_ICiCDocStore
|
|
// IID_ICiCDocStoreEx
|
|
// IID_ICiCPropertyStorage
|
|
// IID_ICiCDocNameToWorkidTranslator
|
|
// IID_ICiCDocNameToWorkidTranslatorEx
|
|
// IID_ICiCAdviseStatus
|
|
// IID_IFsCiAdmin
|
|
// IID_ICiCLangRes
|
|
//
|
|
// History: 12-03-96 srikants Created
|
|
// 2-14-97 mohamedn ICiCLangRes
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
STDMETHODIMP CClientDocStore::QueryInterface( REFIID riid, void **ppvObject )
|
|
{
|
|
Win4Assert( 0 != ppvObject );
|
|
|
|
if ( IID_ICiCDocStore == riid )
|
|
*ppvObject = (void *)((ICiCDocStore *)this);
|
|
else if ( IID_ICiCPropertyStorage == riid )
|
|
*ppvObject = (void *)((ICiCPropertyStorage *)this);
|
|
else if ( IID_ICiCDocNameToWorkidTranslator == riid )
|
|
*ppvObject = (void *)((ICiCDocNameToWorkidTranslator *)this);
|
|
else if ( IID_ICiCDocNameToWorkidTranslatorEx == riid )
|
|
*ppvObject = (void *)((ICiCDocNameToWorkidTranslatorEx *)this);
|
|
else if ( IID_ICiCAdviseStatus == riid )
|
|
*ppvObject = (void *)((ICiCAdviseStatus *)this);
|
|
else if ( IID_IFsCiAdmin == riid )
|
|
*ppvObject = (void *)((IFsCiAdmin *)this);
|
|
else if ( IID_ICiCLangRes == riid )
|
|
*ppvObject = (void *) ((ICiCLangRes *)this);
|
|
else if ( IID_ICiCDocStoreEx == riid )
|
|
*ppvObject = (void *)((IUnknown *) (ICiCDocStoreEx *)this);
|
|
else if ( IID_ICiCResourceMonitor == riid )
|
|
*ppvObject = (void *)((IUnknown *) (ICiCResourceMonitor *)this);
|
|
else if ( IID_IUnknown == riid )
|
|
*ppvObject = (void *)((IUnknown *) (ICiCDocStore *)this);
|
|
else
|
|
{
|
|
*ppvObject = 0;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
AddRef();
|
|
return S_OK;
|
|
} //QueryInterface
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::AddRef
|
|
//
|
|
// History: 12-03-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP_(ULONG) CClientDocStore::AddRef()
|
|
{
|
|
return InterlockedIncrement(&_refCount);
|
|
} //AddRef
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::Release
|
|
//
|
|
// History: 12-03-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP_(ULONG) CClientDocStore::Release()
|
|
{
|
|
Win4Assert( _refCount > 0 );
|
|
|
|
LONG refCount = InterlockedDecrement(&_refCount);
|
|
|
|
if ( refCount <= 0 )
|
|
delete this;
|
|
|
|
return (ULONG) refCount;
|
|
} //Release
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::CClientDocStore
|
|
//
|
|
// Synopsis: Constructor of the NULL CiCDocStore object.
|
|
//
|
|
// Arguments: None
|
|
//
|
|
// History: Jul-09-97 KrishnaN Created
|
|
// 01-Nov-98 KLam Need to instantiate a CDiskFreeStatus
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
CClientDocStore::CClientDocStore()
|
|
: _sigClientDocStore(eSigClientDocStore),
|
|
_refCount(1),
|
|
_fNoQuery( FALSE ),
|
|
_state(eUpdatesDisabled),
|
|
_pCiCat(0),
|
|
_pCiNullCat(0)
|
|
{
|
|
// Create the CiNullCat object.
|
|
_pCiNullCat = new CiNullCat( *this );
|
|
XPtr<CiNullCat> xCat( _pCiNullCat );
|
|
|
|
// Map std props only (second param)
|
|
_xPropMapper.Set( new CFwPropertyMapper( _pCiNullCat->GetPidLookupTable(), TRUE ) );
|
|
|
|
// Create CI Manager object.
|
|
_CreateCiManager();
|
|
|
|
_pCiNullCat->StartupCiFrameWork( _xCiManager.GetPointer() );
|
|
|
|
//
|
|
// Startup content index.
|
|
//
|
|
ICiStartup * pCiStartup;
|
|
SCODE sc = _xCiManager->QueryInterface( IID_ICiStartup, (void **) &pCiStartup );
|
|
XInterface<ICiStartup> xStartup( pCiStartup );
|
|
if ( S_OK != sc )
|
|
{
|
|
THROW( CException(sc) );
|
|
}
|
|
|
|
CI_STARTUP_INFO startupInfo;
|
|
RtlZeroMemory( &startupInfo, sizeof(startupInfo) );
|
|
|
|
startupInfo.clsidDaemonClientMgr = clsidStorageFilterObject;
|
|
startupInfo.startupFlags = CI_CONFIG_ENABLE_QUERYING ;
|
|
|
|
BOOL fAbort = FALSE;
|
|
|
|
sc = pCiStartup->StartupNullContentIndex( &startupInfo,0, &fAbort );
|
|
|
|
if ( FAILED(sc) )
|
|
{
|
|
THROW( CException(sc) );
|
|
}
|
|
|
|
xCat->SetAdviseStatus();
|
|
xCat.Acquire();
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::CClientDocStore
|
|
//
|
|
// Synopsis: Constructor of the CiCDocStore object.
|
|
//
|
|
// Arguments: [pwszPath] - Path of the directory where the files must be
|
|
// created.
|
|
// [pwszName] - Name of the Content Index.
|
|
//
|
|
// History: 12-03-96 srikants Created
|
|
// 02-17-98 kitmanh Added code to deal with read-only
|
|
// catalogs (the readOnly flag is
|
|
// passed down from CiCat to startupInfo)
|
|
// 07-Jan-99 klam Logged and event that initialization failed
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
CClientDocStore::CClientDocStore( WCHAR const * pwszPath,
|
|
BOOL fOpenForReadOnly,
|
|
CDrvNotifArray & DrvNotifArray,
|
|
WCHAR const * pwszName )
|
|
: _sigClientDocStore(eSigClientDocStore),
|
|
_refCount(1),
|
|
_state(eUpdatesDisabled),
|
|
_pCiCat(0),
|
|
_pCiNullCat(0),
|
|
_fNoQuery( FALSE ),
|
|
_pDrvNotifArray( &DrvNotifArray )
|
|
{
|
|
ciDebugOut(( DEB_ITRACE, "CClientDocStore::CClinetDocStore.. fOpenForReadOnly == %d\n", fOpenForReadOnly ));
|
|
ciDebugOut(( DEB_ITRACE, "CClientDocStore::CClinetDocStore.. pwszPath == %ws\n", pwszPath ));
|
|
ciDebugOut(( DEB_ITRACE, "CClientDocStore::CClinetDocStore.. pwszName == %ws\n", pwszName ));
|
|
|
|
// Check if the catalog directory exists
|
|
|
|
if ( ( wcslen( pwszPath ) + wcslen( CAT_DIR ) ) >= MAX_PATH )
|
|
THROW( CException( STATUS_INVALID_PARAMETER ) );
|
|
|
|
WCHAR awcCatDir[ MAX_PATH ];
|
|
wcscpy( awcCatDir, pwszPath );
|
|
wcscat( awcCatDir, CAT_DIR );
|
|
|
|
WIN32_FILE_ATTRIBUTE_DATA fData;
|
|
if ( GetFileAttributesEx( awcCatDir, GetFileExInfoStandard, &fData ) )
|
|
{
|
|
// Is the catalog path a file or a directory?
|
|
|
|
if ( 0 == ( fData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY ) )
|
|
THROW( CException( HRESULT_FROM_WIN32( ERROR_PATH_NOT_FOUND ) ) );
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// You can get back both errors depending on what parts of the
|
|
// directory tree currently exist.
|
|
//
|
|
|
|
if ( ( ERROR_FILE_NOT_FOUND == GetLastError() ) ||
|
|
( ERROR_PATH_NOT_FOUND == GetLastError() ) )
|
|
{
|
|
// create the catalog directory with proper acls
|
|
|
|
CMachineAdmin admin;
|
|
admin.CreateSubdirs( pwszPath );
|
|
}
|
|
else
|
|
THROW( CException() );
|
|
}
|
|
|
|
BOOL fLeaveCorruptCatalog;
|
|
{
|
|
// By default, delete corrupt catalogs.
|
|
|
|
CRegAccess reg( RTL_REGISTRY_CONTROL, wcsRegAdmin );
|
|
fLeaveCorruptCatalog = reg.Read( wcsLeaveCorruptCatalog, (ULONG) FALSE );
|
|
}
|
|
|
|
//
|
|
// Create the CiCat object.
|
|
//
|
|
BOOL fVersionChange = FALSE;
|
|
|
|
_pCiCat = new CiCat( *this,
|
|
_workMan,
|
|
pwszPath,
|
|
fVersionChange,
|
|
fOpenForReadOnly,
|
|
*_pDrvNotifArray,
|
|
pwszName,
|
|
fLeaveCorruptCatalog );
|
|
|
|
XPtr<CiCat> xCat( _pCiCat );
|
|
|
|
// Create a CDiskFreeStatus object
|
|
_xDiskStatus.Set( new CDiskFreeStatus( pwszPath,
|
|
xCat->GetRegParams()->GetMinDiskSpaceToLeave() ) );
|
|
|
|
// Map std props AND others (second param set to false)
|
|
_xPropMapper.Set( new CFwPropertyMapper( xCat->GetPidLookupTable(), FALSE ) );
|
|
|
|
//
|
|
// Create the perform counters object. This must be done before
|
|
// starting up rest of the content index.
|
|
//
|
|
Win4Assert( 0 != xCat->GetCatalogName() );
|
|
CPerfMon * pPerfMon = new CPerfMon( xCat->GetCatalogName() );
|
|
|
|
_xPerfMon.Set( pPerfMon );
|
|
|
|
//
|
|
// Create CI Manager object.
|
|
//
|
|
_CreateCiManager();
|
|
|
|
xCat->StartupCiFrameWork( _xCiManager.GetPointer() );
|
|
|
|
//
|
|
// Startup content index.
|
|
//
|
|
|
|
ICiStartup * pCiStartup;
|
|
SCODE sc = _xCiManager->QueryInterface( IID_ICiStartup,
|
|
(void **) &pCiStartup );
|
|
XInterface<ICiStartup> xStartup( pCiStartup );
|
|
if ( S_OK != sc )
|
|
THROW( CException(sc) );
|
|
|
|
BOOL fFullScanNeeded = FALSE;
|
|
|
|
CI_STARTUP_INFO startupInfo;
|
|
RtlZeroMemory( &startupInfo, sizeof(startupInfo) );
|
|
|
|
startupInfo.clsidDaemonClientMgr = clsidStorageFilterObject;
|
|
startupInfo.startupFlags = CI_CONFIG_ENABLE_INDEXING |
|
|
CI_CONFIG_ENABLE_QUERYING ;
|
|
|
|
if ( xCat->IsReadOnly() )
|
|
startupInfo.startupFlags |= CI_CONFIG_READONLY;
|
|
|
|
if ( xCat->IsCiDataCorrupt() || xCat->IsFsCiDataCorrupt() || fVersionChange )
|
|
{
|
|
if ( fLeaveCorruptCatalog )
|
|
{
|
|
Win4Assert( !"leaving corrupt catalog" );
|
|
THROW( CException( CI_CORRUPT_CATALOG ) );
|
|
}
|
|
|
|
if ( xCat->IsCiDataCorrupt() ||
|
|
xCat->IsFsCiDataCorrupt() )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Persistent CI/FSCI Data Corruption. It will be emptied \n" ));
|
|
//Win4Assert( !"Persistent CI/FSCI Data Corruption" );
|
|
}
|
|
|
|
startupInfo.startupFlags |= (ULONG) CI_CONFIG_EMPTY_DATA;
|
|
xCat->LogEvent( CCiStatusMonitor::eCiRemoved );
|
|
fFullScanNeeded = TRUE;
|
|
}
|
|
|
|
BOOL fAbort = FALSE;
|
|
|
|
#if CIDBG==1
|
|
#if 0
|
|
_FillLoadFilesInfo( startupInfo );
|
|
#endif // 0
|
|
#endif // CIDBG==1
|
|
|
|
sc = pCiStartup->StartupContentIndex( xCat->GetCatDir(), &startupInfo,0, &fAbort );
|
|
|
|
#if CIDBG==1
|
|
#if 0
|
|
if ( SUCCEEDED(sc) )
|
|
{
|
|
WCHAR wszBackupPath[MAX_PATH];
|
|
wcscpy( wszBackupPath, xCat->GetCatDir() );
|
|
wcscat( wszBackupPath, L"\\backup");
|
|
ICiPersistIncrFile * pIPersist = 0;
|
|
sc = _xCiManager->QueryInterface( IID_ICiPersistIncrFile,
|
|
(void **) &pIPersist );
|
|
Win4Assert( SUCCEEDED(sc) );
|
|
_xSaveTest.Set( new CCiSaveTest( wszBackupPath,
|
|
pIPersist,
|
|
xCat.GetReference() ) );
|
|
pIPersist->Release();
|
|
}
|
|
_ClearLoadFilesInfo( startupInfo );
|
|
#endif // 0
|
|
#endif // CIDBG==1
|
|
|
|
|
|
if ( FAILED(sc) )
|
|
{
|
|
if ( !IsCiCorruptStatus( sc ) && sc != CI_INCORRECT_VERSION )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Failed to startupci. Error 0x%X\n", sc ));
|
|
xCat->LogEvent ( CCiStatusMonitor::eCiError, sc );
|
|
THROW( CException(sc) );
|
|
}
|
|
|
|
if ( fLeaveCorruptCatalog )
|
|
{
|
|
Win4Assert( !"leaving corrupt catalog" );
|
|
THROW( CException( sc ) );
|
|
}
|
|
|
|
ciDebugOut(( DEB_ERROR, "ContentIndex is corrupt. It will be emptied\n" ));
|
|
//Win4Assert( !"Startup CI Data Corruption" );
|
|
xCat->LogEvent( CCiStatusMonitor::eCiRemoved );
|
|
|
|
//
|
|
// Content Index is corrupt. Ask CI to delete the contentIndex and
|
|
// start afresh.
|
|
//
|
|
startupInfo.startupFlags |= CI_CONFIG_EMPTY_DATA;
|
|
sc = pCiStartup->StartupContentIndex( xCat->GetCatDir(),
|
|
&startupInfo,
|
|
0,
|
|
&fAbort );
|
|
|
|
if ( FAILED(sc) )
|
|
{
|
|
THROW( CException(sc) );
|
|
}
|
|
|
|
fFullScanNeeded = TRUE;
|
|
}
|
|
|
|
xCat->ClearCiDataCorrupt();
|
|
|
|
xCat->InitIf( fLeaveCorruptCatalog );
|
|
|
|
//
|
|
// Optimization - we may want to just force an "add" of the documents in the
|
|
// property store rather than a full scan of the corpus.
|
|
//
|
|
if ( fFullScanNeeded )
|
|
xCat->MarkFullScanNeeded();
|
|
|
|
xCat.Acquire();
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::_CreateCiManager
|
|
//
|
|
// Synopsis: Creates the CI manager by doing a CoCreateInstance of the
|
|
// ICiControl.
|
|
//
|
|
// History: 1-31-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
void CClientDocStore::_CreateCiManager()
|
|
{
|
|
//
|
|
// We have to create the ICiManager also now.
|
|
//
|
|
ICiControl * pICiControl = 0;
|
|
GUID clsIdCiControl = CLSID_CiControl;
|
|
SCODE sc = CoCreateInstance( clsIdCiControl,
|
|
NULL,
|
|
CLSCTX_INPROC_SERVER,
|
|
IID_ICiControl,
|
|
(void **) &pICiControl );
|
|
|
|
if ( 0 == pICiControl )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Cannot CoCreateInstance IID_ICiControl. Error (0x%X)\n",
|
|
sc ));
|
|
THROW( CException(sc) );
|
|
}
|
|
|
|
XInterface<ICiControl> xCiControl( pICiControl );
|
|
|
|
ICiManager * pICiManager = 0;
|
|
|
|
sc = xCiControl->CreateContentIndex( this, &pICiManager );
|
|
if ( 0 == pICiManager )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Cannot Get ContentIndex. Error 0x%X\n",
|
|
sc ));
|
|
|
|
THROW( CException(sc) );
|
|
}
|
|
|
|
_xCiManager.Set( pICiManager );
|
|
} //_CreateCiManager
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::~CClientDocStore
|
|
//
|
|
// Synopsis: Destructor of the client document store.
|
|
//
|
|
// History: 12-03-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
CClientDocStore::~CClientDocStore()
|
|
{
|
|
delete _pCiCat;
|
|
delete _pCiNullCat;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::IsPropertyCached
|
|
//
|
|
// Synopsis: Tests if the given property is cached in the property
|
|
// store or not.
|
|
//
|
|
// Arguments: [pPropSpec] - Property to test.
|
|
// [pfValue] - Set to TRUE if cached; FALSE o/w
|
|
//
|
|
// Returns: S_OK if successful;
|
|
// Some other error code if not in a valid state.
|
|
//
|
|
// History: 12-03-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
STDMETHODIMP CClientDocStore::IsPropertyCached(
|
|
const FULLPROPSPEC * pPropSpec,
|
|
BOOL * pfValue )
|
|
{
|
|
Win4Assert( 0 != pfValue );
|
|
Win4Assert( 0 != pPropSpec );
|
|
|
|
Win4Assert( 0 != _pCiCat );
|
|
|
|
|
|
CFullPropSpec const & ps = *((CFullPropSpec const *) pPropSpec);
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
*pfValue = _pCiCat->IsPropertyCached( ps );
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: StoreProperty
|
|
//
|
|
// Synopsis: Stores the given property for the workid.
|
|
//
|
|
// Arguments: [workid] - WorkId of the document
|
|
// [pPropSpec] - Property to be stored
|
|
// [pPropVariant] - Value of the property
|
|
//
|
|
// Returns: S_OK if successful
|
|
// CI_E_PROPERTY_NOT_CACHED if it is not a cached property.
|
|
// Other error code
|
|
//
|
|
// History: 12-03-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::StoreProperty(
|
|
WORKID workid,
|
|
const FULLPROPSPEC * pPropSpec,
|
|
const PROPVARIANT * pPropVariant )
|
|
{
|
|
Win4Assert( 0 != pPropSpec );
|
|
Win4Assert( 0 != pPropVariant );
|
|
|
|
Win4Assert( 0 != _pCiCat );
|
|
|
|
CFullPropSpec const & ps = *((CFullPropSpec const *) pPropSpec);
|
|
CStorageVariant const & var = *(ConvertToStgVariant( pPropVariant ));
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
BOOL fStored = _pCiCat->StoreValue( workid, ps, var );
|
|
if ( !fStored )
|
|
sc = CI_E_PROPERTY_NOT_CACHED;
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::FlushPropertyStore
|
|
//
|
|
// Synopsis: Causes the property store to flush.
|
|
//
|
|
// History: 06-30-97 KrishnaN Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
STDMETHODIMP CClientDocStore::FlushPropertyStore (void)
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
Win4Assert(_pCiCat);
|
|
|
|
TRY
|
|
{
|
|
//
|
|
// Flushing the property store at this point (just before a shadow
|
|
// merge) is necessary for Push model filtering, but not for clients
|
|
// like fsci that do Pull model filtering. The changelog,
|
|
// scopetable, and property store are already tightly linked and
|
|
// flushed appropriately.
|
|
//
|
|
|
|
// _pCiCat->FlushPropertyStore();
|
|
}
|
|
CATCH( CException, e)
|
|
{
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::GetClientStatus
|
|
//
|
|
// Synopsis: Retrieves the client status information
|
|
//
|
|
// Arguments: [pStatus] - Will have the status information on output.
|
|
//
|
|
// History: 12-05-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
STDMETHODIMP CClientDocStore::GetClientStatus(
|
|
CI_CLIENT_STATUS * pStatus )
|
|
{
|
|
Win4Assert( 0 != pStatus );
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
ULONG cPendingScans, state;
|
|
if (_pCiNullCat)
|
|
_pCiNullCat->CatalogState( pStatus->cDocuments,
|
|
cPendingScans,
|
|
state );
|
|
else
|
|
_pCiCat->CatalogState( pStatus->cDocuments,
|
|
cPendingScans,
|
|
state );
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::GetContentIndex
|
|
//
|
|
// Synopsis: Returns the ICiManager if there is one and we are not
|
|
// in a shutdown sequence.
|
|
//
|
|
// Arguments: [ppICiManager] - ICiManager pointer
|
|
//
|
|
//
|
|
// Returns: S_OK if successful
|
|
// CI_E_SHUTDOWN if shutdown
|
|
// CI_E_NOT_INITIALIZED if not yet initialized
|
|
//
|
|
// History: 12-10-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
STDMETHODIMP CClientDocStore::GetContentIndex(
|
|
ICiManager ** ppICiManager)
|
|
{
|
|
Win4Assert( 0 != ppICiManager );
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
CLock lock(_mutex);
|
|
|
|
if ( !_IsShutdown() && _xCiManager.GetPointer() )
|
|
{
|
|
_xCiManager->AddRef();
|
|
*ppICiManager = _xCiManager.GetPointer();
|
|
}
|
|
else
|
|
{
|
|
*ppICiManager = 0;
|
|
if ( _IsShutdown() )
|
|
sc = CI_E_SHUTDOWN;
|
|
else sc = CI_E_NOT_INITIALIZED;
|
|
}
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::EnableUpdates
|
|
//
|
|
// Synopsis: Enables updates from document store.
|
|
//
|
|
// History: 12-09-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
STDMETHODIMP CClientDocStore::EnableUpdates()
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
Win4Assert(_pCiCat);
|
|
|
|
BOOL fEnableUpdateNotifies = FALSE;
|
|
|
|
TRY
|
|
{
|
|
// =============================================================
|
|
{
|
|
CLock lock(_mutex);
|
|
if ( _IsShutdown() )
|
|
{
|
|
ciDebugOut(( DEB_ERROR,
|
|
"CClientDocStore::EnableUpdates called after shutdown\n" ));
|
|
THROW( CException( CI_E_SHUTDOWN ) );
|
|
}
|
|
|
|
fEnableUpdateNotifies = _AreUpdatesDisabled();
|
|
_state = eUpdatesEnabled;
|
|
}
|
|
// =============================================================
|
|
|
|
//
|
|
// Notifications are not disabled on DisableUpdates and so only scans/usns
|
|
// need to be scheduled, which is done by NoLokClearDiskFull
|
|
//
|
|
|
|
if ( fEnableUpdateNotifies )
|
|
_pCiCat->NoLokClearDiskFull();
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR,
|
|
"CClientDocStore::EnableUpdates caught error (0x%X)\n",
|
|
e.GetErrorCode() ));
|
|
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
if ( S_OK != sc && fEnableUpdateNotifies )
|
|
{
|
|
//
|
|
// There was a failure while enabling udpates. We must set
|
|
// the state back to indicate that updates are not enabled.
|
|
//
|
|
CLock lock(_mutex);
|
|
_LokDisableUpdates();
|
|
}
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::DisableUpdates
|
|
//
|
|
// Synopsis: Disables further updates and prevents update notifications
|
|
// until enabled via the "EnableUpdates" call.
|
|
//
|
|
// History: 12-31-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
CClientDocStore::DisableUpdates( BOOL fIncremental,
|
|
CI_DISABLE_UPDATE_REASON dwReason )
|
|
{
|
|
Win4Assert(_pCiCat);
|
|
|
|
if ( _IsShutdown() )
|
|
return CI_E_SHUTDOWN;
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
{
|
|
CLock lock(_mutex);
|
|
_LokDisableUpdates();
|
|
}
|
|
|
|
if ( fIncremental )
|
|
{
|
|
_pCiCat->MarkIncrScanNeeded();
|
|
_pCiCat->NoLokProcessDiskFull();
|
|
}
|
|
else
|
|
{
|
|
if ( dwReason == CI_CORRUPT_INDEX )
|
|
_pCiCat->HandleError( CI_CORRUPT_DATABASE );
|
|
else
|
|
{
|
|
_pCiCat->MarkFullScanNeeded();
|
|
_pCiCat->NoLokProcessDiskFull();
|
|
}
|
|
}
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
sc = e.GetErrorCode();
|
|
ciDebugOut(( DEB_ERROR,
|
|
"Error (0x%X) while disabling updates\n",
|
|
sc ));
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::ProcessCiDaemonTermination
|
|
//
|
|
// Synopsis: Processes the death of CiDaemon. Creates a work item to
|
|
// restart the filter daemon.
|
|
//
|
|
// History: 12-23-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
STDMETHODIMP CClientDocStore::ProcessCiDaemonTermination( DWORD dwStatus )
|
|
{
|
|
//
|
|
// The CiDaemon process is dead; Just start filtering again.
|
|
//
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
CStartFilterDaemon * pWorkItem = new CStartFilterDaemon( *this, _workMan );
|
|
_workMan.AddToWorkList( pWorkItem );
|
|
|
|
pWorkItem->AddToWorkQueue();
|
|
pWorkItem->Release();
|
|
}
|
|
CATCH( CException, e)
|
|
{
|
|
ciDebugOut(( DEB_ERROR,
|
|
"Failed to create a work item for start filter daemon. Error 0x%X\n",
|
|
e.GetErrorCode() ));
|
|
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::CheckPointChangesFlushed
|
|
//
|
|
// Synopsis: Processes a changelog flush.
|
|
//
|
|
// History: 12-23-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::CheckPointChangesFlushed(
|
|
FILETIME ftFlushed,
|
|
ULONG cEntries,
|
|
USN_FLUSH_INFO const * const *ppUsnEntries)
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
Win4Assert(_pCiCat);
|
|
|
|
TRY
|
|
{
|
|
_pCiCat->ProcessChangesFlush( ftFlushed, cEntries, ppUsnEntries );
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR,
|
|
"CheckPointChangesFlushed failed. Error 0x%X\n",
|
|
e.GetErrorCode() ));
|
|
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::MarkDocUnReachable
|
|
//
|
|
// Synopsis: Marks that the document was not reachable when an attempt
|
|
// was made to filter it.
|
|
//
|
|
// Arguments: [wid] - The WORKID which could not be reached.
|
|
//
|
|
// History: 12-10-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::MarkDocUnReachable(
|
|
WORKID wid )
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
Win4Assert(_pCiCat);
|
|
|
|
TRY
|
|
{
|
|
_pCiCat->MarkUnReachable( wid );
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR,
|
|
"CClientDocStore::MarkDocUnReachable caught exception (0x%X)\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::StoreSecurity
|
|
//
|
|
// Synopsis: Stores the given security data for the workid.
|
|
//
|
|
// Arguments: [wid] - WorkId of the document
|
|
// [pbData] - Security data buffer
|
|
// [cbData] - NUmber of bytes in the security data buffer
|
|
//
|
|
// History: 1-15-97 srikants Created
|
|
//
|
|
// Notes: We may want to eliminate this call and instead have the
|
|
// security be stored as a special property.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
STDMETHODIMP CClientDocStore::StoreSecurity(
|
|
WORKID wid,
|
|
BYTE const * pbData,
|
|
ULONG cbData )
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
Win4Assert(_pCiCat);
|
|
|
|
TRY
|
|
{
|
|
_pCiCat->StoreSecurity( wid, (PSECURITY_DESCRIPTOR) pbData, cbData );
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR,
|
|
"CClientDocStore::MarkDocUnReachable caught exception (0x%X)\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::IsNoQuery
|
|
//
|
|
// Synopsis: Check if the docstore is set to NoQuery
|
|
//
|
|
// Arguments: [fNoQuery] - Output
|
|
//
|
|
// History: 05-26-98 kitmanh Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::IsNoQuery(
|
|
BOOL * fNoQuery )
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
if ( !fNoQuery )
|
|
sc = STATUS_INVALID_PARAMETER;
|
|
else
|
|
*fNoQuery = _fNoQuery;
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::GetPropertyMapper
|
|
//
|
|
// Synopsis: Retrieves the interface to IPropertyMapper.
|
|
//
|
|
// Arguments: [ppIPropertyMapper] - On output, will have the IPropertyMapper.
|
|
//
|
|
// Returns: S_OK if successful;
|
|
// E_NOTIMPL if property mapper is not supported by doc store.
|
|
// Other error code if there is a failure.
|
|
// Modifies:
|
|
//
|
|
// History: 12-31-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::GetPropertyMapper(
|
|
IPropertyMapper ** ppIPropertyMapper)
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
Win4Assert( 0 != _xPropMapper.GetPointer() );
|
|
|
|
*ppIPropertyMapper = _xPropMapper.GetPointer();
|
|
_xPropMapper->AddRef();
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR,
|
|
"CClientDocStore::MarkDocUnReachable caught exception (0x%X)\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
// ICiCDocNameToWorkidTranslatorEx methods.
|
|
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::QueryDocName
|
|
//
|
|
// Synopsis: Creates a new doc name object and returns its interface.
|
|
//
|
|
// Arguments: [ppICiCDocName] - [out] Will have a pointer to the
|
|
// ICiCDocName object filled in.
|
|
//
|
|
// History: 12-05-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::QueryDocName(
|
|
ICiCDocName ** ppICiCDocName )
|
|
{
|
|
Win4Assert( 0 != ppICiCDocName );
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
*ppICiCDocName = new CCiCDocName;
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::WorkIdToDocName
|
|
//
|
|
// Synopsis: Translates a WorkId to a document name.
|
|
//
|
|
// Arguments: [workid] - [in] WorkId to translate
|
|
// [pICiCDocName] - [out] Will be filled in with the document
|
|
// name on output.
|
|
//
|
|
// Returns: S_OK if successfully converted.
|
|
// CI_E_BUFFERTOOSMALL if the buffer is not big enough
|
|
// Other error code.
|
|
//
|
|
// History: 12-05-96 srikants Created
|
|
//
|
|
// Notes: This method may be one of the most frequently called methods.
|
|
// Look for optimizations (esp. can the TRY/CATCH be avoided?)
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::WorkIdToDocName(
|
|
WORKID workid,
|
|
ICiCDocName * pICiCDocName )
|
|
{
|
|
return InternalWorkIdToDocName( workid, pICiCDocName, FALSE );
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::DocNameToWorkId
|
|
//
|
|
// Synopsis: Converts a document name to a WorkId.
|
|
//
|
|
// Arguments: [pICiCDocName] - Document Name
|
|
// [pWorkid] - [out] Will have the workid on output.
|
|
//
|
|
// Returns: S_OK if successful; Error code otherwise.
|
|
//
|
|
// History: 12-05-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::DocNameToWorkId(
|
|
ICiCDocName const * pICiCDocName,
|
|
WORKID * pWorkid )
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
Win4Assert( 0 != pICiCDocName );
|
|
Win4Assert( 0 != pWorkid );
|
|
|
|
TRY
|
|
{
|
|
CCiCDocName const * pDocName = (CCiCDocName const *) pICiCDocName;
|
|
*pWorkid = _pCiCat->PathToWorkId( pDocName->GetPath(), TRUE );
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::WorkIdToAccurateDocName
|
|
//
|
|
// Synopsis: Translates a WorkId 'accurately' to a document name.
|
|
//
|
|
// Arguments: [workid] - [in] WorkId to translate
|
|
// [pICiCDocName] - [out] Will be filled in with the document
|
|
// name on output.
|
|
//
|
|
// Returns: S_OK if successfully converted.
|
|
// CI_E_BUFFERTOOSMALL if the buffer is not big enough
|
|
// Other error code.
|
|
//
|
|
// History: 31-Dec-1998 KyleP Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::WorkIdToAccurateDocName(
|
|
WORKID workid,
|
|
ICiCDocName * pICiCDocName )
|
|
{
|
|
return InternalWorkIdToDocName( workid, pICiCDocName, TRUE );
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::_GetPerfIndex
|
|
//
|
|
// Synopsis: An internal helper function to get the offset of the perfmon
|
|
// counter in the perfmon shared memory.
|
|
//
|
|
// Arguments: [name] - Name of the counter.
|
|
// [index] - Index of the name
|
|
//
|
|
// Returns: TRUE if successfully looked up; FALSE on failure.
|
|
//
|
|
// History: 12-06-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
BOOL CClientDocStore::_GetPerfIndex( CI_PERF_COUNTER_NAME name, ULONG & index )
|
|
{
|
|
BOOL fOk= TRUE;
|
|
|
|
ULONG offset = 0;
|
|
|
|
switch ( name )
|
|
{
|
|
case CI_PERF_NUM_WORDLIST:
|
|
|
|
offset = NUM_WORDLIST;
|
|
break;
|
|
|
|
case CI_PERF_NUM_PERSISTENT_INDEXES:
|
|
|
|
offset = NUM_PERSISTENT_INDEX;
|
|
break;
|
|
|
|
case CI_PERF_INDEX_SIZE:
|
|
|
|
offset = INDEX_SIZE;
|
|
break;
|
|
|
|
case CI_PERF_FILES_TO_BE_FILTERED:
|
|
|
|
offset = FILES_TO_BE_FILTERED;
|
|
break;
|
|
|
|
case CI_PERF_NUM_UNIQUE_KEY:
|
|
|
|
offset = NUM_UNIQUE_KEY;
|
|
break;
|
|
|
|
case CI_PERF_RUNNING_QUERIES:
|
|
|
|
offset = RUNNING_QUERIES;
|
|
break;
|
|
|
|
case CI_PERF_MERGE_PROGRESS:
|
|
|
|
offset = MERGE_PROGRESS;
|
|
break;
|
|
|
|
case CI_PERF_DOCUMENTS_FILTERED:
|
|
|
|
offset = DOCUMENTS_FILTERED;
|
|
break;
|
|
|
|
case CI_PERF_NUM_DOCUMENTS:
|
|
|
|
offset = NUM_DOCUMENTS;
|
|
break;
|
|
|
|
case CI_PERF_TOTAL_QUERIES:
|
|
|
|
offset = TOTAL_QUERIES;
|
|
break;
|
|
|
|
case CI_PERF_DEFERRED_FILTER_FILES:
|
|
|
|
offset = DEFERRED_FILTER_FILES;
|
|
break;
|
|
|
|
default:
|
|
|
|
fOk = FALSE;
|
|
}
|
|
|
|
if ( fOk )
|
|
index = offset + KERNEL_USER_INDEX;
|
|
|
|
return fOk;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::SetPerfCounterValue
|
|
//
|
|
// Synopsis: Sets the value of the perfmon counter.
|
|
//
|
|
// Arguments: [name] - Name of the counter
|
|
// [value] - Value to be set.
|
|
//
|
|
// Returns: S_OK if a valid perfmon name; E_INVALIDARG if the perfmon
|
|
// name is not correct.
|
|
//
|
|
// History: 12-06-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::SetPerfCounterValue(
|
|
CI_PERF_COUNTER_NAME name,
|
|
long value )
|
|
{
|
|
// true for the NULL catalog
|
|
|
|
if ( _xPerfMon.IsNull() )
|
|
return S_OK;
|
|
SCODE sc = S_OK;
|
|
ULONG index;
|
|
|
|
//
|
|
// CPerfMon::Update must not throw.
|
|
//
|
|
|
|
if ( _GetPerfIndex( name, index ) )
|
|
_xPerfMon->Update( index, value );
|
|
else
|
|
sc = E_INVALIDARG;
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::IncrementPerfCounterValue
|
|
//
|
|
// Synopsis: Increments the value of the perfmon counter.
|
|
//
|
|
// Arguments: [name] - Name of the counter
|
|
//
|
|
// Returns: S_OK if a valid perfmon name; E_INVALIDARG if the perfmon
|
|
// name is not correct.
|
|
//
|
|
// History: 1-15-97 dlee Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::IncrementPerfCounterValue(
|
|
CI_PERF_COUNTER_NAME name )
|
|
{
|
|
// true for the NULL catalog
|
|
|
|
if ( _xPerfMon.IsNull() )
|
|
return S_OK;
|
|
|
|
SCODE sc = S_OK;
|
|
ULONG index;
|
|
|
|
//
|
|
// CPerfMon::Update must not throw.
|
|
//
|
|
|
|
if ( _GetPerfIndex( name, index ) )
|
|
_xPerfMon->Increment( index );
|
|
else
|
|
sc = E_INVALIDARG;
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::DecrementPerfCounterValue
|
|
//
|
|
// Synopsis: Decrements the value of the perfmon counter.
|
|
//
|
|
// Arguments: [name] - Name of the counter
|
|
//
|
|
// Returns: S_OK if a valid perfmon name; E_INVALIDARG if the perfmon
|
|
// name is not correct.
|
|
//
|
|
// History: 1-15-97 dlee Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::DecrementPerfCounterValue(
|
|
CI_PERF_COUNTER_NAME name )
|
|
{
|
|
// true for the NULL catalog
|
|
|
|
if ( _xPerfMon.IsNull() )
|
|
return S_OK;
|
|
|
|
SCODE sc = S_OK;
|
|
ULONG index;
|
|
|
|
//
|
|
// CPerfMon::Update must not throw.
|
|
//
|
|
|
|
if ( _GetPerfIndex( name, index ) )
|
|
_xPerfMon->Decrement( index );
|
|
else
|
|
sc = E_INVALIDARG;
|
|
|
|
return sc;
|
|
} //DecrementPerfCounterValue
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::GetPerfCounterValue
|
|
//
|
|
// Synopsis: Retrieves the value of the perfmon counter.
|
|
//
|
|
// Arguments: [name] - [see SetPerfCounterValue]
|
|
// [pValue] - "
|
|
//
|
|
// History: 12-06-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
STDMETHODIMP CClientDocStore::GetPerfCounterValue(
|
|
CI_PERF_COUNTER_NAME name,
|
|
long * pValue )
|
|
{
|
|
// true for the NULL catalog
|
|
|
|
if ( _xPerfMon.IsNull() )
|
|
return S_OK;
|
|
|
|
Win4Assert( pValue );
|
|
|
|
ULONG index;
|
|
SCODE sc = S_OK;
|
|
|
|
if ( _GetPerfIndex( name, index ) )
|
|
*pValue = _xPerfMon->GetCurrValue( index );
|
|
else
|
|
sc = E_INVALIDARG;
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::NotifyEvent
|
|
//
|
|
// Synopsis: Reports the passed in event and arguments to eventlog
|
|
//
|
|
// Arguments: [fType ] - Type of event
|
|
// [eventId] - Message file event identifier
|
|
// [nParams] - Number of substitution arguments being passed
|
|
// [aParams] - pointer to PROPVARIANT array of substitution args.
|
|
// [cbData ] - number of bytes in supplemental raw data.
|
|
// [data ] - pointer to block of supplemental data.
|
|
//
|
|
// Returns: S_OK upon success, value of the exception if an exception
|
|
// is thrown.
|
|
//
|
|
// History: 12-30-96 mohamedn Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::NotifyEvent( WORD fType,
|
|
DWORD eventId,
|
|
ULONG nParams,
|
|
const PROPVARIANT *aParams,
|
|
ULONG cbData,
|
|
void* data)
|
|
|
|
{
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
CClientNotifyEvent notifyEvent(fType,eventId,nParams,aParams,cbData,data);
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Exception 0x%X in CClientDocStore::NotifyEvent()\n",
|
|
e.GetErrorCode() ));
|
|
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::NotifyStatus
|
|
//
|
|
// Synopsis: When a special status is being notified.
|
|
//
|
|
// Returns: S_OK always.
|
|
//
|
|
// History: 12-05-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::NotifyStatus(
|
|
CI_NOTIFY_STATUS_VALUE status,
|
|
ULONG nParams,
|
|
const PROPVARIANT * aParams )
|
|
{
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
switch ( status )
|
|
{
|
|
case CI_NOTIFY_FILTERING_FAILURE:
|
|
|
|
Win4Assert( nParams == 1 );
|
|
Win4Assert( aParams[0].vt == VT_I4 || aParams[0].vt == VT_UI4 );
|
|
|
|
if ( 1 == nParams )
|
|
_ReportFilteringFailure( aParams[0].ulVal );
|
|
break;
|
|
|
|
case CI_NOTIFY_FILTER_TOO_MANY_BLOCKS:
|
|
case CI_NOTIFY_FILTER_EMBEDDING_FAILURE:
|
|
//
|
|
// This is possible in the in-proc filtering because
|
|
// the advise status is provided by docstore for filtering
|
|
// also.
|
|
//
|
|
{
|
|
CStorageFilterObjNotifyStatus notify( this );
|
|
sc = notify.NotifyStatus( status, nParams, aParams );
|
|
}
|
|
break;
|
|
|
|
default:
|
|
|
|
Win4Assert( !"Invalid Case Stmt" );
|
|
break;
|
|
};
|
|
|
|
return sc;
|
|
}
|
|
|
|
//
|
|
// IFsCiAdmin methods.
|
|
//
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::ForceMerge
|
|
//
|
|
// Synopsis: Forces a master merge on the given partition id.
|
|
//
|
|
// History: 2-12-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::ForceMerge( PARTITIONID partId )
|
|
{
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
if ( 0 == _pCiCat )
|
|
THROW( CException( E_INVALIDARG ) );
|
|
|
|
sc = _pCiCat->ForceMerge( partId );
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Error 0x%X in CClientDocStore::ForceMerge\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::AbortMerge
|
|
//
|
|
// Synopsis: Aborts any in-progress merge on the given partition id.
|
|
//
|
|
// History: 2-12-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::AbortMerge( PARTITIONID partId )
|
|
{
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
if ( 0 == _pCiCat )
|
|
THROW( CException( E_INVALIDARG ) );
|
|
|
|
sc = _pCiCat->AbortMerge( partId );
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Error 0x%X in CClientDocStore::AbortMerge\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::CiState
|
|
//
|
|
// Synopsis: Retrieves the CI_STATE information.
|
|
//
|
|
// Arguments: [pCiState] - On output, will contain the cistate data.
|
|
//
|
|
// History: 2-12-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::CiState( CI_STATE * pCiState )
|
|
{
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
if (_pCiNullCat)
|
|
sc = _pCiNullCat->CiState( *pCiState );
|
|
else
|
|
sc = _pCiCat->CiState( *pCiState );
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Error 0x%X in CClientDocStore::CiState\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::UpdateDocuments
|
|
//
|
|
// Synopsis: Forces a scan on the given root.
|
|
//
|
|
// Arguments: [rootPath] - Root to force a scan on.
|
|
// [flag] - Indicating if this is a full scan or a partial
|
|
// scan.
|
|
//
|
|
// History: 2-12-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::UpdateDocuments(
|
|
const WCHAR *rootPath,
|
|
ULONG flag )
|
|
{
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
if ( 0 == _pCiCat )
|
|
THROW( CException( E_INVALIDARG ) );
|
|
|
|
_pCiCat->UpdateDocuments( rootPath, flag);
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Error 0x%X in CClientDocStore::UpdateDocuments\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::AddScopeToCI
|
|
//
|
|
// Synopsis: Adds the given scope to ContentIndex for indexing.
|
|
//
|
|
// Arguments: [rootPath] - Path to add
|
|
//
|
|
// History: 2-12-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::AddScopeToCI(
|
|
const WCHAR *rootPath )
|
|
{
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
if ( 0 == _pCiCat )
|
|
THROW( CException( E_INVALIDARG ) );
|
|
|
|
_pCiCat->AddScopeToCI( rootPath );
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Error 0x%X in CClientDocStore::AddScopeToCI\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::RemoveScopeFromCI
|
|
//
|
|
// Synopsis: Removes the given scope from CI.
|
|
//
|
|
// Arguments: [rootPath] - Scope to remove.
|
|
//
|
|
// History: 2-12-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::RemoveScopeFromCI(
|
|
const WCHAR *rootPath )
|
|
{
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
if ( 0 == _pCiCat )
|
|
THROW( CException( E_INVALIDARG ) );
|
|
|
|
_pCiCat->RemoveScopeFromCI( rootPath, FALSE );
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Error 0x%X in CClientDocStore::RemoveScopeFromCI\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::BeginCacheTransaction
|
|
//
|
|
// Synopsis: Begins a property cache transaction.
|
|
//
|
|
// Arguments: [pulToken] - Output - Transaction "cookie".
|
|
//
|
|
// History: 2-12-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::BeginCacheTransaction(
|
|
ULONG_PTR * pulToken )
|
|
{
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
if ( 0 == _pCiCat )
|
|
THROW( CException( E_INVALIDARG ) );
|
|
|
|
*pulToken = _pCiCat->BeginCacheTransaction();
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Error 0x%X in CClientDocStore::BeginCacheTransaction\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::SetupCache
|
|
//
|
|
// Synopsis: Sets up the property for storing the property cache.
|
|
//
|
|
// Arguments: [ps] -
|
|
// [vt] -
|
|
// [cbMaxLen] -
|
|
// [ulToken] -
|
|
//
|
|
// History: 2-12-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::SetupCache(
|
|
const FULLPROPSPEC *ps,
|
|
ULONG vt,
|
|
ULONG cbMaxLen,
|
|
ULONG_PTR ulToken,
|
|
BOOL fCanBeModified,
|
|
DWORD dwStoreLevel)
|
|
{
|
|
Win4Assert(PRIMARY_STORE == dwStoreLevel ||
|
|
SECONDARY_STORE == dwStoreLevel);
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
if ( 0 == _pCiCat )
|
|
THROW( CException( E_INVALIDARG ) );
|
|
|
|
CFullPropSpec * pFullPropSpec = (CFullPropSpec *)ps;
|
|
_pCiCat->SetupCache( *pFullPropSpec,
|
|
vt,
|
|
cbMaxLen,
|
|
ulToken,
|
|
fCanBeModified,
|
|
dwStoreLevel );
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Error 0x%X in CClientDocStore::RemoveVirtualScope\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::EndCacheTransaction
|
|
//
|
|
// Synopsis: Ends the property cache transaction.
|
|
//
|
|
// Arguments: [ulToken] -
|
|
// [fCommit] -
|
|
//
|
|
// History: 2-12-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::EndCacheTransaction(
|
|
ULONG_PTR ulToken,
|
|
BOOL fCommit )
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
if ( 0 == _pCiCat )
|
|
THROW( CException( E_INVALIDARG ) );
|
|
|
|
_pCiCat->EndCacheTransaction( ulToken, fCommit );
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Error 0x%X in CClientDocStore::RemoveVirtualScope\n",
|
|
e.GetErrorCode() ));
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Method: CClientDocStore::IsIoHigh, public
|
|
//
|
|
// Returns: S_OK if system is in a high i/o state, S_FALSE if it is not.
|
|
//
|
|
// History: 21-Jul-1998 KyleP Created
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
SCODE CClientDocStore::IsIoHigh( BOOL * pfAbort )
|
|
{
|
|
// Windows XP removed support for the IOCTL we used to measure disk
|
|
// usage.
|
|
|
|
return S_FALSE;
|
|
|
|
} //IsIoHigh
|
|
|
|
//
|
|
// Non-Interface methods.
|
|
//
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::_StartFiltering
|
|
//
|
|
// Synopsis: Asks the CiManager to start filter daemon and resume
|
|
// filtering.
|
|
//
|
|
// History: 12-09-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
void CClientDocStore::_StartFiltering()
|
|
{
|
|
XInterface<ICiManager> xManager;
|
|
|
|
// =============================================
|
|
{
|
|
CLock lock(_mutex);
|
|
if ( !_IsShutdown() && _xCiManager.GetPointer() )
|
|
{
|
|
_xCiManager->AddRef();
|
|
xManager.Set( _xCiManager.GetPointer() );
|
|
}
|
|
else
|
|
{
|
|
ciDebugOut(( DEB_WARN,
|
|
"Already Shutdown or CI not started. Cannot start filtering\n" ));
|
|
return;
|
|
}
|
|
}
|
|
// =============================================
|
|
|
|
//
|
|
// Get the startup data and serialize it.
|
|
//
|
|
WCHAR const * pwszName = _pCiCat->GetName();
|
|
WCHAR const * pwszCatDir = _pCiCat->GetDriveName();
|
|
|
|
|
|
CDaemonStartupData startupData( pwszCatDir,
|
|
pwszName );
|
|
|
|
ULONG cbData;
|
|
BYTE * pbData = startupData.Serialize( cbData );
|
|
XArray<BYTE> xBuf;
|
|
xBuf.Set(cbData,pbData);
|
|
|
|
xManager->StartFiltering( cbData, pbData );
|
|
|
|
//
|
|
// Note: Should we be worried about StartFiltering failing ??
|
|
//
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::Shutdown
|
|
//
|
|
// Synopsis: Intiates a shutdown of the document store and the
|
|
// CiManager associated with the document store.
|
|
//
|
|
// History: 12-05-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
void CClientDocStore::Shutdown()
|
|
{
|
|
TRY
|
|
{
|
|
Win4Assert( !_IsShutdown() );
|
|
|
|
// =============================================
|
|
{
|
|
CLock lock(_mutex);
|
|
_LokMarkShutdown();
|
|
}
|
|
// =============================================
|
|
|
|
#if CIDBG==1
|
|
if ( !_xSaveTest.IsNull() )
|
|
_xSaveTest->InitiateShutdown();
|
|
#endif // CIDBG==1
|
|
|
|
if (_pCiNullCat)
|
|
_pCiNullCat->ShutdownPhase1();
|
|
else
|
|
_pCiCat->ShutdownPhase1();
|
|
|
|
if ( _xCiManager.GetPointer() )
|
|
{
|
|
_xCiManager->Shutdown();
|
|
_xCiManager.Free();
|
|
}
|
|
|
|
#if CIDBG==1
|
|
if ( !_xSaveTest.IsNull() )
|
|
_xSaveTest->WaitForDeath();
|
|
#endif //CIDBG==1
|
|
|
|
if (_pCiNullCat)
|
|
_pCiNullCat->ShutdownPhase2();
|
|
else
|
|
_pCiCat->ShutdownPhase2();
|
|
}
|
|
CATCH( CException, e)
|
|
{
|
|
ciDebugOut(( DEB_ERROR, "Error (0x%X) in CClientDocStore::Shutdown\n",
|
|
e.GetErrorCode() ));
|
|
}
|
|
END_CATCH
|
|
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::GetName
|
|
//
|
|
// Synopsis: Gets the catalog name (if any)
|
|
//
|
|
// History: 12-05-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
WCHAR const * CClientDocStore::GetName()
|
|
{
|
|
if (_pCiNullCat)
|
|
return _pCiNullCat->GetName();
|
|
else
|
|
return _pCiCat->GetName();
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::_SetCiCatRecovered
|
|
//
|
|
// Synopsis: Sets that the CiCat is recovered and if updates were
|
|
// enabled by Content Index, it asks CiCat to start scans
|
|
// and notifications.
|
|
//
|
|
// History: 12-09-96 srikants Created
|
|
// 02-25-98 kitmanh if catalog is read-only, don't filter
|
|
//
|
|
// Notes: This method MUST be called in SYSTEM context only. This
|
|
// starts filtering which in turn creates the filter daemon
|
|
// proxy thread and that must be in SYSTEM context.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
void CClientDocStore::_SetCiCatRecovered()
|
|
{
|
|
Win4Assert( 0 != _pCiCat );
|
|
|
|
Win4Assert( !CImpersonateSystem::IsImpersonated() );
|
|
|
|
BOOL fEnableFilering = FALSE;
|
|
|
|
//don't start filtering if catalog is read-only
|
|
if ( !_pCiCat->IsReadOnly() )
|
|
{
|
|
|
|
// ==================================================
|
|
{
|
|
CLock lock(_mutex);
|
|
|
|
fEnableFilering = _AreUpdatesEnabled();
|
|
}
|
|
// ==================================================
|
|
|
|
//
|
|
// Since CiCat has been recovered fully, filtering can be started
|
|
// if we are running with "indexing" enabled.
|
|
//
|
|
_StartFiltering();
|
|
|
|
if ( fEnableFilering )
|
|
_pCiCat->EnableUpdateNotifies();
|
|
}
|
|
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::IsLowOnDiskSpace
|
|
//
|
|
// Synopsis: Returns the status of free space on the disk used by
|
|
// docstore (client) based on the last check.
|
|
//
|
|
// Returns: TRUE if low on free disk space; FALSE o/w
|
|
//
|
|
// History: 12-10-96 srikants Created
|
|
//
|
|
// Notes: This does not actually make an I/O. Just returns an in-memory
|
|
// status.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
BOOL CClientDocStore::IsLowOnDiskSpace() const
|
|
{
|
|
Win4Assert ( !_xDiskStatus.IsNull() );
|
|
return _xDiskStatus->IsLow();
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::VerifyIfLowOnDiskSpace
|
|
//
|
|
// Synopsis: Verifies the free disk space situation by making an I/O call
|
|
// to the O/S.
|
|
//
|
|
// Returns: TRUE if disk is getting to be full; FALSE o/w
|
|
//
|
|
// History: 12-10-96 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
|
|
BOOL CClientDocStore::VerifyIfLowOnDiskSpace()
|
|
{
|
|
Win4Assert ( !_xDiskStatus.IsNull() );
|
|
_xDiskStatus->UpdateDiskLowInfo();
|
|
return _xDiskStatus->IsLow();
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::_ReportFilteringFailure
|
|
//
|
|
// Synopsis: Reports that there was filtering failure on the given wid.
|
|
//
|
|
// Arguments: [wid] - Workid which failed to filter.
|
|
//
|
|
// History: 1-24-97 srikants Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
void CClientDocStore::_ReportFilteringFailure( WORKID wid )
|
|
{
|
|
ciDebugOut(( DEB_IWARN,
|
|
"Warning: unsuccessful attempts to filter workid 0x%X; "
|
|
"cancelling\n", wid ));
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::GetQuerySession
|
|
//
|
|
// Synopsis: Returns a query session object
|
|
//
|
|
// Returns: [ppICiCQuerySession] -- Session object is returned here
|
|
//
|
|
// History: 22-Jan-97 SitaramR Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
SCODE STDMETHODCALLTYPE CClientDocStore::GetQuerySession( ICiCQuerySession **ppICiCQuerySession)
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
CQuerySession *pQuerySession = _pCiNullCat ? new CQuerySession(*_pCiNullCat) :
|
|
new CQuerySession(*_pCiCat );
|
|
XInterface<CQuerySession> xSession( pQuerySession );
|
|
|
|
sc = pQuerySession->QueryInterface( IID_ICiCQuerySession, (void **) ppICiCQuerySession );
|
|
|
|
if ( FAILED(sc) ) {
|
|
vqDebugOut(( DEB_ERROR, "GetQuerySession - QI failed 0x%x\n", sc ));
|
|
}
|
|
}
|
|
CATCH( CException,e )
|
|
{
|
|
sc = e.GetErrorCode();
|
|
|
|
vqDebugOut(( DEB_ERROR,
|
|
"CClientDocStore::GetQuerySession - Exception caught 0x%x\n",
|
|
sc ));
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
|
|
} //GetQuerySession
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: RegisterDLL
|
|
//
|
|
// Synopsis: Calls DllRegisterServer on the fully qualified path
|
|
//
|
|
// History: 19-Jun-97
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
DWORD RegisterDLL( WCHAR const * pwcDLL )
|
|
{
|
|
// All Index Server dlls are currently in system32
|
|
|
|
DWORD dwErr = NO_ERROR;
|
|
|
|
HINSTANCE hDll = LoadLibraryEx( pwcDLL,
|
|
0,
|
|
LOAD_WITH_ALTERED_SEARCH_PATH );
|
|
|
|
if( 0 != hDll )
|
|
{
|
|
SCODE (STDAPICALLTYPE *pfnDllRegisterServer)();
|
|
pfnDllRegisterServer = (HRESULT (STDAPICALLTYPE *)())
|
|
GetProcAddress(hDll, "DllRegisterServer");
|
|
|
|
if ( 0 != pfnDllRegisterServer )
|
|
{
|
|
TRY
|
|
{
|
|
SCODE sc = (*pfnDllRegisterServer)();
|
|
if ( S_OK != sc )
|
|
{
|
|
// no way to map a scode to a win32 error
|
|
dwErr = ERROR_INVALID_FUNCTION;
|
|
}
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
Win4Assert( !"DllRegisterServer threw an exception" );
|
|
ciDebugOut(( DEB_ERROR, "caught 0x%x registering '%ws'\n",
|
|
e.GetErrorCode(), pwcDLL ));
|
|
}
|
|
END_CATCH;
|
|
}
|
|
else
|
|
dwErr = GetLastError();
|
|
|
|
FreeLibrary( hDll );
|
|
}
|
|
else
|
|
{
|
|
dwErr = GetLastError();
|
|
}
|
|
|
|
return dwErr;
|
|
} //RegisterDll
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: RegisterKnownDlls
|
|
//
|
|
// Synopsis: Gets the value of the DLL's from the registry and then
|
|
// registers all of them if not done so already.
|
|
//
|
|
// History: 19-Jun-97 t-elainc Created
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
void RegisterKnownDlls()
|
|
{
|
|
static fKnownDllsRegistered = FALSE;
|
|
|
|
if ( fKnownDllsRegistered )
|
|
return;
|
|
|
|
fKnownDllsRegistered = TRUE;
|
|
|
|
TRY
|
|
{
|
|
// get registry value
|
|
|
|
HKEY hkey;
|
|
int errorcode = RegOpenKeyEx( HKEY_LOCAL_MACHINE,
|
|
wcsRegAdminSubKey,
|
|
0,
|
|
KEY_QUERY_VALUE |
|
|
KEY_ENUMERATE_SUB_KEYS,
|
|
&hkey );
|
|
|
|
if (NO_ERROR == errorcode)
|
|
{
|
|
SRegKey xKey( hkey );
|
|
|
|
WCHAR awcPath[4000];
|
|
DWORD cwcPath = sizeof awcPath / sizeof WCHAR;
|
|
|
|
int anothererrorcode = RegQueryValueEx( hkey,
|
|
L"DLLsToRegister",
|
|
0,
|
|
0,
|
|
(BYTE *)awcPath,
|
|
&cwcPath );
|
|
if (NO_ERROR == anothererrorcode)
|
|
{
|
|
// parse the string
|
|
WCHAR* pwcCurrFile = awcPath;
|
|
|
|
//for each file call RegisterDLL
|
|
while ( 0 != *pwcCurrFile )
|
|
{
|
|
RegisterDLL(pwcCurrFile);
|
|
pwcCurrFile += wcslen( pwcCurrFile) + 1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
CATCH( CException, ex )
|
|
{
|
|
ciDebugOut(( DEB_FORCE, "exception %#x registering dlls\n",
|
|
ex.GetErrorCode() ));
|
|
}
|
|
END_CATCH;
|
|
} //RegisterKnownDlls
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OpenCatalogsOnRemovableDrives
|
|
//
|
|
// Synopsis: Opens the catalogs found on the roots of removable drives
|
|
// that are not yet opened.
|
|
//
|
|
// History: 28-Apr-99 dlee Created.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
void OpenCatalogsOnRemovableDrives()
|
|
{
|
|
CRegAccess reg( RTL_REGISTRY_CONTROL, wcsRegAdmin );
|
|
|
|
if ( 0 == reg.Read( wcsMountRemovableCatalogs,
|
|
CI_AUTO_MOUNT_CATALOGS_DEFAULT ) )
|
|
return;
|
|
|
|
// Determine which drives exist in this bitmask
|
|
|
|
DWORD dwDriveMask = GetLogicalDrives();
|
|
dwDriveMask >>= 2;
|
|
|
|
WCHAR awcPath[5];
|
|
wcscpy( awcPath, L"c:\\" );
|
|
|
|
// loop through all the drives c-z
|
|
|
|
for ( WCHAR wc = L'C'; wc <= L'Z'; wc++ )
|
|
{
|
|
DWORD dwTemp = ( dwDriveMask & 1 );
|
|
dwDriveMask >>= 1;
|
|
|
|
if ( 0 != dwTemp )
|
|
{
|
|
if ( IsRemovableDrive( wc ) )
|
|
{
|
|
awcPath[0] = wc;
|
|
|
|
if ( 0 == Catalogs.GetDocStore( awcPath ) )
|
|
Catalogs.TryToStartCatalogOnRemovableVol( wc, g_pFSCIRequestQueue );
|
|
}
|
|
}
|
|
}
|
|
} //OpenCatalogsOnRemovableDrives
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OpenCatalogsInRegistry
|
|
//
|
|
// Synopsis: Opens the catalogs listed in the registry. If there are
|
|
// problems opening a catalog, it is skipped -- no sense
|
|
// bringing down the service over a messed-up registry entry.
|
|
//
|
|
// History: 10-Oct-96 dlee Created.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
void OpenCatalogsInRegistry( BOOL fOpenForReadOnly = FALSE )
|
|
{
|
|
//
|
|
// Side effect of starting up -- register dlls
|
|
//
|
|
|
|
RegisterKnownDlls();
|
|
|
|
ciDebugOut(( DEB_ITRACE, "OpenCatalogsInRegistry: fOpenForReadOnly == %d\n",
|
|
fOpenForReadOnly ));
|
|
|
|
HKEY hKey;
|
|
if ( ERROR_SUCCESS == RegOpenKeyEx( HKEY_LOCAL_MACHINE,
|
|
wcsRegCatalogsSubKey,
|
|
0,
|
|
KEY_QUERY_VALUE |
|
|
KEY_ENUMERATE_SUB_KEYS,
|
|
&hKey ) )
|
|
{
|
|
SRegKey xKey( hKey );
|
|
DWORD iSubKey = 0;
|
|
|
|
do
|
|
{
|
|
FILETIME ft;
|
|
WCHAR awcName[MAX_PATH];
|
|
DWORD cwcName = sizeof awcName / sizeof WCHAR;
|
|
LONG err = RegEnumKeyEx( hKey,
|
|
iSubKey,
|
|
awcName,
|
|
&cwcName,
|
|
0, 0, 0, &ft );
|
|
|
|
// either error or end of enumeration
|
|
|
|
if ( ERROR_SUCCESS != err )
|
|
break;
|
|
|
|
iSubKey++;
|
|
|
|
HKEY hCatName;
|
|
if ( ERROR_SUCCESS == RegOpenKeyEx( hKey,
|
|
awcName,
|
|
0,
|
|
KEY_QUERY_VALUE,
|
|
&hCatName ) )
|
|
{
|
|
SRegKey xCatNameKey( hCatName );
|
|
WCHAR awcPath[MAX_PATH];
|
|
DWORD cbPath = sizeof awcPath;
|
|
if ( ERROR_SUCCESS == RegQueryValueEx( hCatName,
|
|
wcsCatalogLocation,
|
|
0,
|
|
0,
|
|
(BYTE *)awcPath,
|
|
&cbPath ) )
|
|
{
|
|
Catalogs.GetNamedOne( awcPath, awcName, fOpenForReadOnly );
|
|
}
|
|
}
|
|
} while ( TRUE );
|
|
}
|
|
|
|
OpenCatalogsOnRemovableDrives();
|
|
} //OpenCatalogsInRegistry
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: OpenOneCatalog
|
|
//
|
|
// Synopsis: Opens the catalog specified for R/W or R/O.
|
|
// set the fNoQuery flag in the docstore if fNoQuery (passed in)
|
|
// is TRUE
|
|
//
|
|
// Arguments: [wcCatName] -- name of the catalog
|
|
// [fReadOnly] -- opening for readOnly
|
|
// [fNoQuery] -- opening the catalog as NoQuery
|
|
//
|
|
// History: 27-Apr-98 kitmanh Created.
|
|
// 12-May-98 kitmanh Added fNoQuery
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
void OpenOneCatalog( WCHAR const * wcCatName, BOOL fReadOnly, BOOL fNoQuery = FALSE )
|
|
{
|
|
ciDebugOut(( DEB_ITRACE, "OpenOneCatalog\n" ));
|
|
ciDebugOut(( DEB_ITRACE, "fNoQuery is %d\n", fNoQuery ));
|
|
|
|
HKEY hKey;
|
|
WCHAR wcsKey[MAX_PATH];
|
|
|
|
if ( ( wcslen( wcsRegCatalogsSubKey ) + wcslen( wcCatName ) + 1 ) >= MAX_PATH )
|
|
THROW( CException( STATUS_INVALID_PARAMETER ) );
|
|
|
|
wcscpy( wcsKey, wcsRegCatalogsSubKey );
|
|
wcscat( wcsKey, L"\\" );
|
|
wcscat( wcsKey, wcCatName );
|
|
|
|
if ( ERROR_SUCCESS == RegOpenKeyEx( HKEY_LOCAL_MACHINE,
|
|
wcsKey,
|
|
0,
|
|
KEY_QUERY_VALUE,
|
|
&hKey ) )
|
|
{
|
|
SRegKey xKey( hKey );
|
|
|
|
WCHAR awcPath[MAX_PATH];
|
|
DWORD cwcPath = sizeof awcPath / sizeof WCHAR;
|
|
if ( ERROR_SUCCESS == RegQueryValueEx( hKey,
|
|
wcsCatalogLocation,
|
|
0,
|
|
0,
|
|
(BYTE *)awcPath,
|
|
&cwcPath ) )
|
|
{
|
|
Catalogs.GetNamedOne( awcPath, wcCatName, fReadOnly, fNoQuery );
|
|
}
|
|
}
|
|
} //OpenOneCatalog
|
|
|
|
// Mutex and Event to synchronize start, stop, pause and continue
|
|
CStaticMutexSem g_mtxStartStop;
|
|
CEventSem * g_pevtPauseContinue = 0;
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: StartCiSvcWork
|
|
//
|
|
// Synopsis: Entry point for doing the CI service work. The thread does
|
|
// not exit until StopCiSvcWork is called.
|
|
//
|
|
// Argument: [DrvNotifArray] -- reference to the only one
|
|
// DriveNotificationArray
|
|
//
|
|
// History: 16-Sep-96 dlee Created.
|
|
// 16-Sep-98 kitmanh Added parameter DrvNotifArray
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
void StartCiSvcWork( CDrvNotifArray & DrvNotifArray )
|
|
{
|
|
TRY
|
|
{
|
|
//
|
|
// This lock is released when cisvc is initialized enough that
|
|
// StopCiSvcWork can safely be called.
|
|
//
|
|
|
|
CReleasableLock lock( g_mtxStartStop );
|
|
ciDebugOut(( DEB_ITRACE, "StartCiSvcWork got the lock\n" ));
|
|
|
|
Catalogs.Init();
|
|
TheFrameworkClientWorkQueue.Init();
|
|
|
|
XCom xcom;
|
|
|
|
// Create the CRequestQueue here, so it'll stay around for
|
|
// the whole time
|
|
|
|
ULONG cMaxCachedServerItems, cMaxSimultaneousRequests,
|
|
cmsDefaultClientTimeout, cMinClientIdleTime, cmsStartupDelay;
|
|
BOOL fMinimizeWorkingSet;
|
|
|
|
{
|
|
CRegAccess reg( RTL_REGISTRY_CONTROL, wcsRegAdmin );
|
|
cMaxCachedServerItems = reg.Read( wcsMaxCachedPipes,
|
|
CI_MAX_CACHED_PIPES_DEFAULT,
|
|
CI_MAX_CACHED_PIPES_MIN,
|
|
CI_MAX_CACHED_PIPES_MAX );
|
|
cMaxSimultaneousRequests = reg.Read( wcsMaxSimultaneousRequests,
|
|
CI_MAX_SIMULTANEOUS_REQUESTS_DEFAULT,
|
|
CI_MAX_SIMULTANEOUS_REQUESTS_MIN,
|
|
CI_MAX_SIMULTANEOUS_REQUESTS_MAX );
|
|
cmsDefaultClientTimeout = reg.Read( wcsRequestTimeout,
|
|
CI_REQUEST_TIMEOUT_DEFAULT,
|
|
CI_REQUEST_TIMEOUT_MIN,
|
|
CI_REQUEST_TIMEOUT_MAX );
|
|
fMinimizeWorkingSet = reg.Read( wcsMinimizeWorkingSet,
|
|
(DWORD)CI_MINIMIZE_WORKINGSET_DEFAULT );
|
|
|
|
// convert seconds to milliseconds
|
|
|
|
cMinClientIdleTime = 1000 * reg.Read( wcsMinClientIdleTime,
|
|
CI_MIN_CLIENT_IDLE_TIME );
|
|
cmsStartupDelay = reg.Read( wcsStartupDelay,
|
|
CI_STARTUP_DELAY_DEFAULT,
|
|
CI_STARTUP_DELAY_MIN,
|
|
CI_STARTUP_DELAY_MAX );
|
|
}
|
|
|
|
Catalogs.SetDrvNotifArray( &DrvNotifArray );
|
|
|
|
// Create the CRequestQueue here, so it'll stay around for the
|
|
// whole time.
|
|
|
|
CRequestQueue queue( cMaxCachedServerItems,
|
|
cMaxSimultaneousRequests,
|
|
cmsDefaultClientTimeout,
|
|
fMinimizeWorkingSet,
|
|
cMinClientIdleTime,
|
|
cmsStartupDelay,
|
|
guidStorageDocStoreLocatorObject );
|
|
|
|
g_pFSCIRequestQueue = &queue;
|
|
|
|
while ( TRUE )
|
|
{
|
|
// request lock to prevent problems of sequence of
|
|
// net pause, net stop and net start (in any order
|
|
// and combo.)
|
|
|
|
if ( !lock.IsHeld() )
|
|
lock.Request();
|
|
|
|
ciDebugOut(( DEB_ITRACE, "About to call StartFWCiSvcWork\n" ));
|
|
|
|
StartFWCiSvcWork( lock, &queue, *g_pevtPauseContinue);
|
|
|
|
ciDebugOut(( DEB_ITRACE, "StartCiSvcWork:: fell out from StartFWCiSvcWork\n" ));
|
|
|
|
// In the case where a net pause happened
|
|
// instead of checking IsNetPause and IsNetcontinue, loop thru the
|
|
// dynarray to reopen docstores. and clear the SCarry when done
|
|
// note if array.count == 0, work as normal.
|
|
SCWorkItem * WorkItem;
|
|
WCHAR wcCatName[MAX_PATH];
|
|
|
|
if ( !queue.IsShutdown() )
|
|
{
|
|
// something is on the _stateChangeArray to be handled
|
|
for ( unsigned i = 0; i < queue.SCArrayCount(); i++ )
|
|
{
|
|
WorkItem = queue.GetSCItem(i);
|
|
if ( WorkItem->StoppedCat ) {
|
|
wcsncpy( wcCatName, WorkItem->StoppedCat, sizeof wcCatName / sizeof WCHAR );
|
|
wcCatName[ (sizeof wcCatName / sizeof WCHAR) - 1 ] = 0;
|
|
}
|
|
else if ( eNoCatWork != WorkItem->type ) {
|
|
wcsncpy( wcCatName, ( (CClientDocStore *)(WorkItem->pDocStore) )->GetName(), sizeof wcCatName / sizeof WCHAR );
|
|
wcCatName[ (sizeof wcCatName / sizeof WCHAR) - 1 ] = 0;
|
|
}
|
|
switch ( WorkItem->type )
|
|
{
|
|
case eCatRO:
|
|
//flush cat
|
|
if ( !WorkItem->StoppedCat )
|
|
Catalogs.FlushOne( WorkItem->pDocStore );
|
|
else
|
|
Catalogs.RmFromStopArray( wcCatName );
|
|
|
|
OpenOneCatalog( wcCatName, TRUE );
|
|
break;
|
|
case eCatW:
|
|
if ( !WorkItem->StoppedCat )
|
|
Catalogs.FlushOne( WorkItem->pDocStore );
|
|
else
|
|
Catalogs.RmFromStopArray( wcCatName );
|
|
|
|
OpenOneCatalog( wcCatName, FALSE );
|
|
break;
|
|
case eStopCat:
|
|
if ( !WorkItem->StoppedCat )
|
|
{
|
|
Catalogs.FlushOne( WorkItem->pDocStore );
|
|
ciDebugOut(( DEB_ITRACE, "Catalogs.IsCatStopped( %ws ) == %d\n",
|
|
wcCatName, Catalogs.IsCatStopped( wcCatName ) ));
|
|
}
|
|
break;
|
|
case eNoQuery:
|
|
if ( !WorkItem->StoppedCat )
|
|
Catalogs.FlushOne( WorkItem->pDocStore );
|
|
else
|
|
Catalogs.RmFromStopArray( wcCatName );
|
|
|
|
OpenOneCatalog( wcCatName, !(WorkItem->fNoQueryRW), TRUE );
|
|
break;
|
|
case eNetPause:
|
|
case eNetContinue:
|
|
case eNetStop:
|
|
case eNoCatWork:
|
|
// These cases are handled later
|
|
break;
|
|
default:
|
|
Win4Assert( !"The eCisvcActionType specified is unknown" );
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( queue.IsNetPause() )
|
|
{
|
|
Win4Assert( !queue.IsNetContinue() );
|
|
|
|
Catalogs.ClearStopArray();
|
|
// reset the queue
|
|
|
|
queue.ReStart();
|
|
|
|
//closecatalog
|
|
Catalogs.Flush();
|
|
|
|
OpenCatalogsInRegistry( TRUE ); //reopen catalogs for r/o
|
|
|
|
//Win4Assert( !"Done Pausing" );
|
|
}
|
|
// a net continue happened
|
|
else if ( queue.IsNetContinue() )
|
|
{
|
|
Win4Assert( !queue.IsNetPause() );
|
|
|
|
Catalogs.ClearStopArray();
|
|
// reset the queue
|
|
queue.ReStart();
|
|
|
|
ciDebugOut(( DEB_ITRACE, "YES, net continued*********\n" ));
|
|
ciDebugOut(( DEB_ITRACE, "About to flush catalogs.\n" ));
|
|
//closecatalog
|
|
Catalogs.Flush();
|
|
ciDebugOut(( DEB_ITRACE, "About to OpenCatalogsInRegistry for r/w \n" ));
|
|
|
|
OpenCatalogsInRegistry( FALSE ); //reopen catalogs for r/w
|
|
|
|
ciDebugOut(( DEB_ITRACE, "Done opening catalogs in registry\n" ));
|
|
|
|
//Win4Assert( !"Done Continuing" );
|
|
}
|
|
else
|
|
{
|
|
if ( !( queue.IsShutdown() || queue.IsNetStop() ) )
|
|
queue.ReStart();
|
|
else
|
|
{
|
|
ciDebugOut(( DEB_ITRACE, "StartCisvcWork: Breaking out of the loop and ready to shutdown the service\n" ));
|
|
Catalogs.ClearStopArray();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
CATCH( CException, ex )
|
|
{
|
|
ciDebugOut(( DEB_WARN, "StartCiSvcWork2 exception error 0x%x\n",
|
|
ex.GetErrorCode() ));
|
|
}
|
|
END_CATCH;
|
|
} //StartCiSvcWork
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: StopCiSvcWork
|
|
//
|
|
// Synopsis: Entry point for stopping the CI service work.
|
|
//
|
|
// Arguments: [type] -- type of work to do
|
|
// [wcVol] -- volume letter (for eLockVol only)
|
|
//
|
|
// History: 16-Sep-96 dlee Created.
|
|
// 12-Aug-98 kitmanh Returned an SCODE
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
SCODE StopCiSvcWork( ECiSvcActionType type, WCHAR wcVol )
|
|
{
|
|
ciDebugOut(( DEB_ITRACE, "StopCiSvcWork is trying to get the lock\n" ));
|
|
CReleasableLock lock( g_mtxStartStop );
|
|
ciDebugOut(( DEB_ITRACE, "StopCiSvcWork got the lock\n" ));
|
|
|
|
// If the main SCM thread didn't get very far, this will be the case.
|
|
|
|
if ( 0 == g_pevtPauseContinue )
|
|
return S_OK;
|
|
|
|
g_pevtPauseContinue->Reset();
|
|
|
|
SCODE sc = StopFWCiSvcWork( type, &lock, g_pevtPauseContinue, wcVol );
|
|
|
|
lock.Release();
|
|
|
|
ciDebugOut(( DEB_ITRACE, "StopCiSvcWork.. After StopFwCiSvcWork has returned. type == %d\n", type ));
|
|
|
|
// Do not block if shutdown has intialized
|
|
if ( STATUS_TOO_LATE == sc )
|
|
return S_OK;
|
|
|
|
// block if pausing or continuing
|
|
if ( eNetStop != type )
|
|
{
|
|
ciDebugOut(( DEB_ITRACE, "StopCiSvcWork.. After StopFwCiSvcWork has returned. Block on g_pevtPauseContinue\n" ));
|
|
g_pevtPauseContinue->Wait();
|
|
lock.Request();
|
|
g_pevtPauseContinue->Reset();
|
|
lock.Release();
|
|
ciDebugOut(( DEB_ITRACE, "StopCiSvcWork.. After StopFwCiSvcWork has returned. Reusme on g_pevtPauseContinue\n" ));
|
|
}
|
|
|
|
return sc;
|
|
} //StopCiSvcWork
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::InternalWorkIdToDocName
|
|
//
|
|
// Synopsis: Translates a WorkId to a document name.
|
|
//
|
|
// Arguments: [workid] - [in] WorkId to translate
|
|
// [pICiCDocName] - [out] Will be filled in with the document
|
|
// name on output.
|
|
// [fAccurate] - Use the slow and accurate call
|
|
//
|
|
// Returns: S_OK if successfully converted.
|
|
// CI_E_BUFFERTOOSMALL if the buffer is not big enough
|
|
// Other error code.
|
|
//
|
|
// History: 31-Dec-1998 KyleP Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
SCODE CClientDocStore::InternalWorkIdToDocName( WORKID workid,
|
|
ICiCDocName * pICiCDocName,
|
|
BOOL fAccurate )
|
|
{
|
|
Win4Assert( 0 != pICiCDocName );
|
|
Win4Assert(_pCiCat);
|
|
|
|
CCiCDocName * pDocName = (CCiCDocName *) pICiCDocName;
|
|
|
|
CLowerFunnyPath funnyPath;
|
|
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
//
|
|
// We should avoid a memory copy here. Copy into the CCiCDocName
|
|
// directly. If this turns out to be a performance problem fix
|
|
// it. It isn't now.
|
|
//
|
|
|
|
unsigned cwc;
|
|
|
|
if ( fAccurate )
|
|
cwc = _pCiCat->WorkIdToAccuratePath( workid, funnyPath );
|
|
else
|
|
cwc = _pCiCat->WorkIdToPath( workid, funnyPath );
|
|
//
|
|
// CiCat::WorkIdToPath makes an EMPTY string when a doc
|
|
// is deleted.
|
|
//
|
|
if ( cwc > 1 )
|
|
{
|
|
// this is not a deleted file.
|
|
|
|
pDocName->SetPath( funnyPath.GetActualPath(), cwc );
|
|
}
|
|
else
|
|
{
|
|
// this is a deleted file.
|
|
|
|
sc = CI_S_WORKID_DELETED;
|
|
}
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: CClientDocStore::ClearNonStoragePropertiesForWid
|
|
//
|
|
// Synopsis: Clear non-storage properties from the property storage for wid
|
|
//
|
|
// Arguments: [wid] - workid
|
|
//
|
|
// Returns: S_OK if successful;
|
|
//
|
|
// History: 10-06-2000 kitmanh Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP CClientDocStore::ClearNonStoragePropertiesForWid( WORKID wid )
|
|
{
|
|
SCODE sc = S_OK;
|
|
|
|
TRY
|
|
{
|
|
_pCiCat->ClearNonStoragePropertiesForWid( wid );
|
|
}
|
|
CATCH( CException, e )
|
|
{
|
|
sc = e.GetErrorCode();
|
|
}
|
|
END_CATCH
|
|
|
|
return sc;
|
|
}
|
|
|