Skip to content
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 attributes information on setter of a property (in FSharpMemberOrFunctionOrValue type) #18157

Open
MangelMaxime opened this issue Dec 17, 2024 · 10 comments
Labels
Bug Impact-Low (Internal MS Team use only) Describes an issue with limited impact on existing code.
Milestone

Comments

@MangelMaxime
Copy link

Hello,

I am not sure how to report this bug because this is related to the internals of F# Compiler itself, and happening inside of Fable transformation.

In Fable, we want to erase types and properties decorated with [<Erase>].

For example, this code should emit nothing.

[<Erase>]
type internal LanguageInjectionAttribute() =
    inherit Attribute()
    
    [<Erase>]
    member val Prefix = "" with get, set

However, currently it does not seems possible to do it for the setter because it is missing the [<Erase>] attribute information:

Image

Please tell me, if I can do anything to reword my issue or help you identify the cause of this problem.

@vzarytovskii
Copy link
Member

Yeah, I don't think attributes are universally propagated to property methods. One of the reasons is that targets on attributes can be incorrect for methods. Which is fine from the runtime perspective, but semantically incorrect.

@T-Gro
Copy link
Member

T-Gro commented Jan 2, 2025

You should be able to find the attribute on the symbol for the whole Property (and not just getter/setter).
Is it there?

@MangelMaxime
Copy link
Author

@T-Gro What do you mean by Symbol in this context?

I tried to look but FSharpMemberOrFunctionOrValue does not seems to have a Symbol property for access.

However, I found that there is a DeclaringEntity which seems to have the Attributes information, would this be a good place to look for Attributes if we detect that the FSharpMemberOrFunctionOrValue is a setter?

Image

@T-Gro
Copy link
Member

T-Gro commented Jan 3, 2025

.DeclaringEntity.Attributes gets attributes of the type, not of the property - I assume this is not what you want.

.DeclaringEntity.MembersFunctionsAndValues should yield a collection of members, one of which will be a P with PropInfo (i.e. the full property, not just the set_... method. Can you locate the attribute like that?

(locate the one you want via .LogicalName and .IsProperty)

@MangelMaxime
Copy link
Author

.DeclaringEntity.Attributes gets attributes of the type, not of the property - I assume this is not what you want.

This explains why I had a strange result when removing [<Erase>] from the top level entity.

.DeclaringEntity.MembersFunctionsAndValues should yield a collection of members, one of which will be a P with PropInfo (i.e. the full property, not just the set_... method. Can you locate the attribute like that?

(locate the one you want via .LogicalName and .IsProperty)

This seems possible indeed, it just make the code more verbose compared to the other case where Attributes is propagated.

let isErasedPropertySetter (memb : FSharpMemberOrFunctionOrValue) =
    if memb.IsPropertySetterMethod then
        match memb.DeclaringEntity with
        | Some ent ->
            // Remove the "set_" prefix
            let logicalName = memb.LogicalName.Substring(4)

            ent.MembersFunctionsAndValues
            |> Seq.tryFind(fun (m : FSharpMemberOrFunctionOrValue) ->
                m.LogicalName = logicalName && hasAttrib Atts.erase m.Attributes
            )
            |> Option.isSome
        | _ -> false
    else
        false

Depending on if the attributes should be propagated to the setter, this can be a workaround or the final solution.

Thank you for the guidance.

@T-Gro
Copy link
Member

T-Gro commented Jan 3, 2025

I think automatic propagation might be a breaking change, since AttributeTargets.Property and AttributeTargets.Method are two distinct options.

https://learn.microsoft.com/en-us/dotnet/api/system.attributetargets?view=net-8.0

(i.e. it might lead to an error saying that attribute intended for properties only appeared on a method)

@MangelMaxime
Copy link
Author

I think automatic propagation might be a breaking change, since AttributeTargets.Property and AttributeTargets.Method are two distinct options.

Is it because in F# it could be a Method, and then map it to both a Method / Property in the "generated C#" code?

My first thinking when learning about AttributeTargets which was from the first comment on this issue, was that perhaps "automatic propagation" could check if the AttributeTargets is correct or not. If yes, they the attributes is forwarded.

This is probably a naive idea and I don't know if this would be worth the effort or even fix the issue I originally faced.

@edgarfgp
Copy link
Contributor

edgarfgp commented Jan 6, 2025

I think automatic propagation might be a breaking change, since AttributeTargets.Property and AttributeTargets.Method are two distinct options.

https://learn.microsoft.com/en-us/dotnet/api/system.attributetargets?view=net-8.0

(i.e. it might lead to an error saying that attribute intended for properties only appeared on a method)

I think the attribute not appearing on members here is a bug(regardless the targets it should be using). It is will be a breaking change in the sense that user defined attributes will have to adjust the targets but like we have done in F#9 for AttributeTarget changes is a small price to pay to make sure we address the still long standing issues with attributes in F#

@edgarfgp
Copy link
Contributor

edgarfgp commented Jan 6, 2025

I’d be happy to investigate and potentially fix this if we want to address this

@T-Gro
Copy link
Member

T-Gro commented Jan 6, 2025

They really should not be mixed up.
Properties and methods are different elements in the IL, and they can each carry different attributes.

See e.g. here:
https://sharplab.io/#v2:EYLgtghglgdgNAFxFANgHwAICYCMBYAKEIG0BBBBAJymAFcEBTAVQGcIBzBgCnKpvoYAVCJU4IWAOgAKlAPYAHBpQQBPAJQBdQhgDMAAmx6Asit6Uz/RnpB6LdRoQDehAL6F3BXQax6AwnucCPWDiEzM7AS0g4K8MHAAGAH5jFRkFJVUAsQBuFgYEbLcCFyA

The usability aspect which @MangelMaxime mentions is IMO the right target here - the exposed Symbols API should make it easier to traverse the data from method to property or vice versa (without having to do the string slicing which is visible in the snippet)

@abonie abonie added Impact-Low (Internal MS Team use only) Describes an issue with limited impact on existing code. and removed Needs-Triage labels Jan 6, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Bug Impact-Low (Internal MS Team use only) Describes an issue with limited impact on existing code.
Projects
Status: New
Development

No branches or pull requests

5 participants