polardbxoperator/pkg/probe/prober.go

206 lines
4.3 KiB
Go

/*
Copyright 2021 Alibaba Group Holding Limited.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package probe
import (
"context"
"database/sql"
"errors"
"fmt"
"io"
"net/http"
"strconv"
"strings"
"time"
_ "github.com/go-sql-driver/mysql"
"github.com/alibaba/polardbx-operator/pkg/probe/xstore_ext"
_ "github.com/alibaba/polardbx-operator/pkg/probe/xstore_ext/plugin"
"github.com/alibaba/polardbx-operator/pkg/util/defaults"
)
const (
TypePolarDBX = "polardbx"
TypeXStore = "xstore"
TypeSelf = "server"
TypeCdc = "cdc"
)
type Prober struct {
target string
extra string
user string
host string
port int
timeout time.Duration
db *sql.DB
ctx context.Context
cancel context.CancelFunc
}
func (p *Prober) valid() bool {
for _, t := range []string{
TypePolarDBX, TypeXStore, TypeSelf, TypeCdc,
} {
if p.target == t {
return true
}
}
return false
}
func (p *Prober) defaultUser() string {
switch p.target {
case TypeXStore:
return "root"
case TypePolarDBX:
return "polardbx_root"
default:
return ""
}
}
func NewProber(r *http.Request) (*Prober, error) {
p := &Prober{
target: r.Header.Get("Probe-Target"),
extra: r.Header.Get("Probe-Extra"),
}
if !p.valid() {
return nil, errors.New("invalid probe target: %s" + p.target)
}
// Extract parameters from http headers.
p.host = defaults.NonEmptyStrOrDefault(r.Header.Get("Probe-Host"), "127.0.0.1")
p.user = defaults.NonEmptyStrOrDefault(r.Header.Get("Probe-User"), p.defaultUser())
var err error
probePort := defaults.NonEmptyStrOrDefault(r.Header.Get("Probe-Port"), "3306")
p.port, err = strconv.Atoi(probePort)
if err != nil {
return nil, errors.New("invalid probe parameters: failed to parse port, " + err.Error())
}
probeTimeout := defaults.NonEmptyStrOrDefault(r.Header.Get("Probe-Timeout"), "5s")
timeout, err := time.ParseDuration(probeTimeout)
if err != nil {
return nil, errors.New("invalid probe parameters: failed to parse timeout, " + err.Error())
}
p.timeout = timeout
if timeout > 0 {
p.ctx, p.cancel = context.WithTimeout(context.Background(), timeout)
} else {
p.ctx, p.cancel = context.WithCancel(context.Background())
}
return p, nil
}
func (p *Prober) Close() error {
if p.cancel != nil {
p.cancel()
}
if p.db != nil {
return p.db.Close()
}
return nil
}
func (p *Prober) connect() error {
db, err := sql.Open("mysql", fmt.Sprintf("%s@tcp(%s:%d)/?timeout=%s",
p.user, p.host, p.port, p.timeout))
if err != nil {
return err
}
p.db = db
return nil
}
func (p *Prober) ping() error {
return p.db.PingContext(p.ctx)
}
func (p *Prober) cdcConnect() error {
httpClient := http.Client{Timeout: p.timeout}
url := fmt.Sprintf("http://%s:%d/status", p.host, p.port)
resp, err := httpClient.Get(url)
if err != nil {
return err
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
if err != nil {
return err
}
bodyStr := strings.ToUpper(strings.TrimSpace(string(body)))
if bodyStr != "OK" {
return fmt.Errorf("unhealthy status body=%s , url=%s", bodyStr, url)
}
return nil
}
func (p *Prober) Liveness() error {
switch p.target {
case TypeXStore, TypePolarDBX:
if err := p.connect(); err != nil {
return err
}
// Return if already timeout.
select {
case <-p.ctx.Done():
return p.ctx.Err()
default:
}
return p.ping()
case TypeCdc:
return p.cdcConnect()
case TypeSelf:
return nil
default:
return errors.New("unknown probe type")
}
}
func (p *Prober) extraReadinessForXStore() error {
xstoreExt := xstore_ext.GetXStoreExt(p.extra)
if xstoreExt == nil {
return nil
}
return xstoreExt.Readiness(p.ctx, p.host, p.db)
}
func (p *Prober) ProbeReadiness() error {
err := p.Liveness()
if err != nil {
return err
}
if p.target == TypeXStore {
err = p.extraReadinessForXStore()
if err != nil {
return err
}
}
return nil
}