Johannes Kirschbauer
294c5548b9
Some checks failed
buildbot/nix-build .#checks.aarch64-darwin.nixos-test_install_machine Build done.
buildbot/nix-build .#checks.aarch64-darwin.nixos-iso-installer Build done.
buildbot/nix-build .#checks.aarch64-darwin.nixos-flash-installer Build done.
buildbot/nix-build .#checks.aarch64-linux.nixos-test_install_machine Build done.
buildbot/nix-build .#checks.x86_64-linux.check-for-breakpoints Build done.
buildbot/nix-build .#checks.aarch64-linux.nixos-flash-installer Build done.
buildbot/nix-build .#checks.x86_64-linux.package-default Build done.
buildbot/nix-build .#checks.x86_64-linux.lib-jsonschema-nix-unit-tests Build done.
buildbot/nix-build .#checks.x86_64-linux.package-gui-installer-apk Build done.
buildbot/nix-build .#checks.x86_64-linux.devShell-docs Build done.
buildbot/nix-build .#checks.x86_64-linux.package-gui-installer-deb Build done.
buildbot/nix-build .#checks.aarch64-linux.nixos-iso-installer Build done.
buildbot/nix-build .#checks.x86_64-linux.lib-jsonschema-example-valid Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-app-no-breakpoints Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-bash Build done.
buildbot/nix-build .#checks.x86_64-linux.package-module-docs Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-fakeroot Build done.
buildbot/nix-build .#checks.aarch64-darwin.nixos-test-backup Build done.
buildbot/nix-build .#checks.x86_64-linux.package-gui-installer-archlinux Build done.
buildbot/nix-build .#checks.x86_64-linux.package-gui-installer-rpm Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-git Build done.
buildbot/nix-build .#checks.x86_64-linux.renderClanOptions Build done.
buildbot/nix-build .#checks.aarch64-linux.nixos-test-backup Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-nix Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-openssh Build done.
buildbot/nix-build .#checks.x86_64-linux."clan-dep-python3.11-mypy" Build done.
buildbot/nix-build .#checks.x86_64-linux."clan-dep-python3.11-qemu" Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-rsync Build done.
buildbot/nix-build .#checks.x86_64-linux.package-clan-cli Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-sops Build done.
buildbot/nix-build .#checks.x86_64-linux.package-clan-app Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-sshpass Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-tor Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-zbar Build done.
buildbot/nix-build .#checks.x86_64-linux.nixos-test-backup Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-age Build done.
buildbot/nix-build .#checks.x86_64-linux.devShell-webview-ui Build done.
buildbot/nix-build .#checks.x86_64-linux.container Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-dep-e2fsprogs Build done.
buildbot/nix-build .#checks.x86_64-linux.devShell-clan-cli Build done.
buildbot/nix-build .#checks.x86_64-linux.devShell-inventory-schema Build done.
buildbot/nix-build .#checks.x86_64-linux.inventory-schema-checks Build done.
buildbot/nix-build .#checks.x86_64-linux.devShell-clan-app Build done.
buildbot/nix-build .#checks.x86_64-linux.devShell-default Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-app-pytest Build done.
buildbot/nix-build .#checks.x86_64-linux.borgbackup Build done.
buildbot/nix-build .#checks.x86_64-linux.matrix-synapse Build done.
buildbot/nix-build .#checks.x86_64-linux.package-clan-cli-docs Build done.
buildbot/nix-build .#checks.x86_64-linux.package-clan-ts-api Build done.
buildbot/nix-build .#checks.x86_64-linux.module-schema Build done.
buildbot/nix-build .#checks.x86_64-linux.package-editor Build done.
buildbot/nix-build .#checks.x86_64-linux.package-impure-checks Build done.
buildbot/nix-build .#checks.x86_64-linux.treefmt Build done.
buildbot/nix-build .#checks.x86_64-linux.package-inventory-schema Build done.
buildbot/nix-build .#checks.x86_64-linux.package-deploy-docs Build done.
buildbot/nix-build .#checks.x86_64-linux.nixos-test_install_machine Build done.
buildbot/nix-build .#checks.x86_64-linux.package-merge-after-ci Build done.
buildbot/nix-build .#checks.x86_64-linux.package-moonlight-sunshine-accept Build done.
buildbot/nix-build .#checks.x86_64-linux.package-zerotier-members Build done.
buildbot/nix-build .#checks.x86_64-linux.package-zt-tcp-relay Build done.
buildbot/nix-build .#checks.x86_64-linux.package-function-schema Build done.
buildbot/nix-build .#checks.x86_64-linux.package-module-schema Build done.
buildbot/nix-build .#checks.x86_64-linux.nixos-iso-installer Build done.
buildbot/nix-build .#checks.x86_64-linux.secrets Build done.
buildbot/nix-build .#checks.x86_64-linux.wayland-proxy-virtwl Build done.
buildbot/nix-build .#checks.x86_64-linux.package-iso-installer Build done.
buildbot/nix-build .#checks.x86_64-linux.template-minimal Build done.
buildbot/nix-build .#checks.x86_64-linux.zt-tcp-relay Build done.
buildbot/nix-build .#checks.x86_64-linux.package-gui-install-test-ubuntu-22-04 Build done.
buildbot/nix-build .#checks.x86_64-linux.package-docs Build done.
buildbot/nix-build .#checks.x86_64-linux.package-pending-reviews Build done.
buildbot/nix-build .#checks.x86_64-linux.package-tea-create-pr Build done.
buildbot/nix-build .#checks.x86_64-linux.package-webview-ui Build done.
buildbot/nix-build .#checks.x86_64-linux.deltachat Build done.
buildbot/nix-build .#checks.x86_64-linux.package-zerotierone Build done.
buildbot/nix-build .#checks.x86_64-linux.postgresql Build done.
buildbot/nix-build .#checks.x86_64-linux.nixos-flash-installer Build done.
buildbot/nix-build .#checks.x86_64-linux.syncthing Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-pytest-without-core Build done.
buildbot/nix-build .#checks.x86_64-linux.clan-pytest-with-core Build done.
buildbot/nix-build .#checks.x86_64-linux.test-backups Build done.
checks / checks-impure (pull_request) Successful in 2m15s
buildbot/nix-build .#checks.x86_64-linux.flash Build done.
buildbot/nix-build .#checks.x86_64-linux.test-installation Build done.
buildbot/nix-eval Build done.
331 lines
9.6 KiB
Nix
331 lines
9.6 KiB
Nix
{
|
|
lib ? import <nixpkgs/lib>,
|
|
excludedTypes ? [
|
|
"functionTo"
|
|
"package"
|
|
],
|
|
}:
|
|
let
|
|
# remove _module attribute from options
|
|
clean = opts: builtins.removeAttrs opts [ "_module" ];
|
|
|
|
# throw error if option type is not supported
|
|
notSupported =
|
|
option:
|
|
lib.trace option throw ''
|
|
option type '${option.type.name}' ('${option.type.description}') not supported by jsonschema converter
|
|
location: ${lib.concatStringsSep "." option.loc}
|
|
'';
|
|
|
|
# Exclude the option if its type is in the excludedTypes list
|
|
# or if the option has a defaultText attribute
|
|
isExcludedOption =
|
|
option: ((lib.elem (option.type.name or null) excludedTypes) || (option ? defaultText));
|
|
|
|
filterExcluded = lib.filter (opt: !isExcludedOption opt);
|
|
|
|
filterExcludedAttrs = lib.filterAttrs (_name: opt: !isExcludedOption opt);
|
|
|
|
# Filter out options where the visible attribute is set to false
|
|
filterInvisibleOpts = lib.filterAttrs (_name: opt: opt.visible or true);
|
|
|
|
allBasicTypes = [
|
|
"boolean"
|
|
"integer"
|
|
"number"
|
|
"string"
|
|
"array"
|
|
"object"
|
|
"null"
|
|
];
|
|
in
|
|
rec {
|
|
|
|
# parses a nixos module to a jsonschema
|
|
parseModule =
|
|
module:
|
|
let
|
|
evaled = lib.evalModules { modules = [ module ]; };
|
|
in
|
|
{ "$schema" = "http://json-schema.org/draft-07/schema#"; } // parseOptions evaled.options;
|
|
|
|
# parses a set of evaluated nixos options to a jsonschema
|
|
parseOptions =
|
|
options':
|
|
let
|
|
options = filterInvisibleOpts (filterExcludedAttrs (clean options'));
|
|
# parse options to jsonschema properties
|
|
properties = lib.mapAttrs (_name: option: parseOption option) options;
|
|
# TODO: figure out how to handle if prop.anyOf is used
|
|
isRequired = prop: !(prop ? default || prop.type or null == "object");
|
|
requiredProps = lib.filterAttrs (_: prop: isRequired prop) properties;
|
|
required = lib.optionalAttrs (requiredProps != { }) { required = lib.attrNames requiredProps; };
|
|
in
|
|
# return jsonschema
|
|
required
|
|
// {
|
|
type = "object";
|
|
inherit properties;
|
|
additionalProperties = false;
|
|
};
|
|
|
|
# parses and evaluated nixos option to a jsonschema property definition
|
|
parseOption =
|
|
option:
|
|
let
|
|
default = lib.optionalAttrs (option ? default) { inherit (option) default; };
|
|
example = lib.optionalAttrs (option ? example) {
|
|
examples =
|
|
if (builtins.typeOf option.example) == "list" then option.example else [ option.example ];
|
|
};
|
|
description = lib.optionalAttrs (option ? description) {
|
|
description = option.description.text or option.description;
|
|
};
|
|
in
|
|
|
|
# either type
|
|
# TODO: if all nested options are excluded, the parent should be excluded too
|
|
if
|
|
option.type.name or null == "either" || option.type.name or null == "coercedTo"
|
|
# return jsonschema property definition for either
|
|
then
|
|
let
|
|
optionsList' = [
|
|
{
|
|
type = option.type.nestedTypes.left or option.type.nestedTypes.coercedType;
|
|
_type = "option";
|
|
loc = option.loc;
|
|
}
|
|
{
|
|
type = option.type.nestedTypes.right or option.type.nestedTypes.finalType;
|
|
_type = "option";
|
|
loc = option.loc;
|
|
}
|
|
];
|
|
optionsList = filterExcluded optionsList';
|
|
in
|
|
default // example // description // { anyOf = map parseOption optionsList; }
|
|
|
|
# handle nested options (not a submodule)
|
|
else if !option ? _type then
|
|
parseOptions option
|
|
|
|
# throw if not an option
|
|
else if option._type != "option" && option._type != "option-type" then
|
|
throw "parseOption: not an option"
|
|
|
|
# parse nullOr
|
|
else if
|
|
option.type.name == "nullOr"
|
|
# return jsonschema property definition for nullOr
|
|
then
|
|
let
|
|
nestedOption = {
|
|
type = option.type.nestedTypes.elemType;
|
|
_type = "option";
|
|
loc = option.loc;
|
|
};
|
|
in
|
|
default
|
|
// example
|
|
// description
|
|
// {
|
|
anyOf = [
|
|
{ type = "null"; }
|
|
] ++ (lib.optional (!isExcludedOption nestedOption) (parseOption nestedOption));
|
|
}
|
|
|
|
# parse bool
|
|
else if
|
|
option.type.name == "bool"
|
|
# return jsonschema property definition for bool
|
|
then
|
|
default // example // description // { type = "boolean"; }
|
|
|
|
# parse float
|
|
else if
|
|
option.type.name == "float"
|
|
# return jsonschema property definition for float
|
|
then
|
|
default // example // description // { type = "number"; }
|
|
|
|
# parse int
|
|
else if
|
|
(option.type.name == "int" || option.type.name == "positiveInt")
|
|
# return jsonschema property definition for int
|
|
then
|
|
default // example // description // { type = "integer"; }
|
|
|
|
# TODO: Add support for intMatching in jsonschema
|
|
# parse port type aka. "unsignedInt16"
|
|
else if
|
|
option.type.name == "unsignedInt16"
|
|
|| option.type.name == "unsignedInt"
|
|
|| option.type.name == "pkcs11"
|
|
|| option.type.name == "intBetween"
|
|
then
|
|
default // example // description // { type = "integer"; }
|
|
|
|
# parse string
|
|
# TODO: parse more precise string types
|
|
else if
|
|
option.type.name == "str"
|
|
|| option.type.name == "singleLineStr"
|
|
|| option.type.name == "passwdEntry str"
|
|
|| option.type.name == "passwdEntry path"
|
|
# return jsonschema property definition for string
|
|
then
|
|
default // example // description // { type = "string"; }
|
|
|
|
# TODO: Add support for stringMatching in jsonschema
|
|
# parse stringMatching
|
|
else if lib.strings.hasPrefix "strMatching" option.type.name then
|
|
default // example // description // { type = "string"; }
|
|
|
|
# TODO: Add support for separatedString in jsonschema
|
|
else if lib.strings.hasPrefix "separatedString" option.type.name then
|
|
default // example // description // { type = "string"; }
|
|
|
|
# parse string
|
|
else if
|
|
option.type.name == "path"
|
|
# return jsonschema property definition for path
|
|
then
|
|
default // example // description // { type = "string"; }
|
|
|
|
# parse anything
|
|
else if
|
|
option.type.name == "anything"
|
|
# return jsonschema property definition for anything
|
|
then
|
|
default // example // description // { type = allBasicTypes; }
|
|
|
|
# parse unspecified
|
|
else if
|
|
option.type.name == "unspecified"
|
|
# return jsonschema property definition for unspecified
|
|
then
|
|
default // example // description // { type = allBasicTypes; }
|
|
|
|
# parse raw
|
|
else if
|
|
option.type.name == "raw"
|
|
# return jsonschema property definition for raw
|
|
then
|
|
default // example // description // { type = allBasicTypes; }
|
|
|
|
# parse enum
|
|
else if
|
|
option.type.name == "enum"
|
|
# return jsonschema property definition for enum
|
|
then
|
|
default // example // description // { enum = option.type.functor.payload; }
|
|
|
|
# parse listOf submodule
|
|
else if
|
|
option.type.name == "listOf" && option.type.functor.wrapped.name == "submodule"
|
|
# return jsonschema property definition for listOf submodule
|
|
then
|
|
default
|
|
// example
|
|
// description
|
|
// {
|
|
type = "array";
|
|
items = parseOptions (option.type.functor.wrapped.getSubOptions option.loc);
|
|
}
|
|
|
|
# parse list
|
|
else if
|
|
(option.type.name == "listOf")
|
|
# return jsonschema property definition for list
|
|
then
|
|
let
|
|
nestedOption = {
|
|
type = option.type.functor.wrapped;
|
|
_type = "option";
|
|
loc = option.loc;
|
|
};
|
|
in
|
|
default
|
|
// example
|
|
// description
|
|
// {
|
|
type = "array";
|
|
}
|
|
// (lib.optionalAttrs (!isExcludedOption nestedOption) { items = parseOption nestedOption; })
|
|
|
|
# parse list of unspecified
|
|
else if
|
|
(option.type.name == "listOf") && (option.type.functor.wrapped.name == "unspecified")
|
|
# return jsonschema property definition for list
|
|
then
|
|
default // example // description // { type = "array"; }
|
|
|
|
# parse attrsOf submodule
|
|
else if
|
|
option.type.name == "attrsOf" && option.type.nestedTypes.elemType.name == "submodule"
|
|
# return jsonschema property definition for attrsOf submodule
|
|
then
|
|
default
|
|
// example
|
|
// description
|
|
// {
|
|
type = "object";
|
|
additionalProperties = parseOptions (option.type.nestedTypes.elemType.getSubOptions option.loc);
|
|
}
|
|
|
|
# parse attrs
|
|
else if
|
|
option.type.name == "attrs"
|
|
# return jsonschema property definition for attrs
|
|
then
|
|
default
|
|
// example
|
|
// description
|
|
// {
|
|
type = "object";
|
|
additionalProperties = true;
|
|
}
|
|
|
|
# parse attrsOf
|
|
# TODO: if nested option is excluded, the parent sould be excluded too
|
|
else if
|
|
option.type.name == "attrsOf" || option.type.name == "lazyAttrsOf"
|
|
# return jsonschema property definition for attrs
|
|
then
|
|
let
|
|
nestedOption = {
|
|
type = option.type.nestedTypes.elemType;
|
|
_type = "option";
|
|
loc = option.loc;
|
|
};
|
|
in
|
|
default
|
|
// example
|
|
// description
|
|
// {
|
|
type = "object";
|
|
additionalProperties =
|
|
if !isExcludedOption nestedOption then
|
|
parseOption {
|
|
type = option.type.nestedTypes.elemType;
|
|
_type = "option";
|
|
loc = option.loc;
|
|
}
|
|
else
|
|
false;
|
|
}
|
|
|
|
# parse submodule
|
|
else if
|
|
option.type.name == "submodule"
|
|
# return jsonschema property definition for submodule
|
|
# then (lib.attrNames (option.type.getSubOptions option.loc).opt)
|
|
then
|
|
parseOptions (option.type.getSubOptions option.loc)
|
|
|
|
# throw error if option type is not supported
|
|
else
|
|
notSupported option;
|
|
}
|