-
Notifications
You must be signed in to change notification settings - Fork 69
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
Refine the handling of pointers in TypeAsDeepType
#251
Conversation
a1 := new(int) | ||
b1 := foo(a1) | ||
print(*b1) | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Earlier, total 5 triggers were created across these two functions, now only the required 3 are being created.
Codecov ReportAll modified and coverable lines are covered by tests ✅
Additional details and impacted files@@ Coverage Diff @@
## main #251 +/- ##
==========================================
- Coverage 87.63% 87.60% -0.03%
==========================================
Files 63 63
Lines 7926 7916 -10
==========================================
- Hits 6946 6935 -11
+ Misses 800 799 -1
- Partials 180 182 +2 ☔ View full report in Codecov by Sentry. |
Guards: consumer.Guards, | ||
GuardMatched: true, | ||
} | ||
consumers[i].GuardMatched = true |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This change is not particularly related to the PR, but it is a small enough code cleanup, hence included in the same PR. The change is as following: since we have made our consumers as pointers, we can simply go ahead and modify its GuardMatched
field, instead of having to override the object.
2fea270
to
eb7ecd5
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice find!! I just have a quick question :)
util/util.go
Outdated
// Only consider pointers to deep types (e.g., `var x *[]int`) as deep type, not pointers to basic types (e.g., `var x *int`) | ||
if _, ok := t.Elem().(*types.Basic); !ok { | ||
return t.Elem(), true | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would this actually work for cases where it's a pointer to struct?
Do we actually want to call TypeAsDeepType
recursively here? return TypeAsDeepType(t.Elem())
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah yes, good point. Now added support for pointer to struct. Added more test cases too, to cover this and a named deep type case.
I actually did do it the recursive way first, but quickly realized that if there exists a circular named type dependency then this goes into an infinite loop. But moreover, we were missing deep type cases like pointer of pointer (**x
), causing a change in number of errors as well. The hypothesis of this PR was to only remove the redundant triggers without affecting the number of errors, hence backtracked from the recursive approach.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
exists a circular named type dependency then this goes into an infinite loop
Hmmm, is it something like this?
type A *B
type B *A
It's definitely weird, and the compiler doesn't complain (it complains if you do type A B; type B A
though...).
(it would probably be good to add the pattern to the test suite to avoid accidental regressions)
But moreover, we were missing deep type cases like pointer of pointer (**x), causing a change in number of errors as well.
Are you suggesting pointer of a named type, where the named type is also a pointer (e.g., type A *B; type B *int
)? I think by default double pointers aren't allowed. I think other helper functions probably suffer from the same double-ptr problem, and we definitely don't need to handle them in this PR :) Also, are we missing cases like named types, i.e., type A []int
where A
should be considered deep as well, right?
Another proposal for simpler handling:
Instead of recursions, should we unwrap the ptr first, and then check it's type (like other utility functions in this package), e.g.,
switch UnwrapPtr(t).(type) {
case ....
// no case for `*types.Pointer`
}
This definitely would miss the cases discussed above:
(1) double-ptrs: we're only unwrapping the ptr once, but it avoids infinite loops (and handling double ptrs should be more complex and in separate PRs).
(2) named types: well, if we do UnwrapPtr(t.Underlying())
, we should be handling it fine? we can put those in separate PRs as well (with more test cases perhaps)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We had discussed this point extensively internally. Updating the logic accordingly, incorporating your suggestion and internal discussion.
d22dfba
to
bb822fa
Compare
3285cec
to
0204bd5
Compare
This PR refines the handling of
*types.Pointer
inTypeAsDeepType
to only consider a pointer type deep if its underlying elements are of a deep type. The implication of always considering a pointer deep was that for type likevar x *int; foo(x)
, unnecessary deep triggers were being created. The change in this PR reduces the creation of these redundant triggers.