1 /* SPDX-License-Identifier: LGPL-2.1-or-later */
7 #include "xdg-autostart-service.h"
9 #include "conf-parser.h"
11 #include "unit-name.h"
12 #include "path-util.h"
14 #include "generator.h"
16 #include "specifier.h"
17 #include "string-util.h"
18 #include "nulstr-util.h"
21 XdgAutostartService
* xdg_autostart_service_free(XdgAutostartService
*s
) {
31 free(s
->working_directory
);
33 strv_free(s
->only_show_in
);
34 strv_free(s
->not_show_in
);
37 free(s
->autostart_condition
);
38 free(s
->kde_autostart_condition
);
40 free(s
->gnome_autostart_phase
);
45 char *xdg_autostart_service_translate_name(const char *name
) {
46 _cleanup_free_
char *c
= NULL
, *escaped
= NULL
;
53 res
= endswith(c
, ".desktop");
57 escaped
= unit_name_escape(c
);
61 return strjoin("app-", escaped
, "-autostart.service");
64 static int xdg_config_parse_bool(
69 unsigned section_line
,
83 if (streq(rvalue
, "true"))
85 else if (streq(rvalue
, "false"))
88 return log_syntax(unit
, LOG_ERR
, filename
, line
, SYNTHETIC_ERRNO(EINVAL
), "Invalid value for boolean: %s", rvalue
);
93 /* Unescapes the string in-place, returns non-zero status on error. */
94 static int xdg_unescape_string(
107 for (; *in
; in
++, out
++) {
109 /* Move forward, and ensure it is a valid escape. */
129 /* Technically only permitted for strv. */
133 return log_syntax(unit
, LOG_ERR
, filename
, line
, SYNTHETIC_ERRNO(EINVAL
), "Undefined escape sequence \\%c.", *in
);
146 /* Note: We do not bother with unescaping the strings, hence the _raw postfix. */
147 static int xdg_config_parse_string(
149 const char *filename
,
152 unsigned section_line
,
159 _cleanup_free_
char *res
= NULL
;
168 /* XDG does not allow duplicate definitions. */
170 log_syntax(unit
, LOG_WARNING
, filename
, line
, 0, "Key %s was defined multiple times, ignoring.", lvalue
);
174 res
= strdup(rvalue
);
178 r
= xdg_unescape_string(unit
, filename
, line
, res
);
182 *out
= TAKE_PTR(res
);
186 static int strv_strndup_unescape_and_push(
188 const char *filename
,
199 _cleanup_free_
char *copy
= NULL
;
202 copy
= strndup(start
, end
- start
);
206 r
= xdg_unescape_string(unit
, filename
, line
, copy
);
210 if (!greedy_realloc((void**) sv
, n_allocated
, *n
+ 2, sizeof(char*))) /* One extra for NULL */
213 (*sv
)[*n
] = TAKE_PTR(copy
);
214 (*sv
)[*n
+ 1] = NULL
;
220 static int xdg_config_parse_strv(
222 const char *filename
,
225 unsigned section_line
,
232 char ***ret_sv
= data
;
240 /* XDG does not allow duplicate definitions. */
242 log_syntax(unit
, LOG_WARNING
, filename
, line
, 0, "Key %s was already defined, ignoring.", lvalue
);
246 size_t n
= 0, n_allocated
= 0;
247 _cleanup_strv_free_
char **sv
= NULL
;
249 if (!GREEDY_REALLOC0(sv
, n_allocated
, 1))
252 /* We cannot use strv_split because it does not handle escaping correctly. */
253 const char *start
= rvalue
, *end
;
255 for (end
= start
; *end
; end
++) {
257 /* Move forward, and ensure it is a valid escape. */
259 if (!strchr("sntr\\;", *end
)) {
260 log_syntax(unit
, LOG_WARNING
, filename
, line
, 0, "Undefined escape sequence \\%c.", *end
);
267 r
= strv_strndup_unescape_and_push(unit
, filename
, line
,
268 &sv
, &n_allocated
, &n
,
277 /* Handle the trailing entry after the last separator */
278 r
= strv_strndup_unescape_and_push(unit
, filename
, line
,
279 &sv
, &n_allocated
, &n
,
284 *ret_sv
= TAKE_PTR(sv
);
288 static int xdg_config_item_table_lookup(
292 ConfigParserCallback
*func
,
299 /* Ignore any keys with [] as those are translations. */
300 if (strchr(lvalue
, '[')) {
307 return config_item_table_lookup(table
, section
, lvalue
, func
, ltype
, data
, userdata
);
310 XdgAutostartService
*xdg_autostart_service_parse_desktop(const char *path
) {
311 _cleanup_(xdg_autostart_service_freep
) XdgAutostartService
*service
= NULL
;
314 service
= new0(XdgAutostartService
, 1);
318 service
->path
= strdup(path
);
322 const ConfigTableItem items
[] = {
323 { "Desktop Entry", "Name", xdg_config_parse_string
, 0, &service
->description
},
324 { "Desktop Entry", "Exec", xdg_config_parse_string
, 0, &service
->exec_string
},
325 { "Desktop Entry", "Path", xdg_config_parse_string
, 0, &service
->working_directory
},
326 { "Desktop Entry", "TryExec", xdg_config_parse_string
, 0, &service
->try_exec
},
327 { "Desktop Entry", "Type", xdg_config_parse_string
, 0, &service
->type
},
328 { "Desktop Entry", "OnlyShowIn", xdg_config_parse_strv
, 0, &service
->only_show_in
},
329 { "Desktop Entry", "NotShowIn", xdg_config_parse_strv
, 0, &service
->not_show_in
},
330 { "Desktop Entry", "Hidden", xdg_config_parse_bool
, 0, &service
->hidden
},
331 { "Desktop Entry", "AutostartCondition", xdg_config_parse_string
, 0, &service
->autostart_condition
},
332 { "Desktop Entry", "X-KDE-autostart-condition", xdg_config_parse_string
, 0, &service
->kde_autostart_condition
},
333 { "Desktop Entry", "X-GNOME-Autostart-Phase", xdg_config_parse_string
, 0, &service
->gnome_autostart_phase
},
334 { "Desktop Entry", "X-systemd-skip", xdg_config_parse_bool
, 0, &service
->systemd_skip
},
336 /* Common entries that we do not use currently. */
337 { "Desktop Entry", "Categories", NULL
, 0, NULL
},
338 { "Desktop Entry", "Comment", NULL
, 0, NULL
},
339 { "Desktop Entry", "Encoding", NULL
, 0, NULL
},
340 { "Desktop Entry", "GenericName", NULL
, 0, NULL
},
341 { "Desktop Entry", "Icon", NULL
, 0, NULL
},
342 { "Desktop Entry", "Keywords", NULL
, 0, NULL
},
343 { "Desktop Entry", "MimeType", NULL
, 0, NULL
},
344 { "Desktop Entry", "NoDisplay", NULL
, 0, NULL
},
345 { "Desktop Entry", "StartupNotify", NULL
, 0, NULL
},
346 { "Desktop Entry", "StartupWMClass", NULL
, 0, NULL
},
347 { "Desktop Entry", "Terminal", NULL
, 0, NULL
},
348 { "Desktop Entry", "URL", NULL
, 0, NULL
},
349 { "Desktop Entry", "Version", NULL
, 0, NULL
},
353 r
= config_parse(NULL
, service
->path
, NULL
,
355 xdg_config_item_table_lookup
, items
,
356 CONFIG_PARSE_WARN
, service
,
358 /* If parsing failed, only hide the file so it will still mask others. */
360 log_warning_errno(r
, "Failed to parse %s, ignoring it", service
->path
);
361 service
->hidden
= true;
364 return TAKE_PTR(service
);
367 int xdg_autostart_format_exec_start(
369 char **ret_exec_start
) {
371 _cleanup_strv_free_
char **exec_split
= NULL
;
378 * Unfortunately, there is a mismatch between systemd's idea of $PATH
379 * and XDGs. i.e. we need to ensure that we have an absolute path to
380 * support cases where $PATH has been modified from the default set.
382 * Note that this is only needed for development environments though;
383 * so while it is important, this should have no effect in production
386 * To be compliant with the XDG specification, we also need to strip
387 * certain parameters and such. Doing so properly makes parsing the
388 * command line unavoidable.
390 * NOTE: Technically, XDG only specifies " as quotes, while this also
393 r
= strv_split_full(&exec_split
, exec
, NULL
, EXTRACT_UNQUOTE
| EXTRACT_RELAX
);
397 if (strv_isempty(exec_split
))
398 return log_warning_errno(SYNTHETIC_ERRNO(EINVAL
), "Exec line is empty");
401 for (i
= n
= 0; exec_split
[i
]; i
++) {
402 _cleanup_free_
char *c
= NULL
, *raw
= NULL
, *p
= NULL
, *escaped
= NULL
, *quoted
= NULL
;
404 r
= cunescape(exec_split
[i
], 0, &c
);
406 return log_debug_errno(r
, "Failed to unescape '%s': %m", exec_split
[i
]);
409 _cleanup_free_
char *executable
= NULL
;
411 /* This is the executable, find it in $PATH */
413 r
= find_executable(c
, &executable
);
415 return log_info_errno(r
, "Exec binary '%s' does not exist: %m", c
);
417 escaped
= cescape(executable
);
422 exec_split
[n
++] = TAKE_PTR(escaped
);
427 * Remove any standardised XDG fields; we assume they never appear as
428 * part of another argument as that just does not make any sense as
429 * they can be empty (GLib will e.g. turn "%f" into an empty argument).
430 * Other implementations may handle this differently.
437 "%i", /* Location of icon, could be implemented. */
438 "%c", /* Translated application name, could be implemented. */
439 "%k", /* Location of desktop file, could be implemented. */
446 * %% -> % and then % -> %% means that we correctly quote any %
447 * and also quote any left over (and invalid) % specifier from
450 raw
= strreplace(c
, "%%", "%");
453 p
= strreplace(raw
, "%", "%%");
456 escaped
= cescape(p
);
460 quoted
= strjoin("\"", escaped
, "\"");
465 exec_split
[n
++] = TAKE_PTR(quoted
);
467 for (; exec_split
[n
]; n
++)
468 exec_split
[n
] = mfree(exec_split
[n
]);
470 res
= strv_join(exec_split
, " ");
474 *ret_exec_start
= res
;
478 static int xdg_autostart_generate_desktop_condition(
480 const char *test_binary
,
481 const char *condition
) {
485 /* Generate an ExecCondition for GNOME autostart condition */
486 if (!isempty(condition
)) {
487 _cleanup_free_
char *gnome_autostart_condition_path
= NULL
, *e_autostart_condition
= NULL
;
489 r
= find_executable(test_binary
, &gnome_autostart_condition_path
);
491 log_full_errno(r
== -ENOENT
? LOG_DEBUG
: LOG_WARNING
, r
,
492 "%s not found: %m", test_binary
);
493 fprintf(f
, "# ExecCondition using %s skipped due to missing binary.\n", test_binary
);
497 e_autostart_condition
= cescape(condition
);
498 if (!e_autostart_condition
)
502 "ExecCondition=%s --condition \"%s\"\n",
503 gnome_autostart_condition_path
,
504 e_autostart_condition
);
510 int xdg_autostart_service_generate_unit(
511 XdgAutostartService
*service
,
514 _cleanup_free_
char *path_escaped
= NULL
, *exec_start
= NULL
, *unit
= NULL
;
515 _cleanup_fclose_
FILE *f
= NULL
;
520 /* Nothing to do for hidden services. */
521 if (service
->hidden
) {
522 log_debug("Not generating service for XDG autostart %s, it is hidden.", service
->name
);
526 if (service
->systemd_skip
) {
527 log_debug("Not generating service for XDG autostart %s, should be skipped by generator.", service
->name
);
531 /* Nothing to do if type is not Application. */
532 if (!streq_ptr(service
->type
, "Application")) {
533 log_debug("Not generating service for XDG autostart %s, only Type=Application is supported.", service
->name
);
537 if (!service
->exec_string
) {
538 log_warning("Not generating service for XDG autostart %s, it is has no Exec= line.", service
->name
);
543 * The TryExec key cannot be checked properly from the systemd unit,
544 * it is trivial to check using find_executable though.
546 if (service
->try_exec
) {
547 r
= find_executable(service
->try_exec
, NULL
);
549 log_full_errno(r
== -ENOENT
? LOG_DEBUG
: LOG_WARNING
, r
,
550 "Not generating service for XDG autostart %s, could not find TryExec= binary %s: %m",
551 service
->name
, service
->try_exec
);
556 r
= xdg_autostart_format_exec_start(service
->exec_string
, &exec_start
);
559 "Not generating service for XDG autostart %s, error parsing Exec= line: %m",
564 if (service
->gnome_autostart_phase
) {
565 /* There is no explicit value for the "Application" phase. */
566 log_debug("Not generating service for XDG autostart %s, startup phases are not supported.",
571 path_escaped
= specifier_escape(service
->path
);
575 unit
= path_join(dest
, service
->name
);
579 f
= fopen(unit
, "wxe");
581 return log_error_errno(errno
, "Failed to create unit file %s: %m", unit
);
584 "# Automatically generated by systemd-xdg-autostart-generator\n\n"
586 "Documentation=man:systemd-xdg-autostart-generator(8)\n"
588 "PartOf=graphical-session.target\n\n",
591 if (service
->description
) {
592 _cleanup_free_
char *t
= NULL
;
594 t
= specifier_escape(service
->description
);
598 fprintf(f
, "Description=%s\n", t
);
601 /* Only start after the session is ready. */
603 "After=graphical-session.target\n");
614 if (service
->working_directory
) {
615 _cleanup_free_
char *e_working_directory
= NULL
;
617 e_working_directory
= cescape(service
->working_directory
);
618 if (!e_working_directory
)
621 fprintf(f
, "WorkingDirectory=-%s\n", e_working_directory
);
624 /* Generate an ExecCondition to check $XDG_CURRENT_DESKTOP */
625 if (!strv_isempty(service
->only_show_in
) || !strv_isempty(service
->not_show_in
)) {
626 _cleanup_free_
char *only_show_in
= NULL
, *not_show_in
= NULL
, *e_only_show_in
= NULL
, *e_not_show_in
= NULL
;
628 only_show_in
= strv_join(service
->only_show_in
, ":");
629 not_show_in
= strv_join(service
->not_show_in
, ":");
630 if (!only_show_in
|| !not_show_in
)
633 e_only_show_in
= cescape(only_show_in
);
634 e_not_show_in
= cescape(not_show_in
);
635 if (!e_only_show_in
|| !e_not_show_in
)
638 /* Just assume the values are reasonably sane */
640 "ExecCondition=" ROOTLIBEXECDIR
"/systemd-xdg-autostart-condition \"%s\" \"%s\"\n",
645 r
= xdg_autostart_generate_desktop_condition(f
,
646 "gnome-systemd-autostart-condition",
647 service
->autostart_condition
);
651 r
= xdg_autostart_generate_desktop_condition(f
,
652 "kde-systemd-start-condition",
653 service
->kde_autostart_condition
);
657 (void) generator_add_symlink(dest
, "xdg-desktop-autostart.target", "wants", service
->name
);