/**************************************************************************** * apps/fsutils/passwd/passwd_find.c * * Copyright (C) 2016 Gregory Nutt. All rights reserved. * Author: Gregory Nutt * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * 3. Neither the name NuttX nor the names of its contributors may be * used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include "passwd.h" /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: passwd_find * * Description: * Find a password in the * * Input Parameters: * * Returned Value: * Zero (OK) is returned on success; a negated errno value is returned on * failure. * ****************************************************************************/ int passwd_find(FAR const char *username, FAR struct passwd_s *passwd) { FAR char *iobuffer; FAR char *name; FAR char *encrypted; FAR char *ptr; FILE *stream; off_t offset; int ret; /* Allocate an I/O buffer for the transfer */ iobuffer = (FAR char *)malloc(CONFIG_FSUTILS_PASSWD_IOBUFFER_SIZE); if (iobuffer == NULL) { return -ENOMEM; } /* Open the password file for reading */ stream = fopen(CONFIG_FSUTILS_PASSWD_PATH, "r"); if (stream == NULL) { int errcode = errno; DEBUGASSERT(errcode > 0); return -errcode; } /* Read the password file line by line until the record with the matching * username is found, or until the end of the file is reached. * * The format of the password file is: * * user:x:uid:gid:home * * Where: * user: User name * x: Encrypted password * uid: User ID * gid: Group ID * home: Login directory */ offset = 0; ret = -ENOENT; while (fgets(iobuffer, CONFIG_FSUTILS_PASSWD_IOBUFFER_SIZE, stream) != NULL) { ptr = iobuffer; name = ptr; /* Skip to the end of the name and properly terminate it,. The name * must be terminated with the field delimiter ':'. */ for (; *ptr != '\0' && *ptr != ':'; ptr++); if (*ptr == '\0') { /* Bad file format? */ continue; } *ptr++ = '\0'; /* Check for a username match */ if (strcmp(username, name) == 0) { /* We have a match. The encrypted password must immediately * follow the ':' delimiter. */ encrypted = ptr; /* Skip to the end of the encrypted password and properly * terminate it. */ for (; *ptr != '\0' && *ptr != ':'; ptr++); if (*ptr == '\0') { /* Bad file format? */ ret = -EINVAL; break; } *ptr++ = '\0'; /* Copy the offset and password into the returned structure */ if (strlen(encrypted) >= MAX_ENCRYPTED) { ret = -E2BIG; break; } passwd->offset = offset; strncpy(passwd->encrypted, encrypted, MAX_ENCRYPTED); passwd->encrypted[MAX_ENCRYPTED] = '\0'; ret = OK; break; } /* Get the next file offset */ offset = ftell(stream); } fclose(stream); free(iobuffer); return ret; }