首页 > 解决方案 > 从配置 txt 文件中解析命令行参数

问题描述

我正在尝试创建一个函数来解析文本文件中的命令行参数。这意味着每个标志和值都需要作为一个数组中的单独项目返回。#如果行为空或以,;或开头,则应忽略它们]

我当前的功能存在多个问题。首先,在 reduce 函数中拆分数组不会像使用 push 那样将数组添加到累加器中,而是将新数组添加到累加器中。其次,引号内的字符串可以拆分为数组,即使它们应该被视为单个参数。

const argsFile = `
# Command line arguments
--download example.com
--pass

--no-fail
--output "C:\\Users\\User\\Desktop\\New Folder"

--binary-location 'C:\\Users\\Desktop\\New Folder\\executable program.exe'
`;

let parsedArguments = argsFile.split(/\r?\n/)
    .filter(argument => (!argument.startsWith('#') && !argument.startsWith(';') && !argument.startsWith(']')))
    .reduce((a, c) => [...a, c.split(' ')])
    .filter(argument => argument !== '');
    
console.dir(parsedArguments)

这是我的功能所需的输出:

[
    "--download",
    "example.com",
    "--pass",
    "--no-fail",
    "--output",
    "C:\\Users\\User\\Desktop\\New Folder",
    "--binary-location",
    "C:\\Users\\Desktop\\New Folder\\executable program.exe"
]

如何修改我的函数以实现所需的输出?如果有一个图书馆可以处理这种情况,我找不到它。

标签: javascriptnode.js

解决方案


Yargs 似乎非常健壮地解析字符串的参数,而且它是相当可配置的。

我想出了以下内容,这似乎产生了您想要的输出。但是,没有使用任何其他字符串对其进行测试:

const parse = require("yargs-parser");

const argsFile = `
# Command line arguments
--download example.com
--pass

--no-fail
--output "C:\\Users\\User\\Desktop\\New Folder"

--binary-location 'C:\\Users\\Desktop\\New Folder\\executable program.exe'
`;

let parsedArguments = argsFile
  .split(/\r?\n/)
  .filter(
    (argument) =>
      !argument.startsWith("#") &&
      !argument.startsWith(";") &&
      !argument.startsWith("]")
  )
  .map((line) => {
    return parse(line, {
      string: true,
      configuration: {
        "boolean-negation": false,
        "camel-case-expansion": false,
      },
    });
  })
  .map((ar) => {
    delete ar._;
    let properties = Object.keys(ar);
    if (properties.length == 0) return [];
    return [
      "--" + properties[0],
      typeof ar[properties[0]] == "boolean" ? "" : ar[properties[0]],
    ];
  })
  .filter((argument) => argument.length != 0);

let flatArgs = [].concat.apply([], parsedArguments).filter((i) => i != "");

console.dir(flatArgs);

产生以下结果:

[ '--download',
  'example.com',
  '--pass',
  '--no-fail',
  '--output',
  'C:\\Users\\User\\Desktop\\New Folder',
  '--binary-location',
  'C:\\Users\\Desktop\\New Folder\\executable program.exe' ]

Yargs 解析器会针对您的特定要求“过于激进”地解析字符串,这就是为什么我们必须通过稍微反转解析器所做的映射来进行映射(以'--'开头,忽略布尔值等)。然后,最后,我们必须将数组“展平”,因为每一行都被解析为自己的数组。

编辑:所以,如果我们还要处理短参数,yargs 将不适合,因为我们在解析后无法访问原始字符串。我们可以使用 yarg 的内部函数来标记字符串,但是(我只需要将它转换为 js):

function tokenizeArgString(argString) {
    if (Array.isArray(argString)) {
        return argString.map(e => typeof e !== 'string' ? e + '' : e);
    }
    argString = argString.trim();
    let i = 0;
    let prevC = null;
    let c = null;
    let opening = null;
    const args = [];
    for (let ii = 0; ii < argString.length; ii++) {
        prevC = c;
        c = argString.charAt(ii);
        // split on spaces unless we're in quotes.
        if (c === ' ' && !opening) {
            if (!(prevC === ' ')) {
                i++;
            }
            continue;
        }
        // don't split the string if we're in matching
        // opening or closing single and double quotes.
        if (c === opening) {
            opening = null;
        }
        else if ((c === "'" || c === '"') && !opening) {
            opening = c;
        }
        if (!args[i])
            args[i] = '';
        args[i] += c;
    }
    return args;
}


const argsFile = `
# Command line arguments
--download example.com
--pass

--no-fail
--output "C:\\Users\\User\\Desktop\\New Folder"

-a test
--binary-location 'C:\\Users\\Desktop\\New Folder\\executable program.exe'
`;

let parsedArguments = argsFile.split(/\r?\n/)
    .filter(argument => (!argument.startsWith('#') && !argument.startsWith(';') && !argument.startsWith(']')))
    .map(line => tokenizeArgString(line))
    .filter(argument => argument.length != 0);

let flatArgsNoQuotes = [].concat.apply([], parsedArguments).map(args => args.replace(/['"]+/g, '')).filter(i => i != "");
    
console.dir(flatArgsNoQuotes)

推荐阅读