#![deny(
missing_debug_implementations,
trivial_numeric_casts,
unstable_features,
unused_import_braces,
unused_qualifications
)]
#[macro_use]
extern crate structopt;
extern crate clap;
extern crate digest;
extern crate futures;
extern crate http;
extern crate hyper;
extern crate libflate;
extern crate notify;
extern crate pbr;
extern crate reqwest;
extern crate serde;
extern crate sha1;
extern crate sha2;
extern crate tar;
extern crate tempfile;
extern crate toml;
#[macro_use]
extern crate serde_derive;
#[macro_use]
extern crate serde_json;
extern crate base_x;
extern crate handlebars;
extern crate indexmap;
extern crate regex;
extern crate unicode_categories;
extern crate walkdir;
extern crate websocket;
#[macro_use]
extern crate lazy_static;
extern crate directories;
extern crate percent_encoding;
extern crate parity_wasm;
#[macro_use]
extern crate log;
extern crate env_logger;
extern crate rustc_demangle;
extern crate ansi_term;
extern crate cargo_metadata;
extern crate memmap;
extern crate semver;
extern crate atty;
extern crate open;
#[macro_use]
extern crate failure;
extern crate mime_guess;
mod cargo_shim;
#[macro_use]
mod utils;
mod build;
mod chrome_devtools;
mod cmd_build;
mod cmd_deploy;
mod cmd_prepare_emscripten;
mod cmd_start;
mod cmd_test;
mod config;
mod deployment;
mod emscripten;
mod error;
mod http_utils;
mod package;
mod project_dirs;
mod test_chromium;
mod wasm;
mod wasm_context;
mod wasm_export_main;
mod wasm_export_table;
mod wasm_gc;
mod wasm_hook_grow;
mod wasm_inline_js;
mod wasm_intrinsics;
mod wasm_js_export;
mod wasm_js_snippet;
mod wasm_runtime;
use std::ffi::OsStr;
use std::net::{IpAddr, ToSocketAddrs};
use std::path::PathBuf;
use build::{Backend, BuildArgs};
use cargo_shim::MessageFormat;
use error::Error;
use wasm_runtime::RuntimeKind;
#[derive(Debug, StructOpt)]
#[structopt(name = "cargo-web")]
#[structopt(about = "A `cargo` subcommand for the client-side web.")]
#[structopt(raw(global_setting = "structopt::clap::AppSettings::ColoredHelp"))]
#[structopt(raw(setting = "structopt::clap::AppSettings::VersionlessSubcommands"))]
#[structopt(rename_all = "kebab-case")]
pub enum CargoWebOpts {
Build(BuildOpts),
Check(CheckOpts),
Deploy(DeployOpts),
PrepareEmscripten(PrepareEmscriptenOpts),
Start(StartOpts),
Test(TestOpts),
#[doc(hidden)]
#[structopt(raw(setting = "structopt::clap::AppSettings::Hidden"))]
__Nonexhaustive,
}
pub fn run(cfg: CargoWebOpts) -> Result<(), Error> {
match cfg {
CargoWebOpts::Build(BuildOpts {
build_args,
build_target,
ext,
}) => cmd_build::command_build(BuildArgs::new(build_args, ext, build_target)?),
CargoWebOpts::Check(CheckOpts {
build_args,
build_target,
ext,
}) => cmd_build::command_check(BuildArgs::new(build_args, ext, build_target)?),
CargoWebOpts::Deploy(DeployOpts { build_args, output }) => {
cmd_deploy::command_deploy(build_args.into(), output)
}
CargoWebOpts::PrepareEmscripten(_) => cmd_prepare_emscripten::command_prepare_emscripten(),
CargoWebOpts::Start(StartOpts {
build_args,
build_target,
auto_reload,
open,
port,
host,
}) => cmd_start::command_start(
BuildArgs::from(build_args).with_target(build_target),
host,
port,
open,
auto_reload,
),
CargoWebOpts::Test(TestOpts {
build_args,
nodejs,
no_run,
passthrough,
}) => {
let pass_os = passthrough.iter().map(OsStr::new).collect::<Vec<_>>();
cmd_test::command_test(build_args.into(), nodejs, no_run, &pass_os)
}
CargoWebOpts::__Nonexhaustive => unreachable!(),
}
}
#[derive(Debug, StructOpt)]
#[structopt(rename_all = "kebab-case")]
pub struct BuildOpts {
#[structopt(flatten)]
build_args: Build,
#[structopt(flatten)]
ext: BuildExt,
#[structopt(flatten)]
build_target: Target,
}
#[derive(Debug, StructOpt)]
#[structopt(rename_all = "kebab-case")]
pub struct CheckOpts {
#[structopt(flatten)]
build_args: Build,
#[structopt(flatten)]
ext: BuildExt,
#[structopt(flatten)]
build_target: Target,
}
#[derive(Debug, StructOpt)]
#[structopt(rename_all = "kebab-case")]
pub struct DeployOpts {
#[structopt(short = "o", long, parse(from_os_str))]
output: Option<PathBuf>,
#[structopt(flatten)]
build_args: Build,
}
#[derive(Debug, StructOpt)]
#[structopt(rename_all = "kebab-case")]
pub struct PrepareEmscriptenOpts {
#[doc(hidden)]
#[structopt(raw(set = "structopt::clap::ArgSettings::Hidden"))]
__reserved: bool,
}
#[derive(Debug, StructOpt)]
#[structopt(rename_all = "kebab-case")]
pub struct StartOpts {
#[structopt(
long,
parse(try_from_str = "resolve_host"),
default_value = "localhost"
)]
host: IpAddr,
#[structopt(long, default_value = "8000")]
port: u16,
#[structopt(long)]
open: bool,
#[structopt(long)]
auto_reload: bool,
#[structopt(flatten)]
build_target: Target,
#[structopt(flatten)]
build_args: Build,
}
#[derive(Debug, StructOpt)]
#[structopt(rename_all = "kebab-case")]
pub struct TestOpts {
#[structopt(long)]
no_run: bool,
#[structopt(long)]
nodejs: bool,
#[structopt(flatten)]
build_args: Build,
passthrough: Vec<String>,
}
#[derive(Debug, StructOpt)]
#[structopt(rename_all = "kebab-case")]
struct Target {
#[structopt(long, group = "target_type")]
lib: bool,
#[structopt(long, group = "target_type")]
bin: Option<String>,
#[structopt(long, group = "target_type")]
example: Option<String>,
#[structopt(long, group = "target_type")]
test: Option<String>,
#[structopt(long, group = "target_type")]
bench: Option<String>,
}
impl Default for Target {
fn default() -> Self {
Self {
lib: false,
bin: None,
example: None,
test: None,
bench: None,
}
}
}
#[derive(Debug, StructOpt)]
#[structopt(rename_all = "kebab-case")]
struct BuildExt {
#[structopt(
long,
default_value = "human",
parse(try_from_str),
raw(possible_values = "&[\"human\", \"json\"]"),
raw(set = "structopt::clap::ArgSettings::NextLineHelp")
)]
message_format: MessageFormat,
#[structopt(
long,
parse(try_from_str),
raw(set = "structopt::clap::ArgSettings::NextLineHelp")
)]
runtime: Option<RuntimeKind>,
}
impl Default for BuildExt {
fn default() -> Self {
Self {
message_format: MessageFormat::Json,
runtime: None,
}
}
}
#[derive(Debug, StructOpt)]
#[structopt(rename_all = "kebab-case")]
struct Build {
#[structopt(short = "p", long)]
pub package: Option<String>,
#[structopt(long, group = "build_features")]
pub features: Option<String>,
#[structopt(long, group = "build_features")]
pub all_features: bool,
#[structopt(long)]
pub no_default_features: bool,
#[structopt(long)]
pub use_system_emscripten: bool,
#[structopt(long)]
pub release: bool,
#[structopt(
long,
parse(try_from_str),
raw(
possible_values = "&[\"wasm32-unknown-unknown\", \"wasm32-unknown-emscripten\", \"asmjs-unknown-emscripten\"]"
),
raw(set = "structopt::clap::ArgSettings::NextLineHelp")
)]
pub target: Option<Backend>,
#[structopt(short = "v", long)]
pub verbose: bool,
}
impl Default for Build {
fn default() -> Self {
Self {
package: None,
features: None,
all_features: false,
no_default_features: false,
use_system_emscripten: false,
release: false,
target: None,
verbose: false,
}
}
}
fn resolve_host(host: &str) -> std::io::Result<IpAddr> {
(host, 0)
.to_socket_addrs()
.map(|itr| itr.map(|a| a.ip()).collect::<Vec<_>>()[0])
}