123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536 |
- /*
- paths to configuration files
- Copyright (C) 2010-2017
- Free Software Foundation, Inc.
- Written by:
- Slava Zanko <slavazanko@gmail.com>, 2010.
- This file is part of the Midnight Commander.
- The Midnight Commander 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 3 of the License,
- or (at your option) any later version.
- The Midnight Commander 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 <http://www.gnu.org/licenses/>.
- */
- #include <config.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <errno.h>
- #include "lib/global.h"
- #include "lib/fileloc.h"
- #include "lib/vfs/vfs.h"
- #include "lib/util.h" /* unix_error_string() */
- #include "lib/mcconfig.h"
- /*** global variables ****************************************************************************/
- /*** file scope macro definitions ****************************************************************/
- #define MC_OLD_USERCONF_DIR ".mc"
- /*** file scope type declarations ****************************************************************/
- /*** file scope variables ************************************************************************/
- static gboolean xdg_vars_initialized = FALSE;
- static char *mc_config_str = NULL;
- static char *mc_cache_str = NULL;
- static char *mc_data_str = NULL;
- static gboolean config_dir_present = FALSE;
- static const struct
- {
- const char *old_filename;
- char **new_basedir;
- const char *new_filename;
- } mc_config_files_reference[] =
- {
- /* *INDENT-OFF* */
- /* config */
- { "ini", &mc_config_str, MC_CONFIG_FILE},
- { "filehighlight.ini", &mc_config_str, MC_FHL_INI_FILE},
- { "hotlist", &mc_config_str, MC_HOTLIST_FILE},
- { "mc.keymap", &mc_config_str, GLOBAL_KEYMAP_FILE},
- { "menu", &mc_config_str, MC_USERMENU_FILE},
- { "cedit" PATH_SEP_STR "Syntax", &mc_config_str, EDIT_SYNTAX_FILE},
- { "cedit" PATH_SEP_STR "menu", &mc_config_str, EDIT_HOME_MENU},
- { "cedit" PATH_SEP_STR "edit.indent.rc", &mc_config_str, EDIT_DIR PATH_SEP_STR "edit.indent.rc"},
- { "cedit" PATH_SEP_STR "edit.spell.rc", &mc_config_str, EDIT_DIR PATH_SEP_STR "edit.spell.rc"},
- { "panels.ini", &mc_config_str, MC_PANELS_FILE},
- /* User should move this file with applying some changes in file */
- { "", &mc_config_str, MC_FILEBIND_FILE},
- /* data */
- { "skins", &mc_data_str, MC_SKINS_SUBDIR},
- { "fish", &mc_data_str, FISH_PREFIX},
- { "ashrc", &mc_data_str, "ashrc"},
- { "bashrc", &mc_data_str, "bashrc"},
- { "inputrc", &mc_data_str, "inputrc"},
- { "extfs.d", &mc_data_str, MC_EXTFS_DIR},
- { "history", &mc_data_str, MC_HISTORY_FILE},
- { "filepos", &mc_data_str, MC_FILEPOS_FILE},
- { "cedit" PATH_SEP_STR "cooledit.clip", &mc_data_str, EDIT_CLIP_FILE},
- { "", &mc_data_str, MC_MACRO_FILE},
- /* cache */
- { "log", &mc_cache_str, "mc.log"},
- { "Tree", &mc_cache_str, MC_TREESTORE_FILE},
- { "cedit" PATH_SEP_STR "cooledit.temp", &mc_cache_str, EDIT_TEMP_FILE},
- { "cedit" PATH_SEP_STR "cooledit.block", &mc_cache_str, EDIT_BLOCK_FILE},
- {NULL, NULL, NULL}
- /* *INDENT-ON* */
- };
- #if MC_HOMEDIR_XDG
- static const struct
- {
- char **old_basedir;
- const char *filename;
- char **new_basedir;
- } mc_config_migrate_rules_fix[] =
- {
- /* *INDENT-OFF* */
- { &mc_data_str, MC_USERMENU_FILE, &mc_config_str},
- { &mc_data_str, EDIT_SYNTAX_FILE, &mc_config_str},
- { &mc_data_str, EDIT_HOME_MENU, &mc_config_str},
- { &mc_data_str, EDIT_DIR PATH_SEP_STR "edit.indent.rc", &mc_config_str},
- { &mc_data_str, EDIT_DIR PATH_SEP_STR "edit.spell.rc", &mc_config_str},
- { &mc_data_str, MC_FILEBIND_FILE, &mc_config_str},
- { &mc_cache_str, MC_HISTORY_FILE, &mc_data_str},
- { &mc_cache_str, MC_FILEPOS_FILE, &mc_data_str},
- { &mc_cache_str, EDIT_CLIP_FILE, &mc_data_str},
- { &mc_cache_str, MC_PANELS_FILE, &mc_config_str},
- {NULL, NULL, NULL}
- /* *INDENT-ON* */
- };
- #endif /* MC_HOMEDIR_XDG */
- /*** file scope functions *********************************************************************** */
- /* --------------------------------------------------------------------------------------------- */
- static void
- mc_config_mkdir (const char *directory_name, GError ** mcerror)
- {
- mc_return_if_error (mcerror);
- if ((!g_file_test (directory_name, G_FILE_TEST_EXISTS | G_FILE_TEST_IS_DIR)) &&
- (g_mkdir_with_parents (directory_name, 0700) != 0))
- mc_propagate_error (mcerror, 0, _("Cannot create %s directory"), directory_name);
- }
- /* --------------------------------------------------------------------------------------------- */
- static char *
- mc_config_init_one_config_path (const char *path_base, const char *subdir, GError ** mcerror)
- {
- char *full_path;
- mc_return_val_if_error (mcerror, FALSE);
- full_path = g_build_filename (path_base, subdir, (char *) NULL);
- if (g_file_test (full_path, G_FILE_TEST_EXISTS))
- {
- if (g_file_test (full_path, G_FILE_TEST_IS_DIR))
- {
- config_dir_present = TRUE;
- }
- else
- {
- fprintf (stderr, "%s %s\n", _("FATAL: not a directory:"), full_path);
- exit (EXIT_FAILURE);
- }
- }
- mc_config_mkdir (full_path, mcerror);
- if (mcerror != NULL && *mcerror != NULL)
- MC_PTR_FREE (full_path);
- return full_path;
- }
- /* --------------------------------------------------------------------------------------------- */
- static char *
- mc_config_get_deprecated_path (void)
- {
- return g_build_filename (mc_config_get_home_dir (), MC_OLD_USERCONF_DIR, (char *) NULL);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- mc_config_copy (const char *old_name, const char *new_name, GError ** mcerror)
- {
- mc_return_if_error (mcerror);
- if (g_file_test (old_name, G_FILE_TEST_IS_REGULAR))
- {
- char *contents = NULL;
- size_t length;
- if (g_file_get_contents (old_name, &contents, &length, mcerror))
- g_file_set_contents (new_name, contents, length, mcerror);
- g_free (contents);
- return;
- }
- if (g_file_test (old_name, G_FILE_TEST_IS_DIR))
- {
- GDir *dir;
- const char *dir_name;
- dir = g_dir_open (old_name, 0, mcerror);
- if (dir == NULL)
- return;
- if (g_mkdir_with_parents (new_name, 0700) == -1)
- {
- g_dir_close (dir);
- mc_propagate_error (mcerror, 0,
- _("An error occurred while migrating user settings: %s"),
- unix_error_string (errno));
- return;
- }
- while ((dir_name = g_dir_read_name (dir)) != NULL)
- {
- char *old_name2, *new_name2;
- old_name2 = g_build_filename (old_name, dir_name, (char *) NULL);
- new_name2 = g_build_filename (new_name, dir_name, (char *) NULL);
- mc_config_copy (old_name2, new_name2, mcerror);
- g_free (new_name2);
- g_free (old_name2);
- }
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- #if MC_HOMEDIR_XDG
- static void
- mc_config_fix_migrated_rules (void)
- {
- size_t rule_index;
- for (rule_index = 0; mc_config_migrate_rules_fix[rule_index].old_basedir != NULL; rule_index++)
- {
- char *old_name;
- old_name =
- g_build_filename (*mc_config_migrate_rules_fix[rule_index].old_basedir,
- mc_config_migrate_rules_fix[rule_index].filename, (char *) NULL);
- if (g_file_test (old_name, G_FILE_TEST_EXISTS))
- {
- char *new_name;
- const char *basedir = *mc_config_migrate_rules_fix[rule_index].new_basedir;
- const char *filename = mc_config_migrate_rules_fix[rule_index].filename;
- new_name = g_build_filename (basedir, filename, (char *) NULL);
- rename (old_name, new_name);
- g_free (new_name);
- }
- g_free (old_name);
- }
- }
- #endif /* MC_HOMEDIR_XDG */
- /* --------------------------------------------------------------------------------------------- */
- static gboolean
- mc_config_deprecated_dir_present (void)
- {
- char *old_dir;
- gboolean is_present;
- old_dir = mc_config_get_deprecated_path ();
- is_present = g_file_test (old_dir, G_FILE_TEST_EXISTS | G_FILE_TEST_IS_DIR);
- g_free (old_dir);
- return is_present && !config_dir_present;
- }
- /* --------------------------------------------------------------------------------------------- */
- /*** public functions ****************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- void
- mc_config_init_config_paths (GError ** mcerror)
- {
- const char *profile_root;
- char *dir;
- #if MC_HOMEDIR_XDG == 0
- char *defined_userconf_dir;
- #endif
- mc_return_if_error (mcerror);
- if (xdg_vars_initialized)
- return;
- profile_root = mc_get_profile_root ();
- #if MC_HOMEDIR_XDG
- if (strcmp (profile_root, mc_config_get_home_dir ()) != 0)
- {
- /*
- * The user overrode the default profile root.
- *
- * In this case we can't use GLib's g_get_user_{config,cache,data}_dir()
- * as these functions use the user's home dir as the root.
- */
- dir = g_build_filename (profile_root, ".config", (char *) NULL);
- mc_config_str = mc_config_init_one_config_path (dir, MC_USERCONF_DIR, mcerror);
- g_free (dir);
- dir = g_build_filename (profile_root, ".cache", (char *) NULL);
- mc_cache_str = mc_config_init_one_config_path (dir, MC_USERCONF_DIR, mcerror);
- g_free (dir);
- dir = g_build_filename (profile_root, ".local", "share", (char *) NULL);
- mc_data_str = mc_config_init_one_config_path (dir, MC_USERCONF_DIR, mcerror);
- g_free (dir);
- }
- else
- {
- mc_config_str =
- mc_config_init_one_config_path (g_get_user_config_dir (), MC_USERCONF_DIR, mcerror);
- mc_cache_str =
- mc_config_init_one_config_path (g_get_user_cache_dir (), MC_USERCONF_DIR, mcerror);
- mc_data_str =
- mc_config_init_one_config_path (g_get_user_data_dir (), MC_USERCONF_DIR, mcerror);
- }
- mc_config_fix_migrated_rules ();
- #else /* MC_HOMEDIR_XDG */
- defined_userconf_dir = tilde_expand (MC_USERCONF_DIR);
- if (g_path_is_absolute (defined_userconf_dir))
- dir = defined_userconf_dir;
- else
- {
- g_free (defined_userconf_dir);
- dir = g_build_filename (profile_root, MC_USERCONF_DIR, (char *) NULL);
- }
- mc_data_str = mc_cache_str = mc_config_str = mc_config_init_one_config_path (dir, "", mcerror);
- g_free (dir);
- #endif /* MC_HOMEDIR_XDG */
- xdg_vars_initialized = TRUE;
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- mc_config_deinit_config_paths (void)
- {
- if (!xdg_vars_initialized)
- return;
- g_free (mc_config_str);
- #if MC_HOMEDIR_XDG
- g_free (mc_cache_str);
- g_free (mc_data_str);
- #endif /* MC_HOMEDIR_XDG */
- g_free (mc_global.share_data_dir);
- g_free (mc_global.sysconfig_dir);
- xdg_vars_initialized = FALSE;
- }
- /* --------------------------------------------------------------------------------------------- */
- const char *
- mc_config_get_data_path (void)
- {
- if (!xdg_vars_initialized)
- mc_config_init_config_paths (NULL);
- return (const char *) mc_data_str;
- }
- /* --------------------------------------------------------------------------------------------- */
- const char *
- mc_config_get_cache_path (void)
- {
- if (!xdg_vars_initialized)
- mc_config_init_config_paths (NULL);
- return (const char *) mc_cache_str;
- }
- /* --------------------------------------------------------------------------------------------- */
- const char *
- mc_config_get_home_dir (void)
- {
- static const char *homedir = NULL;
- if (homedir == NULL)
- {
- /* Prior to GLib 2.36, g_get_home_dir() ignores $HOME, which is why
- * we read it ourselves. As that function's documentation explains,
- * using $HOME is good for compatibility with other programs and
- * for running from test frameworks. */
- homedir = g_getenv ("HOME");
- if (homedir == NULL || *homedir == '\0')
- homedir = g_get_home_dir ();
- }
- return homedir;
- }
- /* --------------------------------------------------------------------------------------------- */
- const char *
- mc_config_get_path (void)
- {
- if (!xdg_vars_initialized)
- mc_config_init_config_paths (NULL);
- return (const char *) mc_config_str;
- }
- /* --------------------------------------------------------------------------------------------- */
- gboolean
- mc_config_migrate_from_old_place (GError ** mcerror, char **msg)
- {
- char *old_dir;
- size_t rule_index;
- mc_return_val_if_error (mcerror, FALSE);
- if (!mc_config_deprecated_dir_present ())
- return FALSE;
- old_dir = mc_config_get_deprecated_path ();
- g_free (mc_config_init_one_config_path (mc_config_str, EDIT_DIR, mcerror));
- #if MC_HOMEDIR_XDG
- g_free (mc_config_init_one_config_path (mc_cache_str, EDIT_DIR, mcerror));
- g_free (mc_config_init_one_config_path (mc_data_str, EDIT_DIR, mcerror));
- #endif /* MC_HOMEDIR_XDG */
- mc_return_val_if_error (mcerror, FALSE);
- for (rule_index = 0; mc_config_files_reference[rule_index].old_filename != NULL; rule_index++)
- {
- char *old_name;
- if (*mc_config_files_reference[rule_index].old_filename == '\0')
- continue;
- old_name =
- g_build_filename (old_dir, mc_config_files_reference[rule_index].old_filename,
- (char *) NULL);
- if (g_file_test (old_name, G_FILE_TEST_EXISTS))
- {
- char *new_name;
- const char *basedir = *mc_config_files_reference[rule_index].new_basedir;
- const char *filename = mc_config_files_reference[rule_index].new_filename;
- new_name = g_build_filename (basedir, filename, (char *) NULL);
- mc_config_copy (old_name, new_name, mcerror);
- g_free (new_name);
- }
- g_free (old_name);
- }
- #if MC_HOMEDIR_XDG
- *msg = g_strdup_printf (_("Your old settings were migrated from %s\n"
- "to Freedesktop recommended dirs.\n"
- "To get more info, please visit\n"
- "http://standards.freedesktop.org/basedir-spec/basedir-spec-latest.html"),
- old_dir);
- #else /* MC_HOMEDIR_XDG */
- *msg = g_strdup_printf (_("Your old settings were migrated from %s\n"
- "to %s\n"), old_dir, mc_config_str);
- #endif /* MC_HOMEDIR_XDG */
- g_free (old_dir);
- return TRUE;
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Get full path to config file by short name.
- *
- * @param config_name short name
- * @return full path to config file
- */
- char *
- mc_config_get_full_path (const char *config_name)
- {
- size_t rule_index;
- if (config_name == NULL)
- return NULL;
- if (!xdg_vars_initialized)
- mc_config_init_config_paths (NULL);
- for (rule_index = 0; mc_config_files_reference[rule_index].old_filename != NULL; rule_index++)
- {
- if (strcmp (config_name, mc_config_files_reference[rule_index].new_filename) == 0)
- {
- return g_build_filename (*mc_config_files_reference[rule_index].new_basedir,
- mc_config_files_reference[rule_index].new_filename,
- (char *) NULL);
- }
- }
- return NULL;
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Get full path to config file by short name.
- *
- * @param config_name short name
- * @return object with full path to config file
- */
- vfs_path_t *
- mc_config_get_full_vpath (const char *config_name)
- {
- vfs_path_t *ret_vpath;
- char *str_path;
- str_path = mc_config_get_full_path (config_name);
- ret_vpath = vfs_path_from_str (str_path);
- g_free (str_path);
- return ret_vpath;
- }
- /* --------------------------------------------------------------------------------------------- */
|