| 
									
										
										
										
											2017-11-17 14:18:46 +02:00
										 |  |  | // Copyright 2017 The go-ethereum Authors | 
					
						
							|  |  |  | // This file is part of the go-ethereum library. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							|  |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-17 04:07:11 -08:00
										 |  |  | package rpc | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"net/http" | 
					
						
							|  |  |  | 	"net/http/httptest" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-28 04:43:21 -04:00
										 |  |  | func confirmStatusCode(t *testing.T, got, want int) { | 
					
						
							|  |  |  | 	t.Helper() | 
					
						
							|  |  |  | 	if got == want { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if gotName := http.StatusText(got); len(gotName) > 0 { | 
					
						
							|  |  |  | 		if wantName := http.StatusText(want); len(wantName) > 0 { | 
					
						
							|  |  |  | 			t.Fatalf("response status code: got %d (%s), want %d (%s)", got, gotName, want, wantName) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	t.Fatalf("response status code: got %d, want %d", got, want) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func confirmRequestValidationCode(t *testing.T, method, contentType, body string, expectedStatusCode int) { | 
					
						
							|  |  |  | 	t.Helper() | 
					
						
							|  |  |  | 	request := httptest.NewRequest(method, "http://url.com", strings.NewReader(body)) | 
					
						
							|  |  |  | 	if len(contentType) > 0 { | 
					
						
							|  |  |  | 		request.Header.Set("Content-Type", contentType) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	code, err := validateRequest(request) | 
					
						
							|  |  |  | 	if code == 0 { | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			t.Errorf("validation: got error %v, expected nil", err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else if err == nil { | 
					
						
							|  |  |  | 		t.Errorf("validation: code %d: got nil, expected error", code) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	confirmStatusCode(t, code, expectedStatusCode) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-17 04:07:11 -08:00
										 |  |  | func TestHTTPErrorResponseWithDelete(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2020-04-28 04:43:21 -04:00
										 |  |  | 	confirmRequestValidationCode(t, http.MethodDelete, contentType, "", http.StatusMethodNotAllowed) | 
					
						
							| 
									
										
										
										
											2017-11-17 04:07:11 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestHTTPErrorResponseWithPut(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2020-04-28 04:43:21 -04:00
										 |  |  | 	confirmRequestValidationCode(t, http.MethodPut, contentType, "", http.StatusMethodNotAllowed) | 
					
						
							| 
									
										
										
										
											2017-11-17 04:07:11 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestHTTPErrorResponseWithMaxContentLength(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2018-03-13 13:23:44 +02:00
										 |  |  | 	body := make([]rune, maxRequestContentLength+1) | 
					
						
							| 
									
										
										
										
											2020-04-28 04:43:21 -04:00
										 |  |  | 	confirmRequestValidationCode(t, | 
					
						
							| 
									
										
										
										
											2017-12-12 21:12:32 +03:00
										 |  |  | 		http.MethodPost, contentType, string(body), http.StatusRequestEntityTooLarge) | 
					
						
							| 
									
										
										
										
											2017-11-17 04:07:11 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestHTTPErrorResponseWithEmptyContentType(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2020-04-28 04:43:21 -04:00
										 |  |  | 	confirmRequestValidationCode(t, http.MethodPost, "", "", http.StatusUnsupportedMediaType) | 
					
						
							| 
									
										
										
										
											2017-11-17 04:07:11 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestHTTPErrorResponseWithValidRequest(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2020-04-28 04:43:21 -04:00
										 |  |  | 	confirmRequestValidationCode(t, http.MethodPost, contentType, "", 0) | 
					
						
							| 
									
										
										
										
											2017-11-17 04:07:11 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-28 04:43:21 -04:00
										 |  |  | func confirmHTTPRequestYieldsStatusCode(t *testing.T, method, contentType, body string, expectedStatusCode int) { | 
					
						
							|  |  |  | 	t.Helper() | 
					
						
							|  |  |  | 	s := Server{} | 
					
						
							|  |  |  | 	ts := httptest.NewServer(&s) | 
					
						
							|  |  |  | 	defer ts.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	request, err := http.NewRequest(method, ts.URL, strings.NewReader(body)) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("failed to create a valid HTTP request: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(contentType) > 0 { | 
					
						
							|  |  |  | 		request.Header.Set("Content-Type", contentType) | 
					
						
							| 
									
										
										
										
											2017-11-17 04:07:11 -08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-04-28 04:43:21 -04:00
										 |  |  | 	resp, err := http.DefaultClient.Do(request) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatalf("request failed: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	confirmStatusCode(t, resp.StatusCode, expectedStatusCode) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestHTTPResponseWithEmptyGet(t *testing.T) { | 
					
						
							|  |  |  | 	confirmHTTPRequestYieldsStatusCode(t, http.MethodGet, "", "", http.StatusOK) | 
					
						
							| 
									
										
										
										
											2017-11-17 04:07:11 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-02-26 13:40:35 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // This checks that maxRequestContentLength is not applied to the response of a request. | 
					
						
							|  |  |  | func TestHTTPRespBodyUnlimited(t *testing.T) { | 
					
						
							|  |  |  | 	const respLength = maxRequestContentLength * 3 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s := NewServer() | 
					
						
							|  |  |  | 	defer s.Stop() | 
					
						
							|  |  |  | 	s.RegisterName("test", largeRespService{respLength}) | 
					
						
							|  |  |  | 	ts := httptest.NewServer(s) | 
					
						
							|  |  |  | 	defer ts.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c, err := DialHTTP(ts.URL) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer c.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var r string | 
					
						
							|  |  |  | 	if err := c.Call(&r, "test_largeResp"); err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(r) != respLength { | 
					
						
							|  |  |  | 		t.Fatalf("response has wrong length %d, want %d", len(r), respLength) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-21 14:51:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Tests that an HTTP error results in an HTTPError instance | 
					
						
							|  |  |  | // being returned with the expected attributes. | 
					
						
							|  |  |  | func TestHTTPErrorResponse(t *testing.T) { | 
					
						
							|  |  |  | 	ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { | 
					
						
							|  |  |  | 		http.Error(w, "error has occurred!", http.StatusTeapot) | 
					
						
							|  |  |  | 	})) | 
					
						
							|  |  |  | 	defer ts.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c, err := DialHTTP(ts.URL) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var r string | 
					
						
							|  |  |  | 	err = c.Call(&r, "test_method") | 
					
						
							|  |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		t.Fatal("error was expected") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	httpErr, ok := err.(HTTPError) | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		t.Fatalf("unexpected error type %T", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if httpErr.StatusCode != http.StatusTeapot { | 
					
						
							|  |  |  | 		t.Error("unexpected status code", httpErr.StatusCode) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if httpErr.Status != "418 I'm a teapot" { | 
					
						
							|  |  |  | 		t.Error("unexpected status text", httpErr.Status) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if body := string(httpErr.Body); body != "error has occurred!\n" { | 
					
						
							|  |  |  | 		t.Error("unexpected body", body) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if errMsg := httpErr.Error(); errMsg != "418 I'm a teapot: error has occurred!\n" { | 
					
						
							|  |  |  | 		t.Error("unexpected error message", errMsg) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |