cmd/go: add package forwarding #60696
Replies: 7 comments 22 replies
-
| 
 Is it like aliases, where the old and new are interchangeable, and can be used together in the same package? 
 Can you remind us what +incompatible versions are? 
 You mean forwarding declaration, singular, correct? 
 Given 
 why would forward.go already exist? Wouldn't it be the only file left in the old package? I'm not clear on why a subcommand is useful, if all we need is a single file with a special package comment. 
 I don't understand what this is saying, or asking. | 
Beta Was this translation helpful? Give feedback.
-
| Can you provide real-world examples where this would have been/would be useful? | 
Beta Was this translation helpful? Give feedback.
        
          
            
              This comment has been minimized.
            
          
            
        
      
    
            
              This comment has been minimized.
            
          
            
        -
| 这个提案不错,但是会存在一些潜在问题,假设这个模块需要最低的go版本是1.21,那么如果1.21以下的用户对这个项目进行编译,并不会有异常提示,只会导致编译结果与预期不一致 比如 old模块是fmt.Println(1) 低于1.21编译出来的将会是 fmt.Println(1)而不是fmt.Println(2) 比如以下这种情况 >>>go version
go version go1.20 windows/amd64go.mod module Test
go 1.22main.go package main
import (
	"Test/TestModule"
	"fmt"
)
func main() {
	Test("Test")
}
func Test(Value interface{}) {
	fmt.Println(Value)
	fmt.Println(testModule.Test())
}TestModule/Test.go //go:forward example.com/to/q
package testModule
func Test() map[int]string {
	return make(map[int]string)
} | 
Beta Was this translation helpful? Give feedback.
-
| This proposal needs to describe the interaction between  | 
Beta Was this translation helpful? Give feedback.
-
| Is the forward used to rewrite the importing code's go.mod? In other words if my go.mod contains  If not then this inline forwarding mechanism is going to create new difficulties tracing dependency incompatibilities. At a minimum the forwarding needs to be exposed by  | 
Beta Was this translation helpful? Give feedback.
-
| Thanks for the discussion, this has been very helpful for gathering feedback. Based on the discussion, we're not entirely sure what the right path forward is here, but we'll close this and think some more. | 
Beta Was this translation helpful? Give feedback.
Uh oh!
There was an error while loading. Please reload this page.
Uh oh!
There was an error while loading. Please reload this page.
-
Overview:
The aim of this discussion is talk about a potential proposal to add a mechanism to add forwarding declarations from one package to another package. When the go build system encounters a forwarding declaration on a package, it replaces all imports of that package in the build with imports of the forwarded package. A package that has been moved or forked to a different module can forward itself to the destination or fork, +incompatible versions of modules can forward their packages to versions with major version suffixes, and GitHub repository modules that have changed their name can forward themselves to their new name.
Proposal:
We add a forwarding directive that would be added to a package statement. As an example, consider the package
example.com/from/pwith the following package declaration:The
//go:forwarddeclaration tells the Go build system that all imports of the import pathexample.com/from/pshould be treated as imports of the import pathexample.com/to/q. (This proposal targets the Go build system; if it is accepted, other build systems, such as Bazel would need to implement special support for forwarding to allow importing packages using forwarding directives.) Any uses of the importexample.com/from/pthat are found would be internally rewritten as imports of example.com/to/q. [We internally would create aloadPkgfor example.com/from/p that records that it forwards to example.com/to/q] Packages are allowed to forward to internal packages if they would be able to import those packages.When the build system encounters a
//go:forwarddeclaration, it also enforces that the package (after build constraint resolution) does not contain any code.The
go listoutput for a forwarded package would contain a newForwardfield for forwarded packages that points to the package it's forwarded to. Tools that make use ofgo listoutput such as those based ongolang.org/x/tools/go/packageswould need to be updated to understand the forwarding directives and forward to the target package.We will also add a subcommand to the go command to aid in adding forwarding declarations:
go forward <directory path> <destination path>[/...]will either create new packages with forwarding declarations, or add forwarding declarations to already existing packages, pointing to the destination path. The/…pattern, can only appear at the end of the destination path, will recursively forward subdirectories directory path to the corresponding subdirectories of the destination package's directory. It will not cross module boundaries.If the from directories already exist,
go forwardwould place the forwarding declarations in a new go fileforward.goand delete all source files (Go and non-Go) in the directory that are recognized by the Go build system.Open question: we don't allow declarations in the same package, should we allow declarations in tests? probably not
Use cases:
Standard use case:
Module example.com/old stops development and development moves to example.com/new. Forwarding declarations are added to all the packages of example.com/old on the last published version of that package using
. When a user tries to update example.com/old to the latest version, the forwarding declarations are pulled in from that last published version and all references to the old path are rewritten to point to the new path, "upgrading" the module to the new module. A requirement to the new module is pulled in.
The user can also add requirements on all 'old' module versions forwarding to the new module in the module being forwarded to. We could do this via a test that the old import works. Adding the requirements ensures that if there are imports of the old module in any dependencies of users of the new module, the forwarding declarations are seen and forward those to the new module. That will mean the new module packages are used uniformly for both import paths.
Old module is no longer maintained, support a migration:
In this use case, module example.com/old stops development, but they haven't picked a successor module. A fork of the module example.com/new wants to be able to support a migration of the usages of the module to the new path using a mix of the old and new paths in the interim. They could release an 0.1.0 version of example.com/new that forwards all corresponding packages to the old module. Then users of the old module can add a requirement on v0.1.0 of the new module, and slowly migrate their imports to the old module to point to paths in the new module. Because of the forwarding declarations in v0.1.0, both imports of the old and new module would be uniformly resolved to the old module. This would allow imports to be migrated piecemeal until all of them become imports of the new module. Once that is the case, the requirements can be upgraded to a v1.0.0 version of the new module that no longer has forwarding declaration, and perhaps introduces new functionality/fixes/etc. The author of the module could additionally include the new code in v0.1.0, but under a build constraint, for example under the tag examplecomnew. Then users could run go test -tags=examplecomnew to verify the individual migrations before they upgrade to 1.0.0.
Github Repository renamed with the same major version.
Module github.com/olduser/oldmodule is renamed to github.com/newuser/newmodule. To facilitate this, a "final" version of the module is released that declares itself as github.com/olduser/oldmodule in the go.mod module declaration. This version will forward all packages to their corresponding package in github.com/newuser/newmodule. And then a following version is released (without the forwarding declarations, but instead the normal package code) that declares itself as github.com/newuser/newmodule in its go.mod file. An upgrade of github.com/olduser/oldmodule would upgrade the module to the latest version without errors, which would be the version with the forwarding declaration (That's because upgrade upgrades to the latest version without errors, which doesn't include any of the versions with the new module path because they are module name mismatches with the module requested. It is important to note that this behavior can be O(N) with the number of releases of the new module at the same major version because each version is checked for errors in order going down until one without errors is found). That would in turn add forwarding declarations to the new module which would introduce a requirement on the new module, so that we'd upgrade to the latest version of the new module.
Allowing replacements to point to other active modules (Issue #26904)
In this use case, module example.com/old is forked to example.com/new, but the owners of example.com/old don't want to add forwarding declarations to example.com/new. The user of example.com/new wants all uses of example.com/old in their transitive dependencies to point to example.com/new. They can't just do a replace to example.com/new because a replacement can't point to another module that's also in use in the module's transitive dependencies. But what they can do is replace example.com/old with a module that has forwarding declarations corresponding to all of example.com/old's packages. Then all uses of example.com/old will be replaced and then forwarded to example.com/new.
Beta Was this translation helpful? Give feedback.
All reactions