Как реализовать как позиционные аргументы, так и опции флагов в командной строке python3 с помощью argparse? - PullRequest
0 голосов
/ 30 апреля 2020

Я хочу реализовать командную строку python3 с помощью argparse, которая поддерживает позиционные аргументы и параметры флагов.

Например:

usage: mycmd [-v] [-h] [-o] text

mycmd text       print text
mycmd -h         show help message and exit
mycmd -v         show version info
mycmd -o text    print text and dump text to file mycmd.txt

Вот моя реализация:

import argparse

class MyCmd:
  def __init__(self):
    self.parser = argparse.ArgumentParser()
    self.parser.add_argument('text', help='print text')  # mycmd text
    self.parser.add_argument('-v', action='store_true', dest='version', help='show version info') # mycmd -v
    self.parser.add_argument('-o', dest='output', help='print text and dump text to file mycmd.txt')
  def parse(self):
    return self.parser.parse_args()
  def text(self, text):
    print(text)
  def version(self):
    print('version info: mycmd-0.0.1')
  def output(self, text):
    print(text)
    fp = open('mycmd.txt', 'w')
    fp.write(text + '\n')
    fp.close()

if __name__=='__main__':
  mycmd = MyCmd()
  (options, args) = mycmd.parse()
  if options.text is not None:
    mycmd.text(text)
  elif options.version is not None:
    mycmd.version()
  elif options.output is not None:
    mycmd.output(options.output)

Когда я проверяю это с помощью:

$ ./mycmd -v

Дает мне ошибку:

usage: mycmd [-h] [-o OUTPUT]
            [-v]
            text
mycmd: error: the following arguments are required: text

Почему mycmd не может потреблять mycmd -v?

1 Ответ

1 голос
/ 30 апреля 2020

В исходной реализации отсутствует шаг включения c подпоследовательности для обработки текста потока.

Вот более упрощенная версия (без класса):

import argparse


def version():
    return 'v0.0.1'

def print_and_save(text, file_path):
    print(text)
    if file_path:
        with open(file_path, 'w') as fp:
            fp.write(text + '\n')
            fp.close()


if __name__=='__main__':
    # Root (rt)
    root_parser = argparse.ArgumentParser()
    root_parser.add_argument('-v', '--version', action='store_true', help='show version info')

    root_sbparsers = root_parser.add_subparsers(help='sub-command help')

    # Adding sub-commands
    text_parser = root_sbparsers.add_parser('process', help='print text')
    text_parser.add_argument('text', type=str, help='text you want to work with')
    text_parser.add_argument('-o', type=str, default=None, help='print text and dump text to file mycmd.txt')

    args = root_parser.parse_args()

    if args.version:
        print(version())
    elif args.text:
        print_and_save(args.text, args.o)

Я не уверен, почему вы должны встраивать его в целый класс, но вот версия с ним:

import argparse

class MyCmd(object):
    def __init__(self):
        # Root (rt)
        self.parser = argparse.ArgumentParser()
        self.parser.add_argument('-v', '--version', action='store_true', help='show version info')

        root_sbparsers = self.parser.add_subparsers(help='sub-command help')

        # Adding sub-commands
        text_parser = root_sbparsers.add_parser('process', help='print text')
        text_parser.add_argument('text', type=str, help='text you want to work with')
        text_parser.add_argument('-o', type=str, default=None, help='print text and dump text to file mycmd.txt')

    def parse(self):
        return self.parser.parse_args()

    def text(self, text):
        print(text)

    def version(self):
        print('version info: mycmd-0.0.1')

    def output(self, text, file_path):
        with open(file_path, 'w') as fp:
            fp.write(text + '\n')
            fp.close()


if __name__=='__main__':
    mycmd = MyCmd()
    args = mycmd.parse()

    if args.version:
        mycmd.version()
    elif args.text:
        mycmd.text(args.text)
        if args.o:
            mycmd.output(args.text, args.o)
...