javascript - 从配置 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"
]
如何修改我的函数以实现所需的输出?如果有一个图书馆可以处理这种情况,我找不到它。
解决方案
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)
推荐阅读
- python - 使用 XMLRPC 创建带有标签的 WordPress 帖子
- r - 带有变量 magrittr 的管道
- javascript - 为什么 Array.prototype.includes.bind 行为异常?
- c# - 分号后不包含逗号的匹配值
- mysql - 哪种方法更好?
- javascript - 我的 javascript 文件未执行,其中包括表单刷新
- jmeter-5.0 - “如何使用 VPN 服务器在 Jmeter 中录制脚本”
- python - 无法在 Windows 上安装 psutil
- vb.net - 在水晶报表中创建子报表总计和总计
- reactjs - syncfusion 输入字段未连接到状态