forked from XTLS/Xray-core
-
Notifications
You must be signed in to change notification settings - Fork 2
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Fix data leak between mux.cool connections (XTLS#3718)
Fix XTLS#116
- Loading branch information
1 parent
bdca0d1
commit 4a088d1
Showing
3 changed files
with
144 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,124 @@ | ||
package mux_test | ||
|
||
import ( | ||
"context" | ||
"testing" | ||
|
||
"github.com/xtls/xray-core/common" | ||
"github.com/xtls/xray-core/common/buf" | ||
"github.com/xtls/xray-core/common/mux" | ||
"github.com/xtls/xray-core/common/net" | ||
"github.com/xtls/xray-core/common/session" | ||
"github.com/xtls/xray-core/features/routing" | ||
"github.com/xtls/xray-core/transport" | ||
"github.com/xtls/xray-core/transport/pipe" | ||
) | ||
|
||
func newLinkPair() (*transport.Link, *transport.Link) { | ||
opt := pipe.WithoutSizeLimit() | ||
uplinkReader, uplinkWriter := pipe.New(opt) | ||
downlinkReader, downlinkWriter := pipe.New(opt) | ||
|
||
uplink := &transport.Link{ | ||
Reader: uplinkReader, | ||
Writer: downlinkWriter, | ||
} | ||
|
||
downlink := &transport.Link{ | ||
Reader: downlinkReader, | ||
Writer: uplinkWriter, | ||
} | ||
|
||
return uplink, downlink | ||
} | ||
|
||
type TestDispatcher struct { | ||
OnDispatch func(ctx context.Context, dest net.Destination) (*transport.Link, error) | ||
} | ||
|
||
func (d *TestDispatcher) Dispatch(ctx context.Context, dest net.Destination) (*transport.Link, error) { | ||
return d.OnDispatch(ctx, dest) | ||
} | ||
|
||
func (d *TestDispatcher) DispatchLink(ctx context.Context, destination net.Destination, outbound *transport.Link) error { | ||
return nil | ||
} | ||
|
||
func (d *TestDispatcher) Start() error { | ||
return nil | ||
} | ||
|
||
func (d *TestDispatcher) Close() error { | ||
return nil | ||
} | ||
|
||
func (*TestDispatcher) Type() interface{} { | ||
return routing.DispatcherType() | ||
} | ||
|
||
func TestRegressionOutboundLeak(t *testing.T) { | ||
originalOutbounds := []*session.Outbound{{}} | ||
serverCtx := session.ContextWithOutbounds(context.Background(), originalOutbounds) | ||
|
||
websiteUplink, websiteDownlink := newLinkPair() | ||
|
||
dispatcher := TestDispatcher{ | ||
OnDispatch: func(ctx context.Context, dest net.Destination) (*transport.Link, error) { | ||
// emulate what DefaultRouter.Dispatch does, and mutate something on the context | ||
ob := session.OutboundsFromContext(ctx)[0] | ||
ob.Target = dest | ||
return websiteDownlink, nil | ||
}, | ||
} | ||
|
||
muxServerUplink, muxServerDownlink := newLinkPair() | ||
_, err := mux.NewServerWorker(serverCtx, &dispatcher, muxServerUplink) | ||
common.Must(err) | ||
|
||
client, err := mux.NewClientWorker(*muxServerDownlink, mux.ClientStrategy{}) | ||
common.Must(err) | ||
|
||
clientCtx := session.ContextWithOutbounds(context.Background(), []*session.Outbound{{ | ||
Target: net.TCPDestination(net.DomainAddress("www.example.com"), 80), | ||
}}) | ||
|
||
muxClientUplink, muxClientDownlink := newLinkPair() | ||
|
||
ok := client.Dispatch(clientCtx, muxClientUplink) | ||
if !ok { | ||
t.Error("failed to dispatch") | ||
} | ||
|
||
{ | ||
b := buf.FromBytes([]byte("hello")) | ||
common.Must(muxClientDownlink.Writer.WriteMultiBuffer(buf.MultiBuffer{b})) | ||
} | ||
|
||
resMb, err := websiteUplink.Reader.ReadMultiBuffer() | ||
common.Must(err) | ||
res := resMb.String() | ||
if res != "hello" { | ||
t.Error("upload: ", res) | ||
} | ||
|
||
{ | ||
b := buf.FromBytes([]byte("world")) | ||
common.Must(websiteUplink.Writer.WriteMultiBuffer(buf.MultiBuffer{b})) | ||
} | ||
|
||
resMb, err = muxClientDownlink.Reader.ReadMultiBuffer() | ||
common.Must(err) | ||
res = resMb.String() | ||
if res != "world" { | ||
t.Error("download: ", res) | ||
} | ||
|
||
outbounds := session.OutboundsFromContext(serverCtx) | ||
if outbounds[0] != originalOutbounds[0] { | ||
t.Error("outbound got reassigned: ", outbounds[0]) | ||
} | ||
|
||
if outbounds[0].Target.Address != nil { | ||
t.Error("outbound target got leaked: ", outbounds[0].Target.String()) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters