Leaked source code of windows server 2003
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.
 
 
 
 
 
 

1985 lines
49 KiB

//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1997 - 1999
//
// File: tcertcli.cpp
//
//--------------------------------------------------------------------------
#include <stdlib.h>
#include <string.h>
#include <windows.h>
#include <stdio.h>
#include <certca.h>
#include <winldap.h>
#include <dsrole.h>
#include <dsgetdc.h>
#include <lmaccess.h>
#include <lmapibuf.h>
#include <shobjidl.h>
#include <shellapi.h>
#include "..\..\..\..\..\win32\ntcrypto\autoenrl\pautoenr\autoenro.h"
#include "cryptuiapi.h"
//--------------------------------------------------------------------
HRESULT
myRobustLdapBindEx(
OUT LDAP ** ppldap,
OPTIONAL OUT LPWSTR* ppszForestDNSName,
IN BOOL fGC)
{
HRESULT hr;
BOOL fForceRediscovery = FALSE;
DWORD dwGetDCFlags = DS_RETURN_DNS_NAME;
PDOMAIN_CONTROLLER_INFO pDomainInfo = NULL;
LDAP *pld = NULL;
WCHAR const *pwszDomainControllerName = NULL;
ULONG ldaperr;
if (fGC)
{
dwGetDCFlags |= DS_GC_SERVER_REQUIRED;
}
do {
if (fForceRediscovery)
{
dwGetDCFlags |= DS_FORCE_REDISCOVERY;
}
ldaperr = LDAP_SERVER_DOWN;
// netapi32!DsGetDcName is delay loaded, so wrap
__try
{
// Get the GC location
hr = DsGetDcName(
NULL, // Delayload wrapped
NULL,
NULL,
NULL,
dwGetDCFlags,
&pDomainInfo);
}
__except(EXCEPTION_EXECUTE_HANDLER)
{
hr=E_UNEXPECTED;
}
if (S_OK != hr)
{
hr = HRESULT_FROM_WIN32(hr);
if (fForceRediscovery)
{
goto error;
}
fForceRediscovery = TRUE;
continue;
}
if (NULL == pDomainInfo ||
(fGC && 0 == (DS_GC_FLAG & pDomainInfo->Flags)) ||
0 == (DS_DNS_CONTROLLER_FLAG & pDomainInfo->Flags) ||
NULL == pDomainInfo->DomainControllerName)
{
if (!fForceRediscovery)
{
fForceRediscovery = TRUE;
continue;
}
hr = HRESULT_FROM_WIN32(ERROR_CANT_ACCESS_DOMAIN_INFO);
goto error;
}
pwszDomainControllerName = pDomainInfo->DomainControllerName;
// skip past forward slashes (why are they there?)
while (L'\\' == *pwszDomainControllerName)
{
pwszDomainControllerName++;
}
// bind to ds
pld = ldap_init(
const_cast<WCHAR *>(pwszDomainControllerName),
fGC? LDAP_GC_PORT : LDAP_PORT);
if (NULL == pld)
{
ldaperr = LdapGetLastError();
}
else
{
// do this because we're explicitly setting DC name
ldaperr = ldap_set_option(pld, LDAP_OPT_AREC_EXCLUSIVE, LDAP_OPT_ON);
ldaperr = ldap_bind_s(pld, NULL, NULL, LDAP_AUTH_NEGOTIATE);
}
hr = HRESULT_FROM_WIN32(LdapMapErrorToWin32(ldaperr));
if (fForceRediscovery)
{
break;
}
fForceRediscovery = TRUE;
} while (LDAP_SERVER_DOWN == ldaperr);
// everything's cool, party down
if (S_OK == hr)
{
*ppldap = pld;
pld = NULL;
}
error:
if (NULL != pld)
{
ldap_unbind(pld);
}
// we know netapi32 was already loaded safely (that's where we got
// pDomainInfo), so no need to wrap
if (NULL != pDomainInfo)
{
NetApiBufferFree(pDomainInfo); // Delayload wrapped
}
return(hr);
}
//--------------------------------------------------------------------
HRESULT
myRobustLdapBind(
OUT LDAP ** ppldap,
IN BOOL fGC)
{
return(myRobustLdapBindEx(ppldap, NULL, fGC));
}
//--------------------------------------------------------------------
void PrintHelp(void) {
wprintf(
L"tcertcli <testID>\n"
L" Available tests:\n"
L" OID - test CAOIDxxxx functions\n"
L" Template - test CACertTypexxxx functions\n"
L" Query - test CACertTypeQuery functions without pld\n"
L" QueryLDAP - test CACertTypeQuery functions with pld\n"
L" CAEnum <CAName> - test CAEnumCertTypesForCA functions without pld\n"
L" CAEnumLDAP <CAName> - test CAEnumCertTypesForCAEx functions with pld\n"
L" TemplateDes - test the description property of templates\n"
L" Clone <TemplateName> - test the clone without pld\n"
L" CloneLDAP <TemplateName> - test the clone with pld\n"
L" ACRS - test create/delete autoenrollment object from ACRS store\n"
L" OIDURL - test URL code for OID container\n"
L" PROPERTY <#1> <#2> - update the CERT_AUTO_ENROLL_RETRY_PROP_ID property of certificates in local machine My store\n"
L" <#1> The # of retrial. 0 clears the property\n"
L" <#2> The # of seconds to wait. \n"
L" SHOW PROPERTY - display the CERT_AUTO_ENROLL_RETRY_PROP_ID property of certificates in local machine My store\n"
);
}
//--------------------------------------------------------------------
BOOL TemplateTest()
{
BOOL fSuccess=FALSE;
HRESULT hr=S_OK;
DWORD dwProp=0;
LPWSTR rgwszProp[4];
CERT_ENHKEY_USAGE KeyUsage;
LPSTR szOID="1.2.3.4.5.6";
FILETIME time1;
FILETIME time2;
DWORD dwNameFlag;
HCERTTYPE hCertType=NULL;
HANDLE hClientToken=NULL;
HANDLE hHandle = NULL;
PCERT_EXTENSIONS pCertExtensions=NULL;
LPWSTR *pwszProp=NULL;
LPWSTR *pwszProp1=NULL;
PSECURITY_DESCRIPTOR pSD=NULL;
LPWSTR pwszOID=NULL;
LPWSTR pwsz=NULL;
DWORD dwType=0;
//get the client token
hHandle = GetCurrentThread();
if (NULL == hHandle)
{
hr = HRESULT_FROM_WIN32(GetLastError());
}
else
{
if (!OpenThreadToken(hHandle,
TOKEN_QUERY,
TRUE, // open as self
&hClientToken))
{
hr = HRESULT_FROM_WIN32(GetLastError());
CloseHandle(hHandle);
hHandle = NULL;
}
}
if(hr != S_OK)
{
hHandle = GetCurrentProcess();
if (NULL == hHandle)
{
hr = HRESULT_FROM_WIN32(GetLastError());
}
else
{
HANDLE hProcessToken = NULL;
hr = S_OK;
if (!OpenProcessToken(hHandle,
TOKEN_DUPLICATE,
&hProcessToken))
{
hr = HRESULT_FROM_WIN32(GetLastError());
CloseHandle(hHandle);
hHandle = NULL;
}
else
{
if(!DuplicateToken(hProcessToken,
SecurityImpersonation,
&hClientToken))
{
hr = HRESULT_FROM_WIN32(GetLastError());
CloseHandle(hHandle);
hHandle = NULL;
}
CloseHandle(hProcessToken);
}
}
}
if(S_OK != hr)
goto error;
//find a certifcate type admin
if(S_OK != CAFindCertTypeByName(
wszCERTTYPE_ADMIN,
NULL,
CT_ENUM_USER_TYPES,
&hCertType
))
{
wprintf(L"Can not find template %ws\n", wszCERTTYPE_ADMIN);
goto error;
}
//get the name flag
if(S_OK != CAGetCertTypeFlagsEx(
hCertType,
CERTTYPE_SUBJECT_NAME_FLAG,
&dwNameFlag
))
{
wprintf(L"Can not find template %ws\n", wszCERTTYPE_ADMIN);
goto error;
}
//get all extensions
if(S_OK != CAGetCertTypeExtensionsEx(
hCertType,
0,
NULL,
&pCertExtensions
))
{
wprintf(L"Can not find extensions %ws\n", wszCERTTYPE_ADMIN);
goto error;
}
if(S_OK != (CAFreeCertTypeExtensions(hCertType, pCertExtensions)))
goto error;
pCertExtensions=NULL;
//get template extensions
if(S_OK != CAGetCertTypeExtensionsEx(
hCertType,
CT_EXTENSION_TEMPLATE,
NULL,
&pCertExtensions
))
{
wprintf(L"Can not find extensions %ws\n", wszCERTTYPE_ADMIN);
goto error;
}
if(S_OK != (CAFreeCertTypeExtensions(hCertType, pCertExtensions)))
goto error;
pCertExtensions=NULL;
//get selected extension
if(S_OK != CAGetCertTypeExtensionsEx(
hCertType,
CT_EXTENSION_BASIC_CONTRAINTS | CT_EXTENSION_APPLICATION_POLICY,
NULL,
&pCertExtensions
))
{
wprintf(L"Can not find extensions %ws\n", wszCERTTYPE_ADMIN);
}
else
{
if(S_OK != (CAFreeCertTypeExtensions(hCertType, pCertExtensions)))
goto error;
}
pCertExtensions=NULL;
//get all extension from the old way
if(S_OK != CAGetCertTypeExtensions(
hCertType,
&pCertExtensions
))
{
wprintf(L"Can not find extensions %ws\n", wszCERTTYPE_ADMIN);
goto error;
}
if(S_OK != (CAFreeCertTypeExtensions(hCertType, pCertExtensions)))
goto error;
pCertExtensions=NULL;
if(S_OK != CAGetCertTypeFlagsEx(
hCertType,
CERTTYPE_SUBJECT_NAME_FLAG,
&dwNameFlag
))
{
wprintf(L"Can not find template %ws\n", wszCERTTYPE_ADMIN);
goto error;
}
if(S_OK != CAGetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_FRIENDLY_NAME,
&pwszProp))
{
wprintf(L"Can not get friendly name for template %ws\n", wszCERTTYPE_ADMIN);
goto error;
}
wprintf(L"The friendly name for %ws is %ws\n", wszCERTTYPE_ADMIN, pwszProp[0]);
CAFreeCertTypeProperty(hCertType, pwszProp);
pwszProp=NULL;
CACloseCertType(hCertType);
hCertType=NULL;
//delete a certifcate type
if(S_OK != CAFindCertTypeByName(
wszCERTTYPE_USER,
NULL,
CT_ENUM_USER_TYPES,
&hCertType
))
goto error;
if(S_OK != CADeleteCertType(hCertType))
goto error;
if(S_OK != CACloseCertType(hCertType))
goto error;
hCertType=NULL;
//testing find cert type by oid
if(S_OK != CAFindCertTypeByName(
wszCERTTYPE_CA_EXCHANGE,
NULL,
CT_ENUM_USER_TYPES | CT_FLAG_NO_CACHE_LOOKUP | CT_ENUM_MACHINE_TYPES,
&hCertType
))
goto error;
if(S_OK != CAGetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_OID,
&pwszProp))
goto error;
if(S_OK != CACloseCertType(hCertType))
goto error;
hCertType=NULL;
if(S_OK != CAFindCertTypeByName(
pwszProp[0],
NULL,
CT_FIND_BY_OID | CT_ENUM_MACHINE_TYPES,
&hCertType
))
goto error;
if(S_OK != CAGetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_OID,
&pwszProp1))
goto error;
if(0!=wcscmp(pwszProp[0], pwszProp1[0]))
goto error;
if(S_OK != CAFreeCertTypeProperty(hCertType, pwszProp))
goto error;
if(S_OK != CAFreeCertTypeProperty(hCertType, pwszProp1))
goto error;
if(S_OK != CACloseCertType(hCertType))
goto error;
hCertType=NULL;
//create a certificate type
if(S_OK != CACreateCertType(L"NewCertType",
NULL,
0,
&hCertType))
goto error;
if(S_OK != CAUpdateCertType(hCertType))
goto error;
if(S_OK != CACloseCertType(hCertType))
goto error;
hCertType=NULL;
//retrieve V1 certifcate type: EFS
//access check on the EFS cert type
if(S_OK != CAFindCertTypeByName(
wszCERTTYPE_EFS,
NULL,
CT_ENUM_USER_TYPES,
&hCertType
))
goto error;
if(S_OK != CAGetCertTypeFlagsEx(
hCertType,
CERTTYPE_GENERAL_FLAG,
&dwProp))
goto error;
printf("The general flag for EFS is: %d\n", dwProp);
if(S_OK != CAGetCertTypeFlagsEx(
hCertType,
CERTTYPE_PRIVATE_KEY_FLAG,
&dwProp))
goto error;
printf("The private key flag for EFS is: %d\n", dwProp);
if(S_OK != CAGetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_SCHEMA_VERSION,
&dwProp))
goto error;
printf("The schema version for EFS is: %d\n", dwProp);
if(S_OK != CACertTypeAccessCheckEx(
hCertType,
hClientToken,
CERTTYPE_ACCESS_CHECK_ENROLL))
goto error;
//no autoenrollment
if(S_OK == CACertTypeAccessCheckEx(
hCertType,
hClientToken,
CERTTYPE_ACCESS_CHECK_AUTO_ENROLL))
goto error;
if(S_OK != CACloseCertType(hCertType))
goto error;
hCertType=NULL;
//retrieve V2 certifcate type: EFS
//access check on the EFS cert type
if(S_OK != CAFindCertTypeByName(
wszCERTTYPE_CROSS_CA,
NULL,
CT_ENUM_MACHINE_TYPES | CT_ENUM_USER_TYPES,
&hCertType
))
goto error;
if(S_OK != CACertTypeAccessCheckEx(
hCertType,
hClientToken,
CERTTYPE_ACCESS_CHECK_ENROLL))
goto error;
//no autoenrollment
if(S_OK == CACertTypeAccessCheckEx(
hCertType,
hClientToken,
CERTTYPE_ACCESS_CHECK_AUTO_ENROLL))
goto error;
if(S_OK != CACloseCertType(hCertType))
goto error;
hCertType=NULL;
//clone a certificate type
if(S_OK != CAFindCertTypeByName(
wszCERTTYPE_SUBORDINATE_CA,
NULL,
CT_ENUM_MACHINE_TYPES,
&hCertType
))
goto error;
rgwszProp[0]=L"ClonedCertType";
rgwszProp[1]=NULL;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_CN,
rgwszProp))
goto error;
rgwszProp[0]=L"ClonedCertType Friendly";
rgwszProp[1]=NULL;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_FRIENDLY_NAME,
rgwszProp))
goto error;
rgwszProp[0]=L"1.2.3.4.5";
rgwszProp[1]=NULL;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_OID,
rgwszProp))
goto error;
rgwszProp[0]=L"1.2.3.4.5.6.7.8.9.10";
rgwszProp[1]=NULL;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_APPLICATION_POLICY,
rgwszProp))
goto error;
rgwszProp[0]=L"1.2.3.4.5.6.7.8.9.10.11";
rgwszProp[1]=NULL;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_RA_APPLICATION_POLICY,
rgwszProp))
goto error;
rgwszProp[0]=L"1.2.3.4.5.6.7.8.9.10.11.12";
rgwszProp[1]=NULL;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_RA_POLICY,
rgwszProp))
goto error;
rgwszProp[0]=NULL;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_SUPERSEDE,
rgwszProp))
goto error;
rgwszProp[0]=L"CloneSuper1";
rgwszProp[1]=L"CloneSuper2";
rgwszProp[2]=L"CloneSuper3";
rgwszProp[3]=NULL;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_SUPERSEDE,
rgwszProp))
goto error;
dwProp=2048;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_MIN_KEY_SIZE,
&dwProp))
goto error;
if(S_OK != CAGetCertTypeExtensions(
hCertType,
&pCertExtensions))
goto error;
KeyUsage.cUsageIdentifier=1;
KeyUsage.rgpszUsageIdentifier=&szOID;
if(S_OK != CASetCertTypeExtension(
hCertType,
TEXT(szOID_ENHANCED_KEY_USAGE),
CA_EXT_FLAG_CRITICAL,
&KeyUsage))
goto error;
if(S_OK != CAUpdateCertType(hCertType))
goto error;
if(S_OK != CACloseCertType(hCertType))
goto error;
hCertType=NULL;
//edit V2 certificate type: KeyRecoveryAgent
//update SD, Expiration,
if(S_OK != CAFindCertTypeByName(
wszCERTTYPE_KEY_RECOVERY_AGENT,
NULL,
CT_ENUM_USER_TYPES,
&hCertType
))
goto error;
if(S_OK != CACertTypeAccessCheckEx(
hCertType,
hClientToken,
CERTTYPE_ACCESS_CHECK_ENROLL))
goto error;
dwProp=103;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_REVISION,
&dwProp))
goto error;
if(S_OK != CASetCertTypeFlagsEx(
hCertType,
CERTTYPE_ENROLLMENT_FLAG,
0))
goto error;
if(S_OK != CASetCertTypeFlagsEx(
hCertType,
CERTTYPE_SUBJECT_NAME_FLAG,
0))
goto error;
if(S_OK != CAGetCertTypeExpiration(
hCertType,
&time1,
&time2))
goto error;
if(S_OK != CASetCertTypeExpiration(
hCertType,
&time1,
&time2))
goto error;
if(S_OK != CACertTypeGetSecurity(
hCertType,
&pSD))
goto error;
if(S_OK != CACertTypeSetSecurity(
hCertType,
pSD))
goto error;
if(S_OK != CAGetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_CSP_LIST,
&pwszProp))
goto error;
if(pwszProp && pwszProp[0])
printf("The CSP for KRA is: %S\n", pwszProp[0]);
if(S_OK != CAFreeCertTypeProperty(
hCertType,
pwszProp))
goto error;
pwszProp=NULL;
if(S_OK != CAGetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_RA_POLICY,
&pwszProp))
goto error;
if(pwszProp && pwszProp[0])
printf("The RAPolicy for KRA is: %S\n", pwszProp[0]);
if(S_OK != CAFreeCertTypeProperty(
hCertType,
pwszProp))
goto error;
pwszProp=NULL;
rgwszProp[0]=L"1.2.3.4.5.6.7.8.9.10";
rgwszProp[1]=L"2.2.3.4.5.6.7.8.9.10";
rgwszProp[2]=NULL;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_APPLICATION_POLICY,
rgwszProp))
goto error;
rgwszProp[0]=L"1.2.3.4.5.6.7.8.9.10.11";
rgwszProp[1]=L"2.2.3.4.5.6.7.8.9.10.11";
rgwszProp[2]=NULL;
if(S_OK != CASetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_RA_APPLICATION_POLICY,
rgwszProp))
goto error;
if(S_OK != CAGetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_RA_APPLICATION_POLICY,
&pwszProp))
goto error;
if(pwszProp && pwszProp[0])
printf("The RAAppPolicy for KRA is: %S\n", pwszProp[0]);
if(S_OK != CAFreeCertTypeProperty(
hCertType,
pwszProp))
goto error;
pwszProp=NULL;
if(S_OK != CAGetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_APPLICATION_POLICY,
&pwszProp))
goto error;
if(pwszProp && pwszProp[0])
printf("The AppPolicy for KRA is: %S\n", pwszProp[0]);
if(S_OK != CAFreeCertTypeProperty(
hCertType,
pwszProp))
goto error;
pwszProp=NULL;
if(S_OK != CAUpdateCertType(hCertType))
goto error;
if(S_OK != CACloseCertType(hCertType))
goto error;
hCertType=NULL;
//get the KRA properties again
if(S_OK != CAFindCertTypeByName(
wszCERTTYPE_KEY_RECOVERY_AGENT,
NULL,
CT_ENUM_USER_TYPES,
&hCertType
))
goto error;
if(S_OK != CAGetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_RA_APPLICATION_POLICY,
&pwszProp))
goto error;
if(pwszProp && pwszProp[0])
printf("The RAAppPolicy for KRA is: %S\n", pwszProp[0]);
if(S_OK != CAFreeCertTypeProperty(
hCertType,
pwszProp))
goto error;
pwszProp=NULL;
if(S_OK != CAGetCertTypePropertyEx(
hCertType,
CERTTYPE_PROP_APPLICATION_POLICY,
&pwszProp))
goto error;
if(pwszProp && pwszProp[0])
printf("The AppPolicy for KRA is: %S\n", pwszProp[0]);
if(S_OK != CAFreeCertTypeProperty(
hCertType,
pwszProp))
goto error;
pwszProp=NULL;
fSuccess=TRUE;
error:
if(pwszOID)
LocalFree(pwszOID);
if(pCertExtensions)
CAFreeCertTypeExtensions(hCertType,pCertExtensions);
if(hCertType)
CACloseCertType(hCertType);
if(pSD)
LocalFree(pSD);
if(hHandle)
CloseHandle(hHandle);
if(hClientToken)
CloseHandle(hClientToken);
return fSuccess;
}
//--------------------------------------------------------------------
BOOL OIDTest()
{
BOOL fSuccess=FALSE;
HRESULT hr=S_OK;
DWORD dwProp=0;
LPWSTR rgwszProp[4];
CERT_ENHKEY_USAGE KeyUsage;
LPSTR szOID="1.2.3.4.5.6";
FILETIME time1;
FILETIME time2;
LPWSTR *pwszProp=NULL;
LPWSTR *pwszProp1=NULL;
LPWSTR pwszOID=NULL;
LPWSTR pwsz=NULL;
DWORD dwType=0;
//oid manipulation
//create
if(S_OK != CAOIDCreateNew(CERT_OID_TYPE_TEMPLATE, 0,
&pwszOID))
goto error;
//set/get property test
if(S_OK != CAOIDSetProperty(
pwszOID,
CERT_OID_PROPERTY_DISPLAY_NAME,
L"MyNewOIDFriendlyName"))
goto error;
if(S_OK != CAOIDSetProperty(
pwszOID,
CERT_OID_PROPERTY_CPS,
L"MYCSPStatement"))
goto error;
if(S_OK != CAOIDGetProperty(
pwszOID,
CERT_OID_PROPERTY_CPS,
&pwsz))
goto error;
printf("The CPS statement is: %S\n", pwsz);
if(S_OK != CAOIDFreeProperty(pwsz))
goto error;
pwsz=NULL;
if(S_OK != CAOIDGetProperty(
pwszOID,
CERT_OID_PROPERTY_TYPE,
&dwType))
goto error;
printf("The property type is: %d\n", dwType);
if(S_OK == CAOIDSetProperty(
pwszOID,
CERT_OID_PROPERTY_TYPE,
L"MyNewOIDFriendlyName"))
goto error;
if(S_OK == CAOIDSetProperty(
L"1.2",
CERT_OID_PROPERTY_DISPLAY_NAME,
L"MyNewOIDFriendlyName"))
goto error;
//add and delete
if(S_OK != CAOIDAdd(CERT_OID_TYPE_ISSUER_POLICY, 0, L"1.2.3"))
goto error;
if(S_OK != CAOIDAdd(CERT_OID_TYPE_ISSUER_POLICY, 0, L"1.2.3.4"))
goto error;
if(S_OK != CAOIDSetProperty(L"1.2.3.4",
CERT_OID_PROPERTY_DISPLAY_NAME,
L"MyNewIssuerPolicyOid"))
goto error;
if(S_OK != CAOIDGetProperty(L"1.2.3.4",
CERT_OID_PROPERTY_DISPLAY_NAME,
&pwsz))
goto error;
printf("The display name is: %S\n", pwsz);
if(S_OK != CAOIDSetProperty(L"1.2.3.4",
CERT_OID_PROPERTY_CPS,
L"The DS Issuer Policy String"))
goto error;
if(S_OK != CAOIDSetProperty(L"1.2.3.4",
CERT_OID_PROPERTY_CPS,
NULL))
goto error;
if(S_OK != CAOIDSetProperty(L"1.2.3.4",
CERT_OID_PROPERTY_CPS,
L"New CPS"))
goto error;
if(S_OK != CAOIDFreeProperty(pwsz))
goto error;
pwsz=NULL;
if(CRYPT_E_EXISTS != CAOIDAdd(CERT_OID_TYPE_ISSUER_POLICY, 0, L"1.2.3"))
goto error;
if(S_OK != CAOIDDelete(L"1.2.3"))
goto error;
if(S_OK != CAOIDDelete(L"1.2.3.4"))
goto error;
//URL testing
if(S_OK != CAOIDGetLdapURL(CERT_OID_TYPE_TEMPLATE,
0,
&pwsz))
goto error;
printf("The URL for template is: %S\n", pwsz);
if(S_OK != CAOIDFreeLdapURL(pwsz))
goto error;
if(S_OK != CAOIDGetLdapURL(CERT_OID_TYPE_ALL,
0,
&pwsz))
goto error;
printf("The URL for all is: %S\n", pwsz);
if(S_OK != CAOIDFreeLdapURL(pwsz))
goto error;
if(S_OK != CAOIDGetLdapURL(CERT_OID_TYPE_APPLICATION_POLICY,
0,
&pwsz))
goto error;
printf("The URL for application policy is: %S\n", pwsz);
if(S_OK != CAOIDFreeLdapURL(pwsz))
goto error;
fSuccess=TRUE;
error:
if(pwszOID)
LocalFree(pwszOID);
return fSuccess;
}
//--------------------------------------------------------------------
BOOL QueryTest(BOOL fBind)
{
HRESULT hr=E_FAIL;
BOOL fResult = FALSE;
DWORD dwNumber = 0;
DWORD dwIndex=0;
LDAP *pldap=NULL;
HCERTTYPEQUERY hCertTypeQuery=NULL;
if(fBind)
{
if(S_OK != (hr = myRobustLdapBind(&pldap, FALSE)))
{
wprintf(L"myRobustLdapBind failed with 0x%08X. \n",hr);
goto error;
}
}
if(S_OK != CACertTypeRegisterQuery(0, pldap, &hCertTypeQuery))
{
wprintf(L"CACertTypeRegisterQury failed with 0x%08X. \n",hr);
goto error;
}
for(dwIndex=0; dwIndex < 3; dwIndex++)
{
if(S_OK != CACertTypeQuery(hCertTypeQuery, &dwNumber))
{
wprintf(L"CACertTypeQuery failed with 0x%08X. \n",hr);
goto error;
}
wprintf(L"CACertTypeQuery returned %d. \n", dwNumber);
wprintf(L"Wait for 20 seconds. \n");
Sleep(20 * 1000);
}
fResult = TRUE;
error:
if(hCertTypeQuery)
CACertTypeUnregisterQuery(hCertTypeQuery);
if(pldap)
ldap_unbind(pldap);
return fResult;
}
//--------------------------------------------------------------------
//
//
// CAEnumTest
//
//
//--------------------------------------------------------------------
BOOL CAEnumTest(BOOL fBind, LPWSTR pwszCA)
{
HRESULT hr=E_FAIL;
BOOL fResult = FALSE;
DWORD dwCount = 0;
HCAINFO hCAInfo = NULL;
HCERTTYPE hCertType = NULL;
LDAP *pldap=NULL;
LPWSTR *awszProp=NULL;
if(fBind)
{
if(S_OK != (hr = myRobustLdapBind(&pldap, FALSE)))
{
wprintf(L"myRobustLdapBind failed with 0x%08X. \n",hr);
goto error;
}
}
if(S_OK != (hr = CAFindByName(
pwszCA,
NULL,
CA_FIND_LOCAL_SYSTEM,
&hCAInfo)))
{
wprintf(L"CAFindByName failed with 0x%08X. \n",hr);
goto error;
}
if(NULL==hCAInfo)
{
wprintf(L"CAFindByName return NULL hCAInfo. \n");
goto error;
}
if(fBind)
{
hr = CAEnumCertTypesForCAEx(
hCAInfo,
(LPCWSTR)pldap,
CT_FIND_LOCAL_SYSTEM | CT_ENUM_MACHINE_TYPES | CT_ENUM_USER_TYPES |
CT_FLAG_SCOPE_IS_LDAP_HANDLE | CT_FLAG_NO_CACHE_LOOKUP,
&hCertType);
}
else
{
hr = CAEnumCertTypesForCA(
hCAInfo,
CT_FIND_LOCAL_SYSTEM | CT_ENUM_MACHINE_TYPES | CT_ENUM_USER_TYPES,
&hCertType);
}
if( (S_OK != hr) || (NULL == hCertType))
{
wprintf(L"CAEnumCertTyes failed with 0x%08X. \n",hr);
goto error;
}
dwCount = CACountCertTypes(hCertType);
if(0 == dwCount)
{
wprintf(L"Error: CACountCertTypes returns 0 templates.\n");
goto error;
}
wprintf(L"CACountCertTypes returns %d templates.\n", dwCount);
//get the CA properties
hr=CAGetCAProperty(hCAInfo,
CA_PROP_DNSNAME,
&awszProp);
if((S_OK != hr) || (NULL==awszProp) || (NULL==awszProp[0]))
{
wprintf(L"CAGetCAProperty failed with 0x%08X. \n",hr);
goto error;
}
wprintf(L"CA's DNS name is %ws.\n", awszProp[0]);
hr=CAFreeCAProperty(hCAInfo, awszProp);
if(S_OK != hr)
{
wprintf(L"CAFreeCAProperty failed with 0x%08X. \n",hr);
goto error;
}
//name
hr=CAGetCAProperty(hCAInfo,
CA_PROP_NAME,
&awszProp);
if((S_OK != hr) || (NULL==awszProp) || (NULL==awszProp[0]))
{
wprintf(L"CAGetCAProperty failed with 0x%08X. \n",hr);
goto error;
}
wprintf(L"CA's CN name is %ws.\n", awszProp[0]);
hr=CAFreeCAProperty(hCAInfo, awszProp);
if(S_OK != hr)
{
wprintf(L"CAFreeCAProperty failed with 0x%08X. \n",hr);
goto error;
}
//display name
hr=CAGetCAProperty(hCAInfo,
CA_PROP_DISPLAY_NAME,
&awszProp);
if((S_OK != hr) || (NULL==awszProp) || (NULL==awszProp[0]))
{
wprintf(L"CAGetCAProperty failed with 0x%08X. \n",hr);
goto error;
}
wprintf(L"CA's display name is %ws.\n", awszProp[0]);
hr=CAFreeCAProperty(hCAInfo, awszProp);
if(S_OK != hr)
{
wprintf(L"CAFreeCAProperty failed with 0x%08X. \n",hr);
goto error;
}
//cert types
hr=CAGetCAProperty(hCAInfo,
CA_PROP_CERT_TYPES,
&awszProp);
if((S_OK != hr) || (NULL==awszProp) || (NULL==awszProp[0]))
{
wprintf(L"CAGetCAProperty failed with 0x%08X. \n",hr);
goto error;
}
wprintf(L"CA's cert types name is %ws.\n", awszProp[0]);
hr=CAFreeCAProperty(hCAInfo, awszProp);
if(S_OK != hr)
{
wprintf(L"CAFreeCAProperty failed with 0x%08X. \n",hr);
goto error;
}
fResult = TRUE;
error:
if(hCertType)
CACloseCertType(hCertType);
if(hCAInfo)
{
CACloseCA(hCAInfo);
}
if(pldap)
ldap_unbind(pldap);
return fResult;
}
//--------------------------------------------------------------------
//
// CloneTest
//
//
//--------------------------------------------------------------------
BOOL CloneTest(BOOL fBind, LPWSTR pwszCertType)
{
HRESULT hr=E_FAIL;
BOOL fResult = FALSE;
DWORD dwFindCT=CT_ENUM_MACHINE_TYPES | CT_ENUM_USER_TYPES;
WCHAR wszName[100];
WCHAR wszFriendlyName[100];
LPWSTR *awszFriendlyName=NULL;
LDAP *pldap=NULL;
HCERTTYPE hCertType=NULL;
HCERTTYPE hNewCertType=NULL;
if(fBind)
{
if(S_OK != (hr = myRobustLdapBind(&pldap, FALSE)))
{
wprintf(L"myRobustLdapBind failed with 0x%08X. \n",hr);
goto error;
}
}
if(S_OK != (hr = CAFindCertTypeByName(
pwszCertType,
NULL,
dwFindCT,
&hCertType)))
{
wprintf(L"CAFindCertTypeByName failed with 0x%08X. \n",hr);
goto error;
}
wcscpy(wszName, pwszCertType);
wcscat(wszName, L"_Clone");
wcscpy(wszFriendlyName, pwszCertType);
wcscat(wszFriendlyName, L"_CloneFriendly");
if(S_OK != (hr=CACloneCertType(hCertType,
wszName,
wszFriendlyName,
pldap,
fBind? CT_CLONE_KEEP_AUTOENROLLMENT_SETTING | CT_CLONE_KEEP_SUBJECT_NAME_SETTING : 0,
&hNewCertType)))
{
wprintf(L"CACloneCertType failed with 0x%08X. \n",hr);
goto error;
}
if(S_OK != (hr=CAUpdateCertType(hNewCertType)))
{
wprintf(L"CAUpdateCertType failed with 0x%08X. \n",hr);
goto error;
}
//close the tempate
if(S_OK != (hr=CACloseCertType(hNewCertType)))
{
hNewCertType=NULL;
wprintf(L"CACloseCertType failed with 0x%08X. \n",hr);
goto error;
}
hNewCertType=NULL;
if(S_OK != (hr = CAFindCertTypeByName(
wszName,
NULL,
dwFindCT,
&hNewCertType)))
{
wprintf(L"CAFindCertTypeByName for the cloned template failed with 0x%08X. \n",hr);
goto error;
}
if(S_OK != (hr=CAGetCertTypePropertyEx(
hNewCertType,
CERTTYPE_PROP_FRIENDLY_NAME,
&awszFriendlyName)))
{
wprintf(L"CAGetCertTypePropertyEx for the cloned template failed with 0x%08X. \n",hr);
goto error;
}
if(0 != (wcscmp(awszFriendlyName[0], wszFriendlyName)))
{
wprintf(L"The friendly name for the cloned template does not match the original. \n");
hr=E_FAIL;
goto error;
}
fResult = TRUE;
error:
if(pldap)
ldap_unbind(pldap);
if(awszFriendlyName)
CAFreeCertTypeProperty(hNewCertType, awszFriendlyName);
if(hCertType)
CACloseCertType(hCertType);
if(hNewCertType)
CACloseCertType(hNewCertType);
return fResult;
}
//--------------------------------------------------------------------
//
// TemplateDesTest
//
//
//--------------------------------------------------------------------
BOOL TemplateDesTest()
{
BOOL fResult = FALSE;
HRESULT hr = E_FAIL;
DWORD dwCount=0;
DWORD dwIndex=0;
HCERTTYPE hCertType=NULL;
HCERTTYPE hNextCertType=NULL;
LPWSTR *pwszFriendlyName=NULL;
LPWSTR *pwszDescription=NULL;
HANDLE hClientToken=NULL;
HANDLE hHandle = NULL;
HCAINFO hCAInfo=NULL;
//get the client token
/* hHandle = GetCurrentThread();
if (NULL == hHandle)
{
hr = HRESULT_FROM_WIN32(GetLastError());
}
else
{
if (!OpenThreadToken(hHandle,
TOKEN_QUERY,
TRUE, // open as self
&hClientToken))
{
hr = HRESULT_FROM_WIN32(GetLastError());
CloseHandle(hHandle);
hHandle = NULL;
}
}
if(hr != S_OK)
{
hHandle = GetCurrentProcess();
if (NULL == hHandle)
{
hr = HRESULT_FROM_WIN32(GetLastError());
}
else
{
HANDLE hProcessToken = NULL;
hr = S_OK;
if (!OpenProcessToken(hHandle,
TOKEN_DUPLICATE,
&hProcessToken))
{
hr = HRESULT_FROM_WIN32(GetLastError());
CloseHandle(hHandle);
hHandle = NULL;
}
else
{
if(!DuplicateToken(hProcessToken,
SecurityImpersonation,
&hClientToken))
{
hr = HRESULT_FROM_WIN32(GetLastError());
CloseHandle(hHandle);
hHandle = NULL;
}
CloseHandle(hProcessToken);
}
}
}
if(S_OK != hr)
goto error;
hr = CAFindCertTypeByName(L"WindowsTestBuildSigning",
NULL,
CT_ENUM_USER_TYPES,
&hCertType);
if(S_OK != hr)
{
wprintf(L"CAFindCertTypeByName failed with 0x%08X. \n",hr);
goto error;
}
hr = CACertTypeAccessCheck(
hCertType,
hClientToken
);
if(S_OK != hr)
{
wprintf(L"CACertTypeAccessCheck failed with 0x%08X. \n",hr);
goto error;
}
hr = CAFindByName(
L"Microsoft Windows VBL03 !0028DS!0029",
NULL,
0,
&hCAInfo);
if((S_OK != hr) || (NULL==hCAInfo))
{
wprintf(L"CAFindByName for %ws failed with 0x%08X. \n", hr);
goto error;
}
hr=CAAccessCheck(hCAInfo,
hClientToken);
if(S_OK != hr)
{
wprintf(L"CAAccessCheck failed with 0x%08X. \n",hr);
goto error;
} */
hr = CAEnumCertTypes(
CT_ENUM_MACHINE_TYPES | CT_ENUM_USER_TYPES,
&hCertType);
if((S_OK != hr) || (NULL==hCertType))
{
wprintf(L"CAEnumCertTyes failed with 0x%08X. \n",hr);
goto error;
}
dwCount = CACountCertTypes(hCertType);
wprintf(L"We have 0x%d cert types. \n", dwCount);
if(0 == dwCount)
goto error;
for(dwIndex=0; dwIndex < dwCount; dwIndex++)
{
if(dwIndex!=0)
{
hr = CAEnumNextCertType(hCertType, &hNextCertType);
if(S_OK != hr)
{
wprintf(L"CAEnumNextCertType failed with 0x%08X. \n",hr);
goto error;
}
CACloseCertType(hCertType);
hCertType=hNextCertType;
}
hr = CAGetCertTypePropertyEx(hCertType,
CERTTYPE_PROP_FRIENDLY_NAME,
&pwszFriendlyName);
if((S_OK != hr) || (NULL==pwszFriendlyName) || (NULL==pwszFriendlyName[0]))
{
wprintf(L"Friendly Name property failed with 0x%08X. \n",hr);
goto error;
}
hr = CAGetCertTypePropertyEx(hCertType,
CERTTYPE_PROP_DESCRIPTION,
&pwszDescription);
if((S_OK != hr) || (NULL==pwszDescription) || (NULL==pwszDescription[0]))
{
wprintf(L"Description property failed with 0x%08X. \n",hr);
goto error;
}
wprintf(L"%ws has the description of %ws. \n",pwszFriendlyName[0], pwszDescription[0]);
CAFreeCertTypeProperty(hCertType, pwszFriendlyName);
CAFreeCertTypeProperty(hCertType, pwszDescription);
}
fResult = TRUE;
error:
if(hHandle)
CloseHandle(hHandle);
if(hClientToken)
CloseHandle(hClientToken);
return fResult;
}
//--------------------------------------------------------------------
//
// OIDURLTest
//
//
//--------------------------------------------------------------------
BOOL OIDURLTest()
{
BOOL fResult=FALSE;
HRESULT hr=E_FAIL;
DWORD dwIndex=0;
LPWSTR pwsz=NULL;
for(dwIndex=0; dwIndex < 14; dwIndex++)
{
//URL testing
if(S_OK != (hr=CAOIDGetLdapURL(CERT_OID_TYPE_TEMPLATE,
0,
&pwsz)))
{
wprintf(L"CAOIDGetLdapURL failed with 0x%08X. \n",hr);
}
printf("The URL for template is: %S\n", pwsz);
if(S_OK != (hr=CAOIDFreeLdapURL(pwsz)))
{
wprintf(L"CAOIDFreeLdapURL failed with 0x%08X. \n",hr);
goto error;
}
//sleep for 1 second
Sleep(1000);
}
fResult=TRUE;
error:
return fResult;
}
//--------------------------------------------------------------------
//
// ACRSTest
//
//
//--------------------------------------------------------------------
BOOL ACRSTest()
{
BOOL fResult=FALSE;
HRESULT hr=E_FAIL;
/* hr = CACreateLocalAutoEnrollmentObject(
wszCERTTYPE_DC, // DC certificate
NULL, // any CA
NULL, // reserved
CERT_SYSTEM_STORE_LOCAL_MACHINE);
if(S_OK != hr)
{
wprintf(L"CreateLocalAutoEnrollmentObject failed with 0x%08X. \n",hr);
goto error;
} */
hr = CADeleteLocalAutoEnrollmentObject(
wszCERTTYPE_DC, // DC certificate
NULL, // any CA
NULL, // reserved
CERT_SYSTEM_STORE_LOCAL_MACHINE);
if(S_OK != hr)
{
wprintf(L"DeleteLocalAutoEnrollmentObject failed with 0x%08X. \n",hr);
goto error;
}
fResult=TRUE;
error:
return fResult;
}
//--------------------------------------------------------------------
//
//
// PropertyTest()
//
//
//--------------------------------------------------------------------
BOOL PropertyTest(LPWSTR pwszRetry, LPWSTR pwszSecond)
{
BOOL fResult=FALSE;
CRYPT_DATA_BLOB blobProp;
DWORD dwRetry=0;
DWORD dwSecond=0;
AE_RETRY_INFO AE_Retry_Info;
ULARGE_INTEGER ftTime;
ULARGE_INTEGER DueTime;
ULONG lConvert=0;
HCERTSTORE hMyStore=NULL;
PCCERT_CONTEXT pCertContext=NULL;
hMyStore = CertOpenStore(CERT_STORE_PROV_SYSTEM_W,
X509_ASN_ENCODING | PKCS_7_ASN_ENCODING,
NULL,
CERT_SYSTEM_STORE_LOCAL_MACHINE,
L"MY");
if(NULL == hMyStore)
goto Ret;
dwRetry=_wtoi(pwszRetry);
dwSecond=_wtoi(pwszSecond);
if(0 != dwRetry)
{
memset(&blobProp, 0, sizeof(CRYPT_DATA_BLOB));
blobProp.cbData=sizeof(AE_RETRY_INFO);
blobProp.pbData=(BYTE *)&AE_Retry_Info;
memset(&AE_Retry_Info, 0, sizeof(AE_RETRY_INFO));
AE_Retry_Info.cbSize=sizeof(AE_RETRY_INFO);
AE_Retry_Info.dwRetry=dwRetry;
//get the current time
GetSystemTimeAsFileTime((LPFILETIME)&ftTime);
// convert to 10^-7s.
lConvert=10000*1000;
DueTime.QuadPart=Int32x32To64(dwSecond, lConvert);
(AE_Retry_Info.dueTime).QuadPart = ftTime.QuadPart + DueTime.QuadPart;
}
while(pCertContext=CertEnumCertificatesInStore(hMyStore, pCertContext))
{
//copy the property on the certificate
if(0 == dwRetry)
{
if(!CertSetCertificateContextProperty(
pCertContext,
CERT_AUTO_ENROLL_RETRY_PROP_ID,
0,
NULL))
{
goto Ret;
}
}
else
{
if(!CertSetCertificateContextProperty(
pCertContext,
CERT_AUTO_ENROLL_RETRY_PROP_ID,
0,
&blobProp))
{
goto Ret;
}
}
}
fResult=TRUE;
Ret:
if(pCertContext)
CertFreeCertificateContext(pCertContext);
if(hMyStore)
CertCloseStore(hMyStore, 0);
return fResult;
}
//-----------------------------------------------------------------------
//
// GetRetryProperty
//
//-----------------------------------------------------------------------
BOOL GetRetryProperty(PCCERT_CONTEXT pCertContext,
AE_RETRY_INFO **ppAE_Retry_Info)
{
BOOL fResult=FALSE;
DWORD cbData=0;
AE_RETRY_INFO *pRetry_Info=NULL;
if((NULL==pCertContext) || (NULL==ppAE_Retry_Info))
goto Ret;
*ppAE_Retry_Info=NULL;
if(!CertGetCertificateContextProperty(
pCertContext,
CERT_AUTO_ENROLL_RETRY_PROP_ID,
NULL,
&cbData))
goto Ret;
pRetry_Info=(AE_RETRY_INFO *)LocalAlloc(LPTR, cbData);
if(NULL == pRetry_Info)
goto Ret;
if(!CertGetCertificateContextProperty(
pCertContext,
CERT_AUTO_ENROLL_RETRY_PROP_ID,
pRetry_Info,
&cbData))
goto Ret;
//verify the integrity of the property on the certificate
if(cbData < sizeof(AE_RETRY_INFO))
goto Ret;
if((pRetry_Info->cbSize) < sizeof(AE_RETRY_INFO))
goto Ret;
*ppAE_Retry_Info=pRetry_Info;
pRetry_Info=NULL;
fResult=TRUE;
Ret:
if(pRetry_Info)
LocalFree(pRetry_Info);
return fResult;
}
//--------------------------------------------------------------------
//
//
// ShowPropertyTest()
//
//
//--------------------------------------------------------------------
BOOL ShowPropertyTest()
{
BOOL fResult=FALSE;
DWORD i=0;
ULARGE_INTEGER dwSeconds;
ULARGE_INTEGER one;
ULARGE_INTEGER two;
ULARGE_INTEGER three;
ULARGE_INTEGER ftTime;
HCERTSTORE hMyStore=NULL;
PCCERT_CONTEXT pCertContext=NULL;
AE_RETRY_INFO *pAE_Retry_Info=NULL;
hMyStore = CertOpenStore(CERT_STORE_PROV_SYSTEM_W,
X509_ASN_ENCODING | PKCS_7_ASN_ENCODING,
NULL,
CERT_SYSTEM_STORE_LOCAL_MACHINE,
L"MY");
if(NULL == hMyStore)
goto Ret;
while(pCertContext=CertEnumCertificatesInStore(hMyStore, pCertContext))
{
//display the property
if(!GetRetryProperty(pCertContext, &pAE_Retry_Info))
{
wprintf(L"===================== %d =========================================\n", i);
wprintf(L"The certificate has no CERT_AUTO_ENROLL_RETRY_PROP_ID property.\n");
wprintf(L"===============================================================\n\n");
}
else
{
GetSystemTimeAsFileTime((LPFILETIME)&ftTime);
dwSeconds.QuadPart = pAE_Retry_Info->dueTime.QuadPart-ftTime.QuadPart;
one.QuadPart=dwSeconds.QuadPart/10000;
two.QuadPart=one.QuadPart/1000;
wprintf(L"===================== %d =========================================\n", i);
wprintf(L"The certificate has the following CERT_AUTO_ENROLL_RETRY_PROP_ID property.\n");
wprintf(L"Size = %d.\n", pAE_Retry_Info->cbSize);
wprintf(L"Retry = %d.\n", pAE_Retry_Info->dwRetry);
wprintf(L"dueTime = %d seconds. \n", (DWORD)(two.QuadPart));
wprintf(L"===============================================================\n\n");
}
//display the certificate
CryptUIDlgViewContext(
CERT_STORE_CERTIFICATE_CONTEXT,
pCertContext,
NULL,
NULL,
0,
NULL);
i++;
if(pAE_Retry_Info)
LocalFree(pAE_Retry_Info);
}
fResult=TRUE;
Ret:
if(pCertContext)
CertFreeCertificateContext(pCertContext);
if(hMyStore)
CertCloseStore(hMyStore, 0);
return fResult;
}
//--------------------------------------------------------------------
extern "C" int __cdecl wmain(int nArgs, WCHAR ** rgwszArgs)
{
BOOL fResult;
if (1 == nArgs || 0==wcscmp(rgwszArgs[1], L"/?") || 0==wcscmp(rgwszArgs[1], L"-?")) {
PrintHelp();
goto done;
}
if (0==_wcsicmp(L"OID", rgwszArgs[1])) {
fResult=OIDTest();
} else if (0==_wcsicmp(L"Template", rgwszArgs[1])) {
fResult=TemplateTest();
} else if (0==_wcsicmp(L"Query", rgwszArgs[1])) {
fResult=QueryTest(FALSE);
} else if (0==_wcsicmp(L"QueryLDAP", rgwszArgs[1])) {
fResult=QueryTest(TRUE);
} else if (0==_wcsicmp(L"CAEnum", rgwszArgs[1])) {
fResult=CAEnumTest(FALSE, rgwszArgs[2]);
} else if (0==_wcsicmp(L"CAEnumLDAP", rgwszArgs[1])) {
fResult=CAEnumTest(TRUE, rgwszArgs[2]);
} else if (0==_wcsicmp(L"Clone", rgwszArgs[1])) {
fResult=CloneTest(FALSE, rgwszArgs[2]);
} else if (0==_wcsicmp(L"CloneLDAP", rgwszArgs[1])) {
fResult=CloneTest(TRUE, rgwszArgs[2]);
} else if (0==_wcsicmp(L"TemplateDes", rgwszArgs[1])) {
fResult=TemplateDesTest();
} else if (0==_wcsicmp(L"ACRS", rgwszArgs[1])) {
fResult=ACRSTest();
} else if (0==_wcsicmp(L"OIDURL", rgwszArgs[1])) {
fResult=OIDURLTest();
} else if (0==_wcsicmp(L"PROPERTY", rgwszArgs[1])) {
fResult=PropertyTest(rgwszArgs[2], rgwszArgs[3]);
} else if (0==_wcsicmp(L"SHOW PROPERTY", rgwszArgs[1])) {
fResult=ShowPropertyTest();
} else {
wprintf(L"Command '%ws' unknown.\n", rgwszArgs[1]);
goto done;
}
if (!fResult) {
wprintf(L"Command '%ws' failed \n", rgwszArgs[1]);
} else {
wprintf(L"Command '%ws' completed successfully.\n", rgwszArgs[1]);
}
done:
return fResult;
}