UNPKG

@cdktf/hcl2json

Version:
177 lines • 21.7 kB
"use strict"; // Copyright (c) HashiCorp, Inc // SPDX-License-Identifier: MPL-2.0 // eslint-disable-next-line @typescript-eslint/triple-slash-reference /// <reference lib="dom" /> var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.getExpressionAst = exports.getReferencesInExpression = exports.convertFiles = exports.parse = void 0; // Inspired by // https://github.com/ts-terraform/ts-terraform // https://github.com/aaronpowell/webpack-golang-wasm-async-loader const fs = __importStar(require("fs-extra")); const path_1 = require("path"); const deepmerge_1 = require("./deepmerge"); const zlib_1 = require("zlib"); const references_1 = require("./references"); const util_1 = require("./util"); // eslint-disable-next-line @typescript-eslint/no-unsafe-function-type const jsRoot = {}; function sleep() { return new Promise((resolve) => { setTimeout(resolve, 0); }); } function goBridge(getBytes) { let ready = false; async function init() { // After: https://github.com/golang/go/commit/680caf15355057ca84857a2a291b6f5c44e73329 // Go 1.19+ has a different entrypoint await Promise.resolve().then(() => __importStar(require(`../wasm/bridge_wasm_exec.js`))); const go = new global.Go(); const bytes = await getBytes; const result = await WebAssembly.instantiate(bytes, go.importObject); global.__parse_terraform_config_wasm__ = jsRoot; void go.run(result.instance); ready = true; } init().catch((error) => { throw error; }); const proxy = new Proxy({}, { get: (_, key) => { return async (...args) => { while (!ready) { await sleep(); } if (!(key in jsRoot)) { throw new Error(`There is nothing defined with the name "${key.toString()}"`); } if (typeof jsRoot[key] !== "function") { return jsRoot[key]; } return new Promise((resolve, reject) => { const cb = (err, ...msg) => // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore err ? reject(new Error(err)) : resolve(...msg); const run = () => { jsRoot[key].apply(undefined, [...args, cb]); }; run(); }); }; }, }); return proxy; } const loadWasm = async () => { return (0, zlib_1.gunzipSync)(await fs.readFile((0, path_1.join)(__dirname, "..", "main.wasm.gz"))); }; const wasm = goBridge(loadWasm()); async function parse(filename, contents) { const res = await wasm.parse(filename, contents); return JSON.parse(res); } exports.parse = parse; async function convertFiles(workingDirectory) { let tfFileContents = ""; const tfJSONFileContents = []; for (const file of fs.readdirSync(workingDirectory)) { const filePath = (0, path_1.resolve)(workingDirectory, file); if (!fs.lstatSync(filePath).isDirectory()) { if (file.match(/\.tf$/)) { tfFileContents += fs.readFileSync(filePath, "utf-8"); tfFileContents += "\n"; } else if (file.match(/\.tf\.json$/)) { tfJSONFileContents.push(JSON.parse(fs.readFileSync(filePath, "utf-8"))); } } } if (tfFileContents === "" && tfJSONFileContents.length === 0) { console.error(`No '.tf' or '.tf.json' files found in ${workingDirectory}`); return {}; } return (0, deepmerge_1.deepMerge)(await parse("hcl2json.tf", tfFileContents), ...tfJSONFileContents); } exports.convertFiles = convertFiles; /** * Parse a Terraform expression and return the AST. This function expects a string input, and will wrap the expression in quotes if it is not already quoted. * @param filename The filename to use for the expression. This is used for error reporting. * @param expression The expression to parse. * @returns An array of References found in the expression. */ async function getReferencesInExpression(filename, expression) { // We have to do this twice because of the problem with HEREDOCS // Our current hcl2json implementation removes HEREDOCS and replaces them // with a multi-line string, which is causing all kinds of problems let offset = 0; let quoteWrappedExpression = expression; if (!expression.startsWith('"')) { quoteWrappedExpression = `"${expression}"`; offset = 1; } const { wrap: wrappedExpression, wrapOffset: startOffset } = (0, util_1.wrapTerraformExpression)(`${quoteWrappedExpression}`); offset += startOffset; const ast = await getExpressionAst(filename, wrappedExpression); if (!ast) { return []; } const refs = (0, references_1.findAllReferencesInAst)(expression, ast); if (wrappedExpression === expression) { return refs; } return refs.map((ref) => { return { ...ref, startPosition: ref.startPosition - offset, endPosition: ref.endPosition - offset, }; }); } exports.getReferencesInExpression = getReferencesInExpression; /** * Parse a Terraform expression and return the AST. The expression does not need to be a Terraform string. * @param filename The filename to use for the expression. This is used for error reporting. * @param expression The expression to parse. * @returns The AST for the expression. * * The returned AST has the following structure: * - type: The type of the node. This is a string. * - range: This contains the start and end of the node in the expression. * - children: This contains the children of the node. This is an array of nodes. * - meta: This contains metadata about the node. This is an object, and varies depending on the type of the node. */ async function getExpressionAst(filename, expression) { const res = await wasm.getExpressionAst(filename, expression); const ast = JSON.parse(res); if (!ast) { return null; } return ast; } exports.getExpressionAst = getExpressionAst; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bridge.js","sourceRoot":"","sources":["bridge.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright (c) HashiCorp, Inc\n// SPDX-License-Identifier: MPL-2.0\n// eslint-disable-next-line @typescript-eslint/triple-slash-reference\n/// <reference lib=\"dom\" />\n\n// Inspired by\n// https://github.com/ts-terraform/ts-terraform\n// https://github.com/aaronpowell/webpack-golang-wasm-async-loader\n\nimport * as fs from \"fs-extra\";\nimport { join, resolve } from \"path\";\nimport { deepMerge } from \"./deepmerge\";\nimport { gunzipSync } from \"zlib\";\nimport { Reference, findAllReferencesInAst } from \"./references\";\nimport { ExpressionType } from \"./syntax-tree\";\nimport { wrapTerraformExpression } from \"./util\";\n\ninterface GoBridge {\n  parse: (filename: string, hcl: string) => Promise<string>;\n  parseExpression: (filename: string, hcl: string) => Promise<string>;\n  getExpressionAst: (filename: string, hcl: string) => Promise<string>;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-unsafe-function-type\nconst jsRoot: Record<string, Function> = {};\n\nfunction sleep() {\n  return new Promise((resolve) => {\n    setTimeout(resolve, 0);\n  });\n}\n\nfunction goBridge(getBytes: Promise<Buffer>) {\n  let ready = false;\n\n  async function init() {\n    // After: https://github.com/golang/go/commit/680caf15355057ca84857a2a291b6f5c44e73329\n    // Go 1.19+ has a different entrypoint\n    await import(`../wasm/bridge_wasm_exec.js`);\n    const go = new (global as any).Go();\n    const bytes = await getBytes;\n    const result = await WebAssembly.instantiate(bytes, go.importObject);\n    (global as any).__parse_terraform_config_wasm__ = jsRoot;\n    void go.run(result.instance);\n    ready = true;\n  }\n\n  init().catch((error) => {\n    throw error;\n  });\n\n  const proxy = new Proxy({} as GoBridge, {\n    get: (_, key: string) => {\n      return async (...args: unknown[]) => {\n        while (!ready) {\n          await sleep();\n        }\n\n        if (!(key in jsRoot)) {\n          throw new Error(\n            `There is nothing defined with the name \"${key.toString()}\"`,\n          );\n        }\n\n        if (typeof jsRoot[key] !== \"function\") {\n          return jsRoot[key];\n        }\n\n        return new Promise((resolve, reject) => {\n          const cb = (err: string, ...msg: string[]) =>\n            // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n            // @ts-ignore\n            err ? reject(new Error(err)) : resolve(...msg);\n\n          const run = () => {\n            jsRoot[key].apply(undefined, [...args, cb]);\n          };\n\n          run();\n        });\n      };\n    },\n  });\n\n  return proxy;\n}\n\nconst loadWasm = async () => {\n  return gunzipSync(await fs.readFile(join(__dirname, \"..\", \"main.wasm.gz\")));\n};\n\nconst wasm = goBridge(loadWasm());\n\nexport async function parse(\n  filename: string,\n  contents: string,\n): Promise<Record<string, any>> {\n  const res = await wasm.parse(filename, contents);\n  return JSON.parse(res);\n}\n\nexport async function convertFiles(\n  workingDirectory: string,\n): Promise<Record<string, any>> {\n  let tfFileContents = \"\";\n  const tfJSONFileContents: Record<string, any>[] = [];\n\n  for (const file of fs.readdirSync(workingDirectory)) {\n    const filePath = resolve(workingDirectory, file);\n    if (!fs.lstatSync(filePath).isDirectory()) {\n      if (file.match(/\\.tf$/)) {\n        tfFileContents += fs.readFileSync(filePath, \"utf-8\");\n        tfFileContents += \"\\n\";\n      } else if (file.match(/\\.tf\\.json$/)) {\n        tfJSONFileContents.push(JSON.parse(fs.readFileSync(filePath, \"utf-8\")));\n      }\n    }\n  }\n\n  if (tfFileContents === \"\" && tfJSONFileContents.length === 0) {\n    console.error(`No '.tf' or '.tf.json' files found in ${workingDirectory}`);\n    return {};\n  }\n\n  return deepMerge(\n    await parse(\"hcl2json.tf\", tfFileContents),\n    ...tfJSONFileContents,\n  );\n}\n\n/**\n * Parse a Terraform expression and return the AST. This function expects a string input, and will wrap the expression in quotes if it is not already quoted.\n * @param filename The filename to use for the expression. This is used for error reporting.\n * @param expression The expression to parse.\n * @returns An array of References found in the expression.\n */\nexport async function getReferencesInExpression(\n  filename: string,\n  expression: string,\n): Promise<Reference[]> {\n  // We have to do this twice because of the problem with HEREDOCS\n  // Our current hcl2json implementation removes HEREDOCS and replaces them\n  // with a multi-line string, which is causing all kinds of problems\n  let offset = 0;\n  let quoteWrappedExpression = expression;\n  if (!expression.startsWith('\"')) {\n    quoteWrappedExpression = `\"${expression}\"`;\n    offset = 1;\n  }\n\n  const { wrap: wrappedExpression, wrapOffset: startOffset } =\n    wrapTerraformExpression(`${quoteWrappedExpression}`);\n\n  offset += startOffset;\n\n  const ast = await getExpressionAst(filename, wrappedExpression);\n  if (!ast) {\n    return [];\n  }\n\n  const refs = findAllReferencesInAst(expression, ast);\n  if (wrappedExpression === expression) {\n    return refs;\n  }\n\n  return refs.map((ref) => {\n    return {\n      ...ref,\n      startPosition: ref.startPosition - offset,\n      endPosition: ref.endPosition - offset,\n    };\n  });\n}\n\n/**\n * Parse a Terraform expression and return the AST. The expression does not need to be a Terraform string.\n * @param filename The filename to use for the expression. This is used for error reporting.\n * @param expression The expression to parse.\n * @returns The AST for the expression.\n *\n *   The returned AST has the following structure:\n *   - type: The type of the node. This is a string.\n *   - range: This contains the start and end of the node in the expression.\n *   - children: This contains the children of the node. This is an array of nodes.\n *   - meta: This contains metadata about the node. This is an object, and varies depending on the type of the node.\n */\nexport async function getExpressionAst(\n  filename: string,\n  expression: string,\n): Promise<ExpressionType | null> {\n  const res = await wasm.getExpressionAst(filename, expression);\n  const ast = JSON.parse(res) as ExpressionType;\n\n  if (!ast) {\n    return null;\n  }\n\n  return ast;\n}\n"]}