@aws-cdk/cx-api
Version:
Cloud executable protocol
199 lines • 28.1 kB
JavaScript
;
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"]}