* Fix Windows PowerShell 5.1 compatibility in installer Use $env:PROCESSOR_ARCHITECTURE for arch detection instead of RuntimeInformation::OSArchitecture which may not be loaded in every Windows PowerShell 5.1 session. Also fix null-reference when user PATH environment variable is empty. https://claude.ai/code/session_01VFiRDM2ZweyacXN5JneVoP * Fix executable resolution and tar extraction on Windows resolveExecutable() used `sh -lc "command -v ..."` which doesn't work on Windows (no sh). Now uses `cmd /c where` on win32. Also make tar workspace restoration tolerate symlink failures on Windows — .bin/ symlinks can't be created without Developer Mode, but the actual package directories are extracted fine. https://claude.ai/code/session_01VFiRDM2ZweyacXN5JneVoP * Broad Windows compatibility fixes across the codebase - runtime.ts: Use path.delimiter instead of hardcoded ":" for PATH construction — was completely broken on Windows - executables.ts: Add Windows fallback paths for Chrome, Edge, Brave, and Pandoc in Program Files; skip macOS-only paths on win32 - node-version.ts, check-node-version.mjs, bin/feynman.js: Show Windows-appropriate install instructions (irm | iex, nodejs.org) instead of nvm/curl on win32 - preview.ts: Support winget for pandoc auto-install on Windows, and apt on Linux (was macOS/brew only) - launch.ts: Catch unsupported signal errors on Windows - README.md: Add Windows PowerShell commands alongside macOS/Linux for all install instructions https://claude.ai/code/session_01VFiRDM2ZweyacXN5JneVoP * fix: complete windows bootstrap hardening --------- Co-authored-by: Claude <noreply@anthropic.com> Co-authored-by: Advait Paliwal <advaitspaliwal@gmail.com>
183 lines
5.9 KiB
PowerShell
183 lines
5.9 KiB
PowerShell
param(
|
|
[string]$Version = "edge"
|
|
)
|
|
|
|
$ErrorActionPreference = "Stop"
|
|
|
|
function Normalize-Version {
|
|
param([string]$RequestedVersion)
|
|
|
|
if (-not $RequestedVersion) {
|
|
return "edge"
|
|
}
|
|
|
|
switch ($RequestedVersion.ToLowerInvariant()) {
|
|
"edge" { return "edge" }
|
|
"latest" { return "latest" }
|
|
"stable" { return "latest" }
|
|
default { return $RequestedVersion.TrimStart("v") }
|
|
}
|
|
}
|
|
|
|
function Resolve-ReleaseMetadata {
|
|
param(
|
|
[string]$RequestedVersion,
|
|
[string]$AssetTarget,
|
|
[string]$BundleExtension
|
|
)
|
|
|
|
$normalizedVersion = Normalize-Version -RequestedVersion $RequestedVersion
|
|
|
|
if ($normalizedVersion -eq "edge") {
|
|
$release = Invoke-RestMethod -Uri "https://api.github.com/repos/getcompanion-ai/feynman/releases/tags/edge"
|
|
$asset = $release.assets | Where-Object { $_.name -like "feynman-*-$AssetTarget.$BundleExtension" } | Select-Object -First 1
|
|
if (-not $asset) {
|
|
throw "Failed to resolve the latest Feynman edge bundle."
|
|
}
|
|
|
|
$archiveName = $asset.name
|
|
$suffix = ".$BundleExtension"
|
|
$bundleName = $archiveName.Substring(0, $archiveName.Length - $suffix.Length)
|
|
$resolvedVersion = $bundleName.Substring("feynman-".Length)
|
|
$resolvedVersion = $resolvedVersion.Substring(0, $resolvedVersion.Length - ("-$AssetTarget").Length)
|
|
|
|
return [PSCustomObject]@{
|
|
ResolvedVersion = $resolvedVersion
|
|
BundleName = $bundleName
|
|
ArchiveName = $archiveName
|
|
DownloadUrl = $asset.browser_download_url
|
|
}
|
|
}
|
|
|
|
if ($normalizedVersion -eq "latest") {
|
|
$release = Invoke-RestMethod -Uri "https://api.github.com/repos/getcompanion-ai/feynman/releases/latest"
|
|
if (-not $release.tag_name) {
|
|
throw "Failed to resolve the latest Feynman release version."
|
|
}
|
|
|
|
$resolvedVersion = $release.tag_name.TrimStart("v")
|
|
} else {
|
|
$resolvedVersion = $normalizedVersion
|
|
}
|
|
|
|
$bundleName = "feynman-$resolvedVersion-$AssetTarget"
|
|
$archiveName = "$bundleName.$BundleExtension"
|
|
$baseUrl = if ($env:FEYNMAN_INSTALL_BASE_URL) { $env:FEYNMAN_INSTALL_BASE_URL } else { "https://github.com/getcompanion-ai/feynman/releases/download/v$resolvedVersion" }
|
|
|
|
return [PSCustomObject]@{
|
|
ResolvedVersion = $resolvedVersion
|
|
BundleName = $bundleName
|
|
ArchiveName = $archiveName
|
|
DownloadUrl = "$baseUrl/$archiveName"
|
|
}
|
|
}
|
|
|
|
function Get-ArchSuffix {
|
|
# Prefer PROCESSOR_ARCHITECTURE which is always available on Windows.
|
|
# RuntimeInformation::OSArchitecture requires .NET 4.7.1+ and may not
|
|
# be loaded in every Windows PowerShell 5.1 session.
|
|
$envArch = $env:PROCESSOR_ARCHITECTURE
|
|
if ($envArch) {
|
|
switch ($envArch) {
|
|
"AMD64" { return "x64" }
|
|
"ARM64" { return "arm64" }
|
|
}
|
|
}
|
|
|
|
try {
|
|
$arch = [System.Runtime.InteropServices.RuntimeInformation]::OSArchitecture
|
|
switch ($arch.ToString()) {
|
|
"X64" { return "x64" }
|
|
"Arm64" { return "arm64" }
|
|
}
|
|
} catch {}
|
|
|
|
throw "Unsupported architecture: $envArch"
|
|
}
|
|
|
|
$archSuffix = Get-ArchSuffix
|
|
$assetTarget = "win32-$archSuffix"
|
|
$release = Resolve-ReleaseMetadata -RequestedVersion $Version -AssetTarget $assetTarget -BundleExtension "zip"
|
|
$resolvedVersion = $release.ResolvedVersion
|
|
$bundleName = $release.BundleName
|
|
$archiveName = $release.ArchiveName
|
|
$downloadUrl = $release.DownloadUrl
|
|
|
|
$installRoot = Join-Path $env:LOCALAPPDATA "Programs\feynman"
|
|
$installBinDir = Join-Path $installRoot "bin"
|
|
$bundleDir = Join-Path $installRoot $bundleName
|
|
|
|
$tmpDir = Join-Path ([System.IO.Path]::GetTempPath()) ("feynman-install-" + [System.Guid]::NewGuid().ToString("N"))
|
|
New-Item -ItemType Directory -Path $tmpDir | Out-Null
|
|
|
|
try {
|
|
$archivePath = Join-Path $tmpDir $archiveName
|
|
Write-Host "==> Downloading $archiveName"
|
|
try {
|
|
Invoke-WebRequest -Uri $downloadUrl -OutFile $archivePath
|
|
} catch {
|
|
throw @"
|
|
Failed to download $archiveName from:
|
|
$downloadUrl
|
|
|
|
The win32-$archSuffix bundle is missing from the GitHub release.
|
|
This usually means the release exists, but not all platform bundles were uploaded.
|
|
|
|
Workarounds:
|
|
- try again after the release finishes publishing
|
|
- install via pnpm instead: pnpm add -g @companion-ai/feynman
|
|
- install via bun instead: bun add -g @companion-ai/feynman
|
|
"@
|
|
}
|
|
|
|
New-Item -ItemType Directory -Path $installRoot -Force | Out-Null
|
|
if (Test-Path $bundleDir) {
|
|
Remove-Item -Recurse -Force $bundleDir
|
|
}
|
|
|
|
Write-Host "==> Extracting $archiveName"
|
|
Expand-Archive -LiteralPath $archivePath -DestinationPath $installRoot -Force
|
|
|
|
New-Item -ItemType Directory -Path $installBinDir -Force | Out-Null
|
|
|
|
$shimPath = Join-Path $installBinDir "feynman.cmd"
|
|
Write-Host "==> Linking feynman into $installBinDir"
|
|
@"
|
|
@echo off
|
|
"$bundleDir\feynman.cmd" %*
|
|
"@ | Set-Content -Path $shimPath -Encoding ASCII
|
|
|
|
$currentUserPath = [Environment]::GetEnvironmentVariable("Path", "User")
|
|
$alreadyOnPath = $false
|
|
if ($currentUserPath) {
|
|
$alreadyOnPath = $currentUserPath.Split(';') -contains $installBinDir
|
|
}
|
|
if (-not $alreadyOnPath) {
|
|
$updatedPath = if ([string]::IsNullOrWhiteSpace($currentUserPath)) {
|
|
$installBinDir
|
|
} else {
|
|
"$currentUserPath;$installBinDir"
|
|
}
|
|
[Environment]::SetEnvironmentVariable("Path", $updatedPath, "User")
|
|
Write-Host "Updated user PATH. Open a new shell to run feynman."
|
|
} else {
|
|
Write-Host "$installBinDir is already on PATH."
|
|
}
|
|
|
|
$resolvedCommand = Get-Command feynman -ErrorAction SilentlyContinue
|
|
if ($resolvedCommand -and $resolvedCommand.Source -ne $shimPath) {
|
|
Write-Warning "Current shell resolves feynman to $($resolvedCommand.Source)"
|
|
Write-Host "Run in a new shell, or run: `$env:Path = '$installBinDir;' + `$env:Path"
|
|
Write-Host "Then run: feynman"
|
|
if ($resolvedCommand.Source -like "*node_modules*@companion-ai*feynman*") {
|
|
Write-Host "If that path is an old global npm install, remove it with: npm uninstall -g @companion-ai/feynman"
|
|
}
|
|
}
|
|
|
|
Write-Host "Feynman $resolvedVersion installed successfully."
|
|
} finally {
|
|
if (Test-Path $tmpDir) {
|
|
Remove-Item -Recurse -Force $tmpDir
|
|
}
|
|
}
|