mirror of
https://github.com/pbatard/Fido.git
synced 2025-09-16 14:18:02 +02:00
Improve compatibility with vanilla Windows platforms
* Add a notice about PowerShell 3.0 being required. * Enable script to run on platforms without Microsoft.mshtml assembly. * Add a -DisableFirstRunCustomize option to prevent an exception when the script is run on a platform where IE has never been launched. * Also display a notice about Microsoft anti bulk download
This commit is contained in:
117
Fido.ps1
117
Fido.ps1
@@ -1,5 +1,5 @@
|
||||
#
|
||||
# Fido v1.02 - Retail Windows ISO Downloader
|
||||
# Fido v1.03 - Retail Windows ISO Downloader
|
||||
# Copyright © 2019 Pete Batard <pete@akeo.ie>
|
||||
# ConvertTo-ImageSource: Copyright © 2016 Chris Carter
|
||||
#
|
||||
@@ -31,6 +31,10 @@ param(
|
||||
# (Optional) Name of a pipe the download URL should be sent to.
|
||||
# If not provided, a browser window is opened instead.
|
||||
[string]$PipeName,
|
||||
# (Optional) Disable IE First Run Customize so that Invoke-WebRequest
|
||||
# doesn't throw an exception if the user has never launched IE.
|
||||
# Note that this requires the script to run elevated.
|
||||
[switch]$DisableFirstRunCustomize,
|
||||
# (Optional) Toggle expert mode (additional ISOs to choose).
|
||||
[switch]$Expert = $False
|
||||
)
|
||||
@@ -81,6 +85,13 @@ Add-Type -AssemblyName PresentationFramework
|
||||
[Gui.Utils]::ShowWindow(([System.Diagnostics.Process]::GetCurrentProcess() | Get-Process).MainWindowHandle, 0) | Out-Null
|
||||
#endregion
|
||||
|
||||
# Make sure PowerShell 3.0 or later is used (for Invoke-WebRequest)
|
||||
if ($PSVersionTable.PSVersion.Major -lt 3) {
|
||||
Write-Host Error: PowerShell 3.0 or later is required to run this script.
|
||||
[System.Windows.MessageBox]::Show("PowerShell 3.0 or later is required to run this script.`nYou can download it from: https://www.microsoft.com/en-us/download/details.aspx?id=34595", "Error", "OK", "Error") | Out-Null
|
||||
exit -1
|
||||
}
|
||||
|
||||
#region Data
|
||||
$WindowsVersions = @(
|
||||
@(
|
||||
@@ -366,6 +377,17 @@ function Get-Translation([string]$Text)
|
||||
return $Text
|
||||
}
|
||||
|
||||
# Some PowerShells don't have Microsoft.mshtml assembly (comes with MS Office?)
|
||||
# so we can't use ParsedHtml or IHTMLDocument[2|3] features there...
|
||||
function GetElementById([object]$Request, [string]$Id)
|
||||
{
|
||||
try {
|
||||
return $Request.ParsedHtml.IHTMLDocument3_GetElementByID($Id)
|
||||
} catch {
|
||||
return $Request.AllElements | ? {$_.id -eq $Id}
|
||||
}
|
||||
}
|
||||
|
||||
function Error([string]$ErrorMessage)
|
||||
{
|
||||
Write-Host $ErrorMessage
|
||||
@@ -378,6 +400,17 @@ function Error([string]$ErrorMessage)
|
||||
$script:Stage = -1
|
||||
$Continue.IsEnabled = $True
|
||||
}
|
||||
|
||||
function Get-RandomDate()
|
||||
{
|
||||
[DateTime]$Min = "1/1/2008"
|
||||
[DateTime]$Max = [DateTime]::Now
|
||||
|
||||
$RandomGen = new-object random
|
||||
$RandomTicks = [Convert]::ToInt64( ($Max.ticks * 1.0 - $Min.Ticks * 1.0 ) * $RandomGen.NextDouble() + $Min.Ticks * 1.0 )
|
||||
$Date = new-object DateTime($RandomTicks)
|
||||
return $Date.ToString("yyyyMMdd")
|
||||
}
|
||||
#endregion
|
||||
|
||||
#region Form
|
||||
@@ -401,14 +434,20 @@ $MaxStage = 4
|
||||
$SessionId = ""
|
||||
$ExitCode = -1
|
||||
$Locale = "en-US"
|
||||
|
||||
$DFRCKey = "HKLM:\Software\Policies\Microsoft\Internet Explorer\Main\"
|
||||
$DFRCName = "DisableFirstRunCustomize"
|
||||
$DFRCAdded = $False
|
||||
$RequestData = @{}
|
||||
$RequestData["GetLangs"] = @("a8f8f489-4c7f-463a-9ca6-5cff94d8d041", "GetSkuInformationByProductEdition" )
|
||||
$RequestData["GetLangs"] = @("a8f8f489-4c7f-463a-9ca6-5cff94d8d041", "getskuinformationbyproductedition" )
|
||||
$RequestData["GetLinks"] = @("cfa9e580-a81e-4a4b-a846-7b21bf4e2e5b", "GetProductDownloadLinksBySku" )
|
||||
# Create a semi-random Linux User-Agent string
|
||||
$FirefoxVersion = Get-Random -Minimum 30 -Maximum 60
|
||||
$FirefoxDate = Get-RandomDate
|
||||
$UserAgent = "Mozilla/5.0 (X11; Linux i586; rv:$FirefoxVersion.0) Gecko/$FirefoxDate Firefox/$FirefoxVersion.0"
|
||||
#endregion
|
||||
|
||||
# Localization
|
||||
$EnglishMessages = "en-US|Version|Release|Edition|Language|Architecture|Download|Continue|Back|Close|Cancel|Error|Please wait...|Download using a browser"
|
||||
$EnglishMessages = "en-US|Version|Release|Edition|Language|Architecture|Download|Continue|Back|Close|Cancel|Error|Please wait...|Download using a browser|Temporarily banned by Microsoft for requesting too many downloads - Please try again later..."
|
||||
[string[]]$English = $EnglishMessages.Split('|')
|
||||
[string[]]$Localized = $null
|
||||
if ($LocData -and (-not $LocData.StartsWith("en-US"))) {
|
||||
@@ -420,6 +459,20 @@ if ($LocData -and (-not $LocData.StartsWith("en-US"))) {
|
||||
$Locale = $Localized[0]
|
||||
}
|
||||
|
||||
# If asked, disable IE first run customize as it interferes with Invoke-WebRequest
|
||||
if ($DisableFirstRunCustomize) {
|
||||
try {
|
||||
# Only create the key if it doesn't already exist
|
||||
Get-ItemProperty -Path $DFRCKey -Name $DFRCName -ErrorActionPreference "Stop"
|
||||
} catch {
|
||||
if (-not (Test-Path $DFRCKey)) {
|
||||
New-Item -Path $DFRCKey -Force | Out-Null
|
||||
}
|
||||
Set-ItemProperty -Path $DFRCKey -Name $DFRCName -Value 1
|
||||
$DFRCAdded = $True
|
||||
}
|
||||
}
|
||||
|
||||
# Form creation
|
||||
$XMLForm = [Windows.Markup.XamlReader]::Load((New-Object System.Xml.XmlNodeReader $XAML))
|
||||
$XAML.SelectNodes("//*[@Name]") | ForEach-Object { Set-Variable -Name ($_.Name) -Value $XMLForm.FindName($_.Name) -Scope Script }
|
||||
@@ -470,10 +523,12 @@ $Continue.add_click({
|
||||
Write-Host Querying $url
|
||||
|
||||
try {
|
||||
$r = Invoke-WebRequest -SessionVariable "Session" $url
|
||||
$script:SessionId = $r.ParsedHtml.IHTMLDocument3_GetElementById("session-id").Value
|
||||
# Note: We can't use -UseBasicParsing since we need JS to create the session-id
|
||||
# TODO: Use -Headers @{"Cache-Control"="no-cache"}?
|
||||
$r = Invoke-WebRequest -UserAgent $UserAgent -SessionVariable "Session" $url
|
||||
$script:SessionId = $(GetElementById -Request $r -Id "session-id").Value
|
||||
if (-not $SessionId) {
|
||||
$ErrorMessage = $r.ParsedHtml.IHTMLDocument3_GetElementByID("errorModalMessage").innerHtml
|
||||
$ErrorMessage = $(GetElementById -Request $r -Id "errorModalMessage").InnerText
|
||||
if ($ErrorMessage) {
|
||||
Write-Host "$(Get-Translation("Error")): ""$ErrorMessage"""
|
||||
}
|
||||
@@ -526,11 +581,18 @@ $Continue.add_click({
|
||||
$i = 0
|
||||
$SelectedIndex = 0
|
||||
try {
|
||||
$r = Invoke-WebRequest -WebSession $Session $url
|
||||
foreach ($var in $r.ParsedHtml.IHTMLDocument3_GetElementByID("product-languages")) {
|
||||
$r = Invoke-WebRequest -UserAgent $UserAgent -WebSession $Session $url
|
||||
# Go through an XML conversion to keep all PowerShells happy...
|
||||
if (-not $($r.AllElements | ? {$_.id -eq "product-languages"})) {
|
||||
throw "Unexpected server response"
|
||||
}
|
||||
$html = $($r.AllElements | ? {$_.id -eq "product-languages"}).InnerHTML
|
||||
$html = "<options>" + $html.Replace("selected value", "value") + "</options>"
|
||||
$xml = [xml]$html
|
||||
foreach ($var in $xml.options.option) {
|
||||
$json = $var.value | ConvertFrom-Json;
|
||||
if ($json) {
|
||||
$array += @(New-Object PsObject -Property @{ DisplayLanguage = $var.text; Language = $json.language; Id = $json.id })
|
||||
$array += @(New-Object PsObject -Property @{ DisplayLanguage = $var.InnerText; Language = $json.language; Id = $json.id })
|
||||
if (Select-Language($json.language)) {
|
||||
$SelectedIndex = $i
|
||||
}
|
||||
@@ -538,7 +600,7 @@ $Continue.add_click({
|
||||
}
|
||||
}
|
||||
if ($array.Length -eq 0) {
|
||||
$ErrorMessage = $r.ParsedHtml.IHTMLDocument3_GetElementByID("errorModalMessage").innerHtml
|
||||
$ErrorMessage = $(GetElementById -Request $r -Id "errorModalMessage").innerText
|
||||
if ($ErrorMessage) {
|
||||
Write-Host "$(Get-Translation("Error")): ""$ErrorMessage"""
|
||||
}
|
||||
@@ -568,10 +630,27 @@ $Continue.add_click({
|
||||
$SelectedIndex = 0
|
||||
$array = @()
|
||||
try {
|
||||
$r = Invoke-WebRequest -WebSession $Session $url
|
||||
foreach ($var in $r.ParsedHtml.IHTMLDocument3_GetElementsByTagName("span") | Where-Object { $_.className -eq "product-download-type" }) {
|
||||
$Link = $var.ParentNode | Select -Expand href
|
||||
$Type = $var.innerText
|
||||
$r = Invoke-WebRequest -UserAgent $UserAgent -WebSession $Session $url
|
||||
if (-not $($r.AllElements | ? {$_.id -eq "expiration-time"})) {
|
||||
$ErrorMessage = $(GetElementById -Request $r -Id "errorModalMessage").innerText
|
||||
if ($ErrorMessage) {
|
||||
Write-Host "$(Get-Translation("Error")): ""$ErrorMessage"""
|
||||
}
|
||||
throw Get-Translation($English[14])
|
||||
}
|
||||
$html = $($r.AllElements | ? {$_.tagname -eq "input"}).outerHTML
|
||||
# Need to fix the HTML and JSON data so that it is well-formed
|
||||
$html = $html.Replace("class=product-download-hidden", "")
|
||||
$html = $html.Replace("type=hidden", "")
|
||||
$html = $html.Replace(">", "/>")
|
||||
$html = $html.Replace(": I", ": ""I")
|
||||
$html = $html.Replace(" }", """ }")
|
||||
$html = "<inputs>" + $html + "</inputs>"
|
||||
$xml = [xml]$html
|
||||
foreach ($var in $xml.inputs.input) {
|
||||
$json = $var.value | ConvertFrom-Json;
|
||||
if ($json) {
|
||||
$Type = $json.DownloadType
|
||||
# Maybe Microsoft will provide public ARM/ARM64 retail ISOs one day...
|
||||
if ($Type -like "*arm64*") {
|
||||
$Type = "Arm64"
|
||||
@@ -594,11 +673,12 @@ $Continue.add_click({
|
||||
$SelectedIndex = $i
|
||||
}
|
||||
}
|
||||
$array += @(New-Object PsObject -Property @{ Type = $Type; Link = $Link })
|
||||
$array += @(New-Object PsObject -Property @{ Type = $Type; Link = $json.Uri })
|
||||
$i++
|
||||
}
|
||||
}
|
||||
if ($array.Length -eq 0) {
|
||||
$ErrorMessage = $r.ParsedHtml.IHTMLDocument3_GetElementByID("errorModalMessage").innerHtml
|
||||
$ErrorMessage = $(GetElementById -Request $r -Id "errorModalMessage").innerText
|
||||
if ($ErrorMessage) {
|
||||
Write-Host "$(Get-Translation("Error")): ""$ErrorMessage"""
|
||||
}
|
||||
@@ -699,4 +779,7 @@ $XMLForm.ShowDialog() | Out-Null
|
||||
if (-not $PipeName) {
|
||||
Stop-Job -Job $Job
|
||||
}
|
||||
if ($DFRCAdded) {
|
||||
Remove-ItemProperty -Path $DFRCKey -Name $DFRCName
|
||||
}
|
||||
exit $ExitCode
|
||||
|
Reference in New Issue
Block a user