[ 
https://issues.apache.org/jira/browse/GROOVY-11985?page=com.atlassian.jira.plugin.system.issuetabpanels:comment-tabpanel&focusedCommentId=18080015#comment-18080015
 ] 

ASF GitHub Bot commented on GROOVY-11985:
-----------------------------------------

codecov-commenter commented on PR #2529:
URL: https://github.com/apache/groovy/pull/2529#issuecomment-4420729381

   ## 
[Codecov](https://app.codecov.io/gh/apache/groovy/pull/2529?dropdown=coverage&src=pr&el=h1&utm_medium=referral&utm_source=github&utm_content=comment&utm_campaign=pr+comments&utm_term=apache)
 Report
   :white_check_mark: All modified and coverable lines are covered by tests.
   :white_check_mark: Project coverage is 68.1662%. Comparing base 
([`09e9122`](https://app.codecov.io/gh/apache/groovy/commit/09e912275e770773ca0eaafbcf170176c05906e9?dropdown=coverage&el=desc&utm_medium=referral&utm_source=github&utm_content=comment&utm_campaign=pr+comments&utm_term=apache))
 to head 
([`9115a3d`](https://app.codecov.io/gh/apache/groovy/commit/9115a3dedd8c665c7821dffcf699437e98fe5290?dropdown=coverage&el=desc&utm_medium=referral&utm_source=github&utm_content=comment&utm_campaign=pr+comments&utm_term=apache)).
   :warning: Report is 4 commits behind head on master.
   
   <details><summary>Additional details and impacted files</summary>
   
   
   
   [![Impacted file tree 
graph](https://app.codecov.io/gh/apache/groovy/pull/2529/graphs/tree.svg?width=650&height=150&src=pr&token=1r45138NfQ&utm_medium=referral&utm_source=github&utm_content=comment&utm_campaign=pr+comments&utm_term=apache)](https://app.codecov.io/gh/apache/groovy/pull/2529?src=pr&el=tree&utm_medium=referral&utm_source=github&utm_content=comment&utm_campaign=pr+comments&utm_term=apache)
   
   ```diff
   @@                Coverage Diff                 @@
   ##               master      #2529        +/-   ##
   ==================================================
   - Coverage     68.1761%   68.1662%   -0.0099%     
   - Complexity      32715      32720         +5     
   ==================================================
     Files            1498       1499         +1     
     Lines          125079     125109        +30     
     Branches        22542      22551         +9     
   ==================================================
   + Hits            85274      85282         +8     
   - Misses          32317      32339        +22     
     Partials         7488       7488                
   ```
   
   | [Files with missing 
lines](https://app.codecov.io/gh/apache/groovy/pull/2529?dropdown=coverage&src=pr&el=tree&utm_medium=referral&utm_source=github&utm_content=comment&utm_campaign=pr+comments&utm_term=apache)
 | Coverage Δ | |
   |---|---|---|
   | 
[...oovy/transform/trait/TraitReceiverTransformer.java](https://app.codecov.io/gh/apache/groovy/pull/2529?src=pr&el=tree&filepath=src%2Fmain%2Fjava%2Forg%2Fcodehaus%2Fgroovy%2Ftransform%2Ftrait%2FTraitReceiverTransformer.java&utm_medium=referral&utm_source=github&utm_content=comment&utm_campaign=pr+comments&utm_term=apache#diff-c3JjL21haW4vamF2YS9vcmcvY29kZWhhdXMvZ3Jvb3Z5L3RyYW5zZm9ybS90cmFpdC9UcmFpdFJlY2VpdmVyVHJhbnNmb3JtZXIuamF2YQ==)
 | `89.0173% <100.0000%> (+0.3946%)` | :arrow_up: |
   
   ... and [9 files with indirect coverage 
changes](https://app.codecov.io/gh/apache/groovy/pull/2529/indirect-changes?src=pr&el=tree-more&utm_medium=referral&utm_source=github&utm_content=comment&utm_campaign=pr+comments&utm_term=apache)
   </details>
   <details><summary> :rocket: New features to boost your workflow: </summary>
   
   - :snowflake: [Test 
Analytics](https://docs.codecov.com/docs/test-analytics): Detect flaky tests, 
report on failures, and find test suite problems.
   - :package: [JS Bundle 
Analysis](https://docs.codecov.com/docs/javascript-bundle-analysis): Save 
yourself from yourself by tracking and limiting bundle sizes in JS merges.
   </details>




> Static method override on trait implementer ignored when called via this in 
> trait body
> --------------------------------------------------------------------------------------
>
>                 Key: GROOVY-11985
>                 URL: https://issues.apache.org/jira/browse/GROOVY-11985
>             Project: Groovy
>          Issue Type: Bug
>            Reporter: Paul King
>            Priority: Major
>
> Analysis by AI of the description in the Grails canary build comments yielded 
> the following:
> h2. Summary
> In Groovy 4.x, calling {{this.someStaticMethod()}} from inside a trait's 
> static method body dispatched dynamically and honoured a static method 
> override declared on the implementing class. In Groovy 5.x and 
> 6.0.0-SNAPSHOT, the same call is rewritten by {{TraitReceiverTransformer}} to 
> dispatch through the trait helper, which can never see the implementing-class 
> override. The override is silently lost — no exception, no compile warning, 
> the trait's default value just always wins.
> This was discovered as part of the Grails 8 / Groovy 5 migration 
> (apache/grails-core PRs 
> [#15557|https://github.com/apache/grails-core/pull/15557] and 
> [#15558|https://github.com/apache/grails-core/pull/15558]) and motivated a 
> reflection-based workaround in 
> {{{}Validateable.resolveDefaultNullable(Class){}}}.
> h2. Reproducer
> Standalone repro: 
> [https://github.com/jamesfredley/groovy-trait-static-method-override-bug]
> {code:java|title=Validateable.groovy}
> trait Validateable {
>     static boolean defaultNullable() {
>         false
>     }
>     static boolean defaultNullableSeenByTrait() {
>         // expected to dispatch to the implementing class override
>         this.defaultNullable()
>     }
> }
> {code}
> {code:java|title=MyNullableValidateable.groovy}
> class MyNullableValidateable implements Validateable {
>     static boolean defaultNullable() {
>         true
>     }
> }
> {code}
> {code:java|title=Driver}
> assert MyNullableValidateable.defaultNullable()           // direct call: 
> true on every version
> assert MyNullableValidateable.defaultNullableSeenByTrait() // expected true; 
> gets false on 5.x / 6.0
> {code}
> h2. Observed behaviour
> ||Groovy version||Direct call||{{this.defaultNullable()}} from trait body||
> |4.0.27|true (PASS)|true (PASS)|
> |5.0.5|true (PASS)|*false (FAIL)*|
> |6.0.0-SNAPSHOT|true (PASS)|*false (FAIL)*|
> h2. Root cause
> The bytecode emitted for the trait helper's 
> {{defaultNullableSeenByTrait(Class)}} method changed shape:
> {noformat}
> // Groovy 4.0.27                                 // Groovy 5.0.5 / 
> 6.0.0-SNAPSHOT
> 0: aload_0                                       0: ldc          // 
> Validateable$Trait$Helper.class
> 1: invokedynamic invoke:                         2: aload_0
>    (Ljava/lang/Class;)Ljava/lang/Object;         3: invokedynamic invoke:
>                                                     
> (Ljava/lang/Class;Ljava/lang/Class;)Ljava/lang/Object;
> {noformat}
> In 4.x the indy receiver is {{aload_0}} — the implementing class — so the 
> dynamic dispatch resolves {{defaultNullable()}} against 
> {{MyNullableValidateable}} and finds the override. In 5.x+ the receiver is 
> hard-coded {{Validateable$Trait$Helper.class}} (an {{{}ldc{}}}) and the 
> implementing class is demoted to an argument, so the dispatch resolves 
> {{defaultNullable(Class)}} on the trait helper itself and always lands on the 
> lowered trait default.
> The behaviour change was introduced by commit {{0aa78d0a33}} (GROOVY-8854, 
> Sep 2023):
> {quote}write {{T.m(p)}} as {{this.m($static$self,p)}} not {{$self.m(p)}}
> {quote}
> That commit rewrote {{TraitReceiverTransformer.transformMethodCallOnThis}} so 
> that {{this.someStaticMethod()}} inside a trait body is rewritten as {{(this 
> | T$Trait$Helper).m((Class)$self.getClass(), args)}} — routed through the 
> trait helper's lowered static, with the implementing class passed as the 
> {{$static$self}} argument. The existing trait static method test coverage in 
> {{TraitASTTransformationTest.testTraitStaticMethod}} (including the 
> GROOVY-8854 case at line 2218) doesn't exercise the 
> override-on-implementing-class scenario, so the regression wasn't caught.
> h2. Tradeoff
> Groovy 5's behaviour is arguably closer to Java semantics: static methods are 
> not virtual, and {{this}} inside a Java static method doesn't exist, so 
> "{{{}this.staticMethod(){}}} virtually dispatches to the implementing class's 
> static" was always Groovy-specific magic that relied on MOP. However:
>  * The Groovy 4 behaviour was depended on by real code — Grails 
> {{Validateable}} is the visible canary; the same idiom likely exists 
> elsewhere.
>  * The failure mode is silent — no exception, no compile warning, the trait 
> default just wins every time.
>  * The change surfaced as a side effect of GROOVY-8854 (whose ticket was 
> about something else), not as a deliberate "trait statics are no longer 
> virtual" decision, and it isn't called out in the Groovy 5 release notes.
> h2. Suggested options
>  # Restore Groovy 4 semantics for {{this.staticMethod()}} in trait bodies — 
> emit a dynamic lookup whose receiver is {{$static$self}} (the implementing 
> class) rather than the trait helper.
>  # At minimum, emit a compile-time warning when a trait body calls a 
> same-named static and the dispatch will provably not hit any override on the 
> implementing class.
>  # Document the new contract explicitly in the Groovy 5 release notes and 
> trait docs, so consumers can adapt at the trait level instead of debugging 
> silent no-ops.
> h2. Workaround (already applied in Grails)
> Bypass {{TraitReceiverTransformer}} via plain Java reflection, which it can't 
> see through:
> {code:groovy}
> private static boolean resolveDefaultNullable(Class<?> clazz) {
>     try {
>         return clazz.getMethod('defaultNullable').invoke(null) as boolean
>     } catch (NoSuchMethodException ignored) {
>         return false
>     }
> }
> {code}



--
This message was sent by Atlassian Jira
(v8.20.10#820010)

Reply via email to