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

2450 lines
66 KiB
C++

///////////////////////////////////////////////////////////////////////////////
//
// Criteria.cpp
//
///////////////////////////////////////////////////////////////////////////////
#include <pch.hxx>
#include "criteria.h"
#include "ruleutil.h"
#include <xpcomm.h>
#include <flagconv.h>
#include <bodyutil.h>
#include <demand.h>
static const int CRIT_GROW = 16;
BOOL FMatchCritItem(CRIT_ITEM * pItem, LPCSTR pszAcct, MESSAGEINFO * pMsgInfo,
IMessageFolder * pFolder, IMimePropertySet * pIMPropSet,
IMimeMessage * pIMMsg, ULONG cbMsgSize);
BOOL FCritLoad_Account(IStream * pIStm, PROPVARIANT * ppropvar);
BOOL FCritSave_Account(IStream * pIStm, PROPVARIANT * ppropvar);
BOOL FCritLoad_Default(IStream * pIStm, PROPVARIANT * ppropvar);
BOOL FCritSave_Default(IStream * pIStm, PROPVARIANT * ppropvar);
DWORD DwGetFlagsFromMessage(IMimeMessage * pIMMsg);
///////////////////////////////////////////////////////////////////////////////
//
// HrCreateCriteria
//
// This creates a criteria container.
//
// ppICriteria - pointer to return the criteria container
//
// Returns: S_OK, on success
// E_OUTOFMEMORY, if can't create the Criteria object
//
///////////////////////////////////////////////////////////////////////////////
HRESULT HrCreateCriteria(IOECriteria ** ppICriteria)
{
COECriteria * pCriteria = NULL;
HRESULT hr = S_OK;
// Check the incoming params
if (NULL == ppICriteria)
{
hr = E_INVALIDARG;
goto exit;
}
// Initialize outgoing params
*ppICriteria = NULL;
// Create the rules manager object
pCriteria = new COECriteria;
if (NULL == pCriteria)
{
hr = E_OUTOFMEMORY;
goto exit;
}
// Get the rules manager interface
hr = pCriteria->QueryInterface(IID_IOECriteria, (void **) ppICriteria);
if (FAILED(hr))
{
goto exit;
}
pCriteria = NULL;
// Set the proper return value
hr = S_OK;
exit:
if (NULL != pCriteria)
{
delete pCriteria;
}
return hr;
}
COECriteria::~COECriteria()
{
AssertSz(m_cRef == 0, "Somebody still has a hold of us!!");
Reset();
}
STDMETHODIMP_(ULONG) COECriteria::AddRef()
{
return ::InterlockedIncrement(&m_cRef);
}
STDMETHODIMP_(ULONG) COECriteria::Release()
{
LONG cRef = 0;
cRef = ::InterlockedDecrement(&m_cRef);
if (0 == cRef)
{
delete this;
return cRef;
}
return cRef;
}
STDMETHODIMP COECriteria::QueryInterface(REFIID riid, void ** ppvObject)
{
HRESULT hr = S_OK;
// Check the incoming params
if (NULL == ppvObject)
{
hr = E_INVALIDARG;
goto exit;
}
// Initialize outgoing param
*ppvObject = NULL;
if ((riid == IID_IUnknown) || (riid == IID_IOECriteria))
{
*ppvObject = static_cast<IOECriteria *>(this);
}
else if ((riid == IID_IPersistStream) || (riid == IID_IPersist))
{
*ppvObject = static_cast<IPersistStream *>(this);
}
else
{
hr = E_NOINTERFACE;
goto exit;
}
reinterpret_cast<IUnknown *>(*ppvObject)->AddRef();
hr = S_OK;
exit:
return hr;
}
STDMETHODIMP COECriteria::Reset(void)
{
HRESULT hr = S_OK;
// See if there is something to do
if (0 == m_cItems)
{
Assert(NULL == m_rgItems);
hr = S_OK;
goto exit;
}
RuleUtil_HrFreeCriteriaItem(m_rgItems, m_cItems);
SafeMemFree(m_rgItems);
m_cItems = 0;
m_cItemsAlloc = 0;
exit:
return hr;
}
STDMETHODIMP COECriteria::GetState(DWORD * pdwState)
{
HRESULT hr = S_OK;
DWORD dwState = CRIT_STATE_NULL;
ULONG ulIndex = 0;
// Check incoming params
if (NULL == pdwState)
{
hr = E_INVALIDARG;
goto exit;
}
// Init the outgoing param
*pdwState = CRIT_STATE_NULL;
// See if there is something to do
if (0 == m_cItems)
{
Assert(NULL == m_rgItems);
hr = S_OK;
goto exit;
}
// Walk through the actions to figure out the state
for (ulIndex = 0; ulIndex < m_cItems; ulIndex++)
{
if ((CRIT_TYPE_SECURE == m_rgItems[ulIndex].type) ||
(CRIT_TYPE_BODY == m_rgItems[ulIndex].type) ||
(CRIT_TYPE_ATTACH == m_rgItems[ulIndex].type))
{
dwState = CRIT_STATE_ALL;
}
else if (CRIT_STATE_ALL != dwState)
{
dwState = CRIT_STATE_HEADER;
}
}
// Set the outgoing param
*pdwState = dwState;
// Set the proper return value
hr = S_OK;
exit:
return hr;
}
STDMETHODIMP COECriteria::GetCriteria(DWORD dwFlags, PCRIT_ITEM * ppItem, ULONG * pcItem)
{
HRESULT hr = S_OK;
CRIT_ITEM * pItemNew = NULL;
// Check incoming params
if ((NULL == ppItem) || (0 != dwFlags))
{
hr = E_INVALIDARG;
goto exit;
}
// Initialize the out params
*ppItem = NULL;
if (NULL != pcItem)
{
*pcItem = 0;
}
// If we don't have any criteria, then return
if (0 == m_cItems)
{
hr = E_FAIL;
goto exit;
}
// Allocate space for the criteria
hr = RuleUtil_HrDupCriteriaItem(m_rgItems, m_cItems, &pItemNew);
if (FAILED(hr))
{
goto exit;
}
// Save the criteria
*ppItem = pItemNew;
pItemNew = NULL;
if (NULL != pcItem)
{
*pcItem = m_cItems;
}
exit:
RuleUtil_HrFreeCriteriaItem(pItemNew, m_cItems);
SafeMemFree(pItemNew);
return hr;
}
STDMETHODIMP COECriteria::SetCriteria(DWORD dwFlags, CRIT_ITEM * pItem, ULONG cItem)
{
HRESULT hr = S_OK;
CRIT_ITEM * pItemNew = NULL;
// Check incoming params
if ((NULL == pItem) || (0 == cItem) || (0 != dwFlags))
{
hr = E_INVALIDARG;
goto exit;
}
// If we have any criteria already, then reset
if (0 != m_cItems)
{
Reset();
}
// Allocate space for the criteria
hr = RuleUtil_HrDupCriteriaItem(pItem, cItem, &pItemNew);
if (FAILED(hr))
{
goto exit;
}
// Save the criteria
m_rgItems = pItemNew;
pItemNew = NULL;
m_cItems = cItem;
m_cItemsAlloc = cItem;
exit:
RuleUtil_HrFreeCriteriaItem(pItemNew, cItem);
SafeMemFree(pItemNew);
return hr;
}
///////////////////////////////////////////////////////////////////////////////
//
// ValidateCriteria
//
// This verifies each of the criteria values
//
// Returns: S_OK, if the criteria were valid
// S_FALSE, otherwise
//
///////////////////////////////////////////////////////////////////////////////
STDMETHODIMP COECriteria::Validate(DWORD dwFlags)
{
HRESULT hr = S_OK;
ULONG ulIndex = 0;
LPSTR pszText = NULL;
IImnAccount * pAccount = NULL;
FOLDERINFO Folder = {0};
LPTSTR pszWalk = NULL;
ULONG cchText = 0;
RULEFOLDERDATA * prfdData = NULL;
// If we don't have any criteria, then we must be valid
if (0 == m_cItems)
{
hr = S_OK;
goto exit;
}
for (ulIndex = 0; ulIndex < m_cItems; ulIndex++)
{
if (0 != (m_rgItems[ulIndex].dwFlags & ~(CRIT_FLAG_INVERT | CRIT_FLAG_MULTIPLEAND)))
{
hr = S_FALSE;
goto exit;
}
switch(m_rgItems[ulIndex].type)
{
case CRIT_TYPE_NEWSGROUP:
if ((VT_BLOB != m_rgItems[ulIndex].propvar.vt) ||
(0 == m_rgItems[ulIndex].propvar.blob.cbSize))
{
hr = S_FALSE;
goto exit;
}
// Make life simpler
prfdData = (RULEFOLDERDATA *) (m_rgItems[ulIndex].propvar.blob.pBlobData);
// Validate the rule folder data
if (S_OK != RuleUtil_HrValidateRuleFolderData(prfdData))
{
hr = S_FALSE;
goto exit;
}
// Does the folder exist
hr = g_pStore->GetFolderInfo(prfdData->idFolder, &Folder);
if (FAILED(hr))
{
hr = S_FALSE;
goto exit;
}
// Are we subscribed?
if (0 == (Folder.dwFlags & FOLDER_SUBSCRIBED))
{
hr = S_FALSE;
goto exit;
}
break;
case CRIT_TYPE_ALL:
case CRIT_TYPE_JUNK:
case CRIT_TYPE_READ:
case CRIT_TYPE_REPLIES:
case CRIT_TYPE_DOWNLOADED:
case CRIT_TYPE_DELETED:
case CRIT_TYPE_ATTACH:
case CRIT_TYPE_FLAGGED:
if (VT_EMPTY != m_rgItems[ulIndex].propvar.vt)
{
hr = S_FALSE;
goto exit;
}
break;
case CRIT_TYPE_SUBJECT:
case CRIT_TYPE_BODY:
case CRIT_TYPE_TO:
case CRIT_TYPE_CC:
case CRIT_TYPE_TOORCC:
case CRIT_TYPE_FROM:
if ((VT_BLOB != m_rgItems[ulIndex].propvar.vt) ||
(0 == m_rgItems[ulIndex].propvar.blob.cbSize) ||
(NULL == m_rgItems[ulIndex].propvar.blob.pBlobData) ||
('\0' == m_rgItems[ulIndex].propvar.blob.pBlobData[0]))
{
hr = S_FALSE;
goto exit;
}
// Spin through each item making sure it is perfect
cchText = 0;
for (pszWalk = (LPTSTR) m_rgItems[ulIndex].propvar.blob.pBlobData;
'\0' != pszWalk[0]; pszWalk += lstrlen(pszWalk) + 1)
{
cchText += lstrlen(pszWalk) + 1;
}
// For the terminator
if ('\0' == pszWalk[0])
{
cchText++;
}
if ('\0' == pszWalk[1])
{
cchText++;
}
if (cchText != m_rgItems[ulIndex].propvar.blob.cbSize)
{
hr = S_FALSE;
goto exit;
}
break;
case CRIT_TYPE_SIZE:
case CRIT_TYPE_THREADSTATE:
case CRIT_TYPE_LINES:
case CRIT_TYPE_PRIORITY:
case CRIT_TYPE_AGE:
case CRIT_TYPE_SECURE:
if (VT_UI4 != m_rgItems[ulIndex].propvar.vt)
{
hr = S_FALSE;
goto exit;
}
break;
case CRIT_TYPE_ACCOUNT:
if ((VT_LPSTR != m_rgItems[ulIndex].propvar.vt) ||
(NULL == m_rgItems[ulIndex].propvar.pszVal))
{
hr = S_FALSE;
goto exit;
}
Assert(g_pAcctMan);
if (FAILED(g_pAcctMan->FindAccount(AP_ACCOUNT_ID, m_rgItems[ulIndex].propvar.pszVal, &pAccount)))
{
hr = S_FALSE;
goto exit;
}
SafeRelease(pAccount);
break;
case CRIT_TYPE_SENDER:
{
LPWSTR pwszText = NULL,
pwszVal = NULL;
if ((VT_LPSTR != m_rgItems[ulIndex].propvar.vt) ||
(NULL == m_rgItems[ulIndex].propvar.pszVal))
{
AssertSz(VT_LPWSTR != m_rgItems[ulIndex].propvar.vt, "We are getting UNICODE here.");
hr = S_FALSE;
goto exit;
}
// Verify the email string
pwszVal = PszToUnicode(CP_ACP, m_rgItems[ulIndex].propvar.pszVal);
if (!pwszVal)
{
hr = S_FALSE;
goto exit;
}
hr = RuleUtil_HrParseEmailString(pwszVal, 0, &pwszText, NULL);
MemFree(pwszText);
MemFree(pwszVal);
if (FAILED(hr))
{
hr = S_FALSE;
goto exit;
}
break;
}
default:
hr = S_FALSE;
goto exit;
break;
}
}
// If we got here, then we must be AOK
hr = S_OK;
exit:
g_pStore->FreeRecord(&Folder);
SafeRelease(pAccount);
return hr;
}
STDMETHODIMP COECriteria::AppendCriteria(DWORD dwFlags, CRIT_LOGIC logic,
CRIT_ITEM * pItem, ULONG cItem, ULONG * pcItemAppended)
{
HRESULT hr = S_OK;
CRIT_ITEM * pItemNew = NULL;
// Check incoming parameters
if ((0 != dwFlags) || (CRIT_LOGIC_NULL == logic) || (NULL == pItem) || (0 == cItem))
{
hr = E_INVALIDARG;
goto exit;
}
// Let's init our outgoing parameters
if (NULL != pcItemAppended)
{
*pcItemAppended = 0;
}
// Do we have to add more items?
if (m_cItems == m_cItemsAlloc)
{
hr = HrRealloc((LPVOID *) &m_rgItems, sizeof(CRIT_ITEM) * (m_cItemsAlloc + CRIT_GROW));
if (FAILED(hr))
{
goto exit;
}
ZeroMemory(m_rgItems + m_cItemsAlloc, sizeof(CRIT_ITEM) * CRIT_GROW);
m_cItemsAlloc += CRIT_GROW;
}
// Let's duplicate the items that need to be added
hr = RuleUtil_HrDupCriteriaItem(pItem, cItem, &pItemNew);
if (FAILED(hr))
{
goto exit;
}
// Let's add them to the criteria array
if (0 != m_cItems)
{
m_rgItems[m_cItems - 1].logic = logic;
}
CopyMemory(m_rgItems + m_cItems, pItemNew, sizeof(CRIT_ITEM) * cItem);
m_cItems += cItem;
// Set the proper outgoing parameter
if (NULL != pcItemAppended)
{
*pcItemAppended = cItem;
}
// Set the proper return value
hr = S_OK;
exit:
SafeMemFree(pItemNew);
return hr;
}
STDMETHODIMP COECriteria::MatchMessage(LPCSTR pszAcct, MESSAGEINFO * pMsgInfo, IMessageFolder * pFolder,
IMimePropertySet * pIMPropSet, IMimeMessage * pIMMsg, ULONG cbMsgSize)
{
HRESULT hr = S_OK;
ULONG ulIndex = 0;
BOOL fResult = FALSE;
BOOL fResultNew = FALSE;
CRIT_LOGIC logic;
// Check incoming parameters
if (((NULL == pMsgInfo) && (NULL == pIMPropSet)) || (0 == cbMsgSize))
{
hr = E_INVALIDARG;
goto exit;
}
// Let's go through the criteria and see if we match
fResult = FALSE;
logic = CRIT_LOGIC_OR;
for (ulIndex = 0; ulIndex < m_cItems; ulIndex++)
{
// Call matching function for this criteria item
fResultNew = FMatchCritItem(&(m_rgItems[ulIndex]), pszAcct, pMsgInfo, pFolder, pIMPropSet, pIMMsg, cbMsgSize);
// Slap it together with the old result
if (CRIT_LOGIC_AND == logic)
{
fResult = (fResult && fResultNew);
}
else
{
Assert(CRIT_LOGIC_OR == logic);
fResult = (fResult || fResultNew);
}
// Save of the next logical operation
logic = m_rgItems[ulIndex].logic;
}
// Set the proper return value
hr = (FALSE != fResult) ? S_OK : S_FALSE;
exit:
return hr;
}
///////////////////////////////////////////////////////////////////////////////
//
// LoadReg
//
// This loads in the criteria from the registry. It loads in the criteria
// order from the Order value. The string contains space delimitied values
// and each value contains the subkey name for each criterion. Each criterion
// is loaded in the order that is contained in the Order value. The criterion
// are loaded with the Criterion Type and Logical Operator. The Criterion Value
// Type is loaded if it exists. If a Criterion Value Type exists, then the
// corresponding Criterion Value is loaded in.
//
// pszRegPath - the path to load the criteria from
//
// Returns: S_OK, if the criteria was loaded without problems
// E_OUTOFMEMORY, if we couldn't allocate memory to hold the criteria
// E_FAIL, otherwise
//
///////////////////////////////////////////////////////////////////////////////
STDMETHODIMP COECriteria::LoadReg(LPCSTR pszRegPath)
{
HRESULT hr = S_OK;
LONG lErr = 0;
HKEY hkeyRoot = NULL;
ULONG cbData = 0;
LPSTR pszOrder = NULL;
ULONG cOrder = 0;
LPSTR pszWalk = NULL;
CRIT_ITEM * pItems = NULL;
LPSTR pszNext = NULL;
ULONG ulOrder = 0;
HKEY hkeyCriteria = NULL;
CRIT_TYPE typeCrit;
CRIT_LOGIC logicCrit;
PROPVARIANT propvar;
DWORD dwType = 0;
BYTE * pbData = NULL;
DWORD dwFlags = CRIT_FLAG_DEFAULT;
// Check incoming param
if (NULL == pszRegPath)
{
hr = E_INVALIDARG;
goto exit;
}
// Should we fail if we're already loaded?
AssertSz(0 == (m_dwState & CRIT_STATE_LOADED), "We're already loaded!!!");
// Open the reg key from the path
lErr = AthUserOpenKey(pszRegPath, KEY_ALL_ACCESS, &hkeyRoot);
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
// Get the criteria order
hr = RuleUtil_HrGetRegValue(hkeyRoot, c_szCriteriaOrder, NULL, (BYTE **) &pszOrder, &cbData);
if (FAILED(hr))
{
goto exit;
}
// Make sure we actually have something to load
if ('\0' == *pszOrder)
{
AssertSz(FALSE, "The order string for the criteria is mis-formatted in the registry");
hr = E_FAIL;
goto exit;
}
// Convert the criteria string to a more useful format
pszWalk = pszOrder;
cOrder = 1;
for (pszWalk = StrStr(pszOrder, g_szSpace); NULL != pszWalk; pszWalk = StrStr(pszWalk, g_szSpace))
{
// Terminate the order item
*pszWalk = '\0';
pszWalk++;
cOrder++;
}
// Allocate the space to hold all the criteria
hr = HrAlloc((void **) &pItems, cOrder * sizeof(CRIT_ITEM));
if (FAILED(hr))
{
goto exit;
}
// Initialize it to a known value
ZeroMemory(pItems, cOrder * sizeof(CRIT_ITEM));
// For each criteria in the order string
pszWalk = pszOrder;
for (ulOrder = 0, pszWalk = pszOrder; ulOrder < cOrder; ulOrder++, pszWalk += lstrlen(pszWalk) + 1)
{
// Open up the criteria reg key
lErr = RegOpenKeyEx(hkeyRoot, pszWalk, 0, KEY_READ, &hkeyCriteria);
if (ERROR_SUCCESS != lErr)
{
AssertSz(FALSE, "Part of the criteria is mis-formatted in the registry");
hr = E_FAIL;
goto exit;
}
// Get the criteria type
cbData = sizeof(typeCrit);
lErr = RegQueryValueEx(hkeyCriteria, c_szCriteriaType, 0, NULL,
(BYTE *) &(typeCrit), &cbData);
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
// Get the criteria logicial op
cbData = sizeof(logicCrit);
lErr = RegQueryValueEx(hkeyCriteria, c_szCriteriaLogic, 0, NULL,
(BYTE *) &(logicCrit), &cbData);
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
// Get the criteria flags
cbData = sizeof(dwFlags);
lErr = RegQueryValueEx(hkeyCriteria, c_szCriteriaFlags, 0, NULL,
(BYTE *) &(dwFlags), &cbData);
if ((ERROR_SUCCESS != lErr) && (ERROR_FILE_NOT_FOUND != lErr))
{
hr = E_FAIL;
goto exit;
}
// If it didn't exist then assign it to the default
if (ERROR_FILE_NOT_FOUND == lErr)
{
dwFlags = CRIT_FLAG_DEFAULT;
}
// Initialize the new space to a known value
ZeroMemory(&propvar, sizeof(propvar));
// Does a criteria value type exist
lErr = RegQueryValueEx(hkeyCriteria, c_szCriteriaValueType, 0, NULL, NULL, &cbData);
if ((ERROR_SUCCESS == lErr) && (0 != cbData))
{
// Load the criteria value in
cbData = sizeof(dwType);
lErr = RegQueryValueEx(hkeyCriteria, c_szCriteriaValueType, 0, NULL,
(BYTE *) &dwType, &cbData);
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
propvar.vt = (VARTYPE) dwType;
switch (propvar.vt)
{
case VT_UI4:
// Get the criteria value
cbData = sizeof(propvar.ulVal);
lErr = RegQueryValueEx(hkeyCriteria, c_szCriteriaValue, 0, NULL,
(BYTE * ) &(propvar.ulVal), &cbData);
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
break;
case VT_LPSTR:
case VT_BLOB:
// Get the criteria value
hr = RuleUtil_HrGetRegValue(hkeyCriteria, c_szCriteriaValue, NULL, (BYTE **) &pbData, &cbData);
if (FAILED(hr))
{
goto exit;
}
// Save the space so we can free it
if (VT_LPSTR == propvar.vt)
{
propvar.pszVal = (LPSTR) pbData;
}
else
{
propvar.blob.cbSize = cbData;
propvar.blob.pBlobData = pbData;
}
pbData = NULL;
break;
default:
AssertSz(FALSE, "Why are we loading in a invalid criteria type?");
hr = E_FAIL;
goto exit;
break;
}
}
// Save the value into the criteria array
pItems[ulOrder].type = typeCrit;
pItems[ulOrder].dwFlags = dwFlags;
pItems[ulOrder].logic = logicCrit;
pItems[ulOrder].propvar = propvar;
// Close the criteria
SideAssert(ERROR_SUCCESS == RegCloseKey(hkeyCriteria));
hkeyCriteria = NULL;
}
// Free up the current criteria
SafeMemFree(m_rgItems);
// Save the new values
m_rgItems = pItems;
pItems = NULL;
m_cItems = cOrder;
// Make sure we clear the dirty bit
m_dwState &= ~CRIT_STATE_DIRTY;
// Note that we have been loaded
m_dwState |= CRIT_STATE_LOADED;
// Set the return value
hr = S_OK;
exit:
SafeMemFree(pbData);
RuleUtil_HrFreeCriteriaItem(pItems, cOrder);
SafeMemFree(pItems);
SafeMemFree(pszOrder);
if (NULL != hkeyCriteria)
{
RegCloseKey(hkeyCriteria);
}
if (NULL != hkeyRoot)
{
RegCloseKey(hkeyRoot);
}
return hr;
}
STDMETHODIMP COECriteria::SaveReg(LPCSTR pszRegPath, BOOL fClearDirty)
{
HRESULT hr = S_OK;
LONG lErr = 0;
HKEY hkeyRoot = NULL;
DWORD dwDisp = 0;
LPSTR pszOrder = NULL;
ULONG ulIndex = 0;
CRIT_ITEM * pItem = NULL;
CHAR rgchTag[CCH_CRIT_ORDER];
HKEY hkeyCriteria = NULL;
ULONG cbData = 0;
BYTE * pbData = NULL;
// Check incoming param
if (NULL == pszRegPath)
{
hr = E_INVALIDARG;
goto exit;
}
// If there's nothing to save, then fail
if (NULL == m_rgItems)
{
hr = E_FAIL;
goto exit;
}
// Let's make sure we clear out the key first
AthUserDeleteKey(pszRegPath);
// Create the reg key from the path
lErr = AthUserCreateKey(pszRegPath, KEY_ALL_ACCESS, &hkeyRoot, &dwDisp);
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
Assert(REG_CREATED_NEW_KEY == dwDisp);
Assert(m_cItems < CRIT_COUNT_MAX);
// Allocate space to hold the order
DWORD cchSize = (m_cItems * CCH_CRIT_ORDER);
hr = HrAlloc((void **) &pszOrder, cchSize * sizeof(*pszOrder));
if (FAILED(hr))
{
goto exit;
}
pszOrder[0] = '\0';
// Write out each of the criteria
for (ulIndex = 0, pItem = m_rgItems; ulIndex < m_cItems; ulIndex++, pItem++)
{
// Get the new criteria tag
wnsprintf(rgchTag, ARRAYSIZE(rgchTag), "%03X", ulIndex);
// Add the new tag to the order
if (0 != ulIndex)
{
StrCatBuff(pszOrder, g_szSpace, cchSize);
}
StrCatBuff(pszOrder, rgchTag, cchSize);
// Create the new criteria
lErr = RegCreateKeyEx(hkeyRoot, rgchTag, 0, NULL,
REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hkeyCriteria, &dwDisp);
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
Assert(REG_CREATED_NEW_KEY == dwDisp);
// Write out the criteria type
lErr = RegSetValueEx(hkeyCriteria, c_szCriteriaType, 0, REG_DWORD,
(BYTE *) &(pItem->type), sizeof(pItem->type));
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
// Write out the criteria logicial op
lErr = RegSetValueEx(hkeyCriteria, c_szCriteriaLogic, 0, REG_DWORD,
(BYTE *) &(pItem->logic), sizeof(pItem->logic));
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
// Write out the criteria flags
lErr = RegSetValueEx(hkeyCriteria, c_szCriteriaFlags, 0, REG_DWORD,
(BYTE *) &(pItem->dwFlags), sizeof(pItem->dwFlags));
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
// Do we have a criteria value?
if (VT_EMPTY != pItem->propvar.vt)
{
// Write out the criteria value type
dwDisp = pItem->propvar.vt;
lErr = RegSetValueEx(hkeyCriteria, c_szCriteriaValueType, 0, REG_DWORD, (BYTE *) &dwDisp, sizeof(dwDisp));
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
// Write out the criteria value
switch (pItem->propvar.vt)
{
case VT_UI4:
dwDisp = REG_DWORD;
pbData = (BYTE * ) &(pItem->propvar.ulVal);
cbData = sizeof(pItem->propvar.ulVal);
break;
case VT_LPSTR:
dwDisp = REG_SZ;
pbData = (BYTE * ) (pItem->propvar.pszVal);
cbData = lstrlen(pItem->propvar.pszVal) + 1;
break;
case VT_BLOB:
dwDisp = REG_BINARY;
pbData = pItem->propvar.blob.pBlobData;
cbData = pItem->propvar.blob.cbSize;
break;
default:
AssertSz(FALSE, "Why are we trying to save in a invalid criteria type?");
hr = E_FAIL;
goto exit;
break;
}
// Write out the criteria value
lErr = RegSetValueEx(hkeyCriteria, c_szCriteriaValue, 0, dwDisp, pbData, cbData);
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
}
// Close the criteria
SideAssert(ERROR_SUCCESS == RegCloseKey(hkeyCriteria));
hkeyCriteria = NULL;
}
// Write out the order string.
lErr = RegSetValueEx(hkeyRoot, c_szCriteriaOrder, 0, REG_SZ,
(BYTE *) pszOrder, lstrlen(pszOrder) + 1);
if (ERROR_SUCCESS != lErr)
{
hr = E_FAIL;
goto exit;
}
// Should we clear the dirty bit?
if (FALSE != fClearDirty)
{
m_dwState &= ~CRIT_STATE_DIRTY;
}
// Set the return value
hr = S_OK;
exit:
if (NULL != hkeyCriteria)
{
RegCloseKey(hkeyCriteria);
}
SafeMemFree(pszOrder);
if (NULL != hkeyRoot)
{
RegCloseKey(hkeyRoot);
}
return hr;
}
STDMETHODIMP COECriteria::Clone(IOECriteria ** ppICriteria)
{
HRESULT hr = S_OK;
COECriteria * pCriteria = NULL;
// Check incoming params
if (NULL == ppICriteria)
{
hr = E_INVALIDARG;
goto exit;
}
// Initialize the outgoing params
*ppICriteria = NULL;
// Create a new criteria
pCriteria = new COECriteria;
if (NULL == pCriteria)
{
hr = E_OUTOFMEMORY;
goto exit;
}
// Copy over the list of criteria
hr = pCriteria->SetCriteria(0, m_rgItems, m_cItems);
if (FAILED(hr))
{
goto exit;
}
// Get the criteria interface
hr = pCriteria->QueryInterface(IID_IOECriteria, (void **) ppICriteria);
if (FAILED(hr))
{
goto exit;
}
pCriteria = NULL;
// Set the proper return value
hr = S_OK;
exit:
if (NULL != pCriteria)
{
delete pCriteria;
}
return hr;
}
STDMETHODIMP COECriteria::GetClassID(CLSID * pclsid)
{
HRESULT hr = S_OK;
if (NULL == pclsid)
{
hr = E_INVALIDARG;
goto exit;
}
*pclsid = CLSID_OECriteria;
// Set the proper return value
hr = S_OK;
exit:
return hr;
}
STDMETHODIMP COECriteria::IsDirty(void)
{
HRESULT hr = S_OK;
hr = (CRIT_STATE_DIRTY == (m_dwState & CRIT_STATE_DIRTY)) ? S_OK : S_FALSE;
return hr;
}
STDMETHODIMP COECriteria::Load(IStream * pStm)
{
HRESULT hr = S_OK;
ULONG cbData = 0;
ULONG cbRead = 0;
DWORD dwData = 0;
ULONG cItems = 0;
CRIT_ITEM * pItems = NULL;
ULONG ulIndex = 0;
CRIT_ITEM * pItem = NULL;
CRIT_TYPE typeCrit;
CRIT_LOGIC logicCrit;
DWORD dwFlags = CRIT_FLAG_DEFAULT;
PROPVARIANT propvar = {0};
BYTE * pbData = NULL;
// Check incoming param
if (NULL == pStm)
{
hr = E_INVALIDARG;
goto exit;
}
// Verify we have the correct version
hr = pStm->Read(&dwData, sizeof(dwData), &cbRead);
if (FAILED(hr))
{
goto exit;
}
if ((cbRead != sizeof(dwData)) || (dwData != CRIT_VERSION))
{
hr = E_FAIL;
goto exit;
}
// Get the number of criteria
hr = pStm->Read(&cItems, sizeof(cItems), &cbRead);
if (FAILED(hr))
{
goto exit;
}
if ((cbRead != sizeof(cItems)) || (0 == cItems))
{
hr = E_FAIL;
goto exit;
}
// Allocate space to hold all the criteria
hr = HrAlloc( (void **) &pItems, cItems * sizeof(*pItems));
if (FAILED(hr))
{
goto exit;
}
// Initialize the criteria to a known value
ZeroMemory(pItems, cItems * sizeof(*pItems));
// for each criteria
for (ulIndex = 0, pItem = pItems; ulIndex < cItems; ulIndex++, pItem++)
{
// Read in the criteria type
hr = pStm->Read(&typeCrit, sizeof(typeCrit), &cbRead);
if ((FAILED(hr)) || (cbRead != sizeof(typeCrit)))
{
hr = E_FAIL;
goto exit;
}
// Read in the criteria logical op
hr = pStm->Read(&logicCrit, sizeof(logicCrit), &cbRead);
if ((FAILED(hr)) || (cbRead != sizeof(logicCrit)))
{
hr = E_FAIL;
goto exit;
}
// Read in the criteria flags
hr = pStm->Read(&dwFlags, sizeof(dwFlags), &cbRead);
if ((FAILED(hr)) || (cbRead != sizeof(dwFlags)))
{
hr = E_FAIL;
goto exit;
}
// Read in the proper criteria value
switch(typeCrit)
{
case CRIT_TYPE_ACCOUNT:
if (FALSE == FCritLoad_Account(pStm, &propvar))
{
hr = E_FAIL;
goto exit;
}
break;
default:
if (FALSE == FCritLoad_Default(pStm, &propvar))
{
hr = E_FAIL;
goto exit;
}
break;
}
// Assign the values
pItem->type = typeCrit;
pItem->logic = logicCrit;
pItem->dwFlags = dwFlags;
pItem->propvar = propvar;
ZeroMemory(&propvar, sizeof(propvar));
}
// Free up the current criteria
SafeMemFree(m_rgItems);
// Save the new values
m_rgItems = pItems;
pItems = NULL;
m_cItems = cItems;
// Make sure we clear the dirty bit
m_dwState &= ~CRIT_STATE_DIRTY;
// Note that we have been loaded
m_dwState |= CRIT_STATE_LOADED;
// Set the return value
hr = S_OK;
exit:
RuleUtil_HrFreeCriteriaItem(pItems, cItems);
SafeMemFree(pItems);
PropVariantClear(&propvar);
return hr;
}
STDMETHODIMP COECriteria::Save(IStream * pStm, BOOL fClearDirty)
{
HRESULT hr = S_OK;
ULONG cbData = 0;
ULONG cbWritten = 0;
DWORD dwData = 0;
ULONG ulIndex = 0;
CRIT_ITEM * pItem = NULL;
BYTE * pbData = NULL;
// Check incoming param
if (NULL == pStm)
{
hr = E_INVALIDARG;
goto exit;
}
// Write out the version
dwData = CRIT_VERSION;
hr = pStm->Write(&dwData, sizeof(dwData), &cbWritten);
if (FAILED(hr))
{
goto exit;
}
Assert(cbWritten == sizeof(dwData));
// Write out the count of criteria
hr = pStm->Write(&m_cItems, sizeof(m_cItems), &cbWritten);
if (FAILED(hr))
{
goto exit;
}
Assert(cbWritten == sizeof(m_cItems));
// Loop through each of the criteria
for (ulIndex = 0, pItem = m_rgItems; ulIndex < m_cItems; ulIndex++, pItem++)
{
// Write out the criteria type
hr = pStm->Write(&(pItem->type), sizeof(pItem->type), &cbWritten);
if (FAILED(hr))
{
goto exit;
}
Assert(cbWritten == sizeof(pItem->type));
// Write out the criteria logical op
hr = pStm->Write(&(pItem->logic), sizeof(pItem->logic), &cbWritten);
if (FAILED(hr))
{
goto exit;
}
Assert(cbWritten == sizeof(pItem->logic));
// Write out the criteria flags
hr = pStm->Write(&(pItem->dwFlags), sizeof(pItem->dwFlags), &cbWritten);
if (FAILED(hr))
{
goto exit;
}
Assert(cbWritten == sizeof(pItem->dwFlags));
// Write out the proper criteria value
switch(pItem->type)
{
case CRIT_TYPE_ACCOUNT:
if (FALSE == FCritSave_Account(pStm, &(pItem->propvar)))
{
hr = E_FAIL;
goto exit;
}
break;
default:
if (FALSE == FCritSave_Default(pStm, &(pItem->propvar)))
{
hr = E_FAIL;
goto exit;
}
break;
}
}
// Should we clear out the dirty bit
if (FALSE != fClearDirty)
{
m_dwState &= ~CRIT_STATE_DIRTY;
}
// Set the return value
hr = S_OK;
exit:
return hr;
}
BOOL CritFunc_Query(CRIT_ITEM * pItem, LPCSTR pszQuery, IMimePropertySet * pIMPropSet);
BOOL CritFunc_Text(CRIT_ITEM * pItem, LPSTR pszText);
BOOL CritFunc_Sender(CRIT_ITEM * pItem, LPSTR pszAddr);
BOOL CritFunc_Priority(CRIT_ITEM * pItem, WORD wPriority);
BOOL CritFunc_Secure(CRIT_ITEM * pItem, DWORD dwFlags);
BOOL CritFunc_Age(CRIT_ITEM * pItem, FILETIME * pftSent);
BOOL CritFunc_Body(CRIT_ITEM * pItem, IMimeMessage * pIMMsg);
BOOL _FMatchBlobString(CRIT_ITEM * pItem, LPSTR pszText);
BOOL _FQueryBlobString(CRIT_ITEM * pItem, LPCSTR pszQuery, IMimePropertySet * pIMPropSet);
BOOL FMatchCritItem(CRIT_ITEM * pItem, LPCSTR pszAcct, MESSAGEINFO * pMsgInfo,
IMessageFolder * pFolder, IMimePropertySet * pIMPropSet,
IMimeMessage * pIMMsg, ULONG cbMsgSize)
{
BOOL fRet = FALSE;
ULONG ulIndex = 0;
PROPVARIANT propvar = {0};
ADDRESSLIST addrList = {0};
FOLDERID idFolder = 0;
RULEFOLDERDATA * prfdData = NULL;
Assert((NULL != pItem) && ((NULL != pMsgInfo) || (NULL != pIMPropSet)) && (0 != cbMsgSize))
switch (pItem->type)
{
case CRIT_TYPE_ALL:
Assert(VT_EMPTY == pItem->propvar.vt);
fRet = TRUE;
break;
case CRIT_TYPE_ACCOUNT:
Assert(VT_LPSTR == pItem->propvar.vt);
fRet = FALSE;
if ((NULL != pszAcct) && (NULL != pItem->propvar.pszVal))
{
fRet = (0 == lstrcmpi(pItem->propvar.pszVal, pszAcct));
}
break;
case CRIT_TYPE_NEWSGROUP:
Assert(VT_BLOB == pItem->propvar.vt);
fRet = FALSE;
if ((NULL != pFolder) && (0 != pItem->propvar.blob.cbSize))
{
// Make life simpler
prfdData = (RULEFOLDERDATA *) (pItem->propvar.blob.pBlobData);
// Validate the rule folder data
if (S_OK != RuleUtil_HrValidateRuleFolderData(prfdData))
{
fRet = FALSE;
}
else if (SUCCEEDED(pFolder->GetFolderId(&idFolder)))
{
fRet = (idFolder == prfdData->idFolder);
}
}
break;
case CRIT_TYPE_SIZE:
Assert(VT_UI4 == pItem->propvar.vt);
// Set the size of the message to Kilobytes
cbMsgSize = cbMsgSize / 1024;
fRet = (cbMsgSize > pItem->propvar.ulVal);
break;
case CRIT_TYPE_LINES:
Assert(VT_UI4 == pItem->propvar.vt);
fRet = FALSE;
if (NULL != pMsgInfo)
{
fRet = (pMsgInfo->cLines > pItem->propvar.ulVal);
}
break;
case CRIT_TYPE_AGE:
Assert(VT_UI4 == pItem->propvar.vt);
fRet = FALSE;
if (NULL != pMsgInfo)
{
fRet = CritFunc_Age(pItem, &(pMsgInfo->ftSent));
}
else if ((NULL != pIMPropSet) && (SUCCEEDED(pIMPropSet->GetProp(PIDTOSTR(PID_ATT_SENTTIME), 0, &propvar))))
{
fRet = CritFunc_Age(pItem, &(propvar.filetime));
}
break;
case CRIT_TYPE_ATTACH:
Assert(VT_EMPTY == pItem->propvar.vt);
fRet = TRUE;
if (NULL != pMsgInfo)
{
fRet = (0 != (pMsgInfo->dwFlags & ARF_HASATTACH));
}
else if (NULL != pIMMsg)
{
fRet = (0 != (DwGetFlagsFromMessage(pIMMsg) & ARF_HASATTACH));
}
break;
case CRIT_TYPE_PRIORITY:
Assert(VT_UI4 == pItem->propvar.vt);
fRet = FALSE;
if (NULL != pMsgInfo)
{
fRet = CritFunc_Priority(pItem, pMsgInfo->wPriority);
}
else if ((NULL != pIMPropSet) && (SUCCEEDED(pIMPropSet->GetProp(PIDTOSTR(PID_ATT_PRIORITY), 0, &propvar))))
{
fRet = CritFunc_Priority(pItem, (WORD) (propvar.ulVal));
}
break;
case CRIT_TYPE_SECURE:
Assert(VT_UI4 == pItem->propvar.vt);
fRet = FALSE;
if (NULL != pMsgInfo)
{
fRet = CritFunc_Secure(pItem, pMsgInfo->dwFlags);
}
else if (NULL != pIMMsg)
{
fRet = CritFunc_Secure(pItem, DwGetFlagsFromMessage(pIMMsg));
}
break;
case CRIT_TYPE_TOORCC:
Assert(VT_BLOB == pItem->propvar.vt);
fRet = FALSE;
if (NULL != pIMPropSet)
{
fRet = _FQueryBlobString(pItem, PIDTOSTR(PID_HDR_TO), pIMPropSet);
if (((0 != (pItem->dwFlags & CRIT_FLAG_INVERT)) && (FALSE != fRet)) ||
((0 == (pItem->dwFlags & CRIT_FLAG_INVERT)) && (FALSE == fRet)))
{
fRet = _FQueryBlobString(pItem, PIDTOSTR(PID_HDR_CC), pIMPropSet);
}
}
break;
case CRIT_TYPE_SENDER:
Assert(VT_LPSTR == pItem->propvar.vt);
fRet = FALSE;
if ((NULL == pItem->propvar.pszVal) || ('\0' == pItem->propvar.pszVal[0]))
{
Assert(FALSE);
}
else if (S_OK == RuleUtil_HrMatchSender(pItem->propvar.pszVal, pMsgInfo, pIMMsg, pIMPropSet))
{
fRet = TRUE;
}
break;
case CRIT_TYPE_SUBJECT:
Assert(VT_BLOB == pItem->propvar.vt);
fRet = FALSE;
if ((0 == pItem->propvar.blob.cbSize) ||
(NULL == pItem->propvar.blob.pBlobData) ||
('\0' == pItem->propvar.blob.pBlobData[0]))
{
Assert(FALSE);
fRet = FALSE;
}
else if ((NULL != pMsgInfo) && (NULL != pMsgInfo->pszSubject))
{
fRet = _FMatchBlobString(pItem, pMsgInfo->pszSubject);
}
else
{
fRet = _FQueryBlobString(pItem, PIDTOSTR(PID_HDR_SUBJECT), pIMPropSet);
}
break;
case CRIT_TYPE_BODY:
Assert(VT_BLOB == pItem->propvar.vt);
fRet = FALSE;
if ((0 == pItem->propvar.blob.cbSize) ||
(NULL == pItem->propvar.blob.pBlobData) ||
('\0' == pItem->propvar.blob.pBlobData[0]))
{
Assert(FALSE);
fRet = FALSE;
}
else if (NULL != pIMMsg)
{
fRet = CritFunc_Body(pItem, pIMMsg);
}
break;
case CRIT_TYPE_FROM:
Assert(VT_BLOB == pItem->propvar.vt);
fRet = FALSE;
if ((0 == pItem->propvar.blob.cbSize) ||
(NULL == pItem->propvar.blob.pBlobData) ||
('\0' == pItem->propvar.blob.pBlobData[0]))
{
Assert(FALSE);
fRet = FALSE;
}
else if ((NULL != pMsgInfo) && (NULL != pMsgInfo->pszFromHeader))
{
fRet = _FMatchBlobString(pItem, pMsgInfo->pszFromHeader);
}
else
{
fRet = _FQueryBlobString(pItem, PIDTOSTR(PID_HDR_FROM), pIMPropSet);
}
break;
case CRIT_TYPE_TO:
Assert(VT_BLOB == pItem->propvar.vt);
fRet = FALSE;
if ((0 == pItem->propvar.blob.cbSize) ||
(NULL == pItem->propvar.blob.pBlobData) ||
('\0' == pItem->propvar.blob.pBlobData[0]))
{
Assert(FALSE);
fRet = FALSE;
}
else
{
fRet = _FQueryBlobString(pItem, PIDTOSTR(PID_HDR_TO), pIMPropSet);
}
break;
case CRIT_TYPE_CC:
Assert(VT_BLOB == pItem->propvar.vt);
fRet = FALSE;
if ((0 == pItem->propvar.blob.cbSize) ||
(NULL == pItem->propvar.blob.pBlobData) ||
('\0' == pItem->propvar.blob.pBlobData[0]))
{
Assert(FALSE);
fRet = FALSE;
}
else
{
fRet = _FQueryBlobString(pItem, PIDTOSTR(PID_HDR_CC), pIMPropSet);
}
break;
default:
fRet = FALSE;
break;
}
PropVariantClear(&propvar);
return fRet;
}
BOOL CritFunc_Query(CRIT_ITEM * pItem, LPCSTR pszQuery, IMimePropertySet * pIMPropSet)
{
BOOL fRet = FALSE;
LPSTR pszWalk = NULL;
LPSTR pszAddr = NULL;
LPSTR pszTerm = NULL;
HRESULT hr = S_OK;
if (NULL == pIMPropSet)
{
fRet = FALSE;
goto exit;
}
// Dup the string
pszAddr = PszDupA(pItem->propvar.pszVal);
if (NULL == pszAddr)
{
fRet = FALSE;
goto exit;
}
pszWalk = pszAddr;
pszTerm = pszWalk;
while (NULL != pszTerm)
{
pszTerm = StrStr(pszWalk, g_szComma);
if (NULL != pszTerm)
{
pszTerm[0] = '\0';
}
fRet = (S_OK == pIMPropSet->QueryProp(pszQuery, pszWalk, TRUE, FALSE));
if (FALSE == fRet)
{
break;
}
pszWalk = pszWalk + lstrlen(pszWalk) + 1;
}
exit:
SafeMemFree(pszAddr);
return fRet;
}
BOOL CritFunc_Priority(CRIT_ITEM * pItem, WORD wPriority)
{
BOOL fRet = FALSE;
Assert(NULL != pItem);
Assert(VT_UI4 == pItem->propvar.vt);
if (CRIT_DATA_HIPRI == pItem->propvar.ulVal)
{
fRet = (wPriority == (WORD) IMSG_PRI_HIGH);
}
else if (CRIT_DATA_LOPRI == pItem->propvar.ulVal)
{
fRet = (wPriority == (WORD) IMSG_PRI_LOW);
}
else
{
fRet = (wPriority == (WORD) IMSG_PRI_NORMAL);
}
return fRet;
}
BOOL CritFunc_Secure(CRIT_ITEM * pItem, DWORD dwFlags)
{
BOOL fRet = FALSE;
Assert(NULL != pItem);
Assert(VT_UI4 == pItem->propvar.vt);
// Should we be checking signed messages
if (0 != (pItem->propvar.ulVal & CRIT_DATA_SIGNEDSECURE))
{
fRet = (0 != (dwFlags & ARF_SIGNED));
}
else if (0 != (pItem->propvar.ulVal & CRIT_DATA_ENCRYPTSECURE))
// Should we be checking encrypted messages
{
fRet = (0 != (dwFlags & ARF_ENCRYPTED));
}
else
{
fRet = (0 == (dwFlags & (ARF_ENCRYPTED | ARF_SIGNED)));
}
return fRet;
}
BOOL CritFunc_Age(CRIT_ITEM * pItem, FILETIME * pftSent)
{
BOOL fRet = FALSE;
SYSTEMTIME sysTime = {0};
FILETIME ftTime = {0};
ULONG ulSeconds;
Assert(VT_UI4 == pItem->propvar.vt);
if ((NULL == pftSent) || ((0 == pftSent->dwLowDateTime) && (0 == pftSent->dwHighDateTime)))
{
fRet = FALSE;
goto exit;
}
// Get the current time
GetSystemTime(&sysTime);
SystemTimeToFileTime(&sysTime, &ftTime);
ulSeconds = UlDateDiff(pftSent, &ftTime);
fRet = ((ulSeconds / SECONDS_INA_DAY) > pItem->propvar.ulVal);
exit:
return fRet;
}
BOOL CritFunc_Sender(CRIT_ITEM * pItem, LPSTR pszAddr)
{
BOOL fRet = FALSE;
ULONG cchVal = 0;
ULONG cchEmail = 0;
CHAR chTest = 0;
Assert(VT_LPSTR == pItem->propvar.vt);
// Check to make sure that there's something to match
if ((NULL == pszAddr) || ('\0' == pszAddr[0]))
{
fRet = FALSE;
goto exit;
}
// Check to see if it is an address
if (NULL != StrStr(pItem->propvar.pszVal, "@"))
{
fRet = (0 == lstrcmpi(pItem->propvar.pszVal, pszAddr));
}
else
{
cchVal = lstrlen(pItem->propvar.pszVal);
cchEmail = lstrlen(pszAddr);
if (cchVal <= cchEmail)
{
fRet = (0 == lstrcmpi(pItem->propvar.pszVal, pszAddr + (cchEmail - cchVal)));
if ((FALSE != fRet) && (cchVal != cchEmail))
{
chTest = *(pszAddr + (cchEmail - cchVal - 1));
if (('@' != chTest) && ('.' != chTest))
{
fRet = FALSE;
}
}
}
}
exit:
return fRet;
}
BOOL _FMatchBlobString(CRIT_ITEM * pItem, LPSTR pszText)
{
BOOL fRet = FALSE;
LPSTR pszWalk = NULL;
// Walk each of the strings looking for a match
for (pszWalk = (LPSTR) (pItem->propvar.blob.pBlobData); '\0' != pszWalk[0];
pszWalk = pszWalk + lstrlen(pszWalk) + 1)
{
// Do the comparison
fRet = (NULL != StrStrI(pszText, pszWalk));
// If we are doing an AND of the multiple criteria
if (0 != (pItem->dwFlags & CRIT_FLAG_MULTIPLEAND))
{
// if we don't have a match, then we're done
if (FALSE == fRet)
{
break;
}
}
else
{
// if we do have a match, then we're done
if (FALSE != fRet)
{
break;
}
}
}
// Invert the result if needed
if (0 != (pItem->dwFlags & CRIT_FLAG_INVERT))
{
fRet = !fRet;
}
return fRet;
}
BOOL _FQueryBlobString(CRIT_ITEM * pItem, LPCSTR pszQuery, IMimePropertySet * pIMPropSet)
{
BOOL fRet = FALSE;
LPSTR pszWalk = NULL;
if (NULL == pIMPropSet)
{
fRet = FALSE;
goto exit;
}
// Walk each of the strings looking for a match
for (pszWalk = (LPSTR) (pItem->propvar.blob.pBlobData); '\0' != pszWalk[0];
pszWalk = pszWalk + lstrlen(pszWalk) + 1)
{
// Do the comparison
fRet = (S_OK == pIMPropSet->QueryProp(pszQuery, pszWalk, TRUE, FALSE));
// If we are doing an AND of the multiple criteria
if (0 != (pItem->dwFlags & CRIT_FLAG_MULTIPLEAND))
{
// if we don't have a match, then we're done
if (FALSE == fRet)
{
break;
}
}
else
{
// if we do have a match, then we're done
if (FALSE != fRet)
{
break;
}
}
}
// Invert the result if needed
if (0 != (pItem->dwFlags & CRIT_FLAG_INVERT))
{
fRet = !fRet;
}
exit:
return fRet;
}
BOOL CritFunc_Text(CRIT_ITEM * pItem, LPSTR pszText)
{
BOOL fRet = FALSE;
LPSTR pszWalk = NULL;
LPSTR pszAddr = NULL;
LPSTR pszTerm = NULL;
// Dup the string
pszAddr = PszDupA(pItem->propvar.pszVal);
if (NULL == pszAddr)
{
fRet = FALSE;
goto exit;
}
pszWalk = pszAddr;
pszTerm = pszWalk;
while (NULL != pszTerm)
{
pszTerm = StrStr(pszWalk, g_szComma);
if (NULL != pszTerm)
{
pszTerm[0] = '\0';
}
fRet = (NULL != StrStrI(pszText, pszWalk));
if (FALSE == fRet)
{
break;
}
pszWalk = pszWalk + lstrlen(pszWalk) + 1;
}
exit:
SafeMemFree(pszAddr);
return fRet;
}
BOOL CritFunc_Body(CRIT_ITEM * pItem, IMimeMessage * pIMMsg)
{
BOOL fRet = FALSE;
LPSTR pszWalk = NULL;
IStream * pStream = NULL;
IStream * pStreamHtml = NULL;
pszWalk = (LPTSTR) (pItem->propvar.blob.pBlobData);
if (NULL == pszWalk)
{
fRet = FALSE;
goto exit;
}
// Try to Get the Plain Text Stream
if (FAILED(pIMMsg->GetTextBody(TXT_PLAIN, IET_DECODED, &pStream, NULL)))
{
// Try to get the HTML stream and convert it to text...
if (SUCCEEDED(pIMMsg->GetTextBody(TXT_HTML, IET_DECODED, &pStreamHtml, NULL)))
{
if (FAILED(HrConvertHTMLToPlainText(pStreamHtml, &pStream, CF_TEXT)))
{
fRet = FALSE;
goto exit;
}
}
}
if (NULL == pStream)
{
fRet = FALSE;
goto exit;
}
for (; '\0' != pszWalk[0]; pszWalk += lstrlen(pszWalk) + 1)
{
fRet = StreamSubStringMatch(pStream, pszWalk);
// If we are doing an AND of the multiple criteria
if (0 != (pItem->dwFlags & CRIT_FLAG_MULTIPLEAND))
{
// if we don't have a match, then we're done
if (FALSE == fRet)
{
break;
}
}
else
{
// if we do have a match, then we're done
if (FALSE != fRet)
{
break;
}
}
}
// Invert the result if needed
if (0 != (pItem->dwFlags & CRIT_FLAG_INVERT))
{
fRet = !fRet;
}
exit:
SafeRelease(pStreamHtml);
SafeRelease(pStream);
return fRet;
}
BOOL FCrit_GetAcctInfo(DWORD dwServerTypes, DWORD * pdwServerType, DWORD * pdwPropTag)
{
BOOL fRet = FALSE;
Assert((NULL != pdwServerType) && (NULL != pdwPropTag));
// Figure out the type of the account
// and the server property
if (0 != (dwServerTypes & SRV_NNTP))
{
*pdwServerType = SRV_NNTP;
*pdwPropTag = AP_NNTP_SERVER;
}
else if (0 != (dwServerTypes & SRV_IMAP))
{
*pdwServerType = SRV_IMAP;
*pdwPropTag = AP_IMAP_SERVER;
}
else if (0 != (dwServerTypes & SRV_POP3))
{
*pdwServerType = SRV_POP3;
*pdwPropTag = AP_POP3_SERVER;
}
else if (0 != (dwServerTypes & SRV_HTTPMAIL))
{
*pdwServerType = SRV_HTTPMAIL;
*pdwPropTag = AP_HTTPMAIL_SERVER;
}
else
{
Assert(FALSE);
fRet = FALSE;
goto exit;
}
// Set the return value
fRet = TRUE;
exit:
return fRet;
}
BOOL FCritLoad_Account(IStream * pIStm, PROPVARIANT * ppropvar)
{
BOOL fRet = FALSE;
HRESULT hr = S_OK;
DWORD dwData = 0;
DWORD dwPropTag = 0;
ULONG cbRead = 0;
BYTE * pbData = NULL;
ULONG cbData = 0;
IImnEnumAccounts * pIEnumAcct = NULL;
IImnAccount * pAccount = NULL;
CHAR szAccount[CCHMAX_SERVER_NAME];
LPSTR pszAcct = NULL;
BOOL fFound = FALSE;
// Check the incoming params
if ((NULL == pIStm) || (NULL == ppropvar))
{
fRet = FALSE;
goto exit;
}
// Initialize the outgoing param
ZeroMemory(ppropvar, sizeof(*ppropvar));
// Read in the account server type
hr = pIStm->Read(&dwData, sizeof(dwData), &cbRead);
if ((FAILED(hr)) || (cbRead != sizeof(dwData)))
{
fRet = FALSE;
goto exit;
}
// Figure out the type of the account
// and the server property
fRet = FCrit_GetAcctInfo(dwData, &dwData, &dwPropTag);
if (FALSE == fRet)
{
goto exit;
}
// Get the size of the server name
hr = pIStm->Read(&cbData, sizeof(cbData), &cbRead);
if ((FAILED(hr)) || (cbRead != sizeof(cbData)))
{
fRet = FALSE;
goto exit;
}
// Allocate the space to hold the server name
hr = HrAlloc((VOID **) &pbData, cbData);
if (FAILED(hr))
{
fRet = FALSE;
goto exit;
}
// Read in the server name
hr = pIStm->Read(pbData, cbData, &cbRead);
if ((FAILED(hr)) || (cbRead != cbData))
{
fRet = FALSE;
goto exit;
}
// Get an account enumerator
Assert(g_pAcctMan);
if (FAILED(g_pAcctMan->Enumerate(dwData, &pIEnumAcct)))
{
fRet = FALSE;
goto exit;
}
// Search each account for the server name
while(SUCCEEDED(pIEnumAcct->GetNext(&pAccount)))
{
// We can get back NULL accounts
if (NULL == pAccount)
{
break;
}
// Get the server name
if (FAILED(pAccount->GetPropSz(dwPropTag, szAccount, sizeof(szAccount))))
{
SafeRelease(pAccount);
continue;
}
// Do we have a match?
if (0 == lstrcmpi(szAccount, (LPSTR) pbData))
{
fFound = TRUE;
break;
}
// We have a match
// Release it
SafeRelease(pAccount);
}
// Did we find anything?
if (FALSE == fFound)
{
fRet = FALSE;
goto exit;
}
// Get the account
if (FAILED(pAccount->GetPropSz(AP_ACCOUNT_ID, szAccount, sizeof(szAccount))))
{
fRet = FALSE;
goto exit;
}
// Save off the account ID
pszAcct = PszDupA(szAccount);
if (NULL == pszAcct)
{
fRet = FALSE;
goto exit;
}
// Set the outgoing param
ppropvar->vt = VT_LPSTR;
ppropvar->pszVal = pszAcct;
pszAcct = NULL;
// Set the return value
fRet = TRUE;
exit:
SafeMemFree(pszAcct);
SafeRelease(pAccount);
SafeRelease(pIEnumAcct);
SafeMemFree(pbData);
return fRet;
}
BOOL FCritSave_Account(IStream * pIStm, PROPVARIANT * ppropvar)
{
BOOL fRet = FALSE;
HRESULT hr = S_OK;
IImnAccount * pAccount = NULL;
DWORD dwServerTypes = 0;
DWORD dwPropTag = 0;
LPSTR pszServer = NULL;
ULONG cbWritten = 0;
ULONG cbData = 0;
// Check the incoming params
if ((NULL == pIStm) || (NULL == ppropvar))
{
fRet = FALSE;
goto exit;
}
Assert(g_pAcctMan);
if (FAILED(g_pAcctMan->FindAccount(AP_ACCOUNT_ID, ppropvar->pszVal, &pAccount)))
{
fRet = FALSE;
goto exit;
}
// Get the server type
if (FAILED(pAccount->GetServerTypes(&dwServerTypes)))
{
fRet = FALSE;
goto exit;
}
// Figure out the type of the account
// and the server property
fRet = FCrit_GetAcctInfo(dwServerTypes, &dwServerTypes, &dwPropTag);
if (FALSE == fRet)
{
goto exit;
}
// Allocate space to hold the server name
if (FAILED(HrAlloc((void **) &pszServer, CCHMAX_SERVER_NAME + 1)))
{
fRet = FALSE;
goto exit;
}
// Get the server name
if (FAILED(pAccount->GetPropSz(dwPropTag, pszServer, CCHMAX_SERVER_NAME)))
{
fRet = FALSE;
goto exit;
}
// Write out the server type
hr = pIStm->Write(&(dwServerTypes), sizeof(dwServerTypes), &cbWritten);
if (FAILED(hr))
{
fRet = FALSE;
goto exit;
}
Assert(cbWritten == sizeof(dwServerTypes));
// Write out the count of chars in the name
cbData = lstrlen(pszServer) + 1;
hr = pIStm->Write(&cbData, sizeof(cbData), &cbWritten);
if (FAILED(hr))
{
fRet = TRUE;
goto exit;
}
Assert(cbWritten == sizeof(cbData));
// Write out the server name
hr = pIStm->Write((BYTE *) pszServer, cbData, &cbWritten);
if (FAILED(hr))
{
fRet = TRUE;
goto exit;
}
Assert(cbWritten == cbData);
// Set the return value
fRet = TRUE;
exit:
SafeMemFree(pszServer);
SafeRelease(pAccount);
return fRet;
}
BOOL FCritLoad_Default(IStream * pIStm, PROPVARIANT * ppropvar)
{
BOOL fRet = FALSE;
HRESULT hr = S_OK;
DWORD dwData = 0;
ULONG cbRead = 0;
BYTE * pbData = NULL;
ULONG cbData = 0;
// Check the incoming params
if ((NULL == pIStm) || (NULL == ppropvar))
{
fRet = FALSE;
goto exit;
}
// Initialize the outgoing param
ZeroMemory(ppropvar, sizeof(*ppropvar));
// Read in the criteria value type
hr = pIStm->Read(&dwData, sizeof(dwData), &cbRead);
if ((FAILED(hr)) || (cbRead != sizeof(dwData)))
{
fRet = FALSE;
goto exit;
}
// Do we have any more data to get
if (dwData != VT_EMPTY)
{
ppropvar->vt = (VARTYPE) dwData;
// Get the size of the criteria value
hr = pIStm->Read(&cbData, sizeof(cbData), &cbRead);
if ((FAILED(hr)) || (cbRead != sizeof(cbData)))
{
fRet = FALSE;
goto exit;
}
// Allocate space to hold the criteria value data
switch (ppropvar->vt)
{
case VT_UI4:
pbData = (BYTE * ) &(ppropvar->ulVal);
break;
case VT_BLOB:
case VT_LPSTR:
// Allocate the space to hold the data
hr = HrAlloc((void **) &pbData, cbData);
if (FAILED(hr))
{
fRet = FALSE;
goto exit;
}
// Make sure we don't lose the allocated memory
if (VT_LPSTR == ppropvar->vt)
{
ppropvar->pszVal = (LPSTR) pbData;
}
else
{
ppropvar->blob.cbSize = cbData;
ppropvar->blob.pBlobData = pbData;
}
break;
default:
AssertSz(FALSE, "Why are we trying to save in a invalid criteria type?");
fRet = FALSE;
goto exit;
break;
}
// Read in the criteria value
hr = pIStm->Read(pbData, cbData, &cbRead);
if ((FAILED(hr)) || (cbRead != cbData))
{
fRet = FALSE;
goto exit;
}
}
// Set the return value
fRet = TRUE;
exit:
return fRet;
}
BOOL FCritSave_Default(IStream * pIStm, PROPVARIANT * ppropvar)
{
BOOL fRet = FALSE;
HRESULT hr = S_OK;
DWORD dwData = 0;
ULONG cbWritten = 0;
BYTE * pbData = NULL;
ULONG cbData = 0;
// Check the incoming params
if ((NULL == pIStm) || (NULL == ppropvar))
{
fRet = FALSE;
goto exit;
}
// Write out the value type
dwData = ppropvar->vt;
hr = pIStm->Write(&(dwData), sizeof(dwData), &cbWritten);
if (FAILED(hr))
{
fRet = FALSE;
goto exit;
}
Assert(cbWritten == sizeof(dwData));
// We don't have to save out the criteria value
// if we don't have one
if (VT_EMPTY == ppropvar->vt)
{
fRet = TRUE;
goto exit;
}
// Figure out the size of the criteria value
switch (ppropvar->vt)
{
case VT_UI4:
pbData = (BYTE * ) &(ppropvar->ulVal);
cbData = sizeof(ppropvar->ulVal);
break;
case VT_LPSTR:
pbData = (BYTE * ) (ppropvar->pszVal);
cbData = lstrlen(ppropvar->pszVal) + 1;
break;
case VT_BLOB:
pbData = ppropvar->blob.pBlobData;
cbData = ppropvar->blob.cbSize;
break;
default:
AssertSz(FALSE, "Why are we trying to save in a invalid criteria type?");
fRet = FALSE;
goto exit;
break;
}
// Write out the criteria value size
hr = pIStm->Write(&cbData, sizeof(cbData), &cbWritten);
if (FAILED(hr))
{
fRet = TRUE;
goto exit;
}
Assert(cbWritten == sizeof(cbData));
// Write out the criteria value
hr = pIStm->Write(pbData, cbData, &cbWritten);
if (FAILED(hr))
{
fRet = TRUE;
goto exit;
}
Assert(cbWritten == cbData);
// Set the return value
fRet = TRUE;
exit:
return fRet;
}
DWORD DwGetFlagsFromMessage(IMimeMessage * pIMMsg)
{
DWORD dwFlags = 0;
DWORD dwImf = 0;
Assert(NULL != pIMMsg);
if (SUCCEEDED(pIMMsg->GetFlags(&dwImf)))
{
dwFlags = ConvertIMFFlagsToARF(dwImf);
}
return dwFlags;
}