feat: AOT 单文件静态链接 + Release 引导脚本

- csproj 静态链接 Skia/HarfBuzz/ANGLE(CoreUtils.*.Static),AOT 产出真单文件
- 发布后清理 .dll/.lib/.pdb,只留 notify.exe
- bin/notify.cmd 与 notify.sh:首次运行从 Release 下载 notify.exe
  原子下载(临时文件+改名)+ mkdir 锁,并发不重复下载,带陈旧锁恢复
- 修复 cmd 引导脚本管道 stdin 丢失:避免调 exe 前向前 goto,改用 call 子程序
- hooks.json 指向 notify.cmd,超时放宽以容纳首次下载
- .gitignore 放行两个脚本但忽略 notify.exe;.gitattributes 固定脚本换行符
This commit is contained in:
2026-06-22 17:35:57 +08:00
Unverified
parent 79992680c9
commit 0de6fe4064
6 changed files with 133 additions and 10 deletions
+2
View File
@@ -1,3 +1,5 @@
# 批处理脚本必须用 CRLF,否则 cmd 解析会出错
*.bat text eol=crlf
*.cmd text eol=crlf
# shell 脚本必须用 LF
*.sh text eol=lf
+6
View File
@@ -475,3 +475,9 @@ $RECYCLE.BIN/
# Windows shortcuts
*.lnk
# 引导脚本要进库(notify.exe 等仍被上面的 [Bb]in/ 忽略)
!/bin/
/bin/*
!/bin/notify.cmd
!/bin/notify.sh
+34
View File
@@ -44,4 +44,38 @@
<PackageReference Include="Irihi.Ursa.Themes.Semi" Version="2.0.1" />
</ItemGroup>
<!--
AOT 单文件:把 Skia / HarfBuzz / ANGLE 三个原生库静态链接进 exe
CoreUtils.SkiaSharp.Static 含 skia + libHarfBuzzSharp 的 .libCoreUtils.ANGLE.Static 含 ANGLE
两包各自的 .targets 会在 PublishAot 时自动追加 NativeLibrary,这里只补 DirectPInvoke 与系统 lib
版本对应:Avalonia 12 → SkiaSharp 3.119
-->
<ItemGroup>
<PackageReference Include="CoreUtils.SkiaSharp.Static" Version="3.119.0.1" />
<PackageReference Include="CoreUtils.ANGLE.Static" Version="7151.0.1" />
</ItemGroup>
<ItemGroup Condition="'$(PublishAot)' == 'true'">
<DirectPInvoke Include="libSkiaSharp" />
<DirectPInvoke Include="libHarfBuzzSharp" />
<DirectPInvoke Include="av_libglesv2" />
<!-- ANGLE 静态库需要的系统库 -->
<LinkerArg Include="gdi32.lib" />
<LinkerArg Include="user32.lib" />
<LinkerArg Include="d3d9.lib" />
<LinkerArg Include="dxgi.lib" />
<LinkerArg Include="dxguid.lib" />
<LinkerArg Include="synchronization.lib" />
</ItemGroup>
<!-- 静态链接后清理发布目录:动态原生 DLL、链接期 .lib、调试 .pdb 都不需要,只留 notify.exe -->
<Target Name="CleanAotSingleFileOutput" AfterTargets="Publish" Condition="'$(PublishAot)' == 'true'">
<ItemGroup>
<_AotJunk Include="$(PublishDir)*.dll" />
<_AotJunk Include="$(PublishDir)*.lib" />
<_AotJunk Include="$(PublishDir)*.pdb" />
</ItemGroup>
<Delete Files="@(_AotJunk)" />
</Target>
</Project>
+43
View File
@@ -0,0 +1,43 @@
@echo off
rem ============================================================
rem Download URL (notify.exe is fetched from here on first run) -- edit as needed
set "DOWNLOAD_URL=https://github.com/OWNER/REPO/releases/latest/download/notify.exe"
rem ============================================================
setlocal
set "EXE=%~dp0notify.exe"
set "LOCK=%~dp0notify.download.lock"
rem only bootstrap on first run; the common path runs the exe directly (keeps piped stdin intact)
if not exist "%EXE%" call :bootstrap
if exist "%EXE%" "%EXE%" %*
endlocal
exit /b
:bootstrap
set "TMP=%~dp0notify.exe.%RANDOM%.tmp"
rem mkdir is atomic; success = this process downloads, failure = someone else is downloading
mkdir "%LOCK%" 2>nul
if errorlevel 1 goto :waitdl
rem double-check in case it just finished
if exist "%EXE%" ( rmdir "%LOCK%" 2>nul & exit /b )
rem download to temp then atomic rename, so no half-written exe is ever seen
curl -fsSL "%DOWNLOAD_URL%" -o "%TMP%"
if errorlevel 1 ( del "%TMP%" 2>nul & rmdir "%LOCK%" 2>nul & exit /b )
move /y "%TMP%" "%EXE%" >nul
rmdir "%LOCK%" 2>nul
exit /b
:waitdl
rem did not get the lock; wait for the exe to appear (up to ~60s)
set /a _w=0
:waitloop
if exist "%EXE%" exit /b
if %_w% geq 120 (
rem timed out; a killed download may have left a stale lock, clear it for next time
rmdir "%LOCK%" 2>nul
exit /b
)
ping -n 2 127.0.0.1 >nul
set /a _w+=1
goto :waitloop
+38
View File
@@ -0,0 +1,38 @@
#!/bin/sh
# ============================================================
# 下载地址(首次运行从这里拉取 notify.exe)—— 按需修改
DOWNLOAD_URL="https://github.com/OWNER/REPO/releases/latest/download/notify.exe"
# ============================================================
DIR="$(cd "$(dirname "$0")" && pwd)"
EXE="$DIR/notify.exe"
LOCK="$DIR/notify.download.lock"
if [ ! -f "$EXE" ]; then
# mkdir 是原子操作,用作锁:成功=本进程负责下载,失败=已有进程在下
if mkdir "$LOCK" 2>/dev/null; then
# 双重检查,避免刚好别人下完
if [ ! -f "$EXE" ]; then
TMP="$DIR/notify.exe.$$.tmp"
if curl -fsSL "$DOWNLOAD_URL" -o "$TMP"; then
mv -f "$TMP" "$EXE" # 原子改名,避免半截 exe
chmod +x "$EXE" 2>/dev/null
else
rm -f "$TMP"
fi
fi
rmdir "$LOCK" 2>/dev/null
else
# 没抢到锁:等 exe 出现(最多约 60 秒)
i=0
while [ ! -f "$EXE" ] && [ "$i" -lt 120 ]; do
sleep 0.5
i=$((i + 1))
done
# 超时仍没下好:可能上次下载被杀留下陈旧锁,清掉让下次重下
[ ! -f "$EXE" ] && rmdir "$LOCK" 2>/dev/null
fi
fi
# 转发全部参数与 stdin 给真正的 exe
[ -f "$EXE" ] && exec "$EXE" "$@"
+10 -10
View File
@@ -6,8 +6,8 @@
"hooks": [
{
"type": "command",
"command": "${CLAUDE_PLUGIN_ROOT}/bin/notify.exe save",
"timeout": 5
"command": "${CLAUDE_PLUGIN_ROOT}/bin/notify.cmd save",
"timeout": 30
}
]
}
@@ -18,8 +18,8 @@
"hooks": [
{
"type": "command",
"command": "${CLAUDE_PLUGIN_ROOT}/bin/notify.exe input",
"timeout": 10
"command": "${CLAUDE_PLUGIN_ROOT}/bin/notify.cmd input",
"timeout": 30
}
]
}
@@ -30,8 +30,8 @@
"hooks": [
{
"type": "command",
"command": "${CLAUDE_PLUGIN_ROOT}/bin/notify.exe input",
"timeout": 10
"command": "${CLAUDE_PLUGIN_ROOT}/bin/notify.cmd input",
"timeout": 30
}
]
}
@@ -42,8 +42,8 @@
"hooks": [
{
"type": "command",
"command": "${CLAUDE_PLUGIN_ROOT}/bin/notify.exe notify",
"timeout": 10
"command": "${CLAUDE_PLUGIN_ROOT}/bin/notify.cmd notify",
"timeout": 30
}
]
}
@@ -54,8 +54,8 @@
"hooks": [
{
"type": "command",
"command": "${CLAUDE_PLUGIN_ROOT}/bin/notify.exe cleanup",
"timeout": 5
"command": "${CLAUDE_PLUGIN_ROOT}/bin/notify.cmd cleanup",
"timeout": 10
}
]
}