From b14b795d36a9d25508c1c704ba01a0db5fff95cf Mon Sep 17 00:00:00 2001
From: Zeke Morton <morton30@llnl.gov>
Date: Tue, 10 Oct 2023 16:36:55 -0700
Subject: [PATCH] rq: rq2 structure with match allocate

Problem: resource query does not utilize reapi. This means that
new functionality must be implemented in both places and that
reapi is not tested with automated ci tests and code coverage.

Create a new resource query that uses reapi.
---
 resource/.gitignore               |   1 +
 resource/utilities/CMakeLists.txt |   8 +
 resource/utilities/rq2.cpp        | 631 ++++++++++++++++++++++++++++++
 resource/utilities/rq2.hpp        |  28 ++
 4 files changed, 668 insertions(+)
 create mode 100644 resource/utilities/rq2.cpp
 create mode 100644 resource/utilities/rq2.hpp

diff --git a/resource/.gitignore b/resource/.gitignore
index d4ca2a927..8e886d40e 100644
--- a/resource/.gitignore
+++ b/resource/.gitignore
@@ -6,3 +6,4 @@ schema/test/schema_test01
 schema/test/schema_test02
 utilities/grug2dot
 utilities/resource-query
+utilities/rq2
diff --git a/resource/utilities/CMakeLists.txt b/resource/utilities/CMakeLists.txt
index 8d8064ed2..7f5c6a050 100644
--- a/resource/utilities/CMakeLists.txt
+++ b/resource/utilities/CMakeLists.txt
@@ -19,6 +19,14 @@ target_link_libraries(resource-query
   Boost::filesystem
   Boost::headers
   )
+add_executable(rq2
+  rq2.cpp
+  rq2.hpp
+  )
+target_link_libraries(rq2
+  resource
+  PkgConfig::LIBEDIT
+  )
 
 add_subdirectory(test)
 
