@echo off rem ============================================================ rem Download URL (notify.exe is fetched from here on first run) -- edit as needed set "DOWNLOAD_URL=https://git.pchuan.top/cc-tools/notify/releases/download/v1.0.0/notify.exe" rem ============================================================ setlocal set "BIN=%~dp0..\bin" set "EXE=%BIN%\notify.exe" set "PART=%BIN%\notify.exe.partial" set "LOCK=%BIN%\notify.download.lock" rem hidden self-reinvocation: detached background downloader (see :downloader) if "%~1"=="__download" goto downloader rem common path: exe present -> run directly (keeps piped stdin intact) if exist "%EXE%" ( "%EXE%" %* endlocal & exit /b ) rem ---- exe missing: never block the hook ---- rem kick off the download once (atomic mkdir lock); a detached worker survives the rem hook timeout. then report progress to Claude and return immediately. if not exist "%BIN%" mkdir "%BIN%" 2>nul rem self-heal: if a previous worker was hard-killed (shutdown/crash) it may leave a rem stale lock that blocks all retries. reclaim it so -C - can resume the .partial. if exist "%LOCK%" call :reclaim rem atomic lock: only the first hook spawns the worker; others fall through and just rem report progress -> no duplicate downloads even when hooks fire concurrently. rem rem must use Start-Process (not `start /b`): a child started by cmd inherits the rem hook's stdout pipe handle, so Claude won't see EOF until the download ends -> rem the hook would block. Start-Process spawns WITHOUT inheriting handles, so the rem hook returns immediately while curl keeps running detached. mkdir "%LOCK%" 2>nul && powershell -nop -w hidden -c "Start-Process -WindowStyle Hidden -FilePath '%~f0' -ArgumentList '__download'" >nul 2>&1 rem downloaded size so far (from the .partial file), shown as X.X MB set "DLBYTES=0" if exist "%PART%" for %%A in ("%PART%") do set "DLBYTES=%%~zA" set /a DLKB=DLBYTES/1024 set /a DLMB=DLKB/1024 set /a DLF=(DLKB*10/1024)%%10 rem exit 0 so Claude parses this JSON; systemMessage is shown to the user, the rem notification itself is skipped this time (suppressOutput hides raw stdout) echo {"suppressOutput":true,"systemMessage":"[Claude Code Notify] notifier not ready, downloading in background: %DLMB%.%DLF% MB / ~14 MB done. Works automatically once finished; this notification is skipped."} endlocal & exit /b 0 :downloader rem detached worker: resume-capable download (-C -), atomic install, always free lock. rem if curl fails (slow/flaky net), the .partial is kept and the next hook resumes it. curl -fsSL -C - "%DOWNLOAD_URL%" -o "%PART%" if not errorlevel 1 move /y "%PART%" "%EXE%" >nul 2>&1 rmdir "%LOCK%" 2>nul endlocal & exit /b :reclaim rem no .partial yet => worker died before downloading anything; reclaim immediately if not exist "%PART%" ( rmdir "%LOCK%" 2>nul & goto :eof ) rem .partial idle for >120s => worker is dead (a live curl writes continuously); reclaim for /f "delims=" %%T in ('powershell -nop -c "[int]((Get-Date)-(Get-Item '%PART%').LastWriteTime).TotalSeconds" 2^>nul') do set "IDLE=%%T" if not defined IDLE goto :eof if %IDLE% geq 120 rmdir "%LOCK%" 2>nul goto :eof