UNPKG

@aws-cdk/cx-api

Version:

Cloud executable protocol

199 lines 28.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); /** * Generate FEATURE_FLAGS.md, a report of all current feature flags */ const fs_1 = require("fs"); const path = require("path"); const validate_flags_1 = require("./validate-flags"); const feats = require("../lib/features"); const flag_modeling_1 = require("../lib/private/flag-modeling"); async function main() { (0, validate_flags_1.validateFlags)(); await updateMarkdownFile(path.join(__dirname, '..', 'FEATURE_FLAGS.md'), { table: flagsTable(), details: flagsDetails(), json: recommendedJson(), removed: removedFlags(), diff: changedFlags(), migratejson: migrateJson(), }); // Write to the package root await updateRecommendedFlagsFile(path.join(__dirname, '..', '..', 'recommended-feature-flags.json')); } function flagsTable() { return renderTable([ ['Flag', 'Summary', 'Since', 'Type'], ...v2flags().map(([name, flag]) => [ renderLink(mdEsc(name), githubHeadingLink(flagDetailsHeading(name, flag))), flag.summary, flag.introducedIn.v2 ?? '', renderType(flag.type, 'short'), ]), ]); } function removedFlags() { const removedInV2 = flags(flag => flag.introducedIn.v2 === undefined && flag.introducedIn.v1 !== undefined); return renderTable([ ['Flag', 'Summary', 'Type', 'Since'], ...removedInV2.map(([name, flag]) => [ renderLink(mdEsc(name), githubHeadingLink(flagDetailsHeading(name, flag))), flag.summary, renderType(flag.type, 'short'), flag.introducedIn.v1 ?? '', ]), ]); } function changedFlags() { const changedInV2 = flags(flag => !!flag.unconfiguredBehavesLike?.v2 && !!flag.introducedIn.v2); return renderTable([ ['Flag', 'Summary', 'Type', 'Since', 'v1 default', 'v2 default'], ...changedInV2.map(([name, flag]) => [ renderLink(mdEsc(name), githubHeadingLink(flagDetailsHeading(name, flag))), flag.summary, renderType(flag.type, 'short'), flag.introducedIn.v1 ?? '', renderValue(false), renderValue(flag.unconfiguredBehavesLike?.v2), ]), ]); } function migrateJson() { const changedInV2 = flags(flag => !!flag.unconfiguredBehavesLike?.v2 && !!flag.introducedIn.v2 && !!flag.introducedIn.v1); const context = Object.fromEntries(changedInV2.map(([name, _]) => [name, false])); return [ '```json', JSON.stringify({ context }, undefined, 2), '```', ].join('\n'); } function flagsDetails() { const allFlags = flags(_ => true); return allFlags.flatMap(([name, flag]) => [ `### ${flagDetailsHeading(name, flag)}`, '', `*${flag.summary}*`, '', `Flag type: ${renderType(flag.type, 'long')}`, '', dedent(flag.detailsMd), '', renderTable([ ['Since', 'Unset behaves like', 'Recommended value'], // V1 flag.introducedIn.v1 ? [flag.introducedIn.v1, renderValue(false), renderValue(flag.recommendedValue)] : ['(not in v1)', '', ''], // V2 flag.introducedIn.v2 ? [flag.introducedIn.v2, renderValue(flag.unconfiguredBehavesLike?.v2 ?? false), renderValue(flag.recommendedValue)] : flag.unconfiguredBehavesLike?.v2 !== undefined ? ['(not configurable in v2)', renderValue(flag.unconfiguredBehavesLike?.v2), ''] : ['(not in v2)', '', ''], ]), ...oldBehavior(flag) ? [ `**Compatibility with old behavior:** ${oldBehavior(flag)}`, '', ] : [], '', ]).join('\n'); } function oldBehavior(flag) { switch (flag.type) { case flag_modeling_1.FlagType.ApiDefault: return flag.compatibilityWithOldBehaviorMd; case flag_modeling_1.FlagType.BugFix: return flag.compatibilityWithOldBehaviorMd; case flag_modeling_1.FlagType.VisibleContext: return undefined; case flag_modeling_1.FlagType.Temporary: return flag.compatibilityWithOldBehaviorMd; } } function recommendedJson() { return [ '```json', JSON.stringify({ context: feats.CURRENTLY_RECOMMENDED_FLAGS }, undefined, 2), '```', ].join('\n'); } function v2flags() { return flags(flag => flag.introducedIn.v2 !== undefined); } function flags(pred) { const entries = Object.entries(feats.FLAGS) .filter(([_, flag]) => pred(flag)); entries.sort((a, b) => firstCmp( // Sort by versions first (0, flag_modeling_1.compareVersions)(a[1].introducedIn.v2, b[1].introducedIn.v2), (0, flag_modeling_1.compareVersions)(a[1].introducedIn.v1, b[1].introducedIn.v1), // Then sort by name a[0].localeCompare(b[0]))); return entries; } function renderType(type, flavor) { switch (type) { case flag_modeling_1.FlagType.ApiDefault: return longShort('New default behavior', 'new default'); case flag_modeling_1.FlagType.BugFix: return longShort('Backwards incompatible bugfix', 'fix'); case flag_modeling_1.FlagType.VisibleContext: return longShort('Configuration option', 'config'); case flag_modeling_1.FlagType.Temporary: return longShort('Temporary flag', 'temporary'); } function longShort(long, short) { return flavor === 'long' ? long : short; } } function renderTable(rows) { return [ '', '| ' + rows[0].join(' | ') + ' |', '| ' + rows[0].map(_ => '-----').join(' | ') + ' |', ...rows.slice(1).map(row => '| ' + row.join(' | ') + ' |'), '', ].join('\n'); } /** * Return the heading that will be used to caption this flag's details */ function flagDetailsHeading(name, _) { return name; } /** * Return a link that is valid on GitHub to refer to a heading */ function githubHeadingLink(heading) { return `#${heading.toLowerCase().replace(/ /g, '-').replace(/[^a-z0-9_-]/g, '')}`; } /** * Remove shared leading whitespace from all non-empty lines */ function dedent(body) { const lines = body.split('\n').filter((x) => x.trim() !== ''); const leadingWs = lines.map(x => x.match(/^ */)?.[0].length ?? 0); const sharedWs = Math.min(...leadingWs); const re = new RegExp('^' + ' '.repeat(sharedWs), 'mg'); return body.replace(re, '').trim(); } function renderValue(x) { return `\`${JSON.stringify(x)}\``; } function renderLink(caption, link) { return `[${caption}](${link})`; } function mdEsc(x) { return x.replace(/_/g, '\\_'); } async function updateMarkdownFile(filename, sections) { let contents = await fs_1.promises.readFile(filename, { encoding: 'utf-8' }); for (const [section, value] of Object.entries(sections)) { const re = new RegExp(`<!-- BEGIN ${section} -->(.*)<!-- END ${section} -->`, 's'); contents = contents.replace(re, `<!-- BEGIN ${section} -->\n${value}\n<!-- END ${section} -->`); } await fs_1.promises.writeFile(filename, contents, { encoding: 'utf-8' }); } async function updateRecommendedFlagsFile(filename) { await fs_1.promises.writeFile(filename, JSON.stringify(feats.CURRENTLY_RECOMMENDED_FLAGS, undefined, 2), { encoding: 'utf-8' }); } function firstCmp(...xs) { return xs.find(x => x !== 0) ?? 0; } main().catch(e => { // eslint-disable-next-line no-console console.error(e); process.exitCode = 1; }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"flag-report.js","sourceRoot":"","sources":["flag-report.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Generate FEATURE_FLAGS.md, a report of all current feature flags\n */\nimport { promises as fs } from 'fs';\nimport * as path from 'path';\nimport { validateFlags } from './validate-flags';\nimport * as feats from '../lib/features';\nimport { FlagInfo, FlagType, compareVersions } from '../lib/private/flag-modeling';\n\nasync function main() {\n  validateFlags();\n\n  await updateMarkdownFile(path.join(__dirname, '..', 'FEATURE_FLAGS.md'), {\n    table: flagsTable(),\n    details: flagsDetails(),\n    json: recommendedJson(),\n    removed: removedFlags(),\n    diff: changedFlags(),\n    migratejson: migrateJson(),\n  });\n\n  // Write to the package root\n  await updateRecommendedFlagsFile(path.join(__dirname, '..', '..', 'recommended-feature-flags.json'));\n}\n\nfunction flagsTable() {\n  return renderTable([\n    ['Flag', 'Summary', 'Since', 'Type'],\n    ...v2flags().map(([name, flag]) =>\n      [\n        renderLink(mdEsc(name), githubHeadingLink(flagDetailsHeading(name, flag))),\n        flag.summary,\n        flag.introducedIn.v2 ?? '',\n        renderType(flag.type, 'short'),\n      ],\n    ),\n  ]);\n}\n\nfunction removedFlags() {\n  const removedInV2 = flags(flag => flag.introducedIn.v2 === undefined && flag.introducedIn.v1 !== undefined);\n\n  return renderTable([\n    ['Flag', 'Summary', 'Type', 'Since'],\n    ...removedInV2.map(([name, flag]) => [\n      renderLink(mdEsc(name), githubHeadingLink(flagDetailsHeading(name, flag))),\n      flag.summary,\n      renderType(flag.type, 'short'),\n      flag.introducedIn.v1 ?? '',\n    ]),\n  ]);\n}\n\nfunction changedFlags() {\n  const changedInV2 = flags(flag => !!flag.unconfiguredBehavesLike?.v2 && !!flag.introducedIn.v2);\n\n  return renderTable([\n    ['Flag', 'Summary', 'Type', 'Since', 'v1 default', 'v2 default'],\n    ...changedInV2.map(([name, flag]) => [\n      renderLink(mdEsc(name), githubHeadingLink(flagDetailsHeading(name, flag))),\n      flag.summary,\n      renderType(flag.type, 'short'),\n      flag.introducedIn.v1 ?? '',\n      renderValue(false),\n      renderValue(flag.unconfiguredBehavesLike?.v2),\n    ]),\n  ]);\n}\n\nfunction migrateJson() {\n  const changedInV2 = flags(flag => !!flag.unconfiguredBehavesLike?.v2 && !!flag.introducedIn.v2 && !!flag.introducedIn.v1);\n\n  const context = Object.fromEntries(changedInV2.map(([name, _]) => [name, false]));\n\n  return [\n    '```json',\n    JSON.stringify({ context }, undefined, 2),\n    '```',\n  ].join('\\n');\n}\n\nfunction flagsDetails() {\n  const allFlags = flags(_ => true);\n\n  return allFlags.flatMap(([name, flag]) => [\n    `### ${flagDetailsHeading(name, flag)}`,\n    '',\n    `*${flag.summary}*`,\n    '',\n    `Flag type: ${renderType(flag.type, 'long')}`,\n    '',\n    dedent(flag.detailsMd),\n    '',\n    renderTable([\n      ['Since', 'Unset behaves like', 'Recommended value'],\n\n      // V1\n      flag.introducedIn.v1\n        ? [flag.introducedIn.v1, renderValue(false), renderValue(flag.recommendedValue)]\n        : ['(not in v1)', '', ''],\n\n      // V2\n      flag.introducedIn.v2\n        ? [flag.introducedIn.v2, renderValue(flag.unconfiguredBehavesLike?.v2 ?? false), renderValue(flag.recommendedValue)]\n        : flag.unconfiguredBehavesLike?.v2 !== undefined\n          ? ['(not configurable in v2)', renderValue(flag.unconfiguredBehavesLike?.v2), '']\n          : ['(not in v2)', '', ''],\n    ]),\n    ...oldBehavior(flag) ? [\n      `**Compatibility with old behavior:** ${oldBehavior(flag)}`,\n      '',\n    ] : [],\n    '',\n  ]).join('\\n');\n}\n\nfunction oldBehavior(flag: FlagInfo): string | undefined {\n  switch (flag.type) {\n    case FlagType.ApiDefault: return flag.compatibilityWithOldBehaviorMd;\n    case FlagType.BugFix: return flag.compatibilityWithOldBehaviorMd;\n    case FlagType.VisibleContext: return undefined;\n    case FlagType.Temporary: return flag.compatibilityWithOldBehaviorMd;\n  }\n}\n\nfunction recommendedJson() {\n  return [\n    '```json',\n    JSON.stringify({ context: feats.CURRENTLY_RECOMMENDED_FLAGS }, undefined, 2),\n    '```',\n  ].join('\\n');\n}\n\nfunction v2flags() {\n  return flags(flag => flag.introducedIn.v2 !== undefined);\n}\n\nfunction flags(pred: (x: FlagInfo) => boolean) {\n  const entries = Object.entries(feats.FLAGS)\n    .filter(([_, flag]) => pred(flag));\n\n  entries.sort((a, b) => firstCmp(\n    // Sort by versions first\n    compareVersions(a[1].introducedIn.v2, b[1].introducedIn.v2),\n    compareVersions(a[1].introducedIn.v1, b[1].introducedIn.v1),\n    // Then sort by name\n    a[0].localeCompare(b[0])));\n\n  return entries;\n}\n\nfunction renderType(type: FlagType, flavor: 'short' | 'long'): string {\n  switch (type) {\n    case FlagType.ApiDefault: return longShort('New default behavior', 'new default');\n    case FlagType.BugFix: return longShort('Backwards incompatible bugfix', 'fix');\n    case FlagType.VisibleContext: return longShort('Configuration option', 'config');\n    case FlagType.Temporary: return longShort('Temporary flag', 'temporary');\n  }\n\n  function longShort(long: string, short: string) {\n    return flavor === 'long' ? long : short;\n  }\n}\n\nfunction renderTable(rows: string[][]) {\n  return [\n    '',\n    '| ' + rows[0].join(' | ') + ' |',\n    '| ' + rows[0].map(_ => '-----').join(' | ') + ' |',\n    ...rows.slice(1).map(row => '| ' + row.join(' | ') + ' |'),\n    '',\n  ].join('\\n');\n}\n\n/**\n * Return the heading that will be used to caption this flag's details\n */\nfunction flagDetailsHeading(name: string, _: FlagInfo) {\n  return name;\n}\n\n/**\n * Return a link that is valid on GitHub to refer to a heading\n */\nfunction githubHeadingLink(heading: string) {\n  return `#${heading.toLowerCase().replace(/ /g, '-').replace(/[^a-z0-9_-]/g, '')}`;\n}\n\n/**\n * Remove shared leading whitespace from all non-empty lines\n */\nfunction dedent(body: string) {\n  const lines = body.split('\\n').filter((x) => x.trim() !== '');\n  const leadingWs = lines.map(x => x.match(/^ */)?.[0].length ?? 0);\n  const sharedWs = Math.min(...leadingWs);\n  const re = new RegExp('^' + ' '.repeat(sharedWs), 'mg');\n  return body.replace(re, '').trim();\n}\n\nfunction renderValue(x: any) {\n  return `\\`${JSON.stringify(x)}\\``;\n}\n\nfunction renderLink(caption: string, link: string) {\n  return `[${caption}](${link})`;\n}\n\nfunction mdEsc(x: string) {\n  return x.replace(/_/g, '\\\\_');\n}\n\nasync function updateMarkdownFile(filename: string, sections: Record<string, string>) {\n  let contents = await fs.readFile(filename, { encoding: 'utf-8' });\n\n  for (const [section, value] of Object.entries(sections)) {\n    const re = new RegExp(`<!-- BEGIN ${section} -->(.*)<!-- END ${section} -->`, 's');\n    contents = contents.replace(re, `<!-- BEGIN ${section} -->\\n${value}\\n<!-- END ${section} -->`);\n  }\n\n  await fs.writeFile(filename, contents, { encoding: 'utf-8' });\n}\n\nasync function updateRecommendedFlagsFile(filename: string) {\n  await fs.writeFile(filename, JSON.stringify(feats.CURRENTLY_RECOMMENDED_FLAGS, undefined, 2), { encoding: 'utf-8' });\n}\n\nfunction firstCmp(...xs: number[]) {\n  return xs.find(x => x !== 0) ?? 0;\n}\n\nmain().catch(e => {\n  // eslint-disable-next-line no-console\n  console.error(e);\n  process.exitCode = 1;\n});\n"]}