package bytecounter

import (
	"errors"
	"testing"

	"github.com/ooni/probe-cli/v3/internal/model/mocks"
)

func TestWrappedConnWorksOnSuccess(t *testing.T) {
	counter := New()
	underlying := &mocks.Conn{
		MockRead: func(b []byte) (int, error) {
			return 10, nil
		},
		MockWrite: func(b []byte) (int, error) {
			return 4, nil
		},
	}
	conn := &wrappedConn{
		Conn:    underlying,
		Counter: counter,
	}
	if _, err := conn.Read(make([]byte, 128)); err != nil {
		t.Fatal(err)
	}
	if _, err := conn.Write(make([]byte, 1024)); err != nil {
		t.Fatal(err)
	}
	if counter.BytesReceived() != 10 {
		t.Fatal("unexpected number of bytes received")
	}
	if counter.BytesSent() != 4 {
		t.Fatal("unexpected number of bytes sent")
	}
}

func TestWrappedConnWorksOnFailure(t *testing.T) {
	readError := errors.New("read error")
	writeError := errors.New("write error")
	counter := New()
	underlying := &mocks.Conn{
		MockRead: func(b []byte) (int, error) {
			return 0, readError
		},
		MockWrite: func(b []byte) (int, error) {
			return 0, writeError
		},
	}
	conn := &wrappedConn{
		Conn:    underlying,
		Counter: counter,
	}
	if _, err := conn.Read(make([]byte, 128)); !errors.Is(err, readError) {
		t.Fatal("not the error we expected", err)
	}
	if _, err := conn.Write(make([]byte, 1024)); !errors.Is(err, writeError) {
		t.Fatal("not the error we expected", err)
	}
	if counter.BytesReceived() != 0 {
		t.Fatal("unexpected number of bytes received")
	}
	if counter.BytesSent() != 0 {
		t.Fatal("unexpected number of bytes sent")
	}
}

func TestWrapConn(t *testing.T) {
	conn := &mocks.Conn{}
	counter := New()
	nconn := WrapConn(conn, counter)
	_, good := nconn.(*wrappedConn)
	if !good {
		t.Fatal("did not wrap")
	}
}

func TestMaybeWrapConn(t *testing.T) {
	t.Run("with nil counter", func(t *testing.T) {
		conn := &mocks.Conn{}
		nconn := MaybeWrapConn(conn, nil)
		_, good := nconn.(*mocks.Conn)
		if !good {
			t.Fatal("did not wrap")
		}
	})

	t.Run("with legit counter", func(t *testing.T) {
		conn := &mocks.Conn{}
		counter := New()
		nconn := MaybeWrapConn(conn, counter)
		_, good := nconn.(*wrappedConn)
		if !good {
			t.Fatal("did not wrap")
		}
	})
}