Packaging rules in a module#
PSRule supports distribution of rules within modules. Using a module, rules can be published and installed using standard PowerShell cmdlets.
You should consider packaging rules into a module to:
- Version rules. PowerShell modules support semantic versioning (semver).
- Reuse rules across projects, pipelines or teams.
- Publish rules to external consumers via the PowerShell Gallery.
This scenario covers the following:
- Creating a module manifest
- Including rules and baselines
- Defining a module configuration
- Including documentation
Creating a module manifest#
When creating a PowerShell module, a module manifest is an optional file that stores module metadata.
Module manifests use the .psd1
file extension.
When packaging rules in a module, a module manifest is required for PSRule discover the module.
Creating the manifest file#
A module manifest can be created from PowerShell using the New-ModuleManifest
cmdlet.
Additionally, Visual Studio Code and many other tools also include snippets for creating a module manifest.
For example:
# Create a directory for the module
md ./Enterprise.Rules;
# Create the manifest
New-ModuleManifest -Path ./Enterprise.Rules/Enterprise.Rules.psd1 -Tags 'PSRule-rules';
The example above creates a module manifest for a module named Enterprise.Rules tagged with PSRule-rules
.
The use of the PSRule-rules
tag is explained in the following section.
Setting module tags#
When PSRule cmdlets are used with the -Module
parameter, PSRule discovers rule modules.
If the module is already imported, that module is used.
If the module is not imported, PSRule will import the highest version of the module automatically.
For a module to be discovered by PSRule, tag the module with PSRule-rules
.
To tag modules, find the Tags
section the PSData
hashtable in the module manifest and add PSRule-rules
.
An updated module manifest may look like this:
# Private data to pass to the module specified in RootModule/ModuleToProcess. This may also contain a PSData hashtable with additional module metadata used by PowerShell.
PrivateData = @{
PSData = @{
# Tags applied to this module. These help with module discovery in online galleries.
Tags = @('PSRule-rules')
}
}
Including rules and baselines#
Rules and baselines can be included anywhere within the module directory structure. Such as in the root directory of the module or in a nested sub-directory.
By convention, consider including rules and baselines within a rules
sub-directory within the module.
For example:
- Enterprise.Rules/
- rules/
- Baseline.Rule.yaml
- Config.Rule.yaml
- Standards.Rule.ps1
- Enterprise.Rules.psd1
- rules/
File names#
For PSRule to find rules included in a module, rule file names must end with the .Rule.ps1
suffix.
We recommend using the exact case .Rule.ps1
.
This is because some file systems are case-sensitive.
For example, on Linux Standards.rule.ps1
would be ignored by PSRule.
Similarly, when including baselines within a module use the .Rule.yaml
suffix.
Defining a module configuration#
A module configuration that sets options defaults and can be optionally packaged with a module.
To set a module configuration, define a ModuleConfig
resource within an included .Rule.yaml
file.
A module configuration .Rule.yaml
file must be distributed within the module directory structure.
PSRule only supports a single ModuleConfig
resource.
The name of the ModuleConfig
must match the name of the module.
Additional ModuleConfig
resources or with an alternative name are ignored.
PSRule does not support module configurations distributed outside of a module.
Example
---
# Synopsis: Example module configuration for Enterprise.Rules module.
apiVersion: github.com/microsoft/PSRule/2025-01-01
kind: ModuleConfig
metadata:
name: Enterprise.Rules
spec:
binding:
targetName:
- ResourceName
- FullName
- name
targetType:
- ResourceType
- type
- Extension
field:
resourceId: [ 'ResourceId' ]
subscriptionId: [ 'SubscriptionId' ]
resourceGroupName: [ 'ResourceGroupName' ]
rule:
baseline: Enterprise.Default
The following options are allowed within a ModuleConfig
:
Binding.Field
Binding.IgnoreCase
Binding.NameSeparator
Binding.PreferTargetInfo
Binding.TargetName
Binding.TargetType
Binding.UseQualifiedName
Configuration
Output.Culture
Rule.Baseline
Setting a default baseline#
Optionally, baselines can be included in rule modules. If a baseline contains configuration or binding options then setting a default baseline is often desirable. When a default baseline is set, PSRule will use the named baseline automatically when processing rules from that module. This feature removes the need for users to specify it manually.
To set a default baseline, set the Rule.Baseline
property of the ModuleConfig
resource.
Example
---
# Synopsis: Example module configuration for Enterprise.Rules module.
apiVersion: github.com/microsoft/PSRule/2025-01-01
kind: ModuleConfig
metadata:
name: Enterprise.Rules
spec:
binding:
targetName:
- ResourceName
- FullName
- name
targetType:
- ResourceType
- type
- Extension
field:
resourceId: [ 'ResourceId' ]
subscriptionId: [ 'SubscriptionId' ]
resourceGroupName: [ 'ResourceGroupName' ]
rule:
baseline: Enterprise.Default
This examples set the default baseline to Enterprise.Default
.
The default baseline must be included in file ending with .Rule.yaml
within the module directory structure.
Including documentation#
PSRule supports write and packaging rule modules with markdown documentation. Markdown documentation is automatically interpreted by PSRule and included in output.
When including markdown, files are copied into a directory structure based on the target culture.
For example, store documentation targeted to the culture en-US
in a directory named en-US
.
Similarly, documentation for cultures such as en-AU
, en-GB
and fr-FR
would be in separate directories.
If a directory for the exact culture en-US
doesn't exist, PSRule will attempt to find the parent culture.
For example, documentation would be read from a directory named en
.
When naming directories for their culture, use exact case.
This is because some file systems are case-sensitive.
For example on Linux en-us
would not match.
For example:
- Enterprise.Rules/
- en/
- Org.Az.Storage.UseHttps.md
- Org.Az.Resource.Tagging.md
- en-US/
- Org.Az.Storage.UseHttps.md
- fr-FR/
- Org.Az.Storage.UseHttps.md
- rules/
- Baseline.Rule.yaml
- Config.Rule.yaml
- Standards.Rule.ps1
- Enterprise.Rules.psd1
- en/
More information#
- Enterprise.Rules.psd1 - An example module manifest.
- Baseline.Rule.yaml - An example baseline.
- Config.Rule.yaml - An example module configuration.