Skip to content

Instantly share code, notes, and snippets.

@mickert
Last active March 4, 2026 23:19
Show Gist options
  • Select an option

  • Save mickert/c502202ca7dd1a56e14b48e53faa411c to your computer and use it in GitHub Desktop.

Select an option

Save mickert/c502202ca7dd1a56e14b48e53faa411c to your computer and use it in GitHub Desktop.
PowerShell script to install, update, downgrade, or uninstall smartmontools on Windows with GitHub API discovery, silent install, MD5 validation, JSON output, and optional cleanup. (c) 2026 Michel Sijmons - https://mickert.dev - License: Apache 2.0

PowerShell License Status Windows

A robust PowerShell script to install, update, downgrade, or uninstall smartmontools on Windows systems.
Supports GitHub API release discovery, silent installation, MD5 validation, downgrade logic, JSON output, and optional cleanup.

Download URL: https://gists.mickert.dev/c502202ca7dd1a56e14b48e53faa411c#file-zyntriops-smartmontoolsinstaller-ps1 Latest README: https://gists.mickert.dev/c502202ca7dd1a56e14b48e53faa411c#file-readme-md


⚡ Quick Start

Install or update to the latest version

.\ZyntriOps-SmartmontoolsInstaller.ps1 -Verbose

Install a specific version

.\ZyntriOps-SmartmontoolsInstaller.ps1 -TargetVersion 7.4 -Verbose

Downgrade (automatic uninstall → install)

.\ZyntriOps-SmartmontoolsInstaller.ps1 -TargetVersion 7.4 -Verbose

Uninstall smartmontools

.\ZyntriOps-SmartmontoolsInstaller.ps1 -Uninstall -Verbose

JSON output for RMM/CI/CD

.\ZyntriOps-SmartmontoolsInstaller.ps1 -Json

Cleanup temporary files

.\ZyntriOps-SmartmontoolsInstaller.ps1 -Cleanup -Verbose

✨ Features

  • GitHub API asset discovery
    Automatically locates the correct installer for any version (including future 7.5.x, 7.6, etc.).

  • Install / Update / Downgrade / Uninstall
    Full lifecycle support with silent installation.

  • Downgrade detection
    Automatically uninstalls the current version before installing an older one.

  • MD5 integrity validation
    Verifies installer integrity using upstream smartmontools checksums.

  • Mark-of-the-Web removal
    Prevents SmartScreen warnings and execution blocks.

  • JSON output
    Ideal for automation platforms, monitoring systems, and CI/CD pipelines.

  • Service restart support
    Optionally restarts a Windows service after install, downgrade, or uninstall.

  • Cleanup mode
    Removes installer, checksum, and stdout/stderr logs after successful execution.

  • Verbose, timestamped logging
    Logs are written to:

    C:\ProgramData\ZyntriOps\Logs\Installer\SmartmontoolsInstaller.log
    
  • WhatIf / Confirm support
    Fully integrated with PowerShell’s ShouldProcess model.


🧠 How It Works

The script performs the following steps:

  1. Detects any existing smartmontools installation via registry, where.exe, and default paths.
  2. Retrieves release metadata from the GitHub API.
  3. Determines the correct installer asset for the requested version.
  4. Downloads the installer and MD5 checksum.
  5. Validates integrity using MD5.
  6. Removes Mark-of-the-Web from the installer.
  7. Executes a silent installation (/S).
  8. Optionally restarts a service.
  9. Optionally cleans up temporary files.
  10. Outputs structured JSON when requested.

🚀 Usage Examples

Install with verbose logging

.\ZyntriOps-SmartmontoolsInstaller.ps1 -Verbose

Downgrade + JSON + Cleanup

.\ZyntriOps-SmartmontoolsInstaller.ps1 -TargetVersion 7.4 -Json -Cleanup

Uninstall + service restart

.\ZyntriOps-SmartmontoolsInstaller.ps1 -Uninstall -RestartServiceName "DiskMonitor"

WhatIf dry-run

.\ZyntriOps-SmartmontoolsInstaller.ps1 -WhatIf

📊 JSON Output Example

{
  "Status": "Installed",
  "OldVersion": "7.5",
  "NewVersion": "7.4",
  "DesiredVersion": "7.4",
  "LatestVersion": "7.5",
  "Downgrade": true,
  "ServiceRestarted": "",
  "ServiceRestartOK": false,
  "InstallerExitCode": 0,
  "RebootRequired": false,
  "CleanupRequested": true,
  "CleanupOK": true
}

🛠️ Requirements

  • Windows 10/11 or Windows Server 2016+
  • PowerShell 5.1 or PowerShell 7+
  • Administrator privileges
  • Internet access (GitHub API + downloads)

📁 File Locations

File Location
Installer $env:TEMP\smartmontools-setup.exe
MD5 checksum $env:TEMP\smartmontools-setup.exe.md5
Stdout/stderr logs $env:TEMP\smartmontools-stdout.log
Script log C:\ProgramData\ZyntriOps\Logs\Installer\SmartmontoolsInstaller.log

🧪 Test Matrix Summary

Scenario Result
Install latest
Install specific version
Downgrade
Upgrade
Uninstall
Cleanup
JSON output
WhatIf
Verbose logging
GitHub API discovery

📜 License (Apache 2.0)

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at:

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

📝 Notes

  • MD5 is used because smartmontools publishes MD5 checksums only.
  • Downgrade logic is fully automatic and safe.
  • Logging is audit-friendly and suitable for production environments.
  • Cleanup is intentionally skipped on failure to preserve artifacts for debugging.

👋 Hey visitor - Michel here

If any of my tools, gists, or guides helped you out - saved you time or money, solved a headache, or taught you something useful - I’d love it if you:

Explore my work (Gists + GitHub projects) and maybe drop a ⭐ or follow https://gists.mickert.dev/https://github.mickert.dev/

Support my work on Ko-fi https://support.mickert.dev/https://ko-fi.com/michelsprojects

Your support helps me create more tools like this.

Kind regards,
Michel Sijmons
https://mickert.dev

