为了账号安全,请及时绑定邮箱和手机立即绑定

从 config txt 文件解析命令行参数

从 config txt 文件解析命令行参数

慕姐4208626 2023-03-03 13:19:49
我正在尝试创建一个函数来解析文本文件中的命令行参数。这意味着每个标志和值都需要作为一个数组中的单独项返回。如果行为空或以#,;或开头,则应忽略这些行]。我当前的功能存在多个问题。首先,在 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"]如何修改我的函数以获得所需的输出?如果有一个图书馆可以处理这种情况,我还没有找到。
查看完整描述

1 回答

?
暮色呼如

TA贡献1853条经验 获得超9个赞

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)


查看完整回答
反对 回复 2023-03-03
  • 1 回答
  • 0 关注
  • 211 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号