2021-08-02 13:07:28 +08:00
|
|
|
package plugin
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"fmt"
|
2024-06-12 17:30:10 +08:00
|
|
|
"net/http"
|
2021-08-02 13:07:28 +08:00
|
|
|
"strconv"
|
|
|
|
|
2023-02-27 14:44:16 +08:00
|
|
|
"github.com/onsi/ginkgo/v2"
|
2022-08-29 01:02:53 +08:00
|
|
|
|
2021-08-02 13:07:28 +08:00
|
|
|
"github.com/fatedier/frp/pkg/transport"
|
|
|
|
"github.com/fatedier/frp/test/e2e/framework"
|
|
|
|
"github.com/fatedier/frp/test/e2e/framework/consts"
|
|
|
|
"github.com/fatedier/frp/test/e2e/mock/server/httpserver"
|
|
|
|
"github.com/fatedier/frp/test/e2e/pkg/cert"
|
|
|
|
"github.com/fatedier/frp/test/e2e/pkg/port"
|
|
|
|
"github.com/fatedier/frp/test/e2e/pkg/request"
|
|
|
|
)
|
|
|
|
|
2022-08-29 01:02:53 +08:00
|
|
|
var _ = ginkgo.Describe("[Feature: Client-Plugins]", func() {
|
2021-08-02 13:07:28 +08:00
|
|
|
f := framework.NewDefaultFramework()
|
|
|
|
|
2022-08-29 01:02:53 +08:00
|
|
|
ginkgo.Describe("UnixDomainSocket", func() {
|
|
|
|
ginkgo.It("Expose a unix domain socket echo server", func() {
|
2021-08-02 13:07:28 +08:00
|
|
|
serverConf := consts.DefaultServerConfig
|
|
|
|
clientConf := consts.DefaultClientConfig
|
|
|
|
|
|
|
|
getProxyConf := func(proxyName string, portName string, extra string) string {
|
|
|
|
return fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
[[proxies]]
|
|
|
|
name = "%s"
|
|
|
|
type = "tcp"
|
|
|
|
remotePort = {{ .%s }}
|
2024-03-12 15:14:13 +08:00
|
|
|
`+extra, proxyName, portName) + fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
[proxies.plugin]
|
|
|
|
type = "unix_domain_socket"
|
|
|
|
unixPath = "{{ .%s }}"
|
2024-03-12 15:14:13 +08:00
|
|
|
`, framework.UDSEchoServerAddr)
|
2021-08-02 13:07:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
proxyName string
|
|
|
|
portName string
|
|
|
|
extraConfig string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
proxyName: "normal",
|
|
|
|
portName: port.GenName("Normal"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
proxyName: "with-encryption",
|
|
|
|
portName: port.GenName("WithEncryption"),
|
2023-09-13 16:32:39 +08:00
|
|
|
extraConfig: "transport.useEncryption = true",
|
2021-08-02 13:07:28 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
proxyName: "with-compression",
|
|
|
|
portName: port.GenName("WithCompression"),
|
2023-09-13 16:32:39 +08:00
|
|
|
extraConfig: "transport.useCompression = true",
|
2021-08-02 13:07:28 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
proxyName: "with-encryption-and-compression",
|
|
|
|
portName: port.GenName("WithEncryptionAndCompression"),
|
|
|
|
extraConfig: `
|
2023-09-13 16:32:39 +08:00
|
|
|
transport.useEncryption = true
|
|
|
|
transport.useCompression = true
|
2021-08-02 13:07:28 +08:00
|
|
|
`,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
// build all client config
|
|
|
|
for _, test := range tests {
|
|
|
|
clientConf += getProxyConf(test.proxyName, test.portName, test.extraConfig) + "\n"
|
|
|
|
}
|
|
|
|
// run frps and frpc
|
|
|
|
f.RunProcesses([]string{serverConf}, []string{clientConf})
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
framework.NewRequestExpect(f).Port(f.PortByName(test.portName)).Ensure()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2022-08-29 01:02:53 +08:00
|
|
|
ginkgo.It("http_proxy", func() {
|
2021-08-02 13:07:28 +08:00
|
|
|
serverConf := consts.DefaultServerConfig
|
|
|
|
clientConf := consts.DefaultClientConfig
|
|
|
|
|
|
|
|
remotePort := f.AllocPort()
|
|
|
|
clientConf += fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
[[proxies]]
|
|
|
|
name = "tcp"
|
|
|
|
type = "tcp"
|
|
|
|
remotePort = %d
|
|
|
|
[proxies.plugin]
|
|
|
|
type = "http_proxy"
|
|
|
|
httpUser = "abc"
|
|
|
|
httpPassword = "123"
|
2021-08-02 13:07:28 +08:00
|
|
|
`, remotePort)
|
|
|
|
|
|
|
|
f.RunProcesses([]string{serverConf}, []string{clientConf})
|
|
|
|
|
|
|
|
// http proxy, no auth info
|
|
|
|
framework.NewRequestExpect(f).PortName(framework.HTTPSimpleServerPort).RequestModify(func(r *request.Request) {
|
|
|
|
r.HTTP().Proxy("http://127.0.0.1:" + strconv.Itoa(remotePort))
|
|
|
|
}).Ensure(framework.ExpectResponseCode(407))
|
|
|
|
|
|
|
|
// http proxy, correct auth
|
|
|
|
framework.NewRequestExpect(f).PortName(framework.HTTPSimpleServerPort).RequestModify(func(r *request.Request) {
|
|
|
|
r.HTTP().Proxy("http://abc:123@127.0.0.1:" + strconv.Itoa(remotePort))
|
|
|
|
}).Ensure()
|
|
|
|
|
|
|
|
// connect TCP server by CONNECT method
|
|
|
|
framework.NewRequestExpect(f).PortName(framework.TCPEchoServerPort).RequestModify(func(r *request.Request) {
|
|
|
|
r.TCP().Proxy("http://abc:123@127.0.0.1:" + strconv.Itoa(remotePort))
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2022-08-29 01:02:53 +08:00
|
|
|
ginkgo.It("socks5 proxy", func() {
|
2021-08-02 13:07:28 +08:00
|
|
|
serverConf := consts.DefaultServerConfig
|
|
|
|
clientConf := consts.DefaultClientConfig
|
|
|
|
|
|
|
|
remotePort := f.AllocPort()
|
|
|
|
clientConf += fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
[[proxies]]
|
|
|
|
name = "tcp"
|
|
|
|
type = "tcp"
|
|
|
|
remotePort = %d
|
|
|
|
[proxies.plugin]
|
|
|
|
type = "socks5"
|
|
|
|
username = "abc"
|
|
|
|
password = "123"
|
2021-08-02 13:07:28 +08:00
|
|
|
`, remotePort)
|
|
|
|
|
|
|
|
f.RunProcesses([]string{serverConf}, []string{clientConf})
|
|
|
|
|
|
|
|
// http proxy, no auth info
|
|
|
|
framework.NewRequestExpect(f).PortName(framework.TCPEchoServerPort).RequestModify(func(r *request.Request) {
|
|
|
|
r.TCP().Proxy("socks5://127.0.0.1:" + strconv.Itoa(remotePort))
|
|
|
|
}).ExpectError(true).Ensure()
|
|
|
|
|
|
|
|
// http proxy, correct auth
|
|
|
|
framework.NewRequestExpect(f).PortName(framework.TCPEchoServerPort).RequestModify(func(r *request.Request) {
|
|
|
|
r.TCP().Proxy("socks5://abc:123@127.0.0.1:" + strconv.Itoa(remotePort))
|
|
|
|
}).Ensure()
|
|
|
|
})
|
|
|
|
|
2022-08-29 01:02:53 +08:00
|
|
|
ginkgo.It("static_file", func() {
|
2021-08-02 13:07:28 +08:00
|
|
|
vhostPort := f.AllocPort()
|
|
|
|
serverConf := consts.DefaultServerConfig + fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
vhostHTTPPort = %d
|
2021-08-02 13:07:28 +08:00
|
|
|
`, vhostPort)
|
|
|
|
clientConf := consts.DefaultClientConfig
|
|
|
|
|
|
|
|
remotePort := f.AllocPort()
|
|
|
|
f.WriteTempFile("test_static_file", "foo")
|
|
|
|
clientConf += fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
[[proxies]]
|
|
|
|
name = "tcp"
|
|
|
|
type = "tcp"
|
|
|
|
remotePort = %d
|
|
|
|
[proxies.plugin]
|
|
|
|
type = "static_file"
|
|
|
|
localPath = "%s"
|
|
|
|
|
|
|
|
[[proxies]]
|
|
|
|
name = "http"
|
|
|
|
type = "http"
|
|
|
|
customDomains = ["example.com"]
|
|
|
|
[proxies.plugin]
|
|
|
|
type = "static_file"
|
|
|
|
localPath = "%s"
|
|
|
|
|
|
|
|
[[proxies]]
|
|
|
|
name = "http-with-auth"
|
|
|
|
type = "http"
|
|
|
|
customDomains = ["other.example.com"]
|
|
|
|
[proxies.plugin]
|
|
|
|
type = "static_file"
|
|
|
|
localPath = "%s"
|
|
|
|
httpUser = "abc"
|
|
|
|
httpPassword = "123"
|
2021-08-02 13:07:28 +08:00
|
|
|
`, remotePort, f.TempDirectory, f.TempDirectory, f.TempDirectory)
|
|
|
|
|
|
|
|
f.RunProcesses([]string{serverConf}, []string{clientConf})
|
|
|
|
|
|
|
|
// from tcp proxy
|
|
|
|
framework.NewRequestExpect(f).Request(
|
|
|
|
framework.NewHTTPRequest().HTTPPath("/test_static_file").Port(remotePort),
|
|
|
|
).ExpectResp([]byte("foo")).Ensure()
|
|
|
|
|
|
|
|
// from http proxy without auth
|
|
|
|
framework.NewRequestExpect(f).Request(
|
|
|
|
framework.NewHTTPRequest().HTTPHost("example.com").HTTPPath("/test_static_file").Port(vhostPort),
|
|
|
|
).ExpectResp([]byte("foo")).Ensure()
|
|
|
|
|
|
|
|
// from http proxy with auth
|
|
|
|
framework.NewRequestExpect(f).Request(
|
2022-05-26 23:57:30 +08:00
|
|
|
framework.NewHTTPRequest().HTTPHost("other.example.com").HTTPPath("/test_static_file").Port(vhostPort).HTTPAuth("abc", "123"),
|
2021-08-02 13:07:28 +08:00
|
|
|
).ExpectResp([]byte("foo")).Ensure()
|
|
|
|
})
|
|
|
|
|
2022-08-29 01:02:53 +08:00
|
|
|
ginkgo.It("http2https", func() {
|
2021-08-02 13:07:28 +08:00
|
|
|
serverConf := consts.DefaultServerConfig
|
|
|
|
vhostHTTPPort := f.AllocPort()
|
|
|
|
serverConf += fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
vhostHTTPPort = %d
|
2021-08-02 13:07:28 +08:00
|
|
|
`, vhostHTTPPort)
|
|
|
|
|
|
|
|
localPort := f.AllocPort()
|
|
|
|
clientConf := consts.DefaultClientConfig + fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
[[proxies]]
|
|
|
|
name = "http2https"
|
|
|
|
type = "http"
|
|
|
|
customDomains = ["example.com"]
|
|
|
|
[proxies.plugin]
|
|
|
|
type = "http2https"
|
|
|
|
localAddr = "127.0.0.1:%d"
|
2021-08-02 13:07:28 +08:00
|
|
|
`, localPort)
|
|
|
|
|
|
|
|
f.RunProcesses([]string{serverConf}, []string{clientConf})
|
|
|
|
|
|
|
|
tlsConfig, err := transport.NewServerTLSConfig("", "", "")
|
|
|
|
framework.ExpectNoError(err)
|
|
|
|
localServer := httpserver.New(
|
|
|
|
httpserver.WithBindPort(localPort),
|
2022-08-29 01:02:53 +08:00
|
|
|
httpserver.WithTLSConfig(tlsConfig),
|
2021-08-02 13:07:28 +08:00
|
|
|
httpserver.WithResponse([]byte("test")),
|
|
|
|
)
|
|
|
|
f.RunServer("", localServer)
|
|
|
|
|
|
|
|
framework.NewRequestExpect(f).
|
|
|
|
Port(vhostHTTPPort).
|
|
|
|
RequestModify(func(r *request.Request) {
|
|
|
|
r.HTTP().HTTPHost("example.com")
|
|
|
|
}).
|
|
|
|
ExpectResp([]byte("test")).
|
|
|
|
Ensure()
|
|
|
|
})
|
|
|
|
|
2022-08-29 01:02:53 +08:00
|
|
|
ginkgo.It("https2http", func() {
|
2021-08-02 13:07:28 +08:00
|
|
|
generator := &cert.SelfSignedCertGenerator{}
|
|
|
|
artifacts, err := generator.Generate("example.com")
|
|
|
|
framework.ExpectNoError(err)
|
|
|
|
crtPath := f.WriteTempFile("server.crt", string(artifacts.Cert))
|
|
|
|
keyPath := f.WriteTempFile("server.key", string(artifacts.Key))
|
|
|
|
|
|
|
|
serverConf := consts.DefaultServerConfig
|
|
|
|
vhostHTTPSPort := f.AllocPort()
|
|
|
|
serverConf += fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
vhostHTTPSPort = %d
|
2021-08-02 13:07:28 +08:00
|
|
|
`, vhostHTTPSPort)
|
|
|
|
|
|
|
|
localPort := f.AllocPort()
|
|
|
|
clientConf := consts.DefaultClientConfig + fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
[[proxies]]
|
|
|
|
name = "https2http"
|
|
|
|
type = "https"
|
|
|
|
customDomains = ["example.com"]
|
|
|
|
[proxies.plugin]
|
|
|
|
type = "https2http"
|
|
|
|
localAddr = "127.0.0.1:%d"
|
|
|
|
crtPath = "%s"
|
|
|
|
keyPath = "%s"
|
2021-08-02 13:07:28 +08:00
|
|
|
`, localPort, crtPath, keyPath)
|
|
|
|
|
|
|
|
f.RunProcesses([]string{serverConf}, []string{clientConf})
|
|
|
|
|
|
|
|
localServer := httpserver.New(
|
|
|
|
httpserver.WithBindPort(localPort),
|
|
|
|
httpserver.WithResponse([]byte("test")),
|
|
|
|
)
|
|
|
|
f.RunServer("", localServer)
|
|
|
|
|
|
|
|
framework.NewRequestExpect(f).
|
|
|
|
Port(vhostHTTPSPort).
|
|
|
|
RequestModify(func(r *request.Request) {
|
|
|
|
r.HTTPS().HTTPHost("example.com").TLSConfig(&tls.Config{
|
|
|
|
ServerName: "example.com",
|
|
|
|
InsecureSkipVerify: true,
|
|
|
|
})
|
|
|
|
}).
|
|
|
|
ExpectResp([]byte("test")).
|
|
|
|
Ensure()
|
|
|
|
})
|
|
|
|
|
2022-08-29 01:02:53 +08:00
|
|
|
ginkgo.It("https2https", func() {
|
2021-08-02 13:07:28 +08:00
|
|
|
generator := &cert.SelfSignedCertGenerator{}
|
|
|
|
artifacts, err := generator.Generate("example.com")
|
|
|
|
framework.ExpectNoError(err)
|
|
|
|
crtPath := f.WriteTempFile("server.crt", string(artifacts.Cert))
|
|
|
|
keyPath := f.WriteTempFile("server.key", string(artifacts.Key))
|
|
|
|
|
|
|
|
serverConf := consts.DefaultServerConfig
|
|
|
|
vhostHTTPSPort := f.AllocPort()
|
|
|
|
serverConf += fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
vhostHTTPSPort = %d
|
2021-08-02 13:07:28 +08:00
|
|
|
`, vhostHTTPSPort)
|
|
|
|
|
|
|
|
localPort := f.AllocPort()
|
|
|
|
clientConf := consts.DefaultClientConfig + fmt.Sprintf(`
|
2023-09-13 16:32:39 +08:00
|
|
|
[[proxies]]
|
|
|
|
name = "https2https"
|
|
|
|
type = "https"
|
|
|
|
customDomains = ["example.com"]
|
|
|
|
[proxies.plugin]
|
|
|
|
type = "https2https"
|
|
|
|
localAddr = "127.0.0.1:%d"
|
|
|
|
crtPath = "%s"
|
|
|
|
keyPath = "%s"
|
2021-08-02 13:07:28 +08:00
|
|
|
`, localPort, crtPath, keyPath)
|
|
|
|
|
|
|
|
f.RunProcesses([]string{serverConf}, []string{clientConf})
|
|
|
|
|
|
|
|
tlsConfig, err := transport.NewServerTLSConfig("", "", "")
|
|
|
|
framework.ExpectNoError(err)
|
|
|
|
localServer := httpserver.New(
|
|
|
|
httpserver.WithBindPort(localPort),
|
|
|
|
httpserver.WithResponse([]byte("test")),
|
2022-08-29 01:02:53 +08:00
|
|
|
httpserver.WithTLSConfig(tlsConfig),
|
2021-08-02 13:07:28 +08:00
|
|
|
)
|
|
|
|
f.RunServer("", localServer)
|
|
|
|
|
|
|
|
framework.NewRequestExpect(f).
|
|
|
|
Port(vhostHTTPSPort).
|
|
|
|
RequestModify(func(r *request.Request) {
|
|
|
|
r.HTTPS().HTTPHost("example.com").TLSConfig(&tls.Config{
|
|
|
|
ServerName: "example.com",
|
|
|
|
InsecureSkipVerify: true,
|
|
|
|
})
|
|
|
|
}).
|
|
|
|
ExpectResp([]byte("test")).
|
|
|
|
Ensure()
|
|
|
|
})
|
2024-06-12 17:30:10 +08:00
|
|
|
|
|
|
|
ginkgo.Describe("http2http", func() {
|
|
|
|
ginkgo.It("host header rewrite", func() {
|
|
|
|
serverConf := consts.DefaultServerConfig
|
|
|
|
|
|
|
|
localPort := f.AllocPort()
|
|
|
|
remotePort := f.AllocPort()
|
|
|
|
clientConf := consts.DefaultClientConfig + fmt.Sprintf(`
|
|
|
|
[[proxies]]
|
|
|
|
name = "http2http"
|
|
|
|
type = "tcp"
|
|
|
|
remotePort = %d
|
|
|
|
[proxies.plugin]
|
|
|
|
type = "http2http"
|
|
|
|
localAddr = "127.0.0.1:%d"
|
|
|
|
hostHeaderRewrite = "rewrite.test.com"
|
|
|
|
`, remotePort, localPort)
|
|
|
|
|
|
|
|
f.RunProcesses([]string{serverConf}, []string{clientConf})
|
|
|
|
|
|
|
|
localServer := httpserver.New(
|
|
|
|
httpserver.WithBindPort(localPort),
|
|
|
|
httpserver.WithHandler(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
|
|
|
|
_, _ = w.Write([]byte(req.Host))
|
|
|
|
})),
|
|
|
|
)
|
|
|
|
f.RunServer("", localServer)
|
|
|
|
|
|
|
|
framework.NewRequestExpect(f).
|
|
|
|
Port(remotePort).
|
|
|
|
RequestModify(func(r *request.Request) {
|
|
|
|
r.HTTP().HTTPHost("example.com")
|
|
|
|
}).
|
|
|
|
ExpectResp([]byte("rewrite.test.com")).
|
|
|
|
Ensure()
|
|
|
|
})
|
|
|
|
|
|
|
|
ginkgo.It("set request header", func() {
|
|
|
|
serverConf := consts.DefaultServerConfig
|
|
|
|
|
|
|
|
localPort := f.AllocPort()
|
|
|
|
remotePort := f.AllocPort()
|
|
|
|
clientConf := consts.DefaultClientConfig + fmt.Sprintf(`
|
|
|
|
[[proxies]]
|
|
|
|
name = "http2http"
|
|
|
|
type = "tcp"
|
|
|
|
remotePort = %d
|
|
|
|
[proxies.plugin]
|
|
|
|
type = "http2http"
|
|
|
|
localAddr = "127.0.0.1:%d"
|
|
|
|
requestHeaders.set.x-from-where = "frp"
|
|
|
|
`, remotePort, localPort)
|
|
|
|
|
|
|
|
f.RunProcesses([]string{serverConf}, []string{clientConf})
|
|
|
|
|
|
|
|
localServer := httpserver.New(
|
|
|
|
httpserver.WithBindPort(localPort),
|
|
|
|
httpserver.WithHandler(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
|
|
|
|
_, _ = w.Write([]byte(req.Header.Get("x-from-where")))
|
|
|
|
})),
|
|
|
|
)
|
|
|
|
f.RunServer("", localServer)
|
|
|
|
|
|
|
|
framework.NewRequestExpect(f).
|
|
|
|
Port(remotePort).
|
|
|
|
RequestModify(func(r *request.Request) {
|
|
|
|
r.HTTP().HTTPHost("example.com")
|
|
|
|
}).
|
|
|
|
ExpectResp([]byte("frp")).
|
|
|
|
Ensure()
|
|
|
|
})
|
|
|
|
})
|
2021-08-02 13:07:28 +08:00
|
|
|
})
|