summaryrefslogtreecommitdiffstats
path: root/os_win32/update-smart-drivedb.ps1.in
diff options
context:
space:
mode:
Diffstat (limited to 'os_win32/update-smart-drivedb.ps1.in')
-rw-r--r--os_win32/update-smart-drivedb.ps1.in841
1 files changed, 841 insertions, 0 deletions
diff --git a/os_win32/update-smart-drivedb.ps1.in b/os_win32/update-smart-drivedb.ps1.in
new file mode 100644
index 0000000..ffec165
--- /dev/null
+++ b/os_win32/update-smart-drivedb.ps1.in
@@ -0,0 +1,841 @@
+#
+# smartmontools drive database update script for Windows
+#
+# Home page of code is: https://www.smartmontools.org
+#
+# Copyright (C) 2022-23 Christian Franke
+#
+# SPDX-License-Identifier: GPL-2.0-or-later
+#
+# $Id: update-smart-drivedb.ps1.in 5515 2023-07-24 12:58:25Z chrfranke $
+#
+
+<#
+ .SYNOPSIS
+ Update smartmontools @VERSION@ drive database.
+
+ .DESCRIPTION
+ update-smart-drivedb.ps1 updates SCRIPTDIR/drivedb.h from
+ branches/@DRIVEDB_BRANCH@ of smartmontools SVN repository.
+
+ The downloaded file is verified with OpenPGP/GPG key ID 721042C5.
+ The public key block is included in the script.
+
+ .PARAMETER Smartctl
+ Use this smartctl executable for syntax check ('-Smartctl -' to
+ disable). The default is smartctl.exe in script directory.
+
+ .PARAMETER UrlOf
+ Use the URL of one of the following locations for download:
+ github (GitHub mirror of SVN repository)
+ sf (Sourceforge code browser)
+ svn (SVN repository)
+ svni (SVN repository via HTTP instead of HTTPS)
+ trac (Trac code browser)
+ The default is 'svn'.
+
+ .PARAMETER Url
+ Download from this URL. A valid OpenPGP/GPG signature with '.raw.asc'
+ extension must also exist unless '-NoVerify' is also specified.
+
+ .PARAMETER File
+ Copy from local file. A valid OpenPGP/GPG signature 'FILE.raw.asc'
+ must also exist unless '-NoVerify' is also specified.
+
+ .PARAMETER Trunk
+ Download from SVN trunk. This requires '-NoVerify' because the trunk
+ versions are not signed.
+
+ .PARAMETER Branch
+ If '-Branch X.Y' is specified, download from
+ branches/RELEASE_X_Y_DRIVEDB. This also selects the OpenPGP/GPG key
+ for older branches (5.40 to 6.6: Key ID DFD22559).
+
+ .PARAMETER Insecure
+ Don't abort download if certificate verification fails. This option
+ is also required if a HTTP URL is selected with '-UrlOf' option.
+ If a HTTPS URL is selected, PowerShell >= 6.0 is required.
+
+ .PARAMETER NoVerify
+ Don't verify signature with GnuPG.
+
+ .PARAMETER Force
+ Allow downgrades. By default, the database is not replaced with an
+ older version of the same branch.
+
+ .PARAMETER ExportKey
+ Print the OpenPGP/GPG public key block.
+
+ .PARAMETER DryRun
+ Print download commands only.
+
+ .PARAMETER Quiet
+ Suppress info messages.
+
+ .PARAMETER Drivedb
+ Destination drive database file.
+ The default is drivedb.h in script directory.
+
+ .EXAMPLE
+ update-smart-drivedb.ps1
+ /INSTALLPATH/drivedb.h 7.2/5225 updated to 7.2/5237
+
+ (Regular update)
+
+ .EXAMPLE
+ update-smart-drivedb.ps1 -Force -File /INSTALLPATH/drivedb.h.old
+ /INSTALLPATH/drivedb.h 7.2/5237 downgraded to 7.2/5225
+
+ (Revert to previous version)
+
+ .EXAMPLE
+ update-smart-drivedb -Trunk -NoVerify -Smartctl - drivedb-trunk.h
+ drivedb-trunk.h 7.3/5254 newly installed (NOT VERIFIED)
+
+ (Download the database from SVN trunk to current directory)
+
+ .LINK
+ https://www.smartmontools.org/
+#>
+
+param (
+ [string]$Smartctl,
+ [string]$UrlOf,
+ [string]$Url,
+ [string]$File,
+ [switch]$Trunk,
+ [string]$Branch,
+ [switch]$Insecure,
+ [switch]$NoVerify,
+ [switch]$Force,
+ [switch]$ExportKey,
+ [switch]$DryRun,
+ [switch]$Quiet,
+ #[switch]$Verbose, # Common parameter
+ [Parameter(Position=0)][string]$Drivedb
+)
+
+$ErrorActionPreference = "Stop"
+
+# Set by config.status
+# Default drivedb.h update branch
+$default_branch="@DRIVEDB_BRANCH@"
+
+# GnuPG used to verify signature (disabled if empty)
+$gpg = "@gnupg@"
+
+# Name and Directory of this script
+$myname = $MyInvocation.MyCommand.Name
+$mydir = Split-Path -Path $MyInvocation.MyCommand.Source
+if (!$mydir) {
+ throw "Unknown script directory"
+}
+
+# Default drivedb location
+$default_drivedb = "$mydir\drivedb.h"
+
+# Default command used for syntax check
+$default_smartctl = "$mydir\smartctl"
+
+function error($message)
+{
+ # Echo error messages to stdout because 'Write-Error'
+ # always writes a full 'ErrorRecord' to stderr.
+ echo "${myname}: $message"
+ exit 1
+}
+
+function warning($message)
+{
+ echo "${myname}: (Warning) $message"
+}
+
+function iecho($message)
+{
+ if (!$script:Quiet) {
+ echo $message
+ }
+}
+
+function vecho($message)
+{
+ $message | Write-Verbose
+}
+
+function test_f($file)
+{
+ return Test-Path -PathType Leaf -LiteralPath $file
+}
+
+function touch($file)
+{
+ if (Test-Path -PathType Leaf -LiteralPath $file) {
+ (Get-Item $file).LastWriteTime = Get-Date
+ } else {
+ New-Item -ItemType File -Path $file | Out-Null
+ }
+}
+
+function cmp($file1, $file2)
+{
+ if (!( (Test-Path -PathType Leaf -LiteralPath $file1) `
+ -and (Test-Path -PathType Leaf -LiteralPath $file2))) {
+ return $false
+ }
+ return (Get-FileHash $file1).hash -eq (Get-FileHash $file2).hash
+}
+
+function rm_f # FILE...
+{
+ foreach ($file in $args) {
+ if (Test-Path -PathType Leaf -LiteralPath $file) {
+ Remove-Item $file
+ }
+ }
+}
+
+function rm_rf($dir)
+{
+ if (Test-Path -PathType Container -LiteralPath $dir) {
+ Remove-Item -Recurse $dir
+ }
+}
+
+function mv_f($oldfile, $newfile)
+{
+ rm_f $newfile
+ Rename-Item $oldfile $newfile
+}
+
+function selecturl($url_of, [ref]$url)
+{
+ switch ($url_of) {
+ 'github' {
+ # https://github.com/smartmontools/smartmontools/raw/origin/$branch/smartmontools/drivedb.h
+ # https://github.com/smartmontools/smartmontools/raw/master/smartmontools/drivedb.h
+ # redirected to:
+ $u = 'https://raw.githubusercontent.com/smartmontools/smartmontools/master/smartmontools/drivedb.h'
+ }
+ 'sf' { $u = 'https://sourceforge.net/p/smartmontools/code/HEAD/tree/trunk/smartmontools/drivedb.h?format=raw' }
+ 'svn' { $u = 'https://svn.code.sf.net/p/smartmontools/code/trunk/smartmontools/drivedb.h' }
+ 'svni' { $u = 'http://svn.code.sf.net/p/smartmontools/code/trunk/smartmontools/drivedb.h' }
+ 'trac' { $u = 'https://www.smartmontools.org/export/HEAD/trunk/smartmontools/drivedb.h' }
+ default { error "${url_of}: is none of 'github sf svn svni trac'" }
+ }
+ $url.Value = $u
+}
+
+function vcopy($src, $dest)
+{
+ if ($script:DryRun) {
+ echo "Copy-Item $src $dest"
+ } else {
+ vecho "Copy-Item $src $dest"
+ Copy-Item $src $dest
+ }
+}
+
+function download($url, $file, [ref]$errmsg)
+{
+ $req = @{ Uri = $url; OutFile = $file; MaximumRedirection = 0 }
+ if ($script:Insecure) {
+ # #Requires -Version 6
+ $req += @{ SkipCertificateCheck = $true }
+ }
+
+ $errmsg.Value = ""
+ if ($script:DryRun) {
+ echo "Invoke-WebRequest $(echo @req)"
+ } else {
+ try {
+ Invoke-WebRequest @req
+ } catch {
+ if ($_.Exception.Message) {
+ $errmsg.Value = $_.Exception.Message
+ } else {
+ $errmsg.Value = "Unknown error"
+ }
+ }
+ }
+}
+
+function check_file($file, $firstchar, $minsize, $maxsize)
+{
+ # Check file size
+ $size = (Get-Item -LiteralPath $file).Length
+ if ($size -lt $minsize) {
+ return "too small file size $size bytes"
+ }
+ if ($size -gt $maxsize) {
+ return "too large file size $size bytes"
+ }
+
+ # Check first chars
+ switch ((Get-Content -LiteralPath $file -TotalCount 1).ToCharArray()[0]) {
+ "$firstchar" { }
+ "<" { return "HTML error message" }
+ default { return "unknown file contents" }
+ }
+
+ return ""
+}
+
+function unexpand_svn_id($source, $dest)
+{
+ # For -NoNewLine:
+ #Requires -Version 5
+ (Get-Content -Raw -Path $source) -replace `
+ ('\$'+'Id: drivedb\.h [0-9][0-9]* 2[-0-9]* [012][:0-9]*Z [a-z][a-z0-9]* \$'),('$'+'Id'+'$') `
+ | Set-Content -NoNewLine -Path $dest
+}
+
+function selectkey($branch, [ref]$key)
+{
+ switch -RegEx ($branch) {
+ '^RELEASE_(5_4[0-3]|6_[0-6])_DRIVEDB$' {
+# Smartmontools Signing Key (ext. to 2024) <smartmontools-database@listi.jpberlin.de>
+# Smartmontools Signing Key (through 2018) <smartmontools-database@listi.jpberlin.de>
+# Smartmontools Signing Key (through 2018) <smartmontools-database@lists.sourceforge.net>
+# Key ID DFD22559
+ $key.Value = `
+'-----BEGIN PGP PUBLIC KEY BLOCK-----
+
+mQENBFgOYoEBCAC93841SlFmpp6640hKUvZ8PbZR6OGnZnMXD6QRVzpibXGZXUDB
+f6unujun5Ql4ObAWt6QuRqz5Gk2gF8tcOfN6edR/uK5gyX2rlWVLoZKOV91a3aDI
+iIDh018tLWOpHg3VxgHL6f0iMcFogUYnD5zhC5Z2GVhFb/cVpj+ocZWcxQLQGPVv
+uZPUQWrvdpFEzcnxPMtJJDqXEChzhrdFTXGm69ERxULOro7yDmG1Y5xWmhdGnPPM
+cuCXVVlADz/Gh1w+ay7RqFnzPjqjQmHAuggns467TJEcS0yiX4LJnEoKLyPGen9L
+FH6z38xHCNt4Da05/OeRgXwVLH9M95lu8d6TABEBAAG0U1NtYXJ0bW9udG9vbHMg
+U2lnbmluZyBLZXkgKGV4dC4gdG8gMjAyNCkgPHNtYXJ0bW9udG9vbHMtZGF0YWJh
+c2VAbGlzdGkuanBiZXJsaW4uZGU+iQFBBBMBAgArAhsDBQkPZe4NBgsJCAcDAgYV
+CAIJCgsEFgIDAQIeAQIXgAUCXheK5gIZAQAKCRDzh2PO39IlWdUTCAC8v9Oa7umW
++/tXBiEtElDW/U2rEOC3OHWSzPvqE4iGjWc5fbvrAKS7bfccZM8Aq0a1t2pSbIlB
+MvRrsNTGdQSPsOdhxPD8pEJW0uH9Z5VyPzoO9VIaoqi1irRdWnXCfhBJX9PLySAb
+9BPQZXXQypmACieRDv31E4hiB+vYet/SpVuRyfL57XU3jmwFREip9OiFOp+61X2+
+oIlgvNU60JZy2vXpTo6PNbDGetEycfH6Y8vfCXniihMkSfeOnNqWI/hycBDprFB5
+CB5ShIH71vhCOPnVGwtYY30wlJ1+Ybg2ZAIi6JN8E38Dpx382IzeT2LydnZydiC6
+PcLCr7mbsX3hiQEcBBMBAgAGBQJeF4sWAAoJEC/N7AvTrxqr7ZAH/jB4xFtBTo1x
+w8CGwslZCJ+/BeEZ5XpV+8zLdeRV2tXegUFjGZ9FI6UpzBeVyK2R1qGbcdSf2S45
+KutcM2gjKETW+ZwW76qHJD52mYihPPLXu2pRAG2WyH5GDnqNMj5iQ1inoPdZOTpi
+evBMTv1YHJML6SiF6t/HoKorl5ffvHBE/1onBfUzLwQ/ct14sZ2UXHzyxdHo73vm
+XWgcjQ1TQhCSdLqucQbwR78EyUa9tYxk/NWBqfc5YHt7t+KTVTLlp7Buk1wscLkj
+NTlxl+IjAxRwsWc6PWnyRdAgXxtt2q6llYgFahWM21OyJVLVjbMGVF+oBtFumqq3
+lQy6H6tp/1uJAhwEEwECAAYFAl4XiyMACgkQvwsznGS8qosSiw//QjbWDldB2gHf
+3Tfs+LaFdzkDbioWdnj96DiCynTSwZF8d5ISqwA+QTL/43Y0msU26WBMvIRBg2Xm
++r4TMMfWF4a1Yjq6cisKEaUsbjV9ztzH/XB2ydo8HgnxZuVKQoIuh1sSrE7p6mpQ
+YUrV5eWRpqc79AI9ZzRBM5nhbBejqLVw2F8dyz6c3lfGM9IOenp+Y8N43SdNpBcp
+DuHnzbQIMtkyoX7tTKDDv5gnoRNCsdBsCduTyNWYOIEdhRiCfo5Ce7kufIoo4ZqV
+BM8dzwm1RrcYa0kMKPZAucJDRjwevEYDbOg7vmEYsuGPRbVmOFdx4uMx4gX8vF5+
+AG3rTSA805zkwD+WQXyYQohVZxNjeK7P/ukr6NCZx226gwAiw1ms7PYOo8snjK8e
+nRlMTLKiGiMIH7xJu55JliVlcEvn3G7WO0n4qQOJj3Msh+xflBSfZmzBDAzPgxwC
+m/RSmonGV0uZVJFDHCpqus35E6bzFF6yO3yXvpngAMTBrpX6Nzgea1SzlK2Iquls
+te1GYAx/IXaY7cVYo4iEv/m346SINzLGHpXZkbbcenSgljBfHLCz7vF33IotfEWh
+C7Kb4iKbEjERa+zzqR+vK+nDj6YG9Mvguj1EqnM47oDwgMaqWY6oPfefLCD8Tg51
+rlAAGFdcWb9g034vgtK8l+ooUtn63PKJAhwEEwECAAYFAl4XiywACgkQ6nSrJXIQ
+QsUuTRAAsSMmQ7jsvmljExwrmIu6Oyh+1J5D/GPBRYhSyip/bnxCscCBnpjEk8+7
+VG9JtGTCa0zVY14Y3Cl4obND25QN9LhiE/y8olnIgJ2adtmpi6+zFpdGWVYUpDgZ
+IMePUVKyZenTjezFwRlLsYsxbSb9wIR1iofP1l/dQF8DwhwFL9AGRmHTcWM1ZYoc
+fv80A5SAposnspnkKKcuC3q2+pMsUtbHT9t/+iusVXBDERh+FPlvtYh+Khze3c8z
+g4M9RsQLCanMp4jZhzgSakjeg9tCr33SIJIEKpn6MUftX9QC82S75UNwxXgC38EA
+s2t+BjPLUaXENSdOe3l+KKY5ozbmRpRmQIHw7jlT3+9C0RUHGTPQYCidsx8OdYA0
+4wDRWcjCQcXWxTaUoeaoMJcE1iv5IIf/X0MXYMlCPG8OKAlDE2Kkrx0A8agPp7JH
+0UAOaqpAA74kZnpuvJ6BqrX2hMbNbyVg1rWu1BQA3qESa41rKiWyEtjiLdQ/NtNu
+6BsPhDGvaQqGbu4t0GfJ1PhbFnHrVkLW8v1NzYZRpLXAFJGZdD6Ue/L6bHFOJ6SJ
+JwAHjH26nxSMuDV779AUrnOcmoXIkj6sdAwDZ5Z2ri7b2MgkrJzeapKd0SItnWUQ
+TMe7YUl8B+kUATj01YWMLtHsX9yciFP0iDagW14/rFJHtchOBcu0U1NtYXJ0bW9u
+dG9vbHMgU2lnbmluZyBLZXkgKHRocm91Z2ggMjAxOCkgPHNtYXJ0bW9udG9vbHMt
+ZGF0YWJhc2VAbGlzdGkuanBiZXJsaW4uZGU+iQE+BBMBAgAoAhsDBgsJCAcDAgYV
+CAIJCgsEFgIDAQIeAQIXgAUJD2XuDQUCXheK5gAKCRDzh2PO39IlWTDxCACtkOGn
+vUs/m/uE7IHoSM6wj/6OXXo+TEM1rgnl40oySVoMgyonx7PSwi9rSoDC8AfRhN2q
+bFLEQcrGI8V7PxLpjsz5Z0m/ZnZJAP7TB5WhLRJdu3w2cssjekhIRc+I2B00gcRl
+H//okXyvGte3kr1JdgaownbslwcZRxyNdvWigQH/Vnz91lKAujGULJyl7hv6Kl02
+HYynYmxGmES3pd5VEOpA/DR7n54T2J+Vubh99RT+RH2v46e7LnPhZhN2uxvIiJKE
+8Lp67l1aeMXfgZv6dQ7Dl+pu5lUUyyMQ+nUMBGKZBWftyqhekZrvYcVnTJYU93kU
+41QULaRVIwg888kUiQEcBBMBAgAGBQJZ7kylAAoJEC/N7AvTrxqroQQH/jrZAGT5
+t8uyzRTzJCf3Bco8FqwKcfw8hhpF1Uaypa+quxkpYz9PtP+3e9lGxl0XSEzOwHjf
+gGWXISUOM1ufVxo2hSLG87yO7naFAtylL8l0Zny8Fb6kmT9f3vMktbHdXHUTDNrC
+UkoElEwwDK3qaur8IPUaIKeSTC3C8E/DVnasLs9cpOs2LPIKr3ishbqbHNeWOgGy
+HbA4KCtvQzBhun9drmtQJW6OyCC9FcIoqPSFM/bs2KHf7qATNu9kSMg/YWw7WLAD
+4GPqH9us1GigQ0h6Y4KG5EgmkFvuQFPLHvT4rtqv51zzs1iwFh4+GIagFp+HJ2jn
+lp+GcZcySlwfnemJAT4EEwECACgFAlnuSe4CGwMFCQQcDQAGCwkIBwMCBhUIAgkK
+CwQWAgMBAh4BAheAAAoJEPOHY87f0iVZVMQIAK5wPezq0ROsxiCYPLcR9dF/Qdp2
+1pLfodi6wsC9FAlTVJ3fk2vkNQDb5rMkNvZ/MHf2EWoVIFHvPZcJ6paBjZlapvGF
+qDNrU6hDbakO0PIej5yy+qVeIYcSQpNZeHchAhOOJcnN0o8H6SzZik38b4Hb8H5X
+do78LsZJwU0jsKG6LH3gjiWJtrC+WCXCMYzEGjAJXev2npU2DMVVwxsfYLfdZWq7
+FJJINv8R9EUjtSQQIynJAwb2lFvZB+jC6u8Vv9N1Wid6wh5lF5ejMt6KXqWOvNn+
+YreopmQfbn2XJZxpyn9d7Ev91epYW11E5qG4xNI3m3AmtEGjMTGjfMUstNK0V1Nt
+YXJ0bW9udG9vbHMgU2lnbmluZyBLZXkgKHRocm91Z2ggMjAxOCkgPHNtYXJ0bW9u
+dG9vbHMtZGF0YWJhc2VAbGlzdHMuc291cmNlZm9yZ2UubmV0PokBPgQTAQIAKAIb
+AwYLCQgHAwIGFQgCCQoLBBYCAwECHgECF4AFAl4XiZMFCQ9l7g0ACgkQ84djzt/S
+JVnl5Qf+PVRoLmEpDIqQ+58DMIwz98+yajCJ1vQvEOKjMcgeePOn475eV5Phkvsp
+KtW6TedWhN9l/NcDZzEPCpkhrz24WJDLFV+o16B4MZwSkGTl4/3qijERKsd8M+MS
+tiLr3+eUCFi4dAp0uhPytETvUmtj3ByA0R2luoOK+kEutq6i2x9BPr8Qc55Lqdwt
+SK8pPU05WSaCu1m2oThJhkELVklOQ2cj+D8MrQdJGd3plEb9j5oUbhj7LW/y0i4M
+lqk1rQCQKnY3vTFQBpj1o7T6kLiGqQCOLTX0B6RQ8vt+PEzXPHi0lIdwOrQk5l7h
+utnjwXmWaWEpRjlsuQ5PBrFDsD9N+IkBHAQTAQIABgUCWA5kYwAKCRDfDxpJxKSQ
+Op+/CADTlsgisoXI6b+0oohRaD4ZVl5eBtkvTrxNQf6EF7Z1uPkVOqi1OLWFGyAm
+beLcRmN6c4/DVcaa6GAG7GA+KQwVPRCyC+9Ibsn/+uG6ZFXAez+0eG9NxOfkCnYH
+8ZP8o2VH+9uKJlGGujh9o5r1SNGVifoLGTc8NkWCW+MAKj8dw8WW+wDc80YrdCRr
+SyLrRU9NLTSE4pIJWKcHLwG63xkXHQPPR1lsJgzdAalfEv1TQdIF3sM+GXp4lZ6b
+uahFDiILBh1vj+5C9TdpWZAlqHDYFICa7Rv/MvQa4O9UUl3SlN3sed8zwAmL3Heo
+XE5tBu8iatMaS9e3BmSsVYlhd/q+iQEcBBMBAgAGBQJYDmSWAAoJEC/N7AvTrxqr
+8HsH+QGQuhHYt9Syccd8AF36psyT03mqgbGLMZL8H9ngoa9ZqVMq7O8Aqz23SGTt
+uNuw6EyrcHo7Dy1311GftshI6arsFNJxE2ZNGIfGocRxu9m3Ez+AysWT9sxz/haH
+E+d58NTg+/7R8YWS1q+Tk6m8dA0Xyf3tMBsIJfj0zJvuGMbCLmd93Yw4nk76qtSn
+9UHbnf76UJN5SctAd8+gK3uO6O4XDcZqC06xkWKl193lzcC8sZJBdI15NszC3y/e
+pnILDDMBUNQMBm/XlCYQUetyrJnAVzFGXurtjEXQ/DDnbfy2Z8efoG8rtq7v3fxS
+1TC5jSVOIEqOE4TwzRz1Y/dfqSU=
+=3Lcg
+-----END PGP PUBLIC KEY BLOCK-----
+'
+ }
+
+ '^RELEASE_7_[023]_DRIVEDB$' {
+# Smartmontools Signing Key (through 2025) <smartmontools-database@listi.jpberlin.de>
+# Smartmontools Signing Key (through 2020) <smartmontools-database@listi.jpberlin.de>
+# Key ID 721042C5
+ $key.Value = `
+'-----BEGIN PGP PUBLIC KEY BLOCK-----
+
+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
+90iSgZdt5cZgoH04weXCMwDugn/+Q3rzKvRUTrEfSOivJYg65D/mhbz6HoUTs4JD
+SstTYa9qNCwKQGRSeis4PAgu0hCpnDAhZuN3Ja5AFC2Wi2szQ7R+Zx/JucIBm5S4
+U30W66MtsyUHeulSJ3AV3HrbFfnqu6zfQM4XLw7MpAtQUNJceS/lWfGIquAp3tY/
+IjZIHwgZqKB3czWDhM83wBzCWgAmxyzIrpb4MBYJ5PGuCyC7R/YTdtPJXxsPQl2l
+znsX/9ssa4kBHAQTAQIABgUCX+CfSAAKCRDzh2PO39IlWVcuB/9UkLaPtGY4sDDV
+/A7qjSvSy93mv8gkaIj9dhqoZw+r7cLiEtX04Cz9PqocOFgCYJXKrufHNNkHke2A
+jE9EJfRKiPU/bkeWmrACvtrOd/DZbdmXfxTOekOr516D2ip/U8GBPw6zxfCQVot6
+htpBpB6zzMDtzMOeLnkOxoxR4EMu5K6eJ48bHvG/lbGBByyfRzhtqPh6AAA9G1CC
+IdhNkaA5W1qums3N1mCXrTBnWyjaFhdnttGQfrMdHvTQ77HeL0c2axT2y5PYfrXY
+2ZfZowYLEtFXRSTpDaJfgG+qem3N+pMv6SMOG/4CvlH4/3Hq0aCNvKcY5KUXfIgT
+xmc3/n/wtFNTbWFydG1vbnRvb2xzIFNpZ25pbmcgS2V5ICh0aHJvdWdoIDIwMjAp
+IDxzbWFydG1vbnRvb2xzLWRhdGFiYXNlQGxpc3RpLmpwYmVybGluLmRlPokCPgQT
+AQIAKAIbAwYLCQgHAwIGFQgCCQoLBBYCAwECHgECF4AFAl/gnzAFCQ0umpsACgkQ
+6nSrJXIQQsVK7RAAqbZfT3wZEfJkw8MK2JlvgGWH76fHKn5ZoH5i0mA4AvN4QLbU
+5Q20HmqHnO9mfAZQ6u4Tn/aFcYT7nlSsEsEmFX+s5QU2y6m2Tx9ThDbZ03ezREOS
+0wNf0FOQunV9ZVPT/7cKIgWJa5mZy+LClor9OHllyGUfs9tKNzwxaHh1zBrCNJow
+Fi/1bkWy3iMc7vZhWHASwPSp64KHjB4UdMz2hV4pROiUhWi7BY0exIHyZrkcANMP
+Hhl9lP32ZvNbOy8osBdPUgXyK3HePD+ftcwJMkoc4mFQXYi9UY7NQpk7STRO10cx
+Kq/CgDDvYxbnViRjQoJ0sfwKCaOfsnY/gea7I0aCx8uNISYpHO9iMidd/tJ7+lgx
+NiKZTI0EppHYvkyMY15/NGb0gTJbYjuVYdbqDS9mnLuLQAjAX43+n9ND2NjX1o0q
+Z9bBqV2VFioNmnxKqGphhRFX9jEzTklieOjhpRrd8v9ljprT6vLFNpYpeLkel8om
+VFXrHxrfzKtVFcto5wqHVOcyZyE2zm1QmsS8qvWOTrNfY6p2q9MA2rysqdfgfvN7
+pNDaXutK6ooQi6YlyyTA2ANnHFKa0ncRH+dg+5OF9rhNvM7RyaBXgxF7+5gnU5Gb
+VQRKbJ+LOtSKkj0pApR5AKSwyGslZ2bNVlKsADWhk5xj8QlHVlNWiht+i/6JARwE
+EwECAAYFAlwmhpwACgkQL83sC9OvGqsVOggAqLB5eQrUv8E9ikD6kJCito827bzD
+WF29yD7PvfhjXaz5in54jOVpwg3o9CsqIjjRW0/1bBVswC8ZL0sAdZ+GDSDMw5F2
+IpkD77gjnFY79M/e6C9xYyxYzHC7emDPSz9IroOvdkkEgrB+OABKkaOCcS18P4Lk
+3WNHaPw5c7aI0z1iJP52EmSfvB8r86mtUFJB+f15eD/4vaRfkZLFjF9FQ3kgEK1U
++rV4s1O2bCFfP3WPDcc83NgwRUvtXmcSOSOIoXnemJzyJr+JnqCWVET4XWF6i20m
+RFXVEpWtf5AkJYgR3z/jW0djELbBWA/35bAnpXy5pDHv9NbZsTkBZxK/kokBHAQT
+AQIABgUCXCaGnQAKCRAY7NpGy/a6xn4lB/90tXTnZsgmoftol9uivfQrPdR88WmO
+ZLYmUeQAd1rqSFMxe+KzO/qLuU8s6OF4nznwL2cPfbGZxezM4PiYmAmbbEU/3gTO
+NwjVBBA0Gfimy/fITEezFtCigo1thkaJ195g/dqY+zE3Vt4rzC03j1vx8mUHRPU6
+kkvKj8cP0j+XHX2xQDsTXTstfnom29wBmGnvSZ9HgcdL71e1VXJXwikmnO3P4J/1
+C2LeCOlWrGqWZ2c0WBLKdJnsYUx7Dm/OvkkB4lF+zWp98zS8jS/5h+1apVgEzrdT
+MvT8ydTkUr7ObKGkIhK+L+Xo5BD+V9Qf6xKGYPwhhdj/E5/kyjULrm10iQEcBBMB
+AgAGBQJcJoadAAoJEPOHY87f0iVZfiUH/3yKS5wGvTeRInse8+W1WzKuto3XzqXL
+ngb9QXWw7nCwqmNS7PbzDnufQi2ThKrMfcK14WgNYABNZPU75I+6bcb0oCB5tloo
+IUEV/2Ut/5Hl/83zFFoNA/kQKVz8kIDqgRcxC+zY2VJ4eTKHyQDvXygVk8wnKTBa
+e3gX+CIZqJHPXiiygHlbl31Mi3G1Iaxu57dP6ocV0vX1dytKSwd4Rbviwwb4L76o
+/tVT9t3GwFM15uK1SqtnAaiaktEdMi3XI4d01H3VUVz/iR0XQbf13RZoEM6CJWms
+Q/qvYlwkbKOdlahjoHrFlkhADSBaO9N1OZp3OYDjziIujMdt2IPKnmM=
+=7MQk
+-----END PGP PUBLIC KEY BLOCK-----
+'
+ }
+
+ default {
+ error "No known public key for branches/${branch}"
+ }
+ }
+}
+
+function run_join_out_err($cmd) # $arg1, $arg2, ...
+{
+ $cmdobj = Get-Command -CommandType Application -Name $cmd
+ # Don't prepend $input with BOM
+ $enc = [Console]::InputEncoding
+ [Console]::InputEncoding = [System.Text.UTF8Encoding]::new()
+ $ErrorActionPreference = "Continue" # Don't abort command on first stderr output
+ $LASTEXITCODE = 42
+ # Run command and convert ErrorRecords from stderr to plain Strings
+ $($input | & $cmdobj @args 2>&1) | %{ $_.ToString() }
+ $ErrorActionPreference = "Stop"
+ [Console]::InputEncoding = $enc
+}
+
+function gpg_verify($ascfile, $file, [ref]$ok)
+{
+ # Create temp home dir
+ if (!$env:TEMP) { error "Environment variable TEMP is not set" }
+ $gnupgtmp = Join-Path $env:TEMP "gnupg.$(New-Guid)"
+ rm_rf $gnupgtmp
+ New-Item -Type Directory -Path $gnupgtmp | Out-Null
+
+ # Import public key
+ $env:LC_MESSAGES = "C"
+ $out = ($public_key | run_join_out_err $gpg "--batch" "--no-tty" "--homedir=$gnupgtmp" "--import")
+ if ($LASTEXITCODE -ne 0) {
+ echo $out
+ exit 1
+ }
+ vecho $out
+
+ # Verify
+ $out = run_join_out_err $gpg "--batch" "--no-tty" "--homedir=$gnupgtmp" "--verify" $ascfile $file
+ if ($LASTEXITCODE -eq 0) {
+ vecho $out
+ $ok.Value = $true
+ }
+ else {
+ # print gpg error always
+ echo $out
+ $ok.Value = $false
+ }
+
+ # Stop the gpg-agent possibly started by gpg
+ if ($gpgconf) {
+ $out = run_join_out_err $gpgconf "--homedir=$gnupgtmp" "--kill" "gpg-agent"
+ if ($LASTEXITCODE -ne 0) {
+ echo $out
+ }
+ }
+
+ # Remove temp home dir
+ try {
+ rm_rf "$gnupgtmp"
+ } catch {
+ warning "$_"
+ }
+}
+
+function get_db_version($file)
+{
+ $x = (Select-String -CaseSensitive -Pattern '^[ {]*"VERSION: *[^"]*"' -Path $file).Line `
+ -replace '^[ {]*"VERSION: ([1-9][./0-9]* [^"]*)".*$','$1'
+ $v = $x -replace ' .*$',''
+ if ($v -match '^[1-9][.0-9]*$') { # trunk: get rev from expanded SVN-Id
+ $r = $x -replace '^[^$]*\$Id: drivedb\.h ([1-9][0-9]*) .*$','$1'
+ if (!($r -match '^[1-9][0-9]*$')) {
+ $r = "?"
+ }
+ $v = "$v/$r"
+ } elseif (!($v -match '^[1-9][./0-9]*$')) {
+ return ""
+ }
+ return $v
+}
+
+function mv_all($prefix, $old, $new)
+{
+ mv_f "${prefix}${old}" "${prefix}${new}"
+ mv_f "${prefix}${old}.raw" "${prefix}${new}.raw"
+ if (test_f "${prefix}${old}.raw.asc") {
+ mv_f "${prefix}${old}.raw.asc" "${prefix}${new}.raw.asc"
+ } else {
+ rm_f "${prefix}${new}.raw.asc"
+ }
+}
+
+# Set defaults
+if (!$Smartctl) {
+ $Smartctl = $default_smartctl
+}
+
+if ($Branch) {
+ $brname = $Branch -replace '^([567])\.([0-9][0-9]*)$','RELEASE_$1_$2_DRIVEDB'
+ if ($brname -eq $Branch) {
+ error "invalid branch version '${Branch}'"
+ }
+} else {
+ $brname = $default_branch
+}
+
+if (!$Drivedb) {
+ $Drivedb = $default_drivedb
+}
+
+if ($ExportKey) {
+ $key = ""
+ selectkey $brname ([ref]$key)
+ echo $key
+ exit 0
+}
+
+# Check selected source
+if (!$Url -and !$File) {
+ if (!$UrlOf) {
+ $UrlOf = "svn"
+ }
+ $Url = ""
+ selecturl $UrlOf ([ref]$Url)
+ if (!$Trunk) {
+ $Url = $Url -replace "/trunk/","/branches/$brname/"
+ $Url = $Url -replace "/master/","/origin/$brname/"
+ } elseif (!$NoVerify) {
+ error "'-Trunk' requires '-NoVerify'"
+ }
+} elseif (!$UrlOf -and $Url -and !$File) {
+ if (!($Url -match '^[a-z][a-z0-9]*:[^ ][^ ]*$')) {
+ error "${Url}: Invalid URL"
+ }
+} elseif (!$UrlOf -and !$Url -and $File) {
+} else {
+ error "only one of '-UrlOf', '-Url', '-File' is allowed"
+}
+
+# Determine path of signature file
+$FileAsc = ""
+$UrlAsc = ""
+if (!$NoVerify) {
+ if (!$Url) {
+ $FileAsc = "${File}.raw.asc"
+ } elseif ($Url -match '\?') {
+ $UrlAsc = $Url -replace '\?','.raw.asc?'
+ } else {
+ $UrlAsc = "${Url}.raw.asc"
+ }
+}
+
+# Check option compatibility
+if ($UrlOf -eq "svni") {
+ if ($Insecure) {
+ $Insecure = $false
+ }
+ else {
+ error "'-UrlOf svni' requires '-Insecure'"
+ }
+}
+
+# Check for smartctl
+if ($Smartctl -ne "-") {
+ if (!(Get-Command -Type Application -Name $Smartctl -ErrorAction Ignore)) {
+ error "${Smartctl}: not found ('-Smartctl -' to ignore)"
+ }
+}
+
+# Check for GnuPG
+$gpgconf = ""
+if (!$NoVerify) {
+ $gpgobj = $null
+ if ($gpg) {
+ $gpgobj = Get-Command -Type Application -Name $gpg -ErrorAction Ignore
+ }
+ if (!$gpgobj) {
+ error "GnuPG is not available ('-NoVerify' to ignore)"
+ }
+ $public_key = ""
+ selectkey $brname ([ref]$public_key)
+ # Check for gpgconf in same directory
+ $gpgconf = "gpgconf"
+ $gpgconfobj = Get-Command -Type Application -Name $gpgconf -ErrorAction Ignore;
+ if (!($gpgconfobj -and ((Split-Path $gpgobj.Source) -eq (Split-Path $gpgconfobj.Source)))) {
+ $gpgconf = ""
+ }
+}
+
+# Remove possible garbage from last download
+if (!$DryRun) {
+ rm_f "${Drivedb}.new" "${Drivedb}.new.raw" "${Drivedb}.new.raw.asc"
+}
+
+if ($Url) {
+ # Download
+ vecho "Download drivedb.h"
+ $errmsg = ""
+ download $Url "${Drivedb}.new" ([ref]$errmsg)
+ if ($errmsg) {
+ rm_f "${Drivedb}.new"
+ error "drivedb.h: download failed: $errmsg"
+ }
+
+ if ($UrlAsc) {
+ vecho "Download drivedb.h.raw.asc"
+ download $UrlAsc "${Drivedb}.new.raw.asc" ([ref]$errmsg)
+ if ($errmsg) {
+ rm_f "${Drivedb}.new" "${Drivedb}.new.raw.asc"
+ error "drivedb.h.raw.asc: download failed: $errmsg ('-NoVerify' to ignore)"
+ }
+ }
+} else {
+ # Copy from local file
+ if (!(test_f $File)) {
+ error "${File}: file not found"
+ }
+ if ($FileAsc -and !(test_f $FileAsc)) {
+ error "${FileAsc}: file not found ('-NoVerify' to ignore)"
+ }
+
+ vcopy $File "${Drivedb}.new"
+ if ($FileAsc) {
+ vcopy $FileAsc "${Drivedb}.new.raw.asc"
+ }
+}
+
+if ($DryRun) {
+ exit 0
+}
+
+# Check files and adjust timestamps
+$errmsg = check_file "${Drivedb}.new" '/' 10000 1000000
+if ($errmsg) {
+ rm_f "${Drivedb}.new.raw.asc"
+ mv_f "${Drivedb}.new" "${Drivedb}.error"
+ error "${Drivedb}.error: $errmsg"
+}
+touch "${Drivedb}.new"
+
+if (test_f "${Drivedb}.new.raw.asc") {
+ $errmsg = check_file "${Drivedb}.new.raw.asc" '-' 200 2000
+ if ($errmsg) {
+ rm_f "${Drivedb}.new"
+ mv_f "${Drivedb}.new.raw.asc" "${Drivedb}.error.raw.asc"
+ error "${Drivedb}.error.raw.asc: $errmsg"
+ }
+ touch "${Drivedb}.new.raw.asc"
+}
+
+# Create raw file with unexpanded SVN Id
+# (This assumes newlines are LF and not CR/LF)
+unexpand_svn_id "${Drivedb}.new" "${Drivedb}.new.raw"
+
+# Check whether installed file is identical
+$equal = $false
+if (test_f $Drivedb) {
+ if (!(test_f "${Drivedb}.raw")) {
+ # Create missing raw file
+ unexpand_svn_id "${Drivedb}" "${Drivedb}.raw"
+ }
+ # Ignore missing Id keyword expansion in new file
+ if ( (cmp "${Drivedb}.raw" "${Drivedb}.new.raw") `
+ -and ( (cmp "${Drivedb}" "${Drivedb}.new") `
+ -or (cmp "${Drivedb}.raw" "${Drivedb}.new"))) {
+ $equal = $true
+ }
+}
+
+if (!$NoVerify) {
+ # Verify raw file
+ $ok = $false
+ gpg_verify "${Drivedb}.new.raw.asc" "${Drivedb}.new.raw" ([ref]$ok)
+ if (!$ok) {
+ mv_all $Drivedb ".new" ".error"
+ if ($equal) {
+ warning "${Drivedb}: *** installed file is identical to broken new file ***"
+ }
+ error "${Drivedb}.error.raw: *** BAD signature or outdated key ***"
+ }
+}
+
+# Get version
+$newver = get_db_version "${Drivedb}.new"
+if (!$newver) {
+ if (!$Force) {
+ mv_all $Drivedb ".new" ".error"
+ error "${Drivedb}.error: no VERSION information found ('-Force' to ignore)"
+ }
+ $newver = "?/?"
+} elseif ($newver -match '/\?$') {
+ if (!$Trunk) {
+ mv_all $Drivedb ".new" ".error"
+ error "${Drivedb}.error: VERSION information is incomplete ('-Trunk' to ignore)"
+ }
+}
+
+if ($Smartctl -ne "-") {
+ # Check syntax
+ run_join_out_err $Smartctl -B "${Drivedb}.new" -P showall | Out-Null
+ if (!$?) {
+ mv_all $Drivedb ".new" ".error"
+ error "${Drivedb}.error: rejected by $Smartctl, probably no longer compatible"
+ }
+ vecho "${Smartctl}: syntax OK"
+}
+
+# Always install if missing
+rm_f "${Drivedb}.lastcheck"
+if (!(Test-Path -PathType Leaf -Path $Drivedb)) {
+ mv_all $Drivedb ".new" ""
+ iecho "$Drivedb $newver newly installed$(if ($NoVerify) {" (NOT VERIFIED)"})"
+ exit 0
+}
+
+# Keep old file if identical
+if ($equal) {
+ if (test_f "${Drivedb}.new.raw.asc") {
+ if (!(cmp "${Drivedb}.new.raw.asc" "${Drivedb}.raw.asc")) {
+ mv_f "${Drivedb}.new.raw.asc" "${Drivedb}.raw.asc"
+ iecho "${Drivedb}.raw.asc $newver updated"
+ }
+ }
+ rm_f "${Drivedb}.new" "${Drivedb}.new.raw" "${Drivedb}.new.raw.asc"
+ touch "${Drivedb}.lastcheck"
+ iecho "$Drivedb $newver is already up to date$(if ($NoVerify) {" (NOT VERIFIED)"})"
+ exit 0
+}
+
+# Check branch and file version
+$oldver = $(get_db_version "${Drivedb}")
+if (!$oldver) {
+ $oldver = "?/?"
+}
+
+if ( ($newver -match '/\?$') `
+ -or ($oldver -match '/\?$') `
+ -or (($newver -replace '/.*$','') -ne ($oldver -replace '/.*$',''))) {
+ # Always install from trunk or other branch
+ $updmsg = "replaced with"
+} elseif ((($newver -replace '^.*/','') - ($oldver -replace '^.*/','')) -lt 0) {
+ # Install older file only if '-Force' is used
+ if (!$Force) {
+ rm_f "${Drivedb}.new" "${Drivedb}.new.raw" "${Drivedb}.new.raw.asc"
+ iecho "$Drivedb $oldver not downgraded to $newver ('-Force' to override)"
+ exit 0
+ }
+ $updmsg = "downgraded to"
+} else {
+ $updmsg = "updated to"
+}
+
+mv_all $Drivedb "" ".old"
+mv_all $Drivedb ".new" ""
+iecho "$Drivedb $oldver $updmsg $newver$(if ($NoVerify) {" (NOT VERIFIED)"})"
+exit 0