]> git.ipfire.org Git - thirdparty/systemd.git/blob - src/shared/edit-util.c
2887e7fa292d596b4bf6bf528565e7ac6e818c09
[thirdparty/systemd.git] / src / shared / edit-util.c
1 /* SPDX-License-Identifier: LGPL-2.1-or-later */
2
3 #include <stdio.h>
4 #include <stdlib.h>
5 #include <sys/stat.h>
6 #include <unistd.h>
7
8 #include "alloc-util.h"
9 #include "copy.h"
10 #include "edit-util.h"
11 #include "errno-util.h"
12 #include "fd-util.h"
13 #include "fileio.h"
14 #include "fs-util.h"
15 #include "log.h"
16 #include "mkdir-label.h"
17 #include "path-util.h"
18 #include "process-util.h"
19 #include "string-util.h"
20 #include "strv.h"
21 #include "tmpfile-util-label.h"
22
23 typedef struct EditFile {
24 EditFileContext *context;
25 char *path;
26 char *original_path;
27 char **comment_paths;
28 char *temp;
29 unsigned line;
30 } EditFile;
31
32 void edit_file_context_done(EditFileContext *context) {
33 int r;
34
35 assert(context);
36
37 FOREACH_ARRAY(i, context->files, context->n_files) {
38 unlink_and_free(i->temp);
39
40 if (context->remove_parent) {
41 _cleanup_free_ char *parent = NULL;
42
43 r = path_extract_directory(i->path, &parent);
44 if (r < 0)
45 log_debug_errno(r, "Failed to extract directory from '%s', ignoring: %m", i->path);
46 else if (rmdir(parent) < 0 && !IN_SET(errno, ENOENT, ENOTEMPTY))
47 log_debug_errno(errno, "Failed to remove parent directory '%s', ignoring: %m", parent);
48 }
49
50 free(i->path);
51 free(i->original_path);
52 strv_free(i->comment_paths);
53 }
54
55 context->files = mfree(context->files);
56 context->n_files = 0;
57 }
58
59 bool edit_files_contains(const EditFileContext *context, const char *path) {
60 assert(context);
61 assert(path);
62
63 FOREACH_ARRAY(i, context->files, context->n_files)
64 if (path_equal(i->path, path))
65 return true;
66
67 return false;
68 }
69
70 int edit_files_add(
71 EditFileContext *context,
72 const char *path,
73 const char *original_path,
74 char * const *comment_paths) {
75
76 _cleanup_free_ char *new_path = NULL, *new_original_path = NULL;
77 _cleanup_strv_free_ char **new_comment_paths = NULL;
78
79 assert(context);
80 assert(path);
81
82 if (edit_files_contains(context, path))
83 return 0;
84
85 if (!GREEDY_REALLOC(context->files, context->n_files + 1))
86 return log_oom();
87
88 new_path = strdup(path);
89 if (!new_path)
90 return log_oom();
91
92 if (original_path) {
93 new_original_path = strdup(original_path);
94 if (!new_original_path)
95 return log_oom();
96 }
97
98 if (comment_paths) {
99 new_comment_paths = strv_copy(comment_paths);
100 if (!new_comment_paths)
101 return log_oom();
102 }
103
104 context->files[context->n_files] = (EditFile) {
105 .context = context,
106 .path = TAKE_PTR(new_path),
107 .original_path = TAKE_PTR(new_original_path),
108 .comment_paths = TAKE_PTR(new_comment_paths),
109 .line = 1,
110 };
111 context->n_files++;
112
113 return 1;
114 }
115
116 static int populate_edit_temp_file(EditFile *e, FILE *f, const char *filename) {
117 assert(e);
118 assert(e->context);
119 assert(!e->context->read_from_stdin);
120 assert(e->path);
121 assert(f);
122 assert(filename);
123
124 bool has_original = e->original_path && access(e->original_path, F_OK) >= 0;
125 bool has_target = access(e->path, F_OK) >= 0;
126 const char *source;
127 int r;
128
129 if (has_original && (!has_target || e->context->overwrite_with_origin))
130 /* We are asked to overwrite target with original_path or target doesn't exist. */
131 source = e->original_path;
132 else if (has_target)
133 /* Target exists and shouldn't be overwritten. */
134 source = e->path;
135 else
136 source = NULL;
137
138 if (e->comment_paths) {
139 _cleanup_free_ char *source_contents = NULL;
140
141 if (source) {
142 r = read_full_file(source, &source_contents, NULL);
143 if (r < 0)
144 return log_error_errno(r, "Failed to read source file '%s': %m", source);
145 }
146
147 fprintf(f,
148 "### Editing %s\n"
149 "%s\n"
150 "\n"
151 "%s%s"
152 "\n"
153 "%s\n",
154 e->path,
155 e->context->marker_start,
156 strempty(source_contents),
157 source_contents && endswith(source_contents, "\n") ? "" : "\n",
158 e->context->marker_end);
159
160 e->line = 4; /* Start editing at the contents area */
161
162 STRV_FOREACH(path, e->comment_paths) {
163 _cleanup_free_ char *comment = NULL;
164
165 /* Skip the file which is being edited and the source file (can be the same) */
166 if (PATH_IN_SET(*path, e->path, source))
167 continue;
168
169 r = read_full_file(*path, &comment, NULL);
170 if (r < 0)
171 return log_error_errno(r, "Failed to read comment file '%s': %m", *path);
172
173 fprintf(f, "\n\n### %s", *path);
174
175 if (!isempty(comment)) {
176 _cleanup_free_ char *c = NULL;
177
178 c = strreplace(strstrip(comment), "\n", "\n# ");
179 if (!c)
180 return log_oom();
181
182 fprintf(f, "\n# %s", c);
183 }
184 }
185 } else if (source) {
186 r = copy_file_fd(source, fileno(f), COPY_REFLINK);
187 if (r < 0) {
188 assert(r != -ENOENT);
189 return log_error_errno(r, "Failed to copy file '%s' to temporary file '%s': %m",
190 source, filename);
191 }
192 }
193
194 return 0;
195 }
196
197 static int create_edit_temp_file(EditFile *e, const char *contents, size_t contents_size) {
198 _cleanup_(unlink_and_freep) char *temp = NULL;
199 _cleanup_fclose_ FILE *f = NULL;
200 int r;
201
202 assert(e);
203 assert(e->context);
204 assert(e->path);
205 assert(!e->comment_paths || (e->context->marker_start && e->context->marker_end));
206 assert(contents || contents_size == 0);
207 assert(e->context->read_from_stdin == !!contents);
208
209 if (e->temp)
210 return 0;
211
212 r = mkdir_parents_label(e->path, 0755);
213 if (r < 0)
214 return log_error_errno(r, "Failed to create parent directories for '%s': %m", e->path);
215
216 r = fopen_temporary_label(e->path, e->path, &f, &temp);
217 if (r < 0)
218 return log_error_errno(r, "Failed to create temporary file for '%s': %m", e->path);
219
220 if (fchmod(fileno(f), 0644) < 0)
221 return log_error_errno(errno, "Failed to change mode of temporary file '%s': %m", temp);
222
223 if (e->context->read_from_stdin) {
224 if (fwrite(contents, 1, contents_size, f) != contents_size)
225 return log_error_errno(SYNTHETIC_ERRNO(EIO),
226 "Failed to write stdin data to temporary file '%s'.", temp);
227 } else {
228 r = populate_edit_temp_file(e, f, temp);
229 if (r < 0)
230 return r;
231 }
232
233 r = fflush_and_check(f);
234 if (r < 0)
235 return log_error_errno(r, "Failed to write to temporary file '%s': %m", temp);
236
237 e->temp = TAKE_PTR(temp);
238
239 return 0;
240 }
241
242 static int run_editor_child(const EditFileContext *context) {
243 _cleanup_strv_free_ char **args = NULL, **editor = NULL;
244 int r;
245
246 assert(context);
247 assert(context->n_files >= 1);
248
249 /* SYSTEMD_EDITOR takes precedence over EDITOR which takes precedence over VISUAL.
250 * If neither SYSTEMD_EDITOR nor EDITOR nor VISUAL are present, we try to execute
251 * well known editors. */
252 FOREACH_STRING(e, "SYSTEMD_EDITOR", "EDITOR", "VISUAL") {
253 const char *m = empty_to_null(getenv(e));
254 if (m) {
255 editor = strv_split(m, WHITESPACE);
256 if (!editor)
257 return log_oom();
258
259 break;
260 }
261 }
262
263 if (context->n_files == 1 && context->files[0].line > 1) {
264 /* If editing a single file only, use the +LINE syntax to put cursor on the right line */
265 r = strv_extendf(&args, "+%u", context->files[0].line);
266 if (r < 0)
267 return log_oom();
268 }
269
270 FOREACH_ARRAY(i, context->files, context->n_files) {
271 r = strv_extend(&args, i->temp);
272 if (r < 0)
273 return log_oom();
274 }
275
276 size_t editor_n = strv_length(editor);
277 if (editor_n > 0) {
278 /* Strings are owned by 'editor' and 'args' */
279 _cleanup_free_ char **cmdline = new(char*, editor_n + strv_length(args) + 1);
280 if (!cmdline)
281 return log_oom();
282
283 *mempcpy_typesafe(mempcpy_typesafe(cmdline, editor, editor_n), args, strv_length(args)) = NULL;
284
285 execvp(cmdline[0], cmdline);
286 log_warning_errno(errno, "Specified editor '%s' not available, trying fallbacks: %m", editor[0]);
287 }
288
289 bool prepended = false;
290 FOREACH_STRING(name, "editor", "nano", "vim", "vi") {
291 if (!prepended) {
292 r = strv_prepend(&args, name);
293 prepended = true;
294 } else
295 r = free_and_strdup(&args[0], name);
296 if (r < 0)
297 return log_oom();
298
299 execvp(args[0], (char* const*) args);
300
301 /* We do not fail if the editor doesn't exist because we want to try each one of them
302 * before failing. */
303 if (errno != ENOENT)
304 return log_error_errno(errno, "Failed to execute '%s': %m", name);
305 }
306
307 return log_error_errno(SYNTHETIC_ERRNO(ENOENT),
308 "Cannot edit files, no editor available. Please set either $SYSTEMD_EDITOR, $EDITOR or $VISUAL.");
309 }
310
311 static int run_editor(const EditFileContext *context) {
312 int r;
313
314 assert(context);
315
316 r = safe_fork("(editor)", FORK_RESET_SIGNALS|FORK_DEATHSIG_SIGTERM|FORK_RLIMIT_NOFILE_SAFE|FORK_CLOSE_ALL_FDS|FORK_REOPEN_LOG|FORK_LOG|FORK_WAIT, NULL);
317 if (r < 0)
318 return r;
319 if (r == 0) { /* Child */
320 r = run_editor_child(context);
321 _exit(r < 0 ? EXIT_FAILURE : EXIT_SUCCESS);
322 }
323
324 return 0;
325 }
326
327 static int strip_edit_temp_file(EditFile *e) {
328 _cleanup_free_ char *old_contents = NULL, *tmp = NULL, *new_contents = NULL;
329 const char *stripped;
330 bool with_marker;
331 int r;
332
333 assert(e);
334 assert(e->context);
335 assert(!e->context->marker_start == !e->context->marker_end);
336 assert(e->temp);
337
338 r = read_full_file(e->temp, &old_contents, NULL);
339 if (r < 0)
340 return log_error_errno(r, "Failed to read temporary file '%s': %m", e->temp);
341
342 tmp = strdup(old_contents);
343 if (!tmp)
344 return log_oom();
345
346 with_marker = e->context->marker_start && !e->context->read_from_stdin;
347
348 if (with_marker) {
349 /* Trim out the lines between the two markers */
350 char *contents_start, *contents_end;
351
352 contents_start = strstrafter(tmp, e->context->marker_start) ?: tmp;
353
354 contents_end = strstr(contents_start, e->context->marker_end);
355 if (contents_end)
356 *contents_end = '\0';
357
358 stripped = strstrip(contents_start);
359 } else
360 stripped = strstrip(tmp);
361
362 if (isempty(stripped)) {
363 /* People keep coming back to #24208 due to edits outside of markers. Let's detect this
364 * and point them in the right direction. */
365 if (with_marker)
366 for (const char *p = old_contents;;) {
367 p = skip_leading_chars(p, WHITESPACE);
368 if (*p == '\0')
369 break;
370 if (*p != '#') {
371 log_warning("Found modifications outside of the staging area, which would be discarded.");
372 break;
373 }
374
375 /* Skip the whole line if commented out */
376 p = strchr(p, '\n');
377 if (!p)
378 break;
379 p++;
380 }
381
382 return 0; /* File is empty (has no real changes) */
383 }
384
385 /* Trim prefix and suffix, but ensure suffixed by single newline */
386 new_contents = strjoin(stripped, "\n");
387 if (!new_contents)
388 return log_oom();
389
390 if (streq(old_contents, new_contents)) /* Don't touch the file if the above didn't change a thing */
391 return 1; /* Contents have real changes */
392
393 r = write_string_file(e->temp, new_contents,
394 WRITE_STRING_FILE_TRUNCATE | WRITE_STRING_FILE_AVOID_NEWLINE);
395 if (r < 0)
396 return log_error_errno(r, "Failed to strip temporary file '%s': %m", e->temp);
397
398 return 1; /* Contents have real changes */
399 }
400
401 static int edit_file_install_one(EditFile *e) {
402 int r;
403
404 assert(e);
405 assert(e->path);
406 assert(e->temp);
407
408 r = strip_edit_temp_file(e);
409 if (r <= 0)
410 return r;
411
412 r = RET_NERRNO(rename(e->temp, e->path));
413 if (r < 0)
414 return log_error_errno(r,
415 "Failed to rename temporary file '%s' to target file '%s': %m",
416 e->temp, e->path);
417 e->temp = mfree(e->temp);
418
419 return 1;
420 }
421
422 static int edit_file_install_one_stdin(EditFile *e, const char *contents, size_t contents_size, int *fd) {
423 int r;
424
425 assert(e);
426 assert(e->path);
427 assert(contents || contents_size == 0);
428 assert(fd);
429
430 if (contents_size == 0)
431 return 0;
432
433 if (*fd >= 0) {
434 r = mkdir_parents_label(e->path, 0755);
435 if (r < 0)
436 return log_error_errno(r, "Failed to create parent directories for '%s': %m", e->path);
437
438 r = copy_file_atomic_at(*fd, NULL, AT_FDCWD, e->path, 0644, COPY_REFLINK|COPY_REPLACE|COPY_MAC_CREATE);
439 if (r < 0)
440 return log_error_errno(r, "Failed to copy stdin contents to '%s': %m", e->path);
441
442 return 1;
443 }
444
445 r = create_edit_temp_file(e, contents, contents_size);
446 if (r < 0)
447 return r;
448
449 _cleanup_close_ int tfd = open(e->temp, O_PATH|O_CLOEXEC);
450 if (tfd < 0)
451 return log_error_errno(errno, "Failed to pin temporary file '%s': %m", e->temp);
452
453 r = edit_file_install_one(e);
454 if (r <= 0)
455 return r;
456
457 *fd = TAKE_FD(tfd);
458
459 return 1;
460 }
461
462 int do_edit_files_and_install(EditFileContext *context) {
463 _cleanup_free_ char *stdin_data = NULL;
464 size_t stdin_size = 0;
465 int r;
466
467 assert(context);
468
469 if (context->n_files == 0)
470 return log_debug_errno(SYNTHETIC_ERRNO(ENOENT), "Got no files to edit.");
471
472 if (context->read_from_stdin) {
473 r = read_full_stream(stdin, &stdin_data, &stdin_size);
474 if (r < 0)
475 return log_error_errno(r, "Failed to read stdin: %m");
476 } else {
477 FOREACH_ARRAY(editfile, context->files, context->n_files) {
478 r = create_edit_temp_file(editfile, /* contents = */ NULL, /* contents_size = */ 0);
479 if (r < 0)
480 return r;
481 }
482
483 r = run_editor(context);
484 if (r < 0)
485 return r;
486 }
487
488 _cleanup_close_ int stdin_data_fd = -EBADF;
489
490 FOREACH_ARRAY(editfile, context->files, context->n_files) {
491 if (context->read_from_stdin) {
492 r = edit_file_install_one_stdin(editfile, stdin_data, stdin_size, &stdin_data_fd);
493 if (r == 0) {
494 log_notice("Stripped stdin content is empty, not writing file.");
495 return 0;
496 }
497 } else {
498 r = edit_file_install_one(editfile);
499 if (r == 0) {
500 log_notice("%s: after editing, new contents are empty, not writing file.",
501 editfile->path);
502 continue;
503 }
504 }
505 if (r < 0)
506 return r;
507
508 log_info("Successfully installed edited file '%s'.", editfile->path);
509 }
510
511 return 0;
512 }