checker-ping/checker/rule.go
Pierre-Olivier Mercier 2aa596afd5 Address publication review feedback
Add the AGPL LICENSE file and a deployment-security note in the README
to clarify that the unauthenticated /collect endpoint must run on a
trusted network.

Fix the IPv6 reachability rule so it consults the IP actually probed:
PingTargetResult now carries ResolvedIP populated from pinger.IPAddr(),
which lets the rule classify hostname targets correctly instead of
always reporting "No IPv6 target pinged".

Tighten error handling: ipsFromService now propagates JSON errors,
ExtractMetrics wraps decode failures, the count option returns an
explicit error when out of range instead of silently clamping, and the
"all pings failed" message no longer concatenates every per-target
error. Threshold validation is factored into validateThresholdPair and
shared between the RTT and packet-loss rules.

Add unit tests covering address resolution, threshold validation, and
each rule's evaluation paths.
2026-04-26 18:05:47 +07:00

92 lines
3 KiB
Go

// This file is part of the happyDomain (R) project.
// Copyright (c) 2020-2026 happyDomain
// Authors: Pierre-Olivier Mercier, et al.
//
// This program is offered under a commercial and under the AGPL license.
// For commercial licensing, contact us at <contact@happydomain.org>.
//
// For AGPL licensing:
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
package checker
import (
"context"
"fmt"
"net"
sdk "git.happydns.org/checker-sdk-go/checker"
)
// Rules returns the full list of CheckRules exposed by the ping checker.
// Each rule covers a single concern so callers see at a glance which
// aspects passed and which did not, instead of sharing a single monolithic
// rule result.
func Rules() []sdk.CheckRule {
return []sdk.CheckRule{
&reachabilityRule{},
&packetLossRule{},
&rttRule{},
&ipv6ReachabilityRule{},
}
}
// validateThresholdPair checks that warn and crit are within [min, max] and
// that crit is strictly greater than warn. The names are used in error
// messages so callers get diagnostics naming their actual options.
func validateThresholdPair(warnName, critName string, warn, crit, min, max float64) error {
if warn < min || warn > max {
return fmt.Errorf("%s must be between %v and %v", warnName, min, max)
}
if crit < min || crit > max {
return fmt.Errorf("%s must be between %v and %v", critName, min, max)
}
if crit <= warn {
return fmt.Errorf("%s (%v) must be greater than %s (%v)", critName, crit, warnName, warn)
}
return nil
}
// loadPingData fetches the ping observation. On error, returns a CheckState
// the caller should emit to short-circuit its rule.
func loadPingData(ctx context.Context, obs sdk.ObservationGetter) (*PingData, *sdk.CheckState) {
var data PingData
if err := obs.Get(ctx, ObservationKeyPing, &data); err != nil {
return nil, &sdk.CheckState{
Status: sdk.StatusError,
Message: fmt.Sprintf("failed to load ping observation: %v", err),
Code: "ping.observation_error",
}
}
return &data, nil
}
// noTargetsState is returned when the observation has no targets at all.
func noTargetsState(code string) sdk.CheckState {
return sdk.CheckState{
Status: sdk.StatusUnknown,
Message: "No targets to ping",
Code: code,
}
}
// isIPv6 reports whether addr parses as an IPv6 address (excluding
// IPv4-mapped representations).
func isIPv6(addr string) bool {
ip := net.ParseIP(addr)
if ip == nil {
return false
}
return ip.To4() == nil
}