Home | History | Annotate | Download | only in xdg_user_dirs
      1 /*
      2   This file is not licenced under the GPL like the rest of the code.
      3   Its is under the MIT license, to encourage reuse by cut-and-paste.
      4 
      5   Copyright (c) 2007 Red Hat, inc
      6 
      7   Permission is hereby granted, free of charge, to any person
      8   obtaining a copy of this software and associated documentation files
      9   (the "Software"), to deal in the Software without restriction,
     10   including without limitation the rights to use, copy, modify, merge,
     11   publish, distribute, sublicense, and/or sell copies of the Software,
     12   and to permit persons to whom the Software is furnished to do so,
     13   subject to the following conditions:
     14 
     15   The above copyright notice and this permission notice shall be
     16   included in all copies or substantial portions of the Software.
     17 
     18   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
     19   EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
     20   MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
     21   NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
     22   BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
     23   ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
     24   CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
     25   SOFTWARE.
     26 */
     27 
     28 #include <stdio.h>
     29 #include <stdlib.h>
     30 #include <string.h>
     31 
     32 /**
     33  * xdg_user_dir_lookup_with_fallback:
     34  * @type: a string specifying the type of directory
     35  * @fallback: value to use if the directory isn't specified by the user
     36  * @returns: a newly allocated absolute pathname
     37  *
     38  * Looks up a XDG user directory of the specified type.
     39  * Example of types are "DESKTOP" and "DOWNLOAD".
     40  *
     41  * In case the user hasn't specified any directory for the specified
     42  * type the value returned is @fallback.
     43  *
     44  * The return value is newly allocated and must be freed with
     45  * free(). The return value is never NULL if @fallback != NULL, unless
     46  * out of memory.
     47  **/
     48 static char *
     49 xdg_user_dir_lookup_with_fallback (const char *type, const char *fallback)
     50 {
     51   FILE *file;
     52   char *home_dir, *config_home, *config_file;
     53   char buffer[512];
     54   char *user_dir;
     55   char *p, *d;
     56   int len;
     57   int relative;
     58 
     59   home_dir = getenv ("HOME");
     60 
     61   if (home_dir == NULL)
     62     goto error;
     63 
     64   config_home = getenv ("XDG_CONFIG_HOME");
     65   if (config_home == NULL || config_home[0] == 0)
     66     {
     67       config_file = (char*) malloc (strlen (home_dir) + strlen ("/.config/user-dirs.dirs") + 1);
     68       if (config_file == NULL)
     69         goto error;
     70 
     71       strcpy (config_file, home_dir);
     72       strcat (config_file, "/.config/user-dirs.dirs");
     73     }
     74   else
     75     {
     76       config_file = (char*) malloc (strlen (config_home) + strlen ("/user-dirs.dirs") + 1);
     77       if (config_file == NULL)
     78         goto error;
     79 
     80       strcpy (config_file, config_home);
     81       strcat (config_file, "/user-dirs.dirs");
     82     }
     83 
     84   file = fopen (config_file, "r");
     85   free (config_file);
     86   if (file == NULL)
     87     goto error;
     88 
     89   user_dir = NULL;
     90   while (fgets (buffer, sizeof (buffer), file))
     91     {
     92       /* Remove newline at end */
     93       len = strlen (buffer);
     94       if (len > 0 && buffer[len-1] == '\n')
     95 	buffer[len-1] = 0;
     96 
     97       p = buffer;
     98       while (*p == ' ' || *p == '\t')
     99 	p++;
    100 
    101       if (strncmp (p, "XDG_", 4) != 0)
    102 	continue;
    103       p += 4;
    104       if (strncmp (p, type, strlen (type)) != 0)
    105 	continue;
    106       p += strlen (type);
    107       if (strncmp (p, "_DIR", 4) != 0)
    108 	continue;
    109       p += 4;
    110 
    111       while (*p == ' ' || *p == '\t')
    112 	p++;
    113 
    114       if (*p != '=')
    115 	continue;
    116       p++;
    117 
    118       while (*p == ' ' || *p == '\t')
    119 	p++;
    120 
    121       if (*p != '"')
    122 	continue;
    123       p++;
    124 
    125       relative = 0;
    126       if (strncmp (p, "$HOME/", 6) == 0)
    127 	{
    128 	  p += 6;
    129 	  relative = 1;
    130 	}
    131       else if (*p != '/')
    132 	continue;
    133 
    134       if (relative)
    135 	{
    136 	  user_dir = (char*) malloc (strlen (home_dir) + 1 + strlen (p) + 1);
    137           if (user_dir == NULL)
    138             goto error2;
    139 
    140 	  strcpy (user_dir, home_dir);
    141 	  strcat (user_dir, "/");
    142 	}
    143       else
    144 	{
    145 	  user_dir = (char*) malloc (strlen (p) + 1);
    146           if (user_dir == NULL)
    147             goto error2;
    148 
    149 	  *user_dir = 0;
    150 	}
    151 
    152       d = user_dir + strlen (user_dir);
    153       while (*p && *p != '"')
    154 	{
    155 	  if ((*p == '\\') && (*(p+1) != 0))
    156 	    p++;
    157 	  *d++ = *p++;
    158 	}
    159       *d = 0;
    160     }
    161 error2:
    162   fclose (file);
    163 
    164   if (user_dir)
    165     return user_dir;
    166 
    167  error:
    168   if (fallback)
    169     return strdup (fallback);
    170   return NULL;
    171 }
    172 
    173 /**
    174  * xdg_user_dir_lookup:
    175  * @type: a string specifying the type of directory
    176  * @returns: a newly allocated absolute pathname
    177  *
    178  * Looks up a XDG user directory of the specified type.
    179  * Example of types are "DESKTOP" and "DOWNLOAD".
    180  *
    181  * The return value is always != NULL (unless out of memory),
    182  * and if a directory
    183  * for the type is not specified by the user the default
    184  * is the home directory. Except for DESKTOP which defaults
    185  * to ~/Desktop.
    186  *
    187  * The return value is newly allocated and must be freed with
    188  * free().
    189  **/
    190 char *
    191 xdg_user_dir_lookup (const char *type)
    192 {
    193   char *dir, *home_dir, *user_dir;
    194 
    195   dir = xdg_user_dir_lookup_with_fallback (type, NULL);
    196   if (dir != NULL)
    197     return dir;
    198 
    199   home_dir = getenv ("HOME");
    200 
    201   if (home_dir == NULL)
    202     return strdup ("/tmp");
    203 
    204   /* Special case desktop for historical compatibility */
    205   if (strcmp (type, "DESKTOP") == 0)
    206     {
    207       user_dir = (char*) malloc (strlen (home_dir) + strlen ("/Desktop") + 1);
    208       if (user_dir == NULL)
    209         return NULL;
    210 
    211       strcpy (user_dir, home_dir);
    212       strcat (user_dir, "/Desktop");
    213       return user_dir;
    214     }
    215 
    216   return strdup (home_dir);
    217 }
    218 
    219 #ifdef STANDALONE_XDG_USER_DIR_LOOKUP
    220 int
    221 main (int argc, char *argv[])
    222 {
    223   if (argc != 2)
    224     {
    225       fprintf (stderr, "Usage %s <dir-type>\n", argv[0]);
    226       exit (1);
    227     }
    228 
    229   printf ("%s\n", xdg_user_dir_lookup (argv[1]));
    230   return 0;
    231 }
    232 #endif
    233