<varlistentry>
<term><varname>L</varname></term>
<term><varname>L+</varname></term>
- <listitem><para>Create a symlink if it does not exist
- yet. If suffixed with <varname>+</varname> and a file or
- directory already exists where the symlink is to be created,
- it will be removed and be replaced by the symlink. If the
- argument is omitted, symlinks to files with the same name
- residing in the directory
- <filename>/usr/share/factory/</filename> are created. Note
- that permissions on symlinks are ignored.
- </para></listitem>
+ <term><varname>L?</varname></term>
+ <listitem><para>Create a symlink if it does not exist yet. If suffixed with <varname>+</varname>
+ and a file or directory already exists where the symlink is to be created, it will be removed and
+ be replaced by the symlink. If suffixed with <varname>?</varname> and the source path does not
+ exist, the symlink is not created. If the argument is omitted, symlinks to files with the same name
+ residing in the directory <filename>/usr/share/factory/</filename> are created. Note that
+ permissions on symlinks are ignored.</para></listitem>
</varlistentry>
<varlistentry>
bool purge:1;
+ bool ignore_if_target_missing:1;
+
OperationMask done;
} Item;
RECURSIVE_REMOVE_PATH);
}
+static bool supports_ignore_if_target_missing(ItemType t) {
+ return t == CREATE_SYMLINK;
+}
+
static struct Item* find_glob(OrderedHashmap *h, const char *match) {
ItemArray *j;
assert(c);
assert(i);
+ if (i->ignore_if_target_missing) {
+ r = chase(i->argument, arg_root, CHASE_SAFE|CHASE_PREFIX_ROOT|CHASE_NOFOLLOW, /*ret_path=*/ NULL, /*ret_fd=*/ NULL);
+ if (r == -ENOENT) {
+ /* Silently skip over lines where the source file is missing. */
+ log_info("Symlink source path '%s%s' does not exist, skipping line.", strempty(arg_root), i->argument);
+ return 0;
+ }
+ if (r < 0)
+ return log_error_errno(r, "Failed to check if symlink source path '%s%s' exists: %m", strempty(arg_root), i->argument);
+ }
+
r = path_extract_filename(i->path, &bn);
if (r < 0)
return log_error_errno(r, "Failed to extract filename from path '%s': %m", i->path);
ItemArray *existing;
OrderedHashmap *h;
bool append_or_force = false, boot = false, allow_failure = false, try_replace = false,
- unbase64 = false, from_cred = false, missing_user_or_group = false, purge = false;
+ unbase64 = false, from_cred = false, missing_user_or_group = false, purge = false,
+ ignore_if_target_missing = false;
int r;
assert(fname);
from_cred = true;
else if (action[pos] == '$' && !purge)
purge = true;
+ else if (action[pos] == '?' && !ignore_if_target_missing)
+ ignore_if_target_missing = true;
else {
*invalid_config = true;
return log_syntax(NULL, LOG_ERR, fname, line, SYNTHETIC_ERRNO(EBADMSG),
i.allow_failure = allow_failure;
i.try_replace = try_replace;
i.purge = purge;
+ i.ignore_if_target_missing = ignore_if_target_missing;
r = specifier_printf(path, PATH_MAX-1, specifier_table, arg_root, NULL, &i.path);
if (ERRNO_IS_NEG_NOINFO(r))
"Purge flag '$' combined with line type '%c' which does not support purging.", (char) i.type);
}
+ if (i.ignore_if_target_missing && !supports_ignore_if_target_missing(i.type)) {
+ *invalid_config = true;
+ return log_syntax(NULL, LOG_ERR, fname, line, SYNTHETIC_ERRNO(EBADMSG),
+ "Modifier '?' combined with line type '%c' which does not support this modifier.", (char) i.type);
+ }
+
if (!should_include_path(i.path))
return 0;
if (!i.argument)
return log_oom();
}
+
break;
case COPY_FILES: