xtask: implement xtask run subcommand
This new subcommand provides a convenient way to run your aya application. Features include: - automatic compilation of userspace + bpf code - support for user-defined runners, defaulting to `sudo -E` for convenience - no need to run `sudo cargo run` (gross) or `sudo target/debug/myapp` (also gross), it just works - an optional `-p` flag which automatically supplies the correct --path flag to the application - support for trailing args after a final -- which are passed to the application, cargo run-stylepull/20/head
parent
509073c454
commit
d590d3a1ea
@ -0,0 +1,75 @@
|
|||||||
|
use std::{os::unix::process::CommandExt, process::Command};
|
||||||
|
|
||||||
|
use anyhow::Context as _;
|
||||||
|
use structopt::StructOpt;
|
||||||
|
|
||||||
|
use crate::build_ebpf::{build_ebpf, Architecture, Options as BuildOptions};
|
||||||
|
|
||||||
|
#[derive(StructOpt)]
|
||||||
|
pub struct Options {
|
||||||
|
/// Set the endianness of the BPF target
|
||||||
|
#[structopt(default_value = "bpfel-unknown-none", long)]
|
||||||
|
pub bpf_target: Architecture,
|
||||||
|
/// Build and run the release target
|
||||||
|
#[structopt(long)]
|
||||||
|
pub release: bool,
|
||||||
|
/// The command used to wrap your application
|
||||||
|
#[structopt(short, long, default_value = "sudo -E")]
|
||||||
|
pub runner: String,
|
||||||
|
/// A convenience flag that will supply `--path /path/to/bpf/object` to your application
|
||||||
|
#[structopt(short = "p", long)]
|
||||||
|
pub supply_path: bool,
|
||||||
|
/// Arguments to pass to your application
|
||||||
|
#[structopt(name = "args", last = true)]
|
||||||
|
pub run_args: Vec<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Build the project
|
||||||
|
fn build(opts: &Options) -> Result<(), anyhow::Error> {
|
||||||
|
let mut args = vec!["build"];
|
||||||
|
if opts.release {
|
||||||
|
args.push("--release")
|
||||||
|
}
|
||||||
|
let status = Command::new("cargo")
|
||||||
|
.args(&args)
|
||||||
|
.status()
|
||||||
|
.expect("failed to build userspace");
|
||||||
|
assert!(status.success());
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Build and run the project
|
||||||
|
pub fn run(opts: Options) -> Result<(), anyhow::Error> {
|
||||||
|
// build our ebpf program followed by our application
|
||||||
|
build_ebpf(BuildOptions {
|
||||||
|
target: opts.bpf_target,
|
||||||
|
release: opts.release,
|
||||||
|
})
|
||||||
|
.context("Error while building eBPF program")?;
|
||||||
|
build(&opts).context("Error while building userspace application")?;
|
||||||
|
|
||||||
|
// profile we are building (release or debug)
|
||||||
|
let profile = if opts.release { "release" } else { "debug" };
|
||||||
|
let bin_path = format!("target/{}/{{project-name}}", profile);
|
||||||
|
let bpf_path = format!("target/{}/{}/{{project-name}}", opts.bpf_target, profile);
|
||||||
|
|
||||||
|
// arguments to pass to the application
|
||||||
|
let mut run_args: Vec<_> = opts.run_args.iter().map(String::as_str).collect();
|
||||||
|
if opts.supply_path {
|
||||||
|
run_args.push("--path");
|
||||||
|
run_args.push(bpf_path.as_str());
|
||||||
|
};
|
||||||
|
|
||||||
|
// configure args
|
||||||
|
let mut args: Vec<_> = opts.runner.trim().split_terminator(" ").collect();
|
||||||
|
args.push(bin_path.as_str());
|
||||||
|
args.append(&mut run_args);
|
||||||
|
|
||||||
|
// spawn the command
|
||||||
|
let err = Command::new(args.get(0).expect("No first argument"))
|
||||||
|
.args(args.iter().skip(1))
|
||||||
|
.exec();
|
||||||
|
|
||||||
|
// we shouldn't get here unless the command failed to spawn
|
||||||
|
Err(anyhow::Error::from(err).context(format!("Failed to run `{}`", args.join(" "))))
|
||||||
|
}
|
Loading…
Reference in New Issue