Search
lxdream.org :: lxdream/src/lxpaths.c
lxdream 0.9.1
released Jun 29
Download Now
filename src/lxpaths.c
changeset 1283:2cbafe321d6f
prev1205:a486ac64f34b
next1286:8376a612a79d
author nkeynes
date Tue Mar 27 17:39:05 2012 +1000 (12 years ago)
permissions -rw-r--r--
last change Treat quotes as entering a double-quoted string (mostly redundant, but we
generate them in the path escaping for some reason)
file annotate diff log raw
nkeynes@1041
     1
/**
nkeynes@1041
     2
 * $Id$
nkeynes@1041
     3
 *
nkeynes@1041
     4
 * GUI helper functions that aren't specific to any particular implementation.
nkeynes@1041
     5
 *
nkeynes@1041
     6
 * Copyright (c) 2009 Nathan Keynes.
nkeynes@1041
     7
 *
nkeynes@1041
     8
 * This program is free software; you can redistribute it and/or modify
nkeynes@1041
     9
 * it under the terms of the GNU General Public License as published by
nkeynes@1041
    10
 * the Free Software Foundation; either version 2 of the License, or
nkeynes@1041
    11
 * (at your option) any later version.
nkeynes@1041
    12
 *
nkeynes@1041
    13
 * This program is distributed in the hope that it will be useful,
nkeynes@1041
    14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
nkeynes@1041
    15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
nkeynes@1041
    16
 * GNU General Public License for more details.
nkeynes@1041
    17
 */
nkeynes@1041
    18
nkeynes@1041
    19
#include <ctype.h>
nkeynes@1041
    20
#include <unistd.h>
nkeynes@1205
    21
#include <stdlib.h>
nkeynes@1041
    22
#include <glib/gstrfuncs.h>
nkeynes@1041
    23
#include <glib/gutils.h>
nkeynes@1041
    24
nkeynes@1041
    25
#include "gui.h"
nkeynes@1041
    26
#include "config.h"
nkeynes@1041
    27
nkeynes@1041
    28
static gchar *gui_paths[CONFIG_KEY_MAX];
nkeynes@1041
    29
nkeynes@1041
    30
/**
nkeynes@1041
    31
 * Test if we need to escape a path to prevent substitution mangling. 
nkeynes@1041
    32
 * @return TRUE if the input value contains any character that doesn't
nkeynes@1041
    33
 * match [a-zA-Z0-9._@%/] (this will escape slightly more than it needs to,
nkeynes@1041
    34
 * but is safe)
nkeynes@1041
    35
 */
nkeynes@1041
    36
static gboolean path_needs_escaping( const gchar *value )
nkeynes@1041
    37
{
nkeynes@1041
    38
   const gchar *p = value;
nkeynes@1041
    39
   while( *p ) {
nkeynes@1041
    40
       if( !isalnum(*p) && *p != '.' && *p != '_' &&
nkeynes@1041
    41
               *p != '@' && *p != '%' && *p != '/' ) {
nkeynes@1041
    42
           return TRUE;
nkeynes@1041
    43
       }
nkeynes@1041
    44
       p++;
nkeynes@1041
    45
   }
nkeynes@1041
    46
   return FALSE;
nkeynes@1041
    47
}
nkeynes@1041
    48
nkeynes@1041
    49
gchar *get_escaped_path( const gchar *value )
nkeynes@1041
    50
{
nkeynes@1041
    51
    if( value != NULL && path_needs_escaping(value) ) {
nkeynes@1041
    52
        /* Escape with "", and backslash the remaining characters:
nkeynes@1041
    53
         *   \ " $ `
nkeynes@1041
    54
         */
nkeynes@1041
    55
        char buf[strlen(value)*2+3];  
nkeynes@1041
    56
        const char *s = value;
nkeynes@1041
    57
        char *p = buf;
nkeynes@1041
    58
        *p++ = '\"';
nkeynes@1041
    59
        while( *s ) {
nkeynes@1041
    60
            if( *s == '\\' || *s == '"' || *s == '$' || *s == '`' ) {
nkeynes@1041
    61
                *p++ = '\\';
nkeynes@1041
    62
            }
nkeynes@1041
    63
            *p++ = *s++;
nkeynes@1041
    64
        }
nkeynes@1041
    65
        *p++ = '\"';
nkeynes@1041
    66
        *p = '\0';
nkeynes@1041
    67
        return g_strdup(buf);
nkeynes@1041
    68
    } else {
nkeynes@1041
    69
        return g_strdup(value);
nkeynes@1041
    70
    }
nkeynes@1041
    71
}
nkeynes@1041
    72
