diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/basic/unit-name.c | 45 | ||||
| -rw-r--r-- | src/test/test-unit-name.c | 33 | 
2 files changed, 54 insertions, 24 deletions
| diff --git a/src/basic/unit-name.c b/src/basic/unit-name.c index 5fc3b9d6fd..d4a3062658 100644 --- a/src/basic/unit-name.c +++ b/src/basic/unit-name.c @@ -27,6 +27,7 @@  #include "alloc-util.h"  #include "bus-label.h" +#include "glob-util.h"  #include "hexdecoct.h"  #include "macro.h"  #include "path-util.h" @@ -35,10 +36,22 @@  #include "strv.h"  #include "unit-name.h" +/* Characters valid in a unit name. */  #define VALID_CHARS                             \ -        DIGITS LETTERS                          \ +        DIGITS                                  \ +        LETTERS                                 \          ":-_.\\" +/* The same, but also permits the single @ character that may appear */ +#define VALID_CHARS_WITH_AT                     \ +        "@"                                     \ +        VALID_CHARS + +/* All chars valid in a unit name glob */ +#define VALID_CHARS_GLOB                        \ +        VALID_CHARS_WITH_AT                     \ +        "[]!-*?" +  bool unit_name_is_valid(const char *n, UnitNameFlags flags) {          const char *e, *i, *at; @@ -637,7 +650,7 @@ static char *do_escape_mangle(const char *f, UnitNameMangle allow_globs, char *t          /* We'll only escape the obvious characters here, to play           * safe. */ -        valid_chars = allow_globs == UNIT_NAME_GLOB ? "@" VALID_CHARS "[]!-*?" : "@" VALID_CHARS; +        valid_chars = allow_globs == UNIT_NAME_GLOB ? VALID_CHARS_GLOB : VALID_CHARS_WITH_AT;          for (; *f; f++) {                  if (*f == '/') @@ -672,15 +685,15 @@ int unit_name_mangle_with_suffix(const char *name, UnitNameMangle allow_globs, c          if (!unit_suffix_is_valid(suffix))                  return -EINVAL; -        if (unit_name_is_valid(name, UNIT_NAME_ANY)) { -                /* No mangling necessary... */ -                s = strdup(name); -                if (!s) -                        return -ENOMEM; +        /* Already a fully valid unit name? If so, no mangling is necessary... */ +        if (unit_name_is_valid(name, UNIT_NAME_ANY)) +                goto good; -                *ret = s; -                return 0; -        } +        /* Already a fully valid globbing expression? If so, no mangling is necessary either... */ +        if (allow_globs == UNIT_NAME_GLOB && +            string_is_glob(name) && +            in_charset(name, VALID_CHARS_GLOB)) +                goto good;          if (is_device_path(name)) {                  r = unit_name_from_path(name, ".device", ret); @@ -705,11 +718,21 @@ int unit_name_mangle_with_suffix(const char *name, UnitNameMangle allow_globs, c          t = do_escape_mangle(name, allow_globs, s);          *t = 0; -        if (unit_name_to_type(s) < 0) +        /* Append a suffix if it doesn't have any, but only if this is not a glob, so that we can allow "foo.*" as a +         * valid glob. */ +        if ((allow_globs != UNIT_NAME_GLOB || !string_is_glob(s)) && unit_name_to_type(s) < 0)                  strcpy(t, suffix);          *ret = s;          return 1; + +good: +        s = strdup(name); +        if (!s) +                return -ENOMEM; + +        *ret = s; +        return 0;  }  int slice_build_parent_slice(const char *slice, char **ret) { diff --git a/src/test/test-unit-name.c b/src/test/test-unit-name.c index a2ff8cc299..5287ee5e6f 100644 --- a/src/test/test-unit-name.c +++ b/src/test/test-unit-name.c @@ -27,6 +27,7 @@  #include <string.h>  #include "alloc-util.h" +#include "glob-util.h"  #include "hostname-util.h"  #include "macro.h"  #include "manager.h" @@ -159,34 +160,40 @@ static void test_unit_name_to_path(void) {          test_unit_name_to_path_one("home/foo", NULL, -EINVAL);  } -static void test_unit_name_mangle_one(const char *pattern, const char *expect, int ret) { +static void test_unit_name_mangle_one(UnitNameMangle allow_globs, const char *pattern, const char *expect, int ret) {          _cleanup_free_ char *t = NULL; -        assert_se(unit_name_mangle(pattern, UNIT_NAME_NOGLOB, &t) == ret); +        assert_se(unit_name_mangle(pattern, allow_globs, &t) == ret);          puts(strna(t));          assert_se(streq_ptr(t, expect));          if (t) {                  _cleanup_free_ char *k = NULL; -                assert_se(unit_name_is_valid(t, UNIT_NAME_ANY)); +                assert_se(unit_name_is_valid(t, UNIT_NAME_ANY) || +                          (allow_globs == UNIT_NAME_GLOB && string_is_glob(t))); -                assert_se(unit_name_mangle(t, UNIT_NAME_NOGLOB, &k) == 0); +                assert_se(unit_name_mangle(t, allow_globs, &k) == 0);                  assert_se(streq_ptr(t, k));          }  }  static void test_unit_name_mangle(void) {          puts("-------------------------------------------------"); -        test_unit_name_mangle_one("foo.service", "foo.service", 0); -        test_unit_name_mangle_one("/home", "home.mount", 1); -        test_unit_name_mangle_one("/dev/sda", "dev-sda.device", 1); -        test_unit_name_mangle_one("üxknürz.service", "\\xc3\\xbcxkn\\xc3\\xbcrz.service", 1); -        test_unit_name_mangle_one("foobar-meh...waldi.service", "foobar-meh...waldi.service", 0); -        test_unit_name_mangle_one("_____####----.....service", "_____\\x23\\x23\\x23\\x23----.....service", 1); -        test_unit_name_mangle_one("_____##@;;;,,,##----.....service", "_____\\x23\\x23@\\x3b\\x3b\\x3b\\x2c\\x2c\\x2c\\x23\\x23----.....service", 1); -        test_unit_name_mangle_one("xxx@@@@/////\\\\\\\\\\yyy.service", "xxx@@@@-----\\\\\\\\\\yyy.service", 1); -        test_unit_name_mangle_one("", NULL, -EINVAL); +        test_unit_name_mangle_one(UNIT_NAME_NOGLOB, "foo.service", "foo.service", 0); +        test_unit_name_mangle_one(UNIT_NAME_NOGLOB, "/home", "home.mount", 1); +        test_unit_name_mangle_one(UNIT_NAME_NOGLOB, "/dev/sda", "dev-sda.device", 1); +        test_unit_name_mangle_one(UNIT_NAME_NOGLOB, "üxknürz.service", "\\xc3\\xbcxkn\\xc3\\xbcrz.service", 1); +        test_unit_name_mangle_one(UNIT_NAME_NOGLOB, "foobar-meh...waldi.service", "foobar-meh...waldi.service", 0); +        test_unit_name_mangle_one(UNIT_NAME_NOGLOB, "_____####----.....service", "_____\\x23\\x23\\x23\\x23----.....service", 1); +        test_unit_name_mangle_one(UNIT_NAME_NOGLOB, "_____##@;;;,,,##----.....service", "_____\\x23\\x23@\\x3b\\x3b\\x3b\\x2c\\x2c\\x2c\\x23\\x23----.....service", 1); +        test_unit_name_mangle_one(UNIT_NAME_NOGLOB, "xxx@@@@/////\\\\\\\\\\yyy.service", "xxx@@@@-----\\\\\\\\\\yyy.service", 1); +        test_unit_name_mangle_one(UNIT_NAME_NOGLOB, "", NULL, -EINVAL); + +        test_unit_name_mangle_one(UNIT_NAME_GLOB, "foo.service", "foo.service", 0); +        test_unit_name_mangle_one(UNIT_NAME_GLOB, "foo", "foo.service", 1); +        test_unit_name_mangle_one(UNIT_NAME_GLOB, "foo*", "foo*", 0); +        test_unit_name_mangle_one(UNIT_NAME_GLOB, "ü*", "\\xc3\\xbc*", 1);  }  static int test_unit_printf(void) { | 
