python argparse文件扩展名检查

bph*_*bph 3 python argparse

argparse用于验证文件名cmd行参数的文件扩展名?

例如,如果我有一个python脚本,我从cmd行运行:

$ script.py file.csv
$ script.py file.tab
$ script.py file.txt
Run Code Online (Sandbox Code Playgroud)

我希望argparse接受前两个文件名cmd行选项,但拒绝第三个

我知道你可以这样做:

parser = argparse.ArgumentParser()
parser.add_argument("fn", choices=["csv","tab"])
args = parser.parse_args()
Run Code Online (Sandbox Code Playgroud)

为cmd行选项指定两个有效选项

我想要的是这个:

parser.add_argument("fn", choices=["*.csv","*.tab"])
Run Code Online (Sandbox Code Playgroud)

为cmd行选项指定两个有效的文件扩展名.不幸的是,这不起作用 - 是否有办法实现这一目的argparse

mgi*_*son 8

当然 - 你只需要指定一个合适的函数作为type.

import argparse
import os.path

parser = argparse.ArgumentParser()

def file_choices(choices,fname):
    ext = os.path.splitext(fname)[1][1:]
    if ext not in choices:
       parser.error("file doesn't end with one of {}".format(choices))
    return fname

parser.add_argument('fn',type=lambda s:file_choices(("csv","tab"),s))

parser.parse_args()
Run Code Online (Sandbox Code Playgroud)

演示:

temp $ python test.py test.csv
temp $ python test.py test.foo
usage: test.py [-h] fn
test.py: error: file doesn't end with one of ('csv', 'tab')
Run Code Online (Sandbox Code Playgroud)

这可能是更干净/更通用的方法:

import argparse
import os.path

def CheckExt(choices):
    class Act(argparse.Action):
        def __call__(self,parser,namespace,fname,option_string=None):
            ext = os.path.splitext(fname)[1][1:]
            if ext not in choices:
                option_string = '({})'.format(option_string) if option_string else ''
                parser.error("file doesn't end with one of {}{}".format(choices,option_string))
            else:
                setattr(namespace,self.dest,fname)

    return Act

parser = argparse.ArgumentParser()
parser.add_argument('fn',action=CheckExt({'csv','txt'}))

print parser.parse_args()
Run Code Online (Sandbox Code Playgroud)

这里的缺点是代码在某些方面变得有点复杂 - 结果是当你真正去格式化你的参数时,界面变得更加清晰.


Jon*_*nts 5

定义一个将名称作为字符串的自定义函数-将扩展名拆分以进行比较,如果可以,则仅返回字符串,否则引发argparse期望的异常:

def valid_file(param):
    base, ext = os.path.splitext(param)
    if ext.lower() not in ('.csv', '.tab'):
        raise argparse.ArgumentTypeError('File must have a csv or tab extension')
    return param
Run Code Online (Sandbox Code Playgroud)

然后使用该函数,例如:

parser = argparse.ArgumentParser()
parser.add_argument('filename', type=valid_file)
Run Code Online (Sandbox Code Playgroud)