nkeynes@1041
    73
gchar *get_expanded_path( const gchar *input )
nkeynes@1041
    74
{
nkeynes@1205
    75
    char result[PATH_MAX];
nkeynes@1205
    76
nkeynes@1205
    77
    char *d, *e;
nkeynes@1205
    78
    const char *s;
nkeynes@1205
    79
    d = result;
nkeynes@1205
    80
    e = result+sizeof(result)-1;
nkeynes@1205
    81
    s = input;
nkeynes@1283
    82
    gboolean inDQstring = FALSE;
nkeynes@1205
    83
nkeynes@1205
    84
    if( input == NULL )
nkeynes@1041
    85
        return NULL;
nkeynes@1205
    86
nkeynes@1205
    87
    while( *s ) {
nkeynes@1205
    88
        if( d == e ) {
nkeynes@1205
    89
            return g_strdup(input); /* expansion too long */
nkeynes@1205
    90
        }
nkeynes@1205
    91
        char c = *s++;
nkeynes@1205
    92
        if( c == '$' ) {
nkeynes@1205
    93
            if( *s == '{' ) {
nkeynes@1205
    94
                s++;
nkeynes@1205
    95
                const char *q = s;
nkeynes@1205
    96
                while( *q != '}' ) {
nkeynes@1205
    97
                    if( ! *q ) {
nkeynes@1205
    98
                        return g_strdup(input); /* unterminated variable */
nkeynes@1205
    99
                    }
nkeynes@1205
   100
                    q++;
nkeynes@1205
   101
                }
nkeynes@1205
   102
                char *tmp = g_strndup(s, (q-s));
nkeynes@1205
   103
                s = q+1;
nkeynes@1205
   104
                char *value = getenv(tmp);
nkeynes@1205
   105
                g_free(tmp);
nkeynes@1205
   106
                if( value != NULL ) {
nkeynes@1205
   107
                    int len = strlen(value);
nkeynes@1205
   108
                    if( d + len > e )
nkeynes@1205
   109
                        return g_strdup(input);
nkeynes@1205
   110
                    strcpy(d, value);
nkeynes@1205
   111
                    d+=len;
nkeynes@1205
   112
                } /* Else, empty string */
nkeynes@1205
   113
            } else {
nkeynes@1205
   114
                const char *q = s;
nkeynes@1205
   115
                while( isalnum(*q) || *q == '_' ) {
nkeynes@1205
   116
                    q++;
nkeynes@1205
   117
                }
nkeynes@1205
   118
                if( q == s ) {
nkeynes@1205
   119
                    *d++ = '$';
nkeynes@1205
   120
                } else {
nkeynes@1205
   121
                    char *tmp = g_strndup(s,q-s);
nkeynes@1205
   122
                    s = q;
nkeynes@1205
   123
                    char *value = getenv(tmp);
nkeynes@1205
   124
                    g_free(tmp);
nkeynes@1205
   125
                    if( value != NULL ) {
nkeynes@1205
   126
                        int len = strlen(value);
nkeynes@1205
   127
                        if( d + len > e )
nkeynes@1205
   128
                            return g_strdup(input);
nkeynes@1205
   129
                        strcpy(d, value);
nkeynes@1205
   130
                        d += len;
nkeynes@1205
   131
                    }
nkeynes@1205
   132
                }
nkeynes@1205
   133
            }
nkeynes@1205
   134
        } else if( c == '\\' ) {
nkeynes@1205
   135
            c = *s++;
nkeynes@1205
   136
            if( c ) {
nkeynes@1205
   137
                *d++ = c;
nkeynes@1205
   138
            } else {
nkeynes@1205
   139
                *d++ = '\\';
nkeynes@1205
   140
            }
nkeynes@1283
   141
        } else if( c == '\"' ) {
nkeynes@1283
   142
            /* Unescaped double-quotes start a DQ string. Although we treat the
nkeynes@1283
   143
             * string as if it were double-quoted for most purposes anyway, so
nkeynes@1283
   144
             * this has little effect.
nkeynes@1283
   145
             */
nkeynes@1283
   146
            inDQstring = !inDQstring;
nkeynes@1205
   147
        } else {
nkeynes@1205
   148
            *d++ = c;
nkeynes@1205
   149
        }
nkeynes@1041
   150
    }
nkeynes@1205
   151
    *d = '\0';
nkeynes@1283
   152
    if( inDQstring ) {
nkeynes@1283
   153
        WARN( "Unterminated double-quoted string '%s'", input );
nkeynes@1283
   154
    }
nkeynes@1205
   155
    return g_strdup(result);
nkeynes@1041
   156
}
nkeynes@1041
   157
