/* 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) * * Tool to Manage AD Join/Leave/Query * */ #define _POSIX_PTHREAD_SEMANTICS 1 #include "includes.h" #define LW_PRINTF_STRING(x) ((x) ? (x) : "") static DWORD ParseArgs( int argc, char* argv[], PLW_DOMAIN_INFO_REQUEST pDomainInfoRequest ); static VOID ShowUsage(); static DWORD MapErrorCode( DWORD dwError ); int lw_domain_main( int argc, char* argv[] ) { DWORD dwError = 0; LW_DOMAIN_INFO_REQUEST domainInfoRequest = {0}; size_t dwErrorBufferSize = 0; BOOLEAN bPrintOrigError = TRUE; PSTR pszComputerName = NULL; PSTR pszDnsDomainName = NULL; PSTR pszComputerDN = NULL; dwError = ParseArgs( argc, argv, &domainInfoRequest); BAIL_ON_LSA_ERROR(dwError); switch (domainInfoRequest.taskType) { case LW_DOMAIN_TASK_TYPE_JOIN: // TODO: Canonicalize the OU dwError = LwDomainJoin( domainInfoRequest.args.joinArgs.pszDomainName, domainInfoRequest.args.joinArgs.pszOU, domainInfoRequest.args.joinArgs.pszUsername, domainInfoRequest.args.joinArgs.pszPassword ); BAIL_ON_LSA_ERROR(dwError); break; case LW_DOMAIN_TASK_TYPE_LEAVE: dwError = LwDomainLeave( domainInfoRequest.args.joinArgs.pszUsername, domainInfoRequest.args.joinArgs.pszPassword ); BAIL_ON_LSA_ERROR(dwError); break; case LW_DOMAIN_TASK_TYPE_QUERY: dwError = LwDomainQuery( &pszComputerName, &pszDnsDomainName, &pszComputerDN); BAIL_ON_LSA_ERROR(dwError); fprintf(stdout, "Name = %s\n", LSA_SAFE_LOG_STRING(pszComputerName)); fprintf(stdout, "Domain = %s\n", LSA_SAFE_LOG_STRING(pszDnsDomainName)); fprintf(stdout, "Distinguished Name = %s\n", LSA_SAFE_LOG_STRING(pszComputerDN)); break; default: dwError = LW_ERROR_INVALID_PARAMETER; } cleanup: LwFreeDomainInfoRequest(&domainInfoRequest); LW_SAFE_FREE_STRING(pszComputerName); LW_SAFE_FREE_STRING(pszDnsDomainName); LW_SAFE_FREE_STRING(pszComputerDN); 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 enumerate maps. Error code %u (%s).\n%s", dwError, LW_PRINTF_STRING(LwWin32ExtErrorToName(dwError)), pszErrorBuffer); bPrintOrigError = FALSE; } } LW_SAFE_FREE_STRING(pszErrorBuffer); } if (bPrintOrigError) { fprintf(stderr, "Failed to enumerate maps. Error code %u (%s).\n", dwError, LW_PRINTF_STRING(LwWin32ExtErrorToName(dwError))); } dwError = 1; goto cleanup; } static DWORD ParseArgs( int argc, char* argv[], PLW_DOMAIN_INFO_REQUEST pDomainInfoRequest ) { typedef enum { PARSE_MODE_OPEN = 0, PARSE_MODE_JOIN_OU } ParseMode; DWORD dwError = 0; int iArg = 1; PSTR pszArg = NULL; ParseMode parseMode = PARSE_MODE_OPEN; LW_DOMAIN_INFO_REQUEST domainInfoRequest = {0}; do { pszArg = argv[iArg++]; if (pszArg == NULL || *pszArg == '\0') { break; } switch (parseMode) { case PARSE_MODE_OPEN: if ((strcmp(pszArg, "--help") == 0) || (strcmp(pszArg, "-h") == 0)) { ShowUsage(); exit(0); } else if (!strcmp(pszArg, "--join")) { if (domainInfoRequest.taskType == LW_DOMAIN_TASK_TYPE_UNKNOWN) { domainInfoRequest.taskType = LW_DOMAIN_TASK_TYPE_JOIN; } else { ShowUsage(); exit(1); } } else if (!strcmp(pszArg, "--leave")) { if (domainInfoRequest.taskType == LW_DOMAIN_TASK_TYPE_UNKNOWN) { domainInfoRequest.taskType = LW_DOMAIN_TASK_TYPE_LEAVE; } else { ShowUsage(); exit(1); } } else if (!strcmp(pszArg, "--query")) { if (domainInfoRequest.taskType == LW_DOMAIN_TASK_TYPE_UNKNOWN) { domainInfoRequest.taskType = LW_DOMAIN_TASK_TYPE_QUERY; } else { ShowUsage(); exit(1); } } else if (!strcmp(pszArg, "--ou")) { if (domainInfoRequest.taskType == LW_DOMAIN_TASK_TYPE_JOIN) { ShowUsage(); exit(1); } parseMode = PARSE_MODE_JOIN_OU; } else { if (domainInfoRequest.taskType == LW_DOMAIN_TASK_TYPE_JOIN) { if (LW_IS_NULL_OR_EMPTY_STR(domainInfoRequest.args.joinArgs.pszDomainName)) { dwError = LwAllocateString( pszArg, &domainInfoRequest.args.joinArgs.pszDomainName); BAIL_ON_LSA_ERROR(dwError); } else if (LW_IS_NULL_OR_EMPTY_STR(domainInfoRequest.args.joinArgs.pszUsername)) { dwError = LwAllocateString( pszArg, &domainInfoRequest.args.joinArgs.pszUsername); BAIL_ON_LSA_ERROR(dwError); } else if (LW_IS_NULL_OR_EMPTY_STR(domainInfoRequest.args.joinArgs.pszPassword)) { dwError = LwAllocateString( pszArg, &domainInfoRequest.args.joinArgs.pszPassword); BAIL_ON_LSA_ERROR(dwError); } else { ShowUsage(); exit(1); } } else if (domainInfoRequest.taskType == LW_DOMAIN_TASK_TYPE_LEAVE) { if (LW_IS_NULL_OR_EMPTY_STR(domainInfoRequest.args.leaveArgs.pszUsername)) { dwError = LwAllocateString( pszArg, &domainInfoRequest.args.leaveArgs.pszUsername); BAIL_ON_LSA_ERROR(dwError); } else if (LW_IS_NULL_OR_EMPTY_STR(domainInfoRequest.args.leaveArgs.pszPassword)) { dwError = LwAllocateString( pszArg, &domainInfoRequest.args.leaveArgs.pszPassword); BAIL_ON_LSA_ERROR(dwError); } else { ShowUsage(); exit(1); } } else { ShowUsage(); exit(1); } } break; case PARSE_MODE_JOIN_OU: if (domainInfoRequest.taskType != LW_DOMAIN_TASK_TYPE_JOIN) { ShowUsage(); exit(1); } LW_SAFE_FREE_STRING(domainInfoRequest.args.joinArgs.pszOU); dwError = LwAllocateString( pszArg, &domainInfoRequest.args.joinArgs.pszOU); BAIL_ON_LSA_ERROR(dwError); parseMode = PARSE_MODE_OPEN; break; } } while (iArg < argc); *pDomainInfoRequest = domainInfoRequest; cleanup: return dwError; error: LwFreeDomainInfoRequest(&domainInfoRequest); goto cleanup; } static void ShowUsage( VOID ) { printf("Usage: lw-domain { --join | --leave | --query }\n\n" "Join options: --ou \n" "Leave options: { }\n" ); } static 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; }