aboutsummaryrefslogtreecommitdiff
path: root/.github/devcontainers-action/lib
diff options
context:
space:
mode:
authorJosh Spicer <joshspicer@github.com>2022-06-14 03:01:19 +0300
committerGitHub <noreply@github.com>2022-06-14 03:01:19 +0300
commit2fa323248528604a3548fd1292c74c7b473c611b (patch)
tree39831944b1ed879a3342310b6f2a107e48153aad /.github/devcontainers-action/lib
parentfac5e5a4540ffc79138b94ce8edfd329a209b250 (diff)
Devcontainer docs (#47)
* docs * automatically generate documentation from features.json * trigger doc gen * 1 * no-ci * no-ci * no-ci * no-ci * comment out pull * Automated documentation update * no-ci markdown table * Automated documentation update * no-ci * Automated documentation update * branch to main Co-authored-by: Devcontainers CI <vscr-feedback@microsoft.com>
Diffstat (limited to '.github/devcontainers-action/lib')
-rw-r--r--.github/devcontainers-action/lib/contracts/features.js2
-rw-r--r--.github/devcontainers-action/lib/generateDocs.js135
-rw-r--r--.github/devcontainers-action/lib/main.js14
3 files changed, 151 insertions, 0 deletions
diff --git a/.github/devcontainers-action/lib/contracts/features.js b/.github/devcontainers-action/lib/contracts/features.js
new file mode 100644
index 0000000..c8ad2e5
--- /dev/null
+++ b/.github/devcontainers-action/lib/contracts/features.js
@@ -0,0 +1,2 @@
+"use strict";
+Object.defineProperty(exports, "__esModule", { value: true });
diff --git a/.github/devcontainers-action/lib/generateDocs.js b/.github/devcontainers-action/lib/generateDocs.js
new file mode 100644
index 0000000..707ab12
--- /dev/null
+++ b/.github/devcontainers-action/lib/generateDocs.js
@@ -0,0 +1,135 @@
+"use strict";
+var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
+ if (k2 === undefined) k2 = k;
+ Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
+}) : (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;
+};
+var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
+ return new (P || (P = Promise))(function (resolve, reject) {
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
+ });
+};
+Object.defineProperty(exports, "__esModule", { value: true });
+exports.generateFeaturesDocumentation = void 0;
+const fs = __importStar(require("fs"));
+const github = __importStar(require("@actions/github"));
+const core = __importStar(require("@actions/core"));
+const path = __importStar(require("path"));
+function generateFeaturesDocumentation(basePath) {
+ return __awaiter(this, void 0, void 0, function* () {
+ fs.readdir(basePath, (err, files) => {
+ if (err) {
+ core.error(err.message);
+ core.setFailed(`failed to generate 'features' documentation ${err.message}`);
+ return;
+ }
+ files.forEach(f => {
+ core.info(`Generating docs for feature '${f}'`);
+ if (f !== '.' && f !== '..') {
+ const readmePath = path.join(basePath, f, 'README.md');
+ // Reads in feature.json
+ const featureJsonPath = path.join(basePath, f, 'devcontainer-feature.json');
+ if (!fs.existsSync(featureJsonPath)) {
+ core.error(`devcontainer-feature.json not found at path '${featureJsonPath}'`);
+ return;
+ }
+ let featureJson = undefined;
+ try {
+ featureJson = JSON.parse(fs.readFileSync(featureJsonPath, 'utf8'));
+ }
+ catch (err) {
+ core.error(`Failed to parse ${featureJsonPath}: ${err}`);
+ return;
+ }
+ if (!featureJson || !(featureJson === null || featureJson === void 0 ? void 0 : featureJson.id)) {
+ core.error(`devcontainer-feature.json for feature '${f}' does not contain an 'id'`);
+ return;
+ }
+ const ref = github.context.ref;
+ const owner = github.context.repo.owner;
+ const repo = github.context.repo.repo;
+ // Add tag if parseable
+ let versionTag = 'latest';
+ if (ref.includes('refs/tags/')) {
+ versionTag = ref.replace('refs/tags/', '');
+ }
+ const generateOptionsMarkdown = () => {
+ const options = featureJson === null || featureJson === void 0 ? void 0 : featureJson.options;
+ if (!options) {
+ return '';
+ }
+ const keys = Object.keys(options);
+ const contents = keys
+ .map(k => {
+ const val = options[k];
+ return `| ${k} | ${val.description || '-'} | ${val.type || '-'} | ${val.default || '-'} |`;
+ })
+ .join('\n');
+ return ('| Options Id | Description | Type | Default Value |\n' +
+ '|-----|-----|-----|-----|\n' +
+ contents);
+ };
+ const newReadme = README_TEMPLATE.replace('#{nwo}', `${owner}/${repo}`)
+ .replace('#{versionTag}', versionTag)
+ .replace('#{featureId}', featureJson.id)
+ .replace('#{featureName}', featureJson.name
+ ? `${featureJson.name} (${featureJson.id})`
+ : `${featureJson.id}`)
+ .replace('#{featureDescription}', featureJson.description ? featureJson.description : '')
+ .replace('#{optionsTable}', generateOptionsMarkdown());
+ // Remove previous readme
+ if (fs.existsSync(readmePath)) {
+ fs.unlinkSync(readmePath);
+ }
+ // Write new readme
+ fs.writeFileSync(readmePath, newReadme);
+ }
+ });
+ });
+ });
+}
+exports.generateFeaturesDocumentation = generateFeaturesDocumentation;
+const README_TEMPLATE = `
+# #{featureName}
+
+#{featureDescription}
+
+## Example Usage
+
+\`\`\`json
+"features": [
+ {
+ "id": "#{nwo}/#{featureId}@#{versionTag}",
+ "options": {
+ "version": "latest"
+ }
+ }
+]
+\`\`\`
+
+## Options
+
+#{optionsTable}
+
+---
+
+_Note: This file was auto-generated from the [devcontainer-feature.json](./devcontainer-feature.json)._
+`;
diff --git a/.github/devcontainers-action/lib/main.js b/.github/devcontainers-action/lib/main.js
index 63d938e..2e5d391 100644
--- a/.github/devcontainers-action/lib/main.js
+++ b/.github/devcontainers-action/lib/main.js
@@ -33,12 +33,15 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
};
Object.defineProperty(exports, "__esModule", { value: true });
const core = __importStar(require("@actions/core"));
+const generateDocs_1 = require("./generateDocs");
const utils_1 = require("./utils");
function run() {
return __awaiter(this, void 0, void 0, function* () {
core.debug('Reading input parameters...');
+ // Read inputs
const shouldPublishFeatures = core.getInput('publish-features').toLowerCase() === 'true';
const shouldPublishTemplate = core.getInput('publish-templates').toLowerCase() === 'true';
+ const shouldGenerateDocumentation = core.getInput('generate-docs').toLowerCase() === 'true';
if (shouldPublishFeatures) {
core.info('Publishing features...');
const featuresBasePath = core.getInput('base-path-to-features');
@@ -49,6 +52,17 @@ function run() {
const basePathToDefinitions = core.getInput('base-path-to-templates');
yield packageTemplates(basePathToDefinitions);
}
+ if (shouldGenerateDocumentation) {
+ core.info('Generating documentation...');
+ const featuresBasePath = core.getInput('base-path-to-features');
+ if (featuresBasePath) {
+ yield (0, generateDocs_1.generateFeaturesDocumentation)(featuresBasePath);
+ }
+ else {
+ core.error("'base-path-to-features' input is required to generate documentation");
+ }
+ // TODO: base-path-to-templates
+ }
// TODO: Programatically add feature/template fino with relevant metadata for UX clients.
core.info('Generation metadata file: devcontainer-collection.json');
yield (0, utils_1.addCollectionsMetadataFile)();