1 #include "git-compat-util.h"
6 #include "environment.h"
12 #include "run-command.h"
16 #ifndef DEFAULT_EDITOR
17 #define DEFAULT_EDITOR "vi"
20 int is_terminal_dumb(void)
22 const char *terminal
= getenv("TERM");
23 return !terminal
|| !strcmp(terminal
, "dumb");
26 const char *git_editor(void)
28 const char *editor
= getenv("GIT_EDITOR");
29 int terminal_is_dumb
= is_terminal_dumb();
31 if (!editor
&& editor_program
)
32 editor
= editor_program
;
33 if (!editor
&& !terminal_is_dumb
)
34 editor
= getenv("VISUAL");
36 editor
= getenv("EDITOR");
38 if (!editor
&& terminal_is_dumb
)
42 editor
= DEFAULT_EDITOR
;
47 const char *git_sequence_editor(void)
49 const char *editor
= getenv("GIT_SEQUENCE_EDITOR");
52 git_config_get_string_tmp("sequence.editor", &editor
);
54 editor
= git_editor();
59 static int launch_specified_editor(const char *editor
, const char *path
,
60 struct strbuf
*buffer
, const char *const *env
)
63 return error("Terminal is dumb, but EDITOR unset");
65 if (strcmp(editor
, ":")) {
66 struct strbuf realpath
= STRBUF_INIT
;
67 struct child_process p
= CHILD_PROCESS_INIT
;
69 int print_waiting_for_editor
= advice_enabled(ADVICE_WAITING_FOR_EDITOR
) && isatty(2);
71 if (print_waiting_for_editor
) {
73 * A dumb terminal cannot erase the line later on. Add a
74 * newline to separate the hint from subsequent output.
76 * Make sure that our message is separated with a whitespace
77 * from further cruft that may be written by the editor.
79 const char term
= is_terminal_dumb() ? '\n' : ' ';
82 _("hint: Waiting for your editor to close the file...%c"),
87 strbuf_realpath(&realpath
, path
, 1);
89 strvec_pushl(&p
.args
, editor
, realpath
.buf
, NULL
);
91 strvec_pushv(&p
.env
, (const char **)env
);
93 p
.trace2_child_class
= "editor";
94 if (start_command(&p
) < 0) {
95 strbuf_release(&realpath
);
96 return error("unable to start editor '%s'", editor
);
99 sigchain_push(SIGINT
, SIG_IGN
);
100 sigchain_push(SIGQUIT
, SIG_IGN
);
101 ret
= finish_command(&p
);
102 strbuf_release(&realpath
);
104 sigchain_pop(SIGINT
);
105 sigchain_pop(SIGQUIT
);
106 if (sig
== SIGINT
|| sig
== SIGQUIT
)
109 return error("There was a problem with the editor '%s'.",
112 if (print_waiting_for_editor
&& !is_terminal_dumb())
114 * Erase the entire line to avoid wasting the
122 if (strbuf_read_file(buffer
, path
, 0) < 0)
123 return error_errno("could not read file '%s'", path
);
127 int launch_editor(const char *path
, struct strbuf
*buffer
, const char *const *env
)
129 return launch_specified_editor(git_editor(), path
, buffer
, env
);
132 int launch_sequence_editor(const char *path
, struct strbuf
*buffer
,
133 const char *const *env
)
135 return launch_specified_editor(git_sequence_editor(), path
, buffer
, env
);
138 int strbuf_edit_interactively(struct strbuf
*buffer
, const char *path
,
139 const char *const *env
)
144 if (!is_absolute_path(path
))
145 path
= path2
= xstrdup(git_path("%s", path
));
147 fd
= open(path
, O_WRONLY
| O_CREAT
| O_TRUNC
, 0666);
149 res
= error_errno(_("could not open '%s' for writing"), path
);
150 else if (write_in_full(fd
, buffer
->buf
, buffer
->len
) < 0) {
151 res
= error_errno(_("could not write to '%s'"), path
);
153 } else if (close(fd
) < 0)
154 res
= error_errno(_("could not close '%s'"), path
);
156 strbuf_reset(buffer
);
157 if (launch_editor(path
, buffer
, env
) < 0)
158 res
= error_errno(_("could not edit '%s'"), path
);