]> git.ipfire.org Git - thirdparty/git.git/blob - builtin/bugreport.c
rebase: allow overriding the maximal length of the generated labels
[thirdparty/git.git] / builtin / bugreport.c
1 #include "builtin.h"
2 #include "abspath.h"
3 #include "editor.h"
4 #include "gettext.h"
5 #include "parse-options.h"
6 #include "strbuf.h"
7 #include "help.h"
8 #include "compat/compiler.h"
9 #include "hook.h"
10 #include "hook-list.h"
11 #include "diagnose.h"
12 #include "object-file.h"
13 #include "setup.h"
14 #include "wrapper.h"
15
16 static void get_system_info(struct strbuf *sys_info)
17 {
18 struct utsname uname_info;
19 char *shell = NULL;
20
21 /* get git version from native cmd */
22 strbuf_addstr(sys_info, _("git version:\n"));
23 get_version_info(sys_info, 1);
24
25 /* system call for other version info */
26 strbuf_addstr(sys_info, "uname: ");
27 if (uname(&uname_info))
28 strbuf_addf(sys_info, _("uname() failed with error '%s' (%d)\n"),
29 strerror(errno),
30 errno);
31 else
32 strbuf_addf(sys_info, "%s %s %s %s\n",
33 uname_info.sysname,
34 uname_info.release,
35 uname_info.version,
36 uname_info.machine);
37
38 strbuf_addstr(sys_info, _("compiler info: "));
39 get_compiler_info(sys_info);
40
41 strbuf_addstr(sys_info, _("libc info: "));
42 get_libc_info(sys_info);
43
44 shell = getenv("SHELL");
45 strbuf_addf(sys_info, "$SHELL (typically, interactive shell): %s\n",
46 shell ? shell : "<unset>");
47 }
48
49 static void get_populated_hooks(struct strbuf *hook_info, int nongit)
50 {
51 const char **p;
52
53 if (nongit) {
54 strbuf_addstr(hook_info,
55 _("not run from a git repository - no hooks to show\n"));
56 return;
57 }
58
59 for (p = hook_name_list; *p; p++) {
60 const char *hook = *p;
61
62 if (hook_exists(hook))
63 strbuf_addf(hook_info, "%s\n", hook);
64 }
65 }
66
67 static const char * const bugreport_usage[] = {
68 N_("git bugreport [(-o | --output-directory) <path>] [(-s | --suffix) <format>]\n"
69 " [--diagnose[=<mode>]]"),
70 NULL
71 };
72
73 static int get_bug_template(struct strbuf *template)
74 {
75 const char template_text[] = N_(
76 "Thank you for filling out a Git bug report!\n"
77 "Please answer the following questions to help us understand your issue.\n"
78 "\n"
79 "What did you do before the bug happened? (Steps to reproduce your issue)\n"
80 "\n"
81 "What did you expect to happen? (Expected behavior)\n"
82 "\n"
83 "What happened instead? (Actual behavior)\n"
84 "\n"
85 "What's different between what you expected and what actually happened?\n"
86 "\n"
87 "Anything else you want to add:\n"
88 "\n"
89 "Please review the rest of the bug report below.\n"
90 "You can delete any lines you don't wish to share.\n");
91
92 strbuf_addstr(template, _(template_text));
93 return 0;
94 }
95
96 static void get_header(struct strbuf *buf, const char *title)
97 {
98 strbuf_addf(buf, "\n\n[%s]\n", title);
99 }
100
101 int cmd_bugreport(int argc, const char **argv, const char *prefix)
102 {
103 struct strbuf buffer = STRBUF_INIT;
104 struct strbuf report_path = STRBUF_INIT;
105 int report = -1;
106 time_t now = time(NULL);
107 struct tm tm;
108 enum diagnose_mode diagnose = DIAGNOSE_NONE;
109 char *option_output = NULL;
110 char *option_suffix = "%Y-%m-%d-%H%M";
111 const char *user_relative_path = NULL;
112 char *prefixed_filename;
113 size_t output_path_len;
114 int ret;
115
116 const struct option bugreport_options[] = {
117 OPT_CALLBACK_F(0, "diagnose", &diagnose, N_("mode"),
118 N_("create an additional zip archive of detailed diagnostics (default 'stats')"),
119 PARSE_OPT_OPTARG, option_parse_diagnose),
120 OPT_STRING('o', "output-directory", &option_output, N_("path"),
121 N_("specify a destination for the bugreport file(s)")),
122 OPT_STRING('s', "suffix", &option_suffix, N_("format"),
123 N_("specify a strftime format suffix for the filename(s)")),
124 OPT_END()
125 };
126
127 argc = parse_options(argc, argv, prefix, bugreport_options,
128 bugreport_usage, 0);
129
130 /* Prepare the path to put the result */
131 prefixed_filename = prefix_filename(prefix,
132 option_output ? option_output : "");
133 strbuf_addstr(&report_path, prefixed_filename);
134 strbuf_complete(&report_path, '/');
135 output_path_len = report_path.len;
136
137 strbuf_addstr(&report_path, "git-bugreport-");
138 strbuf_addftime(&report_path, option_suffix, localtime_r(&now, &tm), 0, 0);
139 strbuf_addstr(&report_path, ".txt");
140
141 switch (safe_create_leading_directories(report_path.buf)) {
142 case SCLD_OK:
143 case SCLD_EXISTS:
144 break;
145 default:
146 die(_("could not create leading directories for '%s'"),
147 report_path.buf);
148 }
149
150 /* Prepare diagnostics, if requested */
151 if (diagnose != DIAGNOSE_NONE) {
152 struct strbuf zip_path = STRBUF_INIT;
153 strbuf_add(&zip_path, report_path.buf, output_path_len);
154 strbuf_addstr(&zip_path, "git-diagnostics-");
155 strbuf_addftime(&zip_path, option_suffix, localtime_r(&now, &tm), 0, 0);
156 strbuf_addstr(&zip_path, ".zip");
157
158 if (create_diagnostics_archive(&zip_path, diagnose))
159 die_errno(_("unable to create diagnostics archive %s"), zip_path.buf);
160
161 strbuf_release(&zip_path);
162 }
163
164 /* Prepare the report contents */
165 get_bug_template(&buffer);
166
167 get_header(&buffer, _("System Info"));
168 get_system_info(&buffer);
169
170 get_header(&buffer, _("Enabled Hooks"));
171 get_populated_hooks(&buffer, !startup_info->have_repository);
172
173 /* fopen doesn't offer us an O_EXCL alternative, except with glibc. */
174 report = xopen(report_path.buf, O_CREAT | O_EXCL | O_WRONLY, 0666);
175
176 if (write_in_full(report, buffer.buf, buffer.len) < 0)
177 die_errno(_("unable to write to %s"), report_path.buf);
178
179 close(report);
180
181 /*
182 * We want to print the path relative to the user, but we still need the
183 * path relative to us to give to the editor.
184 */
185 if (!(prefix && skip_prefix(report_path.buf, prefix, &user_relative_path)))
186 user_relative_path = report_path.buf;
187 fprintf(stderr, _("Created new report at '%s'.\n"),
188 user_relative_path);
189
190 free(prefixed_filename);
191 strbuf_release(&buffer);
192
193 ret = !!launch_editor(report_path.buf, NULL, NULL);
194 strbuf_release(&report_path);
195 return ret;
196 }