1 /* SPDX-License-Identifier: LGPL-2.1+ */
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
) {
32 strv_free(s
->only_show_in
);
33 strv_free(s
->not_show_in
);
36 free(s
->autostart_condition
);
37 free(s
->kde_autostart_condition
);
39 free(s
->gnome_autostart_phase
);
44 char *xdg_autostart_service_translate_name(const char *name
) {
45 _cleanup_free_
char *c
= NULL
, *escaped
= NULL
;
52 res
= endswith(c
, ".desktop");
56 escaped
= unit_name_escape(c
);
60 return strjoin("app-", escaped
, "-autostart.service");
63 static int xdg_config_parse_bool(
68 unsigned section_line
,
82 if (streq(rvalue
, "true"))
84 else if (streq(rvalue
, "false"))
87 return log_syntax(unit
, LOG_ERR
, filename
, line
, SYNTHETIC_ERRNO(EINVAL
), "Invalid value for boolean: %s", rvalue
);
92 /* Unescapes the string in-place, returns non-zero status on error. */
93 static int xdg_unescape_string(
106 for (; *in
; in
++, out
++) {
108 /* Move forward, and ensure it is a valid escape. */
128 /* Technically only permitted for strv. */
132 return log_syntax(unit
, LOG_ERR
, filename
, line
, SYNTHETIC_ERRNO(EINVAL
), "Undefined escape sequence \\%c.", *in
);
145 /* Note: We do not bother with unescaping the strings, hence the _raw postfix. */
146 static int xdg_config_parse_string(
148 const char *filename
,
151 unsigned section_line
,
158 _cleanup_free_
char *res
= NULL
;
167 /* XDG does not allow duplicate definitions. */
169 log_syntax(unit
, LOG_ERR
, filename
, line
, 0, "Key %s was defined multiple times, ignoring.", lvalue
);
173 res
= strdup(rvalue
);
177 r
= xdg_unescape_string(unit
, filename
, line
, res
);
181 *out
= TAKE_PTR(res
);
185 static int strv_strndup_unescape_and_push(
187 const char *filename
,
198 _cleanup_free_
char *copy
= NULL
;
201 copy
= strndup(start
, end
- start
);
205 r
= xdg_unescape_string(unit
, filename
, line
, copy
);
209 if (!greedy_realloc((void**) sv
, n_allocated
, *n
+ 2, sizeof(char*))) /* One extra for NULL */
212 (*sv
)[*n
] = TAKE_PTR(copy
);
213 (*sv
)[*n
+ 1] = NULL
;
219 static int xdg_config_parse_strv(
221 const char *filename
,
224 unsigned section_line
,
231 char ***ret_sv
= data
;
239 /* XDG does not allow duplicate definitions. */
241 log_syntax(unit
, LOG_ERR
, filename
, line
, 0, "Key %s was already defined, ignoring.", lvalue
);
245 size_t n
= 0, n_allocated
= 0;
246 _cleanup_strv_free_
char **sv
= NULL
;
248 if (!GREEDY_REALLOC0(sv
, n_allocated
, 1))
251 /* We cannot use strv_split because it does not handle escaping correctly. */
252 const char *start
= rvalue
, *end
;
254 for (end
= start
; *end
; end
++) {
256 /* Move forward, and ensure it is a valid escape. */
258 if (!strchr("sntr\\;", *end
)) {
259 log_syntax(unit
, LOG_ERR
, filename
, line
, 0, "Undefined escape sequence \\%c.", *end
);
266 r
= strv_strndup_unescape_and_push(unit
, filename
, line
,
267 &sv
, &n_allocated
, &n
,
276 /* Handle the trailing entry after the last separator */
277 r
= strv_strndup_unescape_and_push(unit
, filename
, line
,
278 &sv
, &n_allocated
, &n
,
283 *ret_sv
= TAKE_PTR(sv
);
287 static int xdg_config_item_table_lookup(
291 ConfigParserCallback
*func
,
298 /* Ignore any keys with [] as those are translations. */
299 if (strchr(lvalue
, '[')) {
306 return config_item_table_lookup(table
, section
, lvalue
, func
, ltype
, data
, userdata
);
309 XdgAutostartService
*xdg_autostart_service_parse_desktop(const char *path
) {
310 _cleanup_(xdg_autostart_service_freep
) XdgAutostartService
*service
= NULL
;
313 service
= new0(XdgAutostartService
, 1);
317 service
->path
= strdup(path
);
321 const ConfigTableItem items
[] = {
322 { "Desktop Entry", "Name", xdg_config_parse_string
, 0, &service
->description
},
323 { "Desktop Entry", "Exec", xdg_config_parse_string
, 0, &service
->exec_string
},
324 { "Desktop Entry", "TryExec", xdg_config_parse_string
, 0, &service
->try_exec
},
325 { "Desktop Entry", "Type", xdg_config_parse_string
, 0, &service
->type
},
326 { "Desktop Entry", "OnlyShowIn", xdg_config_parse_strv
, 0, &service
->only_show_in
},
327 { "Desktop Entry", "NotShowIn", xdg_config_parse_strv
, 0, &service
->not_show_in
},
328 { "Desktop Entry", "Hidden", xdg_config_parse_bool
, 0, &service
->hidden
},
329 { "Desktop Entry", "AutostartCondition", xdg_config_parse_string
, 0, &service
->autostart_condition
},
330 { "Desktop Entry", "X-KDE-autostart-condition", xdg_config_parse_string
, 0, &service
->kde_autostart_condition
},
331 { "Desktop Entry", "X-GNOME-Autostart-Phase", xdg_config_parse_string
, 0, &service
->gnome_autostart_phase
},
332 { "Desktop Entry", "X-systemd-skip", xdg_config_parse_bool
, 0, &service
->systemd_skip
},
334 /* Common entries that we do not use currently. */
335 { "Desktop Entry", "Categories", NULL
, 0, NULL
},
336 { "Desktop Entry", "Comment", NULL
, 0, NULL
},
337 { "Desktop Entry", "Encoding", NULL
, 0, NULL
},
338 { "Desktop Entry", "GenericName", NULL
, 0, NULL
},
339 { "Desktop Entry", "Icon", NULL
, 0, NULL
},
340 { "Desktop Entry", "Keywords", NULL
, 0, NULL
},
341 { "Desktop Entry", "NoDisplay", NULL
, 0, NULL
},
342 { "Desktop Entry", "StartupNotify", NULL
, 0, NULL
},
343 { "Desktop Entry", "Terminal", NULL
, 0, NULL
},
344 { "Desktop Entry", "Version", NULL
, 0, NULL
},
348 r
= config_parse(NULL
, service
->path
, NULL
,
350 xdg_config_item_table_lookup
, items
,
351 CONFIG_PARSE_WARN
, service
,
353 /* If parsing failed, only hide the file so it will still mask others. */
355 log_warning_errno(r
, "Failed to parse %s, ignoring it", service
->path
);
356 service
->hidden
= true;
359 return TAKE_PTR(service
);
362 int xdg_autostart_format_exec_start(
364 char **ret_exec_start
) {
366 _cleanup_strv_free_
char **exec_split
= NULL
;
373 * Unfortunately, there is a mismatch between systemd's idea of $PATH
374 * and XDGs. i.e. we need to ensure that we have an absolute path to
375 * support cases where $PATH has been modified from the default set.
377 * Note that this is only needed for development environments though;
378 * so while it is important, this should have no effect in production
381 * To be compliant with the XDG specification, we also need to strip
382 * certain parameters and such. Doing so properly makes parsing the
383 * command line unavoidable.
385 * NOTE: Technically, XDG only specifies " as quotes, while this also
388 r
= strv_split_extract(&exec_split
, exec
, NULL
, EXTRACT_UNQUOTE
| EXTRACT_RELAX
);
392 if (strv_isempty(exec_split
))
393 return log_warning_errno(SYNTHETIC_ERRNO(EINVAL
), "Exec line is empty");
396 for (i
= n
= 0; exec_split
[i
]; i
++) {
397 _cleanup_free_
char *c
= NULL
, *raw
= NULL
, *p
= NULL
, *escaped
= NULL
, *quoted
= NULL
;
399 r
= cunescape(exec_split
[i
], 0, &c
);
401 return log_debug_errno(r
, "Failed to unescape '%s': %m", exec_split
[i
]);
404 _cleanup_free_
char *executable
= NULL
;
406 /* This is the executable, find it in $PATH */
408 r
= find_binary(c
, &executable
);
410 return log_info_errno(r
, "Exec binary '%s' does not exist: %m", c
);
412 escaped
= cescape(executable
);
417 exec_split
[n
++] = TAKE_PTR(escaped
);
422 * Remove any standardised XDG fields; we assume they never appear as
423 * part of another argument as that just does not make any sense as
424 * they can be empty (GLib will e.g. turn "%f" into an empty argument).
425 * Other implementations may handle this differently.
432 "%i", /* Location of icon, could be implemented. */
433 "%c", /* Translated application name, could be implemented. */
434 "%k", /* Location of desktop file, could be implemented. */
441 * %% -> % and then % -> %% means that we correctly quote any %
442 * and also quote any left over (and invalid) % specifier from
445 raw
= strreplace(c
, "%%", "%");
448 p
= strreplace(raw
, "%", "%%");
451 escaped
= cescape(p
);
455 quoted
= strjoin("\"", escaped
, "\"");
460 exec_split
[n
++] = TAKE_PTR(quoted
);
462 for (; exec_split
[n
]; n
++)
463 exec_split
[n
] = mfree(exec_split
[n
]);
465 res
= strv_join(exec_split
, " ");
469 *ret_exec_start
= res
;
473 static int xdg_autostart_generate_desktop_condition(
475 const char *test_binary
,
476 const char *condition
) {
480 /* Generate an ExecCondition for GNOME autostart condition */
481 if (!isempty(condition
)) {
482 _cleanup_free_
char *gnome_autostart_condition_path
= NULL
, *e_autostart_condition
= NULL
;
484 r
= find_binary(test_binary
, &gnome_autostart_condition_path
);
486 log_full_errno(r
== -ENOENT
? LOG_INFO
: LOG_WARNING
, r
,
487 "%s not found: %m", test_binary
);
488 fprintf(f
, "# ExecCondition using %s skipped due to missing binary.\n", test_binary
);
492 e_autostart_condition
= cescape(condition
);
493 if (!e_autostart_condition
)
497 "ExecCondition=%s --condition \"%s\"\n",
498 gnome_autostart_condition_path
,
499 e_autostart_condition
);
505 int xdg_autostart_service_generate_unit(
506 XdgAutostartService
*service
,
509 _cleanup_free_
char *path_escaped
= NULL
, *exec_start
= NULL
, *unit
= NULL
;
510 _cleanup_fclose_
FILE *f
= NULL
;
515 /* Nothing to do for hidden services. */
516 if (service
->hidden
) {
517 log_info("Not generating service for XDG autostart %s, it is hidden.", service
->name
);
521 if (service
->systemd_skip
) {
522 log_info("Not generating service for XDG autostart %s, should be skipped by generator.", service
->name
);
526 /* Nothing to do if type is not Application. */
527 if (!streq_ptr(service
->type
, "Application")) {
528 log_info("Not generating service for XDG autostart %s, only Type=Application is supported.", service
->name
);
532 if (!service
->exec_string
) {
533 log_warning("Not generating service for XDG autostart %s, it is has no Exec= line.", service
->name
);
538 * The TryExec key cannot be checked properly from the systemd unit,
539 * it is trivial to check using find_binary though.
541 if (service
->try_exec
) {
542 r
= find_binary(service
->try_exec
, NULL
);
544 log_full_errno(r
== -ENOENT
? LOG_INFO
: LOG_WARNING
, r
,
545 "Not generating service for XDG autostart %s, could not find TryExec= binary %s: %m",
546 service
->name
, service
->try_exec
);
551 r
= xdg_autostart_format_exec_start(service
->exec_string
, &exec_start
);
554 "Not generating service for XDG autostart %s, error parsing Exec= line: %m",
559 if (service
->gnome_autostart_phase
) {
560 /* There is no explicit value for the "Application" phase. */
561 log_info("Not generating service for XDG autostart %s, startup phases are not supported.",
566 path_escaped
= specifier_escape(service
->path
);
570 unit
= path_join(dest
, service
->name
);
574 f
= fopen(unit
, "wxe");
576 return log_error_errno(errno
, "Failed to create unit file %s: %m", unit
);
579 "# Automatically generated by systemd-xdg-autostart-generator\n\n"
581 "Documentation=man:systemd-xdg-autostart-generator(8)\n"
583 "PartOf=graphical-session.target\n\n",
586 if (service
->description
) {
587 _cleanup_free_
char *t
= NULL
;
589 t
= specifier_escape(service
->description
);
593 fprintf(f
, "Description=%s\n", t
);
596 /* Only start after the session is ready. */
598 "After=graphical-session.target\n");
609 /* Generate an ExecCondition to check $XDG_CURRENT_DESKTOP */
610 if (!strv_isempty(service
->only_show_in
) || !strv_isempty(service
->not_show_in
)) {
611 _cleanup_free_
char *only_show_in
= NULL
, *not_show_in
= NULL
, *e_only_show_in
= NULL
, *e_not_show_in
= NULL
;
613 only_show_in
= strv_join(service
->only_show_in
, ":");
614 not_show_in
= strv_join(service
->not_show_in
, ":");
615 if (!only_show_in
|| !not_show_in
)
618 e_only_show_in
= cescape(only_show_in
);
619 e_not_show_in
= cescape(not_show_in
);
620 if (!e_only_show_in
|| !e_not_show_in
)
623 /* Just assume the values are reasonably sane */
625 "ExecCondition=" ROOTLIBEXECDIR
"/systemd-xdg-autostart-condition \"%s\" \"%s\"\n",
630 r
= xdg_autostart_generate_desktop_condition(f
,
631 "gnome-systemd-autostart-condition",
632 service
->autostart_condition
);
636 r
= xdg_autostart_generate_desktop_condition(f
,
637 "kde-systemd-start-condition",
638 service
->kde_autostart_condition
);
642 (void) generator_add_symlink(dest
, "xdg-desktop-autostart.target", "wants", service
->name
);