Bash - エスケープされた文字カラーテキストをBBCodeに変換する

Bash - エスケープされた文字カラーテキストをBBCodeに変換する

をカラーテキストterminal escape-character colored text fileに変換したいです。BBCodeそのために、スイッチはカラーエスケープ文字を追加するAndroidログファイルを作成しましたlogcat -Cd > /sdcard/logcat.txt-C出力は次のとおりです。

[0m[38;5;231mV/Zygote  ( 4666): Switching descriptor 55 to /dev/null
[0m[38;5;231mV/Zygote  ( 4666): Switching descriptor 9 to /dev/null
[0m[38;5;40mI/ggheart ( 1111): onStop
[0m[38;5;40mI/Test    ( 1111): onStop
[0m[38;5;75mD/ActivityThread( 4666): handleBindApplication:com.mxtech.videoplayer.ad
[0m[38;5;75mD/ActivityThread( 4666): setTargetHeapUtilization:0.75
[0m[38;5;75mD/ActivityThread( 4666): setTargetHeapMinFree:2097152

BBCodeカラーコードを私が書いたスクリプトに変換するにはsed

#!/bin/bash
#Use 'logcat -Cd > /sdcard/logcat.txt' as input file

sed '/\x1b/ {
    s/\x1b\[0m\x1b\[38;5;40m/\[COLOR="Green"\]/
    s/\x1b\[0m\x1b\[38;5;196m/\[COLOR="Red"\]/
    s/\x1b\[0m\x1b\[38;5;75m/\[COLOR="Blue"\]/
    s/\x1b\[0m\x1b\[38;5;166m/\[COLOR="Sienna"\]/
    s/\x1b\[0m\x1b\[38;5;231m/\[COLOR="DarkSlateGray"\]/
    s/\x1b\[38;5;40m/\[COLOR="Green"\]/
    s/\x1b\[38;5;196m/\[COLOR="Red"\]/
    s/\x1b\[38;5;75m/\[COLOR="Blue"\]/
    s/\x1b\[38;5;166m/\[COLOR="Sienna"\]/
    s/\x1b\[38;5;231m/\[COLOR="DarkSlateGray"\]/
    s/\x1b\[0m/\[COLOR="Black"\]/
    s/$/\[\/COLOR\]/
    }' <logcat.txt >logcat2.txt

処理された出力テキストは次のとおりです。

[COLOR="DarkSlateGray"]V/Zygote  ( 4666): Switching descriptor 55 to /dev/null[/COLOR]
[COLOR="DarkSlateGray"]V/Zygote  ( 4666): Switching descriptor 9 to /dev/null[/COLOR]
[COLOR="Green"]I/ggheart ( 1111): onStop[/COLOR]
[COLOR="Green"]I/Test    ( 1111): onStop[/COLOR]
[COLOR="Blue"]D/ActivityThread( 4666): handleBindApplication:com.mxtech.videoplayer.ad[/COLOR]
[COLOR="Blue"]D/ActivityThread( 4666): setTargetHeapUtilization:0.75[/COLOR]
[COLOR="Blue"]D/ActivityThread( 4666): setTargetHeapMinFree:2097152[/COLOR]

COLORこれは構文的に正確で、対応するフォーラム掲示板でうまく機能しますが、タグが複数行にわたっていないため(ほとんどの掲示板では制限されています)、最適化されず、あまりにも多くの文字を無駄にします。

同じ色の線が同じCOLORタブを閉じたり再開したりしない場合は、次のように表示されます。

[COLOR="DarkSlateGray"]V/Zygote  ( 4666): Switching descriptor 55 to /dev/null
V/Zygote  ( 4666): Switching descriptor 9 to /dev/null[/COLOR]
[COLOR="Green"]I/ggheart ( 1111): onStop
I/Test    ( 1111): onStop[/COLOR]
[COLOR="Blue"]D/ActivityThread( 4666): handleBindApplication:com.mxtech.videoplayer.ad
D/ActivityThread( 4666): setTargetHeapUtilization:0.75
D/ActivityThread( 4666): setTargetHeapMinFree:2097152[/COLOR]

これを達成する方法についてのアイデアはありますか?ラインベースのテキストプロセッサは可能ですか? shell/bash/cygwin で実行される限り必須ではありません。

答え1

"Easy using awk"の説明はsedに関連しています。

OPはよく文書化されていない機能を説明していますlogcat(参照:犯罪logcat)ログエントリに基づいて各行にハードコードされた色を割り当てるように指示します。優先順位

次のような他のオプションもあります。logcatカラーPID猫そしてカラーログキャット色を変えるフィールドたとえば、行ごとに複数の色の可能性を高めます。

以下は、ハードコーディングされたlogcat色を認識しますが、1行に複数の色を許可するawkスクリプトです。

#!/usr/bin/awk -f
BEGIN {
        colors["0"]        = "Black";
        colors["38;5;40"]  = "Green";
        colors["38;5;196"] = "Red";
        colors["38;5;75"]  = "Blue";
        colors["38;5;166"] = "Sienna";
        colors["38;5;231"] = "DarkSlateGray";
        color = "";
        last = "";
        this = "";
        save = "";
}
/\033/ {
        done = "";
        while ( $0 ~ /\033\[[;0-9]*m/ ) {
                mark = match($0, /\033\[[;0-9]*m/ );
                if ( mark > 1 ) { done = done substr($0, 1, mark - 1); }
                item = substr($0, RSTART + 2, RLENGTH - 3);
                $0 = substr($0, RSTART + RLENGTH);
                if ( match($0, /^\033\[[;0-9]*m/ ) > 0 ) continue;
                color = colors[item];
                if ( done == "" ) this = color;
                if ( item == "0" ) color = "";
                if ( color == "" ) {
                        if ( $0 != "" ) last = color;
                        $0 = "[/COLOR]" $0;
                } else if (color != last) {
                        $0 = "[COLOR=\"" color "\"]" $0;
                        last = color;
                }
        }
        $0 = done $0;
        if ( last != "" ) $0 = $0 "[/COLOR]";
}
{
        if ( NR > 1 ) {
                if ( this == last) sub("\[/COLOR\]$", "", save);
                print save;
        }
        save = $0;
}
END {
        if ( NR > 0 ) print save;
}

もともと、例えば次のようになります。

^[[0m^[[38;5;231mV/Zygote  ( 4666): Switching descriptor 55 to /dev/null
^[[0m^[[38;5;231mV/Zygote  ( 4666): Switching descriptor 9 to /dev/null
^[[0m^[[38;5;40mI/ggheart ( 1111): onStop
^[[0m^[[38;5;40mI/Test    ( 1111): onStop
^[[0m^[[38;5;75mD/ActivityThread( 4666): handleBindApplication:com.mxtech.videoplayer.ad
^[[0m^[[38;5;75mD/ActivityThread( 4666): setTargetHeapUtilization:0.75
^[[0m^[[38;5;75mD/ActivityThread( 4666): setTargetHeapMinFree:2097152

要求された出力を取得します。

[COLOR="DarkSlateGray"]V/Zygote  ( 4666): Switching descriptor 55 to /dev/null
V/Zygote  ( 4666): Switching descriptor 9 to /dev/null
[COLOR="Green"]I/ggheart ( 1111): onStop
I/Test    ( 1111): onStop
[COLOR="Blue"]D/ActivityThread( 4666): handleBindApplication:com.mxtech.videoplayer.ad
D/ActivityThread( 4666): setTargetHeapUtilization:0.75
D/ActivityThread( 4666): setTargetHeapMinFree:2097152[/COLOR]

ただし、入力を次に変更します。

^[[0m^[[38;5;231mV/Zygote  ( 4666): Switching descriptor 55 to /dev/null
^[[0m^[[38;5;231mV/Zygote  ( 4666): Switching descriptor 9 to /dev/null
^[[0m^[[38;5;40mI/ggheart ( ^[[38;5;75mI/ggheart 1111^[[0m): onStop
^[[0m^[[38;5;40mI/Test    ( 1111): onStop
^[[0m^[[38;5;75mD/ActivityThread( 4666): handleBindApplication:com.mxtech.videoplayer.ad
^[[0m^[[38;5;75mD/ActivityThread( 4666): setTargetHeapUtilization:0.75
^[[0m^[[38;5;75mD/ActivityThread( 4666): setTargetHeapMinFree:2097152

次の結果を提供します。

[COLOR="DarkSlateGray"]V/Zygote  ( 4666): Switching descriptor 55 to /dev/null
V/Zygote  ( 4666): Switching descriptor 9 to /dev/null[/COLOR]
[COLOR="Green"]I/ggheart ( [COLOR="Blue"]I/ggheart 1111[/COLOR]): onStop
[COLOR="Green"]I/Test    ( 1111): onStop
[COLOR="Blue"]D/ActivityThread( 4666): handleBindApplication:com.mxtech.videoplayer.ad
D/ActivityThread( 4666): setTargetHeapUtilization:0.75
D/ActivityThread( 4666): setTargetHeapMinFree:2097152[/COLOR]

答え2

簡単に聞こえますperl

perl -0777 -pe '
  BEGIN{
    %c = (
      "38;5;40" => "Green",
      "38;5;196" => "Red",
      "38;5;75" => "Blue",
      "38;5;166" => "Sienna",
      "38;5;231" => "DarkSlateGray",
      "38;5;40" => "Green",
      "38;5;196" => "Red",
      "38;5;75" => "Blue",
      "38;5;166" => "Sienna",
      "38;5;231" => "DarkSlateGray",
      "0" => "Black");
    $esc = qr{\e\[([\d;]*)m};
  }

  s{$esc(.*?)(?=$|$esc)}{
    $ret = $2;
    if ($2 ne "" && $1 ne $last) {
      $ret = (defined($last) && "[/COLOR]") . "[COLOR=\"$c{$1}\"]$2";
      $last = $1
    }
    $ret
  }gse;

  s{$}{[/COLOR]} if $last'

(ここでは怠惰なアプローチをとり、ファイル全体をメモリにロードします。

関連情報