|
use clap::{Args, Parser, Subcommand}; |
|
use fern::colors::{Color, ColoredLevelConfig}; |
|
use futures::executor::block_on; |
|
use graph_craft::document::*; |
|
use graph_craft::graphene_compiler::{Compiler, Executor}; |
|
use graph_craft::proto::ProtoNetwork; |
|
use graph_craft::util::load_network; |
|
use graph_craft::wasm_application_io::EditorPreferences; |
|
use graphene_core::text::FontCache; |
|
use graphene_std::application_io::{ApplicationIo, NodeGraphUpdateMessage, NodeGraphUpdateSender, RenderConfig}; |
|
use graphene_std::wasm_application_io::{WasmApplicationIo, WasmEditorApi}; |
|
use interpreted_executor::dynamic_executor::DynamicExecutor; |
|
use interpreted_executor::util::wrap_network_in_scope; |
|
use std::error::Error; |
|
use std::path::PathBuf; |
|
use std::sync::Arc; |
|
|
|
struct UpdateLogger {} |
|
|
|
impl NodeGraphUpdateSender for UpdateLogger { |
|
fn send(&self, message: NodeGraphUpdateMessage) { |
|
println!("{message:?}"); |
|
} |
|
} |
|
|
|
#[derive(Debug, Parser)] |
|
#[clap(name = "graphene-cli", version)] |
|
pub struct App { |
|
#[clap(flatten)] |
|
global_opts: GlobalOpts, |
|
|
|
#[clap(subcommand)] |
|
command: Command, |
|
} |
|
|
|
#[derive(Debug, Subcommand)] |
|
enum Command { |
|
|
|
Compile { |
|
|
|
#[clap(long, short = 'p')] |
|
print_proto: bool, |
|
|
|
|
|
document: PathBuf, |
|
}, |
|
|
|
Run { |
|
|
|
document: PathBuf, |
|
|
|
|
|
image: Option<PathBuf>, |
|
|
|
|
|
#[clap(long, short = 'l')] |
|
run_loop: bool, |
|
}, |
|
} |
|
|
|
#[derive(Debug, Args)] |
|
struct GlobalOpts { |
|
|
|
#[clap(long, short, global = true, action = clap::ArgAction::Count)] |
|
verbose: u8, |
|
} |
|
|
|
#[tokio::main] |
|
async fn main() -> Result<(), Box<dyn Error>> { |
|
let app = App::parse(); |
|
|
|
let log_level = app.global_opts.verbose; |
|
|
|
init_logging(log_level); |
|
|
|
let document_path = match app.command { |
|
Command::Compile { ref document, .. } => document, |
|
Command::Run { ref document, .. } => document, |
|
}; |
|
|
|
let document_string = std::fs::read_to_string(document_path).expect("Failed to read document"); |
|
|
|
log::info!("creating gpu context",); |
|
let mut application_io = block_on(WasmApplicationIo::new()); |
|
|
|
if let Command::Run { image: Some(ref image_path), .. } = app.command { |
|
application_io.resources.insert("null".to_string(), Arc::from(std::fs::read(image_path).expect("Failed to read image"))); |
|
} |
|
let device = application_io.gpu_executor().unwrap().context.device.clone(); |
|
|
|
let preferences = EditorPreferences { |
|
use_vello: true, |
|
..Default::default() |
|
}; |
|
let editor_api = Arc::new(WasmEditorApi { |
|
font_cache: FontCache::default(), |
|
application_io: Some(application_io.into()), |
|
node_graph_message_sender: Box::new(UpdateLogger {}), |
|
editor_preferences: Box::new(preferences), |
|
}); |
|
|
|
let proto_graph = compile_graph(document_string, editor_api)?; |
|
|
|
match app.command { |
|
Command::Compile { print_proto, .. } => { |
|
if print_proto { |
|
println!("{}", proto_graph); |
|
} |
|
} |
|
Command::Run { run_loop, .. } => { |
|
std::thread::spawn(move || { |
|
loop { |
|
std::thread::sleep(std::time::Duration::from_nanos(10)); |
|
device.poll(wgpu::Maintain::Poll); |
|
} |
|
}); |
|
let executor = create_executor(proto_graph)?; |
|
let render_config = RenderConfig::default(); |
|
|
|
loop { |
|
let result = (&executor).execute(render_config).await?; |
|
if !run_loop { |
|
println!("{:?}", result); |
|
break; |
|
} |
|
std::thread::sleep(std::time::Duration::from_millis(16)); |
|
} |
|
} |
|
} |
|
|
|
Ok(()) |
|
} |
|
|
|
fn init_logging(log_level: u8) { |
|
let default_level = match log_level { |
|
0 => log::LevelFilter::Error, |
|
1 => log::LevelFilter::Info, |
|
2 => log::LevelFilter::Debug, |
|
_ => log::LevelFilter::Trace, |
|
}; |
|
let colors = ColoredLevelConfig::new().debug(Color::Magenta).info(Color::Green).error(Color::Red); |
|
fern::Dispatch::new() |
|
.chain(std::io::stdout()) |
|
.level_for("wgpu", log::LevelFilter::Error) |
|
.level_for("naga", log::LevelFilter::Error) |
|
.level_for("wgpu_hal", log::LevelFilter::Error) |
|
.level_for("wgpu_core", log::LevelFilter::Error) |
|
.level(default_level) |
|
.format(move |out, message, record| { |
|
out.finish(format_args!( |
|
"[{}]{}{} {}", |
|
|
|
colors.color(record.level()), |
|
chrono::Utc::now().format("[%Y-%m-%d %H:%M:%S]"), |
|
record.module_path().unwrap_or(""), |
|
message |
|
)) |
|
}) |
|
.apply() |
|
.unwrap(); |
|
} |
|
|
|
|
|
|
|
fn fix_nodes(network: &mut NodeNetwork) { |
|
for node in network.nodes.values_mut() { |
|
match &mut node.implementation { |
|
|
|
DocumentNodeImplementation::Network(network) => fix_nodes(network), |
|
|
|
|
|
|
|
DocumentNodeImplementation::ProtoNode(proto_node_identifier) |
|
if (proto_node_identifier.name.starts_with("graphene_core::ConstructLayerNode") || proto_node_identifier.name.starts_with("graphene_core::AddArtboardNode")) |
|
&& node.inputs.len() < 3 => |
|
{ |
|
node.inputs.push(NodeInput::Reflection(DocumentNodeMetadata::DocumentNodePath)); |
|
} |
|
_ => {} |
|
} |
|
} |
|
} |
|
fn compile_graph(document_string: String, editor_api: Arc<WasmEditorApi>) -> Result<ProtoNetwork, Box<dyn Error>> { |
|
let mut network = load_network(&document_string); |
|
fix_nodes(&mut network); |
|
|
|
let substitutions = preprocessor::generate_node_substitutions(); |
|
preprocessor::expand_network(&mut network, &substitutions); |
|
|
|
let wrapped_network = wrap_network_in_scope(network.clone(), editor_api); |
|
|
|
let compiler = Compiler {}; |
|
compiler.compile_single(wrapped_network).map_err(|x| x.into()) |
|
} |
|
|
|
fn create_executor(proto_network: ProtoNetwork) -> Result<DynamicExecutor, Box<dyn Error>> { |
|
let executor = block_on(DynamicExecutor::new(proto_network)).map_err(|errors| errors.iter().map(|e| format!("{e:?}")).reduce(|acc, e| format!("{acc}\n{e}")).unwrap_or_default())?; |
|
Ok(executor) |
|
} |
|
|