Skip to content

Commit

Permalink
WIP - validation
Browse files Browse the repository at this point in the history
  • Loading branch information
mvollmer committed Dec 16, 2024
1 parent baa20a0 commit 5dce77b
Show file tree
Hide file tree
Showing 4 changed files with 178 additions and 13 deletions.
179 changes: 172 additions & 7 deletions pkg/shell/manifests.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,8 @@ import { JsonValue, JsonObject } from "cockpit";
export interface ManifestKeyword {
matches: string[];
goto?: string;
weight: number;
translate: boolean;
weight?: number;
translate?: boolean;
}

export interface ManifestDocs {
Expand All @@ -49,9 +49,9 @@ export interface ManifestParentSection {
}

export interface Manifest {
dashboard: ManifestSection;
menu: ManifestSection;
tools: ManifestSection;
dashboard?: ManifestSection;
menu?: ManifestSection;
tools?: ManifestSection;

preload?: string[];
parent?: ManifestParentSection;
Expand All @@ -67,7 +67,172 @@ export interface ShellManifest {
locales?: { [id: string]: string };
}

class ValidationError extends Error { }

const validation_path: string[] = [];

function with_validation_path<T>(p: string, func: () => T): T {
validation_path.push(p);
try {
return func();
} finally {
validation_path.pop();
}
}

function validation_error(msg: string) {
console.error("Validation error:", JSON.stringify(validation_path), msg);
}

function import_string(val: JsonValue, fallback?: string): string {
if (typeof val == "string")
return val;
validation_error(`Not a string: ${JSON.stringify(val)}`);
if (fallback !== undefined)
return fallback;
throw new ValidationError();
}

function import_number(val: JsonValue, fallback?: number): number {
if (typeof val == "number")
return val;
validation_error(`Not a number: ${JSON.stringify(val)}`);
if (fallback !== undefined)
return fallback;
throw new ValidationError();
}

function import_boolean(val: JsonValue, fallback?: boolean): boolean {
if (typeof val == "boolean")
return val;
validation_error(`Not a boolean: ${JSON.stringify(val)}`);
if (fallback !== undefined)

Check notice

Code scanning / CodeQL

Unneeded defensive code Note

This guard always evaluates to true.
return fallback;
throw new ValidationError();
}

function import_json_object(val: JsonValue, fallback?: JsonObject): JsonObject {
if (!!val && typeof val == "object" && val.length === undefined)
return val as JsonObject;
validation_error(`Not an object: ${JSON.stringify(val)}`);
if (fallback !== undefined)
return fallback;
throw new ValidationError();
}

function import_json_array(val: JsonValue, fallback?: JsonValue[]): JsonValue[] {
if (!!val && typeof val == "object" && val.length !== undefined)
return val as JsonValue[];
validation_error(`Not an array: ${JSON.stringify(val)}`);
if (fallback !== undefined)

Check notice

Code scanning / CodeQL

Unneeded defensive code Note

This guard always evaluates to true.
return fallback;
throw new ValidationError();
}

function import_record<T>(val: JsonValue, importer: (val: JsonValue) => T): Record<string, T> {
const obj = import_json_object(val, {});
const res: Record<string, T> = {};
for (const key of Object.keys(obj)) {
try {
with_validation_path(key, () => { res[key] = importer(obj[key]) });
} catch (e) {
if (!(e instanceof ValidationError))
throw e;
}
}
return res;
}

function import_array<T>(val: JsonValue, importer: (val: JsonValue) => T): Array<T> {
const arr = import_json_array(val, []);
const res: Array<T> = [];
for (let i = 0; i < arr.length; i++) {
try {
with_validation_path(`index ${i}`, () => { res.push(importer(arr[i])) });
} catch (e) {
if (!(e instanceof ValidationError))
throw e;
}
}
return res;
}

function import_optional<T, F extends keyof T>(res: T, obj: JsonObject, field: F, importer: (val: JsonValue) => T[F]): void {
if (obj[field as string] === undefined)
return;
with_validation_path(field as string, () => { res[field] = importer(obj[field]) });
}

function import_mandatory<T, V>(obj: JsonObject, field: keyof T, importer: (val: JsonValue) => V): V {
if (obj[field as string] === undefined) {
validation_error(`Field ${String(field)} is missing`);
throw new ValidationError();
}
return with_validation_path(field as string, () => importer(obj[field]));
}

function import_ManifestKeyword(val: JsonValue): ManifestKeyword {
const obj = import_json_object(val);
const res: ManifestKeyword = {
matches: import_mandatory<ManifestKeyword, string[]>(obj, "matches", v => import_array(v, import_string)),
};
import_optional(res, obj, "goto", import_string);
import_optional(res, obj, "weight", v => import_number(v, 0));
import_optional(res, obj, "translate", v => import_boolean(v, false));
return res;
}

function import_ManifestDocs(val: JsonValue): ManifestDocs {
const obj = import_json_object(val);
const res: ManifestDocs = {
label: import_mandatory<ManifestDocs, string>(obj, "label", import_string),
url: import_mandatory<ManifestDocs, string>(obj, "url", import_string),
};
return res;
}

function import_ManifestEntry(val: JsonValue): ManifestEntry {
const obj = import_json_object(val);
const res: ManifestEntry = { };
import_optional(res, obj, "path", import_string);
import_optional(res, obj, "label", import_string);
import_optional(res, obj, "order", v => import_number(v, 0));
import_optional(res, obj, "docs", v => import_array(v, import_ManifestDocs));
import_optional(res, obj, "keywords", v => import_array(v, import_ManifestKeyword));
return res;
}

function import_ManifestSection(val: JsonValue): ManifestSection {
return import_record(val, import_ManifestEntry);
}

function import_ManifestParentSection(val: JsonValue): ManifestParentSection {
const obj = import_json_object(val);
const res: ManifestParentSection = { };
import_optional(res, obj, "component", import_string);
import_optional(res, obj, "docs", v => import_array(v, import_ManifestDocs));
return res;
}

function import_Manifest(val: JsonValue): Manifest {
const obj = import_json_object(val);
const res: Manifest = { };
import_optional(res, obj, "menu", import_ManifestSection);
import_optional(res, obj, "tools", import_ManifestSection);
import_optional(res, obj, "dashboard", import_ManifestSection);
import_optional(res, obj, "preload", v => import_array(v, import_string));
import_optional(res, obj, "parent", import_ManifestParentSection);
return res;
}

export function import_Manifests(val: JsonValue): Manifests {
// TODO - validate against schema
return val as unknown as Manifests;
return with_validation_path("Manifests", () => import_record(val, import_Manifest));
}

export function import_ShellManifest(val: JsonValue): ShellManifest {
const obj = import_json_object(val);
const res: ShellManifest = { };
import_optional(res, obj, "docs", v => import_array(v, import_ManifestDocs));
import_optional(res, obj, "locales", v => import_record(v, import_string));
return res;
}
4 changes: 2 additions & 2 deletions pkg/shell/shell-modals.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ import { SearchIcon } from '@patternfly/react-icons';

import { useInit } from "hooks";

import { ShellManifest } from "./manifests";
import { import_ShellManifest } from "./manifests";

import "menu-select-widget.scss";

Expand Down Expand Up @@ -106,7 +106,7 @@ export const LangModal = ({ dialogResult }) => {
window.location.reload();
}

const manifest = (cockpit.manifests.shell || { }) as ShellManifest;
const manifest = import_ShellManifest(cockpit.manifests.shell || { });

return (
<Modal isOpen position="top" variant="small"
Expand Down
4 changes: 2 additions & 2 deletions pkg/shell/topnav.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ import { Toolbar, ToolbarContent, ToolbarItem } from "@patternfly/react-core/dis
import { CogIcon, ExternalLinkAltIcon, HelpIcon } from '@patternfly/react-icons';

import { ShellState } from "./state";
import { ManifestDocs, ManifestParentSection, ShellManifest } from "./manifests";
import { ManifestDocs, ManifestParentSection, import_ShellManifest } from "./manifests";
import { ActivePagesDialog } from "./active-pages-modal.jsx";
import { CredentialsModal } from './credentials.jsx';
import { AboutCockpitModal, LangModal, OopsModal } from "./shell-modals.jsx";
Expand Down Expand Up @@ -148,7 +148,7 @@ export class TopNav extends React.Component {
{cockpit.format(_("$0 documentation"), this.state.osRelease.NAME)}
</DropdownItem>);

const shell_manifest = (cockpit.manifests.shell || {}) as ShellManifest;
const shell_manifest = import_ShellManifest(cockpit.manifests.shell || {});

// global documentation for cockpit as a whole
(shell_manifest.docs ?? []).forEach(doc => {
Expand Down
4 changes: 2 additions & 2 deletions pkg/shell/util.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -100,9 +100,9 @@ export class CompiledComponents {
this.manifests = manifests || { };
}

load(section: string, getter: (man: Manifest) => ManifestSection): void {
load(section: string, getter: (man: Manifest) => ManifestSection | undefined): void {
Object.entries(this.manifests).forEach(([name, manifest]) => {
const manifest_section = (getter(manifest) || {});
const manifest_section = getter(manifest) || {};
Object.entries(manifest_section).forEach(([prop, info]) => {
const item: ManifestItem = {
path: "", // set below
Expand Down

0 comments on commit 5dce77b

Please sign in to comment.