2011-10-15 14:56:47 +04:00
|
|
|
/*
|
|
|
|
Execution routines for GNU Midnight Commander
|
2010-05-11 16:16:58 +04:00
|
|
|
|
2024-01-01 09:46:17 +03:00
|
|
|
Copyright (C) 2003-2024
|
2014-02-12 10:33:10 +04:00
|
|
|
Free Software Foundation, Inc.
|
2010-05-11 16:16:58 +04:00
|
|
|
|
2013-04-14 17:38:37 +04:00
|
|
|
Written by:
|
|
|
|
Slava Zanko <slavazanko@gmail.com>, 2013
|
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
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,
|
2003-06-22 13:17:46 +04:00
|
|
|
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
|
2011-10-15 14:56:47 +04:00
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2009-02-05 21:28:18 +03:00
|
|
|
/** \file execute.c
|
|
|
|
* \brief Source: execution routines
|
|
|
|
*/
|
|
|
|
|
2003-06-22 13:17:46 +04:00
|
|
|
#include <config.h>
|
2005-02-08 12:04:03 +03:00
|
|
|
|
2009-08-12 15:37:22 +04:00
|
|
|
#include <signal.h>
|
2013-01-11 01:00:00 +04:00
|
|
|
#include <string.h>
|
2009-03-07 15:23:15 +03:00
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/time.h>
|
2005-02-08 12:04:03 +03:00
|
|
|
|
2010-01-20 18:11:52 +03:00
|
|
|
#include "lib/global.h"
|
2009-05-09 19:17:57 +04:00
|
|
|
|
2010-01-08 17:47:19 +03:00
|
|
|
#include "lib/tty/tty.h"
|
|
|
|
#include "lib/tty/key.h"
|
|
|
|
#include "lib/tty/win.h"
|
2011-02-15 16:44:17 +03:00
|
|
|
#include "lib/vfs/vfs.h"
|
2011-11-10 00:08:31 +04:00
|
|
|
#include "lib/mcconfig.h"
|
2010-11-11 16:58:29 +03:00
|
|
|
#include "lib/util.h"
|
2013-01-11 01:00:00 +04:00
|
|
|
#include "lib/strutil.h" /* str_replace_all_substrings() */
|
2010-11-12 11:03:57 +03:00
|
|
|
#include "lib/widget.h"
|
2009-05-09 19:17:57 +04:00
|
|
|
|
2020-08-19 09:33:24 +03:00
|
|
|
#include "filemanager/filemanager.h"
|
2010-11-22 17:15:28 +03:00
|
|
|
#include "filemanager/layout.h" /* use_dash() */
|
2010-01-21 17:20:11 +03:00
|
|
|
#include "consaver/cons.saver.h"
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2015-11-10 15:54:40 +03:00
|
|
|
#include "subshell/subshell.h"
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif
|
2010-11-22 14:45:18 +03:00
|
|
|
#include "setup.h" /* clear_before_exec */
|
|
|
|
|
2003-06-22 13:17:46 +04:00
|
|
|
#include "execute.h"
|
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/*** global variables ****************************************************************************/
|
|
|
|
|
2010-11-22 14:45:18 +03:00
|
|
|
int pause_after_run = pause_on_dumb_terminals;
|
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/*** file scope macro definitions ****************************************************************/
|
|
|
|
|
|
|
|
/*** file scope type declarations ****************************************************************/
|
|
|
|
|
Update template for .c files.
Add section for forward declarations of local functions. This section is
located before file scope variables because functions can be used in
strucutres (see find.c for example):
/*** forward declarations (file scope functions) *************************************************/
/* button callbacks */
static int start_stop (WButton * button, int action);
static int find_do_view_file (WButton * button, int action);
static int find_do_edit_file (WButton * button, int action);
/*** file scope variables ************************************************************************/
static struct
{
...
bcback_fn callback;
} fbuts[] =
{
...
{ B_STOP, NORMAL_BUTTON, N_("S&uspend"), 0, 0, NULL, start_stop },
...
{ B_VIEW, NORMAL_BUTTON, N_("&View - F3"), 0, 0, NULL, find_do_view_file },
{ B_VIEW, NORMAL_BUTTON, N_("&Edit - F4"), 0, 0, NULL, find_do_edit_file }
};
Signed-off-by: Andrew Borodin <aborodin@vmail.ru>
2023-02-24 09:27:11 +03:00
|
|
|
/*** forward declarations (file scope functions) *************************************************/
|
2013-01-10 01:31:51 +04:00
|
|
|
|
|
|
|
void do_execute (const char *shell, const char *command, int flags);
|
2013-01-10 21:20:58 +04:00
|
|
|
void do_executev (const char *shell, int flags, char *const argv[]);
|
|
|
|
char *execute_get_external_cmd_opts_from_config (const char *command,
|
2013-02-16 19:17:30 +04:00
|
|
|
const vfs_path_t * filename_vpath,
|
|
|
|
long start_line);
|
2013-01-10 01:31:51 +04:00
|
|
|
|
Update template for .c files.
Add section for forward declarations of local functions. This section is
located before file scope variables because functions can be used in
strucutres (see find.c for example):
/*** forward declarations (file scope functions) *************************************************/
/* button callbacks */
static int start_stop (WButton * button, int action);
static int find_do_view_file (WButton * button, int action);
static int find_do_edit_file (WButton * button, int action);
/*** file scope variables ************************************************************************/
static struct
{
...
bcback_fn callback;
} fbuts[] =
{
...
{ B_STOP, NORMAL_BUTTON, N_("S&uspend"), 0, 0, NULL, start_stop },
...
{ B_VIEW, NORMAL_BUTTON, N_("&View - F3"), 0, 0, NULL, find_do_view_file },
{ B_VIEW, NORMAL_BUTTON, N_("&Edit - F4"), 0, 0, NULL, find_do_edit_file }
};
Signed-off-by: Andrew Borodin <aborodin@vmail.ru>
2023-02-24 09:27:11 +03:00
|
|
|
/*** file scope variables ************************************************************************/
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/*** file scope functions ************************************************************************/
|
2010-11-10 14:09:42 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2003-06-22 13:17:46 +04:00
|
|
|
|
|
|
|
static void
|
|
|
|
edition_post_exec (void)
|
|
|
|
{
|
2016-05-04 13:29:47 +03:00
|
|
|
tty_enter_ca_mode ();
|
2003-06-22 13:17:46 +04:00
|
|
|
|
|
|
|
/* FIXME: Missing on slang endwin? */
|
2009-05-31 16:23:10 +04:00
|
|
|
tty_reset_prog_mode ();
|
2009-06-02 21:40:37 +04:00
|
|
|
tty_flush_input ();
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2009-05-31 16:23:10 +04:00
|
|
|
tty_keypad (TRUE);
|
2009-05-19 20:12:29 +04:00
|
|
|
tty_raw_mode ();
|
2003-06-22 13:17:46 +04:00
|
|
|
channels_up ();
|
|
|
|
enable_mouse ();
|
2013-10-01 13:31:13 +04:00
|
|
|
enable_bracketed_paste ();
|
2011-09-14 02:07:31 +04:00
|
|
|
if (mc_global.tty.alternate_plus_minus)
|
2010-05-11 16:16:58 +04:00
|
|
|
application_keypad_mode ();
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2003-06-22 13:17:46 +04:00
|
|
|
|
|
|
|
static void
|
|
|
|
edition_pre_exec (void)
|
|
|
|
{
|
|
|
|
if (clear_before_exec)
|
2020-10-31 14:26:10 +03:00
|
|
|
tty_clear_screen ();
|
2010-05-11 16:16:58 +04:00
|
|
|
else
|
|
|
|
{
|
2011-09-14 02:07:31 +04:00
|
|
|
if (!(mc_global.tty.console_flag != '\0' || mc_global.tty.xterm_flag))
|
2010-05-11 16:16:58 +04:00
|
|
|
printf ("\n\n");
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
channels_down ();
|
|
|
|
disable_mouse ();
|
2013-10-01 13:31:13 +04:00
|
|
|
disable_bracketed_paste ();
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2009-05-31 16:23:10 +04:00
|
|
|
tty_reset_shell_mode ();
|
|
|
|
tty_keypad (FALSE);
|
2009-06-02 21:40:37 +04:00
|
|
|
tty_reset_screen ();
|
2003-06-22 13:17:46 +04:00
|
|
|
|
|
|
|
numeric_keypad_mode ();
|
|
|
|
|
|
|
|
/* on xterms: maybe endwin did not leave the terminal on the shell
|
|
|
|
* screen page: do it now.
|
|
|
|
*
|
|
|
|
* Do not move this before endwin: in some systems rmcup includes
|
|
|
|
* a call to clear screen, so it will end up clearing the shell screen.
|
|
|
|
*/
|
2016-05-04 13:29:47 +03:00
|
|
|
tty_exit_ca_mode ();
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2009-08-05 13:28:59 +04:00
|
|
|
static void
|
2011-11-26 16:46:54 +04:00
|
|
|
do_possible_cd (const vfs_path_t * new_dir_vpath)
|
2009-08-05 13:28:59 +04:00
|
|
|
{
|
2020-12-18 14:34:08 +03:00
|
|
|
if (!panel_cd (current_panel, new_dir_vpath, cd_exact))
|
2017-05-25 21:32:46 +03:00
|
|
|
message (D_ERROR, _("Warning"), "%s",
|
2010-05-15 18:45:13 +04:00
|
|
|
_("The Commander can't change to the directory that\n"
|
|
|
|
"the subshell claims you are in. Perhaps you have\n"
|
|
|
|
"deleted your working directory, or given yourself\n"
|
|
|
|
"extra access permissions with the \"su\" command?"));
|
2009-08-05 13:28:59 +04:00
|
|
|
}
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif /* ENABLE_SUBSHELL */
|
2009-08-05 13:28:59 +04:00
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2003-06-22 14:15:04 +04:00
|
|
|
static void
|
2013-01-10 01:31:51 +04:00
|
|
|
do_suspend_cmd (void)
|
|
|
|
{
|
|
|
|
pre_exec ();
|
|
|
|
|
|
|
|
if (mc_global.tty.console_flag != '\0' && !mc_global.tty.use_subshell)
|
|
|
|
handle_console (CONSOLE_RESTORE);
|
|
|
|
|
|
|
|
#ifdef SIGTSTP
|
|
|
|
{
|
|
|
|
struct sigaction sigtstp_action;
|
|
|
|
|
2013-11-17 09:55:18 +04:00
|
|
|
memset (&sigtstp_action, 0, sizeof (sigtstp_action));
|
2013-01-10 01:31:51 +04:00
|
|
|
/* Make sure that the SIGTSTP below will suspend us directly,
|
|
|
|
without calling ncurses' SIGTSTP handler; we *don't* want
|
|
|
|
ncurses to redraw the screen immediately after the SIGCONT */
|
|
|
|
sigaction (SIGTSTP, &startup_handler, &sigtstp_action);
|
|
|
|
|
|
|
|
kill (getpid (), SIGTSTP);
|
|
|
|
|
|
|
|
/* Restore previous SIGTSTP action */
|
|
|
|
sigaction (SIGTSTP, &sigtstp_action, NULL);
|
|
|
|
}
|
|
|
|
#endif /* SIGTSTP */
|
|
|
|
|
|
|
|
if (mc_global.tty.console_flag != '\0' && !mc_global.tty.use_subshell)
|
|
|
|
handle_console (CONSOLE_SAVE);
|
|
|
|
|
|
|
|
edition_post_exec ();
|
|
|
|
}
|
|
|
|
|
2013-01-10 01:48:09 +04:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
static gboolean
|
2013-01-10 21:20:58 +04:00
|
|
|
execute_prepare_with_vfs_arg (const vfs_path_t * filename_vpath, vfs_path_t ** localcopy_vpath,
|
|
|
|
time_t * mtime)
|
2013-01-10 01:48:09 +04:00
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
|
|
|
|
/* Simplest case, this file is local */
|
|
|
|
if ((filename_vpath == NULL && vfs_file_is_local (vfs_get_raw_current_dir ()))
|
|
|
|
|| vfs_file_is_local (filename_vpath))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
/* FIXME: Creation of new files on VFS is not supported */
|
|
|
|
if (filename_vpath == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
*localcopy_vpath = mc_getlocalcopy (filename_vpath);
|
|
|
|
if (*localcopy_vpath == NULL)
|
|
|
|
{
|
2013-04-14 17:38:37 +04:00
|
|
|
message (D_ERROR, MSG_ERROR, _("Cannot fetch a local copy of %s"),
|
|
|
|
vfs_path_as_str (filename_vpath));
|
2013-01-10 01:48:09 +04:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
mc_stat (*localcopy_vpath, &st);
|
|
|
|
*mtime = st.st_mtime;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
static void
|
2013-01-10 21:20:58 +04:00
|
|
|
execute_cleanup_with_vfs_arg (const vfs_path_t * filename_vpath, vfs_path_t ** localcopy_vpath,
|
|
|
|
time_t * mtime)
|
2013-01-10 01:48:09 +04:00
|
|
|
{
|
|
|
|
if (*localcopy_vpath != NULL)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* filename can be an entry on panel, it can be changed by executing
|
|
|
|
* the command, so make a copy. Smarter VFS code would make the code
|
|
|
|
* below unnecessary.
|
|
|
|
*/
|
|
|
|
mc_stat (*localcopy_vpath, &st);
|
|
|
|
mc_ungetlocalcopy (filename_vpath, *localcopy_vpath, *mtime != st.st_mtime);
|
2021-02-21 19:30:18 +03:00
|
|
|
vfs_path_free (*localcopy_vpath, TRUE);
|
2013-01-10 01:48:09 +04:00
|
|
|
*localcopy_vpath = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-11 01:00:00 +04:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
static char *
|
|
|
|
execute_get_opts_from_cfg (const char *command, const char *default_str)
|
|
|
|
{
|
|
|
|
char *str_from_config;
|
|
|
|
|
|
|
|
str_from_config =
|
2016-04-26 09:30:42 +03:00
|
|
|
mc_config_get_string_raw (mc_global.main_config, CONFIG_EXT_EDITOR_VIEWER_SECTION, command,
|
|
|
|
NULL);
|
2013-01-11 01:00:00 +04:00
|
|
|
|
|
|
|
if (str_from_config == NULL)
|
|
|
|
{
|
|
|
|
mc_config_t *cfg;
|
|
|
|
|
2023-01-03 09:10:51 +03:00
|
|
|
cfg = mc_config_init (mc_global.profile_name, TRUE);
|
2013-01-11 01:00:00 +04:00
|
|
|
if (cfg == NULL)
|
|
|
|
return g_strdup (default_str);
|
|
|
|
|
|
|
|
str_from_config =
|
|
|
|
mc_config_get_string_raw (cfg, CONFIG_EXT_EDITOR_VIEWER_SECTION, command, default_str);
|
|
|
|
|
|
|
|
mc_config_deinit (cfg);
|
|
|
|
}
|
|
|
|
|
|
|
|
return str_from_config;
|
|
|
|
}
|
|
|
|
|
2013-01-10 01:31:51 +04:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/*** public functions ****************************************************************************/
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2013-01-10 21:20:58 +04:00
|
|
|
char *
|
|
|
|
execute_get_external_cmd_opts_from_config (const char *command, const vfs_path_t * filename_vpath,
|
2013-01-11 14:31:31 +04:00
|
|
|
long start_line)
|
2013-01-10 21:20:58 +04:00
|
|
|
{
|
2013-01-11 01:00:00 +04:00
|
|
|
char *str_from_config, *return_str;
|
|
|
|
char *parameter;
|
2013-01-10 21:20:58 +04:00
|
|
|
|
|
|
|
if (filename_vpath == NULL)
|
|
|
|
return g_strdup ("");
|
|
|
|
|
2013-12-04 14:27:15 +04:00
|
|
|
parameter = g_shell_quote (vfs_path_get_last_path_str (filename_vpath));
|
|
|
|
|
|
|
|
if (start_line <= 0)
|
|
|
|
return parameter;
|
|
|
|
|
2013-01-11 01:00:00 +04:00
|
|
|
str_from_config = execute_get_opts_from_cfg (command, "%filename");
|
|
|
|
|
|
|
|
return_str = str_replace_all (str_from_config, "%filename", parameter);
|
|
|
|
g_free (parameter);
|
|
|
|
g_free (str_from_config);
|
|
|
|
str_from_config = return_str;
|
|
|
|
|
2013-01-11 14:31:31 +04:00
|
|
|
parameter = g_strdup_printf ("%ld", start_line);
|
2013-01-11 01:00:00 +04:00
|
|
|
return_str = str_replace_all (str_from_config, "%lineno", parameter);
|
|
|
|
g_free (parameter);
|
|
|
|
g_free (str_from_config);
|
|
|
|
|
|
|
|
return return_str;
|
2013-01-10 21:20:58 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2013-01-10 01:31:51 +04:00
|
|
|
void
|
2013-01-10 21:20:58 +04:00
|
|
|
do_executev (const char *shell, int flags, char *const argv[])
|
2003-06-22 13:17:46 +04:00
|
|
|
{
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2011-11-26 16:46:54 +04:00
|
|
|
vfs_path_t *new_dir_vpath = NULL;
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif /* ENABLE_SUBSHELL */
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2011-11-26 16:46:54 +04:00
|
|
|
vfs_path_t *old_vfs_dir_vpath = NULL;
|
2003-06-22 13:17:46 +04:00
|
|
|
|
|
|
|
if (!vfs_current_is_local ())
|
2011-11-26 16:46:54 +04:00
|
|
|
old_vfs_dir_vpath = vfs_path_clone (vfs_get_raw_current_dir ());
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2011-02-10 18:02:54 +03:00
|
|
|
if (mc_global.mc_run_mode == MC_RUN_FULL)
|
2010-05-11 16:16:58 +04:00
|
|
|
save_cwds_stat ();
|
2003-06-22 13:17:46 +04:00
|
|
|
pre_exec ();
|
2011-09-14 02:07:31 +04:00
|
|
|
if (mc_global.tty.console_flag != '\0')
|
2010-05-11 16:16:58 +04:00
|
|
|
handle_console (CONSOLE_RESTORE);
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2013-01-10 21:20:58 +04:00
|
|
|
if (!mc_global.tty.use_subshell && *argv != NULL && (flags & EXECUTE_INTERNAL) == 0)
|
2010-05-11 16:16:58 +04:00
|
|
|
{
|
2013-01-10 21:20:58 +04:00
|
|
|
printf ("%s%s\n", mc_prompt, *argv);
|
2010-05-11 16:16:58 +04:00
|
|
|
fflush (stdout);
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2013-01-10 21:20:58 +04:00
|
|
|
if (mc_global.tty.use_subshell && (flags & EXECUTE_INTERNAL) == 0)
|
2010-05-11 16:16:58 +04:00
|
|
|
{
|
|
|
|
do_update_prompt ();
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2010-05-11 16:16:58 +04:00
|
|
|
/* We don't care if it died, higher level takes care of this */
|
2013-01-10 21:20:58 +04:00
|
|
|
invoke_subshell (*argv, VISIBLY, old_vfs_dir_vpath != NULL ? NULL : &new_dir_vpath);
|
2010-05-11 16:16:58 +04:00
|
|
|
}
|
|
|
|
else
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif /* ENABLE_SUBSHELL */
|
2013-01-10 21:20:58 +04:00
|
|
|
my_systemv_flags (flags, shell, argv);
|
2010-05-11 16:16:58 +04:00
|
|
|
|
2013-01-10 21:20:58 +04:00
|
|
|
if ((flags & EXECUTE_INTERNAL) == 0)
|
2010-05-11 16:16:58 +04:00
|
|
|
{
|
|
|
|
if ((pause_after_run == pause_always
|
2011-09-06 14:35:42 +04:00
|
|
|
|| (pause_after_run == pause_on_dumb_terminals && !mc_global.tty.xterm_flag
|
2011-09-14 02:07:31 +04:00
|
|
|
&& mc_global.tty.console_flag == '\0')) && quit == 0
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2010-05-11 16:16:58 +04:00
|
|
|
&& subshell_state != RUNNING_COMMAND
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif /* ENABLE_SUBSHELL */
|
2010-05-11 16:16:58 +04:00
|
|
|
)
|
|
|
|
{
|
2017-05-25 21:32:46 +03:00
|
|
|
printf ("%s", _("Press any key to continue..."));
|
2010-05-11 16:16:58 +04:00
|
|
|
fflush (stdout);
|
|
|
|
tty_raw_mode ();
|
|
|
|
get_key_code (0);
|
|
|
|
printf ("\r\n");
|
|
|
|
fflush (stdout);
|
|
|
|
}
|
2013-01-10 21:20:58 +04:00
|
|
|
if (mc_global.tty.console_flag != '\0' && output_lines != 0 && mc_global.keybar_visible)
|
2010-05-11 16:16:58 +04:00
|
|
|
{
|
2013-01-10 21:20:58 +04:00
|
|
|
putchar ('\n');
|
|
|
|
fflush (stdout);
|
2010-05-11 16:16:58 +04:00
|
|
|
}
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
|
|
|
|
2011-09-14 02:07:31 +04:00
|
|
|
if (mc_global.tty.console_flag != '\0')
|
2010-05-11 16:16:58 +04:00
|
|
|
handle_console (CONSOLE_SAVE);
|
2003-06-22 13:17:46 +04:00
|
|
|
edition_post_exec ();
|
|
|
|
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2011-11-26 16:46:54 +04:00
|
|
|
if (new_dir_vpath != NULL)
|
|
|
|
{
|
|
|
|
do_possible_cd (new_dir_vpath);
|
2021-02-21 19:30:18 +03:00
|
|
|
vfs_path_free (new_dir_vpath, TRUE);
|
2011-11-26 16:46:54 +04:00
|
|
|
}
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif /* ENABLE_SUBSHELL */
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2011-11-26 16:46:54 +04:00
|
|
|
if (old_vfs_dir_vpath != NULL)
|
2010-05-11 16:16:58 +04:00
|
|
|
{
|
2011-11-26 16:46:54 +04:00
|
|
|
mc_chdir (old_vfs_dir_vpath);
|
2021-02-21 19:30:18 +03:00
|
|
|
vfs_path_free (old_vfs_dir_vpath, TRUE);
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
|
|
|
|
2011-02-10 18:02:54 +03:00
|
|
|
if (mc_global.mc_run_mode == MC_RUN_FULL)
|
2010-05-11 16:16:58 +04:00
|
|
|
{
|
|
|
|
update_panels (UP_OPTIMIZE, UP_KEEPSEL);
|
|
|
|
update_xterm_title_path ();
|
2024-02-04 19:40:38 +03:00
|
|
|
update_terminal_cwd ();
|
2010-05-07 20:54:15 +04:00
|
|
|
}
|
2003-06-22 13:17:46 +04:00
|
|
|
|
|
|
|
do_refresh ();
|
|
|
|
use_dash (TRUE);
|
|
|
|
}
|
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2013-01-10 21:20:58 +04:00
|
|
|
void
|
|
|
|
do_execute (const char *shell, const char *command, int flags)
|
|
|
|
{
|
|
|
|
GPtrArray *args_array;
|
|
|
|
|
|
|
|
args_array = g_ptr_array_new ();
|
|
|
|
g_ptr_array_add (args_array, (char *) command);
|
|
|
|
g_ptr_array_add (args_array, NULL);
|
|
|
|
|
|
|
|
do_executev (shell, flags, (char *const *) args_array->pdata);
|
|
|
|
|
|
|
|
g_ptr_array_free (args_array, TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/** Set up the terminal before executing a program */
|
|
|
|
|
|
|
|
void
|
|
|
|
pre_exec (void)
|
|
|
|
{
|
|
|
|
use_dash (FALSE);
|
|
|
|
edition_pre_exec ();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/** Hide the terminal after executing a program */
|
|
|
|
void
|
|
|
|
post_exec (void)
|
|
|
|
{
|
|
|
|
edition_post_exec ();
|
|
|
|
use_dash (TRUE);
|
2011-09-19 13:53:54 +04:00
|
|
|
repaint_screen ();
|
2010-11-10 14:09:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2003-06-22 13:17:46 +04:00
|
|
|
/* Executes a command */
|
2010-11-10 14:09:42 +03:00
|
|
|
|
2003-06-22 13:17:46 +04:00
|
|
|
void
|
|
|
|
shell_execute (const char *command, int flags)
|
|
|
|
{
|
2004-11-26 14:14:07 +03:00
|
|
|
char *cmd = NULL;
|
2004-11-18 20:47:55 +03:00
|
|
|
|
2024-03-10 12:01:43 +03:00
|
|
|
if ((flags & EXECUTE_HIDE) != 0)
|
2010-05-11 16:16:58 +04:00
|
|
|
{
|
|
|
|
cmd = g_strconcat (" ", command, (char *) NULL);
|
|
|
|
flags ^= EXECUTE_HIDE;
|
2004-11-26 14:14:07 +03:00
|
|
|
}
|
2004-11-18 20:47:55 +03:00
|
|
|
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2011-02-10 18:02:54 +03:00
|
|
|
if (mc_global.tty.use_subshell)
|
2017-05-25 21:32:46 +03:00
|
|
|
{
|
2010-05-11 16:16:58 +04:00
|
|
|
if (subshell_state == INACTIVE)
|
2024-03-10 12:01:43 +03:00
|
|
|
do_execute (mc_global.shell->path, cmd != NULL ? cmd : command,
|
|
|
|
flags | EXECUTE_AS_SHELL);
|
2010-05-11 16:16:58 +04:00
|
|
|
else
|
2017-05-25 21:32:46 +03:00
|
|
|
message (D_ERROR, MSG_ERROR, "%s", _("The shell is already running a command"));
|
|
|
|
}
|
2003-06-22 13:17:46 +04:00
|
|
|
else
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif /* ENABLE_SUBSHELL */
|
2024-03-10 12:01:43 +03:00
|
|
|
do_execute (mc_global.shell->path, cmd != NULL ? cmd : command, flags | EXECUTE_AS_SHELL);
|
2004-11-18 20:47:55 +03:00
|
|
|
|
2009-02-06 01:27:37 +03:00
|
|
|
g_free (cmd);
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2003-06-22 13:17:46 +04:00
|
|
|
|
|
|
|
void
|
2019-08-10 11:57:31 +03:00
|
|
|
toggle_subshell (void)
|
2003-06-22 13:17:46 +04:00
|
|
|
{
|
2019-08-10 11:57:31 +03:00
|
|
|
static gboolean message_flag = TRUE;
|
|
|
|
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2011-11-26 16:46:54 +04:00
|
|
|
vfs_path_t *new_dir_vpath = NULL;
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif /* ENABLE_SUBSHELL */
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2012-10-31 11:33:50 +04:00
|
|
|
SIG_ATOMIC_VOLATILE_T was_sigwinch = 0;
|
2012-10-30 14:56:02 +04:00
|
|
|
|
2019-08-10 11:57:31 +03:00
|
|
|
if (!(mc_global.tty.xterm_flag || mc_global.tty.console_flag != '\0'
|
|
|
|
|| mc_global.tty.use_subshell || output_starts_shell))
|
|
|
|
{
|
|
|
|
if (message_flag)
|
|
|
|
message (D_ERROR, MSG_ERROR,
|
2020-03-07 12:39:22 +03:00
|
|
|
_("Not an xterm or Linux console;\nthe subshell cannot be toggled."));
|
2019-08-10 11:57:31 +03:00
|
|
|
message_flag = FALSE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2003-06-22 13:17:46 +04:00
|
|
|
channels_down ();
|
|
|
|
disable_mouse ();
|
2013-10-01 13:31:13 +04:00
|
|
|
disable_bracketed_paste ();
|
2003-06-22 13:17:46 +04:00
|
|
|
if (clear_before_exec)
|
2020-10-31 14:26:10 +03:00
|
|
|
tty_clear_screen ();
|
2011-09-14 02:07:31 +04:00
|
|
|
if (mc_global.tty.alternate_plus_minus)
|
2010-05-11 16:16:58 +04:00
|
|
|
numeric_keypad_mode ();
|
2003-06-22 13:17:46 +04:00
|
|
|
#ifndef HAVE_SLANG
|
|
|
|
/* With slang we don't want any of this, since there
|
2009-06-07 10:48:51 +04:00
|
|
|
* is no raw_mode supported
|
2003-06-22 13:17:46 +04:00
|
|
|
*/
|
2009-05-31 16:23:10 +04:00
|
|
|
tty_reset_shell_mode ();
|
2010-05-11 16:16:58 +04:00
|
|
|
#endif /* !HAVE_SLANG */
|
2009-06-02 21:40:37 +04:00
|
|
|
tty_noecho ();
|
2009-05-31 16:23:10 +04:00
|
|
|
tty_keypad (FALSE);
|
2009-06-02 21:40:37 +04:00
|
|
|
tty_reset_screen ();
|
2016-05-04 13:29:47 +03:00
|
|
|
tty_exit_ca_mode ();
|
2009-05-19 20:12:29 +04:00
|
|
|
tty_raw_mode ();
|
2011-09-14 02:07:31 +04:00
|
|
|
if (mc_global.tty.console_flag != '\0')
|
2010-05-11 16:16:58 +04:00
|
|
|
handle_console (CONSOLE_RESTORE);
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2011-02-10 18:02:54 +03:00
|
|
|
if (mc_global.tty.use_subshell)
|
2010-05-11 16:16:58 +04:00
|
|
|
{
|
2013-10-15 10:34:04 +04:00
|
|
|
vfs_path_t **new_dir_p;
|
|
|
|
|
2012-07-06 14:56:29 +04:00
|
|
|
new_dir_p = vfs_current_is_local ()? &new_dir_vpath : NULL;
|
2010-06-23 13:54:36 +04:00
|
|
|
invoke_subshell (NULL, VISIBLY, new_dir_p);
|
2010-05-11 16:16:58 +04:00
|
|
|
}
|
|
|
|
else
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif /* ENABLE_SUBSHELL */
|
2003-06-22 13:17:46 +04:00
|
|
|
{
|
2010-05-11 16:16:58 +04:00
|
|
|
if (output_starts_shell)
|
|
|
|
{
|
2017-05-25 21:32:46 +03:00
|
|
|
fputs (_("Type 'exit' to return to the Midnight Commander"), stderr);
|
|
|
|
fputs ("\n\r\n\r", stderr);
|
2010-05-11 16:16:58 +04:00
|
|
|
|
2015-11-10 17:23:09 +03:00
|
|
|
my_system (EXECUTE_INTERNAL, mc_global.shell->path, NULL);
|
2010-05-11 16:16:58 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
get_key_code (0);
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
2010-06-23 13:54:36 +04:00
|
|
|
|
2011-09-14 02:07:31 +04:00
|
|
|
if (mc_global.tty.console_flag != '\0')
|
2010-05-11 16:16:58 +04:00
|
|
|
handle_console (CONSOLE_SAVE);
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2016-05-04 13:29:47 +03:00
|
|
|
tty_enter_ca_mode ();
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2009-05-31 16:23:10 +04:00
|
|
|
tty_reset_prog_mode ();
|
|
|
|
tty_keypad (TRUE);
|
2003-06-22 13:17:46 +04:00
|
|
|
|
|
|
|
/* Prevent screen flash when user did 'exit' or 'logout' within
|
|
|
|
subshell */
|
2010-06-23 13:54:36 +04:00
|
|
|
if ((quit & SUBSHELL_EXIT) != 0)
|
|
|
|
{
|
2013-05-26 10:11:06 +04:00
|
|
|
/* User did 'exit' or 'logout': quit MC */
|
2010-06-23 13:54:36 +04:00
|
|
|
if (quiet_quit_cmd ())
|
|
|
|
return;
|
|
|
|
|
|
|
|
quit = 0;
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2010-06-23 13:54:36 +04:00
|
|
|
/* restart subshell */
|
2011-02-10 18:02:54 +03:00
|
|
|
if (mc_global.tty.use_subshell)
|
2010-06-23 13:54:36 +04:00
|
|
|
init_subshell ();
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif /* ENABLE_SUBSHELL */
|
2010-06-23 13:54:36 +04:00
|
|
|
}
|
2003-06-22 13:17:46 +04:00
|
|
|
|
|
|
|
enable_mouse ();
|
2013-10-01 13:31:13 +04:00
|
|
|
enable_bracketed_paste ();
|
2003-06-22 13:17:46 +04:00
|
|
|
channels_up ();
|
2011-09-14 02:07:31 +04:00
|
|
|
if (mc_global.tty.alternate_plus_minus)
|
2010-05-11 16:16:58 +04:00
|
|
|
application_keypad_mode ();
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2012-10-30 14:56:02 +04:00
|
|
|
/* HACK:
|
|
|
|
* Save sigwinch flag that will be reset in mc_refresh() called via update_panels().
|
|
|
|
* There is some problem with screen redraw in ncurses-based mc in this situation.
|
|
|
|
*/
|
2019-08-17 17:55:50 +03:00
|
|
|
was_sigwinch = tty_got_winch ();
|
2019-09-21 20:44:51 +03:00
|
|
|
tty_flush_winch ();
|
2012-10-30 14:56:02 +04:00
|
|
|
|
2012-10-19 19:24:57 +04:00
|
|
|
#ifdef ENABLE_SUBSHELL
|
2011-02-10 18:02:54 +03:00
|
|
|
if (mc_global.tty.use_subshell)
|
2010-05-11 16:16:58 +04:00
|
|
|
{
|
2019-09-02 15:00:58 +03:00
|
|
|
if (mc_global.mc_run_mode == MC_RUN_FULL)
|
|
|
|
{
|
|
|
|
if (new_dir_vpath != NULL)
|
|
|
|
do_possible_cd (new_dir_vpath);
|
|
|
|
}
|
|
|
|
else if (new_dir_vpath != NULL && mc_chdir (new_dir_vpath) != -1)
|
|
|
|
vfs_setup_cwd ();
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
2011-11-26 16:46:54 +04:00
|
|
|
|
2021-02-21 19:30:18 +03:00
|
|
|
vfs_path_free (new_dir_vpath, TRUE);
|
2012-10-19 19:24:57 +04:00
|
|
|
#endif /* ENABLE_SUBSHELL */
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2011-02-10 18:02:54 +03:00
|
|
|
if (mc_global.mc_run_mode == MC_RUN_FULL)
|
2010-05-11 16:16:58 +04:00
|
|
|
{
|
|
|
|
update_panels (UP_OPTIMIZE, UP_KEEPSEL);
|
|
|
|
update_xterm_title_path ();
|
2024-02-04 19:40:38 +03:00
|
|
|
update_terminal_cwd ();
|
2010-03-06 11:08:18 +03:00
|
|
|
}
|
2012-10-30 14:56:02 +04:00
|
|
|
|
2019-08-17 17:55:50 +03:00
|
|
|
if (was_sigwinch != 0 || tty_got_winch ())
|
2012-10-30 14:56:02 +04:00
|
|
|
dialog_change_screen_size ();
|
|
|
|
else
|
|
|
|
repaint_screen ();
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2011-02-18 15:13:44 +03:00
|
|
|
/* event callback */
|
|
|
|
gboolean
|
|
|
|
execute_suspend (const gchar * event_group_name, const gchar * event_name,
|
|
|
|
gpointer init_data, gpointer data)
|
2003-06-22 13:17:46 +04:00
|
|
|
{
|
2011-02-18 15:13:44 +03:00
|
|
|
(void) event_group_name;
|
|
|
|
(void) event_name;
|
|
|
|
(void) init_data;
|
|
|
|
(void) data;
|
|
|
|
|
2011-02-10 18:02:54 +03:00
|
|
|
if (mc_global.mc_run_mode == MC_RUN_FULL)
|
2010-05-11 16:16:58 +04:00
|
|
|
save_cwds_stat ();
|
2003-06-22 13:17:46 +04:00
|
|
|
do_suspend_cmd ();
|
2011-02-10 18:02:54 +03:00
|
|
|
if (mc_global.mc_run_mode == MC_RUN_FULL)
|
2010-05-11 16:16:58 +04:00
|
|
|
update_panels (UP_OPTIMIZE, UP_KEEPSEL);
|
2003-06-22 13:17:46 +04:00
|
|
|
do_refresh ();
|
2011-02-18 15:13:44 +03:00
|
|
|
|
|
|
|
return TRUE;
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2013-01-10 01:48:09 +04:00
|
|
|
|
2010-11-10 14:09:42 +03:00
|
|
|
/**
|
2003-06-22 13:17:46 +04:00
|
|
|
* Execute command on a filename that can be on VFS.
|
|
|
|
* Errors are reported to the user.
|
|
|
|
*/
|
2010-11-10 14:09:42 +03:00
|
|
|
|
2003-06-22 13:17:46 +04:00
|
|
|
void
|
2011-11-29 01:38:27 +04:00
|
|
|
execute_with_vfs_arg (const char *command, const vfs_path_t * filename_vpath)
|
2003-06-22 13:17:46 +04:00
|
|
|
{
|
2013-01-10 01:48:09 +04:00
|
|
|
vfs_path_t *localcopy_vpath = NULL;
|
|
|
|
const vfs_path_t *do_execute_vpath;
|
2003-06-22 13:17:46 +04:00
|
|
|
time_t mtime;
|
|
|
|
|
2013-01-10 01:48:09 +04:00
|
|
|
if (!execute_prepare_with_vfs_arg (filename_vpath, &localcopy_vpath, &mtime))
|
2010-05-11 16:16:58 +04:00
|
|
|
return;
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2013-01-10 01:48:09 +04:00
|
|
|
do_execute_vpath = (localcopy_vpath == NULL) ? filename_vpath : localcopy_vpath;
|
2011-11-29 01:38:27 +04:00
|
|
|
|
2013-01-10 01:48:09 +04:00
|
|
|
do_execute (command, vfs_path_get_last_path_str (do_execute_vpath), EXECUTE_INTERNAL);
|
2003-06-22 13:17:46 +04:00
|
|
|
|
2013-01-10 01:48:09 +04:00
|
|
|
execute_cleanup_with_vfs_arg (filename_vpath, &localcopy_vpath, &mtime);
|
2003-06-22 13:17:46 +04:00
|
|
|
}
|
2010-11-10 14:09:42 +03:00
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2013-01-10 21:20:58 +04:00
|
|
|
/**
|
|
|
|
* Execute external editor or viewer.
|
|
|
|
*
|
|
|
|
* @param command editor/viewer to run
|
|
|
|
* @param filename_vpath path for edit/view
|
|
|
|
* @param start_line cursor will be placed at the 'start_line' position after opening file
|
2013-12-04 14:27:15 +04:00
|
|
|
* if start_line is 0 or negative, no start line will be passed to editor/viewer
|
2013-01-10 21:20:58 +04:00
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
execute_external_editor_or_viewer (const char *command, const vfs_path_t * filename_vpath,
|
2013-01-11 14:31:31 +04:00
|
|
|
long start_line)
|
2013-01-10 21:20:58 +04:00
|
|
|
{
|
|
|
|
vfs_path_t *localcopy_vpath = NULL;
|
|
|
|
const vfs_path_t *do_execute_vpath;
|
|
|
|
char *extern_cmd_options;
|
2016-12-06 13:17:57 +03:00
|
|
|
time_t mtime = 0;
|
2013-01-10 21:20:58 +04:00
|
|
|
|
|
|
|
if (!execute_prepare_with_vfs_arg (filename_vpath, &localcopy_vpath, &mtime))
|
|
|
|
return;
|
|
|
|
|
|
|
|
do_execute_vpath = (localcopy_vpath == NULL) ? filename_vpath : localcopy_vpath;
|
|
|
|
|
|
|
|
extern_cmd_options =
|
|
|
|
execute_get_external_cmd_opts_from_config (command, do_execute_vpath, start_line);
|
|
|
|
|
|
|
|
if (extern_cmd_options != NULL)
|
|
|
|
{
|
|
|
|
char **argv_cmd_options;
|
|
|
|
int argv_count;
|
|
|
|
|
2013-06-13 15:09:29 +04:00
|
|
|
if (g_shell_parse_argv (extern_cmd_options, &argv_count, &argv_cmd_options, NULL))
|
|
|
|
{
|
|
|
|
do_executev (command, EXECUTE_INTERNAL, argv_cmd_options);
|
|
|
|
g_strfreev (argv_cmd_options);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
do_executev (command, EXECUTE_INTERNAL, NULL);
|
2013-01-10 21:20:58 +04:00
|
|
|
|
2013-06-13 15:09:29 +04:00
|
|
|
g_free (extern_cmd_options);
|
2013-01-10 21:20:58 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
execute_cleanup_with_vfs_arg (filename_vpath, &localcopy_vpath, &mtime);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|