Skip to content

Commit 0f3c032

Browse files
authoredMar 8, 2024
Merge pull request #357 from SteveL-MSFT/dscresource-caps
Add capabilities to DscResource schema
2 parents 569f2ea + e109519 commit 0f3c032

File tree

7 files changed

+56
-11
lines changed

7 files changed

+56
-11
lines changed
 

‎dsc/src/subcommand.rs

+8-8
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ use dsc_lib::dscresources::invoke_result::{
1414
use dsc_lib::{
1515
DscManager,
1616
dscresources::invoke_result::ValidateResult,
17-
dscresources::dscresource::{ImplementedAs, Invoke},
17+
dscresources::dscresource::{Capability, ImplementedAs, Invoke},
1818
dscresources::resource_manifest::{import_manifest, ResourceManifest},
1919
};
2020
use serde_yaml::Value;
@@ -400,13 +400,17 @@ pub fn resource(subcommand: &ResourceSubCommand, stdin: &Option<String>) {
400400
ResourceSubCommand::List { resource_name, description, tags, format } => {
401401

402402
let mut write_table = false;
403-
let mut table = Table::new(&["Type", "Kind", "Version", "Methods", "Requires", "Description"]);
403+
let mut table = Table::new(&["Type", "Kind", "Version", "Caps", "Requires", "Description"]);
404404
if format.is_none() && atty::is(Stream::Stdout) {
405405
// write as table if format is not specified and interactive
406406
write_table = true;
407407
}
408408
for resource in dsc.list_available_resources(&resource_name.clone().unwrap_or_default()) {
409-
let mut methods = "g---".to_string();
409+
let mut capabilities = "g---".to_string();
410+
if resource.capabilities.contains(&Capability::Set) { capabilities.replace_range(1..2, "s"); }
411+
if resource.capabilities.contains(&Capability::Test) { capabilities.replace_range(2..3, "t"); }
412+
if resource.capabilities.contains(&Capability::Export) { capabilities.replace_range(3..4, "e"); }
413+
410414
// if description, tags, or write_table is specified, pull resource manifest if it exists
411415
if let Some(ref resource_manifest) = resource.manifest {
412416
let manifest = match import_manifest(resource_manifest.clone()) {
@@ -438,10 +442,6 @@ pub fn resource(subcommand: &ResourceSubCommand, stdin: &Option<String>) {
438442
}
439443
if !found { continue; }
440444
}
441-
442-
if manifest.set.is_some() { methods.replace_range(1..2, "s"); }
443-
if manifest.test.is_some() { methods.replace_range(2..3, "t"); }
444-
if manifest.export.is_some() { methods.replace_range(3..4, "e"); }
445445
} else {
446446
// resource does not have a manifest but filtering on description or tags was requested - skip such resource
447447
if description.is_some() || tags.is_some() {
@@ -454,7 +454,7 @@ pub fn resource(subcommand: &ResourceSubCommand, stdin: &Option<String>) {
454454
resource.type_name,
455455
format!("{:?}", resource.kind),
456456
resource.version,
457-
methods,
457+
capabilities,
458458
resource.requires.unwrap_or_default(),
459459
resource.description.unwrap_or_default()
460460
]);

‎dsc/tests/dsc_resource_list.tests.ps1

+8
Original file line numberDiff line numberDiff line change
@@ -53,4 +53,12 @@ Describe 'Tests for listing resources' {
5353
$resources.type | Should -BeExactly $expectedType
5454
}
5555
}
56+
57+
It 'Capabilities are returned' {
58+
$resource = dsc resource list Microsoft/OSInfo | ConvertFrom-Json
59+
$LASTEXITCODE | Should -Be 0
60+
$resource.capabilities.Count | Should -Be 2
61+
$resource.capabilities | Should -Contain 'Get'
62+
$resource.capabilities | Should -Contain 'Export'
63+
}
5664
}

‎dsc_lib/src/discovery/command_discovery.rs

+14-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
// Licensed under the MIT License.
33

44
use crate::discovery::discovery_trait::ResourceDiscovery;
5-
use crate::dscresources::dscresource::{DscResource, ImplementedAs};
5+
use crate::dscresources::dscresource::{Capability, DscResource, ImplementedAs};
66
use crate::dscresources::resource_manifest::{import_manifest, Kind, ResourceManifest};
77
use crate::dscresources::command_resource::invoke_command;
88
use crate::dscerror::DscError;
@@ -281,12 +281,25 @@ fn load_manifest(path: &Path) -> Result<DscResource, DscError> {
281281
Kind::Resource
282282
};
283283

284+
// all command based resources are required to support `get`
285+
let mut capabilities = vec![Capability::Get];
286+
if manifest.set.is_some() {
287+
capabilities.push(Capability::Set);
288+
}
289+
if manifest.test.is_some() {
290+
capabilities.push(Capability::Test);
291+
}
292+
if manifest.export.is_some() {
293+
capabilities.push(Capability::Export);
294+
}
295+
284296
let resource = DscResource {
285297
type_name: manifest.resource_type.clone(),
286298
kind,
287299
implemented_as: ImplementedAs::Command,
288300
description: manifest.description.clone(),
289301
version: manifest.version.clone(),
302+
capabilities,
290303
path: path.to_str().unwrap().to_string(),
291304
directory: path.parent().unwrap().to_str().unwrap().to_string(),
292305
manifest: Some(serde_json::to_value(manifest)?),

‎dsc_lib/src/dscresources/dscresource.rs

+11
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,8 @@ pub struct DscResource {
2121
pub kind: Kind,
2222
/// The version of the resource.
2323
pub version: String,
24+
/// The capabilities of the resource.
25+
pub capabilities: Vec<Capability>,
2426
/// The file path to the resource.
2527
pub path: String,
2628
/// The description of the resource.
@@ -40,6 +42,14 @@ pub struct DscResource {
4042
pub manifest: Option<Value>,
4143
}
4244

45+
#[derive(Clone, Debug, PartialEq, Deserialize, Serialize, JsonSchema)]
46+
pub enum Capability {
47+
Get,
48+
Set,
49+
Test,
50+
Export,
51+
}
52+
4353
#[derive(Clone, Debug, PartialEq, Deserialize, Serialize, JsonSchema)]
4454
#[serde(untagged)]
4555
pub enum ImplementedAs {
@@ -56,6 +66,7 @@ impl DscResource {
5666
type_name: String::new(),
5767
kind: Kind::Resource,
5868
version: String::new(),
69+
capabilities: Vec::new(),
5970
description: None,
6071
path: String::new(),
6172
directory: String::new(),

‎powershell-adapter/powershell.resource.ps1

+10-1
Original file line numberDiff line numberDiff line change
@@ -106,12 +106,21 @@ if ($Operation -eq 'List')
106106

107107
$fullResourceTypeName = "$moduleName/$($r.ResourceType)"
108108
$script:ResourceCache[$fullResourceTypeName] = $r
109-
if ($WinPS) {$requiresString = "Microsoft.Windows/WindowsPowerShell"} else {$requiresString = "Microsoft.DSC/PowerShell"}
109+
if ($WinPS) {$requiresString = "Microsoft.Windows/WindowsPowerShell"} else {$requiresString = "Microsoft.DSC/PowerShell"}
110+
111+
$t = [Type]$r.ResourceType
112+
$exportMethod = $t.GetMethod('Export')
113+
114+
$capabilities = @('Get', 'Set', 'Test')
115+
if ($null -ne $exportMethod) {
116+
$capabilities += 'Export'
117+
}
110118

111119
$z = [pscustomobject]@{
112120
type = $fullResourceTypeName;
113121
kind = 'Resource';
114122
version = $version_string;
123+
capabilities = $capabilities;
115124
path = $r.Path;
116125
directory = $r.ParentPath;
117126
implementedAs = $r.ImplementationDetail;

‎tools/test_group_resource/src/main.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ mod args;
66
use args::{Args, SubCommand};
77
use clap::Parser;
88
use dsc_lib::dscresources::resource_manifest::{ResourceManifest, GetMethod, Kind};
9-
use dsc_lib::dscresources::dscresource::{DscResource, ImplementedAs};
9+
use dsc_lib::dscresources::dscresource::{Capability, DscResource, ImplementedAs};
1010

1111
fn main() {
1212
let args = Args::parse();
@@ -16,6 +16,7 @@ fn main() {
1616
type_name: "Test/TestResource1".to_string(),
1717
kind: Kind::Resource,
1818
version: "1.0.0".to_string(),
19+
capabilities: vec![Capability::Get, Capability::Set],
1920
description: Some("This is a test resource.".to_string()),
2021
implemented_as: ImplementedAs::Custom("TestResource".to_string()),
2122
path: "test_resource1".to_string(),
@@ -48,6 +49,7 @@ fn main() {
4849
type_name: "Test/TestResource2".to_string(),
4950
kind: Kind::Resource,
5051
version: "1.0.1".to_string(),
52+
capabilities: vec![Capability::Get, Capability::Set],
5153
description: Some("This is a test resource.".to_string()),
5254
implemented_as: ImplementedAs::Custom("TestResource".to_string()),
5355
path: "test_resource2".to_string(),
@@ -84,6 +86,7 @@ fn main() {
8486
type_name: "InvalidResource".to_string(),
8587
kind: Kind::Resource,
8688
version: "1.0.0".to_string(),
89+
capabilities: vec![Capability::Get],
8790
description: Some("This is a test resource.".to_string()),
8891
implemented_as: ImplementedAs::Custom("TestResource".to_string()),
8992
path: "test_resource1".to_string(),

‎wmi-adapter/wmi.resource.ps1

+1
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,7 @@ if ($Operation -eq 'List')
4949
type = $fullResourceTypeName;
5050
kind = 'Resource';
5151
version = $version_string;
52+
capabilities = @('Get');
5253
path = "";
5354
directory = "";
5455
implementedAs = "";

0 commit comments

Comments
 (0)
Please sign in to comment.