auto_generate_cdp 0.4.6

experimental crate to generate the Chrome Devtools Protocol.
Documentation
use quote::quote;

use std::env;
use std::path::Path;
use std::{fs::OpenOptions, process::Command};

use std::io::prelude::*;

mod types;
mod compile;

pub fn init() {
    const CDP_COMMIT: &str = "bb9208f5db271fc9f10978a11082d1ec57d52df2";
    let out_dir = env::var_os("OUT_DIR").unwrap();
    let out_file = Path::new(&out_dir).join("protocol.rs");
    let mut file = OpenOptions::new()
        .append(true)
        .create(true)
        .open(&out_file)
        .unwrap();

    file.sync_all().unwrap();

    if file.metadata().unwrap().len() == 0 {
        let (js_mods, js_events) = compile::compile_cdp_json("js_protocol.json", CDP_COMMIT);
        let (browser_mods, browser_events) = compile::compile_cdp_json("browser_protocol.json", CDP_COMMIT);

        writeln!(
            file,
            "// Auto-generated from ChromeDevTools/devtools-protocol at commit {CDP_COMMIT}",
        )
        .unwrap();

        let modv = quote! {
            #[allow(unused)]
            #[allow(non_camel_case_types)]
            #[allow(non_snake_case)]

            pub mod cdp {

                pub mod types {
                    use serde::{Deserialize, Serialize};
                    use std::fmt::Debug;

                    pub type JsFloat = f64;
                    pub type JsUInt = u32;

                    pub type WindowId = JsUInt;

                    pub type CallId = JsUInt;


                    #[derive(Serialize, Debug)]
                    pub struct MethodCall<T>
                    where
                    T: Debug,
                    {
                        #[serde(rename = "method")]
                        method_name: &'static str,
                        pub id: CallId,
                        params: T,
                    }

                    impl<T> MethodCall<T>
                    where
                    T: Debug,
                    {
                        pub fn get_params(&self) -> &T {
                        &self.params
                        }
                    }

                    pub trait Method: Debug {
                    const NAME: &'static str;

                    type ReturnObject: serde::de::DeserializeOwned + std::fmt::Debug;


                    fn to_method_call(self, call_id: CallId) -> MethodCall<Self>
                    where
                    Self: std::marker::Sized,
                    {
                        MethodCall {
                            id: call_id,
                             params: self,
                            method_name: Self::NAME,
                            }
                    }

                    }

                    #[derive(Deserialize, Debug, Clone, PartialEq)]
                    #[serde(tag = "method")]
                    #[allow(clippy::large_enum_variant)]
                    pub enum Event {
                        #(#browser_events)*
                        #(#js_events)*
                    }
                }

                #(#js_mods)*
                #(#browser_mods)*
            }
        };

        writeln!(file, "{modv}").unwrap();

        if env::var_os("DO_NOT_FORMAT").is_none() {
            let mut rustfmt = match env::var_os("RUSTFMT") {
                Some(rustfmt) => Command::new(rustfmt),
                None => Command::new("rustfmt"),
            };
            rustfmt.arg(&out_file).output().expect("rustfmt not found");
        }
    }
}

#[cfg(test)]
mod tests {
    #[test]
    fn test() {
        crate::init();
    }
}