- 
                Notifications
    You must be signed in to change notification settings 
- Fork 474
Fix issue #382 (multiple publish output files with same relative path) #1234
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: master
Are you sure you want to change the base?
Conversation
| Thanks for looking into this. #382 has been a long standing issue that a few people (including me) have had a look at but failed to solve, I'm happy to see the fix turns out to be so simple! I'll find some time to test this out soon, I'd really like to get it into the next update if possible. Would you be interested in developing a followup PR with your suggested improvements? Removing the extra installation step and making it just work would be fantastic. | 
| Glad I could help, this issue had been around for a while, let’s see if the tests confirm the fix. One noteI’m not sure why, but the last time I worked on this bug I was able to pack the NuGets with the original paths. This time, however, I had to adjust them. In all the  For example: I hope this doesn’t break the way you build the packages, if it does, please let me know and I’ll restore the src paths to their previous state. TestingAs before, I tested with a MAUI project using LLamaSharp.Backend.Cpu and LLamaSharp.Backend.Android. This time, I added the packages using the usual NuGet Package Manager GUI, and it was not necessary to use  | 
| Hi @LucaMaccarini , does this PR need any additional changes? I'm not sure what you mean by that last note (have you found out if we need to start with  Edit After replacing all  | 
| Hi @m0nsky, Thanks a lot for testing and for the detailed feedback I actually suspected that the path part should have remained  Example of the command I used locally: I’ll fix the initial path in this PR so that it uses  Could you kindly share how you usually generate the NuGet packages on your side? (what script or command do you run). That would help me align with your workflow and avoid mismatches like this. Regarding the point you mentioned about the  <dependencies>
    <dependency id="LLamaSharp.Backend.Cpu" version="$version$" />
</dependencies>That dependency is the reason why the CPU backend is included when packing the CUDA12 backend. | 
| Yes, absolutely, here are my steps: 
 Basically, I put that  What I meant by the CPU/CUDA12 issue, is that the CUDA12 runtimes seem to be completely missing when building my application. When I extract  I'm using the following command to build/publish my application: | 
| Thanks for pointing me to the script for packing all the backends, I tried it and it works. I missunderstood your previous comment: I thought the CUDA files were there and I was asking why the CPU ones were included as well, but as you said the CUDA files are actually missing. I’ll investigate further... | 
| I am attempting to use a locally built version of the LLamaSharp CPU and Vulkan backends to use this fix, but I'm encountering a build error during the publish step. Here is my process: 
 <?xml version="1.0" encoding="utf-8"?>
<configuration>
    <packageSources>
        <clear />
        <!-- Define local source using a relative path -->
        <add key="local-packages" value="./local-packages" />
        <!-- Add the official nuget.org source -->
        <add key="nuget.org" value="https://api.nuget.org/v3/index.json" protocolVersion="3" />
    </packageSources>
    <packageSourceMapping>
        <packageSource key="nuget.org">
            <package pattern="*" />
        </packageSource>
        <packageSource key="local-packages">
            <package pattern="LLamaSharp.Backend.Cpu" />
            <package pattern="LLamaSharp.Backend.Vulkan" />
            <package pattern="LLamaSharp.Backend.Vulkan.Linux" />
            <package pattern="LLamaSharp.Backend.Vulkan.Windows" />
        </packageSource>
    </packageSourceMapping>
</configuration>In my IDE's NuGet Package Manager, I selected the  When I try to publish my application with the following command, the build fails: 
 I receive the  Here is the error output: Any guidance would be greatly appreciated. Edit: Actually never mind, It was a version conflict. When I set the version to 0.25.0, it used cached backends from the main feed. Changing it to 0.25.1 resolved the issue, and it built successfully. However, it is still packing the Linux and macOS binaries into the  I can also confirm that adding any backend (Vulkan in my case) only adds the CPU backend binaries to the build, not the specific backend itself (e.g., Vulkan, CUDA). | 
just corrected the name of .props file inside nuget package
| Thank you both @m0nsky and @LSXAxeller for your feedback. | 
| 
 Great job, I will test this tonight! 
 I could understand not packing Linux/macOS when building for Windows only (if this is possible), however, I'm not sure if we should flatten the AVX levels, as I build my applications once and deploy them on different servers (AVX2 / AVX512) and rely on the automatic AVX level detection. | 