diff --git a/resource/utilities/rq2.cpp b/resource/utilities/rq2.cpp
new file mode 100644
index 000000000..3b1903251
--- /dev/null
+++ b/resource/utilities/rq2.cpp
@@ -0,0 +1,631 @@
+/*****************************************************************************\
+ * Copyright 2022 Lawrence Livermore National Security, LLC
+ * (c.f. AUTHORS, NOTICE.LLNS, LICENSE)
+ *
+ * This file is part of the Flux resource manager framework.
+ * For details, see https://github.com/flux-framework.
+ *
+ * SPDX-License-Identifier: LGPL-3.0
+\*****************************************************************************/
+
+#include "resource/reapi/bindings/c++/reapi_cli.hpp"
+#include "resource/reapi/bindings/c++/reapi_cli_impl.hpp"
+#include "rq2.hpp"
+#include <cerrno>
+#include <getopt.h>
+#include <string.h>
+#include <jansson.h>
+#include <boost/filesystem.hpp>
+#include <editline/readline.h>
+
+namespace fs = boost::filesystem;
+using namespace Flux;
+using namespace Flux::resource_model;
+using namespace Flux::resource_model::detail;
+
+typedef int cmd_func_f (resource_query_t &,
+                        std::vector<std::string> &,
+                        json_t *, std::ostream &);
+
+#define OPTIONS "L:f:W:S:P:F:g:o:p:t:r:edh"
+static const struct option longopts[] = {
+    {"load-file",        required_argument,  0, 'L'},
+    {"load-format",      required_argument,  0, 'f'},
+    {"load-allowlist",   required_argument,  0, 'W'},
+    {"match-subsystems", required_argument,  0, 'S'},
+    {"match-policy",     required_argument,  0, 'P'},
+    {"match-format",     required_argument,  0, 'F'},
+    {"graph-format",     required_argument,  0, 'g'},
+    {"graph-output",     required_argument,  0, 'o'},
+    {"prune-filters",    required_argument,  0, 'p'},
+    {"test-output",      required_argument,  0, 't'},
+    {"reserve-vtx-vec",  required_argument,  0, 'r'},
+    {"elapse-time",      no_argument,        0, 'e'},
+    {"disable-prompt",   no_argument,        0, 'd'},
+    {"help",             no_argument,        0, 'h'},
+    { 0, 0, 0, 0 },
+};
+
+struct command_t {
+    std::string name;
+    std::string abbr;
+    cmd_func_f *cmd;
+    std::string note;
+};
+
+command_t commands[] = {
+    { "match",  "m", match, "Allocate or reserve matching resources (subcmd: "
+"allocate | allocate_with_satisfiability | allocate_orelse_reserve) | "
+"satisfiability: "
+"resource-query> match allocate jobspec"},
+    { "help", "h", help, "Print help message: resource-query> help" },
+    { "quit", "q", quit, "Quit the session: resource-query> quit" },
+    { "NA", "NA", (cmd_func_f *)NULL, "NA" }
+};
+
+
+static void usage (int code)
+{
+    std::cerr <<
+"usage: resource-query [OPTIONS...]\n"
+"\n"
+"Command-line utility that takes in an HPC resource request written in\n"
+"Flux's Canonical Job Specification (or simply a jobspec) (RFC 14) and\n"
+"selects the best-matching compute and other resources in accordance\n"
+"with a selection policy.\n"
+"\n"
+"Read in a resource-graph generation recipe written in the GRUG format\n"
+"and populate the resource-graph data store representing the compute and\n"
+"other HPC resources and their relationships (RFC 4).\n"
+"\n"
+"Provide a simple command-line interface (cli) to allow users to allocate\n"
+"or reserve the resource set in this resource-graph data store \n"
+"using a jobspec as an input.\n"
+"Traverse the resource graph in a predefined order for resource selection.\n"
+"Currently only support one traversal type: depth-first traversal on the\n"
+"dominant subsystem and up-walk traversal on one or more auxiliary \n"
+"subsystems.\n"
+"\n"
+"OPTIONS allow for using a predefined matcher that is configured\n"
+"to use a different set of subsystems as its dominant and/or auxiliary\n"
+"ones to perform the matches on.\n"
+"\n"
+"OPTIONS also allow for instantiating a different resource-matching\n"
+"selection policy--e.g., select resources with high or low IDs first.\n"
+"\n"
+"OPTIONS allow for exporting the filtered graph of the used matcher\n"
+"in a selected graph format at the end of the cli session.\n"
+"\n"
+"To see cli commands, type in \"help\" in the cli: i.e.,\n"
+"  % resource-query> help\n"
+"\n"
+"\n"
+"\n"
+"OPTIONS:\n"
+"    -h, --help\n"
+"            Display this usage information\n"
+"\n"
+"    -L, --load-file=filepath\n"
+"            Input file from which to load the resource graph data store\n"
+"            (default=conf/default)\n"
+"\n"
+"    -f, --load-format=<grug|hwloc|jgf|rv1exec>\n"
+"            Format of the load file (default=grug)\n"
+"\n"
+"    -W, --load-allowlist=<resource1[,resource2[,resource3...]]>\n"
+"            Allowlist of resource types to be loaded\n"
+"            Resources that are not included in this list will be filtered out\n"
+"\n"
+"    -S, --match-subsystems="
+         "<CA|IBA|IBBA|PFS1BA|PA|C+IBA|C+PFS1BA|C+PA|IB+IBBA|"
+              "C+P+IBA|VA|V+PFS1BA|ALL>\n"
+"            Set the predefined matcher to use. Available matchers are:\n"
+"                CA: Containment Aware\n"
+"                IBA: InfiniBand connection-Aware\n"
+"                IBBA: InfiniBand Bandwidth-Aware\n"
+"                PFS1BA: Parallel File System 1 Bandwidth-aware\n"
+"                PA: Power-Aware\n"
+"                C+IBA: Containment- and InfiniBand connection-Aware\n"
+"                C+PFS1BA: Containment- and PFS1 Bandwidth-Aware\n"
+"                C+PA: Containment- and Power-Aware\n"
+"                IB+IBBA: InfiniBand connection and Bandwidth-Aware\n"
+"                C+P+IBA: Containment-, Power- and InfiniBand connection-Aware\n"
+"                VA: Virtual Hierarchy-Aware\n"
+"                V+PFS1BA: Virtual Hierarchy and PFS1 Bandwidth-Aware \n"
+"                ALL: Aware of everything.\n"
+"            (default=CA).\n"
+"\n"
+"    -P, --match-policy=<low|high|lonode|hinode|lonodex|hinodex|first|locality|variation>\n"
+"            Set the resource match selection policy. Available policies are:\n"
+"                low: Select resources with low ID first\n"
+"                high: Select resources with high ID first\n"
+"                lonode: Select resources with lowest node ID first, \n"
+"                        low ID first otherwise (e.g., node-local resource types) \n"
+"                hinode: Select resources with highest node ID first, \n"
+"                        high ID first otherwise (e.g., node-local resource types) \n"
+"                lonodex: Same as lonode except each node is exclusively allocated \n"
+"                hinodex: Same as hinode except each node is exclusively allocated \n"
+"                first: Select the first matching resources and stop the search\n"
+"                locality: Select contiguous resources first in their ID space\n"
+"                variation: Allocate resources based on performance classes.\n"
+"                                (perf_class must be set using set-property).\n"
+"                (default=first).\n"
+"\n"
+"    -F, --match-format=<simple|pretty_simple|jgf|rlite|rv1|rv1_nosched>\n"
+"            Specify the emit format of the matched resource set.\n"
+"            (default=simple).\n"
+"\n"
+"    -p, --prune-filters=<HL-resource1:LL-resource1[,HL-resource2:LL-resource2...]...]>\n"
+"            Install a planner-based filter at each High-Level (HL) resource\n"
+"                vertex which tracks the state of the Low-Level (LL) resources\n"
+"                in aggregate, residing under its subtree. If a jobspec requests\n"
+"                1 node with 4 cores, and the visiting compute-node vertex has\n"
+"                only a total of 2 available cores in aggregate at its\n"
+"                subtree, this filter allows the traverser to prune a further descent\n"
+"                to accelerate the search.\n"
+"                Use the ALL keyword for HL-resource if you want LL-resource to be\n"
+"                tracked at all of the HL-resource vertices. Examples:\n"
+"                    rack:node,node:core\n"
+"                    ALL:core,cluster:node,rack:node\n"
+"                (default=ALL:core).\n"
+"\n"
+"    -g, --graph-format=<dot|graphml>\n"
+"            Specify the graph format of the output file\n"
+"            (default=dot).\n"
+"\n"
+"    -r, --reserve-vtx-vec=<size>\n"
+"            Reserve the graph vertex size to optimize resource graph loading.\n"
+"            The size value must be a non-zero integer up to 2000000.\n"
+"\n"
+"    -e, --elapse-time\n"
+"            Print the elapse time per scheduling operation.\n"
+"\n"
+"    -d, --disable-prompt\n"
+"            Don't print the prompt.\n"
+"\n"
+"    -o, --graph-output=<basename>\n"
+"            Set the basename of the graph output file\n"
+"            For AT&T Graphviz dot, <basename>.dot\n"
+"            For GraphML, <basename>.graphml.\n"
+"\n"
+"    -t, --test-output=<filename>\n"
+"            Set the output filename where allocated or reserved resource\n"
+"            information is stored into.\n"
+"\n";
+    exit (code);
+}
+
+static int string_to_graph_format (std::string st, Flux::resource_model::emit_format_t &format)
+{
+    int rc = 0;
+    if (boost::iequals (st, std::string ("dot")))
+        format = Flux::resource_model::emit_format_t::GRAPHVIZ_DOT;
+    else if (boost::iequals (st, std::string ("graphml")))
+        format = Flux::resource_model::emit_format_t::GRAPH_ML;
+    else
+        rc = -1;
+    return rc;
+}
+
+static int graph_format_to_ext (Flux::resource_model::emit_format_t format, std::string &st)
+{
+    int rc = 0;
+    switch (format) {
+    case Flux::resource_model::emit_format_t::GRAPHVIZ_DOT:
+        st = "dot";
+        break;
+    case Flux::resource_model::emit_format_t::GRAPH_ML:
+        st = "graphml";
+        break;
+    default:
+        rc = -1;
+    }
+    return rc;
+}
+
+static void print_sat_info (resource_query_t &ctx,
+                            std::ostream &out, bool sat, double elapse,
+                            bool print_elapse)
+{
+    unsigned int pre = reapi_cli_t::preorder_count (&ctx);
+    unsigned int post = reapi_cli_t::postorder_count (&ctx);
+    std::string satstr = sat? "Satisfiable" : "Unsatisfiable";
+    out << "INFO:" << " =============================" << std::endl;
+    out << "INFO: " << satstr << " request" << std::endl;
+    out << "INFO:" << " =============================" << std::endl;
+    if (print_elapse) {
+        std::cout << "INFO:" << " ELAPSE=" << std::to_string (elapse)
+                  << std::endl;
+        std::cout << "INFO:" << " PREORDER VISIT COUNT=" << pre
+                  << std::endl;
+        std::cout << "INFO:" << " POSTORDER VISIT COUNT=" << post
+                  << std::endl;
+    }
+}
+
+static void print_schedule_info (resource_query_t &ctx,
+                                 std::ostream &out, uint64_t jobid,
+                                 const std::string &jobspec_fn, bool matched,
+                                 int64_t at, bool sat, double elapse,
+                                 bool print_elapse)
+{
+    unsigned int pre = reapi_cli_t::preorder_count (&ctx);
+    unsigned int post = reapi_cli_t::postorder_count (&ctx);
+    if (matched) {
+        job_lifecycle_t st;
+        std::shared_ptr<job_info_t> job = nullptr;
+        std::string mode = (at == 0)? "ALLOCATED" : "RESERVED";
+        std::string scheduled_at = (at == 0)? "Now" : std::to_string (at);
+        out << "INFO:" << " =============================" << std::endl;
+        out << "INFO:" << " JOBID=" << jobid << std::endl;
+        out << "INFO:" << " RESOURCES=" << mode << std::endl;
+        out << "INFO:" << " SCHEDULED AT=" << scheduled_at << std::endl;
+        if (print_elapse) {
+            std::cout << "INFO:" << " ELAPSE=" << std::to_string (elapse)
+                      << std::endl;
+            std::cout << "INFO:" << " PREORDER VISIT COUNT=" << pre
+                      << std::endl;
+            std::cout << "INFO:" << " POSTORDER VISIT COUNT=" << post
+                      << std::endl;
+        }
+        out << "INFO:" << " =============================" << std::endl;
+        st = (at == 0)? job_lifecycle_t::ALLOCATED : job_lifecycle_t::RESERVED;
+        if (reapi_cli_t::info(&ctx, jobid, job) == 0)
+            job->jobspec_fn = jobspec_fn;
+    } else {
+        out << "INFO:" << " =============================" << std::endl;
+        out << "INFO: " << "No matching resources found" << std::endl;
+        if (!sat)
+            out << "INFO: " << "Unsatisfiable request" << std::endl;
+        out << "INFO:" << " JOBID=" << jobid << std::endl;
+        if (print_elapse) {
+            out << "INFO:" << " ELAPSE=" << std::to_string (elapse)
+                << std::endl;
+            std::cout << "INFO:" << " PREORDER VISIT COUNT=" << pre
+                      << std::endl;
+            std::cout << "INFO:" << " POSTORDER VISIT COUNT=" << post
+                      << std::endl;
+        }
+        out << "INFO:" << " =============================" << std::endl;
+    }
+}
+
+bool r_fname_set (json_t *params)
+{
+    return (json_object_get (params, "r_fname") != NULL);
+}
+
+std::ofstream open_fs (json_t *params) {
+    std::string r_fname = json_string_value (json_object_get (params, "r_fname"));
+    std::ofstream r_out; 
+
+    r_out.exceptions (std::ofstream::failbit |
+                      std::ofstream::badbit);
+    
+    r_out.open (r_fname);
+
+    return r_out;
+}
+int match (resource_query_t &ctx, std::vector<std::string> &args,
+           json_t *params, std::ostream &out)
+{
+    int rc = 0;
+    int64_t at = 0;
+    bool orelse_reserve = false;
+    bool reserved = false;
+    bool sat = true;
+    bool matched = true;
+    std::string R = "";
+    double ov = 0.0;
+    std::stringstream o;
+    struct timeval st, et; 
+        
+    if ( (gettimeofday (&st, NULL)) < 0) {
+        std::cerr << "ERROR: gettimeofday: " << strerror (errno) << std::endl;
+        return 0;
+    }
+
+    if (args.size () != 3) {
+        std::cerr << "ERROR: malformed command" << std::endl;
+        return 0;
+    }
+    std::string subcmd = args[1];
+    if (!(subcmd == "allocate" || subcmd == "allocate_orelse_reserve"
+          || subcmd == "allocate_with_satisfiability"
+          || subcmd == "satisfiability")) {
+        std::cerr << "ERROR: unknown subcmd " << args[1] << std::endl;
+        return 0;
+    }
+
+    if (subcmd == "allocate_orelse_reserve") {
+        orelse_reserve = true;
+    }
+
+    uint64_t jobid = ctx.get_job_counter ();
+    std::string &jobspec_fn = args[2];
+    std::ifstream jobspec_in (jobspec_fn);
+    if (!jobspec_in) {
+        std::cerr << "ERROR: can't open " << jobspec_fn << std::endl;
+        return 0;
+    }
+    std::string jobspec ( (std::istreambuf_iterator<char> (jobspec_in) ),
+                        (std::istreambuf_iterator<char> () ) );
+
+    rc = reapi_cli_t::match_allocate (&ctx, orelse_reserve, jobspec, jobid, reserved, R, at, ov);
+
+    // check for match success 
+    if ( (errno == ENODEV) || (errno == EBUSY) || (errno == EINVAL))
+        matched = false;
+    
+    // check for satisfiability
+    if (errno == ENODEV)
+        sat = false;
+    
+    if (reapi_cli_t::get_err_message () != "") {
+        std::cerr << reapi_cli_t::get_err_message () << std::endl;
+        reapi_cli_t::clear_err_message ();
+    }
+
+    out << R;
+
+    if ( (gettimeofday (&et, NULL)) < 0) {
+        std::cerr << "ERROR: gettimeofday: " << strerror (errno) << std::endl;
+        return 0;
+    }
+
+    if (subcmd != "satisfiability")
+        print_schedule_info (ctx, out, jobid,
+                             jobspec_fn, matched, at, sat,
+                             ov, json_object_get (params, "elapse_time"));
+    else
+        print_sat_info (ctx, out, sat, ov, 
+                        json_object_get (params, "elapse_time"));
+
+    jobspec_in.close ();
+
+    return 0;
+}
+
+int quit (resource_query_t &ctx, std::vector<std::string> &args,
+          json_t *params, std::ostream &out)
+{
+    return -1;
+}
+
+int help (resource_query_t &ctx, std::vector<std::string> &args,
+          json_t *params, std::ostream &out)
+{
+    bool multi = true;
+    bool found = false;
+    std::string cmd = "unknown";
+
+    if (args.size () == 2) {
+        multi = false;
+        cmd = args[1];
+    }
+
+    for (int i = 0; commands[i].name != "NA"; ++i) {
+        if (multi || cmd == commands[i].name || cmd == commands[i].abbr) {
+            std::cout << "INFO: " << commands[i].name << " ("
+                      << commands[i].abbr << ")" << " -- "
+                      << commands[i].note << std::endl;
+            found = true;
+        }
+    }
+    if (!multi && !found)
+        std::cout << "ERROR: unknown command: " << cmd << std::endl;
+
+    return 0;
+}
+
+cmd_func_f *find_cmd (const std::string &cmd_str)
+{
+    for (int i = 0; commands[i].name != "NA"; ++i) {
+        if (cmd_str == commands[i].name)
+            return commands[i].cmd;
+        else if (cmd_str == commands[i].abbr)
+            return commands[i].cmd;
+    }
+    return (cmd_func_f *)NULL;
+}
+
+static void process_args (json_t *options,
+                          int argc, char *argv[])
+{
+    int rc = 0;
+    int ch = 0;
+    std::string token;
+    Flux::resource_model::emit_format_t format;
+
+
+    /* set defaults specific for resource query */
+    json_object_set_new (options, "match_format", json_string ("simple"));
+
+    while ( (ch = getopt_long (argc, argv, OPTIONS, longopts, NULL)) != -1) {
+        switch (ch) {
+            case 'h': /* --help */
+                usage (0);
+                break;
+            case 'L': /* --load-file */
+                json_object_set_new (options, "load_file", json_string (optarg));
+                if (!fs::exists (optarg)) {
+                    std::cerr << "[ERROR] file does not exist for --load-file: ";
+                    std::cerr << optarg << std::endl;
+                    usage (1);
+                } else if (fs::is_directory (optarg)) {
+                    std::cerr << "[ERROR] path passed to --load-file is a directory: ";
+                    std::cerr << optarg << std::endl;
+                    usage (1);
+                }
+                break;
+            case 'f': /* --load-format */
+                json_object_set_new (options, "load_format", json_string (optarg));
+                if (!known_resource_reader (optarg)) {
+                    std::cerr << "[ERROR] unknown format for --load-format: ";
+                    std::cerr << optarg << std::endl;
+                    usage (1);
+                }
+                break;
+            case 'W': /* --hwloc-allowlist */
+                token = optarg;
+                if (token.find_first_not_of (' ') != std::string::npos) {
+                    if (!json_object_get (options,"load_allowlist")){
+                        json_object_set_new (options, "load_allowlist", json_string (""));
+                    }
+                    json_object_set_new (options, "load_allowlist",
+                                        json_pack ("s++",
+                                        json_object_get (options,"load_allowlist"),
+                                         "cluster,", token.c_str ()));
+                }
+                break;
+            case 'S': /* --match-subsystems */
+                json_object_set_new (options, "matcher_name", json_string (optarg));
+                break;
+            case 'P': /* --match-policy */
+                json_object_set_new (options, "matcher_policy", json_string (optarg));
+                break;
+            case 'F': /* --match-format */
+                json_object_set_new (options, "match_format", json_string (optarg));
+                if (!known_match_format (optarg)) {
+                    std::cerr << "[ERROR] unknown format for --match-format: ";
+                    std::cerr << optarg << std::endl;
+                    usage (1);
+                }
+                break;
+            case 'g': /* --graph-format */
+                rc = string_to_graph_format (optarg, format);
+                if ( rc != 0) {
+                    std::cerr << "[ERROR] unknown format for --graph-format: ";
+                    std::cerr << optarg << std::endl;
+                    usage (1);
+                }
+                graph_format_to_ext (format, token);
+                json_object_set_new (options, "o_fext", json_string (token.c_str ()));
+                break;
+            case 'o': /* --graph-output */
+                json_object_set_new (options, "o_fname", json_string (optarg));
+                break;
+            case 'p': /* --prune-filters */
+                token = optarg;
+                if (token.find_first_not_of (' ') != std::string::npos) {
+                    if (!json_object_get (options,"prune_filters")){
+                        json_object_set_new (options, "prune_filters", json_string (token.c_str ()));
+                    }else{
+                      json_object_set_new (options, "prune_filters",
+                                          json_pack ("s++",
+                                          json_object_get (options,"prune_filters"),
+                                           ",", token.c_str ()));
+                    }
+
+                }
+                break;
+            case 't': /* --test-output */
+                json_object_set_new (options, "r_fname", json_string (optarg));
+                break;
+            case 'r': /* --reserve-vtx-vec */
+                // If atoi fails, it defaults to 0, which is fine for us
+                json_object_set_new (options, "match_format", json_integer (atoi (optarg)));
+                if ( (atoi (optarg) < 0)
+                    || (atoi (optarg) > 2000000)) {
+                    json_object_set_new (options, "match_format", json_integer (0));
+                    std::cerr
+                        << "WARN: out of range specified for --reserve-vtx-vec: ";
+                    std::cerr << optarg << std::endl;
+                }
+                break;
+            case 'e': /* --elapse-time */
+                json_object_set_new (options, "elapse_time", json_true ());
+                break;
+            case 'd': /* --disable-prompt */
+                json_object_set_new (options, "disable_prompt", json_true ());
+                break;
+            default:
+                usage (1);
+                break;
+        }
+    }
+
+    if (optind != argc)
+        usage (1);
+}
+
+void get_rgraph (std::string &rgraph, json_t* options)
+{
+    const char *load_file = json_string_value (json_object_get (options, "load_file"));
+    if (load_file == NULL) {
+      load_file = "conf/default";
+    }
+    std::ifstream ifs (load_file);
+    rgraph = std::string( (std::istreambuf_iterator<char> (ifs) ),
+                        (std::istreambuf_iterator<char> () ) );
+    ifs.close();
+}
+
+static void control_loop (resource_query_t &ctx, json_t *params,
+                          std::ostream &out)
+{
+    cmd_func_f *cmd = NULL;
+    while (1) {
+        char *line = json_object_get (params, "disable_prompt")? readline ("")
+                                               : readline ("resource-query> ");
+        if (line == NULL)
+            continue;
+        else if (*line)
+            add_history (line);
+
+        std::vector<std::string> tokens;
+        std::istringstream iss (line);
+        std::copy (std::istream_iterator<std::string> (iss),
+                   std::istream_iterator<std::string> (),
+             back_inserter (tokens));
+        free(line);
+        if (tokens.empty ())
+            continue;
+
+        std::string &cmd_str = tokens[0];
+        if (!(cmd = find_cmd (cmd_str)))
+            continue;
+        if (cmd (ctx, tokens, params, out) != 0)
+            break;
+    }
+}
+
+int main (int argc, char *argv[])
+{
+    json_t *json_options = json_object ();
+    std::string rgraph;
+    resource_query_t *ctx = nullptr;
+    int match_out, info_out = 0;
+    std::string options;
+
+    process_args (json_options, argc, argv);
+    get_rgraph (rgraph, json_options);
+    options = json_dumps (json_options, JSON_COMPACT);
+
+    try {
+        ctx = new resource_query_t (rgraph, options);
+    } catch (std::bad_alloc &e) {
+        errno = ENOMEM;
+        std::cerr << "Memory error\n";
+        return EXIT_FAILURE;
+    } catch (std::runtime_error &e) {
+        errno = EPROTO;
+        std::cerr << ": Runtime error: "
+                     + std::string (e.what ()) + "\n";
+        return EXIT_FAILURE;
+    }
+
+    std::ofstream out_file;
+
+    if (r_fname_set (json_options))
+    {
+        out_file = open_fs (json_options);
+    }
+
+    std::ostream &out = (r_fname_set (json_options))? out_file
+                                     : std::cout;
+
+    control_loop (*ctx, json_options, out);
+
+    return EXIT_SUCCESS;
+}
diff --git a/resource/utilities/rq2.hpp b/resource/utilities/rq2.hpp
new file mode 100644
index 000000000..f22326121
--- /dev/null
+++ b/resource/utilities/rq2.hpp
@@ -0,0 +1,28 @@
+/*****************************************************************************\
+ * Copyright 2022 Lawrence Livermore National Security, LLC
+ * (c.f. AUTHORS, NOTICE.LLNS, LICENSE)
+ *
+ * This file is part of the Flux resource manager framework.
+ * For details, see https://github.com/flux-framework.
+ *
+ * SPDX-License-Identifier: LGPL-3.0
+\*****************************************************************************/
+
+#include <memory>
+#include <cerrno>
+#include <vector>
+#include <map>
+#include "resource/reapi/bindings/c++/reapi_cli.hpp"
+#include "resource/reapi/bindings/c++/reapi_cli_impl.hpp"
+
+using namespace Flux;
+using namespace Flux::resource_model;
+using namespace Flux::resource_model::detail;
+
+int match (resource_query_t &ctx, std::vector<std::string> &args, json_t *params,
+           std::ostream &out);
+int help (resource_query_t &ctx, std::vector<std::string> &args, json_t *params,
+          std::ostream &out);
+int quit (resource_query_t &ctx, std::vector<std::string> &args, json_t *params,
+          std::ostream &out);
+