彩色xtrace输出

时间:2014-09-26 20:38:01

标签: bash shell

我有一个.sh脚本,它使用set -o xtrace来打印以下所有命令。 我想为这些命令着色。我试着像这样使用PS4变量:

export PS4='\[\e[36m\]\+ \[\e[m\]'

但是这只会使+ - 字符变色,如果我遗漏\[\e[m\]我的完整输出并且已执行程序的输出被着色。 是否有另一个变量在打印命令后附加,我可以重置颜色或有另一种方式。 感谢。

4 个答案:

答案 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_STRs/...\(${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

我不确定为什么该函数调用foobar会打印两次,尽管我认为第二个条目是进入该函数的DEBUG陷阱,我们也许可以检测到并抑制它以某种方式,例如通过检查FUNCNAMEBASH_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