Files
starter-workflows/script/validate-data/index.ts
T
2026-04-14 08:05:27 +02:00

187 lines
6.2 KiB
TypeScript
Executable File

#!/usr/bin/env npx ts-node
import { promises as fs } from "fs";
import { safeLoad } from "js-yaml";
import { basename, extname, join, dirname } from "path";
import { Validator as validator } from "jsonschema";
import { endGroup, error, info, setFailed, startGroup } from '@actions/core';
interface WorkflowWithErrors {
id: string;
name: string;
errors: string[];
}
interface WorkflowProperties {
name: string;
description: string;
creator: string;
iconName: string;
categories: string[];
}
const yamlWorkflowExtensions = [".yml", ".yaml"];
function getSupportedWorkflowExtensions(folder: string): string[] {
if (basename(folder).toLowerCase() === "agentic") {
return [...yamlWorkflowExtensions, ".md"];
}
return yamlWorkflowExtensions;
}
const propertiesSchema = {
type: "object",
properties: {
name: { type: "string", required: true , "minLength": 1},
description: { type: "string", required: true },
creator: { type: "string", required: false },
iconName: { type: "string", required: true },
categories: {
anyOf: [
{
type: "array",
items: { type: "string" }
},
{
type: "null",
}
],
required: true
},
}
}
async function checkWorkflows(folders: string[], allowed_categories: object[]): Promise<WorkflowWithErrors[]> {
const result: WorkflowWithErrors[] = []
const workflow_template_names = new Set()
for (const folder of folders) {
const supportedWorkflowExtensions = getSupportedWorkflowExtensions(folder);
const dir = await fs.readdir(folder, {
withFileTypes: true,
});
for (const e of dir) {
if (e.isFile() && supportedWorkflowExtensions.includes(extname(e.name))) {
const fileType = basename(e.name, extname(e.name))
const workflowFilePath = join(folder, e.name);
const propertiesFilePath = join(folder, "properties", `${fileType}.properties.json`)
const workflowWithErrors = await checkWorkflow(workflowFilePath, propertiesFilePath, allowed_categories);
if(workflowWithErrors.name && workflow_template_names.size == workflow_template_names.add(workflowWithErrors.name).size) {
workflowWithErrors.errors.push(`Workflow template name "${workflowWithErrors.name}" already exists`)
}
if (workflowWithErrors.errors.length > 0) {
result.push(workflowWithErrors)
}
}
}
}
return result;
}
function getMarkdownFrontmatter(workflowPath: string, workflowFileContent: string): string {
const frontmatterMatch = workflowFileContent.match(/^---\r?\n([\s\S]*?)\r?\n---(?:\r?\n|$)/);
if (!frontmatterMatch) {
throw new Error(`Markdown workflow ${workflowPath} must start with valid YAML frontmatter`);
}
return frontmatterMatch[1];
}
function validateWorkflowContent(workflowPath: string, workflowFileContent: string): void {
const extension = extname(workflowPath).toLowerCase();
if (yamlWorkflowExtensions.includes(extension)) {
safeLoad(workflowFileContent);
return;
}
if (extension === ".md") {
safeLoad(getMarkdownFrontmatter(workflowPath, workflowFileContent));
return;
}
throw new Error(`Unsupported workflow extension ${extension}`);
}
async function checkWorkflow(workflowPath: string, propertiesPath: string, allowed_categories: object[]): Promise<WorkflowWithErrors> {
let workflowErrors: WorkflowWithErrors = {
id: workflowPath,
name: null,
errors: []
}
try {
const workflowFileContent = await fs.readFile(workflowPath, "utf8");
validateWorkflowContent(workflowPath, workflowFileContent);
const propertiesFileContent = await fs.readFile(propertiesPath, "utf8")
const properties: WorkflowProperties = JSON.parse(propertiesFileContent)
if(properties.name && properties.name.trim().length > 0) {
workflowErrors.name = properties.name
}
let v = new validator();
const res = v.validate(properties, propertiesSchema)
workflowErrors.errors = res.errors.map(e => e.toString())
if (properties.iconName) {
if(! /^octicon\s+/.test(properties.iconName)) {
try {
await fs.access(`../../icons/${properties.iconName}.svg`)
} catch (e) {
workflowErrors.errors.push(`No icon named ${properties.iconName} found`)
}
}
else {
let iconName = properties.iconName.match(/^octicon\s+(.*)/)
if(!iconName || iconName[1].split(".")[0].length <= 0) {
workflowErrors.errors.push(`No icon named ${properties.iconName} found`)
}
}
}
var path = dirname(workflowPath)
var folder_categories = allowed_categories.find( category => category["path"] == path)["categories"]
if (!workflowPath.endsWith("blank.yml")) {
if(!properties.categories || properties.categories.length == 0) {
workflowErrors.errors.push(`Workflow categories cannot be null or empty`)
}
else if(!folder_categories.some(category => properties.categories[0].toLowerCase() == category.toLowerCase())) {
workflowErrors.errors.push(`The first category in properties.json categories for workflow in ${basename(path)} folder must be one of "${folder_categories}. Either move the workflow to an appropriate directory or change the category."`)
}
}
if(basename(path).toLowerCase() == 'deployments' && !properties.creator) {
workflowErrors.errors.push(`The "creator" in properties.json must be present.`)
}
} catch (e) {
workflowErrors.errors.push(e.toString())
}
return workflowErrors;
}
(async function main() {
try {
const settings = require("./settings.json");
const erroredWorkflows = await checkWorkflows(
settings.folders, settings.allowed_categories
)
if (erroredWorkflows.length > 0) {
startGroup(`😟 - Found ${erroredWorkflows.length} workflows with errors:`);
erroredWorkflows.forEach(erroredWorkflow => {
error(`Errors in ${erroredWorkflow.id} - ${erroredWorkflow.errors.map(e => e.toString()).join(", ")}`)
})
endGroup();
setFailed(`Found ${erroredWorkflows.length} workflows with errors`);
} else {
info("🎉🤘 - Found no workflows with errors!")
}
} catch (e) {
error(`Unhandled error while syncing workflows: ${e}`);
setFailed(`Unhandled error`)
}
})();