You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
boundary/internal/host/plugin/service_list_hosts_refresh.go

84 lines
3.3 KiB

// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1
package plugin
import (
"context"
"time"
"github.com/hashicorp/boundary/globals"
"github.com/hashicorp/boundary/internal/errors"
"github.com/hashicorp/boundary/internal/host"
"github.com/hashicorp/boundary/internal/listtoken"
"github.com/hashicorp/boundary/internal/pagination"
ppagination "github.com/hashicorp/boundary/internal/pagination/plugin"
"github.com/hashicorp/boundary/internal/plugin"
"github.com/hashicorp/boundary/internal/types/resource"
)
// ListHostsRefresh lists hosts according to the page size
// and list token, filtering out entries that do not
// pass the filter item fn. It returns a new list token
// based on the old one, the grants hash, and the returned
// hosts. It also returns the plugin associated with the host catalog.
func ListHostsRefresh(
ctx context.Context,
grantsHash []byte,
pageSize int,
filterItemFn ppagination.ListPluginFilterFunc[host.Host],
tok *listtoken.Token,
repo *Repository,
hostCatalogId string,
) (*pagination.ListResponse[host.Host], *plugin.Plugin, error) {
const op = "plugin.ListHostsRefresh"
switch {
case len(grantsHash) == 0:
return nil, nil, errors.New(ctx, errors.InvalidParameter, op, "missing grants hash")
case pageSize < 1:
return nil, nil, errors.New(ctx, errors.InvalidParameter, op, "page size must be at least 1")
case filterItemFn == nil:
return nil, nil, errors.New(ctx, errors.InvalidParameter, op, "missing filter item callback")
case tok == nil:
return nil, nil, errors.New(ctx, errors.InvalidParameter, op, "missing token")
case repo == nil:
return nil, nil, errors.New(ctx, errors.InvalidParameter, op, "missing repo")
case hostCatalogId == "":
return nil, nil, errors.New(ctx, errors.InvalidParameter, op, "missing host catalog ID")
case tok.ResourceType != resource.Host:
return nil, nil, errors.New(ctx, errors.InvalidParameter, op, "token did not have a host resource type")
}
rt, ok := tok.Subtype.(*listtoken.StartRefreshToken)
if !ok {
return nil, nil, errors.New(ctx, errors.InvalidParameter, op, "token did not have a start-refresh token component")
}
listItemsFn := func(ctx context.Context, lastPageItem host.Host, limit int) ([]host.Host, *plugin.Plugin, time.Time, error) {
opts := []Option{
WithLimit(limit),
}
if lastPageItem != nil {
opts = append(opts, WithStartPageAfterItem(lastPageItem))
}
// Add the database read timeout to account for any creations missed due to concurrent
// transactions in the initial pagination phase.
pHosts, plg, listTime, err := repo.listHostsRefresh(ctx, hostCatalogId, rt.PreviousPhaseUpperBound.Add(-globals.RefreshReadLookbackDuration), opts...)
if err != nil {
return nil, nil, time.Time{}, err
}
var hosts []host.Host
for _, host := range pHosts {
hosts = append(hosts, host)
}
return hosts, plg, listTime, nil
}
listDeletedIdsFn := func(ctx context.Context, since time.Time) ([]string, time.Time, error) {
// Add the database read timeout to account for any deletes missed due to concurrent
// transactions in the original list pagination phase.
return repo.listDeletedHostIds(ctx, since.Add(-globals.RefreshReadLookbackDuration))
}
return ppagination.ListPluginRefresh(ctx, grantsHash, pageSize, filterItemFn, listItemsFn, repo.estimatedHostCount, listDeletedIdsFn, tok)
}