Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

DAOS-16038 pool: Enable pool list for clients #14575

Closed
wants to merge 11 commits into from
2 changes: 1 addition & 1 deletion src/client/api/init.c
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ const struct daos_task_api dc_funcs[] = {
/** Management */
{dc_deprecated, 0},
{dc_deprecated, 0},
{dc_deprecated, 0},
{dc_mgmt_pool_list, sizeof(daos_mgmt_pool_list_t)},
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

do you really need to add a user task API for this function though? I'm not really sure this is required and you can probably not worry about changes in the task stuff.
otherwise changes here are not enough and you would need to add a few more things.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure. I was following the pattern set by @kccain when he added the original implementation of pool list. I haven't really dug into it beyond that.

Given that I don't feel strongly either way, I'm happy to follow your guidance on the best way to implement this feature.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@mchaarawi: Ping... Please advise on the recommended approach here. I'm getting ready to rebase all of this work on master, and would prefer to push that PR in as close to a final state as possible rather than iterating on it.

Also, I don't see much in the way of client API tests. Not particularly keen to invent all of that myself, but happy to work with someone who knows more about this side of things. I am planning to add ftests for this feature -- would that be acceptable in lieu of new client API unit tests?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

sorry i didn't know you were still looking for my review on this. as i mentioned, you should just avoid adding this change with the task api, otherwise you still need to update other structures to support that.
for client api tests, im not sure what you mean. i believe you should add some tests there:
https://github.com/daos-stack/daos/blob/master/src/tests/suite/daos_mgmt.c
but if you want to use ftest, that is fine too.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

sorry i didn't know you were still looking for my review on this. as i mentioned, you should just avoid adding this change with the task api, otherwise you still need to update other structures to support that.

OK, is the task API deprecated? I have a superficial understanding of the client API's workings, so as I said I was just following the existing patterns. If I didn't use the task API, would I basically just rework daos_mgmt_list_pools() to just call dc_mgmt_pool_list() directly? Is there any downside to doing it this way?

for client api tests, im not sure what you mean. i believe you should add some tests there: https://github.com/daos-stack/daos/blob/master/src/tests/suite/daos_mgmt.c but if you want to use ftest, that is fine too.

Ah, OK. I was thinking of unit tests under src/client/api/tests. I'll just rework the suite test to use the re-added client API instead of the dmg helper.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK, is the task API deprecated? I have a superficial understanding of the client API's workings, so as I said I was just following the existing patterns.

no the task api is not deprecated. but to add the task API entry for this function, you still need to update:
https://github.com/daos-stack/daos/blob/mjmac/DAOS-15982/src/include/daos/task.h
to be safe.
otherwise you can just remove the task api entry for it, and do as you suggested:

If I didn't use the task API, would I basically just rework daos_mgmt_list_pools() to just call dc_mgmt_pool_list() directly? Is there any downside to doing it this way?

either way is fine for me.

{dc_debug_set_params, sizeof(daos_set_params_t)},
{dc_mgmt_get_bs_state, sizeof(daos_mgmt_get_bs_state_t)},

Expand Down
26 changes: 26 additions & 0 deletions src/client/api/mgmt.c
Original file line number Diff line number Diff line change
Expand Up @@ -79,3 +79,29 @@ daos_mgmt_put_sys_info(struct daos_sys_info *info)
{
dc_mgmt_put_sys_info(info);
}

int
daos_mgmt_list_pools(const char *group, daos_size_t *npools, daos_mgmt_pool_info_t *pools,
daos_event_t *ev)
{
daos_mgmt_pool_list_t *args;
tse_task_t *task;
int rc;

DAOS_API_ARG_ASSERT(*args, MGMT_LIST_POOLS);

if (npools == NULL) {
D_ERROR("npools must be non-NULL\n");
return -DER_INVAL;
}

rc = dc_task_create(dc_mgmt_pool_list, NULL, ev, &task);
if (rc)
return rc;
args = dc_task_get_args(task);
args->grp = group;
args->pools = pools;
args->npools = npools;

return dc_task_schedule(task, true);
}
77 changes: 77 additions & 0 deletions src/control/cmd/daos/pool.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import (
"github.com/daos-stack/daos/src/control/lib/daos"
"github.com/daos-stack/daos/src/control/lib/ranklist"
"github.com/daos-stack/daos/src/control/lib/ui"
"github.com/daos-stack/daos/src/control/logging"
)

/*
Expand Down Expand Up @@ -180,6 +181,7 @@ func (cmd *poolBaseCmd) getAttr(name string) (*attribute, error) {
}

type poolCmd struct {
List poolListCmd `command:"list" description:"list pools to which this user has access"`
Query poolQueryCmd `command:"query" description:"query pool info"`
QueryTargets poolQueryTargetsCmd `command:"query-targets" description:"query pool target info"`
ListConts containerListCmd `command:"list-containers" alias:"list-cont" description:"list all containers in pool"`
Expand Down Expand Up @@ -601,3 +603,78 @@ func (cmd *poolAutoTestCmd) Execute(_ []string) error {

return nil
}

func getPoolList(log logging.Logger, queryEnabled bool) ([]*daos.PoolInfo, error) {
var rc C.int
bufSize := C.size_t(0)

// First, fetch the total number of pools in the system.
// We may not have access to all of them, so this is an upper bound.
rc = C.daos_mgmt_list_pools(nil, &bufSize, nil, nil)
if err := daosError(rc); err != nil {
return nil, err
}

if bufSize < 1 {
return nil, nil
}

// Now, we actually fetch the pools into the buffer that we've created.
cPools := make([]C.daos_mgmt_pool_info_t, bufSize)
rc = C.daos_mgmt_list_pools(nil, &bufSize, &cPools[0], nil)
if err := daosError(rc); err != nil {
return nil, err
}
pools := make([]*daos.PoolInfo, 0, bufSize)
for i := 0; i < int(bufSize); i++ {
poolUUID, err := uuidFromC(cPools[i].mgpi_uuid)
if err != nil {
return nil, err
}
svcRanks, err := rankSetFromC(cPools[i].mgpi_svc)
if err != nil {
return nil, err
}

// Populate the basic info.
pool := &daos.PoolInfo{
UUID: poolUUID,
Label: C.GoString(cPools[i].mgpi_label),
ServiceReplicas: svcRanks.Ranks(),
State: daos.PoolServiceStateReady,
}

if queryEnabled {
// TODO: rework query logic to be callable standalone.
}
pools = append(pools, pool)
}

log.Debugf("fetched %d/%d pools", len(pools), len(cPools))
return pools, nil
}

type poolListCmd struct {
daosCmd
Verbose bool `short:"v" long:"verbose" description:"Add pool UUIDs and service replica lists to display"`
NoQuery bool `short:"n" long:"no-query" description:"Disable query of listed pools"`
}

func (cmd *poolListCmd) Execute(_ []string) error {
pools, err := getPoolList(cmd.Logger, !cmd.NoQuery)
if err != nil {
return err
}

if cmd.JSONOutputEnabled() {
return cmd.OutputJSON(pools, nil)
}

var buf strings.Builder
if err := pretty.PrintPoolList(pools, &buf, cmd.Verbose); err != nil {
return err
}
cmd.Info(buf.String())

return nil
}
184 changes: 184 additions & 0 deletions src/control/cmd/daos/pretty/pool.go
Original file line number Diff line number Diff line change
Expand Up @@ -98,3 +98,187 @@ func PrintPoolQueryTargetInfo(pqti *daos.PoolQueryTargetInfo, out io.Writer) err

return w.Err
}

func poolListCreateRow(pool *daos.PoolInfo, upgrade bool) txtfmt.TableRow {
// display size of the largest non-empty tier
var size uint64
poolUsage := pool.Usage()
for ti := len(poolUsage) - 1; ti >= 0; ti-- {
if poolUsage[ti].Size != 0 {
size = poolUsage[ti].Size
break
}
}

// display usage of the most used tier
var used int
for ti := 0; ti < len(poolUsage); ti++ {
t := poolUsage[ti]
u := float64(t.Size-t.Free) / float64(t.Size)

if int(u*100) > used {
used = int(u * 100)
}
}

// display imbalance of the most imbalanced tier
var imbalance uint32
for ti := 0; ti < len(poolUsage); ti++ {
if poolUsage[ti].Imbalance > imbalance {
imbalance = poolUsage[ti].Imbalance
}
}

row := txtfmt.TableRow{
"Pool": pool.Name(),
"Size": humanize.Bytes(size),
"State": pool.State.String(),
"Used": fmt.Sprintf("%d%%", used),
"Imbalance": fmt.Sprintf("%d%%", imbalance),
"Disabled": fmt.Sprintf("%d/%d", pool.DisabledTargets, pool.TotalTargets),
}

if upgrade {
upgradeString := "None"

if pool.PoolLayoutVer != pool.UpgradeLayoutVer {
upgradeString = fmt.Sprintf("%d->%d", pool.PoolLayoutVer, pool.UpgradeLayoutVer)
}
row["UpgradeNeeded?"] = upgradeString
}

return row
}

func printPoolList(pools []*daos.PoolInfo, out io.Writer) error {
upgrade := false
for _, pool := range pools {
if pool.PoolLayoutVer != pool.UpgradeLayoutVer {
upgrade = true
}
}

titles := []string{"Pool", "Size", "State", "Used", "Imbalance", "Disabled"}
if upgrade {
titles = append(titles, "UpgradeNeeded?")
}
formatter := txtfmt.NewTableFormatter(titles...)

var table []txtfmt.TableRow
for _, pool := range pools {
table = append(table, poolListCreateRow(pool, upgrade))
}

fmt.Fprintln(out, formatter.Format(table))

return nil
}

func addVerboseTierUsage(row txtfmt.TableRow, usage *daos.PoolTierUsage) txtfmt.TableRow {
row[usage.TierName+" Size"] = humanize.Bytes(usage.Size)
row[usage.TierName+" Used"] = humanize.Bytes(usage.Size - usage.Free)
row[usage.TierName+" Imbalance"] = fmt.Sprintf("%d%%", usage.Imbalance)

return row
}

func poolListCreateRowVerbose(pool *daos.PoolInfo, hasSpace, hasRebuild bool) txtfmt.TableRow {
label := pool.Label
if label == "" {
label = "-"
}

svcReps := "N/A"
if len(pool.ServiceReplicas) != 0 {
svcReps = PrintRanks(pool.ServiceReplicas)
}

upgrade := "None"
if pool.PoolLayoutVer != pool.UpgradeLayoutVer {
upgrade = fmt.Sprintf("%d->%d", pool.PoolLayoutVer, pool.UpgradeLayoutVer)
}

row := txtfmt.TableRow{
"Label": label,
"UUID": pool.UUID.String(),
"State": pool.State.String(),
"SvcReps": svcReps,
}
if hasSpace {
row["Disabled"] = fmt.Sprintf("%d/%d", pool.DisabledTargets, pool.TotalTargets)
row["UpgradeNeeded?"] = upgrade
}
if hasRebuild {
row["Rebuild State"] = pool.RebuildState()
}

if hasSpace {
for _, tu := range pool.Usage() {
row = addVerboseTierUsage(row, tu)
}
}

return row
}

func printVerbosePoolList(pools []*daos.PoolInfo, out io.Writer) error {
// Basic pool info should be available without a query.
titles := []string{"Label", "UUID", "State", "SvcReps"}

hasSpaceQuery := false
hasRebuildQuery := false
for _, pool := range pools {
if hasSpaceQuery && hasRebuildQuery {
break
}

if pool.QueryMask.HasOption(daos.PoolQueryOptionSpace) {
hasSpaceQuery = true
}
if pool.QueryMask.HasOption(daos.PoolQueryOptionRebuild) {
hasRebuildQuery = true
}
}

// If any of the pools was queried, then we'll need to show more fields.
if hasSpaceQuery {
for _, t := range pools[0].Usage() {
titles = append(titles,
t.TierName+" Size",
t.TierName+" Used",
t.TierName+" Imbalance")
}
titles = append(titles, "Disabled")
titles = append(titles, "UpgradeNeeded?")
}

if hasRebuildQuery {
titles = append(titles, "Rebuild State")
}

formatter := txtfmt.NewTableFormatter(titles...)

var table []txtfmt.TableRow
for _, pool := range pools {
table = append(table, poolListCreateRowVerbose(pool, hasSpaceQuery, hasRebuildQuery))
}

fmt.Fprintln(out, formatter.Format(table))

return nil
}

// PrintPoolList generates a human-readable representation of the supplied
// slice of daos.PoolInfo structs and writes it to the supplied io.Writer.
func PrintPoolList(pools []*daos.PoolInfo, out io.Writer, verbose bool) error {
if len(pools) == 0 {
fmt.Fprintln(out, "no pools in system")
return nil
}

if verbose {
return printVerbosePoolList(pools, out)
}

return printPoolList(pools, out)
}
Loading
Loading