<#PSScriptInfo
.VERSION 1.3.1
.GUID 8f3c9c3c-4b9d-4c1e-9c4a-2f1f0e8b7a11
.AUTHOR Michel Sijmons (ZyntriOps)
.COMPANYNAME ZyntriOps
.COPYRIGHT Copyright (c) 2026 Michel Sijmons - https://mickert.dev
.LICENSEURI http://www.apache.org/licenses/LICENSE-2.0
.PROJECTURI https://zyntriops.com/tools/smartmontools-installer
.ICONURI https://zyntriops.com/tools/smartmontools-installer/icon.png
.TAGS smartmontools;installer;zyntriops;monitoring;storage;automation
.RELEASENOTES
- Added GitHub API asset discovery (future-proof for 7.5.1+, 7.6, etc.)
- Added uninstall /S enforcement
- Improved logging, cleanup, fail-fast behavior, and installer output capture
- Added JSON output for automation systems
- Added downgrade support with automatic uninstall
- Added MOTW removal and MD5 validation
#>
<#
.SYNOPSIS
Installs, updates, downgrades, or uninstalls smartmontools on Windows systems.
.DESCRIPTION
ZyntriOps-SmartmontoolsInstaller.ps1
This script installs, updates, downgrades, or uninstalls smartmontools on Windows systems.
It retrieves release metadata directly from the GitHub API, discovers the correct
installer asset for the requested version, downloads the installer and its MD5
checksum, validates file integrity, removes Mark-of-the-Web (MOTW), and performs
a fully silent installation.
If a specific version is requested via -TargetVersion, the script will install that version.
If the requested version is lower than the installed version, the script will uninstall the
current version before installing the target. If the requested version equals the installed
version, no action is taken unless -Force is specified.
When -Uninstall is specified, the script will only uninstall smartmontools (if installed).
If -RestartServiceName is provided, the specified service will be restarted after install,
downgrade, or uninstall. This is useful for services that probe for smartmontools on startup.
When -Cleanup is used, temporary files (installer, checksum, stdout/stderr logs)
are removed after a successful operation. Cleanup is skipped on failure to
preserve artifacts for debugging and auditing.
Existing installations are detected through registry keys, where.exe, and default installation
paths.
The script supports JSON output for RMM/CI/CD systems and writes operational logs
to ProgramData with timestamped entries.
This script is licensed under the Apache License, Version 2.0.
See the LICENSE section in this script.
DISCLAIMER:
This script is provided "as-is" without any warranties or guarantees.
Use at your own risk. The author assumes no liability for any damages,
data loss, misconfiguration, or operational impact resulting from the
use of this script. Always test in a controlled environment before
deploying to production.
(c) 2026 Michel Sijmons - https://mickert.dev
.PARAMETER TargetVersion
Optional. Installs a specific smartmontools version instead of the latest.
If the target version is lower than the installed version, the script will
uninstall the current version before installing the requested version.
If the target version equals the installed version, no action is taken
unless -Force is specified.
.PARAMETER Uninstall
Uninstalls smartmontools if it is installed. No installation or update
will be performed when this switch is used. If smartmontools is not
installed, the script exits gracefully. If -RestartServiceName is
specified, the service will be restarted after uninstall (or attempted
uninstall).
.PARAMETER RestartServiceName
Optional. Restarts the specified Windows service after installation,
downgrade, or uninstallation completes. This is useful for services
that probe for smartmontools on startup or require a refresh after
changes.
.PARAMETER Force
Forces installation even if the installed version appears up-to-date or
matches the requested TargetVersion.
.PARAMETER Json
Outputs structured JSON for RMM/CI/CD pipelines.
.PARAMETER Cleanup
Removes downloaded temporary files (installer, checksum, and captured
stdout/stderr logs) from $env:TEMP after a successful installation,
update, downgrade, or uninstall. This option is disabled by default
to preserve artifacts for debugging and auditing.
.PARAMETER Verbose
Enables detailed logging output. This script uses Write-Verbose extensively
to provide timestamped operational logs. Use -Verbose to display them.
.PARAMETER WhatIf
Shows what actions would be performed without making any changes. Applies to
installation, downgrade, uninstall, cleanup, and service restart operations.
.PARAMETER Confirm
Prompts for confirmation before executing actions. Applies to installation,
downgrade, uninstall, cleanup, and service restart operations.
.PARAMETER Debug
Enables debug output. Useful for troubleshooting script behavior.
.EXAMPLE
PS> .\ZyntriOps-SmartmontoolsInstaller.ps1 -Verbose
.EXAMPLE
PS> .\ZyntriOps-SmartmontoolsInstaller.ps1 -TargetVersion 7.4
.EXAMPLE
PS> .\ZyntriOps-SmartmontoolsInstaller.ps1 -Uninstall -RestartServiceName "DiskMonitor"
.EXAMPLE
PS> .\ZyntriOps-SmartmontoolsInstaller.ps1 -Json | ConvertFrom-Json
.NOTES
Author : Michel Sijmons (ZyntriOps) - https://mickert.dev
License : Apache License 2.0
Version : 1.3.1
This script must be run with administrative privileges.
Internet access is required to retrieve release metadata and download installers.
Temporary files are stored in $env:TEMP and removed only when -Cleanup is used.
Logs are written to: $env:ProgramData\ZyntriOps\Logs\Installer\SmartmontoolsInstaller.log
.LINK
https://gists.mickert.dev/c502202ca7dd1a56e14b48e53faa411c#file-zyntriops-smartmontoolsinstaller-ps1
#>
# =====================================================================
# NOTICE
# =====================================================================
# This script is part of the ZyntriOps Toolkit.
# Developed by Michel Sijmons.
# Copyright (c) 2026 Michel Sijmons and licensed under the
# Apache License, Version 2.0. See License section below.
#
# For more information, visit: https://zyntriops.com
# =====================================================================
# DISCLAIMER & LICENSE
# =====================================================================
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at:
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# DISCLAIMER:
# This script is provided "as-is" without any warranties or guarantees.
# Use at your own risk. The author assumes no liability for any damages,
# data loss, misconfiguration, or operational impact resulting from the
# use of this script. Always test in a controlled environment before
# deploying to production.
# =====================================================================
# COPYRIGHT
# =====================================================================
# Copyright (c) 2026 Michel Sijmons
# All rights reserved.
[CmdletBinding(SupportsShouldProcess = $true)]
param(
[string]$TargetVersion,
[string]$RestartServiceName,
[switch]$Force,
[switch]$Json,
[switch]$Uninstall,
[switch]$Cleanup
)
# =====================================================================
# GLOBALS
# =====================================================================
$script:ModuleName = "ZyntriOps.SmartmontoolsInstaller"
$script:ScriptVersion = "1.3.1"
$script:VerboseLoggingEnabled = $false
$script:LogFile = $null
# Initialize log file
try {
$logRoot = Join-Path $env:ProgramData "ZyntriOps\Logs\Installer"
if (-not (Test-Path $logRoot)) {
New-Item -ItemType Directory -Path $logRoot -Force | Out-Null
}
$script:LogFile = Join-Path $logRoot "SmartmontoolsInstaller.log"
}
catch {
$script:LogFile = $null
}
if ($PSBoundParameters.ContainsKey('Verbose')) {
$script:VerboseLoggingEnabled = $true
}
# =====================================================================
# LOGGING
# =====================================================================
function Write-ZOLog {
param(
[Parameter(Mandatory = $true)][string]$Message,
[ValidateSet("INFO","WARN","ERROR","DEBUG")][string]$Level = "INFO"
)
$timestamp = Get-Date -Format o
$line = "[$timestamp] [$Level] $Message"
if ($script:LogFile) {
try { Add-Content -LiteralPath $script:LogFile -Value $line } catch {}
}
switch ($Level) {
"ERROR" {
Write-Error $line
Write-Verbose $line
}
"DEBUG" {
if ($script:VerboseLoggingEnabled) { Write-Verbose $line }
}
default {
Write-Verbose $line
}
}
}
# =====================================================================
# NETWORK
# =====================================================================
function Invoke-ZOWebRequest {
[CmdletBinding()]
[OutputType([bool])]
param(
[string]$Url,
[string]$OutFile
)
Write-ZOLog "Downloading: $Url" "INFO"
try {
Invoke-WebRequest -Uri $Url -OutFile $OutFile -UseBasicParsing -TimeoutSec 60
return $true
}
catch {
Write-ZOLog "Download failed: $($_.Exception.Message)" "ERROR"
return $false
}
}
# =====================================================================
# DETECTION
# =====================================================================
function Get-ZOSmartmontoolsInstallInfo {
[CmdletBinding()]
[OutputType([PSCustomObject])]
param()
$paths = @(
"HKLM:\SOFTWARE\WOW6432Node\Microsoft\Windows\CurrentVersion\Uninstall\smartmontools",
"HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\smartmontools"
)
foreach ($p in $paths) {
if (Test-Path $p) {
$key = Get-ItemProperty $p
return [PSCustomObject]@{
DisplayVersion = $key.DisplayVersion
InstallLocation = $key.InstallLocation
UninstallString = $key.UninstallString
}
}
}
return $null
}
function Get-ZOSmartctlPath {
[CmdletBinding()]
[OutputType([string])]
param()
$info = Get-ZOSmartmontoolsInstallInfo
if ($info -and $info.InstallLocation) {
$candidate = Join-Path $info.InstallLocation "bin\smartctl.exe"
if (Test-Path $candidate) { return $candidate }
}
$where = & where.exe smartctl 2>$null
if ($where -and (Test-Path $where)) { return $where }
$default = "C:\Program Files\smartmontools\bin\smartctl.exe"
if (Test-Path $default) { return $default }
return $null
}
function Get-ZOSmartmontoolsInstalledVersion {
[CmdletBinding()]
[OutputType([string])]
param()
$smartctl = Get-ZOSmartctlPath
if (-not $smartctl) { return $null }
try {
$output = & $smartctl --version 2>$null
if (-not $output) { return $null }
$line = $output | Select-Object -First 1
if ($line -match "smartctl\s+([\d\.]+)") {
return $Matches[1]
}
}
catch {
Write-ZOLog "Failed to query smartctl version: $($_.Exception.Message)" "WARN"
}
return $null
}
# =====================================================================
# UNINSTALL (with /S enforcement)
# =====================================================================
function Uninstall-ZOSmartmontools {
[CmdletBinding(SupportsShouldProcess = $true)]
[OutputType([bool])]
param()
$info = Get-ZOSmartmontoolsInstallInfo
if (-not $info -or -not $info.UninstallString) {
Write-ZOLog "No uninstall information found; skipping uninstall." "WARN"
return $true
}
$uninstallCmd = $info.UninstallString.Trim()
# Ensure /S is present
if ($uninstallCmd -notmatch "/S\b") {
Write-ZOLog "Adding /S to uninstall parameters." "INFO"
$uninstallCmd = "$uninstallCmd /S"
}
Write-ZOLog "Final uninstall command: $uninstallCmd" "DEBUG"
if ($PSCmdlet.ShouldProcess("smartmontools", "Uninstall")) {
try {
$arguments = "/c `"$uninstallCmd`""
$proc = Start-Process -FilePath "cmd.exe" -ArgumentList $arguments -Wait -PassThru
if ($proc.ExitCode -ne 0) {
Write-ZOLog "Uninstall returned exit code $($proc.ExitCode)" "ERROR"
return $false
}
Write-ZOLog "Uninstall completed successfully." "INFO"
return $true
}
catch {
Write-ZOLog "Uninstall failed: $($_.Exception.Message)" "ERROR"
return $false
}
}
return $true
}
# =====================================================================
# GITHUB RELEASE DISCOVERY (future-proof)
# =====================================================================
function Get-ZOSmartmontoolsDownloadUrls {
[CmdletBinding()]
[OutputType([PSCustomObject])]
param(
[Parameter(Mandatory = $true)]
[string]$Version
)
# Convert version to tag format: 7.5.1 → RELEASE_7_5_1
$tag = "RELEASE_" + ($Version -replace "\.", "_")
$api = "https://api.github.com/repos/smartmontools/smartmontools/releases/tags/$tag"
Write-ZOLog "Querying GitHub for release tag $tag" "INFO"
try {
$release = Invoke-RestMethod -Uri $api -Headers @{ "User-Agent" = "ZyntriOps" }
}
catch {
throw "Failed to retrieve release metadata for ${Version}: $($_.Exception.Message)"
}
# Find installer and MD5 assets
$installer = $release.assets | Where-Object { $_.name -match "win32-setup\.exe$" }
$md5 = $release.assets | Where-Object { $_.name -match "win32-setup\.exe\.md5$" }
if (-not $installer) { throw "Could not locate installer asset for version $Version" }
if (-not $md5) { throw "Could not locate MD5 asset for version $Version" }
return [PSCustomObject]@{
Installer = $installer.browser_download_url
Md5 = $md5.browser_download_url
}
}
# =====================================================================
# SERVICE RESTART
# =====================================================================
function Restart-ZOService {
[CmdletBinding(SupportsShouldProcess = $true)]
[OutputType([bool])]
param([string]$Name)
Write-ZOLog "Requested restart of service '$Name'." "INFO"
try {
Get-Service -Name $Name -ErrorAction Stop | Out-Null
}
catch {
Write-ZOLog "Service '$Name' not found: $($_.Exception.Message)" "ERROR"
return $false
}
if ($PSCmdlet.ShouldProcess("Service $Name", "Restart")) {
try {
Restart-Service -Name $Name -Force -ErrorAction Stop
Write-ZOLog "Service '$Name' restarted successfully." "INFO"
return $true
}
catch {
Write-ZOLog "Failed to restart service '$Name': $($_.Exception.Message)" "ERROR"
return $false
}
}
return $false
}
# =====================================================================
# CLEANUP
# =====================================================================
function Remove-ZOTempFile {
[CmdletBinding(SupportsShouldProcess = $true)]
[OutputType([bool])]
param([string]$Path)
if (-not (Test-Path $Path)) {
Write-ZOLog "Temporary file not found: $Path" "WARN"
return $true
}
if ($PSCmdlet.ShouldProcess($Path, "Remove temporary file")) {
try {
Remove-Item -Path $Path -Force -ErrorAction Stop
Write-ZOLog "Removed temporary file: $Path" "INFO"
return $true
}
catch {
Write-ZOLog "Failed to remove temporary file '$Path': $($_.Exception.Message)" "ERROR"
return $false
}
}
return $true
}
# =====================================================================
# MAIN LOGIC — INITIALIZATION
# =====================================================================
Write-ZOLog "Starting $script:ModuleName v$script:ScriptVersion" "INFO"
$installedRaw = Get-ZOSmartmontoolsInstalledVersion
$installedVer = $null
if ($installedRaw) {
try { $installedVer = [version]$installedRaw } catch {}
}
# Determine latest version
try {
$latestApi = "https://api.github.com/repos/smartmontools/smartmontools/releases/latest"
$latestJson = Invoke-RestMethod -Uri $latestApi -Headers @{ "User-Agent" = "ZyntriOps" }
$latestRaw = ($latestJson.tag_name -replace "RELEASE_", "") -replace "_", "."
$latestVer = [version]$latestRaw
}
catch {
Write-ZOLog "GitHub API unavailable, falling back to 7.5" "WARN"
$latestRaw = "7.5"
$latestVer = [version]"7.5"
}
Write-ZOLog "Installed version: $installedRaw" "INFO"
Write-ZOLog "Latest version : $latestRaw" "INFO"
$installerPath = Join-Path $env:TEMP "smartmontools-setup.exe"
$md5Path = Join-Path $env:TEMP "smartmontools-setup.exe.md5"
$stdoutPath = Join-Path $env:TEMP "smartmontools-stdout.log"
$stderrPath = Join-Path $env:TEMP "smartmontools-stderr.log"
$cleanupOK = $false
$serviceRestarted = $false
$rebootRequired = $false
$installerExitCode = 0
# =====================================================================
# UNINSTALL MODE
# =====================================================================
if ($Uninstall) {
Write-ZOLog "Uninstall mode requested." "INFO"
$info = Get-ZOSmartmontoolsInstallInfo
if (-not $info) {
Write-ZOLog "smartmontools is not installed. Nothing to uninstall." "WARN"
if ($RestartServiceName) {
$serviceRestarted = Restart-ZOService -Name $RestartServiceName
if (-not $serviceRestarted) {
if ($Json) {
[PSCustomObject]@{
Status = "NotInstalled"
Action = "Uninstall"
ServiceRestarted = $RestartServiceName
ServiceRestartOK = $serviceRestarted
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 1
}
}
if ($Cleanup) {
$cleanupOK = (
(Remove-ZOTempFile -Path $installerPath) -and
(Remove-ZOTempFile -Path $md5Path) -and
(Remove-ZOTempFile -Path $stdoutPath) -and
(Remove-ZOTempFile -Path $stderrPath)
)
}
if ($Json) {
[PSCustomObject]@{
Status = "NotInstalled"
Action = "Uninstall"
ServiceRestarted = $RestartServiceName
ServiceRestartOK = $serviceRestarted
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 0
}
$uninstallOK = Uninstall-ZOSmartmontools
if (-not $uninstallOK) {
if ($Json) {
[PSCustomObject]@{
Status = "UninstallFailed"
OldVersion = $installedRaw
ServiceRestarted = $RestartServiceName
ServiceRestartOK = $false
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 1
}
if ($RestartServiceName) {
$serviceRestarted = Restart-ZOService -Name $RestartServiceName
if (-not $serviceRestarted) {
if ($Json) {
[PSCustomObject]@{
Status = "Uninstalled"
OldVersion = $installedRaw
ServiceRestarted = $RestartServiceName
ServiceRestartOK = $serviceRestarted
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 1
}
}
if ($Cleanup) {
$cleanupOK = (
(Remove-ZOTempFile -Path $installerPath) -and
(Remove-ZOTempFile -Path $md5Path) -and
(Remove-ZOTempFile -Path $stdoutPath) -and
(Remove-ZOTempFile -Path $stderrPath)
)
}
if ($Json) {
[PSCustomObject]@{
Status = "Uninstalled"
OldVersion = $installedRaw
ServiceRestarted = $RestartServiceName
ServiceRestartOK = $serviceRestarted
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 0
}
# =====================================================================
# INSTALL / UPDATE / DOWNGRADE
# =====================================================================
# Determine desired version
if ($TargetVersion) {
try {
$desiredVer = [version]$TargetVersion
$desiredRaw = $TargetVersion
}
catch {
Write-ZOLog "TargetVersion '$TargetVersion' is not a valid version." "ERROR"
if ($Json) {
[PSCustomObject]@{
Status = "InvalidTargetVersion"
TargetVersion = $TargetVersion
} | ConvertTo-Json -Depth 5
}
exit 1
}
Write-ZOLog "Target version : $desiredRaw" "INFO"
}
else {
$desiredVer = $latestVer
$desiredRaw = $latestRaw
Write-ZOLog "No TargetVersion specified; using latest: $desiredRaw" "INFO"
}
# No action needed
if ($installedVer -and $desiredVer -eq $installedVer -and -not $Force) {
Write-ZOLog "Installed version already matches desired version ($desiredRaw). No action required." "INFO"
if ($Json) {
[PSCustomObject]@{
Status = "UpToDate"
InstalledVersion = $installedRaw
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 0
}
# Downgrade detection
$downgrade = $false
if ($installedVer -and $desiredVer -lt $installedVer) {
$downgrade = $true
Write-ZOLog "Requested version ($desiredRaw) is lower than installed version ($installedRaw). Downgrade will uninstall first." "WARN"
}
# Resolve download URLs via GitHub API
try {
$urls = Get-ZOSmartmontoolsDownloadUrls -Version $desiredRaw
}
catch {
Write-ZOLog "Failed to resolve download URLs for version '$desiredRaw': $($_.Exception.Message)" "ERROR"
if ($Json) {
[PSCustomObject]@{
Status = "UrlResolutionFailed"
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
} | ConvertTo-Json -Depth 5
}
exit 1
}
# Downgrade uninstall
if ($downgrade) {
$uninstallOK = Uninstall-ZOSmartmontools
if (-not $uninstallOK) {
if ($Json) {
[PSCustomObject]@{
Status = "DowngradeUninstallFailed"
OldVersion = $installedRaw
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 1
}
}
# Download installer
if (-not (Invoke-ZOWebRequest -Url $urls.Installer -OutFile $installerPath)) {
if ($Json) {
[PSCustomObject]@{
Status = "InstallerDownloadFailed"
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 1
}
# Download MD5
if (-not (Invoke-ZOWebRequest -Url $urls.Md5 -OutFile $md5Path)) {
if ($Json) {
[PSCustomObject]@{
Status = "Md5DownloadFailed"
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 1
}
# Remove MOTW
try {
Unblock-File -Path $installerPath
Write-ZOLog "Removed Mark-of-the-Web from installer." "INFO"
}
catch {
Write-ZOLog "Failed to remove MOTW: $($_.Exception.Message)" "WARN"
}
# MD5 validation
try {
$expected = (Get-Content $md5Path).Split(" ")[0].Trim()
$actual = (Get-FileHash -Path $installerPath -Algorithm MD5).Hash.ToLower()
if ($expected -ne $actual) {
Write-ZOLog "MD5 checksum mismatch. Expected: $expected, Actual: $actual. Aborting." "ERROR"
if ($Json) {
[PSCustomObject]@{
Status = "Md5Mismatch"
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 1
}
Write-ZOLog "MD5 checksum validated successfully." "INFO"
}
catch {
Write-ZOLog "MD5 validation failed: $($_.Exception.Message)" "ERROR"
if ($Json) {
[PSCustomObject]@{
Status = "Md5ValidationFailed"
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 1
}
# Run installer
if ($PSCmdlet.ShouldProcess("smartmontools $desiredRaw", "Install/Update")) {
Write-ZOLog "Running silent installer for version $desiredRaw..." "INFO"
try {
$proc = Start-Process -FilePath $installerPath `
-ArgumentList "/S" `
-Wait `
-PassThru `
-RedirectStandardOutput $stdoutPath `
-RedirectStandardError $stderrPath
$installerExitCode = $proc.ExitCode
if (Test-Path $stdoutPath) {
Get-Content $stdoutPath | ForEach-Object { Write-ZOLog $_ "INFO" }
}
if (Test-Path $stderrPath) {
Get-Content $stderrPath | ForEach-Object { Write-ZOLog $_ "ERROR" }
}
if ($installerExitCode -ne 0 -and $installerExitCode -ne 3010) {
Write-ZOLog "Installer returned exit code $installerExitCode" "ERROR"
if ($Json) {
[PSCustomObject]@{
Status = "InstallFailed"
OldVersion = $installedRaw
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
InstallerExitCode = $installerExitCode
RebootRequired = $false
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit $installerExitCode
}
if ($installerExitCode -eq 3010) {
Write-ZOLog "Installer indicates reboot required (3010)." "WARN"
$rebootRequired = $true
}
Write-ZOLog "Installation completed successfully." "INFO"
}
catch {
Write-ZOLog "Installer execution failed: $($_.Exception.Message)" "ERROR"
if ($Json) {
[PSCustomObject]@{
Status = "InstallerExecutionFailed"
OldVersion = $installedRaw
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 1
}
}
# Query new version
$newRaw = Get-ZOSmartmontoolsInstalledVersion
Write-ZOLog "New installed version: $newRaw" "INFO"
# Restart service if requested
if ($RestartServiceName) {
$serviceRestarted = Restart-ZOService -Name $RestartServiceName
if (-not $serviceRestarted) {
if ($Json) {
[PSCustomObject]@{
Status = "Installed"
OldVersion = $installedRaw
NewVersion = $newRaw
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
Downgrade = $downgrade
ServiceRestarted = $RestartServiceName
ServiceRestartOK = $serviceRestarted
InstallerExitCode = $installerExitCode
RebootRequired = $rebootRequired
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
exit 1
}
}
# Cleanup
if ($Cleanup) {
$cleanupOK = (
(Remove-ZOTempFile -Path $installerPath) -and
(Remove-ZOTempFile -Path $md5Path) -and
(Remove-ZOTempFile -Path $stdoutPath) -and
(Remove-ZOTempFile -Path $stderrPath)
)
}
# JSON output
if ($Json) {
[PSCustomObject]@{
Status = "Installed"
OldVersion = $installedRaw
NewVersion = $newRaw
DesiredVersion = $desiredRaw
LatestVersion = $latestRaw
Downgrade = $downgrade
ServiceRestarted = $RestartServiceName
ServiceRestartOK = $serviceRestarted
InstallerExitCode = $installerExitCode
RebootRequired = $rebootRequired
CleanupRequested = $Cleanup.IsPresent
CleanupOK = $cleanupOK
} | ConvertTo-Json -Depth 5
}
# Exit code propagation
if ($installerExitCode -ne 0 -and $installerExitCode -ne 3010) {
exit $installerExitCode
}
exit 0
# ZyntriOps Action1 PowerShell Wrapper
# Copyright (c) 2026 Michel Sijmons
# Licensed under Apache License 2.0
# Wrapper generator download: https://gists.mickert.dev/d76e6a710a6cafb5b631049f6aa917e4
#
# wrapped original script: ZyntriOps-SmartmontoolsInstaller1.3.1.ps1
# Read Action1-injected variables
$TargetVersion = ${TargetVersion}
$RestartServiceName = ${RestartServiceName}
$Force = ${Force}
$Json = ${Json}
$Uninstall = ${Uninstall}
$Cleanup = ${Cleanup}
# Normalize boolean-like values
if ($TargetVersion -eq '1') { $TargetVersion = $true } elseif ($TargetVersion -eq '0') { $TargetVersion = $false }
if ($RestartServiceName -eq '1') { $RestartServiceName = $true } elseif ($RestartServiceName -eq '0') { $RestartServiceName = $false }
if ($Force -eq '1') { $Force = $true } elseif ($Force -eq '0') { $Force = $false }
if ($Json -eq '1') { $Json = $true } elseif ($Json -eq '0') { $Json = $false }
if ($Uninstall -eq '1') { $Uninstall = $true } elseif ($Uninstall -eq '0') { $Uninstall = $false }
if ($Cleanup -eq '1') { $Cleanup = $true } elseif ($Cleanup -eq '0') { $Cleanup = $false }
# Build callArgs hashtable
$callArgs = @{}
if ($TargetVersion -ne '' -and $TargetVersion -ne $null) { $callArgs['TargetVersion'] = $TargetVersion }
if ($RestartServiceName -ne '' -and $RestartServiceName -ne $null) { $callArgs['RestartServiceName'] = $RestartServiceName }
if ($Force -ne '' -and $Force -ne $null) { $callArgs['Force'] = $Force }
if ($Json -ne '' -and $Json -ne $null) { $callArgs['Json'] = $Json }
if ($Uninstall -ne '' -and $Uninstall -ne $null) { $callArgs['Uninstall'] = $Uninstall }
if ($Cleanup -ne '' -and $Cleanup -ne $null) { $callArgs['Cleanup'] = $Cleanup }
# Log execution
$jsonArgs = ($callArgs | ConvertTo-Json -Compress)
Write-Host "ZyntriOps Action1 PowerShell Wrapper"
Write-Host "Licensed under Apache License 2.0"
Write-Host "Copyright (c) 2026 Michel Sijmons"
Write-Host "Wrapper generator download: https://gists.mickert.dev/d76e6a710a6cafb5b631049f6aa917e4"
Write-Host ""
$ts = (Get-Date).ToString("yyyy-MM-ddTHH:mm:ss.fffffffK")
Write-Host "[$ts] Executing wrapped script 'ZyntriOps-SmartmontoolsInstaller1.3.1.ps1' with arguments: $jsonArgs"
# Decode original script
$scriptPath = "$env:TEMP\Action1WrappedScript.ps1"
$base64 = "<#PSScriptInfo
.VERSION 1.3.1
.GUID 8f3c9c3c-4b9d-4c1e-9c4a-2f1f0e8b7a11
.AUTHOR Michel Sijmons (ZyntriOps)
.COMPANYNAME ZyntriOps
.COPYRIGHT Copyright (c) 2026 Michel Sijmons - https://mickert.dev
.LICENSEURI http://www.apache.org/licenses/LICENSE-2.0
.PROJECTURI https://zyntriops.com/tools/smartmontools-installer
.ICONURI https://zyntriops.com/tools/smartmontools-installer/icon.png
.TAGS smartmontools;installer;zyntriops;monitoring;storage;automation
.RELEASENOTES
- Added GitHub API asset discovery (future-proof for 7.5.1+, 7.6, etc.)
- Added uninstall /S enforcement
- Improved logging, cleanup, fail-fast behavior, and installer output capture
- Added JSON output for automation systems
- Added downgrade support with automatic uninstall
- Added MOTW removal and MD5 validation
#>

<#
.SYNOPSIS
    Installs, updates, downgrades, or uninstalls smartmontools on Windows systems.

.DESCRIPTION
    ZyntriOps-SmartmontoolsInstaller.ps1
    This script installs, updates, downgrades, or uninstalls smartmontools on Windows systems.
    It retrieves release metadata directly from the GitHub API, discovers the correct
    installer asset for the requested version, downloads the installer and its MD5
    checksum, validates file integrity, removes Mark-of-the-Web (MOTW), and performs
    a fully silent installation.

    If a specific version is requested via -TargetVersion, the script will install that version.
    If the requested version is lower than the installed version, the script will uninstall the
    current version before installing the target. If the requested version equals the installed
    version, no action is taken unless -Force is specified.

    When -Uninstall is specified, the script will only uninstall smartmontools (if installed).
    If -RestartServiceName is provided, the specified service will be restarted after install,
    downgrade, or uninstall. This is useful for services that probe for smartmontools on startup.

    When -Cleanup is used, temporary files (installer, checksum, stdout/stderr logs)
    are removed after a successful operation. Cleanup is skipped on failure to
    preserve artifacts for debugging and auditing.

    Existing installations are detected through registry keys, where.exe, and default installation
    paths.
    
    The script supports JSON output for RMM/CI/CD systems and writes operational logs
    to ProgramData with timestamped entries.

    This script is licensed under the Apache License, Version 2.0.
    See the LICENSE section in this script.

    DISCLAIMER:
    This script is provided "as-is" without any warranties or guarantees.
    Use at your own risk. The author assumes no liability for any damages,
    data loss, misconfiguration, or operational impact resulting from the
    use of this script. Always test in a controlled environment before
    deploying to production.

    (c) 2026 Michel Sijmons - https://mickert.dev

.PARAMETER TargetVersion
    Optional. Installs a specific smartmontools version instead of the latest.
    If the target version is lower than the installed version, the script will
    uninstall the current version before installing the requested version.
    If the target version equals the installed version, no action is taken
    unless -Force is specified.

.PARAMETER Uninstall
    Uninstalls smartmontools if it is installed. No installation or update
    will be performed when this switch is used. If smartmontools is not
    installed, the script exits gracefully. If -RestartServiceName is
    specified, the service will be restarted after uninstall (or attempted
    uninstall).

.PARAMETER RestartServiceName
    Optional. Restarts the specified Windows service after installation,
    downgrade, or uninstallation completes. This is useful for services
    that probe for smartmontools on startup or require a refresh after
    changes.

.PARAMETER Force
    Forces installation even if the installed version appears up-to-date or
    matches the requested TargetVersion.

.PARAMETER Json
    Outputs structured JSON for RMM/CI/CD pipelines.

.PARAMETER Cleanup
    Removes downloaded temporary files (installer, checksum, and captured
    stdout/stderr logs) from $env:TEMP after a successful installation,
    update, downgrade, or uninstall. This option is disabled by default
    to preserve artifacts for debugging and auditing.

.PARAMETER Verbose
    Enables detailed logging output. This script uses Write-Verbose extensively
    to provide timestamped operational logs. Use -Verbose to display them.

.PARAMETER WhatIf
    Shows what actions would be performed without making any changes. Applies to
    installation, downgrade, uninstall, cleanup, and service restart operations.

.PARAMETER Confirm
    Prompts for confirmation before executing actions. Applies to installation,
    downgrade, uninstall, cleanup, and service restart operations.

.PARAMETER Debug
    Enables debug output. Useful for troubleshooting script behavior.

.EXAMPLE
    PS> .\ZyntriOps-SmartmontoolsInstaller.ps1 -Verbose

.EXAMPLE
    PS> .\ZyntriOps-SmartmontoolsInstaller.ps1 -TargetVersion 7.4

.EXAMPLE
    PS> .\ZyntriOps-SmartmontoolsInstaller.ps1 -Uninstall -RestartServiceName "DiskMonitor"

.EXAMPLE
    PS> .\ZyntriOps-SmartmontoolsInstaller.ps1 -Json | ConvertFrom-Json

.NOTES
    Author   : Michel Sijmons (ZyntriOps) - https://mickert.dev
    License  : Apache License 2.0
    Version  : 1.3.1

    This script must be run with administrative privileges.
    Internet access is required to retrieve release metadata and download installers.
    Temporary files are stored in $env:TEMP and removed only when -Cleanup is used.
    Logs are written to: $env:ProgramData\ZyntriOps\Logs\Installer\SmartmontoolsInstaller.log

.LINK
    https://gists.mickert.dev/c502202ca7dd1a56e14b48e53faa411c#file-zyntriops-smartmontoolsinstaller-ps1

#>

# =====================================================================
# NOTICE
# =====================================================================
# This script is part of the ZyntriOps Toolkit.
# Developed by Michel Sijmons.
# Copyright (c) 2026 Michel Sijmons and licensed under the
# Apache License, Version 2.0. See License section below.
#
# For more information, visit: https://zyntriops.com

# =====================================================================
# DISCLAIMER & LICENSE
# =====================================================================
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at:
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# DISCLAIMER:
# This script is provided "as-is" without any warranties or guarantees.
# Use at your own risk. The author assumes no liability for any damages,
# data loss, misconfiguration, or operational impact resulting from the
# use of this script. Always test in a controlled environment before
# deploying to production.

# =====================================================================
# COPYRIGHT
# =====================================================================
# Copyright (c) 2026 Michel Sijmons
# All rights reserved.

[CmdletBinding(SupportsShouldProcess = $true)]
param(
    [string]$TargetVersion,
    [string]$RestartServiceName,
    [switch]$Force,
    [switch]$Json,
    [switch]$Uninstall,
    [switch]$Cleanup
)

# =====================================================================
# GLOBALS
# =====================================================================
$script:ModuleName    = "ZyntriOps.SmartmontoolsInstaller"
$script:ScriptVersion = "1.3.1"
$script:VerboseLoggingEnabled = $false
$script:LogFile = $null

# Initialize log file
try {
    $logRoot = Join-Path $env:ProgramData "ZyntriOps\Logs\Installer"
    if (-not (Test-Path $logRoot)) {
        New-Item -ItemType Directory -Path $logRoot -Force | Out-Null
    }
    $script:LogFile = Join-Path $logRoot "SmartmontoolsInstaller.log"
}
catch {
    $script:LogFile = $null
}

if ($PSBoundParameters.ContainsKey('Verbose')) {
    $script:VerboseLoggingEnabled = $true
}

# =====================================================================
# LOGGING
# =====================================================================
function Write-ZOLog {
    param(
        [Parameter(Mandatory = $true)][string]$Message,
        [ValidateSet("INFO","WARN","ERROR","DEBUG")][string]$Level = "INFO"
    )

    $timestamp = Get-Date -Format o
    $line = "[$timestamp] [$Level] $Message"

    if ($script:LogFile) {
        try { Add-Content -LiteralPath $script:LogFile -Value $line } catch {}
    }

    switch ($Level) {
        "ERROR" {
            Write-Error $line
            Write-Verbose $line
        }
        "DEBUG" {
            if ($script:VerboseLoggingEnabled) { Write-Verbose $line }
        }
        default {
            Write-Verbose $line
        }
    }
}

# =====================================================================
# NETWORK
# =====================================================================
function Invoke-ZOWebRequest {
    [CmdletBinding()]
    [OutputType([bool])]
    param(
        [string]$Url,
        [string]$OutFile
    )

    Write-ZOLog "Downloading: $Url" "INFO"

    try {
        Invoke-WebRequest -Uri $Url -OutFile $OutFile -UseBasicParsing -TimeoutSec 60
        return $true
    }
    catch {
        Write-ZOLog "Download failed: $($_.Exception.Message)" "ERROR"
        return $false
    }
}

# =====================================================================
# DETECTION
# =====================================================================
function Get-ZOSmartmontoolsInstallInfo {
    [CmdletBinding()]
    [OutputType([PSCustomObject])]
    param()

    $paths = @(
        "HKLM:\SOFTWARE\WOW6432Node\Microsoft\Windows\CurrentVersion\Uninstall\smartmontools",
        "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\smartmontools"
    )

    foreach ($p in $paths) {
        if (Test-Path $p) {
            $key = Get-ItemProperty $p
            return [PSCustomObject]@{
                DisplayVersion  = $key.DisplayVersion
                InstallLocation = $key.InstallLocation
                UninstallString = $key.UninstallString
            }
        }
    }

    return $null
}

function Get-ZOSmartctlPath {
    [CmdletBinding()]
    [OutputType([string])]
    param()

    $info = Get-ZOSmartmontoolsInstallInfo
    if ($info -and $info.InstallLocation) {
        $candidate = Join-Path $info.InstallLocation "bin\smartctl.exe"
        if (Test-Path $candidate) { return $candidate }
    }

    $where = & where.exe smartctl 2>$null
    if ($where -and (Test-Path $where)) { return $where }

    $default = "C:\Program Files\smartmontools\bin\smartctl.exe"
    if (Test-Path $default) { return $default }

    return $null
}

function Get-ZOSmartmontoolsInstalledVersion {
    [CmdletBinding()]
    [OutputType([string])]
    param()

    $smartctl = Get-ZOSmartctlPath
    if (-not $smartctl) { return $null }

    try {
        $output = & $smartctl --version 2>$null
        if (-not $output) { return $null }

        $line = $output | Select-Object -First 1
        if ($line -match "smartctl\s+([\d\.]+)") {
            return $Matches[1]
        }
    }
    catch {
        Write-ZOLog "Failed to query smartctl version: $($_.Exception.Message)" "WARN"
    }

    return $null
}

# =====================================================================
# UNINSTALL (with /S enforcement)
# =====================================================================
function Uninstall-ZOSmartmontools {
    [CmdletBinding(SupportsShouldProcess = $true)]
    [OutputType([bool])]
    param()

    $info = Get-ZOSmartmontoolsInstallInfo
    if (-not $info -or -not $info.UninstallString) {
        Write-ZOLog "No uninstall information found; skipping uninstall." "WARN"
        return $true
    }

    $uninstallCmd = $info.UninstallString.Trim()

    # Ensure /S is present
    if ($uninstallCmd -notmatch "/S\b") {
        Write-ZOLog "Adding /S to uninstall parameters." "INFO"
        $uninstallCmd = "$uninstallCmd /S"
    }

    Write-ZOLog "Final uninstall command: $uninstallCmd" "DEBUG"

    if ($PSCmdlet.ShouldProcess("smartmontools", "Uninstall")) {
        try {
            $arguments = "/c `"$uninstallCmd`""
            $proc = Start-Process -FilePath "cmd.exe" -ArgumentList $arguments -Wait -PassThru

            if ($proc.ExitCode -ne 0) {
                Write-ZOLog "Uninstall returned exit code $($proc.ExitCode)" "ERROR"
                return $false
            }

            Write-ZOLog "Uninstall completed successfully." "INFO"
            return $true
        }
        catch {
            Write-ZOLog "Uninstall failed: $($_.Exception.Message)" "ERROR"
            return $false
        }
    }

    return $true
}

# =====================================================================
# GITHUB RELEASE DISCOVERY (future-proof)
# =====================================================================
function Get-ZOSmartmontoolsDownloadUrls {
    [CmdletBinding()]
    [OutputType([PSCustomObject])]
    param(
        [Parameter(Mandatory = $true)]
        [string]$Version
    )

    # Convert version to tag format: 7.5.1 → RELEASE_7_5_1
    $tag = "RELEASE_" + ($Version -replace "\.", "_")

    $api = "https://api.github.com/repos/smartmontools/smartmontools/releases/tags/$tag"
    Write-ZOLog "Querying GitHub for release tag $tag" "INFO"

    try {
        $release = Invoke-RestMethod -Uri $api -Headers @{ "User-Agent" = "ZyntriOps" }
    }
    catch {
        throw "Failed to retrieve release metadata for ${Version}: $($_.Exception.Message)"
    }

    # Find installer and MD5 assets
    $installer = $release.assets | Where-Object { $_.name -match "win32-setup\.exe$" }
    $md5       = $release.assets | Where-Object { $_.name -match "win32-setup\.exe\.md5$" }

    if (-not $installer) { throw "Could not locate installer asset for version $Version" }
    if (-not $md5)       { throw "Could not locate MD5 asset for version $Version" }

    return [PSCustomObject]@{
        Installer = $installer.browser_download_url
        Md5       = $md5.browser_download_url
    }
}

# =====================================================================
# SERVICE RESTART
# =====================================================================
function Restart-ZOService {
    [CmdletBinding(SupportsShouldProcess = $true)]
    [OutputType([bool])]
    param([string]$Name)

    Write-ZOLog "Requested restart of service '$Name'." "INFO"

    try {
        Get-Service -Name $Name -ErrorAction Stop | Out-Null
    }
    catch {
        Write-ZOLog "Service '$Name' not found: $($_.Exception.Message)" "ERROR"
        return $false
    }

    if ($PSCmdlet.ShouldProcess("Service $Name", "Restart")) {
        try {
            Restart-Service -Name $Name -Force -ErrorAction Stop
            Write-ZOLog "Service '$Name' restarted successfully." "INFO"
            return $true
        }
        catch {
            Write-ZOLog "Failed to restart service '$Name': $($_.Exception.Message)" "ERROR"
            return $false
        }
    }

    return $false
}

# =====================================================================
# CLEANUP
# =====================================================================
function Remove-ZOTempFile {
    [CmdletBinding(SupportsShouldProcess = $true)]
    [OutputType([bool])]
    param([string]$Path)

    if (-not (Test-Path $Path)) {
        Write-ZOLog "Temporary file not found: $Path" "WARN"
        return $true
    }

    if ($PSCmdlet.ShouldProcess($Path, "Remove temporary file")) {
        try {
            Remove-Item -Path $Path -Force -ErrorAction Stop
            Write-ZOLog "Removed temporary file: $Path" "INFO"
            return $true
        }
        catch {
            Write-ZOLog "Failed to remove temporary file '$Path': $($_.Exception.Message)" "ERROR"
            return $false
        }
    }

    return $true
}

# =====================================================================
# MAIN LOGIC — INITIALIZATION
# =====================================================================
Write-ZOLog "Starting $script:ModuleName v$script:ScriptVersion" "INFO"

$installedRaw = Get-ZOSmartmontoolsInstalledVersion
$installedVer = $null
if ($installedRaw) {
    try { $installedVer = [version]$installedRaw } catch {}
}

# Determine latest version
try {
    $latestApi = "https://api.github.com/repos/smartmontools/smartmontools/releases/latest"
    $latestJson = Invoke-RestMethod -Uri $latestApi -Headers @{ "User-Agent" = "ZyntriOps" }
    $latestRaw = ($latestJson.tag_name -replace "RELEASE_", "") -replace "_", "."
    $latestVer = [version]$latestRaw
}
catch {
    Write-ZOLog "GitHub API unavailable, falling back to 7.5" "WARN"
    $latestRaw = "7.5"
    $latestVer = [version]"7.5"
}

Write-ZOLog "Installed version: $installedRaw" "INFO"
Write-ZOLog "Latest version   : $latestRaw" "INFO"

$installerPath = Join-Path $env:TEMP "smartmontools-setup.exe"
$md5Path       = Join-Path $env:TEMP "smartmontools-setup.exe.md5"
$stdoutPath    = Join-Path $env:TEMP "smartmontools-stdout.log"
$stderrPath    = Join-Path $env:TEMP "smartmontools-stderr.log"

$cleanupOK = $false
$serviceRestarted = $false
$rebootRequired = $false
$installerExitCode = 0

# =====================================================================
# UNINSTALL MODE
# =====================================================================
if ($Uninstall) {
    Write-ZOLog "Uninstall mode requested." "INFO"

    $info = Get-ZOSmartmontoolsInstallInfo

    if (-not $info) {
        Write-ZOLog "smartmontools is not installed. Nothing to uninstall." "WARN"

        if ($RestartServiceName) {
            $serviceRestarted = Restart-ZOService -Name $RestartServiceName
            if (-not $serviceRestarted) {
                if ($Json) {
                    [PSCustomObject]@{
                        Status           = "NotInstalled"
                        Action           = "Uninstall"
                        ServiceRestarted = $RestartServiceName
                        ServiceRestartOK = $serviceRestarted
                        CleanupRequested = $Cleanup.IsPresent
                        CleanupOK        = $cleanupOK
                    } | ConvertTo-Json -Depth 5
                }
                exit 1
            }
        }

        if ($Cleanup) {
            $cleanupOK = (
                (Remove-ZOTempFile -Path $installerPath) -and
                (Remove-ZOTempFile -Path $md5Path) -and
                (Remove-ZOTempFile -Path $stdoutPath) -and
                (Remove-ZOTempFile -Path $stderrPath)
            )
        }

        if ($Json) {
            [PSCustomObject]@{
                Status           = "NotInstalled"
                Action           = "Uninstall"
                ServiceRestarted = $RestartServiceName
                ServiceRestartOK = $serviceRestarted
                CleanupRequested = $Cleanup.IsPresent
                CleanupOK        = $cleanupOK
            } | ConvertTo-Json -Depth 5
        }
        exit 0
    }

    $uninstallOK = Uninstall-ZOSmartmontools
    if (-not $uninstallOK) {
        if ($Json) {
            [PSCustomObject]@{
                Status           = "UninstallFailed"
                OldVersion       = $installedRaw
                ServiceRestarted = $RestartServiceName
                ServiceRestartOK = $false
                CleanupRequested = $Cleanup.IsPresent
                CleanupOK        = $cleanupOK
            } | ConvertTo-Json -Depth 5
        }
        exit 1
    }

    if ($RestartServiceName) {
        $serviceRestarted = Restart-ZOService -Name $RestartServiceName
        if (-not $serviceRestarted) {
            if ($Json) {
                [PSCustomObject]@{
                    Status           = "Uninstalled"
                    OldVersion       = $installedRaw
                    ServiceRestarted = $RestartServiceName
                    ServiceRestartOK = $serviceRestarted
                    CleanupRequested = $Cleanup.IsPresent
                    CleanupOK        = $cleanupOK
                } | ConvertTo-Json -Depth 5
            }
            exit 1
        }
    }

    if ($Cleanup) {
        $cleanupOK = (
            (Remove-ZOTempFile -Path $installerPath) -and
            (Remove-ZOTempFile -Path $md5Path) -and
            (Remove-ZOTempFile -Path $stdoutPath) -and
            (Remove-ZOTempFile -Path $stderrPath)
        )
    }

    if ($Json) {
        [PSCustomObject]@{
            Status           = "Uninstalled"
            OldVersion       = $installedRaw
            ServiceRestarted = $RestartServiceName
            ServiceRestartOK = $serviceRestarted
            CleanupRequested = $Cleanup.IsPresent
            CleanupOK        = $cleanupOK
        } | ConvertTo-Json -Depth 5
    }
    exit 0
}

# =====================================================================
# INSTALL / UPDATE / DOWNGRADE
# =====================================================================

# Determine desired version
if ($TargetVersion) {
    try {
        $desiredVer = [version]$TargetVersion
        $desiredRaw = $TargetVersion
    }
    catch {
        Write-ZOLog "TargetVersion '$TargetVersion' is not a valid version." "ERROR"
        if ($Json) {
            [PSCustomObject]@{
                Status         = "InvalidTargetVersion"
                TargetVersion  = $TargetVersion
            } | ConvertTo-Json -Depth 5
        }
        exit 1
    }
    Write-ZOLog "Target version   : $desiredRaw" "INFO"
}
else {
    $desiredVer = $latestVer
    $desiredRaw = $latestRaw
    Write-ZOLog "No TargetVersion specified; using latest: $desiredRaw" "INFO"
}

# No action needed
if ($installedVer -and $desiredVer -eq $installedVer -and -not $Force) {
    Write-ZOLog "Installed version already matches desired version ($desiredRaw). No action required." "INFO"
    if ($Json) {
        [PSCustomObject]@{
            Status           = "UpToDate"
            InstalledVersion = $installedRaw
            DesiredVersion   = $desiredRaw
            LatestVersion    = $latestRaw
            CleanupRequested = $Cleanup.IsPresent
            CleanupOK        = $cleanupOK
        } | ConvertTo-Json -Depth 5
    }
    exit 0
}

# Downgrade detection
$downgrade = $false
if ($installedVer -and $desiredVer -lt $installedVer) {
    $downgrade = $true
    Write-ZOLog "Requested version ($desiredRaw) is lower than installed version ($installedRaw). Downgrade will uninstall first." "WARN"
}

# Resolve download URLs via GitHub API
try {
    $urls = Get-ZOSmartmontoolsDownloadUrls -Version $desiredRaw
}
catch {
    Write-ZOLog "Failed to resolve download URLs for version '$desiredRaw': $($_.Exception.Message)" "ERROR"
    if ($Json) {
        [PSCustomObject]@{
            Status         = "UrlResolutionFailed"
            DesiredVersion = $desiredRaw
            LatestVersion  = $latestRaw
        } | ConvertTo-Json -Depth 5
    }
    exit 1
}

# Downgrade uninstall
if ($downgrade) {
    $uninstallOK = Uninstall-ZOSmartmontools
    if (-not $uninstallOK) {
        if ($Json) {
            [PSCustomObject]@{
                Status           = "DowngradeUninstallFailed"
                OldVersion       = $installedRaw
                DesiredVersion   = $desiredRaw
                LatestVersion    = $latestRaw
                CleanupRequested = $Cleanup.IsPresent
                CleanupOK        = $cleanupOK
            } | ConvertTo-Json -Depth 5
        }
        exit 1
    }
}

# Download installer
if (-not (Invoke-ZOWebRequest -Url $urls.Installer -OutFile $installerPath)) {
    if ($Json) {
        [PSCustomObject]@{
            Status           = "InstallerDownloadFailed"
            DesiredVersion   = $desiredRaw
            LatestVersion    = $latestRaw
            CleanupRequested = $Cleanup.IsPresent
            CleanupOK        = $cleanupOK
        } | ConvertTo-Json -Depth 5
    }
    exit 1
}

# Download MD5
if (-not (Invoke-ZOWebRequest -Url $urls.Md5 -OutFile $md5Path)) {
    if ($Json) {
        [PSCustomObject]@{
            Status           = "Md5DownloadFailed"
            DesiredVersion   = $desiredRaw
            LatestVersion    = $latestRaw
            CleanupRequested = $Cleanup.IsPresent
            CleanupOK        = $cleanupOK
        } | ConvertTo-Json -Depth 5
    }
    exit 1
}

# Remove MOTW
try {
    Unblock-File -Path $installerPath
    Write-ZOLog "Removed Mark-of-the-Web from installer." "INFO"
}
catch {
    Write-ZOLog "Failed to remove MOTW: $($_.Exception.Message)" "WARN"
}

# MD5 validation
try {
    $expected = (Get-Content $md5Path).Split(" ")[0].Trim()
    $actual   = (Get-FileHash -Path $installerPath -Algorithm MD5).Hash.ToLower()

    if ($expected -ne $actual) {
        Write-ZOLog "MD5 checksum mismatch. Expected: $expected, Actual: $actual. Aborting." "ERROR"
        if ($Json) {
            [PSCustomObject]@{
                Status           = "Md5Mismatch"
                DesiredVersion   = $desiredRaw
                LatestVersion    = $latestRaw
                CleanupRequested = $Cleanup.IsPresent
                CleanupOK        = $cleanupOK
            } | ConvertTo-Json -Depth 5
        }
        exit 1
    }

    Write-ZOLog "MD5 checksum validated successfully." "INFO"
}
catch {
    Write-ZOLog "MD5 validation failed: $($_.Exception.Message)" "ERROR"
    if ($Json) {
        [PSCustomObject]@{
            Status           = "Md5ValidationFailed"
            DesiredVersion   = $desiredRaw
            LatestVersion    = $latestRaw
            CleanupRequested = $Cleanup.IsPresent
            CleanupOK        = $cleanupOK
        } | ConvertTo-Json -Depth 5
    }
    exit 1
}

# Run installer
if ($PSCmdlet.ShouldProcess("smartmontools $desiredRaw", "Install/Update")) {
    Write-ZOLog "Running silent installer for version $desiredRaw..." "INFO"

    try {
        $proc = Start-Process -FilePath $installerPath `
                              -ArgumentList "/S" `
                              -Wait `
                              -PassThru `
                              -RedirectStandardOutput $stdoutPath `
                              -RedirectStandardError  $stderrPath

        $installerExitCode = $proc.ExitCode

        if (Test-Path $stdoutPath) {
            Get-Content $stdoutPath | ForEach-Object { Write-ZOLog $_ "INFO" }
        }
        if (Test-Path $stderrPath) {
            Get-Content $stderrPath | ForEach-Object { Write-ZOLog $_ "ERROR" }
        }

        if ($installerExitCode -ne 0 -and $installerExitCode -ne 3010) {
            Write-ZOLog "Installer returned exit code $installerExitCode" "ERROR"
            if ($Json) {
                [PSCustomObject]@{
                    Status            = "InstallFailed"
                    OldVersion        = $installedRaw
                    DesiredVersion    = $desiredRaw
                    LatestVersion     = $latestRaw
                    InstallerExitCode = $installerExitCode
                    RebootRequired    = $false
                    CleanupRequested  = $Cleanup.IsPresent
                    CleanupOK         = $cleanupOK
                } | ConvertTo-Json -Depth 5
            }
            exit $installerExitCode
        }

        if ($installerExitCode -eq 3010) {
            Write-ZOLog "Installer indicates reboot required (3010)." "WARN"
            $rebootRequired = $true
        }

        Write-ZOLog "Installation completed successfully." "INFO"
    }
    catch {
        Write-ZOLog "Installer execution failed: $($_.Exception.Message)" "ERROR"
        if ($Json) {
            [PSCustomObject]@{
                Status           = "InstallerExecutionFailed"
                OldVersion       = $installedRaw
                DesiredVersion   = $desiredRaw
                LatestVersion    = $latestRaw
                CleanupRequested = $Cleanup.IsPresent
                CleanupOK        = $cleanupOK
            } | ConvertTo-Json -Depth 5
        }
        exit 1
    }
}

# Query new version
$newRaw = Get-ZOSmartmontoolsInstalledVersion
Write-ZOLog "New installed version: $newRaw" "INFO"

# Restart service if requested
if ($RestartServiceName) {
    $serviceRestarted = Restart-ZOService -Name $RestartServiceName
    if (-not $serviceRestarted) {
        if ($Json) {
            [PSCustomObject]@{
                Status            = "Installed"
                OldVersion        = $installedRaw
                NewVersion        = $newRaw
                DesiredVersion    = $desiredRaw
                LatestVersion     = $latestRaw
                Downgrade         = $downgrade
                ServiceRestarted  = $RestartServiceName
                ServiceRestartOK  = $serviceRestarted
                InstallerExitCode = $installerExitCode
                RebootRequired    = $rebootRequired
                CleanupRequested  = $Cleanup.IsPresent
                CleanupOK         = $cleanupOK
            } | ConvertTo-Json -Depth 5
        }
        exit 1
    }
}

# Cleanup
if ($Cleanup) {
    $cleanupOK = (
        (Remove-ZOTempFile -Path $installerPath) -and
        (Remove-ZOTempFile -Path $md5Path) -and
        (Remove-ZOTempFile -Path $stdoutPath) -and
        (Remove-ZOTempFile -Path $stderrPath)
    )
}

# JSON output
if ($Json) {
    [PSCustomObject]@{
        Status            = "Installed"
        OldVersion        = $installedRaw
        NewVersion        = $newRaw
        DesiredVersion    = $desiredRaw
        LatestVersion     = $latestRaw
        Downgrade         = $downgrade
        ServiceRestarted  = $RestartServiceName
        ServiceRestartOK  = $serviceRestarted
        InstallerExitCode = $installerExitCode
        RebootRequired    = $rebootRequired
        CleanupRequested  = $Cleanup.IsPresent
        CleanupOK         = $cleanupOK
    } | ConvertTo-Json -Depth 5
}

# Exit code propagation
if ($installerExitCode -ne 0 -and $installerExitCode -ne 3010) {
    exit $installerExitCode
}

exit 0"
[IO.File]::WriteAllBytes($scriptPath, [Convert]::FromBase64String($base64))
# Execute original script
& $scriptPath @callArgs
exit $LASTEXITCODE
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment