Compare commits
1 commit
a4ca608937
...
568aaa9f8c
Author | SHA1 | Date | |
---|---|---|---|
568aaa9f8c |
5 changed files with 25 additions and 155 deletions
|
@ -15,7 +15,7 @@ type Notifier struct {
|
||||||
cancel context.CancelFunc
|
cancel context.CancelFunc
|
||||||
tgBot *bot.ConcreteBot
|
tgBot *bot.ConcreteBot
|
||||||
notifChannel int64
|
notifChannel int64
|
||||||
ipGetter *ip.IPGetter
|
ipGetter ip.Getter
|
||||||
}
|
}
|
||||||
|
|
||||||
func (n *Notifier) SendInitMessage() error {
|
func (n *Notifier) SendInitMessage() error {
|
||||||
|
|
|
@ -7,34 +7,29 @@ import (
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const ifconfigURL = "https://ifconfig.me"
|
const ifconfigURL = "https://ifconfig.me"
|
||||||
const httpMaxRead = 100
|
const httpMaxRead = 100
|
||||||
const httpTimeout = 10 * time.Second
|
|
||||||
|
|
||||||
type IPGetter struct {
|
// TODO: use a struct, but a baseHTTPClient inside it to perform unit tests
|
||||||
httpClt *http.Client
|
type Getter interface {
|
||||||
remoteAddress string
|
GetCurrentPublicIP(ctx context.Context) (net.IP, error)
|
||||||
timeout time.Duration
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *IPGetter) GetCurrentPublicIP(ctx context.Context) (net.IP, error) {
|
type concreteIPGetter struct {
|
||||||
reqCtx, cancel := context.WithTimeout(ctx, c.timeout)
|
}
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
req, err := http.NewRequestWithContext(reqCtx, http.MethodGet, c.remoteAddress, nil)
|
func (c *concreteIPGetter) GetCurrentPublicIP(ctx context.Context) (net.IP, error) {
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, ifconfigURL, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to prepare public IP request: %w", err)
|
return nil, fmt.Errorf("failed to prepare public IP request: %w", err)
|
||||||
}
|
}
|
||||||
resp, err := c.httpClt.Do(req)
|
resp, err := http.DefaultClient.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to get current IP from ifconfig: %w", err)
|
return nil, fmt.Errorf("failed to get current IP from ifconfig: %w", err)
|
||||||
}
|
}
|
||||||
if resp.StatusCode != http.StatusOK {
|
|
||||||
return nil, fmt.Errorf("invalid returncode %d", resp.StatusCode)
|
|
||||||
}
|
|
||||||
if resp.ContentLength > httpMaxRead {
|
if resp.ContentLength > httpMaxRead {
|
||||||
return nil, fmt.Errorf("response too big: %d/%d", resp.ContentLength, httpMaxRead)
|
return nil, fmt.Errorf("response too big: %d/%d", resp.ContentLength, httpMaxRead)
|
||||||
}
|
}
|
||||||
|
@ -52,10 +47,6 @@ func (c *IPGetter) GetCurrentPublicIP(ctx context.Context) (net.IP, error) {
|
||||||
return res, nil
|
return res, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func New() *IPGetter {
|
func New() *concreteIPGetter {
|
||||||
return &IPGetter{
|
return &concreteIPGetter{}
|
||||||
httpClt: http.DefaultClient,
|
|
||||||
remoteAddress: ifconfigURL,
|
|
||||||
timeout: httpTimeout,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,134 +0,0 @@
|
||||||
package ip
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"net/http"
|
|
||||||
"net/http/httptest"
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestGetIPOK(t *testing.T) {
|
|
||||||
t.Parallel()
|
|
||||||
|
|
||||||
mockSrv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
w.WriteHeader(http.StatusOK)
|
|
||||||
w.Write([]byte("198.51.100.42"))
|
|
||||||
}))
|
|
||||||
defer mockSrv.Close()
|
|
||||||
|
|
||||||
clt := IPGetter{
|
|
||||||
httpClt: mockSrv.Client(),
|
|
||||||
remoteAddress: mockSrv.URL,
|
|
||||||
timeout: 1 * time.Second,
|
|
||||||
}
|
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
val, err := clt.GetCurrentPublicIP(ctx)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Unexpected error %s", err.Error())
|
|
||||||
}
|
|
||||||
if val.String() != "198.51.100.42" {
|
|
||||||
t.Fatalf("Unexpected public IP %v", val)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGetIPServerErr(t *testing.T) {
|
|
||||||
t.Parallel()
|
|
||||||
|
|
||||||
mockSrv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
w.WriteHeader(http.StatusInternalServerError)
|
|
||||||
}))
|
|
||||||
defer mockSrv.Close()
|
|
||||||
|
|
||||||
clt := IPGetter{
|
|
||||||
httpClt: mockSrv.Client(),
|
|
||||||
remoteAddress: mockSrv.URL,
|
|
||||||
timeout: 1 * time.Second,
|
|
||||||
}
|
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
_, err := clt.GetCurrentPublicIP(ctx)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatal("Unexpected nil error")
|
|
||||||
} else if err.Error() != "invalid returncode 500" {
|
|
||||||
t.Fatalf("Unexpected error %s", err.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGetIPUnreachable(t *testing.T) {
|
|
||||||
t.Parallel()
|
|
||||||
|
|
||||||
mockSrv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
}))
|
|
||||||
mockSrv.Close()
|
|
||||||
|
|
||||||
clt := IPGetter{
|
|
||||||
httpClt: mockSrv.Client(),
|
|
||||||
remoteAddress: mockSrv.URL,
|
|
||||||
timeout: 1 * time.Second,
|
|
||||||
}
|
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
_, err := clt.GetCurrentPublicIP(ctx)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatal("Unexpected nil error")
|
|
||||||
} else if !strings.Contains(err.Error(), "connect: connection refused") {
|
|
||||||
t.Fatalf("Unexpected error %s", err.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGetIPInvalidResponse(t *testing.T) {
|
|
||||||
t.Parallel()
|
|
||||||
|
|
||||||
mockSrv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
w.WriteHeader(http.StatusOK)
|
|
||||||
w.Write([]byte("toto"))
|
|
||||||
}))
|
|
||||||
defer mockSrv.Close()
|
|
||||||
|
|
||||||
clt := IPGetter{
|
|
||||||
httpClt: mockSrv.Client(),
|
|
||||||
remoteAddress: mockSrv.URL,
|
|
||||||
timeout: 1 * time.Second,
|
|
||||||
}
|
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
_, err := clt.GetCurrentPublicIP(ctx)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatal("Unexpected nil error")
|
|
||||||
} else if err.Error() != `got an invalid public IP "toto"` {
|
|
||||||
t.Fatalf("Unexpected error %s", err.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGetIPTimeout(t *testing.T) {
|
|
||||||
t.Parallel()
|
|
||||||
|
|
||||||
mockSrv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
time.Sleep(100 * time.Millisecond)
|
|
||||||
w.WriteHeader(http.StatusOK)
|
|
||||||
w.Write([]byte("toto"))
|
|
||||||
}))
|
|
||||||
defer mockSrv.Close()
|
|
||||||
|
|
||||||
clt := IPGetter{
|
|
||||||
httpClt: mockSrv.Client(),
|
|
||||||
remoteAddress: mockSrv.URL,
|
|
||||||
timeout: 1 * time.Millisecond,
|
|
||||||
}
|
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
_, err := clt.GetCurrentPublicIP(ctx)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatal("Unexpected nil error")
|
|
||||||
} else if !strings.Contains(err.Error(), "context deadline exceeded") {
|
|
||||||
t.Fatalf("Unexpected error %s", err.Error())
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -9,6 +9,10 @@ import (
|
||||||
"git.faercol.me/faercol/public-ip-tracker/tracker/config"
|
"git.faercol.me/faercol/public-ip-tracker/tracker/config"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func testMethod(a int) int {
|
||||||
|
return a + 2
|
||||||
|
}
|
||||||
|
|
||||||
type cliArgs struct {
|
type cliArgs struct {
|
||||||
configPath string
|
configPath string
|
||||||
}
|
}
|
||||||
|
|
9
tracker/main_test.go
Normal file
9
tracker/main_test.go
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func TestTestMethod(t *testing.T) {
|
||||||
|
if testMethod(12) != 14 {
|
||||||
|
t.Fatalf("Unexpected result %d", testMethod((12)))
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue