我有一个.sh脚本,它使用set -o xtrace
来打印以下所有命令。
我想为这些命令着色。我试着像这样使用PS4变量:
export PS4='\[\e[36m\]\+ \[\e[m\]'
但是这只会使+
- 字符变色,如果我遗漏\[\e[m\]
我的完整输出并且已执行程序的输出被着色。
是否有另一个变量在打印命令后附加,我可以重置颜色或有另一种方式。
感谢。
答案 0 :(得分:2)
由于终端着色的工作方式,再加上say bash跟踪的工作方式,实现起来很丑陋,但这是您的操作方法:
首先,您需要找到一些不会出现在跟踪,stdout和stderr中的唯一字符串。在我的示例中,它是@@@
。
然后,如果您的脚本名为example.sh
,则运行
$ ./color_trace.sh example.sh
此脚本./color_trace.sh
在哪里:
#!/bin/bash
RED=1
UNIQUE_STR='@@@'
COLOR=$RED
normal()
{
tput sgr0 # tell the terminal to not style the output
}
colored()
{
tput setaf $COLOR # tell the terminal to set forward color to $COLOR
}
export PS4="+${UNIQUE_STR}" # artificially insert $UNIQUE_STR to the trace
exec &> >(sed "s/\(\+\)*\+\(${UNIQUE_STR}\)\(.*\)$/$(colored)\1+\3$(normal)/") # see below
set -x # set trace
source "$@" # run the commands in the current shell so that all settings apply
这将为跟踪命令red上色。
它的执行方式如下:
$UNIQUE_STR
插入输出exec &> >(...)
)路由所有输出-stdout和stderr-sed
)假定跟踪的形式为+++${UNIQUE_STR} <some interesting trace>
$UNIQUE_STR
(s/...\(${UNIQUE_STR}\).../
)查找行并将其删除(/...\1+\3.../
,其中\1
是++
,而\3
是{{1 }})<some interesting trace>
,请告诉终端将颜色设置为该行之前的红色,并取消设置该行之后的颜色($UNIQUE_STR
)。首先,函数/$(colored)...$(normal)/
和colored()
不接受要着色的文本。相反,他们指示终端将颜色设置为接下来要写入的内容,并在写入后指示其取消设置颜色。这就是Unix终端中着色或任何样式的工作方式。
第二,跟踪到达stderr,因此它将与stderr的其他输出混合。可以使用BASH_XTRACEFD
进行更改,但是在这里无法正常工作。
由于您要着色,因此假定您要将迹线与其他输出区分开,因此假定所有输出(包括stdout,stderr和迹线)都将到达终端。
如果所有内容都将发送到终端,则为了使跟踪与实际输出正确交织,所有输出-stdout,stderr和跟踪-必须在同一文件描述符上执行。这就是为什么我们必须通过normal()
重定向所有内容,而不仅仅是跟踪。如果我们仅重定向跟踪,那么您将从各种命令中获得大量输出,只有在此之后,您才能看到这些命令的所有跟踪,例如
sed
如果不是因为文件描述符问题,我们就不需要... actual output from /etc/passwd...
... actual output from /etc/group...
+source ./script.sh
++cat_file /etc/passwd
++cat /etc/passwd
++cat_file /etc/group
++cat /etc/group
,我们可以使用类似这样的东西:
UNIQUE_STR
值得注意的是,即使在此版本中,我们也必须打开和关闭每一行的颜色,因为“实际输出”的行需要不着色。
答案 1 :(得分:0)
import SwiftUI
struct ContentView: View {
@State private var foo = false
var body: some View {
Form{
Toggle(isOn: $foo, label: {
Text("Label")
})
}
}
}
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
ContentView()
}
}
来自export PS4=$'+\e[33m $BASH_SOURCE:${BASH_LINENO[0]} ${FUNCNAME[0]:-NOFUNC}() \e[0m+ '
man bash
与PS1一样扩展此参数的值并打印该值 在执行跟踪期间每个命令bash显示之前。首先 必要时,将PS4扩展值的特征重复多次,以指示多个间接级别。默认值为``+''。
答案 2 :(得分:0)
PS0
in Bash 4.4 几乎的引入使我们能够满足您的需求; PS0='\e[0m'
在每个命令运行之前向终端写入一个重置代码,但是不幸的是,它是在打印PS4
之前而不是之后打印的,所以这样不好。正如其他人所说,我认为您今天无法通过bash来完成自己想做的事情; PS4
根本不能表现出足够的钩子来正确地为命令跟踪着色,并且没有其他钩子可以在适当的时间触发。
根据您的确切目标,您可能可以使用DEBUG
中的-x
陷阱 。它不能完美地复制-x
,但我们可以得到类似的效果。
$ debug() {
# print a '+' for every element in BASH_LINENO, similar to PS4's behavior
printf '%s' "${BASH_LINENO[@]/*/+}"
# Then print the current command, colored
printf ' \e[36m%s\e[0m\n' "$BASH_COMMAND"
}
$ trap debug DEBUG
$ shopt -s extdebug # necessary for the DEBUG trap to carry into functions
尽管调试陷阱并不能完全反映-x
的行为,但这 大多有效,因此输出有些不同。这是一个示例:
$ foo() { bar "$@"; }
$ bar() { printf '%s\n' "$@" | grep baz; }
$ foo biff bang baz
+ foo biff bang baz
++ foo biff bang baz
++ bar "$@"
+++ bar "$@"
+++ printf '%s\n' "$@"
+++ grep baz
baz
我不确定为什么该函数调用foo
和bar
会打印两次,尽管我认为第二个条目是进入该函数的DEBUG陷阱,我们也许可以检测到并抑制它以某种方式,例如通过检查FUNCNAME
或BASH_LINENO
。如果可以解决,我将更新此答案。
答案 3 :(得分:-1)
我通过使用PS1 var来整理整个终端,下面是我的示例,以便完全清楚是什么颜色(以及在术语中在文件夹名称之前添加git分支):
Color_Off="\[\033[0m\]" # Text Reset
# Regular Colors
Black="\[\033[0;30m\]" # Black
Red="\[\033[0;31m\]" # Red
Green="\[\033[0;32m\]" # Green
Yellow="\[\033[0;33m\]" # Yellow
Blue="\[\033[0;34m\]" # Blue
Purple="\[\033[0;35m\]" # Purple
Cyan="\[\033[0;36m\]" # Cyan
White="\[\033[0;37m\]" # White
# Bold
BBlack="\[\033[1;30m\]" # Black
BRed="\[\033[1;31m\]" # Red
BGreen="\[\033[1;32m\]" # Green
BYellow="\[\033[1;33m\]" # Yellow
BBlue="\[\033[1;34m\]" # Blue
BPurple="\[\033[1;35m\]" # Purple
BCyan="\[\033[1;36m\]" # Cyan
BWhite="\[\033[1;37m\]" # White
# Underline
UBlack="\[\033[4;30m\]" # Black
URed="\[\033[4;31m\]" # Red
UGreen="\[\033[4;32m\]" # Green
UYellow="\[\033[4;33m\]" # Yellow
UBlue="\[\033[4;34m\]" # Blue
UPurple="\[\033[4;35m\]" # Purple
UCyan="\[\033[4;36m\]" # Cyan
UWhite="\[\033[4;37m\]" # White
# Background
On_Black="\[\033[40m\]" # Black
On_Red="\[\033[41m\]" # Red
On_Green="\[\033[42m\]" # Green
On_Yellow="\[\033[43m\]" # Yellow
On_Blue="\[\033[44m\]" # Blue
On_Purple="\[\033[45m\]" # Purple
On_Cyan="\[\033[46m\]" # Cyan
On_White="\[\033[47m\]" # White
# High Intensty
IBlack="\[\033[0;90m\]" # Black
IRed="\[\033[0;91m\]" # Red
IGreen="\[\033[0;92m\]" # Green
IYellow="\[\033[0;93m\]" # Yellow
IBlue="\[\033[0;94m\]" # Blue
IPurple="\[\033[0;95m\]" # Purple
ICyan="\[\033[0;96m\]" # Cyan
IWhite="\[\033[0;97m\]" # White
# Bold High Intensty
BIBlack="\[\033[1;90m\]" # Black
BIRed="\[\033[1;91m\]" # Red
BIGreen="\[\033[1;92m\]" # Green
BIYellow="\[\033[1;93m\]" # Yellow
BIBlue="\[\033[1;94m\]" # Blue
BIPurple="\[\033[1;95m\]" # Purple
BICyan="\[\033[1;96m\]" # Cyan
BIWhite="\[\033[1;97m\]" # White
# High Intensty backgrounds
On_IBlack="\[\033[0;100m\]" # Black
On_IRed="\[\033[0;101m\]" # Red
On_IGreen="\[\033[0;102m\]" # Green
On_IYellow="\[\033[0;103m\]" # Yellow
On_IBlue="\[\033[0;104m\]" # Blue
On_IPurple="\[\033[10;95m\]" # Purple
On_ICyan="\[\033[0;106m\]" # Cyan
On_IWhite="\[\033[0;107m\]" # White
Time12h="\T"
Time12a="\@"
PathShort="\w"
PathFull="\W"
NewLine="\n"
Jobs="\j"
User="\u"
Host="\h"
Dollar="\$ "
gitBranch='`git branch 2> /dev/null | grep -e ^* | sed -E s/^\\\\\*\ \(.+\)$/\(\\\\\1\)\ /`'
export PS1=$BPurple$gitBranch$BRed$PathShort$NewLine$BBlue$Dollar$Color_Off