Skip to content

Commit

Permalink
job-list: support ranks constraint
Browse files Browse the repository at this point in the history
Problem: The job-list module supports filtering jobs only by hostname,
but there are times when filtering by ranks would be more convenient.

Add a "ranks" constraint operator with values in RFC 22 Idset string
form to filter jobs based on assigned ranks.
  • Loading branch information
grondo committed Jun 19, 2024
1 parent 3599702 commit 2b207b0
Show file tree
Hide file tree
Showing 4 changed files with 96 additions and 1 deletion.
1 change: 1 addition & 0 deletions src/modules/job-list/job_data.c
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ void job_destroy (void *data)
free (job->ranks);
free (job->nodelist);
hostlist_destroy (job->nodelist_hl);
idset_destroy (job->ranks_idset);
json_decref (job->annotations);
grudgeset_destroy (job->dependencies);
json_decref (job->jobspec);
Expand Down
2 changes: 2 additions & 0 deletions src/modules/job-list/job_data.h
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
#include <jansson.h>

#include "src/common/libhostlist/hostlist.h"
#include "src/common/libidset/idset.h"
#include "src/common/libutil/grudgeset.h"
#include "src/common/libczmqcontainers/czmq_containers.h"

Expand Down Expand Up @@ -56,6 +57,7 @@ struct job {
char *ranks;
char *nodelist;
struct hostlist *nodelist_hl; /* cache of nodelist in hl form */
struct idset *ranks_idset; /* cache of ranks in idset form */
double expiration;
int wait_status;
bool success;
Expand Down
91 changes: 91 additions & 0 deletions src/modules/job-list/match.c
Original file line number Diff line number Diff line change
Expand Up @@ -514,6 +514,95 @@ static struct list_constraint *create_hostlist_constraint (
return NULL;
}

static int match_ranks (struct list_constraint *c,
const struct job *job,
unsigned int *comparisons,
flux_error_t *errp)
{
struct idset *idset = zlistx_first (c->values);
size_t n, m;

/* ranks may not exist if job never ran */
if (!job->ranks)
return 0;
if (!job->ranks_idset) {
/* hack to remove const */
struct job *jobtmp = (struct job *)job;
if (!(jobtmp->ranks_idset = idset_decode (job->ranks)))
return 0;
}
/* Account for all ranks being compared before calling
* inc_check_comparison. This is the smallest of the job or
* comparison idset
*/
m = idset_count (job->ranks_idset);
n = idset_count (idset);
*comparisons += (m < n ? m : n) - 1;
if (inc_check_comparison (c->mctx, comparisons, errp) < 0)
return -1;
return idset_has_intersection (job->ranks_idset, idset);
}


/* zlistx_set_destructor */
static void wrap_idset_destroy (void **item)
{
if (item) {
struct idset *idset = *item;
idset_destroy (idset);
(*item) = NULL;
}
}

static struct list_constraint *create_ranks_constraint (
struct match_ctx *mctx,
json_t *values,
flux_error_t *errp)
{
struct list_constraint *c;
struct idset *idset = NULL;
json_t *entry;
size_t index;

if (!(c = list_constraint_new (mctx,
match_ranks,
wrap_idset_destroy,
errp)))
return NULL;

if (!(idset = idset_create (0, IDSET_FLAG_AUTOGROW))) {
errprintf (errp, "failed to create idset structure");
goto error;
}
json_array_foreach (values, index, entry) {
const char *ids;
idset_error_t error;
if (!json_is_string (entry)
|| !(ids = json_string_value (entry))) {
errprintf (errp, "ranks value must be a string");
goto error;
}
if (idset_decode_add (idset, ids, -1, &error) < 0) {
errprintf (errp, "ranks value '%s': %s", ids, error.text);
goto error;
}
}
if (idset_count (idset) > mctx->max_hostlist) {
errprintf (errp, "too many ranks specified");
goto error;
}
if (!zlistx_add_end (c->values, idset)) {
errprintf (errp, "failed to append idset structure");
goto error;
}
return c;
error:
idset_destroy (idset);
list_constraint_destroy (c);
return NULL;
}


static int match_timestamp (struct list_constraint *c,
const struct job *job,
unsigned int *comparisons,
Expand Down Expand Up @@ -753,6 +842,8 @@ struct list_constraint *list_constraint_create (struct match_ctx *mctx,
return create_results_constraint (mctx, values, errp);
else if (streq (op, "hostlist"))
return create_hostlist_constraint (mctx, values, errp);
else if (streq (op, "ranks"))
return create_ranks_constraint (mctx, values, errp);
else if (streq (op, "t_submit")
|| streq (op, "t_depend")
|| streq (op, "t_run")
Expand Down
3 changes: 2 additions & 1 deletion src/modules/job-list/state_match.c
Original file line number Diff line number Diff line change
Expand Up @@ -367,7 +367,8 @@ struct state_constraint *state_constraint_create (json_t *constraint, flux_error
if (streq (op, "userid")
|| streq (op, "name")
|| streq (op, "queue")
|| streq (op, "hostlist"))
|| streq (op, "hostlist")
|| streq (op, "ranks"))
return state_constraint_new (match_maybe, NULL, errp);
else if (streq (op, "results"))
return state_constraint_new (match_result, NULL, errp);
Expand Down

0 comments on commit 2b207b0

Please sign in to comment.