sed在匹配单个行字符串后附加多行字符串

时间:2015-12-16 22:44:21

标签: bash sed puppet

如果可能的话,我想将sed用于此解决方案。这是我的情况:

我有一个多行字符串,通过外部方式提供给sed - 可能是puppet shell exec或bash脚本。字符串看起来像:

$string = 'blah blah blah\nblahblah\nblah blah blah\n\n'

我想运行sed命令,在找到匹配项后附加此字符串 -

sed -i "/^\[org\/gnome\/login-screen\]/a banner-message-text='$string'" /etc/dconf/db/gdm.d/00-mybanner

问题是sed\n个字符解释为换行符。请注意,我意识到我在$string命令中声明sed变量的方式可能会因执行方式而有所不同,但即使在命令行上输入字符串也会产生多行输出。所以文件最终看起来像:

[org/gnome/login-screen]
banner-message-text='blah blah blah
blablah
blah blah blah

'
banner-message-enable=true

这会导致dconf update抛出错误。我希望文件看起来像:

[org/gnome/login-screen]
banner-message-text='blah blah blah\nblablah\nblah blah blah\n\n'
banner-message-enable=true

但我似乎无法让sed停止解释\n。 谢谢!

编辑 - 使用Puppet完成已实施解决方案的上下文(@ tom-fenech):

  $banner_msg = 'blah blah blah\nblahblah\nblah blah blah\n\n'

  $comply_cmd = "find /etc/dconf/db/gdm.d -maxdepth 1 -type f -print0 | xargs -0 sed -i '/^banner-message-text=/d' || /bin/true && \
    (grep -q \"^\[org/gnome/login-screen\]\" /etc/dconf/db/gdm.d/99-banner && \
    sed -i \"/^\[org\/gnome\/login-screen\]/a banner-message-text='\$(printf '%q' \"${banner_msg}\")'\" /etc/dconf/db/gdm.d/99-banner) || \
    (echo -e \"\n\[org/gnome/login-screen]\" >> /etc/dconf/db/gdm.d/99-banner && \
    echo \"banner-message-text='${banner_msg}'\" >> /etc/dconf/db/gdm.d/99-banner)"

  $unless_cmd = "grep -q \"^banner-message-text='\$(printf '%q' \"${banner_msg}\")'$\" /etc/dconf/db/gdm.d/99-banner"

  exec { 'gdm-banner-msg':
    path     => '/sbin:/bin:/usr/sbin:/usr/bin',
    umask    => '022',
    provider => shell,
    command  => $comply_cmd,
    unless   => $unless_cmd,
    notify   => Exec['dconf-update'],
  }

谢谢大家!

3 个答案:

答案 0 :(得分:2)

printf%q格式说明符一起使用。这会照顾你的逃脱:

$ echo "$string"
blah blah blah\nblahblah\nblah blah blah\n\n
$ printf '%q' "$string"
blah\ blah\ blah\\nblahblah\\nblah\ blah\ blah\\n\\n

来自man bash

  

%q使printf以格式输出相应的参数   可以作为shell输入重用。

这意味着这些空格也会被反斜杠转义,但这并没有什么坏处。

测试出来:

$ cat file
[org/gnome/login-screen]
$ sed "/^\[org\/gnome\/login-screen\]/a banner-message-text='$(printf '%q' "$string")'" file
[org/gnome/login-screen]
banner-message-text='blah blah blah\nblahblah\nblah blah blah\n\n'

答案 1 :(得分:1)

你需要逃避反斜杠:

$ string='blah blah blah\\nblahblah\\nblah blah blah\\n\\n'
$ echo $string
blah blah blah\\nblahblah\\nblah blah blah\\n\\n
$ cat foo.txt
foo
baz
test

$ sed -i "/^foo/a bar='$string'" foo.txt 
$ cat foo.txt 
foo
bar='blah blah blah\nblahblah\nblah blah blah\n\n'
baz
test

$

答案 2 :(得分:1)

准备初始配置文件:

echo -e "[org/gnome/login-screen]\nbanner-message-enable=true"> /etc/dconf/db/gdm.d/00-mybanner
cat /etc/dconf/db/gdm.d/00-mybanner

[org/gnome/login-screen]
banner-message-enable=true

插入新的横幅信息而不解释\ n字符:

string='blah blah blah\\nblahblah\\nblah blah blah\\n\\n'; sed -i "/^\[org\/gnome\/login-screen\]/a banner-message-text=\'$string\'" /etc/dconf/db/gdm.d/00-mybanner

查看结果:

cat /etc/dconf/db/gdm.d/00-mybanner

[org/gnome/login-screen]
banner-message-text='blah blah blah\nblahblah\nblah blah blah\n\n'
banner-message-enable=true

重点是:在字符串中的每个\ n之前添加一个额外斜杠。这将阻止sed解释\ n字符。