【问题标题】:bash: How to evaluate PS1, PS2, ...? [duplicate]bash:如何评估 PS1、PS2、...? [复制]
【发布时间】:2012-04-21 01:49:18
【问题描述】:

可能重复:
Echo expanded PS1

有没有办法在 bash 脚本中“评估”PS1PS2 等?

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

例如,

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

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

【问题讨论】:

  • 也许其他人知道,但我不知道你在说什么。你想完成什么?
  • 您要评估 $PS1、$PS2 吗?或者您想为脚本中的这些变量赋予值?
  • 假设有一个函数或命令叫做evaluateevaluate PS1 的结果应该是什么?
  • @Gangadhar 是的,我想 eval $PS1$PS2 等。不,我不想从我的脚本中为这些变量赋值。
  • 啊哈,明白了。试试这个:myps1=$(echo -e "PROMPT_COMMAND=\n" | bash -i 2>&1 | head -2 | tail -1)——注意这将包含任何提示着色转义序列。

标签: bash evaluate ps1


【解决方案1】:

开源软件的一大优势是源代码是开放的 :-)

如果您下载 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 就可以出于保修目的否认所有知识 :-) 只需更改一行(我添加了 @ 987654333@位):

echo "#define DISTVERSION \"${float_dist}-pax\""

其次,更改 `builtins/Makefile.in 以添加新的源文件。这需要多个步骤。

(a) 将$(srcdir)/evalps1.def 添加到DEFSRC 的末尾。

(b) 将evalps1.o 添加到OFILES 的末尾。

(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 秒的间隔。
  • 这是您遇到的一系列问题的清单 :-) (1) 是的,我在我的系统上进行了尝试,最后一点是实际的成绩单。 (2) 是的,head -1 不适用于多行提示,您必须对此更加巧妙,并且可能需要预知行数或PS1 的预处理@ 计数@987654354 @ 部分。 (3) 我不确定颜色的问题是什么,尽管您可能必须强制设置TERM。 (4) 我无法保证速度,这取决于启动 shell 需要多长时间。如果第二个解决方案不合适,您可能需要查看第一个。
  • 但我怀疑这可能是更多的努力,尽管至少它应该解决这些问题。您甚至可以编写一个与 C 代码执行类似操作的脚本来翻译任意 PS1
  • “你有很多问题:-)” 只是为你提供反馈,不希望你在这个“洗衣清单”上工作。已经 +1 了你的答案,感谢你花时间写而不是(相当合理地)处理你自己的人们通常拥有的东西的清单:-) 特别是在周末:-))
猜你喜欢
  • 2012-05-17
  • 2017-02-22
  • 2011-04-18
  • 2020-12-11
  • 2019-05-08
  • 1970-01-01
  • 1970-01-01
  • 2014-05-27
  • 2013-05-13
相关资源
最近更新 更多