| Yes, it’s possible to restrict the imported runtimes to only the target platform, just like it’s already done in the  We would just need to adjust the initial  That said, this might be better handled in a separate PR, since the current one should probably stay focused only on fixing the bug without changing the overall import logic. | 
| Just tried the latest version, published to a linux server w/ CUDA, all working as expected. | 
| Let’s wait for @martindevans to run his checks and confirm everything works fine, so this can be included in the next release and close this long standing issue. | 
| While building my .NET application for macOS (target  Since the build is running on GitHub Actions and I don't have a macOS machine to test, I'm not sure if this is a critical issue. Will the missing  
 Edit: Yeah, that would be better as I am actually encountering an issue when publishing a .NET app for macOS ( The build is failing during the  Here's the key error: Full Build Log  | 
| Did you try running the same build on the same GitHub runner with the version currently on main, or were you not even able to compile it and reach the Xcode step because of the issue I’m trying to fix with this PR? I’m asking because the version on main also copies all the .dll and .so files, so if you’re getting that error with my fix, you should also be seeing it on main if it compile... If the version on main fails with the same error, then it’s either due to a misconfiguration/publish issue on your side or it’s a new library issue. Let me know so I can better understand how to proceed | 
| @LucaMaccarini I'm sorry I forgot to share an update—I've tackled this issue and have been busy handling several other issues caused by Apple macOS. I fixed the issue by adding the condition  LLamaSharpBackend.props     <ItemGroup Condition="'$(RuntimeIdentifier)' != ''">
        <Content Include="$(MSBuildThisFileDirectory)..\..\LLamaSharpRuntimes\$(RuntimeIdentifier)\**">
            <CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
            <Visible>false</Visible>
            <Link>runtimes\$(RuntimeIdentifier)\%(RecursiveDir)%(Filename)%(Extension)</Link>
        </Content>
    </ItemGroup>
    <!-- Fallback for when RuntimeIdentifier is not specified -->
    <ItemGroup Condition="'$(RuntimeIdentifier)' == ''">
        <Content Include="$(MSBuildThisFileDirectory)..\..\LLamaSharpRuntimes\**\*.*">
            <CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
            <Visible>false</Visible>
            <Link>runtimes\%(RecursiveDir)%(Filename)%(Extension)</Link>
        </Content>
    </ItemGroup>If you're using the publish command and specifying a RID (like win-x64 or osx-arm64), only the corresponding binaries should be copied to the runtimes folder. However, if your project references a backend like  In my case 
 In that case, you'll need to add the following to the layer's      <PropertyGroup>
        <DetectedArch Condition="'$([System.Runtime.InteropServices.RuntimeInformation]::OSArchitecture)' == 'X64'">x64</DetectedArch>
        <DetectedArch Condition="'$([System.Runtime.InteropServices.RuntimeInformation]::OSArchitecture)' == 'Arm64'">arm64</DetectedArch>
    </PropertyGroup>
    <PropertyGroup>
        <RuntimeIdentifier Condition="$([MSBuild]::IsOSPlatform('Windows'))">win-x64</RuntimeIdentifier>
        <RuntimeIdentifier Condition="$([MSBuild]::IsOSPlatform('Linux'))">linux-x64</RuntimeIdentifier>
        <RuntimeIdentifier Condition="$([MSBuild]::IsOSPlatform('OSX')) AND '$(DetectedArch)' == 'arm64'">osx-arm64</RuntimeIdentifier>
        <RuntimeIdentifier Condition="$([MSBuild]::IsOSPlatform('OSX')) AND '$(DetectedArch)' == 'x64'">osx-x64</RuntimeIdentifier>
    </PropertyGroup>If you'd like to test it quickly, I've uploaded a prebuilt NuGet package here to work with my GitHub build workflow | 
| Hi @LSXPrime, sorry for the late reply! I’ve been quite busy these days, but I’ve been keeping an eye on this PR in my spare time. Thanks for your suggestion: I’ve taken it into account and included it. As you’ll notice, I modified the condition to a slightly more elaborate one that checks for the existence of the runtimes. This is because, as long as you publish using a command like the one you suggested, there are no issues. However, for example, when building an app via Visual Studio and clicking “Run,” Visual Studio on Windows uses a platform identifier like  In my solution, if the folder doesn’t exist, the condition fails and all runtimes are imported, allowing the build to work even when using Visual Studio. We need to decide how to proceed: 
 In my opinion, if the current solution works on Visual Studio for Mac—whether the platform identifier is correct and only the necessary DLLs are imported, or the platform identifier doesn’t match and all DLLs are imported but the build still succeeds—there’s no need to add platform identifier normalization. Adding platform identifier normalization would make the folder names used for the runtime assemblies more restrictive. Changing a folder name would then require updating the normalization system. This is unnecessary because the issue of selectively importing only platform-specific runtimes only occurs during debug, and it’s not a real problem as long as the app builds and runs correctly. When using the dotnet publish command with a clean, explicit platform identifier, the solution already works correctly and only the relevant runtimes are included. We would need to find someone with a Mac and visual studio to test this. | 
Fixes #382: On Windows, runtime files from backend packages are currently flattened during copy. This means that all files from subfolders end up in the same output directory, causing conflicts when multiple files share the same name.
Proposed fix
Correct the file
LLamaSharpBackend.propsas shown in the changes included in this PR.When building for Windows, and after installing the Windows-specific backend packages, make sure to add
ExcludeAssets="Native"to thePackageReferencein the.csproj, as shown below:Although
ExcludeAssets="Native"is strictly necessary only for Windows backends when building on Windows, it shouldn’t cause issues for other backends. To be safe, it can be applied to all imported backend packages. I have tested this with bothLLamaSharp.Backend.CpuandLLamaSharp.Backend.Androidinstalled in the same MAUI project. This attribute tells MSBuild not to copy the runtime files from the NuGet package to the output directory, because copying is already handled correctly in theLLamaSharpBackend.propsfile updated in step 1.How to Apply this fix before it is merged
If you urgently need to fix this issue, you can temporarily apply this change by copying the corrected
LLamaSharpBackend.propscontent (from step 1) into each backend package directory, e.g.:and then import the package as indicated in step 2.
Note: this modification will be lost if the package is updated.
Future improvements:
Avoid using the default
Runtimesfolder, since MSBuild always flattens its contents building for windows causing conflicts. Using a custom folder for runtime files would eliminate the need to setExcludeAssets="Native"in the package reference, and then change theLLamaSharpBackend.propsfile to correctly fetch runtime files from the custom folder.Tests
I tested this fix on a limited scenario with a MAUI project having both
LLamaSharp.Backend.CpuandLLamaSharp.Backend.Androidinstalled. The goal was to ensure the app works on both types of devices. Given the large heterogeneity of the backends, I hope this fix will work for other backends as well.