| /*************************************************************************** |
| * _ _ ____ _ |
| * Project ___| | | | _ \| | |
| * / __| | | | |_) | | |
| * | (__| |_| | _ <| |___ |
| * \___|\___/|_| \_\_____| |
| * |
| * Copyright (C) 1998 - 2020, Daniel Stenberg, <[email protected]>, et al. |
| * |
| * This software is licensed as described in the file COPYING, which |
| * you should have received as part of this distribution. The terms |
| * are also available at https://curl.se/docs/copyright.html. |
| * |
| * You may opt to use, copy, modify, merge, publish, distribute and/or sell |
| * copies of the Software, and permit persons to whom the Software is |
| * furnished to do so, under the terms of the COPYING file. |
| * |
| * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY |
| * KIND, either express or implied. |
| * |
| ***************************************************************************/ |
| #include "tool_setup.h" |
| |
| #ifdef HAVE_PWD_H |
| # undef __NO_NET_API /* required for building for AmigaOS */ |
| # include <pwd.h> |
| #endif |
| |
| #ifdef HAVE_SYS_STAT_H |
| #include <sys/stat.h> |
| #endif |
| #ifdef HAVE_FCNTL_H |
| #include <fcntl.h> |
| #endif |
| |
| #include <curl/mprintf.h> |
| |
| #include "tool_homedir.h" |
| |
| #include "memdebug.h" /* keep this as LAST include */ |
| |
| static char *GetEnv(const char *variable) |
| { |
| char *dupe, *env; |
| |
| env = curl_getenv(variable); |
| if(!env) |
| return NULL; |
| |
| dupe = strdup(env); |
| curl_free(env); |
| return dupe; |
| } |
| |
| /* return the home directory of the current user as an allocated string */ |
| |
| /* |
| * The original logic found a home dir to use (by checking a range of |
| * environment variables and last using getpwuid) and returned that for the |
| * parent to use. |
| * |
| * With the XDG_CONFIG_HOME support (added much later than the other), this |
| * variable is treated differently in order to not ruin existing installations |
| * even if this environment variable is set. If this variable is set, and a |
| * file name is set to check, then only if that file name exists in that |
| * directory will it be returned as a "home directory". |
| * |
| * 1. use CURL_HOME if set |
| * 2. use XDG_CONFIG_HOME if set and fname is present |
| * 3. use HOME if set |
| * 4. Non-windows: use getpwuid |
| * 5. Windows: use APPDATA if set |
| * 6. Windows: use "USERPROFILE\Application Data" is set |
| */ |
| |
| char *homedir(const char *fname) |
| { |
| char *home; |
| |
| home = GetEnv("CURL_HOME"); |
| if(home) |
| return home; |
| |
| if(fname) { |
| home = GetEnv("XDG_CONFIG_HOME"); |
| if(home) { |
| char *c = curl_maprintf("%s" DIR_CHAR "%s", home, fname); |
| if(c) { |
| int fd = open(c, O_RDONLY); |
| curl_free(c); |
| if(fd >= 0) { |
| close(fd); |
| return home; |
| } |
| } |
| free(home); |
| } |
| } |
| |
| home = GetEnv("HOME"); |
| if(home) |
| return home; |
| |
| #if defined(HAVE_GETPWUID) && defined(HAVE_GETEUID) |
| { |
| struct passwd *pw = getpwuid(geteuid()); |
| |
| if(pw) { |
| home = pw->pw_dir; |
| if(home && home[0]) |
| home = strdup(home); |
| else |
| home = NULL; |
| } |
| } |
| #endif /* PWD-stuff */ |
| #ifdef WIN32 |
| home = GetEnv("APPDATA"); |
| if(!home) { |
| char *env = GetEnv("USERPROFILE"); |
| if(env) { |
| char *path = curl_maprintf("%s\\Application Data", env); |
| if(path) { |
| home = strdup(path); |
| curl_free(path); |
| } |
| free(env); |
| } |
| } |
| #endif /* WIN32 */ |
| return home; |
| } |