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

Add ability to replace type constraints #750

Merged
merged 4 commits into from
Feb 16, 2024

Conversation

DustinJSilk
Copy link
Contributor

Description

Adds a config option that can replace generic types with a concrete type.

Type of change

  • Bug fix (non-breaking change which fixes an issue)
  • New feature (non-breaking change which adds functionality)
  • Breaking change (fix or feature that would cause existing functionality to not work as expected)
  • This change requires a documentation update

Version of Golang used when building/testing:

  • 1.11
  • 1.12
  • 1.13
  • 1.14
  • 1.15
  • 1.16
  • 1.17
  • 1.18
  • 1.19
  • 1.20
  • 1.21

How Has This Been Tested?

Please describe the tests that you ran to verify your changes. Provide instructions so we can reproduce. Please also list any relevant details for your test configuration

Checklist

  • My code follows the style guidelines of this project
  • I have performed a self-review of my code
  • I have commented my code, particularly in hard-to-understand areas
  • I have made corresponding changes to the documentation
  • My changes generate no new warnings
  • I have added tests that prove my fix is effective or that my feature works
  • New and existing unit tests pass locally with my changes

@LandonTClipp
Copy link
Collaborator

Excellent! I'll give it a proper review in a few days. Thanks for the contribution.

Copy link

codecov bot commented Jan 24, 2024

Codecov Report

Attention: 82 lines in your changes are missing coverage. Please review.

Comparison is base (3b25f39) 42.45067% compared to head (601d297) 42.65889%.
Report is 4 commits behind head on master.

Files Patch % Lines
...m/vektra/mockery/v2/pkg/fixtures/ReplaceGeneric.go 53.01205% 30 Missing and 9 partials ⚠️
pkg/generator.go 43.47826% 20 Missing and 6 partials ⚠️
...ktra/mockery/v2/pkg/fixtures/ReplaceGenericSelf.go 60.60606% 10 Missing and 3 partials ⚠️
...hub.com/vektra/mockery/v2/pkg/fixtures/Variadic.go 0.00000% 2 Missing ⚠️
...ktra/mockery/v2/pkg/fixtures/VariadicReturnFunc.go 0.00000% 1 Missing and 1 partial ⚠️
Additional details and impacted files
@@                 Coverage Diff                 @@
##              master        #750         +/-   ##
===================================================
+ Coverage   42.45067%   42.65889%   +0.20821%     
===================================================
  Files             61          63          +2     
  Lines           4815        4972        +157     
===================================================
+ Hits            2044        2121         +77     
- Misses          2591        2653         +62     
- Partials         180         198         +18     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

pkg/generator.go Outdated Show resolved Hide resolved
@DustinJSilk
Copy link
Contributor Author

Hey @LandonTClipp
These are the 2 new formats I'd propose for replacing generic types using the replace-type option, please let me know what you think of the 2 examples below:

Replace a type parameter:
module.com/foo.Foo[T]=module.com/baz.Baz

type Foo[T any] struct {}
func (*Foo) Get() T {}
type Foo[T baz.Baz] struct {}
func (*Foo) Get() T {}

Remove a type parameter and set each instance of it:
module.com/foo.Foo[-T]=module.com/baz.Baz

type Foo[T any] struct {}
func (*Foo) Get() T {}
type Foo struct {}
func (*Foo) Get() baz.Baz {}

@DustinJSilk
Copy link
Contributor Author

Hi @LandonTClipp

I've updated the API to the suggestion above which I'm using successfully for my usecase and hopefully it provides more flexibility for others.

I'm happy to change the API if you have any ideas.

@LandonTClipp
Copy link
Collaborator

Hi @DustinJSilk, apologies for the lack of responses the last few weeks. I have been on the hunt for a new job and I haven't been able to devote much time or energy to this project, but luckily my search is coming to a close (hopefully, knock on wood).

As far as your proposals, I think I'm on board with it. The only thing I'm unhappy about (which has nothing to do with your proposal or implementation) is that I don't enjoy these complex, custom parameter strings. I think in hindsight, I would have preferred replace-types to simply be a list of maps with individual parameters, instead of having to parse this string here. That ship has already sailed... it's something I'll probably deprecate once a better implementation is available.

Copy link
Collaborator

@LandonTClipp LandonTClipp left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks great, I think I'm okay with the new format here. I have a few comments, if we can get those addressed then I will merge it!

docs/features.md Outdated Show resolved Hide resolved
pkg/generator.go Outdated
// Match type parameter substitution
match := regexp.MustCompile(`\[(.*?)\]$`).FindStringSubmatch(t)
if len(match) >= 2 {
if match[1][:1] == "-" {
Copy link
Collaborator

@LandonTClipp LandonTClipp Feb 8, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was initially confused why you did that instead of:

match[1][0]

but realized the slicing operator will return a string instead of a single byte. To make this cleaner, I'd instead do this:

if strings.HasPrefix(match[1], "-")

It's a bit more obvious what's happening.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let me know if it still makes sense for you, i was able to put it in one line with strings.CutPrefix

Comment on lines 763 to 788
func (s *GeneratorSuite) TestReplaceTypeGeneric() {
cfg := GeneratorConfig{InPackage: false, ReplaceType: []string{
"github.com/vektra/mockery/v2/pkg/fixtures.ReplaceGeneric[-TImport]=github.com/vektra/mockery/v2/pkg/fixtures/redefined_type_b.B",
"github.com/vektra/mockery/v2/pkg/fixtures.ReplaceGeneric[TConstraint]=github.com/vektra/mockery/v2/pkg/fixtures/constraints.Integer",
"github.com/vektra/mockery/v2/pkg/fixtures.ReplaceGenericSelf[-T]=github.com/vektra/mockery/v2/pkg/fixtures.*ReplaceGenericSelf",
}}

s.checkGenerationRegexWithConfig("generic.go", "ReplaceGeneric", cfg, []regexpExpected{
// type ReplaceGeneric[TConstraint constraints.Integer, TKeep interface{}] struct
{true, regexp.MustCompile(`type ReplaceGeneric\[TConstraint constraints.Integer\, TKeep interface\{\}] struct`)},
// func (_m *ReplaceGeneric[TConstraint, TKeep]) A(t1 test.B) TKeep
{true, regexp.MustCompile(`func \(_m \*ReplaceGeneric\[TConstraint, TKeep\]\) A\(t1 test\.B\) TKeep`)},
// func (_m *ReplaceGeneric[TConstraint, TKeep]) B() test.B
{true, regexp.MustCompile(`func \(_m \*ReplaceGeneric\[TConstraint, TKeep\]\) B\(\) test\.B`)},
// func (_m *ReplaceGeneric[TConstraint, TKeep]) C() TConstraint
{true, regexp.MustCompile(`func \(_m \*ReplaceGeneric\[TConstraint, TKeep\]\) C\(\) TConstraint`)},
})

s.checkGenerationRegexWithConfig("generic.go", "ReplaceGenericSelf", cfg, []regexpExpected{
// type ReplaceGenericSelf struct
{true, regexp.MustCompile(`type ReplaceGenericSelf struct`)},
// func (_m *ReplaceGenericSelf) A() *ReplaceGenericSelf
{true, regexp.MustCompile(`func \(_m \*ReplaceGenericSelf\) A\(\) \*ReplaceGenericSelf`)},
})
}

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I do not like these kinds of tests and am asking people not to do them. Any changes to the formatting can/will cause these kinds of tests to fail. Instead I prefer if we just create a new test file, probably in pkg/fixtures/test that checks these various scenarios through a functional test, rather than string comparison. You can just rely on the compiler to fail if the param types are mismatched, or maybe use type assertion through an empty interface in the test (actually, type assertion might be better in this case so that we don't cause all the tests to fail if the mock implementation is wrong somehow).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good idea! I've added some better test cases and removed these. I didn't quite figure out exactly how we could test just with type assertions, at the moment the compiler would complain if the types no longer match

Copy link
Collaborator

@LandonTClipp LandonTClipp left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Excellent, this looks good to me. Thanks again!

@LandonTClipp
Copy link
Collaborator

The tests are failing due to an unrelated bug that I will fix in a separate PR.

@LandonTClipp LandonTClipp changed the title Add replace-generic config Add ability to replace type constraints Feb 15, 2024
@LandonTClipp LandonTClipp merged commit 8b86cf2 into vektra:master Feb 16, 2024
6 checks passed
@LandonTClipp
Copy link
Collaborator

Good work @DustinJSilk, feature has been released.

@DustinJSilk
Copy link
Contributor Author

Amazing thanks @LandonTClipp

@LandonTClipp
Copy link
Collaborator

A note about v3: I'm having to drop this feature because the implementation in this PR uses the Generator struct which is being cut entirely in favor of the TemplateGenerator. If this functionality is needed in v3, it should be easy to port but it will have to be redone in the new templating scheme.

LandonTClipp added a commit that referenced this pull request Jan 1, 2025
LandonTClipp added a commit that referenced this pull request Jan 20, 2025
* Add back `replace-types`

Port the `replace-type` parameter into mockery v3. This changes the config schema used in the v2 replace-type parameter to be more grokable. This is also a practical matter: the somewhat complex parsing mechanism in v2 is replaced by simple struct attributes.

The implementation here is quite different from v2. The templating system in v3 requires type information for all types, and thus we cannot do simple string replacements. We have to call `packages.Load` on the referenced type so we can get real type information. This means that `replace-type` will incur additional latency. This latency penalty, however, grants us additional correctness of implementation: the mock templates will have full type information, and if something about the `replace-type` parameter is wrong (either package path or type name don't exist), mockery will explicitly log this as an error.

This does _not_ implement replacements of type constraints as done in v2: #750. This work still needs to be done.

Per #864

* Additional documentation

* Add docs and hint in log message on how to enable force-file-write.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Option to replace a generic type parameter with a specific type
2 participants