diff --git a/lib/default.nix b/lib/default.nix index e9fbde4d..ef5fe7ac 100644 --- a/lib/default.nix +++ b/lib/default.nix @@ -5,6 +5,7 @@ ... }: { + evalClanModules = import ./eval-clan-modules { inherit clan-core nixpkgs lib; }; jsonschema = import ./jsonschema { inherit lib; }; modules = import ./description.nix { inherit clan-core lib; }; buildClan = import ./build-clan { inherit clan-core lib nixpkgs; }; diff --git a/lib/eval-clan-modules/default.nix b/lib/eval-clan-modules/default.nix new file mode 100644 index 00000000..814388d7 --- /dev/null +++ b/lib/eval-clan-modules/default.nix @@ -0,0 +1,34 @@ +{ + nixpkgs, + clan-core, + lib, +}: +let + inherit (import nixpkgs { system = "x86_64-linux"; }) pkgs; + + inherit (clan-core) clanModules; + + baseModule = { + imports = (import (pkgs.path + "/nixos/modules/module-list.nix")) ++ [ + { + nixpkgs.hostPlatform = "x86_64-linux"; + clan.core.clanName = "dummy"; + } + ]; + }; + + # This function takes a list of module names and evaluates them + # evalClanModules :: [ String ] -> { config, options, ... } + evalClanModules = + modulenames: + let + evaled = lib.evalModules { + modules = [ + baseModule + clan-core.nixosModules.clanCore + ] ++ (map (name: clanModules.${name}) modulenames); + }; + in + evaled; +in +evalClanModules diff --git a/lib/flake-module.nix b/lib/flake-module.nix index 48437bb1..ef93467b 100644 --- a/lib/flake-module.nix +++ b/lib/flake-module.nix @@ -7,7 +7,7 @@ { imports = [ ./jsonschema/flake-module.nix ]; flake.lib = import ./default.nix { - inherit lib; + inherit lib inputs; inherit (inputs) nixpkgs; clan-core = self; }; diff --git a/lib/jsonschema/default.nix b/lib/jsonschema/default.nix index df3f63b8..0a92da9b 100644 --- a/lib/jsonschema/default.nix +++ b/lib/jsonschema/default.nix @@ -47,7 +47,7 @@ rec { let evaled = lib.evalModules { modules = [ module ]; }; in - parseOptions evaled.options; + { "$schema" = "http://json-schema.org/draft-07/schema#"; } // parseOptions evaled.options; # parses a set of evaluated nixos options to a jsonschema parseOptions = @@ -66,6 +66,7 @@ rec { // { type = "object"; inherit properties; + additionalProperties = false; }; # parses and evaluated nixos option to a jsonschema property definition diff --git a/lib/jsonschema/example-schema.json b/lib/jsonschema/example-schema.json index 0073369e..dbdb7536 100644 --- a/lib/jsonschema/example-schema.json +++ b/lib/jsonschema/example-schema.json @@ -1,5 +1,7 @@ { + "$schema": "http://json-schema.org/draft-07/schema#", "type": "object", + "additionalProperties": false, "properties": { "name": { "type": "string", @@ -38,6 +40,7 @@ }, "services": { "type": "object", + "additionalProperties": false, "properties": { "opt": { "type": "string", @@ -59,9 +62,8 @@ "type": "string" } }, - "required": [ - "repo" - ], + "required": ["repo"], + "additionalProperties": false, "type": "object" }, "default": {}, diff --git a/lib/jsonschema/test_parseOption.nix b/lib/jsonschema/test_parseOption.nix index 8471ed43..930eee44 100644 --- a/lib/jsonschema/test_parseOption.nix +++ b/lib/jsonschema/test_parseOption.nix @@ -278,6 +278,7 @@ in expr = slib.parseOption (evalType (lib.types.submodule subModule) { }); expected = { type = "object"; + additionalProperties = false; properties = { opt = { type = "boolean"; @@ -301,6 +302,7 @@ in expr = slib.parseOption (evalType (lib.types.submodule subModule) { }); expected = { type = "object"; + additionalProperties = false; properties = { opt = { type = "boolean"; @@ -331,6 +333,7 @@ in type = "object"; additionalProperties = { type = "object"; + additionalProperties = false; properties = { opt = { type = "boolean"; @@ -363,6 +366,7 @@ in type = "array"; items = { type = "object"; + additionalProperties = false; properties = { opt = { type = "boolean"; diff --git a/lib/jsonschema/test_parseOptions.nix b/lib/jsonschema/test_parseOptions.nix index 1faf6e3b..9467160f 100644 --- a/lib/jsonschema/test_parseOptions.nix +++ b/lib/jsonschema/test_parseOptions.nix @@ -4,16 +4,9 @@ lib ? (import { }).lib, slib ? import ./. { inherit lib; }, }: -let - evaledOptions = - let - evaledConfig = lib.evalModules { modules = [ ./example-interface.nix ]; }; - in - evaledConfig.options; -in { testParseOptions = { - expr = slib.parseOptions evaledOptions; + expr = slib.parseModule ./example-interface.nix; expected = builtins.fromJSON (builtins.readFile ./example-schema.json); }; @@ -26,8 +19,10 @@ in { expr = slib.parseOptions evaled.options; expected = { + additionalProperties = false; properties = { foo = { + additionalProperties = false; properties = { bar = { type = "boolean"; diff --git a/pkgs/schemas/flake-module.nix b/pkgs/schemas/flake-module.nix index 9ba6f4ef..59c25d01 100644 --- a/pkgs/schemas/flake-module.nix +++ b/pkgs/schemas/flake-module.nix @@ -10,37 +10,21 @@ # borgbackup = self.clanModules.borgbackup; # }; - baseModule = { - imports = (import (pkgs.path + "/nixos/modules/module-list.nix")) ++ [ - { - nixpkgs.hostPlatform = "x86_64-linux"; - clan.core.clanName = "dummy"; - } - ]; - }; - optionsFromModule = - modulename: module: + mName: let - evaled = lib.evalModules { - modules = [ - module - baseModule - self.nixosModules.clanCore - ]; - }; + eval = self.lib.evalClanModules [ mName ]; in - # Filter out "injected" options that are not part of the module - if (evaled.options.clan ? "${modulename}") then evaled.options.clan.${modulename} else { }; + if (eval.options.clan ? "${mName}") then eval.options.clan.${mName} else { }; clanModuleSchemas = lib.mapAttrs ( - modulename: module: self.lib.jsonschema.parseOptions (optionsFromModule modulename module) + modulename: _: self.lib.jsonschema.parseOptions (optionsFromModule modulename) ) clanModules; - clanModuleFunctionSchemas = lib.mapAttrsFlatten (modulename: module: { + clanModuleFunctionSchemas = lib.mapAttrsFlatten (modulename: _: { name = modulename; description = self.lib.modules.getShortDescription modulename; - parameters = self.lib.jsonschema.parseOptions (optionsFromModule modulename module); + parameters = self.lib.jsonschema.parseOptions (optionsFromModule modulename); }) clanModules; in rec {