gchar *get_absolute_path( const gchar *in_path )
nkeynes@1041
   158
{
nkeynes@1041
   159
    char tmp[PATH_MAX];
nkeynes@1041
   160
    if( in_path == NULL ) {
nkeynes@1041
   161
        return NULL;
nkeynes@1041
   162
    }
nkeynes@1041
   163
    if( in_path[0] == '/' || in_path[0] == 0 ) {
nkeynes@1041
   164
        return g_strdup(in_path);
nkeynes@1041
   165
    } else {
nkeynes@1041
   166
        getcwd(tmp, sizeof(tmp));
nkeynes@1041
   167
        return g_strdup_printf("%s%c%s", tmp, G_DIR_SEPARATOR, in_path);
nkeynes@1041
   168
    }
nkeynes@1041
   169
}
nkeynes@1041
   170
nkeynes@1041
   171
gchar *get_filename_at( const gchar *at, const gchar *filename )
nkeynes@1041
   172
{
nkeynes@1041
   173
    char tmp[PATH_MAX];
nkeynes@1056
   174
    char *p = strrchr( at, '/' );
nkeynes@1041
   175
    if( p == NULL ) {
nkeynes@1041
   176
        /* No path at all, so just return filename */
nkeynes@1041
   177
        return g_strdup(filename);
nkeynes@1041
   178
    } else {
nkeynes@1056
   179
        int off = p-at;
nkeynes@1041
   180
        return g_strdup_printf("%.*s%c%s", off, at, G_DIR_SEPARATOR, filename );
nkeynes@1041
   181
    }
nkeynes@1041
   182
}
nkeynes@1041
   183
nkeynes@1041
   184
const gchar *get_gui_path( int key )
nkeynes@1041
   185
{
nkeynes@1041
   186
    if( gui_paths[key] == NULL ) {
nkeynes@1041
   187
        gui_paths[key] = lxdream_get_global_config_path_value(key);
nkeynes@1041
   188
        /* If no path defined, go with the current working directory */
nkeynes@1041
   189
        if( gui_paths[key] == NULL ) {
nkeynes@1041
   190
            gui_paths[key] = get_absolute_path(".");
nkeynes@1041
   191
        }
nkeynes@1041
   192
    }
nkeynes@1041
   193
    return gui_paths[key];
nkeynes@1041
   194
}
nkeynes@1041
   195
nkeynes@1041
   196
void set_gui_path( int key, const gchar *path )
nkeynes@1041
   197
{
nkeynes@1041
   198
    g_free(gui_paths[key]);
nkeynes@1041
   199
    gui_paths[key] = g_strdup(path);
nkeynes@1041
   200
}
nkeynes@1041
   201
nkeynes@1041
   202
void reset_gui_paths()
nkeynes@1041
   203
{
nkeynes@1041
   204
    int i;
nkeynes@1041
   205
    for( i=0; i < CONFIG_KEY_MAX; i++ ) {
nkeynes@1041
   206
        g_free(gui_paths[i]);
nkeynes@1041
   207
        gui_paths[i] = NULL;
nkeynes@1041
   208
    }
nkeynes@1041
   209
}
.