diff --git a/src/functions/Mock.ps1 b/src/functions/Mock.ps1 index 7f299b532..0526a4bd1 100644 --- a/src/functions/Mock.ps1 +++ b/src/functions/Mock.ps1 @@ -1,4 +1,4 @@ - + function Add-MockBehavior { [CmdletBinding()] @@ -1457,13 +1457,50 @@ function Get-MockDynamicParameter { switch ($PSCmdlet.ParameterSetName) { 'Cmdlet' { - Get-DynamicParametersForCmdlet -CmdletName $CmdletName -Parameters $Parameters + $dynamicParams = Get-DynamicParametersForCmdlet -CmdletName $CmdletName -Parameters $Parameters } 'Function' { - Get-DynamicParametersForMockedFunction -DynamicParamScriptBlock $DynamicParamScriptBlock -Parameters $Parameters -Cmdlet $Cmdlet + $dynamicParams = Get-DynamicParametersForMockedFunction -DynamicParamScriptBlock $DynamicParamScriptBlock -Parameters $Parameters -Cmdlet $Cmdlet + } + } + + if ($null -eq $dynamicParams) { + return + } + + Repair-ConflictingDynamicParameters -DynamicParams $dynamicParams +} + +function Repair-ConflictingDynamicParameters { + [OutputType([System.Management.Automation.RuntimeDefinedParameterDictionary])] + param ( + [Parameter(Mandatory = $true)] + [System.Management.Automation.RuntimeDefinedParameterDictionary] + $DynamicParams + ) + + $repairedDynamicParams = New-Object System.Management.Automation.RuntimeDefinedParameterDictionary + $conflictingParams = Get-ConflictingParameterNames + + foreach ($paramName in $DynamicParams.Keys) { + $dynamicParam = $DynamicParams[$paramName] + + if ($conflictingParams -contains $paramName) { + $newName = "_$paramName" + $dynamicParam.Name = $newName + + $aliasAttribute = New-Object System.Management.Automation.AliasAttribute -ArgumentList $paramName + $dynamicParam.Attributes.Add($aliasAttribute) + + $repairedDynamicParams[$newName] = $dynamicParam + } + else { + $repairedDynamicParams[$paramName] = $dynamicParam } } + + return $repairedDynamicParams } function Get-DynamicParametersForCmdlet { diff --git a/tst/functions/Mock.Tests.ps1 b/tst/functions/Mock.Tests.ps1 index b3853dca6..8982b900f 100644 --- a/tst/functions/Mock.Tests.ps1 +++ b/tst/functions/Mock.Tests.ps1 @@ -1,4 +1,4 @@ -Set-StrictMode -Version Latest +Set-StrictMode -Version Latest BeforeAll { $PSDefaultParameterValues = @{ 'Should:ErrorAction' = 'Stop' } function FunctionUnderTest { @@ -3160,3 +3160,72 @@ Describe 'Mocking with nested Pester runs' { Get-Command Get-ChildItem | Should -Not -Be 2 } } + +Describe 'Usage of Alias in DynamicParams' { + # https://github.com/pester/Pester/issues/1274 + + BeforeAll { + function New-DynamicAttr($ParamDictionary, $Name, $Alias = $null) { + $attr = New-Object -Type ` + System.Management.Automation.ParameterAttribute + $attr.Mandatory = $false + $attr.ParameterSetName = '__AllParameterSets' + $attributeCollection = New-Object ` + -Type System.Collections.ObjectModel.Collection[System.Attribute] + $attributeCollection.Add($attr) + + if ($null -ne $Alias) { + $attr = New-Object -Type ` + System.Management.Automation.AliasAttribute -ArgumentList @($Alias) + $attributeCollection.Add($attr) + } + + $dynParam1 = New-Object -Type ` + System.Management.Automation.RuntimeDefinedParameter($Name, [string], + $attributeCollection) + + $ParamDictionary.Add($Name, $dynParam1) + } + + function Test-DynamicParam { + [CmdletBinding()] + param( + [String]$Name + ) + + dynamicparam { + if ($Name.StartsWith("Hello")) { + $paramDictionary = New-Object ` + -Type System.Management.Automation.RuntimeDefinedParameterDictionary + New-DynamicAttr -ParamDictionary $paramDictionary -Name "PSEdition" + + return $paramDictionary + } + } + + process { + if ($PSBoundParameters.PSEdition) { + Write-Host "PSEdition value: $($PSBoundParameters.PSEdition)" + } + } + } + } + + Context 'Mocking with ParameterFilter' { + It 'Mocks Test-DynamicParam with PSEdition set to Desktop' { + Mock Test-DynamicParam { "World" } -ParameterFilter { $PSEdition -eq 'Desktop' } + + Test-DynamicParam -Name "Hello" -PSEdition 'Desktop' | Should -Be 'World' + } + } + + Context 'Validating Mock Invocation' { + It 'Invokes Test-DynamicParam with correct parameters' { + Mock Test-DynamicParam { "World" } + + Test-DynamicParam -Name "Hello" -PSEdition 'Desktop' | Should -Be 'World' + + Should -Invoke Test-DynamicParam -Exactly 1 -Scope It + } + } +}