Skip to content

Commit

Permalink
Update spin up to use new spin-trigger semantics
Browse files Browse the repository at this point in the history
Signed-off-by: Lann Martin <lann.martin@fermyon.com>
  • Loading branch information
lann committed Sep 13, 2022
1 parent 1b69243 commit 9b5a810
Showing 1 changed file with 37 additions and 39 deletions.
76 changes: 37 additions & 39 deletions src/commands/up.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ use anyhow::{bail, Context, Result};
use clap::{CommandFactory, Parser};
use spin_loader::bindle::BindleConnectionInfo;
use spin_manifest::ApplicationTrigger;
use spin_trigger::cli::{SPIN_LOCKED_URL, SPIN_WORKING_DIR};
use tempfile::TempDir;

use crate::opts::*;
Expand Down Expand Up @@ -77,21 +78,23 @@ pub struct UpCommand {
)]
pub insecure: bool,

/// Pass an environment variable (key=value) to all components of the application.
#[clap(short = 'e', long = "env", parse(try_from_str = parse_env_var))]
pub env: Vec<(String, String)>,

/// Temporary directory for the static assets of the components.
#[clap(long = "temp")]
pub tmp: Option<PathBuf>,

/// Set the static assets of the components in the temporary directory as writable.
#[clap(long = "allow-transient-write")]
pub allow_transient_write: bool,

/// All other args, to be passed through to the trigger
#[clap(hide = true)]
pub trigger_args: Vec<OsString>,
}

impl UpCommand {
pub async fn run(self) -> Result<()> {
// For displaying help, first print `spin up`'s own usage text, then
// attempt to load an app and print trigger-type-specific usage.
let help = self.help;
if help {
Self::command()
Expand All @@ -117,49 +120,43 @@ impl UpCommand {
};
let working_dir = working_dir_holder.path();

let app = match (&self.app, &self.bindle) {
let mut app = match (&self.app, &self.bindle) {
(app, None) => {
let manifest_file = app
.as_deref()
.unwrap_or_else(|| DEFAULT_MANIFEST_FILE.as_ref());
let bindle_connection = self.bindle_connection();
spin_loader::from_file(
manifest_file,
working_dir,
&bindle_connection,
self.allow_transient_write,
)
.await?
spin_loader::from_file(manifest_file, working_dir, &bindle_connection).await?
}
(None, Some(bindle)) => match &self.server {
Some(server) => {
spin_loader::from_bindle(
bindle,
server,
working_dir,
self.allow_transient_write,
)
.await?
}
Some(server) => spin_loader::from_bindle(bindle, server, working_dir).await?,
_ => bail!("Loading from a bindle requires a Bindle server URL"),
},
(Some(_), Some(_)) => bail!("Specify only one of app file or bindle ID"),
};

let manifest_url = match app.info.origin {
spin_manifest::ApplicationOrigin::File(path) => {
format!("file:{}", path.canonicalize()?.to_string_lossy())
}
spin_manifest::ApplicationOrigin::Bindle { id, server } => {
format!("bindle+{}?id={}", server, id)
// Apply --env to component environments
if !self.env.is_empty() {
for component in app.components.iter_mut() {
component.wasm.environment.extend(self.env.iter().cloned());
}
};
}

let trigger_type = match app.info.trigger {
ApplicationTrigger::Http(_) => "http",
ApplicationTrigger::Redis(_) => "redis",
};

// Build and write app lock file
let locked_app = spin_trigger::locked::build_locked_app(app, working_dir)?;
let locked_path = working_dir.join("spin.lock");
let locked_app_contents =
serde_json::to_vec_pretty(&locked_app).context("failed to serialize locked app")?;
std::fs::write(&locked_path, locked_app_contents)
.with_context(|| format!("failed to write {:?}", locked_path))?;
let locked_url = format!("file://{}", locked_path.to_string_lossy());

// For `spin up --help`, we just want the executor to dump its own argument usage info
let trigger_args = if self.help {
vec![OsString::from("--help-args-only")]
} else {
Expand All @@ -170,24 +167,16 @@ impl UpCommand {
// via hard-link. I think it should be fine as long as we aren't `setuid`ing this binary.
let mut cmd = std::process::Command::new(std::env::current_exe().unwrap());
cmd.arg("trigger")
.env("SPIN_WORKING_DIR", working_dir)
.env("SPIN_MANIFEST_URL", manifest_url)
.env("SPIN_TRIGGER_TYPE", trigger_type)
.env(
"SPIN_ALLOW_TRANSIENT_WRITE",
self.allow_transient_write.to_string(),
)
.env(SPIN_WORKING_DIR, working_dir)
.env(SPIN_LOCKED_URL, locked_url)
.arg(trigger_type)
.args(trigger_args);

if let Some(bindle_server) = self.server {
cmd.env(BINDLE_URL_ENV, bindle_server);
}

tracing::trace!("Running trigger executor: {:?}", cmd);

let mut child = cmd.spawn().context("Failed to execute trigger")?;

// Terminate trigger executor if `spin up` itself receives a termination signal
#[cfg(not(windows))]
{
// https://github.com/nix-rust/nix/issues/656
Expand Down Expand Up @@ -232,3 +221,12 @@ impl WorkingDirectory {
}
}
}

// Parse the environment variables passed in `key=value` pairs.
fn parse_env_var(s: &str) -> Result<(String, String)> {
let parts: Vec<_> = s.splitn(2, '=').collect();
if parts.len() != 2 {
bail!("Environment variable must be of the form `key=value`");
}
Ok((parts[0].to_owned(), parts[1].to_owned()))
}

0 comments on commit 9b5a810

Please sign in to comment.