Andrey Smirnov 0081ac5fac refactor: extract Talos cluster provisioner as common code
This extracts Docker Talos cluster provisioner as common code
which might be shared between `osctl cluster` and integration-test.

There should be almost no functional changes.

As proof of concept, abstract cluster readiness checks were implemented
based on provisioned cluster state. It implements same checks as
`basic-integration.sh` in pure Go via Talos/K8s clients.

`conditions` package was promoted from machined-internal to
`internal/pkg` as it is used to run the checks.

Signed-off-by: Andrey Smirnov <smirnov.andrey@gmail.com>
2019-12-27 12:14:19 -08:00

87 lines
1.8 KiB
Go

// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
package conditions
import (
"context"
"fmt"
"sync"
"time"
)
// AssertionFunc is called every poll interval until it returns nil.
type AssertionFunc func(ctx context.Context) error
type pollingCondition struct {
lastErrMu sync.Mutex
lastErr error
lastErrSet bool
assertion AssertionFunc
description string
timeout, interval time.Duration
}
func (p *pollingCondition) String() string {
lastErr := "..."
p.lastErrMu.Lock()
if p.lastErrSet {
if p.lastErr != nil {
lastErr = p.lastErr.Error()
} else {
lastErr = "OK"
}
}
p.lastErrMu.Unlock()
return fmt.Sprintf("%s: %s", p.description, lastErr)
}
func (p *pollingCondition) Wait(ctx context.Context) error {
ticker := time.NewTicker(p.interval)
defer ticker.Stop()
timeoutCtx, timeoutCtxCancel := context.WithTimeout(ctx, p.timeout)
defer timeoutCtxCancel()
for {
err := func() error {
runCtx, runCtxCancel := context.WithTimeout(ctx, p.interval)
defer runCtxCancel()
err := p.assertion(runCtx)
p.lastErrMu.Lock()
p.lastErr = err
p.lastErrSet = true
p.lastErrMu.Unlock()
return err
}()
if err == nil {
return nil
}
select {
case <-timeoutCtx.Done():
return timeoutCtx.Err()
case <-ticker.C:
}
}
}
// PollingCondition converts AssertionFunc into Condition by calling it every interval until timeout
// is reached.
func PollingCondition(description string, assertion AssertionFunc, timeout, interval time.Duration) Condition {
return &pollingCondition{
assertion: assertion,
description: description,
timeout: timeout,
interval: interval,
}
}