I have been developing a nested dictionary data structure to build a list of files to work with later in the code. Currently I am using this successfully to create the empty data structure.
$manageLocalAssets = @{
resources = @{
extra = New-Object Collections.ArrayList
new = New-Object Collections.ArrayList
skip = New-Object Collections.ArrayList
update = New-Object Collections.ArrayList
}
definitions = @{
extra = New-Object Collections.ArrayList
new = New-Object Collections.ArrayList
skip = New-Object Collections.ArrayList
update = New-Object Collections.ArrayList
}
}
This works, but I would like to use an Ordered Dictionary rather than Hash Tables, so I can loop through the data in the order defined. Also, I need to support PS2.0 (don't ask) so I can't use the [ordered] type.
I have tried
[Collections.Specialized.OrderedDictionary]$manageLocalAssets = @{
resources = @{
extra = New-Object Collections.ArrayList
new = New-Object Collections.ArrayList
skip = New-Object Collections.ArrayList
update = New-Object Collections.ArrayList
}
definitions = @{
extra = New-Object Collections.ArrayList
new = New-Object Collections.ArrayList
skip = New-Object Collections.ArrayList
update = New-Object Collections.ArrayList
}
}
and I have tried
$manageLocalAssets = [Collections.Specialized.OrderedDictionary]@{
resources = @{
extra = New-Object Collections.ArrayList
new = New-Object Collections.ArrayList
skip = New-Object Collections.ArrayList
update = New-Object Collections.ArrayList
}
definitions = @{
extra = New-Object Collections.ArrayList
new = New-Object Collections.ArrayList
skip = New-Object Collections.ArrayList
update = New-Object Collections.ArrayList
}
}
Both fail to cast the hash table to an ordered dictionary. Is my only option something ugly (IMO) like this
$manageLocalAssets = New-Object Collections.Specialized.OrderedDictionary
$manageLocalAssets.Add('resources', (New-Object Collections.Specialized.OrderedDictionary))
$manageLocalAssets.resources.Add('extra', (New-Object Collections.ArrayList))
[ss64][1] has me thinking -property might be the answer, and this seems to be close
$manageLocalAssets = New-Object Collections.Specialized.OrderedDictionary -property @{
resources = New-Object Collections.Specialized.OrderedDictionary -property @{
extra = New-Object Collections.ArrayList
new = New-Object Collections.ArrayList
skip = New-Object Collections.ArrayList
update = New-Object Collections.ArrayList
}
definitions = New-Object Collections.Specialized.OrderedDictionary -property @{
extra = New-Object Collections.ArrayList
new = New-Object Collections.ArrayList
skip = New-Object Collections.ArrayList
update = New-Object Collections.ArrayList
}
}
But it fails to create the nested keys, extra, skip, etc. Am I going in the right direction, or am I better off just admitting a bunch of separate New-Object and .Add lines are the way to go?
EDIT #1: I had a thought, since at the deepest level the data structure is identical, I tried this
$assets = New-Object Collections.Specialized.OrderedDictionary -property @{
extra = New-Object Collections.ArrayList
new = New-Object Collections.ArrayList
skip = New-Object Collections.ArrayList
update = New-Object Collections.ArrayList
}
$manageLocalAssets = New-Object Collections.Specialized.OrderedDictionary -property @{
resources = $assets
definitions = $assets
}
and the odd things is, the error is
New-Object : The member "skip" was not found for the specified .NET object.
At \\Mac\Support\Px Tools\Dev 4.0\Resources\PxContext_Machine.ps1:413 char:15
+ ... $assets = New-Object Collections.Specialized.OrderedDictionary -pro ...
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+ CategoryInfo : InvalidOperation: (:) [New-Object], InvalidOperationException
+ FullyQualifiedErrorId : InvalidOperationException,Microsoft.PowerShell.Commands.NewObjectCommand
New-Object : The member "resources" was not found for the specified .NET object.
At \\Mac\Support\Px Tools\Dev 4.0\Resources\PxContext_Machine.ps1:419 char:26
+ ... calAssets = New-Object Collections.Specialized.OrderedDictionary -pro ...
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+ CategoryInfo : InvalidOperation: (:) [New-Object], InvalidOperationException
+ FullyQualifiedErrorId : InvalidOperationException,Microsoft.PowerShell.Commands.NewObjectCommand
[1]: https://ss64.com/ps/new-object.html
413 is the $assets = New-Object Collections.Specialized.OrderedDictionary -property @{
line, while
419 is the $manageLocalAssets = New-Object Collections.Specialized.OrderedDictionary -property @{
line.
So, it seems like extra and new are being handled at first, then skip fails in the first assignment, and resources fails right off the bat in the second assignment.
EDIT #2: So, breaking out the repeating part as a different variable lead me to this
$assets = New-Object Collections.Specialized.OrderedDictionary
$assets.Add('extra', (New-Object Collections.ArrayList))
$assets.Add('new', (New-Object Collections.ArrayList))
$assets.Add('skip', (New-Object Collections.ArrayList))
$assets.Add('update', (New-Object Collections.ArrayList))
$manageLocalAssets = New-Object Collections.Specialized.OrderedDictionary
$manageLocalAssets.Add('resources', $assets)
$manageLocalAssets.Add('definitions', $assets)
Which is actually fewer lines, and will be even more efficient as I add to $manageLocalAssets. It doesn't look visually like my other data structure initializations, but I guess in time I will get used to that. And it lets me use a loop structure like this to work with the data in the desired order.
foreach ($asset in $manageLocalAssets.keys) {
foreach ($key in ($manageLocalAssets.$asset).keys) {
Write-Host "$asset $key"
foreach ($file in $manageLocalAssets.$asset.$key) {
Write-Host " $file"
}
}
}
Not "ideal", but then ideal here means "familiar", and that's not a great reason to make decisions.
EDIT #3: Nope. I was wrong. This actually makes the content of both $manageLocalAssets.resources
and $manageLocalAssets.definitions
the same, the $assets variable. Seems I am still looking.
I suppose backward-compatibility is never pretty? This is the best I could come up with, coming somewhat close to a structured definition (using a simple array):
Of course, you could put the actual values (the ArrayLists) in the array definition too, but I suppose that would just be a lot more to write and not look so pretty.
Alternative solution, using a function to create the ordered dictionary with variable levels from a definition. Somewhat more verbose but re-usable: