-
Notifications
You must be signed in to change notification settings - Fork 1.1k
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
Missing Linux TargetFrameworkMoniker
#27061
Comments
Here's the reasoning behind that from the original design:
@terrajobst @mhutch Should we reconsider having a |
@terrajobst @mhutch could you please share your thoughts on this one? Having this functionality would help make my code a lot cleaner. |
I think what's actually missing here is the ability to compile for multiple RIDs. TFMs are about the APIs that are available for your code to compile against, RIDs are about what platform/architecture your code runs on. If your implementation is specific to the platform that you run on, then should be a RID-specific implementation. However, we currently have no tooling for compiling a project for multiple RIDs, and packages that contain multiple RID-specific implementations currently must be constructed from the output of multiple projects. The lack of multi-RID support has les to folks using TFMs for this purpose instead. We have two mechanisms for platform-specific APIs. We can put them in a platform-specific TFM, which is a good solution when exposing a large feature "area" such as an app model that is tightly bound to a single platform. Alternatively, we can make the API available on all platforms but only provide implementations on a subset of platforms. In this case, we use annotations and analyzers to indicate to the developer when they may need runtime checks in their code before calling the API. This is a lighter weight solution as it does not force consuming code to have platform-specific builds, and is great for smaller scale platform-specific feature such as a individual method to set a platform-specific option on a cross-platform feature. There is of course a continuum between these extremes and there's not necessarily a "right" answer to which solution should be used. However, we do not yet have any Linux-specific APIs where a TFM would clearly be the better of the two options, which is why we don't have the TFM (yet). |
@mhutch What would help me a lot is guidance at https://learn.microsoft.com/en-us/dotnet/core/tutorials/libraries#how-to-multitarget or https://learn.microsoft.com/en-us/dotnet/standard/library-guidance/cross-platform-targeting?source=recommendations where it is explained how to create multitargeted code that also has a condition for Linux. If tooling support for that is missing then please consider this issue as a request for that. My idea was a TFM since it fits the idea of "platforms" and automatically handles dependencies but I see now that the concept of RIDs is more correct. The current alternative of requiring separate Linux and non-Linux NuGet packages is a pain as there is no tooling/compile time-supported metadata I can add to my library. |
And I found Better support from Microsoft for multi-targeting with Linux, built-in into the dotnet sdk, would be great! |
This feature definitely needs more attention, as it is making it unreasonable difficult to work on multi-target libraries and applications. Note, this issue and my comment is not about having platform specific API build-in, but about creating cross-platform projects that achieve these APIs by themselves or referencing platform specific packages. While in theory I would agree with @mhutch and the original design idea, it doesn't seem to reflect how Target Frameworks are used in reality. For instance, MAUI supports multiple target frameworks - Windows, Mac Catalyst, Android, iOS, and Tizen. It works pretty well until you want to extend the list of available frameworks. In the end, we have a mix of:
In order to support multiple platforms, developers now need to have a mix of TFM with #if/else directives for the well-supported platforms and OperatingSystem.IsLinux/IsBrowser runtime calls for "net6.0" (i.e. "other"). If the .NET ecosystem switches from TFM to RID - that's fine, but it seems to require way more changes including availability of platform-specific APIs. As an example, there is an interesting project that has to build their own workload with custom TFM in order to achieve expected behavior. |
@dsplaisted we now have a target framework for Browser platform, which I am glad to see. It is now possible to at least dedicate a plain "net8.0" target (without any platform moniker) to Linux. While one can argue, that it's hard to define "Linux" as a platform, .NET ecosystem is already built around "Linux" keyword defining a specific sub-class of Operating Systems at least. |
No, as far as I know we still don't have any plans to add a linux platform to the TargetFramework. We are hoping to eventually add support to allow you to multi-target a project over things that are not the target frameworks that we define: NuGet/Home#12124. This would allow you to build a project twice for .NET 6 via multi-targeting over custom TargetFrameworks that you would define. That would probably help with building projects like this, but this wouldn't flow through NuGet packages so it might not help much with the ecosystem side. @terrajobst FYI |
@dsplaisted thank you for sharing this spec. Looks promissing. The latest use case I found where we need "net8.0-linux", is MAUI Essentials project, Which has compile-time implementations for different essential APIs like FilePicker. And atm if somebody decided to add Linux support there, they would have to do runtime checks, combined with compile-time checks of other platforms. From what I understood from the spec you mentioned, it should be possible to create "net9.0-linux" as an alias for "net9.0", but with a correct RID and put compiled lib to the |
Yes, I think this will be possible. Note that we're not initially planning any built-in support for creating NuGet packages in a situation like this. Likely you would have to set some properties in the project file to tell the Also note that we are making some progress but this isn't currently committed for a specific release. It's possible it could end up in .NET 9 but we don't know yet. |
Is your feature request related to a problem? Please describe.
I am trying to create a cross-platform project with platform-specific implementations of several classes.
For most platforms, it would be possible to use the
TargetFrameworkMoniker
to conditionally include files like MAUI does (see e.g. https://github.com/dotnet/maui/blob/main/src/Essentials/src/Essentials.csproj). However, noTargetFrameworkMoniker
is available for Linux.This currently being unavailable also has the consequence that APIs like https://github.com/dotnet/runtime/pull/69980/files need an
[UnsupportedPlatform("windows")]
attribute rather than only making the APIs available on Unix OSes.Describe the solution you'd like
A
net7.0-linux
TargetFrameworkMoniker
alternative considered:
OperatingSystem.IsLinux()
- this relies on runtime analysis and makes methods larger or requires a wrapper that returns a platform-specific implementation at runtime. This also has the limitation that the called / referenced code needs to be referenced even when being compiled for Windows and vice versa - which leads to impossible situations. Especially for technology like bluetooth or other device support code you need to reference platform-specific TFMs which I believe you can't do from a "neutral"net7.0
project.alternative considered: using inverse #if to assume that
Pseudocode:
#if !(WINDOWS || MACOS || IOS || ANDROID)
. This seems brittle.Additional context
These pages have no pointers about workarounds for this case:
The text was updated successfully, but these errors were encountered: