2022-08-29 14:25:30 +00:00
|
|
|
// stm: #unit
|
2020-08-14 14:06:53 +00:00
|
|
|
package rpcenc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-03-07 15:48:09 +00:00
|
|
|
"fmt"
|
2020-08-14 14:06:53 +00:00
|
|
|
"io"
|
|
|
|
"net/http/httptest"
|
|
|
|
"strings"
|
2023-01-25 15:45:35 +00:00
|
|
|
"sync"
|
2020-08-14 14:06:53 +00:00
|
|
|
"testing"
|
2023-01-25 15:45:35 +00:00
|
|
|
"time"
|
2020-08-14 14:06:53 +00:00
|
|
|
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/stretchr/testify/require"
|
2021-07-30 11:03:31 +00:00
|
|
|
"golang.org/x/xerrors"
|
2020-08-14 14:06:53 +00:00
|
|
|
|
|
|
|
"github.com/filecoin-project/go-jsonrpc"
|
2022-04-19 17:14:15 +00:00
|
|
|
|
2022-06-14 17:41:59 +00:00
|
|
|
"github.com/filecoin-project/lotus/storage/pipeline/lib/nullreader"
|
2020-08-14 14:06:53 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type ReaderHandler struct {
|
2021-07-30 10:58:28 +00:00
|
|
|
readApi func(ctx context.Context, r io.Reader) ([]byte, error)
|
2023-01-25 15:45:35 +00:00
|
|
|
|
|
|
|
cont chan struct{}
|
|
|
|
subErr error
|
2021-07-30 10:58:28 +00:00
|
|
|
}
|
|
|
|
|
2021-07-30 11:27:51 +00:00
|
|
|
func (h *ReaderHandler) ReadAllApi(ctx context.Context, r io.Reader, mustRedir bool) ([]byte, error) {
|
|
|
|
if mustRedir {
|
|
|
|
if err := r.(*RpcReader).MustRedirect(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2021-07-30 10:58:28 +00:00
|
|
|
return h.readApi(ctx, r)
|
2020-08-14 14:06:53 +00:00
|
|
|
}
|
|
|
|
|
2023-01-25 15:45:35 +00:00
|
|
|
func (h *ReaderHandler) ReadAllWaiting(ctx context.Context, r io.Reader, mustRedir bool) ([]byte, error) {
|
|
|
|
if mustRedir {
|
|
|
|
if err := r.(*RpcReader).MustRedirect(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
h.cont <- struct{}{}
|
|
|
|
<-h.cont
|
|
|
|
|
|
|
|
var m []byte
|
|
|
|
m, h.subErr = h.readApi(ctx, r)
|
|
|
|
|
|
|
|
h.cont <- struct{}{}
|
|
|
|
|
|
|
|
return m, h.subErr
|
|
|
|
}
|
|
|
|
|
2021-07-30 11:27:51 +00:00
|
|
|
func (h *ReaderHandler) ReadStartAndApi(ctx context.Context, r io.Reader, mustRedir bool) ([]byte, error) {
|
|
|
|
if mustRedir {
|
|
|
|
if err := r.(*RpcReader).MustRedirect(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-30 11:03:31 +00:00
|
|
|
n, err := r.Read([]byte{0})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if n != 1 {
|
|
|
|
return nil, xerrors.Errorf("not one")
|
|
|
|
}
|
|
|
|
|
|
|
|
return h.readApi(ctx, r)
|
|
|
|
}
|
|
|
|
|
2021-07-30 11:27:51 +00:00
|
|
|
func (h *ReaderHandler) CloseReader(ctx context.Context, r io.Reader) error {
|
|
|
|
return r.(io.Closer).Close()
|
|
|
|
}
|
|
|
|
|
2020-08-14 14:06:53 +00:00
|
|
|
func (h *ReaderHandler) ReadAll(ctx context.Context, r io.Reader) ([]byte, error) {
|
2023-03-07 15:48:09 +00:00
|
|
|
b, err := io.ReadAll(r)
|
|
|
|
if err != nil {
|
|
|
|
return nil, xerrors.Errorf("readall: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return b, nil
|
2020-08-14 14:06:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (h *ReaderHandler) ReadNullLen(ctx context.Context, r io.Reader) (int64, error) {
|
2022-04-19 17:14:15 +00:00
|
|
|
return r.(*nullreader.NullReader).N, nil
|
2020-08-14 14:06:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (h *ReaderHandler) ReadUrl(ctx context.Context, u string) (string, error) {
|
|
|
|
return u, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestReaderProxy(t *testing.T) {
|
|
|
|
var client struct {
|
|
|
|
ReadAll func(ctx context.Context, r io.Reader) ([]byte, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
serverHandler := &ReaderHandler{}
|
|
|
|
|
|
|
|
readerHandler, readerServerOpt := ReaderParamDecoder()
|
|
|
|
rpcServer := jsonrpc.NewServer(readerServerOpt)
|
|
|
|
rpcServer.Register("ReaderHandler", serverHandler)
|
|
|
|
|
|
|
|
mux := mux.NewRouter()
|
|
|
|
mux.Handle("/rpc/v0", rpcServer)
|
|
|
|
mux.Handle("/rpc/streams/v0/push/{uuid}", readerHandler)
|
|
|
|
|
|
|
|
testServ := httptest.NewServer(mux)
|
|
|
|
defer testServ.Close()
|
|
|
|
|
2020-08-14 21:40:41 +00:00
|
|
|
re := ReaderParamEncoder("http://" + testServ.Listener.Addr().String() + "/rpc/streams/v0/push")
|
2020-08-22 03:22:40 +00:00
|
|
|
closer, err := jsonrpc.NewMergeClient(context.Background(), "ws://"+testServ.Listener.Addr().String()+"/rpc/v0", "ReaderHandler", []interface{}{&client}, nil, re)
|
2020-08-14 14:06:53 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer closer()
|
|
|
|
|
|
|
|
read, err := client.ReadAll(context.TODO(), strings.NewReader("pooooootato"))
|
|
|
|
require.NoError(t, err)
|
2020-08-20 04:49:10 +00:00
|
|
|
require.Equal(t, "pooooootato", string(read), "potatoes weren't equal")
|
2020-08-14 14:06:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestNullReaderProxy(t *testing.T) {
|
|
|
|
var client struct {
|
2020-08-14 21:40:41 +00:00
|
|
|
ReadAll func(ctx context.Context, r io.Reader) ([]byte, error)
|
2020-08-14 14:06:53 +00:00
|
|
|
ReadNullLen func(ctx context.Context, r io.Reader) (int64, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
serverHandler := &ReaderHandler{}
|
|
|
|
|
|
|
|
readerHandler, readerServerOpt := ReaderParamDecoder()
|
|
|
|
rpcServer := jsonrpc.NewServer(readerServerOpt)
|
|
|
|
rpcServer.Register("ReaderHandler", serverHandler)
|
|
|
|
|
|
|
|
mux := mux.NewRouter()
|
|
|
|
mux.Handle("/rpc/v0", rpcServer)
|
|
|
|
mux.Handle("/rpc/streams/v0/push/{uuid}", readerHandler)
|
|
|
|
|
|
|
|
testServ := httptest.NewServer(mux)
|
|
|
|
defer testServ.Close()
|
|
|
|
|
2020-08-14 21:40:41 +00:00
|
|
|
re := ReaderParamEncoder("http://" + testServ.Listener.Addr().String() + "/rpc/streams/v0/push")
|
2020-08-22 03:22:40 +00:00
|
|
|
closer, err := jsonrpc.NewMergeClient(context.Background(), "ws://"+testServ.Listener.Addr().String()+"/rpc/v0", "ReaderHandler", []interface{}{&client}, nil, re)
|
2020-08-14 14:06:53 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer closer()
|
|
|
|
|
2022-04-19 17:14:15 +00:00
|
|
|
n, err := client.ReadNullLen(context.TODO(), nullreader.NewNullReader(1016))
|
2020-08-14 14:06:53 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, int64(1016), n)
|
|
|
|
}
|
2021-07-30 10:58:28 +00:00
|
|
|
|
|
|
|
func TestReaderRedirect(t *testing.T) {
|
|
|
|
var allClient struct {
|
|
|
|
ReadAll func(ctx context.Context, r io.Reader) ([]byte, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
allServerHandler := &ReaderHandler{}
|
|
|
|
readerHandler, readerServerOpt := ReaderParamDecoder()
|
|
|
|
rpcServer := jsonrpc.NewServer(readerServerOpt)
|
|
|
|
rpcServer.Register("ReaderHandler", allServerHandler)
|
|
|
|
|
|
|
|
mux := mux.NewRouter()
|
|
|
|
mux.Handle("/rpc/v0", rpcServer)
|
|
|
|
mux.Handle("/rpc/streams/v0/push/{uuid}", readerHandler)
|
|
|
|
|
|
|
|
testServ := httptest.NewServer(mux)
|
|
|
|
defer testServ.Close()
|
|
|
|
|
|
|
|
re := ReaderParamEncoder("http://" + testServ.Listener.Addr().String() + "/rpc/streams/v0/push")
|
|
|
|
closer, err := jsonrpc.NewMergeClient(context.Background(), "ws://"+testServ.Listener.Addr().String()+"/rpc/v0", "ReaderHandler", []interface{}{&allClient}, nil, re)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer closer()
|
|
|
|
}
|
|
|
|
|
|
|
|
var redirClient struct {
|
2021-07-30 11:27:51 +00:00
|
|
|
ReadAllApi func(ctx context.Context, r io.Reader, mustRedir bool) ([]byte, error)
|
|
|
|
ReadStartAndApi func(ctx context.Context, r io.Reader, mustRedir bool) ([]byte, error)
|
|
|
|
CloseReader func(ctx context.Context, r io.Reader) error
|
2021-07-30 10:58:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
allServerHandler := &ReaderHandler{readApi: allClient.ReadAll}
|
|
|
|
readerHandler, readerServerOpt := ReaderParamDecoder()
|
|
|
|
rpcServer := jsonrpc.NewServer(readerServerOpt)
|
|
|
|
rpcServer.Register("ReaderHandler", allServerHandler)
|
|
|
|
|
|
|
|
mux := mux.NewRouter()
|
|
|
|
mux.Handle("/rpc/v0", rpcServer)
|
|
|
|
mux.Handle("/rpc/streams/v0/push/{uuid}", readerHandler)
|
|
|
|
|
|
|
|
testServ := httptest.NewServer(mux)
|
|
|
|
defer testServ.Close()
|
|
|
|
|
|
|
|
re := ReaderParamEncoder("http://" + testServ.Listener.Addr().String() + "/rpc/streams/v0/push")
|
|
|
|
closer, err := jsonrpc.NewMergeClient(context.Background(), "ws://"+testServ.Listener.Addr().String()+"/rpc/v0", "ReaderHandler", []interface{}{&redirClient}, nil, re)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer closer()
|
|
|
|
}
|
|
|
|
|
2021-07-30 11:03:31 +00:00
|
|
|
// redirect
|
2021-07-30 11:27:51 +00:00
|
|
|
read, err := redirClient.ReadAllApi(context.TODO(), strings.NewReader("rediracted pooooootato"), true)
|
2021-07-30 10:58:28 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, "rediracted pooooootato", string(read), "potatoes weren't equal")
|
2021-07-30 11:03:31 +00:00
|
|
|
|
|
|
|
// proxy (because we started reading locally)
|
2021-07-30 11:27:51 +00:00
|
|
|
read, err = redirClient.ReadStartAndApi(context.TODO(), strings.NewReader("rediracted pooooootato"), false)
|
2021-07-30 11:03:31 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, "ediracted pooooootato", string(read), "otatoes weren't equal")
|
2021-07-30 11:27:51 +00:00
|
|
|
|
|
|
|
// check mustredir check; proxy (because we started reading locally)
|
|
|
|
read, err = redirClient.ReadStartAndApi(context.TODO(), strings.NewReader("rediracted pooooootato"), true)
|
|
|
|
require.Error(t, err)
|
|
|
|
require.Contains(t, err.Error(), ErrMustRedirect.Error())
|
|
|
|
require.Empty(t, read)
|
|
|
|
|
|
|
|
err = redirClient.CloseReader(context.TODO(), strings.NewReader("rediracted pooooootato"))
|
|
|
|
require.NoError(t, err)
|
2021-07-30 10:58:28 +00:00
|
|
|
}
|
2023-01-25 15:45:35 +00:00
|
|
|
|
|
|
|
func TestReaderRedirectDrop(t *testing.T) {
|
2023-03-07 15:48:09 +00:00
|
|
|
for i := 0; i < 10; i++ {
|
|
|
|
t.Run(fmt.Sprintf("test %d", i), testReaderRedirectDrop)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func testReaderRedirectDrop(t *testing.T) {
|
2023-01-25 15:45:35 +00:00
|
|
|
// lower timeout so that the dangling connection between client and reader is dropped quickly
|
|
|
|
// after the test. Otherwise httptest.Close is blocked.
|
2023-03-07 15:48:09 +00:00
|
|
|
Timeout = 90 * time.Millisecond
|
2023-01-25 15:45:35 +00:00
|
|
|
|
|
|
|
var allClient struct {
|
|
|
|
ReadAll func(ctx context.Context, r io.Reader) ([]byte, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
allServerHandler := &ReaderHandler{}
|
|
|
|
readerHandler, readerServerOpt := ReaderParamDecoder()
|
|
|
|
rpcServer := jsonrpc.NewServer(readerServerOpt)
|
|
|
|
rpcServer.Register("ReaderHandler", allServerHandler)
|
|
|
|
|
|
|
|
mux := mux.NewRouter()
|
|
|
|
mux.Handle("/rpc/v0", rpcServer)
|
|
|
|
mux.Handle("/rpc/streams/v0/push/{uuid}", readerHandler)
|
|
|
|
|
|
|
|
testServ := httptest.NewServer(mux)
|
|
|
|
defer testServ.Close()
|
|
|
|
t.Logf("test server reading: %s", testServ.URL)
|
|
|
|
|
|
|
|
re := ReaderParamEncoder("http://" + testServ.Listener.Addr().String() + "/rpc/streams/v0/push")
|
|
|
|
closer, err := jsonrpc.NewMergeClient(context.Background(), "ws://"+testServ.Listener.Addr().String()+"/rpc/v0", "ReaderHandler", []interface{}{&allClient}, nil, re)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer closer()
|
|
|
|
}
|
|
|
|
|
|
|
|
var redirClient struct {
|
|
|
|
ReadAllWaiting func(ctx context.Context, r io.Reader, mustRedir bool) ([]byte, error)
|
|
|
|
}
|
|
|
|
contCh := make(chan struct{})
|
|
|
|
|
|
|
|
allServerHandler := &ReaderHandler{readApi: allClient.ReadAll, cont: contCh}
|
|
|
|
readerHandler, readerServerOpt := ReaderParamDecoder()
|
|
|
|
rpcServer := jsonrpc.NewServer(readerServerOpt)
|
|
|
|
rpcServer.Register("ReaderHandler", allServerHandler)
|
|
|
|
|
|
|
|
mux := mux.NewRouter()
|
|
|
|
mux.Handle("/rpc/v0", rpcServer)
|
|
|
|
mux.Handle("/rpc/streams/v0/push/{uuid}", readerHandler)
|
|
|
|
|
|
|
|
testServ := httptest.NewServer(mux)
|
|
|
|
defer testServ.Close()
|
|
|
|
t.Logf("test server redirecting: %s", testServ.URL)
|
|
|
|
|
|
|
|
re := ReaderParamEncoder("http://" + testServ.Listener.Addr().String() + "/rpc/streams/v0/push")
|
|
|
|
closer, err := jsonrpc.NewMergeClient(context.Background(), "http://"+testServ.Listener.Addr().String()+"/rpc/v0", "ReaderHandler", []interface{}{&redirClient}, nil, re)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer closer()
|
|
|
|
|
|
|
|
var done sync.WaitGroup
|
|
|
|
|
|
|
|
// Happy case
|
|
|
|
|
|
|
|
done.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer done.Done()
|
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
read, err := redirClient.ReadAllWaiting(ctx, strings.NewReader("rediracted pooooootato"), true)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, "rediracted pooooootato", string(read), "potatoes weren't equal")
|
|
|
|
}()
|
|
|
|
|
|
|
|
<-contCh // exec enter ReadAllWaiting
|
|
|
|
contCh <- struct{}{} // stert subcall
|
|
|
|
<-contCh // wait for subcall to finish
|
|
|
|
|
|
|
|
done.Wait()
|
|
|
|
|
2023-03-07 15:48:09 +00:00
|
|
|
fmt.Println("---------------------")
|
|
|
|
|
2023-01-25 15:45:35 +00:00
|
|
|
// Redir client drops before subcall
|
|
|
|
done.Add(1)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
defer done.Done()
|
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
_, err := redirClient.ReadAllWaiting(ctx, strings.NewReader("rediracted pooooootato"), true)
|
|
|
|
require.ErrorContains(t, err, "sendRequest failed")
|
|
|
|
}()
|
|
|
|
|
|
|
|
// wait for execution to enter ReadAllWaiting
|
|
|
|
<-contCh
|
|
|
|
|
|
|
|
// kill redirecting server connection
|
|
|
|
testServ.CloseClientConnections()
|
|
|
|
|
|
|
|
// ReadAllWaiting should fail
|
|
|
|
done.Wait()
|
|
|
|
|
|
|
|
// resume execution in ReadAllWaiting, calling redicect
|
|
|
|
contCh <- struct{}{}
|
|
|
|
|
|
|
|
// wait for subcall to finish
|
|
|
|
<-contCh
|
|
|
|
|
2023-03-07 15:48:09 +00:00
|
|
|
estr := allServerHandler.subErr.Error()
|
|
|
|
|
|
|
|
require.True(t,
|
|
|
|
strings.Contains(estr, "decoding params for 'ReaderHandler.ReadAll' (param: 0; custom decoder): context canceled") ||
|
|
|
|
strings.Contains(estr, "readall: unexpected EOF"), "unexpected error: %s", estr)
|
2023-01-25 15:45:35 +00:00
|
|
|
}
|