从键/值对创建复杂对象

Ved*_*ic. 6 javascript node.js docker

我有一个具有键/值对的文件.此文件在process.envvia Docker中加载.但是出于开发目的,我手动加载它,所以最后它们是相同的;

配置:

process.env['ccc.logger.winston.level']='info';
process.env['ccc.logger.winston.transports.type.file']='File';
process.env['ccc.logger.winston.transports.filename']='logs/testOne.log';
process.env['ccc.logger.winston.transports.rotate']='false';

process.env['ccc.logger.winston.transports.type.file']='File';
process.env['ccc.logger.winston.transports.filename']='logs/testTwo.log';
process.env['ccc.logger.winston.transports.rotate']='true';
Run Code Online (Sandbox Code Playgroud)

我的期望是拥有这个目标:

{
  "ccc": {
    "logger": {
      "winston": {
        "level": "info",
        "transports": [
          {
            "type": "File",
            "filename": "logs/testONE.log",
            "rotate": true
          },
          {
            "type": "File",
            "filename": "logs/testTWO.log",
            "rotate": false
          }
        ]
      }
    }
  }
}
Run Code Online (Sandbox Code Playgroud)

我已经提出了可行的解决方案,但如果我有一个对象数组,我遇到了问题,就像上面的例子一样:

循环所有键/值的函数和调用函数来创建对象:

let ENV_FROM_DOCKER = process.env;

for (let property in ENV_FROM_DOCKER) {
  let checkForShallowProperties = property.split(".")[1];

  if (typeof checkForShallowProperties === 'undefined') {
    continue;
  }

  let resultObject = this.expand(property, ENV_FROM_DOCKER[property]););

  emptyConfig = merge(emptyConfig, resultObject);
  let stop;
}
Run Code Online (Sandbox Code Playgroud)

对象创建功能:

expand(str, value) {
  let items = str.split(".") // split on dot notation
  let output = {} // prepare an empty object, to fill later
  let ref = output // keep a reference of the new object

  //  loop through all nodes, except the last one
  for (let i = 0; i < items.length - 1; i++) {
    ref[items[i]] = {}; // create a new element inside the reference
    ref = ref[items[i]]; // shift the reference to the newly created object
  }

  ref[items[items.length - 1]] = value; // apply the final value
  return output // return the full object
}
Run Code Online (Sandbox Code Playgroud)

这个设置工作正常,但如果我有一个对象数组的对象(如上面的例子),它不能正常工作.这是现在的输出:

{
  "ccc": {
    "logger": {
      "winston": {
        "level": "info",
        "transports": {
          "type": {
            "file": "File"
          },
          "filename": "logs/testTwo.log",
          "rotate": "true"
        }
      }
    }
  }
}
Run Code Online (Sandbox Code Playgroud)

我试图让这段代码现在工作几个小时,但只是在圈子里旋转.该ccc对象是一个例子.键/值列表中还有其他对象也可能包含数组.

Chr*_*e55 1

为每个传输分配一个索引

创建环境变量时,您可以将每个变量分配transports.whatnot给数组transports[0].whatnot和中的索引transports[1].whatnot。为了使这项工作正常进行,我们必须像这样解析它:

const ENV = {
  'ccc.logger.winston.level': 'info',
  'ccc.logger.winston.transports[0].type': 'File',
  'ccc.logger.winston.transports[0].filename': 'logs/testOne.log',
  'ccc.logger.winston.transports[0].rotate': 'false',
  'ccc.logger.winston.transports[1].type': 'File',
  'ccc.logger.winston.transports[1].filename': 'logs/testTwo.log',
  'ccc.logger.winston.transports[1].rotate': 'true'
}

for (let property in ENV) {
  let checkForShallowProperties = property.split('.')[1]; 

  if (typeof checkForShallowProperties === 'undefined') {
    continue;
  }

  let resultObject = expand(property, ENV[property])
  console.log(resultObject)
}

function expand(string, value) {
  const items = string.split('.').map(name => {
    const match = name.match(/\[\d+?\]/)
    return {
      name: match ? name.slice(0, match.index) : name,
      usesBrackets: !!match,
      key: match && match[0].slice(1, -1)
    }
  })
  
  const output = {}
  let ref = output
  let parent
  
  for (const item of items) {
    ref[item.name] = {}
    parent = ref
    ref = ref[item.name]
    if (item.usesBrackets) {
      ref[item.key] = {}
      ref = ref[item.key]
    }
  }
  parent[items[items.length - 1].name] = value
  return output
}
Run Code Online (Sandbox Code Playgroud)

这里的输出是 PRE-MERGE

正如您所看到的,它的工作方式是将对象视为自己的数组并将内容放置在索引甚至其他访问器处。

然而,将所有这些转移到文件.json某些内容管理系统很可能符合您的最佳利益,因为这是一种非常不稳定的处理方式。如果您的需求发生变化,您可能需要重写它,使用 JSON 您只需加载 JSON