Compare commits
11 Commits
Author | SHA1 | Date | |
---|---|---|---|
4e17fee681 | |||
a8eed30db5 | |||
c1eaf43058 | |||
9f5aca124d | |||
54958724e7 | |||
7a17672149 | |||
44b807d3d1 | |||
0d847462b3 | |||
deed4207ea | |||
f6363fed07 | |||
1de78f2f06 |
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
||||
.idea
|
464
curl_test.go
Normal file
464
curl_test.go
Normal file
@ -0,0 +1,464 @@
|
||||
package starnet
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/http"
|
||||
"net/http/httptest"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestUrlEncodeRaw(t *testing.T) {
|
||||
input := "hello world!@#$%^&*()_+-=~`"
|
||||
expected := "hello%20world%21%40%23%24%25%5E%26%2A%28%29_%2B-%3D~%60"
|
||||
result := UrlEncodeRaw(input)
|
||||
if result != expected {
|
||||
t.Errorf("UrlEncodeRaw(%q) = %q; want %q", input, result, expected)
|
||||
}
|
||||
}
|
||||
|
||||
func TestUrlEncode(t *testing.T) {
|
||||
input := "hello world!@#$%^&*()_+-=~`"
|
||||
expected := `hello+world%21%40%23%24%25%5E%26%2A%28%29_%2B-%3D~%60`
|
||||
result := UrlEncode(input)
|
||||
if result != expected {
|
||||
t.Errorf("UrlEncode(%q) = %q; want %q", input, result, expected)
|
||||
}
|
||||
}
|
||||
|
||||
func TestUrlDecode(t *testing.T) {
|
||||
input := "hello%20world%21%40%23%24%25%5E%26*%28%29_%2B-%3D~%60"
|
||||
expected := "hello world!@#$%^&*()_+-=~`"
|
||||
result, err := UrlDecode(input)
|
||||
if err != nil {
|
||||
t.Errorf("UrlDecode(%q) returned error: %v", input, err)
|
||||
}
|
||||
if result != expected {
|
||||
t.Errorf("UrlDecode(%q) = %q; want %q", input, result, expected)
|
||||
}
|
||||
|
||||
// Test for error case
|
||||
invalidInput := "%zz"
|
||||
_, err = UrlDecode(invalidInput)
|
||||
if err == nil {
|
||||
t.Errorf("UrlDecode(%q) expected error, got nil", invalidInput)
|
||||
}
|
||||
}
|
||||
|
||||
func TestBuildPostForm_WithValidInput(t *testing.T) {
|
||||
input := map[string]string{
|
||||
"key1": "value1",
|
||||
"key2": "value2",
|
||||
}
|
||||
|
||||
expected := []byte("key1=value1&key2=value2")
|
||||
|
||||
result := BuildPostForm(input)
|
||||
|
||||
if string(result) != string(expected) {
|
||||
t.Errorf("BuildPostForm(%v) = %v; want %v", input, result, expected)
|
||||
}
|
||||
}
|
||||
|
||||
func TestBuildPostForm_WithEmptyInput(t *testing.T) {
|
||||
input := map[string]string{}
|
||||
|
||||
expected := []byte("")
|
||||
|
||||
result := BuildPostForm(input)
|
||||
|
||||
if string(result) != string(expected) {
|
||||
t.Errorf("BuildPostForm(%v) = %v; want %v", input, result, expected)
|
||||
}
|
||||
}
|
||||
|
||||
func TestBuildPostForm_WithNilInput(t *testing.T) {
|
||||
var input map[string]string
|
||||
|
||||
expected := []byte("")
|
||||
|
||||
result := BuildPostForm(input)
|
||||
|
||||
if string(result) != string(expected) {
|
||||
t.Errorf("BuildPostForm(%v) = %v; want %v", input, result, expected)
|
||||
}
|
||||
}
|
||||
|
||||
func TestGetRequest(t *testing.T) {
|
||||
server := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
||||
rw.Write([]byte(`OK`))
|
||||
}))
|
||||
defer server.Close()
|
||||
|
||||
resp, err := Get(server.URL)
|
||||
if err != nil {
|
||||
t.Errorf("Unexpected error: %v", err)
|
||||
}
|
||||
|
||||
body := resp.Body().String()
|
||||
if body != "OK" {
|
||||
t.Errorf("Expected OK, got %v", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestPostRequest(t *testing.T) {
|
||||
server := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
||||
if req.Method != http.MethodPost {
|
||||
t.Errorf("Expected 'POST', got %v", req.Method)
|
||||
}
|
||||
rw.Write([]byte(`OK`))
|
||||
}))
|
||||
defer server.Close()
|
||||
|
||||
resp, err := Post(server.URL)
|
||||
if err != nil {
|
||||
t.Errorf("Unexpected error: %v", err)
|
||||
}
|
||||
|
||||
body := resp.Body().String()
|
||||
if body != "OK" {
|
||||
t.Errorf("Expected OK, got %v", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestOptionsRequestWithValidInput(t *testing.T) {
|
||||
server := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
||||
if req.Method != http.MethodOptions {
|
||||
t.Errorf("Expected 'OPTIONS', got %v", req.Method)
|
||||
}
|
||||
rw.Write([]byte(`OK`))
|
||||
}))
|
||||
defer server.Close()
|
||||
|
||||
resp, err := Options(server.URL)
|
||||
if err != nil {
|
||||
t.Errorf("Unexpected error: %v", err)
|
||||
}
|
||||
|
||||
body := resp.Body().String()
|
||||
if body != "OK" {
|
||||
t.Errorf("Expected OK, got %v", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestPutRequestWithValidInput(t *testing.T) {
|
||||
server := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
||||
if req.Method != http.MethodPut {
|
||||
t.Errorf("Expected 'PUT', got %v", req.Method)
|
||||
}
|
||||
rw.Write([]byte(`OK`))
|
||||
}))
|
||||
defer server.Close()
|
||||
|
||||
resp, err := Put(server.URL)
|
||||
if err != nil {
|
||||
t.Errorf("Unexpected error: %v", err)
|
||||
}
|
||||
|
||||
body := resp.Body().String()
|
||||
if body != "OK" {
|
||||
t.Errorf("Expected OK, got %v", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestDeleteRequestWithValidInput(t *testing.T) {
|
||||
server := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
||||
if req.Method != http.MethodDelete {
|
||||
t.Errorf("Expected 'DELETE', got %v", req.Method)
|
||||
}
|
||||
rw.Write([]byte(`OK`))
|
||||
}))
|
||||
defer server.Close()
|
||||
|
||||
resp, err := Delete(server.URL)
|
||||
if err != nil {
|
||||
t.Errorf("Unexpected error: %v", err)
|
||||
}
|
||||
|
||||
body := resp.Body().String()
|
||||
if body != "OK" {
|
||||
t.Errorf("Expected OK, got %v", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestHeadRequestWithValidInput(t *testing.T) {
|
||||
server := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
||||
if req.Method != http.MethodHead {
|
||||
t.Errorf("Expected 'HEAD', got %v", req.Method)
|
||||
}
|
||||
rw.Write([]byte(`OK`))
|
||||
}))
|
||||
defer server.Close()
|
||||
|
||||
resp, err := Head(server.URL)
|
||||
if err != nil {
|
||||
t.Errorf("Unexpected error: %v", err)
|
||||
}
|
||||
|
||||
body := resp.Body().String()
|
||||
if body == "OK" {
|
||||
t.Errorf("Expected , got %v", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestPatchRequestWithValidInput(t *testing.T) {
|
||||
server := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
||||
if req.Method != http.MethodPatch {
|
||||
t.Errorf("Expected 'PATCH', got %v", req.Method)
|
||||
}
|
||||
rw.Write([]byte(`OK`))
|
||||
}))
|
||||
defer server.Close()
|
||||
|
||||
resp, err := Patch(server.URL)
|
||||
if err != nil {
|
||||
t.Errorf("Unexpected error: %v", err)
|
||||
}
|
||||
|
||||
body := resp.Body().String()
|
||||
if body != "OK" {
|
||||
t.Errorf("Expected OK, got %v", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestTraceRequestWithValidInput(t *testing.T) {
|
||||
server := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
||||
if req.Method != http.MethodTrace {
|
||||
t.Errorf("Expected 'TRACE', got %v", req.Method)
|
||||
}
|
||||
rw.Write([]byte(`OK`))
|
||||
}))
|
||||
defer server.Close()
|
||||
|
||||
resp, err := Trace(server.URL)
|
||||
if err != nil {
|
||||
t.Errorf("Unexpected error: %v", err)
|
||||
}
|
||||
|
||||
body := resp.Body().String()
|
||||
if body != "OK" {
|
||||
t.Errorf("Expected OK, got %v", body)
|
||||
}
|
||||
}
|
||||
|
||||
func TestConnectRequestWithValidInput(t *testing.T) {
|
||||
server := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) {
|
||||
if req.Method != http.MethodConnect {
|
||||
t.Errorf("Expected 'CONNECT', got %v", req.Method)
|
||||
}
|
||||
rw.Write([]byte(`OK`))
|
||||
}))
|
||||
defer server.Close()
|
||||
|
||||
resp, err := Connect(server.URL)
|
||||
if err != nil {
|
||||
t.Errorf("Unexpected error: %v", err)
|
||||
}
|
||||
|
||||
body := resp.Body().String()
|
||||
if body != "OK" {
|
||||
t.Errorf("Expected OK, got %v", body)
|
||||
}
|
||||
}
|
||||
func TestMethodReturnsCorrectValue(t *testing.T) {
|
||||
req := NewReq("https://example.com")
|
||||
req.SetMethodNoError("GET")
|
||||
if req.Method() != "GET" {
|
||||
t.Errorf("Expected 'GET', got %v", req.Method())
|
||||
}
|
||||
}
|
||||
|
||||
func TestSetMethodHandlesInvalidInput(t *testing.T) {
|
||||
req := NewReq("https://example.com")
|
||||
err := req.SetMethod("我是谁")
|
||||
if err == nil {
|
||||
t.Errorf("Expected error, got nil")
|
||||
}
|
||||
}
|
||||
|
||||
func TestSetMethodNoErrorSetsMethodCorrectly(t *testing.T) {
|
||||
req := NewReq("https://example.com")
|
||||
req.SetMethodNoError("POST")
|
||||
if req.Method() != "POST" {
|
||||
t.Errorf("Expected 'POST', got %v", req.Method())
|
||||
}
|
||||
}
|
||||
|
||||
func TestSetMethodNoErrorIgnoresInvalidInput(t *testing.T) {
|
||||
req := NewReq("https://example.com")
|
||||
req.SetMethodNoError("你是谁")
|
||||
if req.Method() != "GET" {
|
||||
t.Errorf("Expected '', got %v", req.Method())
|
||||
}
|
||||
}
|
||||
|
||||
func TestUriReturnsCorrectValue(t *testing.T) {
|
||||
req := NewReq("https://example.com")
|
||||
if req.Uri() != "https://example.com" {
|
||||
t.Errorf("Expected 'https://example.com', got %v", req.Uri())
|
||||
}
|
||||
}
|
||||
|
||||
func TestSetUriHandlesValidInput(t *testing.T) {
|
||||
req := NewReq("https://example.com")
|
||||
err := req.SetUri("https://newexample.com")
|
||||
if err != nil {
|
||||
t.Errorf("Unexpected error: %v", err)
|
||||
}
|
||||
if req.Uri() != "https://newexample.com" {
|
||||
t.Errorf("Expected 'https://newexample.com', got %v", req.Uri())
|
||||
}
|
||||
}
|
||||
|
||||
func TestSetUriHandlesInvalidInput(t *testing.T) {
|
||||
req := NewReq("https://example.com")
|
||||
err := req.SetUri("://invalidurl")
|
||||
if err == nil {
|
||||
t.Errorf("Expected error, got nil")
|
||||
}
|
||||
}
|
||||
|
||||
func TestSetUriNoErrorSetsUriCorrectly(t *testing.T) {
|
||||
req := NewReq("https://example.com")
|
||||
req.SetUriNoError("https://newexample.com")
|
||||
if req.Uri() != "https://newexample.com" {
|
||||
t.Errorf("Expected 'https://newexample.com', got %v", req.Uri())
|
||||
}
|
||||
}
|
||||
|
||||
func TestSetUriNoErrorIgnoresInvalidInput(t *testing.T) {
|
||||
req := NewReq("https://example.com")
|
||||
req.SetUriNoError("://invalidurl")
|
||||
if req.Uri() != "https://example.com" {
|
||||
t.Errorf("Expected 'https://example.com', got %v", req.Uri())
|
||||
}
|
||||
}
|
||||
|
||||
type postmanReply struct {
|
||||
Args struct {
|
||||
} `json:"args"`
|
||||
Form map[string]string `json:"form"`
|
||||
Headers map[string]string `json:"headers"`
|
||||
Url string `json:"url"`
|
||||
}
|
||||
|
||||
func TestGet(t *testing.T) {
|
||||
var reply postmanReply
|
||||
resp, err := NewReq("https://postman-echo.com/get").
|
||||
AddHeader("hello", "nononmo").
|
||||
SetAutoCalcContentLength(true).
|
||||
Do()
|
||||
if err != nil {
|
||||
t.Error(err)
|
||||
}
|
||||
err = resp.Body().Unmarshal(&reply)
|
||||
if err != nil {
|
||||
t.Error(err)
|
||||
}
|
||||
fmt.Println(resp.Body().String())
|
||||
fmt.Println(reply.Headers)
|
||||
fmt.Println(resp.Cookies())
|
||||
}
|
||||
|
||||
type testData struct {
|
||||
name string
|
||||
args *Request
|
||||
want func(*Response) error
|
||||
wantErr bool
|
||||
}
|
||||
|
||||
func headerTestData() []testData {
|
||||
return []testData{
|
||||
{
|
||||
name: "addHeader",
|
||||
args: NewReq("https://postman-echo.com/get").
|
||||
AddHeader("b612", "test-data").
|
||||
AddHeader("b612", "test-header").
|
||||
AddSimpleCookie("b612", "test-cookie").
|
||||
SetHeader("User-Agent", "starnet test"),
|
||||
want: func(resp *Response) error {
|
||||
//fmt.Println(resp.Body().String())
|
||||
if resp == nil {
|
||||
return fmt.Errorf("response is nil")
|
||||
}
|
||||
if resp.StatusCode != 200 {
|
||||
return fmt.Errorf("status code is %d", resp.StatusCode)
|
||||
}
|
||||
var reply postmanReply
|
||||
err := resp.Body().Unmarshal(&reply)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if reply.Headers["b612"] != "test-data, test-header" {
|
||||
return fmt.Errorf("header not found")
|
||||
}
|
||||
if reply.Headers["user-agent"] != "starnet test" {
|
||||
return fmt.Errorf("user-agent not found")
|
||||
}
|
||||
if reply.Headers["cookie"] != "b612=test-cookie" {
|
||||
return fmt.Errorf("cookie not found")
|
||||
}
|
||||
return nil
|
||||
},
|
||||
wantErr: false,
|
||||
},
|
||||
{
|
||||
name: "postForm",
|
||||
args: NewSimpleRequest("https://postman-echo.com/post", "POST").
|
||||
AddHeader("b612", "test-data").
|
||||
AddHeader("b612", "test-header").
|
||||
AddSimpleCookie("b612", "test-cookie").
|
||||
SetHeader("User-Agent", "starnet test").
|
||||
//SetHeader("Content-Type", "application/x-www-form-urlencoded").
|
||||
AddFormData("hello", "world").
|
||||
AddFormData("hello2", "world2").
|
||||
SetMethodNoError("POST"),
|
||||
want: func(resp *Response) error {
|
||||
//fmt.Println(resp.Body().String())
|
||||
if resp == nil {
|
||||
return fmt.Errorf("response is nil")
|
||||
}
|
||||
if resp.StatusCode != 200 {
|
||||
return fmt.Errorf("status code is %d", resp.StatusCode)
|
||||
}
|
||||
var reply postmanReply
|
||||
err := resp.Body().Unmarshal(&reply)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if reply.Headers["b612"] != "test-data, test-header" {
|
||||
return fmt.Errorf("header not found")
|
||||
}
|
||||
if reply.Headers["user-agent"] != "starnet test" {
|
||||
return fmt.Errorf("user-agent not found")
|
||||
}
|
||||
if reply.Headers["cookie"] != "b612=test-cookie" {
|
||||
return fmt.Errorf("cookie not found")
|
||||
}
|
||||
if reply.Form["hello"] != "world" {
|
||||
return fmt.Errorf("form data not found")
|
||||
}
|
||||
if reply.Form["hello2"] != "world2" {
|
||||
return fmt.Errorf("form data not found")
|
||||
}
|
||||
return nil
|
||||
},
|
||||
wantErr: false,
|
||||
},
|
||||
}
|
||||
}
|
||||
func TestCurl(t *testing.T) {
|
||||
for _, tt := range headerTestData() {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
got, err := Curl(tt.args)
|
||||
if (err != nil) != tt.wantErr {
|
||||
t.Errorf("Curl() error = %v, wantErr %v", err, tt.wantErr)
|
||||
return
|
||||
}
|
||||
if tt.want != nil {
|
||||
if err := tt.want(got); err != nil {
|
||||
t.Errorf("Curl() = %v", err)
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
47
go.sum
47
go.sum
@ -1,47 +0,0 @@
|
||||
b612.me/stario v0.0.9 h1:bFDlejUJMwZ12a09snZJspQsOlkqpDAl9qKPEYOGWCk=
|
||||
b612.me/stario v0.0.9/go.mod h1:x4D/x8zA5SC0pj/uJAi4FyG5p4j5UZoMEZfvuRR6VNw=
|
||||
github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
|
||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
|
||||
golang.org/x/crypto v0.19.0/go.mod h1:Iy9bg/ha4yyC70EfRS8jz+B6ybOBKMaSxLj6P6oBDfU=
|
||||
golang.org/x/crypto v0.21.0 h1:X31++rzVUdKhX5sWmSOFZxx8UW/ldWx55cbf08iNAMA=
|
||||
golang.org/x/crypto v0.21.0/go.mod h1:0BP7YvVV9gBbVKyeTG0Gyn+gZm94bibOW5BjDEYAOMs=
|
||||
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
|
||||
golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
|
||||
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
||||
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
|
||||
golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
|
||||
golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
|
||||
golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg=
|
||||
golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44=
|
||||
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||
golang.org/x/sys v0.18.0 h1:DBdB3niSjOA/O0blCZBqDefyWNYveAYMNF1Wum0DYQ4=
|
||||
golang.org/x/sys v0.18.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
|
||||
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
|
||||
golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo=
|
||||
golang.org/x/term v0.17.0/go.mod h1:lLRBjIVuehSbZlaOtGMbcMncT+aqLLLmKrsjNrUguwk=
|
||||
golang.org/x/term v0.18.0 h1:FcHjZXDMxI8mM3nwhX9HlKop4C0YQvCVCdwYl2wOtE8=
|
||||
golang.org/x/term v0.18.0/go.mod h1:ILwASektA3OnRv7amZ1xhE/KTR+u50pbXfZ03+6Nx58=
|
||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
|
||||
golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
|
||||
golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=
|
||||
golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
|
||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
|
||||
golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
|
||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
120
httpguts.go
Normal file
120
httpguts.go
Normal file
@ -0,0 +1,120 @@
|
||||
package starnet
|
||||
|
||||
import "strings"
|
||||
|
||||
var isTokenTable = [127]bool{
|
||||
'!': true,
|
||||
'#': true,
|
||||
'$': true,
|
||||
'%': true,
|
||||
'&': true,
|
||||
'\'': true,
|
||||
'*': true,
|
||||
'+': true,
|
||||
'-': true,
|
||||
'.': true,
|
||||
'0': true,
|
||||
'1': true,
|
||||
'2': true,
|
||||
'3': true,
|
||||
'4': true,
|
||||
'5': true,
|
||||
'6': true,
|
||||
'7': true,
|
||||
'8': true,
|
||||
'9': true,
|
||||
'A': true,
|
||||
'B': true,
|
||||
'C': true,
|
||||
'D': true,
|
||||
'E': true,
|
||||
'F': true,
|
||||
'G': true,
|
||||
'H': true,
|
||||
'I': true,
|
||||
'J': true,
|
||||
'K': true,
|
||||
'L': true,
|
||||
'M': true,
|
||||
'N': true,
|
||||
'O': true,
|
||||
'P': true,
|
||||
'Q': true,
|
||||
'R': true,
|
||||
'S': true,
|
||||
'T': true,
|
||||
'U': true,
|
||||
'W': true,
|
||||
'V': true,
|
||||
'X': true,
|
||||
'Y': true,
|
||||
'Z': true,
|
||||
'^': true,
|
||||
'_': true,
|
||||
'`': true,
|
||||
'a': true,
|
||||
'b': true,
|
||||
'c': true,
|
||||
'd': true,
|
||||
'e': true,
|
||||
'f': true,
|
||||
'g': true,
|
||||
'h': true,
|
||||
'i': true,
|
||||
'j': true,
|
||||
'k': true,
|
||||
'l': true,
|
||||
'm': true,
|
||||
'n': true,
|
||||
'o': true,
|
||||
'p': true,
|
||||
'q': true,
|
||||
'r': true,
|
||||
's': true,
|
||||
't': true,
|
||||
'u': true,
|
||||
'v': true,
|
||||
'w': true,
|
||||
'x': true,
|
||||
'y': true,
|
||||
'z': true,
|
||||
'|': true,
|
||||
'~': true,
|
||||
}
|
||||
|
||||
func IsTokenRune(r rune) bool {
|
||||
i := int(r)
|
||||
return i < len(isTokenTable) && isTokenTable[i]
|
||||
}
|
||||
|
||||
func validMethod(method string) bool {
|
||||
/*
|
||||
Method = "OPTIONS" ; Section 9.2
|
||||
| "GET" ; Section 9.3
|
||||
| "HEAD" ; Section 9.4
|
||||
| "POST" ; Section 9.5
|
||||
| "PUT" ; Section 9.6
|
||||
| "DELETE" ; Section 9.7
|
||||
| "TRACE" ; Section 9.8
|
||||
| "CONNECT" ; Section 9.9
|
||||
| extension-method
|
||||
extension-method = token
|
||||
token = 1*<any CHAR except CTLs or separators>
|
||||
*/
|
||||
return len(method) > 0 && strings.IndexFunc(method, isNotToken) == -1
|
||||
}
|
||||
|
||||
func isNotToken(r rune) bool {
|
||||
return !IsTokenRune(r)
|
||||
}
|
||||
|
||||
func hasPort(s string) bool { return strings.LastIndex(s, ":") > strings.LastIndex(s, "]") }
|
||||
|
||||
// removeEmptyPort strips the empty port in ":port" to ""
|
||||
// as mandated by RFC 3986 Section 6.2.3.
|
||||
func removeEmptyPort(host string) string {
|
||||
if hasPort(host) {
|
||||
return strings.TrimSuffix(host, ":")
|
||||
}
|
||||
return host
|
||||
}
|
325
que.go
325
que.go
@ -1,325 +0,0 @@
|
||||
package starnet
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"context"
|
||||
"encoding/binary"
|
||||
"errors"
|
||||
"fmt"
|
||||
"os"
|
||||
"sync"
|
||||
"time"
|
||||
)
|
||||
|
||||
var ErrDeadlineExceeded error = errors.New("deadline exceeded")
|
||||
|
||||
// 识别头
|
||||
var header = []byte{11, 27, 19, 96, 12, 25, 02, 20}
|
||||
|
||||
// MsgQueue 为基本的信息单位
|
||||
type MsgQueue struct {
|
||||
ID uint16
|
||||
Msg []byte
|
||||
Conn interface{}
|
||||
}
|
||||
|
||||
// StarQueue 为流数据中的消息队列分发
|
||||
type StarQueue struct {
|
||||
maxLength uint32
|
||||
count int64
|
||||
Encode bool
|
||||
msgID uint16
|
||||
msgPool chan MsgQueue
|
||||
unFinMsg sync.Map
|
||||
lastID int //= -1
|
||||
ctx context.Context
|
||||
cancel context.CancelFunc
|
||||
duration time.Duration
|
||||
EncodeFunc func([]byte) []byte
|
||||
DecodeFunc func([]byte) []byte
|
||||
//restoreMu sync.Mutex
|
||||
}
|
||||
|
||||
func NewQueueCtx(ctx context.Context, count int64, maxMsgLength uint32) *StarQueue {
|
||||
var q StarQueue
|
||||
q.Encode = false
|
||||
q.count = count
|
||||
q.maxLength = maxMsgLength
|
||||
q.msgPool = make(chan MsgQueue, count)
|
||||
if ctx == nil {
|
||||
q.ctx, q.cancel = context.WithCancel(context.Background())
|
||||
} else {
|
||||
q.ctx, q.cancel = context.WithCancel(ctx)
|
||||
}
|
||||
q.duration = 0
|
||||
return &q
|
||||
}
|
||||
func NewQueueWithCount(count int64) *StarQueue {
|
||||
return NewQueueCtx(nil, count, 0)
|
||||
}
|
||||
|
||||
// NewQueue 建立一个新消息队列
|
||||
func NewQueue() *StarQueue {
|
||||
return NewQueueWithCount(32)
|
||||
}
|
||||
|
||||
// Uint32ToByte 4位uint32转byte
|
||||
func Uint32ToByte(src uint32) []byte {
|
||||
res := make([]byte, 4)
|
||||
res[3] = uint8(src)
|
||||
res[2] = uint8(src >> 8)
|
||||
res[1] = uint8(src >> 16)
|
||||
res[0] = uint8(src >> 24)
|
||||
return res
|
||||
}
|
||||
|
||||
// ByteToUint32 byte转4位uint32
|
||||
func ByteToUint32(src []byte) uint32 {
|
||||
var res uint32
|
||||
buffer := bytes.NewBuffer(src)
|
||||
binary.Read(buffer, binary.BigEndian, &res)
|
||||
return res
|
||||
}
|
||||
|
||||
// Uint16ToByte 2位uint16转byte
|
||||
func Uint16ToByte(src uint16) []byte {
|
||||
res := make([]byte, 2)
|
||||
res[1] = uint8(src)
|
||||
res[0] = uint8(src >> 8)
|
||||
return res
|
||||
}
|
||||
|
||||
// ByteToUint16 用于byte转uint16
|
||||
func ByteToUint16(src []byte) uint16 {
|
||||
var res uint16
|
||||
buffer := bytes.NewBuffer(src)
|
||||
binary.Read(buffer, binary.BigEndian, &res)
|
||||
return res
|
||||
}
|
||||
|
||||
// BuildMessage 生成编码后的信息用于发送
|
||||
func (q *StarQueue) BuildMessage(src []byte) []byte {
|
||||
var buff bytes.Buffer
|
||||
q.msgID++
|
||||
if q.Encode {
|
||||
src = q.EncodeFunc(src)
|
||||
}
|
||||
length := uint32(len(src))
|
||||
buff.Write(header)
|
||||
buff.Write(Uint32ToByte(length))
|
||||
buff.Write(Uint16ToByte(q.msgID))
|
||||
buff.Write(src)
|
||||
return buff.Bytes()
|
||||
}
|
||||
|
||||
// BuildHeader 生成编码后的Header用于发送
|
||||
func (q *StarQueue) BuildHeader(length uint32) []byte {
|
||||
var buff bytes.Buffer
|
||||
q.msgID++
|
||||
buff.Write(header)
|
||||
buff.Write(Uint32ToByte(length))
|
||||
buff.Write(Uint16ToByte(q.msgID))
|
||||
return buff.Bytes()
|
||||
}
|
||||
|
||||
type unFinMsg struct {
|
||||
ID uint16
|
||||
LengthRecv uint32
|
||||
// HeaderMsg 信息头,应当为14位:8位识别码+4位长度码+2位id
|
||||
HeaderMsg []byte
|
||||
RecvMsg []byte
|
||||
}
|
||||
|
||||
func (q *StarQueue) push2list(msg MsgQueue) {
|
||||
q.msgPool <- msg
|
||||
}
|
||||
|
||||
// ParseMessage 用于解析收到的msg信息
|
||||
func (q *StarQueue) ParseMessage(msg []byte, conn interface{}) error {
|
||||
return q.parseMessage(msg, conn)
|
||||
}
|
||||
|
||||
// parseMessage 用于解析收到的msg信息
|
||||
func (q *StarQueue) parseMessage(msg []byte, conn interface{}) error {
|
||||
tmp, ok := q.unFinMsg.Load(conn)
|
||||
if ok { //存在未完成的信息
|
||||
lastMsg := tmp.(*unFinMsg)
|
||||
headerLen := len(lastMsg.HeaderMsg)
|
||||
if headerLen < 14 { //未完成头标题
|
||||
//传输的数据不能填充header头
|
||||
if len(msg) < 14-headerLen {
|
||||
//加入header头并退出
|
||||
lastMsg.HeaderMsg = bytesMerge(lastMsg.HeaderMsg, msg)
|
||||
q.unFinMsg.Store(conn, lastMsg)
|
||||
return nil
|
||||
}
|
||||
//获取14字节完整的header
|
||||
header := msg[0 : 14-headerLen]
|
||||
lastMsg.HeaderMsg = bytesMerge(lastMsg.HeaderMsg, header)
|
||||
//检查收到的header是否为认证header
|
||||
//若不是,丢弃并重新来过
|
||||
if !checkHeader(lastMsg.HeaderMsg[0:8]) {
|
||||
q.unFinMsg.Delete(conn)
|
||||
if len(msg) == 0 {
|
||||
return nil
|
||||
}
|
||||
return q.parseMessage(msg, conn)
|
||||
}
|
||||
//获得本数据包长度
|
||||
lastMsg.LengthRecv = ByteToUint32(lastMsg.HeaderMsg[8:12])
|
||||
if q.maxLength != 0 && lastMsg.LengthRecv > q.maxLength {
|
||||
q.unFinMsg.Delete(conn)
|
||||
return fmt.Errorf("msg length is %d ,too large than %d", lastMsg.LengthRecv, q.maxLength)
|
||||
}
|
||||
//获得本数据包ID
|
||||
lastMsg.ID = ByteToUint16(lastMsg.HeaderMsg[12:14])
|
||||
//存入列表
|
||||
q.unFinMsg.Store(conn, lastMsg)
|
||||
msg = msg[14-headerLen:]
|
||||
if uint32(len(msg)) < lastMsg.LengthRecv {
|
||||
lastMsg.RecvMsg = msg
|
||||
q.unFinMsg.Store(conn, lastMsg)
|
||||
return nil
|
||||
}
|
||||
if uint32(len(msg)) >= lastMsg.LengthRecv {
|
||||
lastMsg.RecvMsg = msg[0:lastMsg.LengthRecv]
|
||||
if q.Encode {
|
||||
lastMsg.RecvMsg = q.DecodeFunc(lastMsg.RecvMsg)
|
||||
}
|
||||
msg = msg[lastMsg.LengthRecv:]
|
||||
storeMsg := MsgQueue{
|
||||
ID: lastMsg.ID,
|
||||
Msg: lastMsg.RecvMsg,
|
||||
Conn: conn,
|
||||
}
|
||||
//q.restoreMu.Lock()
|
||||
q.push2list(storeMsg)
|
||||
//q.restoreMu.Unlock()
|
||||
q.unFinMsg.Delete(conn)
|
||||
return q.parseMessage(msg, conn)
|
||||
}
|
||||
} else {
|
||||
lastID := int(lastMsg.LengthRecv) - len(lastMsg.RecvMsg)
|
||||
if lastID < 0 {
|
||||
q.unFinMsg.Delete(conn)
|
||||
return q.parseMessage(msg, conn)
|
||||
}
|
||||
if len(msg) >= lastID {
|
||||
lastMsg.RecvMsg = bytesMerge(lastMsg.RecvMsg, msg[0:lastID])
|
||||
if q.Encode {
|
||||
lastMsg.RecvMsg = q.DecodeFunc(lastMsg.RecvMsg)
|
||||
}
|
||||
storeMsg := MsgQueue{
|
||||
ID: lastMsg.ID,
|
||||
Msg: lastMsg.RecvMsg,
|
||||
Conn: conn,
|
||||
}
|
||||
q.push2list(storeMsg)
|
||||
q.unFinMsg.Delete(conn)
|
||||
if len(msg) == lastID {
|
||||
return nil
|
||||
}
|
||||
msg = msg[lastID:]
|
||||
return q.parseMessage(msg, conn)
|
||||
}
|
||||
lastMsg.RecvMsg = bytesMerge(lastMsg.RecvMsg, msg)
|
||||
q.unFinMsg.Store(conn, lastMsg)
|
||||
return nil
|
||||
}
|
||||
}
|
||||
if len(msg) == 0 {
|
||||
return nil
|
||||
}
|
||||
var start int
|
||||
if start = searchHeader(msg); start == -1 {
|
||||
return errors.New("data format error")
|
||||
}
|
||||
msg = msg[start:]
|
||||
lastMsg := unFinMsg{}
|
||||
q.unFinMsg.Store(conn, &lastMsg)
|
||||
return q.parseMessage(msg, conn)
|
||||
}
|
||||
|
||||
func checkHeader(msg []byte) bool {
|
||||
if len(msg) != 8 {
|
||||
return false
|
||||
}
|
||||
for k, v := range msg {
|
||||
if v != header[k] {
|
||||
return false
|
||||
}
|
||||
}
|
||||
return true
|
||||
}
|
||||
|
||||
func searchHeader(msg []byte) int {
|
||||
if len(msg) < 8 {
|
||||
return 0
|
||||
}
|
||||
for k, v := range msg {
|
||||
find := 0
|
||||
if v == header[0] {
|
||||
for k2, v2 := range header {
|
||||
if msg[k+k2] == v2 {
|
||||
find++
|
||||
} else {
|
||||
break
|
||||
}
|
||||
}
|
||||
if find == 8 {
|
||||
return k
|
||||
}
|
||||
}
|
||||
}
|
||||
return -1
|
||||
}
|
||||
|
||||
func bytesMerge(src ...[]byte) []byte {
|
||||
var buff bytes.Buffer
|
||||
for _, v := range src {
|
||||
buff.Write(v)
|
||||
}
|
||||
return buff.Bytes()
|
||||
}
|
||||
|
||||
// Restore 获取收到的信息
|
||||
func (q *StarQueue) Restore() (MsgQueue, error) {
|
||||
if q.duration.Seconds() == 0 {
|
||||
q.duration = 86400 * time.Second
|
||||
}
|
||||
for {
|
||||
select {
|
||||
case <-q.ctx.Done():
|
||||
return MsgQueue{}, errors.New("Stoped By External Function Call")
|
||||
case <-time.After(q.duration):
|
||||
if q.duration != 0 {
|
||||
return MsgQueue{}, ErrDeadlineExceeded
|
||||
}
|
||||
case data, ok := <-q.msgPool:
|
||||
if !ok {
|
||||
return MsgQueue{}, os.ErrClosed
|
||||
}
|
||||
return data, nil
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// RestoreOne 获取收到的一个信息
|
||||
// 兼容性修改
|
||||
func (q *StarQueue) RestoreOne() (MsgQueue, error) {
|
||||
return q.Restore()
|
||||
}
|
||||
|
||||
// Stop 立即停止Restore
|
||||
func (q *StarQueue) Stop() {
|
||||
q.cancel()
|
||||
}
|
||||
|
||||
// RestoreDuration Restore最大超时时间
|
||||
func (q *StarQueue) RestoreDuration(tm time.Duration) {
|
||||
q.duration = tm
|
||||
}
|
||||
|
||||
func (q *StarQueue) RestoreChan() <-chan MsgQueue {
|
||||
return q.msgPool
|
||||
}
|
42
que_test.go
42
que_test.go
@ -1,42 +0,0 @@
|
||||
package starnet
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
func Test_QueSpeed(t *testing.T) {
|
||||
que := NewQueueWithCount(0)
|
||||
stop := make(chan struct{}, 1)
|
||||
que.RestoreDuration(time.Second * 10)
|
||||
var count int64
|
||||
go func() {
|
||||
for {
|
||||
select {
|
||||
case <-stop:
|
||||
//fmt.Println(count)
|
||||
return
|
||||
default:
|
||||
}
|
||||
_, err := que.RestoreOne()
|
||||
if err == nil {
|
||||
count++
|
||||
}
|
||||
}
|
||||
}()
|
||||
cp := 0
|
||||
stoped := time.After(time.Second * 10)
|
||||
data := que.BuildMessage([]byte("hello"))
|
||||
for {
|
||||
select {
|
||||
case <-stoped:
|
||||
fmt.Println(count, cp)
|
||||
stop <- struct{}{}
|
||||
return
|
||||
default:
|
||||
que.ParseMessage(data, "lala")
|
||||
cp++
|
||||
}
|
||||
}
|
||||
}
|
401
tlssniffer.go
Normal file
401
tlssniffer.go
Normal file
@ -0,0 +1,401 @@
|
||||
package starnet
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"context"
|
||||
"crypto/tls"
|
||||
"io"
|
||||
"net"
|
||||
"sync"
|
||||
"time"
|
||||
)
|
||||
|
||||
type myConn struct {
|
||||
reader io.Reader
|
||||
conn net.Conn
|
||||
isReadOnly bool
|
||||
multiReader io.Reader
|
||||
}
|
||||
|
||||
func (c *myConn) Read(p []byte) (int, error) {
|
||||
if c.isReadOnly {
|
||||
return c.reader.Read(p)
|
||||
}
|
||||
if c.multiReader == nil {
|
||||
c.multiReader = io.MultiReader(c.reader, c.conn)
|
||||
}
|
||||
return c.multiReader.Read(p)
|
||||
}
|
||||
|
||||
func (c *myConn) Write(p []byte) (int, error) {
|
||||
if c.isReadOnly {
|
||||
return 0, io.ErrClosedPipe
|
||||
}
|
||||
return c.conn.Write(p)
|
||||
}
|
||||
func (c *myConn) Close() error {
|
||||
if c.isReadOnly {
|
||||
return nil
|
||||
}
|
||||
return c.conn.Close()
|
||||
}
|
||||
func (c *myConn) LocalAddr() net.Addr {
|
||||
if c.isReadOnly {
|
||||
return nil
|
||||
}
|
||||
return c.conn.LocalAddr()
|
||||
}
|
||||
func (c *myConn) RemoteAddr() net.Addr {
|
||||
if c.isReadOnly {
|
||||
return nil
|
||||
}
|
||||
return c.conn.RemoteAddr()
|
||||
}
|
||||
func (c *myConn) SetDeadline(t time.Time) error {
|
||||
if c.isReadOnly {
|
||||
return nil
|
||||
}
|
||||
return c.conn.SetDeadline(t)
|
||||
}
|
||||
func (c *myConn) SetReadDeadline(t time.Time) error {
|
||||
if c.isReadOnly {
|
||||
return nil
|
||||
}
|
||||
return c.conn.SetReadDeadline(t)
|
||||
}
|
||||
func (c *myConn) SetWriteDeadline(t time.Time) error {
|
||||
if c.isReadOnly {
|
||||
return nil
|
||||
}
|
||||
return c.conn.SetWriteDeadline(t)
|
||||
}
|
||||
|
||||
type Listener struct {
|
||||
net.Listener
|
||||
cfg *tls.Config
|
||||
getConfigForClient func(hostname string) *tls.Config
|
||||
allowNonTls bool
|
||||
}
|
||||
|
||||
func (l *Listener) GetConfigForClient() func(hostname string) *tls.Config {
|
||||
return l.getConfigForClient
|
||||
}
|
||||
|
||||
func (l *Listener) SetConfigForClient(getConfigForClient func(hostname string) *tls.Config) {
|
||||
l.getConfigForClient = getConfigForClient
|
||||
}
|
||||
|
||||
func Listen(network, address string) (*Listener, error) {
|
||||
listener, err := net.Listen(network, address)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &Listener{Listener: listener}, nil
|
||||
}
|
||||
|
||||
func ListenTLSWithListenConfig(liscfg net.ListenConfig, network, address string, config *tls.Config, getConfigForClient func(hostname string) *tls.Config, allowNonTls bool) (*Listener, error) {
|
||||
listener, err := liscfg.Listen(context.Background(), network, address)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &Listener{
|
||||
Listener: listener,
|
||||
cfg: config,
|
||||
getConfigForClient: getConfigForClient,
|
||||
allowNonTls: allowNonTls,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func ListenWithListener(listener net.Listener, config *tls.Config, getConfigForClient func(hostname string) *tls.Config, allowNonTls bool) (*Listener, error) {
|
||||
return &Listener{
|
||||
Listener: listener,
|
||||
cfg: config,
|
||||
getConfigForClient: getConfigForClient,
|
||||
allowNonTls: allowNonTls,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func ListenTLSWithConfig(network, address string, config *tls.Config, getConfigForClient func(hostname string) *tls.Config, allowNonTls bool) (*Listener, error) {
|
||||
listener, err := net.Listen(network, address)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &Listener{
|
||||
Listener: listener,
|
||||
cfg: config,
|
||||
getConfigForClient: getConfigForClient,
|
||||
allowNonTls: allowNonTls,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func ListenTLS(network, address string, certFile, keyFile string, allowNonTls bool) (*Listener, error) {
|
||||
config, err := tls.LoadX509KeyPair(certFile, keyFile)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
tlsConfig := &tls.Config{
|
||||
Certificates: []tls.Certificate{config},
|
||||
}
|
||||
|
||||
listener, err := net.Listen(network, address)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return &Listener{
|
||||
Listener: listener,
|
||||
cfg: tlsConfig,
|
||||
allowNonTls: allowNonTls,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func (l *Listener) Accept() (net.Conn, error) {
|
||||
conn, err := l.Listener.Accept()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &Conn{
|
||||
Conn: conn,
|
||||
tlsCfg: l.cfg,
|
||||
getConfigForClient: l.getConfigForClient,
|
||||
allowNonTls: l.allowNonTls,
|
||||
}, nil
|
||||
}
|
||||
|
||||
type Conn struct {
|
||||
net.Conn
|
||||
once sync.Once
|
||||
initErr error
|
||||
isTLS bool
|
||||
tlsCfg *tls.Config
|
||||
tlsConn *tls.Conn
|
||||
buffer *bytes.Buffer
|
||||
noTlsReader io.Reader
|
||||
isOriginal bool
|
||||
getConfigForClient func(hostname string) *tls.Config
|
||||
hostname string
|
||||
allowNonTls bool
|
||||
}
|
||||
|
||||
func (c *Conn) Hostname() string {
|
||||
if c.hostname != "" {
|
||||
return c.hostname
|
||||
}
|
||||
if c.isTLS && c.tlsConn != nil {
|
||||
if c.tlsConn.ConnectionState().ServerName != "" {
|
||||
c.hostname = c.tlsConn.ConnectionState().ServerName
|
||||
return c.hostname
|
||||
}
|
||||
}
|
||||
return ""
|
||||
}
|
||||
|
||||
func (c *Conn) IsTLS() bool {
|
||||
return c.isTLS
|
||||
}
|
||||
|
||||
func (c *Conn) TlsConn() *tls.Conn {
|
||||
return c.tlsConn
|
||||
}
|
||||
|
||||
func (c *Conn) isTLSConnection() (bool, error) {
|
||||
if c.getConfigForClient == nil {
|
||||
peek := make([]byte, 5)
|
||||
n, err := io.ReadFull(c.Conn, peek)
|
||||
if err != nil {
|
||||
return false, err
|
||||
}
|
||||
|
||||
isTLS := n >= 3 && peek[0] == 0x16 && peek[1] == 0x03
|
||||
|
||||
c.buffer = bytes.NewBuffer(peek[:n])
|
||||
return isTLS, nil
|
||||
}
|
||||
|
||||
c.buffer = new(bytes.Buffer)
|
||||
r := io.TeeReader(c.Conn, c.buffer)
|
||||
var hello *tls.ClientHelloInfo
|
||||
tls.Server(&myConn{reader: r, isReadOnly: true}, &tls.Config{
|
||||
GetConfigForClient: func(argHello *tls.ClientHelloInfo) (*tls.Config, error) {
|
||||
hello = new(tls.ClientHelloInfo)
|
||||
*hello = *argHello
|
||||
return nil, nil
|
||||
},
|
||||
}).Handshake()
|
||||
peek := c.buffer.Bytes()
|
||||
n := len(peek)
|
||||
isTLS := n >= 3 && peek[0] == 0x16 && peek[1] == 0x03
|
||||
if hello == nil {
|
||||
return isTLS, nil
|
||||
}
|
||||
c.hostname = hello.ServerName
|
||||
if c.hostname == "" {
|
||||
c.hostname, _, _ = net.SplitHostPort(c.Conn.LocalAddr().String())
|
||||
}
|
||||
return isTLS, nil
|
||||
}
|
||||
|
||||
func (c *Conn) init() {
|
||||
c.once.Do(func() {
|
||||
if c.isOriginal {
|
||||
return
|
||||
}
|
||||
if c.tlsCfg != nil {
|
||||
isTLS, err := c.isTLSConnection()
|
||||
if err != nil {
|
||||
c.initErr = err
|
||||
return
|
||||
}
|
||||
c.isTLS = isTLS
|
||||
}
|
||||
|
||||
if c.isTLS {
|
||||
var cfg = c.tlsCfg
|
||||
if c.getConfigForClient != nil {
|
||||
cfg = c.getConfigForClient(c.hostname)
|
||||
if cfg == nil {
|
||||
cfg = c.tlsCfg
|
||||
}
|
||||
}
|
||||
c.tlsConn = tls.Server(&myConn{
|
||||
reader: c.buffer,
|
||||
conn: c.Conn,
|
||||
isReadOnly: false,
|
||||
}, cfg)
|
||||
} else {
|
||||
if !c.allowNonTls {
|
||||
c.initErr = net.ErrClosed
|
||||
return
|
||||
}
|
||||
c.noTlsReader = io.MultiReader(c.buffer, c.Conn)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func (c *Conn) Read(b []byte) (int, error) {
|
||||
c.init()
|
||||
if c.initErr != nil {
|
||||
return 0, c.initErr
|
||||
}
|
||||
if c.isTLS {
|
||||
return c.tlsConn.Read(b)
|
||||
}
|
||||
return c.noTlsReader.Read(b)
|
||||
}
|
||||
|
||||
func (c *Conn) Write(b []byte) (int, error) {
|
||||
c.init()
|
||||
if c.initErr != nil {
|
||||
return 0, c.initErr
|
||||
}
|
||||
|
||||
if c.isTLS {
|
||||
return c.tlsConn.Write(b)
|
||||
}
|
||||
return c.Conn.Write(b)
|
||||
}
|
||||
|
||||
func (c *Conn) Close() error {
|
||||
if c.isTLS && c.tlsConn != nil {
|
||||
return c.tlsConn.Close()
|
||||
}
|
||||
return c.Conn.Close()
|
||||
}
|
||||
|
||||
func (c *Conn) SetDeadline(t time.Time) error {
|
||||
if c.isTLS && c.tlsConn != nil {
|
||||
return c.tlsConn.SetDeadline(t)
|
||||
}
|
||||
return c.Conn.SetDeadline(t)
|
||||
}
|
||||
|
||||
func (c *Conn) SetReadDeadline(t time.Time) error {
|
||||
if c.isTLS && c.tlsConn != nil {
|
||||
return c.tlsConn.SetReadDeadline(t)
|
||||
}
|
||||
return c.Conn.SetReadDeadline(t)
|
||||
}
|
||||
|
||||
func (c *Conn) SetWriteDeadline(t time.Time) error {
|
||||
if c.isTLS && c.tlsConn != nil {
|
||||
return c.tlsConn.SetWriteDeadline(t)
|
||||
}
|
||||
return c.Conn.SetWriteDeadline(t)
|
||||
}
|
||||
|
||||
func (c *Conn) TlsConnection() (*tls.Conn, error) {
|
||||
if c.initErr != nil {
|
||||
return nil, c.initErr
|
||||
}
|
||||
if !c.isTLS {
|
||||
return nil, net.ErrClosed
|
||||
}
|
||||
return c.tlsConn, nil
|
||||
}
|
||||
|
||||
func (c *Conn) OriginalConn() net.Conn {
|
||||
return c.Conn
|
||||
}
|
||||
|
||||
func NewClientTlsConn(conn net.Conn, cfg *tls.Config) (*Conn, error) {
|
||||
if conn == nil {
|
||||
return nil, net.ErrClosed
|
||||
}
|
||||
c := &Conn{
|
||||
Conn: conn,
|
||||
isTLS: true,
|
||||
tlsCfg: cfg,
|
||||
tlsConn: tls.Client(conn, cfg),
|
||||
isOriginal: true,
|
||||
}
|
||||
return c, nil
|
||||
}
|
||||
|
||||
func NewServerTlsConn(conn net.Conn, cfg *tls.Config) (*Conn, error) {
|
||||
if conn == nil {
|
||||
return nil, net.ErrClosed
|
||||
}
|
||||
c := &Conn{
|
||||
Conn: conn,
|
||||
isTLS: true,
|
||||
tlsCfg: cfg,
|
||||
tlsConn: tls.Server(conn, cfg),
|
||||
isOriginal: true,
|
||||
}
|
||||
c.init()
|
||||
return c, nil
|
||||
}
|
||||
|
||||
func Dial(network, address string) (*Conn, error) {
|
||||
conn, err := net.Dial(network, address)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &Conn{
|
||||
Conn: conn,
|
||||
isTLS: false,
|
||||
tlsCfg: nil,
|
||||
tlsConn: nil,
|
||||
noTlsReader: conn,
|
||||
isOriginal: true,
|
||||
}, nil
|
||||
}
|
||||
|
||||
func DialTLS(network, address string, certFile, keyFile string) (*Conn, error) {
|
||||
config, err := tls.LoadX509KeyPair(certFile, keyFile)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
tlsConfig := &tls.Config{
|
||||
Certificates: []tls.Certificate{config},
|
||||
}
|
||||
|
||||
conn, err := net.Dial(network, address)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return NewClientTlsConn(conn, tlsConfig)
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user