-
Notifications
You must be signed in to change notification settings - Fork 17.8k
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
proposal: cmd/go: pinning replacements in require #68638
Comments
CC @matloob |
If I understand the proposal correctly, this combined require and replace directive makes upgrades of dependencies likely to cause builds to fail. I see why that's desirable for your project because you want to detect upgrades and update your forked version of the repo whenever that happens. But I think we'd want to be really careful before introducing a feature that does that because it could lead to surprises and user confusion. I think it would also encourage building modules that don't work properly as dependencies. I also think mixing require and replace directives can lead to confusion since require directives apply for work and dependency modules while replace directives only apply to work modules. One thing you could do to add similar functionality to your project without the feature being added to the module system is to add a test that ensures that versioned replacements in your |
Certainly take your point about abuse by dependencies, and thanks for the suggestion to use a test to work around this. I'm not particularly invested in the proposed syntax here, but I don't see our motivation as particularly niche. It seems to me that this fragility applies to most potential uses of It's also at odds with the general Go design principle that code behaviour should be minimally dependent on its context. The version attached to a Another possible design would be simply to make the check you suggest as a test a standard behaviour. It's not clear to me why a |
Actually I misremembered - my case 3 is also poorly supported since the replacement version is required if the path is not local - #28176 discusses this. |
A colleague suggested an alternative to ensure that all versioned replacements in your I agree that replace directives can be fragile, but it's our intention that they're used as infrequently as possible, and only for short time frames. They're primarily there for the first use case you mentioned. The other main use case for replaces is for making changes across multiple modules. But both of those use cases are primarily there for temporary changes. I hope that the second use case is an uncommon use case. It's a really hard place to be to plan to have a low level dependency replaced for the long term and I think it would be painful regardless of the level of support the go command has for it. Our hope is that users cooperate with lower level dependencies when possible. An example of Go's philosophy is in this blog post about the principles of Go's module system: https://research.swtch.com/vgo-principles#cooperation. (The example is different from what you're trying to do but I think the principle still applies). |
Hi, could you let us know if the technique above could work for you? I would like to help improve your experience, though I think it's best to implement the requirements using mechanisms that already exist. |
Timed out in state WaitingForInfo. Closing. (I am just a bot, though. Please speak up if this is a mistake or you have the requested information.) |
Proposal Details
Motivation
In the go.mod for our large codebase we have about a dozen
replace
directives, for essentially three reasons:x/crypto
in order to add support for some nonstandard cipher modes. Those will never be upstreamed because they are technically dubious, but nevertheless certain 3rd party connections require us to use them.A recurring problem we have is that cases 1 and 2 here are both rather fragile. E.g. we might have:
But once we upgrade
package1
, which depends on a newer version ofpackage2
, we might automatically get:Now it looks like we're using
v1.5
, but in fact we're still using ourv1.2
fork, thatpackage1
is apparently incompatible with in some unspecified way, that may or may not result in compile time failure. And since in practice therequire
andreplace
directives may be several pages apart in the file (the formatter explicitly encourages grouping them separately), this is not obvious during code review.The
replace
directive supports a version number on its left-hand side, but that results in equally undesirable behaviour:Now the
replace
is just ignored, so we upgraded tov1.5
and our fix is discarded.Proposal
To add a combined syntax that explictly requires, pins and replaces a given version:
In this form, any attempt to implicitly upgrade package2 should fail and require human intervention, because likely they need to go away and ensure that
fix3
has been merged upstream or rebased as appropriate before proceeding.I would expect this normally to be used only in main modules. To avoid surprises, it should be permitted in dependency modules only if either a) the exact same
require
appears in the main module or b) the package has areplace
overriding it.The text was updated successfully, but these errors were encountered: