talos/pkg/conditions/poll.go
Andrey Smirnov 9379cf9ee1 refactor: expose provision as public package
This change is only moving packages and updating import paths.

Goal: expose `internal/pkg/provision` as `pkg/provision` to enable other
projects to import Talos provisioning library.

As cluster checks are almost always required as part of provisioning
process, package `internal/pkg/cluster` was also made public as
`pkg/cluster`.

Other changes were direct dependencies discovered by `importvet` which
were updated.

Public packages (useful, general purpose packages with stable API):

* `internal/pkg/conditions` -> `pkg/conditions`
* `internal/pkg/tail` -> `pkg/tail`

Private packages (used only on provisioning library internally):

* `internal/pkg/inmemhttp` -> `pkg/provision/internal/inmemhttp`
* `internal/pkg/kernel/vmlinuz` -> `pkg/provision/internal/vmlinuz`
* `internal/pkg/cniutils` -> `pkg/provision/internal/cniutils`

Signed-off-by: Andrey Smirnov <smirnov.andrey@gmail.com>
2020-08-12 05:12:05 -07: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,
}
}