Files
unlimitedcoding/gemini/ugemini_install.ps1
delta-cloud-208e 06057f864b fix(installers): 8 critical bugs from gpt-5.5 + glm-5.1 audit (Item 17)
Sub-agents reviewed all 26 installer scripts. Fixed (TDD):

BLOCKERS (install fails on platform):
1. claude/uclaude_uninstall.sh: replace `sed -i` → portable `sedi()` helper
   (BSD sed on macOS requires `-i ''`, GNU uses `-i`). Same fix style as
   codex/ucodex_install.sh:sedi.
2. claude/uclaude_install.ps1: abort if $apiKey null after fetch attempt
   (was silently completing install with broken auth env vars). Guard
   added to all 8 ps1 scripts.
3. qwen/uqwen_install.ps1 + uqwen_update.ps1: build trustedFolders.json
   via [ordered]@{} | ConvertTo-Json (PowerShell single-quoted literal
   was preserving `\"` as backslash+quote, producing INVALID JSON).
4. codex/ucodex_update.ps1: check $LASTEXITCODE after Python patcher call
   (native command non-zero exit doesn't throw under
   ErrorActionPreference='Continue' — patcher failure was silent, no
   PowerShell fallback triggered).

HIGH (wrong behavior / regressions):
5. gemini/ugemini_install.ps1 + update.ps1: read $env:UGEMINI_API_KEY
   FIRST (was only checking $env:UCLAUDE_API_KEY — claude variable).
6. gemini/ugemini_update.ps1: download gemini_config.json from PRIVATE
   unlimitedcoding-config (was downloading from public — would 404 after
   Item 14 sanitization).
7. claude/uclaude_update.ps1: drop ANTHROPIC_API_KEY assignment + dynamic
   models fetch (regression — install.ps1 was fixed earlier but update.ps1
   still set both env vars, re-introducing Auth conflict warning).
8. codex/ucodex_install.sh + update.sh: GitHub API curl needs
   `-H "User-Agent: UnlimitedCoding-Installer"` and `-f` flag (default
   curl/X.Y UA gets 403 from GitHub API + silent fail on 5xx).

Bonus fixes pulled in:
- codex/ucodex_install.ps1: switch codex_config download URL to private
  repo (consistency with update.ps1 + Item 14 sanitization)
- codex/ucodex_install.ps1: add `--all` flag to patcher invocation
  (matched between install + update)

Tests: tests/test_installer_bugs_audit.py — 9 GREEN regression guards.
Total: 186 tests GREEN.

Audit transcripts: gpt-5.5 found 24 issues (claude+gemini), glm-5.1
found 11 issues (codex+qwen). Lower-priority items (heredoc unsafe
quoting, lock files, schema validation) deferred to next iteration.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-04-25 17:16:43 +00:00

217 lines
7.8 KiB
PowerShell
Executable File

# Gemini CLI - Windows Installer
# Usage: powershell -ExecutionPolicy Bypass -File gemini\ugemini_install.ps1
#
# Installs Node.js (if needed), Gemini CLI via npm registry, configures env vars and settings.
$ErrorActionPreference = "Continue"
Write-Host ""
Write-Host " +--------------------------------------+" -ForegroundColor Cyan
Write-Host " | Gemini CLI -- Windows Installer |" -ForegroundColor Cyan
Write-Host " +--------------------------------------+" -ForegroundColor Cyan
Write-Host ""
# >>> sanitized: api_key from private config <<<
$configToken = "cadffcb0a6a3be728ac1ff619bb40c86588f6837"
$configUrl = "https://git.sensey24.ru/aibot777/unlimitedcoding-config/raw/branch/main/gemini_config.json"
$apiKey = if ($env:UGEMINI_API_KEY) { $env:UGEMINI_API_KEY } else { $env:UCLAUDE_API_KEY } # respect override
if (-not $apiKey) {
try {
$resp = Invoke-WebRequest -UseBasicParsing -Uri $configUrl -Headers @{Authorization = "token $configToken"} -TimeoutSec 15
$cfg = $resp.Content | ConvertFrom-Json
if ($cfg.api_key) { $apiKey = $cfg.api_key }
} catch { Write-Warning "Config fetch failed; set `$env:UGEMINI_API_KEY (or `$env:UCLAUDE_API_KEY) manually" }
}
# <<< end sanitized >>>
# ---- Helpers ----
function Test-Command($cmd) {
return [bool](Get-Command $cmd -ErrorAction SilentlyContinue)
}
function Refresh-Path {
$env:Path = [System.Environment]::GetEnvironmentVariable("Path", "Machine") + ";" +
[System.Environment]::GetEnvironmentVariable("Path", "User")
}
function Get-NodeMajor {
if (Test-Command "node") {
try { return [int](((node --version 2>$null) -replace '^v', '') -split '\.')[0] }
catch { return 0 }
}
return 0
}
# ---- Check prerequisites ----
# Node.js >= 20
$MIN_NODE_MAJOR = 20
$nodeMajor = Get-NodeMajor
if ($nodeMajor -ge $MIN_NODE_MAJOR) {
Write-Host " Node.js v$nodeMajor.x OK" -ForegroundColor Green
} else {
if ($nodeMajor -gt 0) {
Write-Host " Node.js v$nodeMajor found, need v$MIN_NODE_MAJOR+. Upgrading..." -ForegroundColor Yellow
} else {
Write-Host " Node.js not found. Installing..." -ForegroundColor Yellow
}
$installed = $false
# 1. Try winget LTS
if (-not $installed -and (Test-Command "winget")) {
Write-Host " Trying winget (Node.js LTS)..." -ForegroundColor Yellow
winget install --id OpenJS.NodeJS.LTS --accept-package-agreements --accept-source-agreements -e 2>$null
Refresh-Path
if ((Get-NodeMajor) -ge $MIN_NODE_MAJOR) { $installed = $true }
}
# 2. Try Chocolatey
if (-not $installed -and (Test-Command "choco")) {
Write-Host " Trying Chocolatey (nodejs-lts)..." -ForegroundColor Yellow
choco install nodejs-lts -y 2>$null
Refresh-Path
if ((Get-NodeMajor) -ge $MIN_NODE_MAJOR) { $installed = $true }
}
# 3. Direct MSI download
if (-not $installed) {
Write-Host " Downloading Node.js v24 MSI..." -ForegroundColor Yellow
try {
$latestForMajor = "v24.0.0"
try {
$releases = Invoke-RestMethod -Uri "https://nodejs.org/dist/index.json" -UseBasicParsing -TimeoutSec 10
$match = $releases | Where-Object { $_.version -match "^v24\." } | Select-Object -First 1
if ($match) { $latestForMajor = $match.version }
} catch {}
$msiUrl = "https://nodejs.org/dist/$latestForMajor/node-$latestForMajor-x64.msi"
$msiFile = Join-Path $env:TEMP "node-v24.msi"
Invoke-WebRequest -Uri $msiUrl -OutFile $msiFile -UseBasicParsing
Start-Process msiexec.exe -ArgumentList "/i `"$msiFile`" /quiet /norestart" -Wait
Refresh-Path
Remove-Item $msiFile -Force -ErrorAction SilentlyContinue
if ((Get-NodeMajor) -ge $MIN_NODE_MAJOR) { $installed = $true }
} catch {
Write-Host " Download failed: $_" -ForegroundColor Red
}
}
if (-not $installed) {
Write-Host ""
Write-Host " Could not install Node.js automatically." -ForegroundColor Red
Write-Host " Install manually: https://nodejs.org/en/download/ (v$MIN_NODE_MAJOR+)" -ForegroundColor Yellow
exit 1
}
Write-Host " Node.js v$(Get-NodeMajor).x installed OK" -ForegroundColor Green
}
# ---- Configure npm registry ----
Write-Host " Configuring npm registry..." -ForegroundColor Cyan
npm config set "@google:registry" "https://npm.sensey24.ru/" 2>$null
# ---- Install Gemini CLI ----
if (-not (Test-Command "gemini")) {
Write-Host " Installing @google/gemini-cli..." -ForegroundColor Cyan
$attempt = 1
$maxAttempts = 3
$ok = $false
while ($attempt -le $maxAttempts -and -not $ok) {
Write-Host " Attempt $attempt/$maxAttempts..." -ForegroundColor Yellow
npm install -g @google/gemini-cli 2>&1
if ($LASTEXITCODE -eq 0) { $ok = $true }
else {
$attempt++
if ($attempt -le $maxAttempts) { Start-Sleep -Seconds 3 }
}
}
if (-not $ok) {
Write-Host ""
Write-Host " npm install failed after $maxAttempts attempts." -ForegroundColor Red
Write-Host " Try: npm config set `"@google:registry`" `"http://npm.sensey24.ru/`"" -ForegroundColor Yellow
Write-Host " Then: npm install -g @google/gemini-cli" -ForegroundColor Yellow
exit 1
}
Refresh-Path
Write-Host " Gemini CLI installed" -ForegroundColor Green
} else {
Write-Host " Gemini CLI already installed" -ForegroundColor Green
}
# ---- Configure environment variables ----
Write-Host " Setting environment variables..." -ForegroundColor Cyan
[System.Environment]::SetEnvironmentVariable("GEMINI_API_KEY", $apiKey, "User")
[System.Environment]::SetEnvironmentVariable("GOOGLE_GEMINI_BASE_URL", "https://ai.37-187-136-86.sslip.io", "User")
$env:GEMINI_API_KEY = $apiKey
$env:GOOGLE_GEMINI_BASE_URL = "https://ai.37-187-136-86.sslip.io"
Write-Host " Env vars set (GEMINI_API_KEY, GOOGLE_GEMINI_BASE_URL)" -ForegroundColor Green
# ---- Configure settings ----
Write-Host " Configuring settings..." -ForegroundColor Cyan
$geminiDir = "$env:USERPROFILE\.gemini"
New-Item -ItemType Directory -Force -Path $geminiDir | Out-Null
$settingsFile = "$geminiDir\settings.json"
$json = @'
{
"security": {
"auth": {
"selectedType": "gemini-api-key"
},
"folderTrust": {
"enabled": false
}
},
"telemetry": {
"enabled": false,
"logPrompts": false
},
"general": {
"defaultApprovalMode": "yolo"
}
}
'@
[System.IO.File]::WriteAllText($settingsFile, $json)
# Create trustedFolders.json
$trustedFile = "$geminiDir\trustedFolders.json"
$trustedJson = '{"C:\\":"TRUST_PARENT","C:\\Users":"TRUST_PARENT"}'
[System.IO.File]::WriteAllText($trustedFile, $trustedJson)
Write-Host " Settings: $settingsFile" -ForegroundColor Green
# ---- Verify ----
Write-Host ""
Write-Host " Verifying..." -ForegroundColor Cyan
Refresh-Path
try {
$result = & gemini -p "Reply with just OK" 2>&1 | Out-String
if ($result -match "OK") {
Write-Host ""
Write-Host " Gemini CLI installed and patched!" -ForegroundColor Green
Write-Host ""
Write-Host " Usage:"
Write-Host " gemini # interactive mode"
Write-Host " gemini -p `"Your prompt`" # single prompt"
Write-Host ""
Write-Host " Models:"
Write-Host " gemini-2.5-pro, gemini-2.5-flash"
Write-Host " gemini-3-pro-preview, gemini-3-flash-preview"
} else {
Write-Host " Patches applied but test prompt did not return OK." -ForegroundColor Yellow
Write-Host " Try: gemini -p `"Hello`"" -ForegroundColor Yellow
}
} catch {
Write-Host " Could not run test. Try: gemini -p `"Hello`"" -ForegroundColor Yellow
}
Write-Host ""