mirror of
https://github.com/systemd/systemd-stable.git
synced 2025-01-11 05:17:44 +03:00
Merge pull request #9901 from peterbaouoft/pr/preset_enable_multiple_instances
install: allow instantiated units to be enabled via presets
This commit is contained in:
commit
030836923d
@ -60,6 +60,7 @@ typedef enum {
|
|||||||
typedef struct {
|
typedef struct {
|
||||||
char *pattern;
|
char *pattern;
|
||||||
PresetAction action;
|
PresetAction action;
|
||||||
|
char **instances;
|
||||||
} PresetRule;
|
} PresetRule;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
@ -87,8 +88,10 @@ static inline void presets_freep(Presets *p) {
|
|||||||
if (!p)
|
if (!p)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
for (i = 0; i < p->n_rules; i++)
|
for (i = 0; i < p->n_rules; i++) {
|
||||||
free(p->rules[i].pattern);
|
free(p->rules[i].pattern);
|
||||||
|
strv_free(p->rules[i].instances);
|
||||||
|
}
|
||||||
|
|
||||||
free(p->rules);
|
free(p->rules);
|
||||||
p->n_rules = 0;
|
p->n_rules = 0;
|
||||||
@ -1673,6 +1676,25 @@ static int install_info_discover(
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int install_info_discover_and_check(
|
||||||
|
UnitFileScope scope,
|
||||||
|
InstallContext *c,
|
||||||
|
const LookupPaths *paths,
|
||||||
|
const char *name,
|
||||||
|
SearchFlags flags,
|
||||||
|
UnitFileInstallInfo **ret,
|
||||||
|
UnitFileChange **changes,
|
||||||
|
size_t *n_changes) {
|
||||||
|
|
||||||
|
int r;
|
||||||
|
|
||||||
|
r = install_info_discover(scope, c, paths, name, flags, ret, changes, n_changes);
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
return install_info_may_process(ret ? *ret : NULL, paths, changes, n_changes);
|
||||||
|
}
|
||||||
|
|
||||||
static int install_info_symlink_alias(
|
static int install_info_symlink_alias(
|
||||||
UnitFileInstallInfo *i,
|
UnitFileInstallInfo *i,
|
||||||
const LookupPaths *paths,
|
const LookupPaths *paths,
|
||||||
@ -2396,26 +2418,20 @@ int unit_file_add_dependency(
|
|||||||
if (!config_path)
|
if (!config_path)
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
|
||||||
r = install_info_discover(scope, &c, &paths, target, SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
r = install_info_discover_and_check(scope, &c, &paths, target, SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
||||||
&target_info, changes, n_changes);
|
&target_info, changes, n_changes);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
r = install_info_may_process(target_info, &paths, changes, n_changes);
|
|
||||||
if (r < 0)
|
|
||||||
return r;
|
|
||||||
|
|
||||||
assert(target_info->type == UNIT_FILE_TYPE_REGULAR);
|
assert(target_info->type == UNIT_FILE_TYPE_REGULAR);
|
||||||
|
|
||||||
STRV_FOREACH(f, files) {
|
STRV_FOREACH(f, files) {
|
||||||
char ***l;
|
char ***l;
|
||||||
|
|
||||||
r = install_info_discover(scope, &c, &paths, *f, SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
r = install_info_discover_and_check(scope, &c, &paths, *f, SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
||||||
&i, changes, n_changes);
|
&i, changes, n_changes);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
r = install_info_may_process(i, &paths, changes, n_changes);
|
|
||||||
if (r < 0)
|
|
||||||
return r;
|
|
||||||
|
|
||||||
assert(i->type == UNIT_FILE_TYPE_REGULAR);
|
assert(i->type == UNIT_FILE_TYPE_REGULAR);
|
||||||
|
|
||||||
@ -2464,13 +2480,10 @@ int unit_file_enable(
|
|||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
|
||||||
STRV_FOREACH(f, files) {
|
STRV_FOREACH(f, files) {
|
||||||
r = install_info_discover(scope, &c, &paths, *f, SEARCH_LOAD|SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
r = install_info_discover_and_check(scope, &c, &paths, *f, SEARCH_LOAD|SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
||||||
&i, changes, n_changes);
|
&i, changes, n_changes);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
r = install_info_may_process(i, &paths, changes, n_changes);
|
|
||||||
if (r < 0)
|
|
||||||
return r;
|
|
||||||
|
|
||||||
assert(i->type == UNIT_FILE_TYPE_REGULAR);
|
assert(i->type == UNIT_FILE_TYPE_REGULAR);
|
||||||
}
|
}
|
||||||
@ -2582,10 +2595,7 @@ int unit_file_set_default(
|
|||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
r = install_info_discover(scope, &c, &paths, name, 0, &i, changes, n_changes);
|
r = install_info_discover_and_check(scope, &c, &paths, name, 0, &i, changes, n_changes);
|
||||||
if (r < 0)
|
|
||||||
return r;
|
|
||||||
r = install_info_may_process(i, &paths, changes, n_changes);
|
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
@ -2755,6 +2765,39 @@ int unit_file_exists(UnitFileScope scope, const LookupPaths *paths, const char *
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int split_pattern_into_name_and_instances(const char *pattern, char **out_unit_name, char ***out_instances) {
|
||||||
|
_cleanup_strv_free_ char **instances = NULL;
|
||||||
|
_cleanup_free_ char *unit_name = NULL;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
assert(pattern);
|
||||||
|
assert(out_instances);
|
||||||
|
assert(out_unit_name);
|
||||||
|
|
||||||
|
r = extract_first_word(&pattern, &unit_name, NULL, 0);
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
/* We handle the instances logic when unit name is extracted */
|
||||||
|
if (pattern) {
|
||||||
|
/* We only create instances when a rule of templated unit
|
||||||
|
* is seen. A rule like enable foo@.service a b c will
|
||||||
|
* result in an array of (a, b, c) as instance names */
|
||||||
|
if (!unit_name_is_valid(unit_name, UNIT_NAME_TEMPLATE))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
instances = strv_split(pattern, WHITESPACE);
|
||||||
|
if (!instances)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
*out_instances = TAKE_PTR(instances);
|
||||||
|
}
|
||||||
|
|
||||||
|
*out_unit_name = TAKE_PTR(unit_name);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int read_presets(UnitFileScope scope, const char *root_dir, Presets *presets) {
|
static int read_presets(UnitFileScope scope, const char *root_dir, Presets *presets) {
|
||||||
_cleanup_(presets_freep) Presets ps = {};
|
_cleanup_(presets_freep) Presets ps = {};
|
||||||
size_t n_allocated = 0;
|
size_t n_allocated = 0;
|
||||||
@ -2824,15 +2867,20 @@ static int read_presets(UnitFileScope scope, const char *root_dir, Presets *pres
|
|||||||
|
|
||||||
parameter = first_word(l, "enable");
|
parameter = first_word(l, "enable");
|
||||||
if (parameter) {
|
if (parameter) {
|
||||||
char *pattern;
|
char *unit_name;
|
||||||
|
char **instances = NULL;
|
||||||
|
|
||||||
pattern = strdup(parameter);
|
/* Unit_name will remain the same as parameter when no instances are specified */
|
||||||
if (!pattern)
|
r = split_pattern_into_name_and_instances(parameter, &unit_name, &instances);
|
||||||
return -ENOMEM;
|
if (r < 0) {
|
||||||
|
log_syntax(NULL, LOG_WARNING, *p, n, 0, "Couldn't parse line '%s'. Ignoring.", line);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
rule = (PresetRule) {
|
rule = (PresetRule) {
|
||||||
.pattern = pattern,
|
.pattern = unit_name,
|
||||||
.action = PRESET_ENABLE,
|
.action = PRESET_ENABLE,
|
||||||
|
.instances = instances,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2868,15 +2916,71 @@ static int read_presets(UnitFileScope scope, const char *root_dir, Presets *pres
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int query_presets(const char *name, const Presets presets) {
|
static int pattern_match_multiple_instances(
|
||||||
|
const PresetRule rule,
|
||||||
|
const char *unit_name,
|
||||||
|
char ***ret) {
|
||||||
|
|
||||||
|
_cleanup_free_ char *templated_name = NULL;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
/* If no ret is needed or the rule itself does not have instances
|
||||||
|
* initalized, we return not matching */
|
||||||
|
if (!ret || !rule.instances)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
r = unit_name_template(unit_name, &templated_name);
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
if (!streq(rule.pattern, templated_name))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* Compose a list of specified instances when unit name is a template */
|
||||||
|
if (unit_name_is_valid(unit_name, UNIT_NAME_TEMPLATE)) {
|
||||||
|
_cleanup_free_ char *prefix = NULL;
|
||||||
|
_cleanup_strv_free_ char **out_strv = NULL;
|
||||||
|
char **iter;
|
||||||
|
|
||||||
|
r = unit_name_to_prefix(unit_name, &prefix);
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
STRV_FOREACH(iter, rule.instances) {
|
||||||
|
_cleanup_free_ char *name = NULL;
|
||||||
|
r = unit_name_build(prefix, *iter, ".service", &name);
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
r = strv_extend(&out_strv, name);
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
*ret = TAKE_PTR(out_strv);
|
||||||
|
return 1;
|
||||||
|
} else {
|
||||||
|
/* We now know the input unit name is an instance name */
|
||||||
|
_cleanup_free_ char *instance_name = NULL;
|
||||||
|
|
||||||
|
r = unit_name_to_instance(unit_name, &instance_name);
|
||||||
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
if (strv_find(rule.instances, instance_name))
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int query_presets(const char *name, const Presets presets, char ***instance_name_list) {
|
||||||
PresetAction action = PRESET_UNKNOWN;
|
PresetAction action = PRESET_UNKNOWN;
|
||||||
size_t i;
|
size_t i;
|
||||||
|
char **s;
|
||||||
if (!unit_name_is_valid(name, UNIT_NAME_ANY))
|
if (!unit_name_is_valid(name, UNIT_NAME_ANY))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
for (i = 0; i < presets.n_rules; i++)
|
for (i = 0; i < presets.n_rules; i++)
|
||||||
if (fnmatch(presets.rules[i].pattern, name, FNM_NOESCAPE) == 0) {
|
if (pattern_match_multiple_instances(presets.rules[i], name, instance_name_list) > 0 ||
|
||||||
|
fnmatch(presets.rules[i].pattern, name, FNM_NOESCAPE) == 0) {
|
||||||
action = presets.rules[i].action;
|
action = presets.rules[i].action;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -2886,6 +2990,10 @@ static int query_presets(const char *name, const Presets presets) {
|
|||||||
log_debug("Preset files don't specify rule for %s. Enabling.", name);
|
log_debug("Preset files don't specify rule for %s. Enabling.", name);
|
||||||
return 1;
|
return 1;
|
||||||
case PRESET_ENABLE:
|
case PRESET_ENABLE:
|
||||||
|
if (instance_name_list && *instance_name_list)
|
||||||
|
STRV_FOREACH(s, *instance_name_list)
|
||||||
|
log_debug("Preset files say enable %s.", *s);
|
||||||
|
else
|
||||||
log_debug("Preset files say enable %s.", name);
|
log_debug("Preset files say enable %s.", name);
|
||||||
return 1;
|
return 1;
|
||||||
case PRESET_DISABLE:
|
case PRESET_DISABLE:
|
||||||
@ -2904,7 +3012,7 @@ int unit_file_query_preset(UnitFileScope scope, const char *root_dir, const char
|
|||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
return query_presets(name, presets);
|
return query_presets(name, presets, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int execute_preset(
|
static int execute_preset(
|
||||||
@ -2964,6 +3072,7 @@ static int preset_prepare_one(
|
|||||||
size_t *n_changes) {
|
size_t *n_changes) {
|
||||||
|
|
||||||
_cleanup_(install_context_done) InstallContext tmp = {};
|
_cleanup_(install_context_done) InstallContext tmp = {};
|
||||||
|
_cleanup_strv_free_ char **instance_name_list = NULL;
|
||||||
UnitFileInstallInfo *i;
|
UnitFileInstallInfo *i;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
@ -2979,19 +3088,26 @@ static int preset_prepare_one(
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
r = query_presets(name, presets);
|
r = query_presets(name, presets, &instance_name_list);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
if (r > 0) {
|
if (r > 0) {
|
||||||
r = install_info_discover(scope, plus, paths, name, SEARCH_LOAD|SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
if (instance_name_list) {
|
||||||
|
char **s;
|
||||||
|
STRV_FOREACH(s, instance_name_list) {
|
||||||
|
r = install_info_discover_and_check(scope, plus, paths, *s, SEARCH_LOAD|SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
||||||
&i, changes, n_changes);
|
&i, changes, n_changes);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
}
|
||||||
r = install_info_may_process(i, paths, changes, n_changes);
|
} else {
|
||||||
|
r = install_info_discover_and_check(scope, plus, paths, name, SEARCH_LOAD|SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
||||||
|
&i, changes, n_changes);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
} else
|
} else
|
||||||
r = install_info_discover(scope, minus, paths, name, SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
r = install_info_discover(scope, minus, paths, name, SEARCH_FOLLOW_CONFIG_SYMLINKS,
|
||||||
&i, changes, n_changes);
|
&i, changes, n_changes);
|
||||||
|
@ -983,6 +983,62 @@ static void test_with_dropin_template(const char *root) {
|
|||||||
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "with-dropin-3@instance-2.service", &state) >= 0 && state == UNIT_FILE_ENABLED);
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "with-dropin-3@instance-2.service", &state) >= 0 && state == UNIT_FILE_ENABLED);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void test_preset_multiple_instances(const char *root) {
|
||||||
|
UnitFileChange *changes = NULL;
|
||||||
|
size_t n_changes = 0;
|
||||||
|
const char *p;
|
||||||
|
UnitFileState state;
|
||||||
|
|
||||||
|
/* Set up template service files and preset file */
|
||||||
|
p = strjoina(root, "/usr/lib/systemd/system/foo@.service");
|
||||||
|
assert_se(write_string_file(p,
|
||||||
|
"[Install]\n"
|
||||||
|
"DefaultInstance=def\n"
|
||||||
|
"WantedBy=multi-user.target\n", WRITE_STRING_FILE_CREATE) >= 0);
|
||||||
|
|
||||||
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "foo@.service", &state) >= 0 && state == UNIT_FILE_DISABLED);
|
||||||
|
|
||||||
|
p = strjoina(root, "/usr/lib/systemd/system-preset/test.preset");
|
||||||
|
assert_se(write_string_file(p,
|
||||||
|
"enable foo@.service bar0 bar1 bartest\n"
|
||||||
|
"enable emptylist@.service\n" /* This line ensures the old functionality for templated unit still works */
|
||||||
|
"disable *\n" , WRITE_STRING_FILE_CREATE) >= 0);
|
||||||
|
|
||||||
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "foo@bar0.service", &state) >= 0 && state == UNIT_FILE_DISABLED);
|
||||||
|
|
||||||
|
/* Preset a single instantiated unit specified in the list */
|
||||||
|
assert_se(unit_file_preset(UNIT_FILE_SYSTEM, 0, root, STRV_MAKE("foo@bar0.service"), UNIT_FILE_PRESET_FULL, &changes, &n_changes) >= 0);
|
||||||
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "foo@bar0.service", &state) >= 0 && state == UNIT_FILE_ENABLED);
|
||||||
|
assert_se(n_changes == 1);
|
||||||
|
assert_se(changes[0].type == UNIT_FILE_SYMLINK);
|
||||||
|
p = strjoina(root, SYSTEM_CONFIG_UNIT_PATH"/multi-user.target.wants/foo@bar0.service");
|
||||||
|
assert_se(streq(changes[0].path, p));
|
||||||
|
unit_file_changes_free(changes, n_changes);
|
||||||
|
changes = NULL; n_changes = 0;
|
||||||
|
|
||||||
|
assert_se(unit_file_disable(UNIT_FILE_SYSTEM, 0, root, STRV_MAKE("foo@bar0.service"), &changes, &n_changes) >= 0);
|
||||||
|
assert_se(n_changes == 1);
|
||||||
|
assert_se(changes[0].type == UNIT_FILE_UNLINK);
|
||||||
|
p = strjoina(root, SYSTEM_CONFIG_UNIT_PATH"/multi-user.target.wants/foo@bar0.service");
|
||||||
|
assert_se(streq(changes[0].path, p));
|
||||||
|
unit_file_changes_free(changes, n_changes);
|
||||||
|
changes = NULL; n_changes = 0;
|
||||||
|
|
||||||
|
/* Check for preset-all case, only instances on the list should be enabled, not including the default instance */
|
||||||
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "foo@def.service", &state) >= 0 && state == UNIT_FILE_DISABLED);
|
||||||
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "foo@bar1.service", &state) >= 0 && state == UNIT_FILE_DISABLED);
|
||||||
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "foo@bartest.service", &state) >= 0 && state == UNIT_FILE_DISABLED);
|
||||||
|
|
||||||
|
assert_se(unit_file_preset_all(UNIT_FILE_SYSTEM, 0, root, UNIT_FILE_PRESET_FULL, &changes, &n_changes) >= 0);
|
||||||
|
assert_se(n_changes > 0);
|
||||||
|
|
||||||
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "foo@def.service", &state) >= 0 && state == UNIT_FILE_DISABLED);
|
||||||
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "foo@bar0.service", &state) >= 0 && state == UNIT_FILE_ENABLED);
|
||||||
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "foo@bar1.service", &state) >= 0 && state == UNIT_FILE_ENABLED);
|
||||||
|
assert_se(unit_file_get_state(UNIT_FILE_SYSTEM, root, "foo@bartest.service", &state) >= 0 && state == UNIT_FILE_ENABLED);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
int main(int argc, char *argv[]) {
|
int main(int argc, char *argv[]) {
|
||||||
char root[] = "/tmp/rootXXXXXX";
|
char root[] = "/tmp/rootXXXXXX";
|
||||||
const char *p;
|
const char *p;
|
||||||
@ -1012,6 +1068,7 @@ int main(int argc, char *argv[]) {
|
|||||||
test_indirect(root);
|
test_indirect(root);
|
||||||
test_preset_and_list(root);
|
test_preset_and_list(root);
|
||||||
test_preset_order(root);
|
test_preset_order(root);
|
||||||
|
test_preset_multiple_instances(root);
|
||||||
test_revert(root);
|
test_revert(root);
|
||||||
test_static_instance(root);
|
test_static_instance(root);
|
||||||
test_with_dropin(root);
|
test_with_dropin(root);
|
||||||
|
Loading…
Reference in New Issue
Block a user