bash:如何评估PS1,PS2等?

9
可能是重复内容:
Echo expanded PS1

有没有办法在Bash脚本中'评估' PS1, PS2等变量?

虽然我可以使用其他方法获取当前 PS1 的所有元素,但我真的希望能够重用它的定义,而不是使用这些替代方法。

例如,

=====================================
 PS1 element -->     Alternate means
=====================================
 \u          -->     $USER
 \h          -->     $HOSTNAME
 \w          -->     $PWD
 ...
=====================================

我可以在我的脚本中使用“替代方法”列,但我不想这样做。例如,在我的PS1中,我使用终端转义序列的粗体蓝色,我希望通过评估PS1来简单地重用它。


假设有一个名为 evaluate 的函数或命令。evaluate PS1 的结果应该是什么? - n. m.
虽然我已经将pax的答案标记为最终答案,但我对更好的最终答案的寻找还没有结束。如果有其他想法,欢迎提出。 - Harry
2
明白了。试试这个:myps1=$(echo -e "PROMPT_COMMAND=\n" | bash -i 2>&1 | head -2 | tail -1) — 注意,这将包含任何提示颜色转义序列。 - n. m.
我实际上不认为这是一个重复的问题。OP明确表示替代方法不是一个选项,我认为这足以证明它的独特性。尤其是因为另一个问题的被接受答案是使用那些替代方法。投票重新打开,但为了完整起见,我将复制(带有修改)我的答案到另一个问题。 - paxdiablo
echo ${PS1@P} 似乎可以解决问题。 - Allan
显示剩余6条评论
1个回答

14

开源软件的一个巨大优势是源代码是公开的。

如果你下载bash代码(我看的是4.2版本),就会有一个包含decode_prompt_string()函数的y.tab.c文件:

char *decode_prompt_string (string) char *string; { ... }

你可以尝试提取那部分代码(以及任何需要的支持程序),并构建一个可执行文件来完成你想要的操作。虽然,从简单的尝试来看,这些支持程序似乎很多,所以这可能是一项困难的任务。


除此之外,你可以使用类似于以下命令的方式“欺骗”bash来为你展开它:

expPS1=$(echo xyzzyplughtwisty | bash -i 2>&1
         | grep xyzzyplughtwisty
         | head -1
         | sed 's/xyzzyplughtwisty//g')

现在我将其拆分成多行以提高可读性,但实际上它是一行完成的。

这样做的作用是运行一个交互式的 bash 实例,并传递(希望是)无效的命令。

因为它是交互式的,所以它会打印提示符,因此我抓取第一行包含命令字符串的内容并删除该命令字符串。剩下的就应该是提示符。

在我的系统上,我得到了以下结果:

pax> expPS1=$(echo xyzzyplughtwisty | bash -i 2>&1 | grep xyzzyplughtwisty | head -1 | sed 's/xyzzyplughtwisty//g')

pax> echo "[$expPS1]"
[pax> ]

pax> 
然而,这种方法在多行提示符存在问题时,实际上会给你常规提示符而不是当前的shell提示符。
如果您想要正确地完成它,可能需要向bash本身添加一点内容。以下是添加内部命令evalps1的步骤。
首先,更改support/mkversion.sh,以便您不会将其与“真正的”bash混淆,并且FSF可以否认所有保修知识 :-)。只需更改一行(我添加了-pax ):
echo "#define DISTVERSION \"${float_dist}-pax\""

其次,修改`builtins/Makefile.in`以添加新的源文件。这涉及到多个步骤。

(a) 在DEFSRC末尾添加$(srcdir)/evalps1.def

(b) 在OFILES末尾添加evalps1.o

(c) 添加所需的依赖项:

evalps1.o: evalps1.def $(topdir)/bashtypes.h $(topdir)/config.h \
           $(topdir)/bashintl.h $(topdir)/shell.h common.h

第三步,将builtins/evalps1.def文件添加进去,这是当您运行evalps1命令时执行的代码:

This file is evalps1.def, from which is created evalps1.c.
It implements the builtin "evalps1" in Bash.

Copyright (C) 1987-2009 Free Software Foundation, Inc.

This file is part of GNU Bash, the Bourne Again SHell.

Bash 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 3 of the License, or
(at your option) any later version.

Bash 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 Bash.  If not, see <http://www.gnu.org/licenses/>.

$PRODUCES evalps1.c

$BUILTIN evalps1
$FUNCTION evalps1_builtin
$SHORT_DOC evalps1
Outputs the fully interpreted PS1 prompt.

Outputs the PS1 prompt, fully evaluated, for whatever nefarious purposes
you require.
$END

#include <config.h>
#include "../bashtypes.h"
#include <stdio.h>
#include "../bashintl.h"
#include "../shell.h"
#include "common.h"

int
evalps1_builtin (list)
     WORD_LIST *list;
{
  char *ps1 = get_string_value ("PS1");
  if (ps1 != 0)
  {
    ps1 = decode_prompt_string (ps1);
    if (ps1 != 0)
    {
      printf ("%s", ps1);
    }
  }
  return 0;
}

这其中主要涉及 GPL 许可证(因为我对 exit.def 进行了修改),最后附加了一个非常简单的函数以获取并解码 PS1

最后,在顶层目录中构建该项目:

./configure
make

出现的bash可以重命名为paxsh,尽管我怀疑它永远不会像它的祖先那样普及 :-)

运行它,您可以看到它在运作:

pax> mv bash paxsh

pax> ./paxsh --version
GNU bash, version 4.2-pax.0(1)-release (i686-pc-linux-gnu)
Copyright (C) 2011 Free Software Foundation, Inc.
License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html>

This is free software; you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law.

pax> ./paxsh

pax> echo $BASH_VERSION
4.2-pax.0(1)-release

pax> echo "[$PS1]"
[pax> ]

pax> echo "[$(evalps1)]"
[pax> ]

pax> PS1="\h: "

paxbox01: echo "[$PS1]"
[\h: ]

paxbox01: echo "[$(evalps1)]"
[paxbox01: ]

当然,对于一些人来说,在bash中进行代码更改以添加内部命令可能被认为过度,但如果您想要对PS1进行准确评估,这肯定是一个选项。


你确定在你的系统上尝试过吗?当我尝试并打印 $expPS1 时,它的值是 "bash: : command not found..."。请注意,在多行提示符的情况下,head -1 可能无法正常工作。另外,我将我的提示符以粗体蓝色打印出来。这也是我发帖问问题的原因:我想完全保留我的 PS1,而不以任何形式重复其元素。顺便提一下,我的 PS1 设置为:"\[\033[0;34m\]\u@\h $? \w\n$\[\033[0m\]"。虽然我确实喜欢你解决问题的创意,但它非常慢:在看到任何结果之前需要等待 1-2 秒钟。 - Harry
你有一大堆问题啊 :-) (1) 是的,我在我的系统上尝试过了,最后一部分是实际的记录。(2) 是的,对于多行提示符,head -1 不起作用,你需要更加聪明一些,可能需要预先知道行数或者对 PS1 进行预处理以计算 \n 段落数。(3) 我不确定颜色的问题是什么,虽然你可能需要强制设置 TERM。(4) 我不能保证速度,这取决于启动 shell 的时间。如果第二个解决方案不够好,你可能需要看看第一个。 - paxdiablo
但我怀疑这可能需要更多的努力,尽管至少它应该解决这些问题。你甚至可以编写一个脚本来执行类似于C代码的操作,以翻译任意的“PS1”。 - paxdiablo
你的问题清单相当长啊 :-),我只是提供反馈,并不希望你去处理这个“问题清单”。我已经点赞了你的回答,感谢你抽出时间写下这些内容,而不是(理所当然地)忙于自己通常在周末会有的事情的“问题清单” :-) - Harry

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接