/* Editor Settings: expandtabs and use 4 spaces for indentation
* ex: set softtabstop=4 tabstop=8 expandtab shiftwidth=4: *
* -*- mode: c, c-basic-offset: 4 -*- */
/*
* Copyright Likewise Software 2004-2008
* All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* for more details. You should have received a copy of the GNU General
* Public License along with this program. If not, see
* .
*
* LIKEWISE SOFTWARE MAKES THIS SOFTWARE AVAILABLE UNDER OTHER LICENSING
* TERMS AS WELL. IF YOU HAVE ENTERED INTO A SEPARATE LICENSE AGREEMENT
* WITH LIKEWISE SOFTWARE, THEN YOU MAY ELECT TO USE THE SOFTWARE UNDER THE
* TERMS OF THAT SOFTWARE LICENSE AGREEMENT INSTEAD OF THE TERMS OF THE GNU
* GENERAL PUBLIC LICENSE, NOTWITHSTANDING THE ABOVE NOTICE. IF YOU
* HAVE QUESTIONS, OR WISH TO REQUEST A COPY OF THE ALTERNATE LICENSING
* TERMS OFFERED BY LIKEWISE SOFTWARE, PLEASE CONTACT LIKEWISE SOFTWARE AT
* license@likewisesoftware.com
*/
/*
* Copyright (C) Likewise Software. All rights reserved.
*
* Module Name:
*
* main.c
*
* Abstract:
*
* Likewise Security and Authentication Subsystem (LSASS)
*
* Test Program for exercising LsaFindGroupByName
*
* Authors: Krishna Ganugapati (krishnag@likewisesoftware.com)
* Sriram Nambakam (snambakam@likewisesoftware.com)
*/
#define _POSIX_PTHREAD_SEMANTICS 1
#include "config.h"
#include "lsasystem.h"
#include "lsadef.h"
#include "lsa/lsa.h"
#include "lsaclient.h"
#include "lsaipc.h"
#define LW_PRINTF_STRING(x) ((x) ? (x) : "")
static
DWORD
ParseArgs(
int argc,
char* argv[],
PSTR* ppszGroupId,
PLSA_FIND_FLAGS pFindFlags,
PDWORD pdwInfoLevel,
PBOOLEAN pbCountOnly
);
static
VOID
ShowUsage();
static
VOID
PrintGroupInfo_0(
PLSA_GROUP_INFO_0 pGroupInfo
);
static
VOID
PrintGroupInfo_1(
PLSA_GROUP_INFO_1 pGroupInfo,
BOOLEAN bCountOnly
);
static
DWORD
MapErrorCode(
DWORD dwError
);
static
BOOLEAN
IsUnsignedInteger(
PCSTR pszIntegerCandidate
);
int
find_group_by_name_main(
int argc,
char* argv[]
)
{
DWORD dwError = 0;
PSTR pszGroupId = NULL;
DWORD dwInfoLevel = 0;
HANDLE hLsaConnection = (HANDLE)NULL;
PVOID pGroupInfo = NULL;
size_t dwErrorBufferSize = 0;
BOOLEAN bPrintOrigError = TRUE;
BOOLEAN bCountOnly = FALSE;
LSA_FIND_FLAGS FindFlags = 0;
dwError = ParseArgs(argc, argv, &pszGroupId, &FindFlags, &dwInfoLevel, &bCountOnly);
BAIL_ON_LSA_ERROR(dwError);
dwError = LsaOpenServer(&hLsaConnection);
BAIL_ON_LSA_ERROR(dwError);
dwError = LsaFindGroupByName(
hLsaConnection,
pszGroupId,
FindFlags,
dwInfoLevel,
&pGroupInfo);
BAIL_ON_LSA_ERROR(dwError);
switch(dwInfoLevel)
{
case 0:
PrintGroupInfo_0((PLSA_GROUP_INFO_0)pGroupInfo);
break;
case 1:
PrintGroupInfo_1((PLSA_GROUP_INFO_1)pGroupInfo, bCountOnly);
break;
default:
fprintf(stderr, "Error: Invalid group info level [%d]\n", dwInfoLevel);
break;
}
cleanup:
if (pGroupInfo) {
LsaFreeGroupInfo(dwInfoLevel, pGroupInfo);
}
if (hLsaConnection != (HANDLE)NULL) {
LsaCloseServer(hLsaConnection);
}
LW_SAFE_FREE_STRING(pszGroupId);
return (dwError);
error:
dwError = MapErrorCode(dwError);
dwErrorBufferSize = LwGetErrorString(dwError, NULL, 0);
if (dwErrorBufferSize > 0)
{
DWORD dwError2 = 0;
PSTR pszErrorBuffer = NULL;
dwError2 = LwAllocateMemory(
dwErrorBufferSize,
(PVOID*)&pszErrorBuffer);
if (!dwError2)
{
DWORD dwLen = LwGetErrorString(dwError, pszErrorBuffer, dwErrorBufferSize);
if ((dwLen == dwErrorBufferSize) && !LW_IS_NULL_OR_EMPTY_STR(pszErrorBuffer))
{
fprintf(stderr,
"Failed to locate group. Error code %u (%s).\n%s\n",
dwError,
LW_PRINTF_STRING(LwWin32ExtErrorToName(dwError)),
pszErrorBuffer);
bPrintOrigError = FALSE;
}
}
LW_SAFE_FREE_STRING(pszErrorBuffer);
}
if (bPrintOrigError)
{
fprintf(stderr,
"Failed to locate group. Error code %u (%s).\n",
dwError,
LW_PRINTF_STRING(LwWin32ExtErrorToName(dwError)));
}
goto cleanup;
}
DWORD
ParseArgs(
int argc,
char* argv[],
PSTR* ppszGroupId,
PLSA_FIND_FLAGS pFindFlags,
PDWORD pdwInfoLevel,
PBOOLEAN pbCountOnly
)
{
DWORD dwError = 0;
int iArg = 1;
PSTR pszArg = NULL;
PSTR pszGroupId = NULL;
LSA_FIND_FLAGS FindFlags = 0;
DWORD dwInfoLevel = 0;
BOOLEAN bCountOnly = FALSE;
for (iArg = 1; iArg < argc; iArg++)
{
pszArg = argv[iArg];
if (!strcmp(pszArg, "--help") ||
!strcmp(pszArg, "-h"))
{
ShowUsage();
exit(0);
}
else if (!strcmp(pszArg, "--count"))
{
bCountOnly = TRUE;
}
else if (!strcmp(pszArg, "--level"))
{
PCSTR pszValue;
if (iArg + 1 >= argc)
{
fprintf(stderr, "Missing argument for %s option.\n", pszArg);
ShowUsage();
exit(1);
}
pszValue = argv[++iArg];
if (!IsUnsignedInteger(pszValue))
{
fprintf(stderr, "Please enter an info level which is an unsigned integer.\n");
ShowUsage();
exit(1);
}
dwInfoLevel = atoi(pszValue);
}
else if (!strcmp(pszArg, "--flags"))
{
PCSTR pszValue;
if (iArg + 1 >= argc)
{
fprintf(stderr, "Missing argument for %s option.\n", pszArg);
ShowUsage();
exit(1);
}
pszValue = argv[++iArg];
if (!IsUnsignedInteger(pszValue))
{
fprintf(stderr, "Please enter a flags value which is an unsigned integer.\n");
ShowUsage();
exit(1);
}
FindFlags = atoi(pszValue);
}
else if (pszArg[0] == '-')
{
fprintf(stderr, "Invalid option '%s'.\n", pszArg);
ShowUsage();
exit(1);
}
else
{
break;
}
}
if ((argc - iArg) < 1)
{
fprintf(stderr, "Missing required group name argument.\n");
ShowUsage();
exit(1);
}
dwError = LwAllocateString(argv[iArg++], &pszGroupId);
BAIL_ON_LSA_ERROR(dwError);
if ((argc - iArg) > 0)
{
fprintf(stderr, "Too many arguments.\n");
ShowUsage();
exit(1);
}
if (LW_IS_NULL_OR_EMPTY_STR(pszGroupId))
{
fprintf(stderr, "Please specify a non-empty group name to query for.\n");
ShowUsage();
exit(1);
}
*ppszGroupId = pszGroupId;
*pFindFlags = FindFlags;
*pdwInfoLevel = dwInfoLevel;
*pbCountOnly = bCountOnly;
cleanup:
return dwError;
error:
*ppszGroupId = NULL;
*pFindFlags = 0;
*pdwInfoLevel = 0;
*pbCountOnly = FALSE;
LW_SAFE_FREE_STRING(pszGroupId);
goto cleanup;
}
void
ShowUsage()
{
PCSTR pszProgramName = "lw-find-group-by-name";
printf("Usage: %s [OPTIONS] \n"
"\n"
" Lookup a group by name.\n"
"\n"
" Options:\n"
"\n"
" --level LEVEL - Output level can be 0 or 1.\n"
" 0 does not include membership info.\n"
" 1 include group membership info.\n"
"\n"
" --count - If used with level 1, shows membership count only\n"
" instead of listing group members.\n"
"\n"
" --flags FLAGS - Find flags can be 0 or 1.\n"
"\n"
" Examples:\n"
"\n"
" %s DOMAIN\\\\groupname\n"
" %s --level 1 groupalias\n"
"\n",
pszProgramName,
pszProgramName,
pszProgramName);
}
VOID
PrintGroupInfo_0(
PLSA_GROUP_INFO_0 pGroupInfo
)
{
printf("Group info (Level 0):\n"
"====================\n"
"Name: %s\n"
"Gid: %u\n"
"SID: %s\n",
LSA_SAFE_LOG_STRING(pGroupInfo->pszName),
(unsigned int)pGroupInfo->gid,
LSA_SAFE_LOG_STRING(pGroupInfo->pszSid));
}
VOID
PrintGroupInfo_1(
PLSA_GROUP_INFO_1 pGroupInfo,
BOOLEAN bCountOnly
)
{
PSTR* ppszMembers = NULL;
DWORD iMember = 0;
printf("Group info (Level 1):\n"
"====================\n"
"Name: %s\n"
"Gid: %u\n"
"SID: %s\n",
LSA_SAFE_LOG_STRING(pGroupInfo->pszName),
(unsigned int)pGroupInfo->gid,
LSA_SAFE_LOG_STRING(pGroupInfo->pszSid));
if (!bCountOnly)
{
printf("Members:\n");
}
ppszMembers = pGroupInfo->ppszMembers;
if (ppszMembers)
{
while (!LW_IS_NULL_OR_EMPTY_STR(ppszMembers[iMember]))
{
if (!bCountOnly)
{
printf("%s\n", ppszMembers[iMember]);
}
iMember++;
}
}
printf("Members Count: %d\n", iMember);
}
DWORD
MapErrorCode(
DWORD dwError
)
{
DWORD dwError2 = dwError;
switch (dwError)
{
case ECONNREFUSED:
case ENETUNREACH:
case ETIMEDOUT:
dwError2 = LW_ERROR_LSA_SERVER_UNREACHABLE;
break;
default:
break;
}
return dwError2;
}
BOOLEAN
IsUnsignedInteger(
PCSTR pszIntegerCandidate
)
{
typedef enum {
PARSE_MODE_LEADING_SPACE = 0,
PARSE_MODE_INTEGER,
PARSE_MODE_TRAILING_SPACE
} ParseMode;
ParseMode parseMode = PARSE_MODE_LEADING_SPACE;
BOOLEAN bIsUnsignedInteger = TRUE;
INT iLength = 0;
INT iCharIdx = 0;
CHAR cNext = '\0';
if (LW_IS_NULL_OR_EMPTY_STR(pszIntegerCandidate))
{
bIsUnsignedInteger = FALSE;
goto error;
}
iLength = strlen(pszIntegerCandidate);
do {
cNext = pszIntegerCandidate[iCharIdx++];
switch(parseMode) {
case PARSE_MODE_LEADING_SPACE:
{
if (isdigit((int)cNext))
{
parseMode = PARSE_MODE_INTEGER;
}
else if (!isspace((int)cNext))
{
bIsUnsignedInteger = FALSE;
}
break;
}
case PARSE_MODE_INTEGER:
{
if (isspace((int)cNext))
{
parseMode = PARSE_MODE_TRAILING_SPACE;
}
else if (!isdigit((int)cNext))
{
bIsUnsignedInteger = FALSE;
}
break;
}
case PARSE_MODE_TRAILING_SPACE:
{
if (!isspace((int)cNext))
{
bIsUnsignedInteger = FALSE;
}
break;
}
}
} while (iCharIdx < iLength && bIsUnsignedInteger == TRUE);
error:
return bIsUnsignedInteger;
}