我正在尝试通过Powershell将文件从本地计算机传输到远程计算机。我正在运行Powershell 3.0版,但无法升级Powershell。
我必须具有指定凭据的远程会话。
下面是我正在使用的代码:
# $credential variable created and set elsewhere
$command = {
param( [System.IO.FileStream]$inputStream, [string]$filePath, [string]$fileName )
$writeStream = [System.IO.File]::Create( "$filePath\$fileName" );
$inputStream.Seek(0, [System.IO.SeekOrigin]::Begin);
$inputStream.CopyTo( $writeStream );
$writeStream.Close();
$writeStream.Dispose();
}
$readStream = [System.IO.File]::OpenRead( "C:\temp\tempfile.txt" );
$path = "C:\temp";
$file = "test.txt";
$session = New-PSSession -ComputerName RemoteMachineName -UseSSL -Credential $credential;
# this works when copying the file/stream locally
Invoke-Command -ArgumentList $readStream, $path, $file -ScriptBlock $command
# this does not work when copying the file/stream remotely
Invoke-Command -Session $session -ArgumentList $readStream, $path, $file -ScriptBlock $command
$readStream.Close();
$readStream.Dispose();
上面的代码在尝试本地复制此文件时有效。但是,当尝试通过会话将此文件复制到远程计算机时,出现以下错误:
无法处理参数'inputStream'上的参数转换。无法将类型为“ Deserialized.System.IO.FileStream”的“ System.IO.FileStream”值转换为类型“ System.IO.FileStream”。 + CategoryInfo:InvalidData:(:)[],ParameterBindin ... mationException + FullyQualifiedErrorId:ParameterArgumentTransformationError + PSComputerName:RemoteMachineName
我在Internet上的任何地方都找不到对“ Deserialized.System.IO.FileStream”类的引用。我的猜测是,流在传输到远程计算机时正在序列化,这就是导致此问题的原因。
有人知道如何正确执行此操作,或者有更好的方法来传输此文件?最终文件很大,因此我不能在不耗尽内存的情况下将它们读入字节数组。另外,我必须使用指定的凭据,并且传输必须经过加密。
答案 0 :(得分:1)
远程会话中的对象被拆除,通过网络发送,然后重建。某些类型可以重建为活动对象,而其他类型则可以怀疑是反序列化版本。有一篇博客文章here that explains it。
请注意,无论使用的是HTTP还是HTTPS协议,初始身份验证之后,PSRemoting均为always encrypted,且使用AES-256。
从downloading a file over PsSession的答案开始,我颠倒了整个过程,即通过网络发送1MB的块并将其写入远程计算机上的流。由于此操作使用了您已经建立的任何会话,因此它应该满足您的要求。可能可以对此进行优化,但是我在1GB文件上进行了一些基本测试,并且按预期工作。
function UploadSingleFile {
param(
[System.Management.Automation.Runspaces.PSSession] $RemoteSession,
[string] $RemoteFile,
[string] $LocalFile,
[int] $ChunkSize = 1mb
)
$FileInfo = Get-Item -LiteralPath $LocalFile
$FileStream = $FileInfo.OpenRead()
try {
$FileReader = New-Object System.IO.BinaryReader $FileStream
try {
for() {
$Chunk = $FileReader.ReadBytes($ChunkSize)
if($Chunk.Length) {
Invoke-Command -Session $RemoteSession -ScriptBlock {
param(
[string] $RemoteFile,
[byte[]] $Chunk
)
$FileStream = [System.IO.FileStream]::new($RemoteFile, [System.IO.FileMode]::Append, [System.IO.FileAccess]::Write)
$FileStream.Write($Chunk, 0, $Chunk.Length)
$FileStream.Close()
} -ArgumentList $RemoteFile, $Chunk
} else {
break;
}
}
} finally {
$FileReader.Close();
}
} finally {
$FileStream.Close();
}
}
调用:
$session = New-PSSession -ComputerName ComputerOne -UseSSL -Credential $credential
UploadSingleFile -Session $session -RemoteFile 'c:\temp\Destination.bin' -LocalFile 'C:\temp\source.bin'
然后我使用Get-FileHash确认文件复制成功。
PS C:\> Get-FileHash C:\temp\source.bin
Algorithm Hash Path
--------- ---- ----
SHA256 C513BFBCF4501A06BCC5F6F7A589532F7D802AA2C032D88143B0A31C1CFBD5F4 C:\temp\source.bin
PS C:\> Get-FileHash '\\ComputerOne\c$\temp\Destination.bin'
Algorithm Hash Path
--------- ---- ----
SHA256 C513BFBCF4501A06BCC5F6F7A589532F7D802AA2C032D88143B0A31C1CFBD5F4 \\ComputerOne\c$\temp\Destination.bin