3

我要运行的 shell 命令,它什么都不返回:

echo helloWorld | grep 'dummy'

铅版:

以下线路工作:

out=(echo["helloWorld"] | grep["h"])().strip()

但是下一行没有,可能是什么原因?

out=(echo["helloWorld"] | grep["dummy"])().strip()
print(out)

我遇到的错误:

Traceback (most recent call last):
  File "dd.py", line 6, in <module>
    out=(echo["helloWorld"] | grep["dummy"])().strip()
  File "/home/user/venv/lib/python3.5/site-packages/plumbum/commands/base.py", line 103, in __call__
    return self.run(args, **kwargs)[1]
  File "/home/user/venv/lib/python3.5/site-packages/plumbum/commands/base.py", line 240, in run
    return p.run()
  File "/home/user/venv/lib/python3.5/site-packages/plumbum/commands/base.py", line 201, in runner
    return run_proc(p, retcode, timeout)
  File "/home/user/venv/lib/python3.5/site-packages/plumbum/commands/processes.py", line 232, in run_proc
    return _check_process(proc, retcode, timeout, stdout, stderr)
  File "/home/user/venv/lib/python3.5/site-packages/plumbum/commands/processes.py", line 23, in _check_process
    proc.verify(retcode, timeout, stdout, stderr)
  File "/home/user/venv/lib/python3.5/site-packages/plumbum/commands/base.py", line 412, in verify
    dstproc_verify(retcode, timeout, stdout, stderr)
  File "/home/user/venv/lib/python3.5/site-packages/plumbum/machines/base.py", line 26, in verify
    stderr)
plumbum.commands.processes.ProcessExecutionError: Command line: ['/bin/grep', 'dummy']
Exit code: 1

[Q]我该如何解决这个错误?

4

2 回答 2

2

发生这种情况是因为 grep 的退出状态是 1 如果它没有找到任何东西,如其手册中所述

如果您愿意,可以在命令行中尝试:

echo helloWorld | grep h; echo $?
echo helloWorld | grep x; echo $?

会导致

helloWorld
0
1

避免这种情况的方法在另一个不错的答案中进行了描述,例如

echo helloWorld | grep x | cat

将产生 0 作为状态。但不幸的是,plumbum 做了一个本地管道机制,所以 grep 输出到 plumbum,然后 plumbum 将它的管道传送到下一个命令——这意味着猫不能吞下退出代码 1,在它之前会抛出一个异常。

所以我的两个想法是要么创建一个 shell 脚本来运行一个 grep,在没有结果的情况下搜索永远不会返回错误:

#!/bin/bash
grep "$@" || test $? = 1

并执行它而不是 grep(在原始答案中称为 c1grep),或者在管道代码周围添加一个 try/except 块并手动处理退出代码 1(ProcessExecutionError)。

于 2019-07-23T09:07:37.050 回答
2

另一个答案是正确的,但不是最优的 - 问题可以在铅中解决,不需要包装的外部脚本grep,也不必捕获异常。

这个想法是使用铅的run方法来做两件事:

  • 强制铅锤接受来自被调用程序的任何退出代码(默认情况下,如果返回的错误代码不为零,则会引发异常)。
  • 捕获一个包含错误代码、标准输出和被调用程序的标准错误的元组,而不仅仅是像通常情况下那样的标准输出。

工作示例:

for s in ["hello","dummy"]:
    exitcode,stdout,stderr = (echo["helloWorld"] | grep[s]).run (retcode=None)

    if exitcode == 0:
        print (stdout.strip())

    if exitcode != 0:
        print (f"string {s} not present")
        print (stderr)

前面的代码返回:

helloWorld
string dummy not present

不引发任何异常。

于 2021-10-30T23:01:27.533 回答