API examples

A simple command-line driven test program is provided in the same .zip file as this document as a sample to demonstrate the usage of these APIs. Here are some code snippets from the sample.

Example 1: Log on a user

The following code provides user credentials to single sign-on (SSO) to log on a user.

BOOL DoLogonSsoUser(const wchar_t* pUsername, const wchar_t * pPassword, const wchar_t *pDomain)
{
DWORD rc;
 DWORD result;
 MSV1_0_INTERACTIVE_LOGON msv1Logon;
 ( !pUsername || ! pPassword || !pDomain)
 return FALSE;
 msv1Logon.UserName.Length = (USHORT)(wcslen(pUsername) *
  sizeof(wchar_t));
  msv1Logon.UserName.MaximumLength = msv1Logon.UserName.Length;
  msv1Logon.UserName.Buffer = (PWSTR)pUsername;
  msv1Logon.Password.Length = (USHORT)(wcslen(pPassword) *
  sizeof(wchar_t));
 msv1Logon.Password.MaximumLength = msv1Logon.Password.Length;
 msv1Logon.Password.Buffer = (PWSTR)pPassword;

 msv1Logon.LogonDomainName.Length = (USHORT)(wcslen(pDomain) *
  sizeof(wchar_t));
msv1Logon.LogonDomainName.MaximumLength = msv1Logon.LogonDomainName.Length;
msv1Logon.LogonDomainName.Buffer = (PWSTR)pDomain;
msv1Logon.MessageType = MsV1_0InteractiveLogon;

printf("calling LogonSsoUser...\n");
rc = LogonSsoUser(&msv1Logon, FALSE, FALSE, &result);

SecureZeroMemory(&msv1Logon, sizeof(msv1Logon));

if(rc != LOGONSSOUSER_OK) {
rintf("\nLogonSsoUser() failed: %d\n", rc);
return FALSE;
}
printf("\n User logged on. Result of LogonSsoUser(): %d\n", result);
return TRUE;
}
<!--NeedCopy-->

Example 2: Log on a user with a smart card

The following code provides user credentials to SSO to log on a user with a smart card.

BOOL DoLogonSsoUserWithPin(const wchar_t* pin)
{
DWORD rc;
DWORD result;
if (InitLibrary())
if(!pin)
{
return FALSE;
}
if (!pfnLogOnUserWithPin)
{
return FALSE;
}
printf("calling LogonSsoUserWithPin...\n");
rc = pfnLogOnUserWithPin(pin, disconnect, restartPna, &result);
if(rc != LOGONSSOUSER_OK)
{
printf("\nLogonSsoUserWithPin() failed: %d\n", rc);
return FALSE;
}
printf("\n User logged on. Result of LogonSsoUserWithPin(): %d\n", result);
return TRUE;
}
printf("InitLibrary() Failure\n");
return FALSE;
}
<!--NeedCopy-->

Example 3: Log off a user

This function performs a log off from an SSO perspective, removing the user’s credentials from the system for the purposes of future SSO-based authorization. Existing authorized sessions can still be used. If the previous user was not logged off, that user’s credentials are restored (see example 4).

ICO/CCM interfaces can be used to perform full log off/disconnect.

void DoLogoffSsoUser()
{
  if ( LogoffSsoUser(0) == LOGONSSOUSER_OK )
printf("\n User logged off\n");
  else
 printf("\n Error logging off user\n”);
}
<!--NeedCopy-->

Example 4: Managing an exclusive endpoint

If the current user is not logged off and a new user is logged on, the previous user’s credentials is saved by the system to be restored when the current user logs off. Only a single level of restore is supported. Logging off a second time without an intervening logon results in clearing the SSO credentials and requires authentication before launching new sessions. This supports the following scenario using Controller, which is a custom application using CtxCredApi, ICO:

  1. A kiosk is auto logged on with a default/generic user (can be the normal Windows Logon or Controller calls LogonSsoUser()).
  2. The kiosk is then available with generic credentials for customer use.
  3. Sales Rep A logs on with their own credentials:
    1. Controller calls LogonSsoUser().
    2. Controller disconnects generic user’s sessions.
    3. Controller roams existing sessions for Sales Rep A.
  4. The kiosk can launch new sessions using Sales Rep A’s credentials.
  5. Sales Rep A logs off:
    1. Controller calls LogoffSsoUser(), removing Sales Rep A’s credentials and restoring the generic user.
    2. Controller restores generic user’s sessions.
  6. The kiosk is back to its initial state and is available for use with generic credentials.
  7. Sales Rep B logs on (repeat steps from 3 through 5 for Sales Rep B).
  8. The kiosk is back to its initial state.

This scenario can be repeated indefinitely.

Example 5: Managing user restore and shared endpoints

If the current user is not logged off and a new user is logged on, the previous user’s credentials is saved by the system to be restored when the current user logs off. Only a single level of restore is supported. Logging off a second time without an intervening logon results in clearing the SSO credentials and requires authentication before launching new sessions. This supports the following scenario using Controller, which is a custom application using CtxCredApi, ICO:

  1. A nurse logs on normally and launches sessions. (Controller calls LogonSsoUser() or can be a Windows Logon).
  2. A doctor “taps in”:
    1. Controller runs LogonSsoUser() with the doctor’s credentials.
    2. Controller does not log off the nurse or disconnect sessions.
    3. Controller causes any existing doctor’s sessions to be roamed to this endpoint.
  3. At this point, both the nurse’s and the doctor’s sessions are active. New sessions are launched using the doctor’s credentials.
  4. The doctor “taps out”:
    1. Controller calls LogoffSsoUser(), removing the doctor’s credentials and restoring the nurse’s.
    2. Controller disconnects the doctor’s sessions.
    3. The nurse’s sessions are still active.
  5. The nurse resumes as at Step 1.

Example 6: Checking for reconnecting sessions

Some applications may need to determine whether sessions are being reconnected before launching new instances. The ICO interface, ICACLient::EnumerateCCMSessions(), reports the IDs of the ICA sessions that already exist and that are being formed. This interface can be used by the caller to heuristically determine whether new ICA sessions are being formed, for example after Citrix Workspace app is restarted and is automatically reconnecting disconnected sessions.

After Citrix Workspace app restarts, the caller may call this operation several times at regular intervals for a reasonable period of time (for example, two seconds). If, during that time, the operation reports new ICA sessions, the caller can conclude that those sessions are the reconnected sessions. If the period of time expires and the operation reports no new ICA sessions, the caller can conclude that no session reconnect attempts are being made. The IsReconnectInProgress() API can also be used to check for reconnecting sessions, as shown in the following example:

for (int i = 0; i < 40000; i += 2000)
 {
 printf("IsReconnectInProgress %d seconds left to wait\n", (40000 - i) /
1000);
 BOOL bReconnectInProgress = FALSE;
 DWORD status = pfnIsReconnectInProgress(2000,
&bReconnectInProgress);
 if (status == ERROR_SUCCESS)
 {
 if (bReconnectInProgress)
  {
 printf("IsReconnectInProgress: Reconnect in Progress\n");
  }
  else
  {
 printf("IsReconnectInProgress: No Reconnect\n");
  }
  break;
 }
 else if (status == ERROR_TIMEOUT)
 {
 printf("IsReconnectInProgress: TIMEOUT\n");
 }
 else
 {
 printf("IsReconnectInProgress: Error: %d\n", status);
 break;
}
}
<!--NeedCopy-->
API examples