如何解决Windows Update API CopyToCache(IStringCollection) - 指定的强制转换无效
为了减少我们 VPN 路由上的流量,我需要从外部服务器下载 Windows 更新,同时向我们的内部服务器报告。
所以我正在做以下事情:
创建一个 UpdateSession 并搜索更新,将它们存储在 $SearchResult 中。 然后我从外部服务器下载更新,然后我想通过 IUpdate2.CopyToCache(IStringCollection)
将它们传递到 Windows Update Api一切都很好,除了将 StringCollection 传递给方法 copyToCache 并以“指定的强制转换无效”结束。-错误。
这是我的代码:
感谢您的帮助! eldo-ob
$UpdateSession = New-Object -ComObject Microsoft.Update.Session
$UpdateSearcher = $UpdateSession.CreateUpdateSearcher()
$UpdateCollection = New-Object -Com Microsoft.Update.UpdateColl
$SearchResult = $UpdateSearcher.Search("IsInstalled=0 and Type='Software'")
$Availibleupdates = [int16]$SearchResult.Updates.Count
$Availibleupdates
$WebClient = New-Object System.Net.WebClient
[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12
@($SearchResult.Updates.Item(0).BundledUpdates) | Foreach {
$_.DownloadContents | Foreach {
$FileName = $_.DownloadUrl.Split("/")[-1]
$downloadFrom = $_.DownloadUrl.Replace("http://contoso-intern.com","https://contoso-extern.com")
$WebClient.DownloadFile($downloadFrom,("C:\temp\WSUS\{0}" -f $FileName))
Write-Host "File Downloaded" -ForegroundColor Green
}
$StringCollection = New-Object System.Collections.Specialized.StringCollection
$StringCollection.Add(("C:\temp\WSUS\{0}" -f $FileName))
$_.copyToCache($StringCollection)
}
错误消息:
Specified cast is not valid.
At line:24 char:19
+ $_.copyToCache($StringCollection)
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+ CategoryInfo : OperationStopped: (:) [],InvalidCastException
+ FullyQualifiedErrorId : system.invalidCastException
更新/解决方案:
# create UpdateSession
$UpdateSession = New-Object -ComObject Microsoft.Update.Session
# create UpdateSearcher
$UpdateSearcher = $UpdateSession.CreateUpdateSearcher()
# search for updates & count updates
$SearchResult = $UpdateSearcher.Search("IsInstalled=0 and Type='Software'")
$Availibleupdates = [int16]$SearchResult.Updates.Count
# create an WebClient instance for downloading Updates from alternative source
$WebClient = New-Object System.Net.WebClient
# fix some tls issues (not for everyone neccessary)
[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12
# iterate the updates in searchresult
$SearchResult.Updates | ForEach-Object {
# iterate bundledupdates
$_.BundledUpdates | ForEach-Object {
# create COM stringcollection
$StringCollection = New-Object -ComObject "Microsoft.Update.StringColl.1"
# iterate downloadcontents
$_.DownloadContents | ForEach-Object {
# get the filename from url
$FileName = $_.DownloadUrl.Split("/")[-1]
# create external downloadlink
$downloadFrom = $_.DownloadUrl.Replace("http://contoso-intern.com","https://contoso-extern.com/wsusreplica")
# download update with webclient
$WebClient.DownloadFile($downloadFrom,("C:\temp\WSUS\{0}" -f $FileName))
# adding downloaded filepath to stringcollection
$StringCollection.Add(("C:\temp\WSUS\{0}" -f $FileName))
}
# copy downloaded file to cache (load into wuapi)
$_.copyToCache($StringCollection)
}
}
# create installer
$Installer = $UpdateSession.CreateUpdateInstaller()
# set the updates
$Installer.Updates = $SearchResult.Updates
# and install
$Installer.Install()
在不使用自选位置的 UpdateDownloader 的情况下成功安装的更新。所以现在我可以通过 vpn 隧道报告和搜索更新,并从外部源下载更新,我们可以在其中路由 vpn 隧道旁边的流量。
解决方法
您正在使用 .NET 对象。在注册表中搜索接口后,然后查找接口的TypeLib,它指向wuapi.dll。然后我搜索了使用 wuapi.dll 作为其 InprocServer32 的 COM 对象。我找到了“Microsoft.Update.StringColl.1”。它有一个 Add()
方法,因此它在代码中的工作方式应该与您的其他方法相同(我认为)。所以,用这个替换你初始化 $StringCollection 的地方:
$StringCollection = new-object -ComObject "Microsoft.Update.StringColl.1"
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。