gpt4 book ai didi

python - 如何从 python 中的字符串中删除 ANSI 转义序列

转载 作者:IT老高 更新时间:2023-10-28 20:27:55 30 4
gpt4 key购买 nike

这是一个包含我的字符串的片段。
'ls\r\n\x1b[00m\x1b[01;31mexamplefile.zip\x1b[00m\r\n\x1b[01;31m'

字符串是从我执行的 SSH 命令返回的。我不能使用当前状态的字符串,因为它包含 ANSI 标准化的转义序列。如何以编程方式删除转义序列,以便字符串的唯一部分是 'examplefile.zip'

最佳答案

用正则表达式删除它们:

import re

# 7-bit C1 ANSI sequences
ansi_escape = re.compile(r'''
\x1B # ESC
(?: # 7-bit C1 Fe (except CSI)
[@-Z\\-_]
| # or [ for CSI, followed by a control sequence
\[
[0-?]* # Parameter bytes
[ -/]* # Intermediate bytes
[@-~] # Final byte
)
''', re.VERBOSE)
result = ansi_escape.sub('', sometext)

或者,不带 VERBOSE 标志,采用压缩形式:

ansi_escape = re.compile(r'\x1B(?:[@-Z\\-_]|\[[0-?]*[ -/]*[@-~])')
result = ansi_escape.sub('', sometext)

演示:

>>> import re
>>> ansi_escape = re.compile(r'\x1B(?:[@-Z\\-_]|\[[0-?]*[ -/]*[@-~])')
>>> sometext = 'ls\r\n\x1b[00m\x1b[01;31mexamplefile.zip\x1b[00m\r\n\x1b[01;31m'
>>> ansi_escape.sub('', sometext)
'ls\r\nexamplefile.zip\r\n'

上述正则表达式涵盖了所有 7 位 ANSI C1 转义序列,但涵盖了 8 位 C1 转义序列开场符。后者在今天的 UTF-8 世界中从未使用过,相同的字节范围具有不同的含义。

如果您也确实需要覆盖 8 位代码(然后可能会使用 bytes 值),那么正则表达式就会变成这样的字节模式:

# 7-bit and 8-bit C1 ANSI sequences
ansi_escape_8bit = re.compile(br'''
(?: # either 7-bit C1, two bytes, ESC Fe (omitting CSI)
\x1B
[@-Z\\-_]
| # or a single 8-bit byte Fe (omitting CSI)
[\x80-\x9A\x9C-\x9F]
| # or CSI + control codes
(?: # 7-bit CSI, ESC [
\x1B\[
| # 8-bit CSI, 9B
\x9B
)
[0-?]* # Parameter bytes
[ -/]* # Intermediate bytes
[@-~] # Final byte
)
''', re.VERBOSE)
result = ansi_escape_8bit.sub(b'', somebytesvalue)

可以压缩成

# 7-bit and 8-bit C1 ANSI sequences
ansi_escape_8bit = re.compile(
br'(?:\x1B[@-Z\\-_]|[\x80-\x9A\x9C-\x9F]|(?:\x1B\[|\x9B)[0-?]*[ -/]*[@-~])'
)
result = ansi_escape_8bit.sub(b'', somebytesvalue)

有关详细信息,请参阅:

您给出的示例包含 4 个 CSI(控制序列引入器)代码,由 \x1B[ESC [ 开头字节标记,并且每个都包含一个 SGR(选择图形再现)代码,因为它们每个都以 m 结尾。这些参数之间的参数(由 ; 分号分隔)告诉您的终端要使用哪些图形再现属性。所以对于每个 \x1B[....m 序列,使用的 3 个代码是:

  • 0(或本例中的00):重置,禁用所有属性
  • 1(或示例中的 01):粗体
  • 31:红色(前景)

但是,ANSI 不仅仅是 CSI SGR 代码。单独使用 CSI,您还可以控制光标、清除线条或整个显示或滚动(当然前提是终端支持)。除了 CSI,还有一些代码可以选择替代字体(SS2SS3),发送“私有(private)消息”(想想密码),与终端通信(DCS)、操作系统(OSC)或应用程序本身(APC,一种应用程序将自定义控制代码捎带到通信中的方式流),以及帮助定义字符串(SOS、字符串开始、ST 字符串终止符)或将所有内容重置回基本状态(RIS)。上面的正则表达式涵盖了所有这些。

请注意,上述正则表达式仅删除了 ANSI C1 代码,但并未删除这些代码可能标记的任何其他数据(例如在 OSC 开启程序和终止 ST 代码之间发送的字符串)。删除这些将需要超出此答案范围的额外工作。

关于python - 如何从 python 中的字符串中删除 ANSI 转义序列,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14693701/

30 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com