Skip to content

Commit d5cdcfd

Browse files
authored
Added support for overriding rule severity level #1180 (#2621)
1 parent 11852a9 commit d5cdcfd

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

59 files changed

+1320
-108
lines changed

README.md

+1
Original file line numberDiff line numberDiff line change
@@ -329,6 +329,7 @@ The following conceptual topics exist in the `PSRule` module:
329329
- [Output.Path](https://aka.ms/ps-rule/options#outputpath)
330330
- [Output.SarifProblemsOnly](https://aka.ms/ps-rule/options#outputsarifproblemsonly)
331331
- [Output.Style](https://aka.ms/ps-rule/options#outputstyle)
332+
- [Override.Level](https://aka.ms/ps-rule/options#overridelevel)
332333
- [Repository.BaseRef](https://aka.ms/ps-rule/options#repositorybaseref)
333334
- [Repository.Url](https://aka.ms/ps-rule/options#repositoryurl)
334335
- [Requires](https://aka.ms/ps-rule/options#requires)

docs/CHANGELOG-v3.md

+6
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,12 @@ See [upgrade notes][1] for helpful information when upgrading from previous vers
2727

2828
## Unreleased
2929

30+
- New features:
31+
- Added support for overriding rule severity level by @BernieWhite.
32+
[#1180](https://github.com/microsoft/PSRule/issues/1180)
33+
- Baselines now accept a new `spec.overrides.level` property which configures severity level overrides.
34+
- Options now accept a new `overrides.level` properties which configures severity level overrides.
35+
- For example, a rule that generates an `Error` can be overridden to `Warning`.
3036
- General improvements:
3137
- Automatically restore missing modules when running CLI by @BernieWhite.
3238
[#2552](https://github.com/microsoft/PSRule/issues/2552)

docs/concepts/PSRule/en-US/about_PSRule_Baseline.md

+5-2
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ A baseline includes a set of rule and configuration options that are used for ev
1414
The following baseline options can be configured:
1515

1616
- [Configuration](about_PSRule_Options.md#configuration)
17+
- [Override.Level](about_PSRule_Options.md#overridelevel)
1718
- [Rule.Include](about_PSRule_Options.md#ruleinclude)
1819
- [Rule.IncludeLocal](about_PSRule_Options.md#ruleincludelocal)
1920
- [Rule.Exclude](about_PSRule_Options.md#ruleexclude)
@@ -48,8 +49,9 @@ metadata:
4849
annotations: { }
4950
spec:
5051
# One or more baseline options
51-
rule: { }
5252
configuration: { }
53+
override: {}
54+
rule: { }
5355
```
5456
5557
For example:
@@ -100,8 +102,9 @@ To define a JSON baseline spec use the following structure:
100102
"annotations": {}
101103
},
102104
"spec": {
105+
"configuration": {},
106+
"override": {},
103107
"rule": {},
104-
"configuration": {}
105108
}
106109
}
107110
]

docs/concepts/PSRule/en-US/about_PSRule_Options.md

+56
Original file line numberDiff line numberDiff line change
@@ -68,6 +68,7 @@ The following workspace options are available for use:
6868
Additionally the following baseline options can be included:
6969

7070
- [Configuration](#configuration)
71+
- [Override.Level](#overridelevel)
7172
- [Rule.Baseline](#rulebaseline)
7273
- [Rule.Include](#ruleinclude)
7374
- [Rule.IncludeLocal](#ruleincludelocal)
@@ -3012,6 +3013,52 @@ variables:
30123013
value: 2
30133014
```
30143015

3016+
### Override.Level
3017+
3018+
This option is used to override the severity level of one or more rules.
3019+
When specified, the severity level of the rule will be set to the value specified.
3020+
Use this option to change the severity level of a rule to be different then originally defined by the author.
3021+
3022+
The following severity levels are available:
3023+
3024+
- `Error` - A serious problem that must be addressed before going forward.
3025+
- `Warning` - A problem that should be addressed.
3026+
- `Information` - A minor problem or an opportunity to improve the code.
3027+
3028+
This option can be specified using:
3029+
3030+
```powershell
3031+
# PowerShell: Using the OverrideLevel parameter
3032+
$option = New-PSRuleOption -OverrideLevel @{ 'rule1' = 'Information' };
3033+
```
3034+
3035+
```powershell
3036+
# PowerShell: Using the OVerride.Level hashtable key
3037+
$option = New-PSRuleOption -Option @{ 'Override.Level.rule1' = 'Information' };
3038+
```
3039+
3040+
```powershell
3041+
# PowerShell: Using the OverrideLevel parameter to set YAML
3042+
Set-PSRuleOption -OverrideLevel @{ 'rule1' = 'Information' };
3043+
```
3044+
3045+
```yaml
3046+
# YAML: Using the override/level property
3047+
override:
3048+
level:
3049+
rule1: Information
3050+
```
3051+
3052+
```bash
3053+
# Bash: Using environment variable
3054+
export PSRULE_OVERRIDE_LEVEL_RULE1='Information'
3055+
```
3056+
3057+
```powershell
3058+
# PowerShell: Using environment variable
3059+
$env:PSRULE_OVERRIDE_LEVEL_RULE1 = 'Information';
3060+
```
3061+
30153062
### Repository.BaseRef
30163063

30173064
This option is used for specify the base branch for pull requests.
@@ -3464,6 +3511,12 @@ output:
34643511
sarifProblemsOnly: false
34653512
style: GitHubActions
34663513
3514+
# Overrides the severity level for rules
3515+
override:
3516+
level:
3517+
Rule1: Error
3518+
Rule2: Warning
3519+
34673520
# Configure rule suppression
34683521
suppression:
34693522
storageAccounts.UseHttps:
@@ -3571,6 +3624,9 @@ output:
35713624
sarifProblemsOnly: true
35723625
style: Detect
35733626
3627+
override:
3628+
level: { }
3629+
35743630
# Configure rule suppression
35753631
suppression: { }
35763632

docs/concepts/options.md

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
# Options
2+
3+
Options are used to customize how rules are evaluated and the resulting output.
4+
You can set options in multiple ways, including:
5+
6+
- Parameters
7+
- Environment variables
8+
- Configuration files
9+
10+
Rules or modules could also have a defaults configured by the rule or module author.
11+
12+
## Option precedence
13+
14+
When setting options, you may have a situation where an option is set to different values.
15+
For example, you may set an option in a configuration file and also set the same option as a parameter.
16+
17+
When this happens, PSRule will use the option with the highest precedence.
18+
19+
Option precedence is as follows:
20+
21+
1. Parameters
22+
2. Explicit baselines
23+
3. Environment variables
24+
4. Configuration files
25+
5. Default baseline
26+
6. Module defaults

docs/concepts/sarif-format.md

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
# SARIF Output
2+
3+
PSRule uses a JSON structured output format called the
4+
5+
SARIF format to report results.
6+
The SARIF format is a standard format for the output of static analysis tools.
7+
The format is designed to be easily consumed by other tools and services.
8+
9+
## Runs
10+
11+
When running PSRule executed a run will be generated in `runs` containing details about PSRule and configuration details.
12+
13+
## Invocation
14+
15+
The `invocation` property reports runtime information about how the run started.
16+
17+
### RuleConfigurationOverrides
18+
19+
When a rule has been overridden in configuration this invocation property will contain any level overrides.

schemas/PSRule-language.schema.json

+33
Original file line numberDiff line numberDiff line change
@@ -209,6 +209,39 @@
209209
}
210210
},
211211
"additionalProperties": false
212+
},
213+
"override": {
214+
"type": "object",
215+
"title": "Overrides",
216+
"description": "Specifies additional rule overrides.",
217+
"properties": {
218+
"level": {
219+
"type": "object",
220+
"title": "Override severity level",
221+
"description": "Overrides the severity level defined by the rule to the level specified. A rule can be configured with one of the following: Error, Warning, Information.",
222+
"markdownDescription": "Overrides the severity level defined by the rule to the level specified.\n\nA rule can be configured with one of the following: `Error`, `Warning`, `Information`.\n\n[See help](https://microsoft.github.io/PSRule/v3/concepts/PSRule/en-US/about_PSRule_Options/#overridelevel)",
223+
"additionalProperties": {
224+
"type": "string",
225+
"title": "Override level by rule",
226+
"description": "Specify the new severity level of the rule. Choose one of the following supported values: Error, Warning, Information.",
227+
"markdownDescription": "Specify the new severity level of the rule.\n\nChoose one of the following supported values: `Error`, `Warning`, `Information`.\n\n[See help](https://microsoft.github.io/PSRule/v3/concepts/PSRule/en-US/about_PSRule_Options/#overridelevel)",
228+
"enum": [
229+
"Error",
230+
"Warning",
231+
"Information"
232+
]
233+
},
234+
"defaultSnippets": [
235+
{
236+
"label": "Override level",
237+
"body": {
238+
"${1:Rule}": "Error"
239+
}
240+
}
241+
]
242+
}
243+
},
244+
"additionalProperties": false
212245
}
213246
},
214247
"additionalProperties": false

schemas/PSRule-options.schema.json

+37
Original file line numberDiff line numberDiff line change
@@ -928,6 +928,39 @@
928928
},
929929
"additionalProperties": false
930930
},
931+
"override-option": {
932+
"type": "object",
933+
"title": "Overrides",
934+
"description": "Specifies additional rule overrides.",
935+
"properties": {
936+
"level": {
937+
"type": "object",
938+
"title": "Override severity level",
939+
"description": "Overrides the severity level defined by the rule to the level specified. A rule can be configured with one of the following: Error, Warning, Information.",
940+
"markdownDescription": "Overrides the severity level defined by the rule to the level specified.\n\nA rule can be configured with one of the following: `Error`, `Warning`, `Information`.\n\n[See help](https://microsoft.github.io/PSRule/v3/concepts/PSRule/en-US/about_PSRule_Options/#overridelevel)",
941+
"additionalProperties": {
942+
"type": "string",
943+
"title": "Override level by rule",
944+
"description": "Specify the new severity level of the rule. Choose one of the following supported values: Error, Warning, Information.",
945+
"markdownDescription": "Specify the new severity level of the rule.\n\nChoose one of the following supported values: `Error`, `Warning`, `Information`.\n\n[See help](https://microsoft.github.io/PSRule/v3/concepts/PSRule/en-US/about_PSRule_Options/#overridelevel)",
946+
"enum": [
947+
"Error",
948+
"Warning",
949+
"Information"
950+
]
951+
},
952+
"defaultSnippets": [
953+
{
954+
"label": "Override level",
955+
"body": {
956+
"${1:Rule}": "Error"
957+
}
958+
}
959+
]
960+
}
961+
},
962+
"additionalProperties": false
963+
},
931964
"options": {
932965
"properties": {
933966
"baseline": {
@@ -982,6 +1015,10 @@
9821015
"type": "object",
9831016
"$ref": "#/definitions/output-option"
9841017
},
1018+
"override": {
1019+
"type": "object",
1020+
"$ref": "#/definitions/override-option"
1021+
},
9851022
"repository": {
9861023
"type": "object",
9871024
"$ref": "#/definitions/repository-option"

src/PSRule.Types/Converters/TypeConverter.cs

+14
Original file line numberDiff line numberDiff line change
@@ -265,6 +265,20 @@ public static bool TryDouble(object o, bool convert, out double value)
265265
return false;
266266
}
267267

268+
/// <summary>
269+
/// Try to get the environment variable as a enum of type <typeparamref name="T"/>.
270+
/// </summary>
271+
public static bool TryEnum<T>(object o, bool convert, out T? value) where T : struct, Enum
272+
{
273+
if (o is T t || convert && o is string s && Enum.TryParse(s, ignoreCase: true, out t))
274+
{
275+
value = t;
276+
return true;
277+
}
278+
value = null;
279+
return false;
280+
}
281+
268282
private static bool TryGetValue(object o, string propertyName, out object? value)
269283
{
270284
value = null;

src/PSRule.Types/Data/EnumMap.cs

+72
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
// Copyright (c) Microsoft Corporation.
2+
// Licensed under the MIT License.
3+
4+
using System.Collections;
5+
using PSRule.Converters;
6+
7+
namespace PSRule.Data;
8+
9+
/// <summary>
10+
/// A mapping of string to string arrays.
11+
/// </summary>
12+
public sealed class EnumMap<T> : KeyMap<T> where T : struct, Enum
13+
{
14+
/// <summary>
15+
/// Create an empty <see cref="EnumMap{T}"/> instance.
16+
/// </summary>
17+
public EnumMap()
18+
: base() { }
19+
20+
/// <summary>
21+
/// Create an instance by copying an existing <see cref="EnumMap{T}"/>.
22+
/// </summary>
23+
internal EnumMap(EnumMap<T> map)
24+
: base(map) { }
25+
26+
/// <summary>
27+
/// Create an instance by copying mapped keys from a string dictionary.
28+
/// </summary>
29+
internal EnumMap(IDictionary<string, T> map)
30+
: base(map) { }
31+
32+
/// <summary>
33+
/// Create an instance by copying mapped keys from a <seealso cref="Hashtable"/>.
34+
/// </summary>
35+
/// <param name="map"></param>
36+
internal EnumMap(Hashtable map)
37+
: base(map) { }
38+
39+
/// <summary>
40+
///
41+
/// </summary>
42+
/// <param name="hashtable"></param>
43+
public static implicit operator EnumMap<T>(Hashtable hashtable)
44+
{
45+
return new EnumMap<T>(hashtable);
46+
}
47+
48+
/// <summary>
49+
/// Convert a hashtable into a <see cref="EnumMap{T}"/> instance.
50+
/// </summary>
51+
public static EnumMap<T> FromHashtable(Hashtable hashtable)
52+
{
53+
return new EnumMap<T>(hashtable);
54+
}
55+
56+
/// <summary>
57+
///
58+
/// </summary>
59+
/// <param name="o"></param>
60+
/// <param name="value"></param>
61+
/// <returns></returns>
62+
protected override bool TryConvertValue(object o, out T value)
63+
{
64+
value = default;
65+
if (TypeConverter.TryEnum<T>(o, convert: true, out var result) && result != null)
66+
{
67+
value = result.Value;
68+
return true;
69+
}
70+
return false;
71+
}
72+
}

0 commit comments

Comments
 (0)