Skip to content

Commit 556a281

Browse files
committed
NSM datapath monitoring on proxy NSC
Introduce optional NSM datapath monitoring/healing (or liveness check) checking if the connection is alive by pinging endpoint from the client. This option is for now available only between the proxy and LBs and has to be enabled via new env variables introduced in the proxy.
1 parent db3066c commit 556a281

File tree

10 files changed

+270
-27
lines changed

10 files changed

+270
-27
lines changed

cmd/proxy/internal/client/fullmesh.go

+3-2
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@ import (
2828
"github.com/google/uuid"
2929
"github.com/networkservicemesh/api/pkg/api/networkservice"
3030
"github.com/networkservicemesh/api/pkg/api/registry"
31+
"github.com/networkservicemesh/sdk/pkg/networkservice/common/heal"
3132
registryrefresh "github.com/networkservicemesh/sdk/pkg/registry/common/refresh"
3233
registrysendfd "github.com/networkservicemesh/sdk/pkg/registry/common/sendfd"
3334
registrychain "github.com/networkservicemesh/sdk/pkg/registry/core/chain"
@@ -237,9 +238,9 @@ func (fmnsc *FullMeshNetworkServiceClient) prepareQuery() *registry.NetworkServi
237238
// monitoring Network Service Endpoints belonging to the Network Service of the request.
238239
// Connects to each new Network Service Endpoint, and closes connection when a known
239240
// endpoint disappears.
240-
func NewFullMeshNetworkServiceClient(ctx context.Context, config *Config, additionalFunctionality ...networkservice.NetworkServiceClient) NetworkServiceClient {
241+
func NewFullMeshNetworkServiceClient(ctx context.Context, config *Config, healOptions []heal.Option, additionalFunctionality ...networkservice.NetworkServiceClient) NetworkServiceClient {
241242
// create base client relying on NSM's client.NewClient API
242-
client := newClient(ctx, config.Name, config.APIClient, additionalFunctionality...)
243+
client := newClient(ctx, config.Name, config.APIClient, healOptions, additionalFunctionality...)
243244

244245
fullMeshNetworkServiceClient := &FullMeshNetworkServiceClient{
245246
networkServiceClients: make(map[string]NetworkServiceClient),

cmd/proxy/internal/client/utils.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -41,15 +41,15 @@ func expirationTimeIsNull(expirationTime *timestamppb.Timestamp) bool {
4141
// Refresh Client comes from the NSM sdk version used. (In case of NSM v1.1.1 the built-in
4242
// refresh might lead to connection issues if the different path segments have different
4343
// maxTokenLifetime configured (unless the NSC side has the lowest maxtokenlifetime)).
44-
func newClient(ctx context.Context, name string, nsmAPIClient *nsm.APIClient, additionalFunctionality ...networkservice.NetworkServiceClient) networkservice.NetworkServiceClient {
44+
func newClient(ctx context.Context, name string, nsmAPIClient *nsm.APIClient, healOptions []heal.Option, additionalFunctionality ...networkservice.NetworkServiceClient) networkservice.NetworkServiceClient {
4545
additionalFunctionality = append(additionalFunctionality,
4646
sendfd.NewClient(),
4747
)
4848

4949
return client.NewClient(ctx,
5050
client.WithClientURL(&nsmAPIClient.Config.ConnectTo),
5151
client.WithName(name),
52-
client.WithHealClient(heal.NewClient(ctx)),
52+
client.WithHealClient(heal.NewClient(ctx, healOptions...)),
5353
client.WithAdditionalFunctionality(additionalFunctionality...),
5454
client.WithDialTimeout(nsmAPIClient.Config.DialTimeout),
5555
client.WithDialOptions(nsmAPIClient.GRPCDialOption...),

cmd/proxy/internal/config/config.go

+24-21
Original file line numberDiff line numberDiff line change
@@ -25,27 +25,30 @@ import (
2525

2626
// Config for the proxy
2727
type Config struct {
28-
Name string `default:"proxy" desc:"Pod Name"`
29-
ServiceName string `default:"proxy" desc:"Name of the Network Service" split_words:"true"`
30-
ConnectTo url.URL `default:"unix:///var/lib/networkservicemesh/nsm.io.sock" desc:"url to connect to NSM" split_words:"true"`
31-
DialTimeout time.Duration `default:"5s" desc:"timeout to dial NSMgr" split_words:"true"`
32-
RequestTimeout time.Duration `default:"15s" desc:"timeout to request NSE" split_words:"true"`
33-
MaxTokenLifetime time.Duration `default:"24h" desc:"maximum lifetime of tokens" split_words:"true"`
34-
IPAMService string `default:"ipam-service:7777" desc:"IP (or domain) and port of the IPAM Service" split_words:"true"`
35-
Host string `default:"" desc:"Host name the proxy is running on" split_words:"true"`
36-
NetworkServiceName string `default:"load-balancer" desc:"Name of the network service the proxy request the connection" split_words:"true"`
37-
Namespace string `default:"default" desc:"Namespace the pod is running on" split_words:"true"`
38-
Trench string `default:"default" desc:"Trench the pod is running on" split_words:"true"`
39-
Conduit string `default:"load-balancer" desc:"Name of the conduit" split_words:"true"`
40-
NSPServiceName string `default:"nsp-service" desc:"IP (or domain) of the NSP Service" split_words:"true"`
41-
NSPServicePort int `default:"7778" desc:"port of the NSP Service" split_words:"true"`
42-
IPFamily string `default:"dualstack" desc:"ip family" envconfig:"ip_family"`
43-
LogLevel string `default:"DEBUG" desc:"Log level" split_words:"true"`
44-
MTU int `default:"1500" desc:"Conduit MTU considered by local NSCs and NSE composing the network mesh" split_words:"true"`
45-
GRPCKeepaliveTime time.Duration `default:"30s" desc:"gRPC keepalive timeout"`
46-
GRPCProbeRPCTimeout time.Duration `default:"1s" desc:"RPC timeout of internal gRPC health probe" envconfig:"grpc_probe_rpc_timeout"`
47-
GRPCMaxBackoff time.Duration `default:"5s" desc:"Upper bound on gRPC connection backoff delay" envconfig:"grpc_max_backoff"`
48-
IPReleaseDelay time.Duration `default:"20s" desc:"delay releasing IP address of NSM connection" envconfig:"ip_release_delay"`
28+
Name string `default:"proxy" desc:"Pod Name"`
29+
ServiceName string `default:"proxy" desc:"Name of the Network Service" split_words:"true"`
30+
ConnectTo url.URL `default:"unix:///var/lib/networkservicemesh/nsm.io.sock" desc:"url to connect to NSM" split_words:"true"`
31+
DialTimeout time.Duration `default:"5s" desc:"timeout to dial NSMgr" split_words:"true"`
32+
RequestTimeout time.Duration `default:"15s" desc:"timeout to request NSE" split_words:"true"`
33+
MaxTokenLifetime time.Duration `default:"24h" desc:"maximum lifetime of tokens" split_words:"true"`
34+
IPAMService string `default:"ipam-service:7777" desc:"IP (or domain) and port of the IPAM Service" split_words:"true"`
35+
Host string `default:"" desc:"Host name the proxy is running on" split_words:"true"`
36+
NetworkServiceName string `default:"load-balancer" desc:"Name of the network service the proxy request the connection" split_words:"true"`
37+
Namespace string `default:"default" desc:"Namespace the pod is running on" split_words:"true"`
38+
Trench string `default:"default" desc:"Trench the pod is running on" split_words:"true"`
39+
Conduit string `default:"load-balancer" desc:"Name of the conduit" split_words:"true"`
40+
NSPServiceName string `default:"nsp-service" desc:"IP (or domain) of the NSP Service" split_words:"true"`
41+
NSPServicePort int `default:"7778" desc:"port of the NSP Service" split_words:"true"`
42+
IPFamily string `default:"dualstack" desc:"ip family" envconfig:"ip_family"`
43+
LogLevel string `default:"DEBUG" desc:"Log level" split_words:"true"`
44+
MTU int `default:"1500" desc:"Conduit MTU considered by local NSCs and NSE composing the network mesh" split_words:"true"`
45+
GRPCKeepaliveTime time.Duration `default:"30s" desc:"gRPC keepalive timeout"`
46+
GRPCProbeRPCTimeout time.Duration `default:"1s" desc:"RPC timeout of internal gRPC health probe" envconfig:"grpc_probe_rpc_timeout"`
47+
GRPCMaxBackoff time.Duration `default:"5s" desc:"Upper bound on gRPC connection backoff delay" envconfig:"grpc_max_backoff"`
48+
IPReleaseDelay time.Duration `default:"20s" desc:"delay releasing IP address of NSM connection" envconfig:"ip_release_delay"`
49+
LivenessCheckInterval time.Duration `default:"2s" desc:"Dataplane liveness check interval" split_words:"true"`
50+
LivenessCheckTimeout time.Duration `default:"1s" desc:"Dataplane liveness check timeout" split_words:"true"`
51+
LivenessCheckEnabled bool `default:"false" desc:"Dataplane liveness check enabled/disabled" split_words:"true"`
4952
}
5053

5154
// IsValid checks if the configuration is valid

cmd/proxy/internal/service/client.go

+3-1
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ import (
2626
"github.com/networkservicemesh/api/pkg/api/networkservice/payload"
2727
"github.com/networkservicemesh/sdk-sriov/pkg/networkservice/common/mechanisms/vfio"
2828
sriovtoken "github.com/networkservicemesh/sdk-sriov/pkg/networkservice/common/token"
29+
"github.com/networkservicemesh/sdk/pkg/networkservice/common/heal"
2930
"github.com/networkservicemesh/sdk/pkg/networkservice/common/mechanisms"
3031
"github.com/networkservicemesh/sdk/pkg/networkservice/common/mechanisms/kernel"
3132
"github.com/networkservicemesh/sdk/pkg/networkservice/core/chain"
@@ -46,6 +47,7 @@ func GetNSC(ctx context.Context,
4647
config *config.Config,
4748
nsmAPIClient *nsm.APIClient,
4849
p *proxy.Proxy,
50+
healOptions []heal.Option,
4951
interfaceMonitorClient networkservice.NetworkServiceClient) client.NetworkServiceClient {
5052

5153
logger := log.FromContextOrGlobal(ctx).WithValues("func", "GetNSC")
@@ -70,7 +72,7 @@ func GetNSC(ctx context.Context,
7072
proxyHealth.NewClient(),
7173
fullmeshtracker.NewClient(),
7274
)
73-
fullMeshClient := client.NewFullMeshNetworkServiceClient(ctx, clientConfig, additionalFunctionality)
75+
fullMeshClient := client.NewFullMeshNetworkServiceClient(ctx, clientConfig, healOptions, additionalFunctionality)
7476

7577
return fullMeshClient
7678
}

cmd/proxy/main.go

+12-1
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@ import (
2828

2929
"github.com/kelseyhightower/envconfig"
3030
"github.com/networkservicemesh/api/pkg/api/networkservice"
31+
"github.com/networkservicemesh/sdk/pkg/networkservice/common/heal"
3132
"github.com/networkservicemesh/sdk/pkg/tools/grpcutils"
3233
nsmlog "github.com/networkservicemesh/sdk/pkg/tools/log"
3334
ipamAPI "github.com/nordix/meridio/api/ipam/v1"
@@ -41,6 +42,7 @@ import (
4142
"github.com/nordix/meridio/pkg/health/probe"
4243
linuxKernel "github.com/nordix/meridio/pkg/kernel"
4344
"github.com/nordix/meridio/pkg/nsm"
45+
kernelheal "github.com/nordix/meridio/pkg/nsm/heal"
4446
"github.com/nordix/meridio/pkg/nsm/interfacemonitor"
4547
nsmmonitor "github.com/nordix/meridio/pkg/nsm/monitor"
4648
"github.com/nordix/meridio/pkg/nsp"
@@ -236,9 +238,18 @@ func main() {
236238
monitorClient := networkservice.NewMonitorConnectionClient(cc)
237239
go nsmmonitor.ConnectionMonitor(ctx, config.Name, monitorClient)
238240

241+
healOptions := []heal.Option{}
242+
if config.LivenessCheckEnabled {
243+
healOptions = []heal.Option{
244+
heal.WithLivenessCheckInterval(config.LivenessCheckInterval),
245+
heal.WithLivenessCheckTimeout(config.LivenessCheckTimeout),
246+
heal.WithLivenessCheck(kernelheal.KernelLivenessCheck),
247+
}
248+
}
249+
239250
// create and start NSC that connects all remote NSE belonging to the right service
240251
interfaceMonitorClient := interfacemonitor.NewClient(interfaceMonitor, p, netUtils)
241-
nsmClient := service.GetNSC(ctx, &config, nsmAPIClient, p, interfaceMonitorClient)
252+
nsmClient := service.GetNSC(ctx, &config, nsmAPIClient, p, healOptions, interfaceMonitorClient)
242253
defer nsmClient.Close()
243254
go func() {
244255
service.StartNSC(nsmClient, config.NetworkServiceName)

docs/components/proxy.md

+8
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,14 @@ NSM_NSP_SERVICE_NAME | string | IP (or domain) of the NSP Service | nsp-service
3939
NSM_NSP_SERVICE_PORT | int | port of the NSP Service | 7778
4040
NSM_IP_FAMILY | string | ip family | dualstack
4141
NSM_LOG_LEVEL | string | Log level | DEBUG
42+
NSM_MTU | string | Conduit MTU considered by local NSCs and NSE composing the network mesh | 1500
43+
NSM_GRPC_KEEPALIVE_TIME | time.Duration | gRPC keepalive timeout | 30s
44+
NSM_GRPC_PROBE_RPC_TIMEOUT | time.Duration | RPC timeout of internal gRPC health probe | 1s
45+
NSM_GRPC_MAX_BACKOFF | time.Duration | Upper bound on gRPC connection backoff delay | 5s
46+
NSM_IP_RELEASE_DELAY | time.Duration | delay releasing IP address of NSM connection | 20s
47+
NSM_LIVENESS_CHECK_INTERVAL | time.Duration | Dataplane liveness check interval | 2s
48+
NSM_LIVENESS_CHECK_TIMEOUT | time.Duration | Dataplane liveness check timeout | 1s
49+
NSM_LIVENESS_CHECK_ENABLED | bool | Dataplane liveness check enabled/disabled | false
4250

4351
## Command Line
4452

go.mod

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ require (
77
github.com/faisal-memon/sviddisk v0.0.0-20211007205134-77ccea0b9271
88
github.com/go-logr/logr v1.4.1
99
github.com/go-logr/zapr v1.3.0
10+
github.com/go-ping/ping v1.0.0
1011
github.com/golang/mock v1.6.0
1112
github.com/google/nftables v0.1.0
1213
github.com/google/uuid v1.3.1

go.sum

+3
Original file line numberDiff line numberDiff line change
@@ -106,6 +106,8 @@ github.com/go-openapi/jsonreference v0.20.2 h1:3sVjiK66+uXK/6oQ8xgcRKcFgQ5KXa2Kv
106106
github.com/go-openapi/jsonreference v0.20.2/go.mod h1:Bl1zwGIM8/wsvqjsOQLJ/SH+En5Ap4rVB5KVcIDZG2k=
107107
github.com/go-openapi/swag v0.22.3 h1:yMBqmnQ0gyZvEb/+KzuWZOXgllrXT4SADYbvDaXHv/g=
108108
github.com/go-openapi/swag v0.22.3/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14=
109+
github.com/go-ping/ping v1.0.0 h1:34GZiqLDqqIHEeL5NZIz7jSnMluK7/p0qDB436yO6H0=
110+
github.com/go-ping/ping v1.0.0/go.mod h1:35JbSyV/BYqHwwRA6Zr1uVDm1637YlNOU61wI797NPI=
109111
github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE=
110112
github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 h1:tfuBGBXKqDEevZMzYi5KSi8KkcZtzBcTgAUUtapy0OI=
111113
github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572/go.mod h1:9Pwr4B2jHnOSGXyyzV8ROjYa2ojvAY6HCGYYfMoC3Ls=
@@ -412,6 +414,7 @@ golang.org/x/net v0.0.0-20191007182048-72f939374954/go.mod h1:z5CRVTTTmAJ677TzLL
412414
golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
413415
golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
414416
golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A=
417+
golang.org/x/net v0.0.0-20200904194848-62affa334b73/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
415418
golang.org/x/net v0.0.0-20201010224723-4f7140c49acb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
416419
golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
417420
golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=

pkg/nsm/heal/liveness_check.go

+206
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,206 @@
1+
// Copyright (c) 2022-2023 Cisco and/or its affiliates.
2+
//
3+
// SPDX-License-Identifier: Apache-2.0
4+
//
5+
// Licensed under the Apache License, Version 2.0 (the "License");
6+
// you may not use this file except in compliance with the License.
7+
// You may obtain a copy of the License at:
8+
//
9+
// http://www.apache.org/licenses/LICENSE-2.0
10+
//
11+
// Unless required by applicable law or agreed to in writing, software
12+
// distributed under the License is distributed on an "AS IS" BASIS,
13+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
// See the License for the specific language governing permissions and
15+
// limitations under the License.
16+
17+
// Package heal contains an implementation of LivenessChecker.
18+
package heal
19+
20+
import (
21+
"context"
22+
"errors"
23+
"net"
24+
"strings"
25+
"time"
26+
27+
"github.com/go-ping/ping"
28+
"github.com/networkservicemesh/api/pkg/api/networkservice"
29+
"github.com/networkservicemesh/api/pkg/api/networkservice/mechanisms/kernel"
30+
"github.com/networkservicemesh/sdk/pkg/tools/log"
31+
)
32+
33+
const (
34+
defaultTimeout = time.Second
35+
packetCount = 4
36+
37+
DatapathSourceIPsKey = "DATAPATH_SOURCE_IPS"
38+
DatapathDestinationIPsKey = "DATAPATH_DESTINATION_IPS"
39+
DatapathIPsSeparator = " "
40+
)
41+
42+
type options struct {
43+
pingerFactory PingerFactory
44+
}
45+
46+
// Option is an option pattern for LivelinessChecker
47+
type Option func(o *options)
48+
49+
// WithPingerFactory - sets any custom pinger factory
50+
func WithPingerFactory(pf PingerFactory) Option {
51+
return func(o *options) {
52+
o.pingerFactory = pf
53+
}
54+
}
55+
56+
// KernelLivenessCheck is an implementation of heal.LivenessCheck
57+
func KernelLivenessCheck(deadlineCtx context.Context, conn *networkservice.Connection) bool {
58+
return KernelLivenessCheckWithOptions(deadlineCtx, conn)
59+
}
60+
61+
// KernelLivenessCheckWithOptions is an implementation with options of heal.LivenessCheck. It sends ICMP
62+
// ping and checks reply. Returns false if didn't get reply.
63+
func KernelLivenessCheckWithOptions(deadlineCtx context.Context, conn *networkservice.Connection, opts ...Option) bool {
64+
// Apply options
65+
o := &options{
66+
pingerFactory: &defaultPingerFactory{},
67+
}
68+
for _, opt := range opts {
69+
opt(o)
70+
}
71+
var pingerFactory = o.pingerFactory
72+
73+
if mechanism := conn.GetMechanism().GetType(); mechanism != kernel.MECHANISM {
74+
log.FromContext(deadlineCtx).Warnf("ping is not supported for mechanism %v", mechanism)
75+
return true
76+
}
77+
78+
sourceIPs, destinationIPs := getSourceDestinationIPs(conn.GetContext().GetExtraContext())
79+
combinationCount := len(sourceIPs) * len(destinationIPs)
80+
if combinationCount == 0 {
81+
log.FromContext(deadlineCtx).Debug("No IP address")
82+
return true
83+
}
84+
85+
deadline, ok := deadlineCtx.Deadline()
86+
if !ok {
87+
deadline = time.Now().Add(defaultTimeout)
88+
}
89+
timeout := time.Until(deadline)
90+
91+
responseCh := make(chan error, combinationCount)
92+
defer close(responseCh)
93+
for _, sourceIP := range sourceIPs {
94+
for _, destinationIP := range destinationIPs {
95+
if (destinationIP.To4() != nil) != (sourceIP.To4() != nil) {
96+
responseCh <- nil
97+
continue
98+
}
99+
100+
go func(srcIP, dstIP string) {
101+
logger := log.FromContext(deadlineCtx).WithField("srcIP", srcIP).WithField("dstIP", dstIP)
102+
pinger := pingerFactory.CreatePinger(srcIP, dstIP, timeout, packetCount)
103+
104+
err := pinger.Run()
105+
if err != nil {
106+
logger.Errorf("Ping failed: %s", err.Error())
107+
responseCh <- err
108+
return
109+
}
110+
111+
if pinger.GetReceivedPackets() == 0 {
112+
err = errors.New("No packets received")
113+
logger.Errorf(err.Error())
114+
responseCh <- err
115+
return
116+
}
117+
responseCh <- nil
118+
}(sourceIP.String(), destinationIP.String())
119+
}
120+
}
121+
122+
// Waiting for all ping results. If at least one fails - return false
123+
return waitForResponses(responseCh)
124+
}
125+
126+
func getSourceDestinationIPs(extraContext map[string]string) ([]net.IP, []net.IP) {
127+
sourceIPs := []net.IP{}
128+
destinationIPs := []net.IP{}
129+
sourceIPsStr := extraContext[DatapathSourceIPsKey]
130+
destinationIPsStr := extraContext[DatapathDestinationIPsKey]
131+
132+
for _, sourceIPStr := range strings.Split(sourceIPsStr, DatapathIPsSeparator) {
133+
sourceIP, _, err := net.ParseCIDR(sourceIPStr)
134+
if err != nil {
135+
continue
136+
}
137+
138+
sourceIPs = append(sourceIPs, sourceIP)
139+
}
140+
141+
for _, destinationIPStr := range strings.Split(destinationIPsStr, DatapathIPsSeparator) {
142+
destinationIP, _, err := net.ParseCIDR(destinationIPStr)
143+
if err != nil {
144+
continue
145+
}
146+
147+
destinationIPs = append(destinationIPs, destinationIP)
148+
}
149+
150+
return sourceIPs, destinationIPs
151+
}
152+
153+
func waitForResponses(responseCh <-chan error) bool {
154+
respCount := cap(responseCh)
155+
success := true
156+
for {
157+
resp, ok := <-responseCh
158+
if !ok {
159+
return false
160+
}
161+
if resp != nil {
162+
success = false
163+
}
164+
respCount--
165+
if respCount == 0 {
166+
return success
167+
}
168+
}
169+
}
170+
171+
// PingerFactory - factory interface for creating pingers
172+
type PingerFactory interface {
173+
CreatePinger(srcIP, dstIP string, timeout time.Duration, count int) Pinger
174+
}
175+
176+
// Pinger - pinger interface
177+
type Pinger interface {
178+
Run() error
179+
GetReceivedPackets() int
180+
}
181+
182+
type defaultPingerFactory struct{}
183+
184+
func (p *defaultPingerFactory) CreatePinger(srcIP, dstIP string, timeout time.Duration, count int) Pinger {
185+
pi := ping.New(dstIP)
186+
pi.Source = srcIP
187+
pi.Timeout = timeout
188+
pi.Count = count
189+
if count != 0 {
190+
pi.Interval = timeout / time.Duration(count)
191+
}
192+
193+
return &defaultPinger{pinger: pi}
194+
}
195+
196+
type defaultPinger struct {
197+
pinger *ping.Pinger
198+
}
199+
200+
func (p *defaultPinger) Run() error {
201+
return p.pinger.Run()
202+
}
203+
204+
func (p *defaultPinger) GetReceivedPackets() int {
205+
return p.pinger.Statistics().PacketsRecv
206+
}

0 commit comments

Comments
 (0)