220 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			220 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
const definitions = require("../src/definitions");
 | 
						|
const flatMap = require("array.prototype.flatmap");
 | 
						|
const {
 | 
						|
  typeSignature,
 | 
						|
  iterateProps,
 | 
						|
  mapProps,
 | 
						|
  filterProps,
 | 
						|
  unique,
 | 
						|
} = require("./util");
 | 
						|
 | 
						|
const stdout = process.stdout;
 | 
						|
 | 
						|
const jsTypes = ["string", "number", "boolean"];
 | 
						|
 | 
						|
const quote = (value) => `"${value}"`;
 | 
						|
 | 
						|
function params(fields) {
 | 
						|
  const optionalDefault = (field) =>
 | 
						|
    field.default ? ` = ${field.default}` : "";
 | 
						|
  return mapProps(fields)
 | 
						|
    .map((field) => `${typeSignature(field)}${optionalDefault(field)}`)
 | 
						|
    .join(",");
 | 
						|
}
 | 
						|
 | 
						|
function assertParamType({ assertNodeType, array, name, type }) {
 | 
						|
  if (array) {
 | 
						|
    // TODO - assert contents of array?
 | 
						|
    return `assert(typeof ${name} === "object" && typeof ${name}.length !== "undefined")\n`;
 | 
						|
  } else {
 | 
						|
    if (jsTypes.includes(type)) {
 | 
						|
      return `assert(
 | 
						|
          typeof ${name} === "${type}",
 | 
						|
          "Argument ${name} must be of type ${type}, given: " + typeof ${name}
 | 
						|
      )`;
 | 
						|
    }
 | 
						|
 | 
						|
    if (assertNodeType === true) {
 | 
						|
      return `assert(
 | 
						|
        ${name}.type === "${type}",
 | 
						|
        "Argument ${name} must be of type ${type}, given: " + ${name}.type
 | 
						|
      )`;
 | 
						|
    }
 | 
						|
 | 
						|
    return "";
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
function assertParam(meta) {
 | 
						|
  const paramAssertion = assertParamType(meta);
 | 
						|
 | 
						|
  if (paramAssertion === "") {
 | 
						|
    return "";
 | 
						|
  }
 | 
						|
 | 
						|
  if (meta.maybe || meta.optional) {
 | 
						|
    return `
 | 
						|
      if (${meta.name} !== null && ${meta.name} !== undefined) {
 | 
						|
        ${paramAssertion};
 | 
						|
      }
 | 
						|
    `;
 | 
						|
  } else {
 | 
						|
    return paramAssertion;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
function assertParams(fields) {
 | 
						|
  return mapProps(fields).map(assertParam).join("\n");
 | 
						|
}
 | 
						|
 | 
						|
function buildObject(typeDef) {
 | 
						|
  const optionalField = (meta) => {
 | 
						|
    if (meta.array) {
 | 
						|
      // omit optional array properties if the constructor function was supplied
 | 
						|
      // with an empty array
 | 
						|
      return `
 | 
						|
        if (typeof ${meta.name} !== "undefined" && ${meta.name}.length > 0) {
 | 
						|
          node.${meta.name} = ${meta.name};
 | 
						|
        }
 | 
						|
      `;
 | 
						|
    } else if (meta.type === "Object") {
 | 
						|
      // omit optional object properties if they have no keys
 | 
						|
      return `
 | 
						|
        if (typeof ${meta.name} !== "undefined" && Object.keys(${meta.name}).length !== 0) {
 | 
						|
          node.${meta.name} = ${meta.name};
 | 
						|
        }
 | 
						|
      `;
 | 
						|
    } else if (meta.type === "boolean") {
 | 
						|
      // omit optional boolean properties if they are not true
 | 
						|
      return `
 | 
						|
        if (${meta.name} === true) {
 | 
						|
          node.${meta.name} = true;
 | 
						|
        }
 | 
						|
      `;
 | 
						|
    } else {
 | 
						|
      return `
 | 
						|
        if (typeof ${meta.name} !== "undefined") {
 | 
						|
          node.${meta.name} = ${meta.name};
 | 
						|
        }
 | 
						|
      `;
 | 
						|
    }
 | 
						|
  };
 | 
						|
 | 
						|
  const fields = mapProps(typeDef.fields)
 | 
						|
    .filter((f) => !f.optional && !f.constant)
 | 
						|
    .map((f) => f.name);
 | 
						|
 | 
						|
  const constants = mapProps(typeDef.fields)
 | 
						|
    .filter((f) => f.constant)
 | 
						|
    .map((f) => `${f.name}: "${f.value}"`);
 | 
						|
 | 
						|
  return `
 | 
						|
    const node: ${typeDef.flowTypeName || typeDef.name} = {
 | 
						|
      type: "${typeDef.name}",
 | 
						|
      ${constants.concat(fields).join(",")}
 | 
						|
    }
 | 
						|
 | 
						|
    ${mapProps(typeDef.fields)
 | 
						|
      .filter((f) => f.optional)
 | 
						|
      .map(optionalField)
 | 
						|
      .join("")}
 | 
						|
  `;
 | 
						|
}
 | 
						|
 | 
						|
function lowerCamelCase(name) {
 | 
						|
  return name.substring(0, 1).toLowerCase() + name.substring(1);
 | 
						|
}
 | 
						|
 | 
						|
function generate() {
 | 
						|
  stdout.write(`
 | 
						|
    // @flow
 | 
						|
 | 
						|
    // THIS FILE IS AUTOGENERATED
 | 
						|
    // see scripts/generateNodeUtils.js
 | 
						|
 | 
						|
    import { assert } from "mamacro";
 | 
						|
 | 
						|
    function isTypeOf(t: string) {
 | 
						|
      return (n: Node) => n.type === t;
 | 
						|
    }
 | 
						|
 | 
						|
    function assertTypeOf(t: string) {
 | 
						|
      return (n: Node) => assert(n.type === t);
 | 
						|
    }
 | 
						|
  `);
 | 
						|
 | 
						|
  // Node builders
 | 
						|
  iterateProps(definitions, (typeDefinition) => {
 | 
						|
    stdout.write(`
 | 
						|
      export function ${lowerCamelCase(typeDefinition.name)} (
 | 
						|
        ${params(filterProps(typeDefinition.fields, (f) => !f.constant))}
 | 
						|
      ): ${typeDefinition.name} {
 | 
						|
 | 
						|
        ${assertParams(filterProps(typeDefinition.fields, (f) => !f.constant))}
 | 
						|
        ${buildObject(typeDefinition)} 
 | 
						|
 | 
						|
        return node;
 | 
						|
      }
 | 
						|
    `);
 | 
						|
  });
 | 
						|
 | 
						|
  // Node testers
 | 
						|
  iterateProps(definitions, (typeDefinition) => {
 | 
						|
    stdout.write(`
 | 
						|
      export const is${typeDefinition.name}: ((n: Node) => boolean) =
 | 
						|
        isTypeOf("${typeDefinition.name}");
 | 
						|
    `);
 | 
						|
  });
 | 
						|
 | 
						|
  // Node union type testers
 | 
						|
  const unionTypes = unique(
 | 
						|
    flatMap(
 | 
						|
      mapProps(definitions).filter((d) => d.unionType),
 | 
						|
      (d) => d.unionType
 | 
						|
    )
 | 
						|
  );
 | 
						|
  unionTypes.forEach((unionType) => {
 | 
						|
    stdout.write(
 | 
						|
      `
 | 
						|
      export const is${unionType} = (node: Node): boolean => ` +
 | 
						|
        mapProps(definitions)
 | 
						|
          .filter((d) => d.unionType && d.unionType.includes(unionType))
 | 
						|
          .map((d) => `is${d.name}(node) `)
 | 
						|
          .join("||") +
 | 
						|
        ";\n\n"
 | 
						|
    );
 | 
						|
  });
 | 
						|
 | 
						|
  // Node assertion
 | 
						|
  iterateProps(definitions, (typeDefinition) => {
 | 
						|
    stdout.write(`
 | 
						|
      export const assert${typeDefinition.name}: ((n: Node) => void) =
 | 
						|
        assertTypeOf("${typeDefinition.name}");
 | 
						|
    `);
 | 
						|
  });
 | 
						|
 | 
						|
  // a map from node type to its set of union types
 | 
						|
  stdout.write(
 | 
						|
    `
 | 
						|
    export const unionTypesMap = {` +
 | 
						|
      mapProps(definitions)
 | 
						|
        .filter((d) => d.unionType)
 | 
						|
        .map((t) => `"${t.name}": [${t.unionType.map(quote).join(",")}]\n`) +
 | 
						|
      `};
 | 
						|
      `
 | 
						|
  );
 | 
						|
 | 
						|
  // an array of all node and union types
 | 
						|
  stdout.write(
 | 
						|
    `
 | 
						|
    export const nodeAndUnionTypes = [` +
 | 
						|
      mapProps(definitions)
 | 
						|
        .map((t) => `"${t.name}"`)
 | 
						|
        .concat(unionTypes.map(quote))
 | 
						|
        .join(",") +
 | 
						|
      `];`
 | 
						|
  );
 | 
						|
}
 | 
						|
 | 
						|
generate();
 |