mirror of
https://github.com/AuxXxilium/eudev.git
synced 2025-01-18 10:07:46 +07:00
exec: extend variable substitution to support splitting variable values into seperate arguments
This commit is contained in:
parent
9d25f5ed7b
commit
c24eb49e6a
12
Makefile.am
12
Makefile.am
@ -77,7 +77,8 @@ noinst_PROGRAMS = \
|
||||
test-ns \
|
||||
test-loopback \
|
||||
test-daemon \
|
||||
test-cgroup
|
||||
test-cgroup \
|
||||
test-env-replace
|
||||
|
||||
if HAVE_PAM
|
||||
pamlib_LTLIBRARIES = \
|
||||
@ -446,6 +447,15 @@ test_cgroup_CFLAGS = \
|
||||
test_cgroup_LDADD = \
|
||||
libsystemd-basic.la
|
||||
|
||||
test_env_replace_SOURCES = \
|
||||
src/test-env-replace.c
|
||||
|
||||
test_env_replace_CFLAGS = \
|
||||
$(AM_CFLAGS)
|
||||
|
||||
test_env_replace_LDADD = \
|
||||
libsystemd-basic.la
|
||||
|
||||
systemd_logger_SOURCES = \
|
||||
src/logger.c \
|
||||
src/sd-daemon.c \
|
||||
|
4
fixme
4
fixme
@ -35,9 +35,9 @@
|
||||
|
||||
* systemctl status $PID, systemctl stop $PID!
|
||||
|
||||
* place /etc/inittab with explaining blurb.
|
||||
/dev/null symlinks supporten
|
||||
|
||||
* In command lines, support both "$FOO" and $FOO
|
||||
* place /etc/inittab with explaining blurb.
|
||||
|
||||
* /etc must always take precedence even if we follow symlinks!
|
||||
|
||||
|
@ -274,10 +274,14 @@
|
||||
<citerefentry><refentrytitle>systemd.unit</refentrytitle><manvolnum>5</manvolnum></citerefentry>. On
|
||||
top of that basic environment variable
|
||||
substitution is supported, where
|
||||
<literal>$(FOO)</literal> is replaced
|
||||
by the value of the environment
|
||||
variable of the same
|
||||
name.</para></listitem>
|
||||
<literal>${FOO}</literal> is replaced
|
||||
by the string value of the environment
|
||||
variable of the same name. Also
|
||||
<literal>$FOO</literal> may appear as
|
||||
seperate word on the command line in
|
||||
which case the variable is replaced by
|
||||
its value split at
|
||||
whitespaces.</para></listitem>
|
||||
</varlistentry>
|
||||
|
||||
<varlistentry>
|
||||
|
59
src/test-env-replace.c
Normal file
59
src/test-env-replace.c
Normal file
@ -0,0 +1,59 @@
|
||||
/*-*- Mode: C; c-basic-offset: 8 -*-*/
|
||||
|
||||
/***
|
||||
This file is part of systemd.
|
||||
|
||||
Copyright 2010 Lennart Poettering
|
||||
|
||||
systemd is free software; you can redistribute it and/or modify it
|
||||
under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation; either version 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
systemd is distributed in the hope that it will be useful, but
|
||||
WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with systemd; If not, see <http://www.gnu.org/licenses/>.
|
||||
***/
|
||||
|
||||
#include <unistd.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "util.h"
|
||||
#include "log.h"
|
||||
#include "strv.h"
|
||||
|
||||
int main(int argc, char *argv[]) {
|
||||
|
||||
const char *env[] = {
|
||||
"FOO=BAR BAR",
|
||||
"BAR=waldo",
|
||||
NULL
|
||||
};
|
||||
|
||||
const char *line[] = {
|
||||
"FOO$FOO",
|
||||
"FOO$FOOFOO",
|
||||
"FOO${FOO}$FOO",
|
||||
"FOO${FOO}",
|
||||
"${FOO}",
|
||||
"$FOO",
|
||||
"$FOO$FOO",
|
||||
"${FOO}${BAR}",
|
||||
"${FOO",
|
||||
NULL
|
||||
};
|
||||
|
||||
char **i, **r;
|
||||
|
||||
r = replace_env_argv((char**) line, (char**) env);
|
||||
|
||||
STRV_FOREACH(i, r)
|
||||
printf("%s\n", *i);
|
||||
|
||||
strv_free(r);
|
||||
|
||||
}
|
51
src/util.c
51
src/util.c
@ -2778,7 +2778,7 @@ void status_welcome(void) {
|
||||
char *replace_env(const char *format, char **env) {
|
||||
enum {
|
||||
WORD,
|
||||
DOLLAR,
|
||||
CURLY,
|
||||
VARIABLE
|
||||
} state = WORD;
|
||||
|
||||
@ -2793,11 +2793,11 @@ char *replace_env(const char *format, char **env) {
|
||||
|
||||
case WORD:
|
||||
if (*e == '$')
|
||||
state = DOLLAR;
|
||||
state = CURLY;
|
||||
break;
|
||||
|
||||
case DOLLAR:
|
||||
if (*e == '(') {
|
||||
case CURLY:
|
||||
if (*e == '{') {
|
||||
if (!(k = strnappend(r, word, e-word-1)))
|
||||
goto fail;
|
||||
|
||||
@ -2821,7 +2821,7 @@ char *replace_env(const char *format, char **env) {
|
||||
break;
|
||||
|
||||
case VARIABLE:
|
||||
if (*e == ')') {
|
||||
if (*e == '}') {
|
||||
char *t;
|
||||
|
||||
if ((t = strv_env_get_with_length(env, word+2, e-word-2))) {
|
||||
@ -2853,12 +2853,49 @@ fail:
|
||||
|
||||
char **replace_env_argv(char **argv, char **env) {
|
||||
char **r, **i;
|
||||
unsigned k = 0;
|
||||
unsigned k = 0, l = 0;
|
||||
|
||||
if (!(r = new(char*, strv_length(argv)+1)))
|
||||
l = strv_length(argv);
|
||||
|
||||
if (!(r = new(char*, l+1)))
|
||||
return NULL;
|
||||
|
||||
STRV_FOREACH(i, argv) {
|
||||
|
||||
/* If $FOO appears as single word, replace it by the split up variable */
|
||||
if ((*i)[0] == '$') {
|
||||
char *e = strv_env_get(env, *i+1);
|
||||
|
||||
if (e) {
|
||||
char **w, **m;
|
||||
unsigned q;
|
||||
|
||||
if (!(m = strv_split_quoted(e))) {
|
||||
r[k] = NULL;
|
||||
strv_free(r);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
q = strv_length(m);
|
||||
l = l + q - 1;
|
||||
|
||||
if (!(w = realloc(r, sizeof(char*) * (l+1)))) {
|
||||
r[k] = NULL;
|
||||
strv_free(r);
|
||||
strv_free(m);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
r = w;
|
||||
memcpy(r + k, m, q * sizeof(char*));
|
||||
free(m);
|
||||
|
||||
k += q;
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
/* If ${FOO} appears as part of a word, replace it by the variable as-is */
|
||||
if (!(r[k++] = replace_env(*i, env))) {
|
||||
strv_free(r);
|
||||
return NULL;
|
||||
|
1
units/.gitignore
vendored
1
units/.gitignore
vendored
@ -6,3 +6,4 @@ graphical.target
|
||||
multi-user.target
|
||||
getty@.service
|
||||
remote-fs.target
|
||||
test-env-replace
|
||||
|
@ -22,4 +22,4 @@ KillMode=process-group
|
||||
|
||||
# Bash ignores SIGTERM, so we send SIGHUP first, to ensure that bash
|
||||
# terminates cleanly.
|
||||
ExecStop=-/bin/kill -HUP $(MAINPID)
|
||||
ExecStop=-/bin/kill -HUP ${MAINPID}
|
||||
|
@ -25,4 +25,4 @@ KillMode=process-group
|
||||
|
||||
# Bash ignores SIGTERM, so we send SIGHUP first, to ensure that bash
|
||||
# terminates cleanly.
|
||||
ExecStop=-/bin/kill -HUP $(MAINPID)
|
||||
ExecStop=-/bin/kill -HUP ${MAINPID}
|
||||
|
Loading…
Reference in New Issue
Block a user