windows/Tests/PSRemoteOperations.Tests.ps1
# run this test under Windows PowerShell [Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSAvoidUsingComputerNameHardcoded', '')] [Diagnostics.CodeAnalysis.SuppressMessageAttribute('PSAvoidUsingConvertToSecureStringwithPlainText', '')] Param() $moduleName = (($MyInvocation.mycommand).name -split "\.")[0] $ModuleManifestName = "$modulename.psd1" $ModuleManifestPath = "$PSScriptRoot\..\..\$ModuleManifestName" If (Get-Module $modulename) { Remove-module $moduleName } import-module $ModuleManifestPath -Force Describe $ModuleName { $myModule = Test-ModuleManifest -Path $ModuleManifestPath Context Manifest { It 'Passes Test-ModuleManifest' { $myModule | Should Not BeNullOrEmpty } It "Should NOT have a root module" { $myModule.RootModule | Should BeNullOrEmpty } It "Contains exported commands" { $myModule.ExportedCommands | Should Not BeNullOrEmpty } It "Should have a module description" { $myModule.Description | Should Not BeNullOrEmpty } It "Should have a company" { $myModule.Description | Should Not BeNullOrEmpty } It "Should have an author of 'Jeff Hicks'" { $mymodule.Author | Should Be 'Jeff Hicks' } It "Should have tags" { $mymodule.Tags | Should Not BeNullOrEmpty } It "Should have a license URI" { $mymodule.LicenseUri | should Not BeNullOrEmpty } It "Should have a project URI" { $mymodule.ProjectUri | Should Not BeNullOrEmpty } } Context Exports { $exported = Get-Command -Module $ModuleName -CommandType Function $names = 'New-PSRemoteOperation', 'Invoke-PSRemoteOperation', 'Get-PSRemoteOperationResult', 'Register-PSRemoteOperationWatcher', 'Wait-PSRemoteOperation','New-PSRemoteOperationForm','Get-PSRemoteOperation', 'Import-PSRemoteOpPath','Register-PSRemoteOpPath' It "Should export $($names.count) functions" { $names.Count -eq $exported.count | Should be $True } foreach ($name in $names) { It "Should have an exported command of $name" { $exported.name | Should Contain $name } } $aliasHash = @{ nro = "New-PSRemoteOperation" iro = "Invoke-PSRemoteOperation" row = "Register-PSRemoteOperationWatcher" gro = "Get-PSRemoteOperationResult" nrof = "New-PSRemoteOperationForm" grop = "Get-PSRemoteOperation" } $aliasHash.GetEnumerator() | foreach-object { It "Should have an alias of $($_.key)" { (Get-Alias -Name $_.key).ResolvedCommandName | Should be $_.value } } It "Should create a RemoteOpResults type extension" { Get-TypeData RemoteOpResult | Should be $True } } Context Structure { It "Should have a Docs folder" { Get-Item $PSScriptRoot\..\..\docs | Should Be $True } foreach ($cmd in $myModule.ExportedFunctions.keys) { It "Should have a markdown help file for $cmd" { "$PSScriptRoot\..\..\docs\$cmd.md" | Should Exist } } It "Should have an external help file" { "$PSScriptRoot\..\..\en-us\*.xml" | Should Exist "$PSScriptRoot\..\..\en-us\*.txt" | Should Exist } It "Should have an about file" { "$PSScriptRoot\..\..\docs\about_$ModuleName.md"| Should Exist } It "Should have a license file" { "$PSScriptRoot\..\..\license.*" | Should Exist } It "Should have a changelog file" { "$PSScriptRoot\..\..\changelog*" | Should Exist } It "Should have a README.md file" { "$PSScriptRoot\..\..\README.md" | Should Exist } } } -Tag module InModuleScope PSRemoteOperations { Describe New-PSRemoteOperation { $params = (Get-Command New-PSRemoteOperation).parameters It "Should have a mandatory Computername parameter" { ($params["computername"].attributes).where( {$_.TypeId -match 'parameter'}).Mandatory | Should Be $True } It "Should have a mandatory ScriptBlock or ScriptPath parameter" { ($params["scriptblock"].attributes).where( {$_.TypeId -match 'parameter'}).Mandatory | Should Be $True ($params["scriptpath"].attributes).where( {$_.TypeId -match 'parameter'}).Mandatory | Should Be $True } It "Should create a psd1 file" { {New-PSRemoteOperation -Computername Foo -Path TestDrive:\ -Scriptblock {1}} | Should Not Throw Test-Path TestDrive:\foo*.psd1 | Should be $True #Should have a filename using the pattern computername_guid.psd1" $guidrx = "[0-9A-Fa-f]{8}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{12}" $template = "foo_$guidrx.psd1" (Get-Item Testdrive:\*.psd1).name -match $Template | Should be $True #get the file contents to use in the next set of tests $script:in = Import-PowerShellDataFile Testdrive:\foo_*.psd1 } It "Should create a file with a Computername value of Foo" { $script:in.Computername | Should Be "Foo" } It "Should have a Scriptblock value of 1" { $script:in.Scriptblock | Should be 1 } It "Should have a UTC CreatedAt value" { $script:in.CreatedAt -match "UTC" | Should be $true $script:in.CreatedAt -match $(get-date -format MM/dd/yyyy) | Should be $true } It "Should accept multiple computernames" { $files = New-PSRemoteOperation -Computername Foo1, Foo2, Foo3 -Path TestDrive:\ -Scriptblock {Get-Volume C:} -Passthru $files.count | Should Be 3 } } -tag command Describe Invoke-PSRemoteOperation { New-PSRemoteOperation -Computername $env:computername -Path TestDrive: -Scriptblock {Get-Service bits} New-Item -path TestDrive:\ -Name Archive -ItemType directory Context Input { $params = (Get-Command Invoke-PSRemoteOperation).parameters It "Should have a mandatory Path parameter ending in .psd1" { ($params["Path"].attributes).where( {$_.TypeId -match 'parameter'}).Mandatory | Should Be $True ($params["Path"].attributes).where( {$_.TypeId -match 'pattern'}).regexPattern | Should Be '\.psd1$' } } Context Process { $file = (Get-childitem testdrive:\*.psd1).fullname Mock 'Get-Date' {[datetime]::new(2019, 1, 1)} #I'd like to mock this but there doesn't appear to be a way #Mock New-PSSession {} Mock Convert-Path { $file } Invoke-PSRemoteOperation -Path $file -ArchivePath testdrive:\archive #-Verbose It "Should call Convert-Path" { Assert-MockCalled Convert-Path } It "Should call Get-Date" { Assert-MockCalled Get-Date } #save result for the next set of tests $script:result = Get-ChildItem TestDrive:\archive\*.psd1 $script:rcontent = get-item $script:result | get-content $script:rdata = Import-powershelldatafile $script:result } Context Output { It "Should create an archive file" { $script:result | Should Not BeNullOrEmpty } It "Should show Completed as True" { $script:rdata.Completed | Should be 'True' } It "Should show the local computername" { $script:rdata.Computername | Should be $env:computername } It "Should show the scriptblock" { $script:rdata.Scriptblock.trim() | Should be "Get-Service bits" } It "Should show a date of 01/01/2019" { $script:rdata.Date -match "01\/01\/2019\s(\d{2}:){2}00\sUTC" | Should be $True } } } -tag command Describe Get-PSRemoteOperationResult { #copy the results from an earlier test to avoid duplication New-Item -path TestDrive: -Name Archive -ItemType directory $fake = Join-Path -Path TestDrive:\Archive -ChildPath "$($env:computername)_$(New-Guid).psd1" $script:rcontent | Out-File -FilePath $fake $r = Get-PSRemoteOperationResult -ArchivePath $fake It "Should get a result" { $fake | Should Exist $r | Should Not BeNullOrEmpty } It "Should write a RemoteOpResult object" { $r.psobject.typenames[0] | Should Be "RemoteOpResult" } It "Should get an array of strings when using -RAW" { $r = Get-PSRemoteOperationResult -ArchivePath $fake -Raw $r | Should BeOftype "String" $r -is [array] | Should Be True } } -tag command Describe Register-PSRemoteOperationWatcher { New-Item -path TestDrive: -Name Archive -ItemType directory Mock New-JobTrigger { [Microsoft.PowerShell.ScheduledJob.ScheduledJobTrigger]::new()} Mock Register-ScheduledJob {} #create a fake credential for the pester test $cred = New-Object PSCredential foo, (ConvertTo-SecureString "password" -AsPlainText -Force) $testparams = @{ Credential = $cred Name = 'TestWatch' Path = 'Testdrive:\' ArchivePath = 'Testdrive:\Archive' } Register-PSRemoteOperationWatcher @testparams It "Should call New-Jobtrigger" { Assert-MockCalled New-JobTrigger } it "Should call Register-ScheduledJob" { Assert-MockCalled Register-ScheduledJob } } -tag command } |