typescript-definitions 0.1.10

serde support for exporting Typescript definitions
Documentation
#![allow(unused)]
use std::process;

use typescript_definitions::{TypeScriptify, TypeScriptifyTrait, TypescriptDefinition};

use serde::Serialize;
// use serde::de::value::Error;
use insta::assert_snapshot_matches;
use wasm_bindgen::prelude::*;

use std::io::Write;
use std::process::{Command, Stdio};
pub fn prettier(s: &str) -> String {
    let mut child = Command::new("prettier")
        .arg("--parser")
        .arg("typescript")
        .stdin(Stdio::piped())
        .stdout(Stdio::piped())
        .spawn()
        .expect("failed to execute prettier");

    // hopefully the pipe buffers don't fill up :)
    {
        // limited borrow of stdin
        let stdin = child.stdin.as_mut().expect("failed to get stdin");
        stdin
            .write_all(s.as_bytes())
            .expect("failed to write to stdin");
    }

    let output = child
        .wait_with_output()
        .expect("failed to wait on prettier");

    String::from_utf8_lossy(&output.stdout).trim().to_string()
}
#[cfg(feature = "type-guards")]
#[test]
fn verify_untagged_enum() {
    use serde_json;
    // use serde_json::Error;
    #[derive(Serialize, TypeScriptify)]
    #[serde(untagged)]
    enum Untagged {
        V1 { id: i32, attr: String },
        V2 { id: i32, attr2: Vec<String> },
    }
    let verify_untagged_enum = prettier(&Untagged::type_script_guard().unwrap());
    assert_snapshot_matches!(verify_untagged_enum,
        @r###"export const isUntagged = (obj: any): obj is Untagged => {
  if (obj == undefined) return false;
  if (
    (() => {
      if (obj.id === undefined) return false;
      {
        const val = obj.id;
        if (!(typeof val === "number")) return false;
      }
      if (obj.attr === undefined) return false;
      {
        const val = obj.attr;
        if (!(typeof val === "string")) return false;
      }
      return true;
    })()
  )
    return true;
  if (
    (() => {
      if (obj.id === undefined) return false;
      {
        const val = obj.id;
        if (!(typeof val === "number")) return false;
      }
      if (obj.attr2 === undefined) return false;
      {
        const val = obj.attr2;
        if (!Array.isArray(val)) return false;
        for (let x of val) {
          if (!(typeof x === "string")) return false;
        }
      }
      return true;
    })()
  )
    return true;
  return false;
};"###
    )
}
#[cfg(feature = "type-guards")]
#[test]
fn verify_first_only() {
    use serde_json;

    #[derive(Serialize, TypeScriptify)]
    struct S {
        #[ts(array_check = "first")]
        vals: Vec<String>,
    }

    let verify_first_only = prettier(&S::type_script_guard().unwrap());
    assert_snapshot_matches!(verify_first_only,
    @r###"export const isS = (obj: any): obj is S => {
  if (obj == undefined) return false;
  if (obj.vals === undefined) return false;
  {
    const val = obj.vals;
    if (!Array.isArray(val)) return false;
    for (let x of val) {
      if (!(typeof x === "string")) return false;
      break;
    }
  }
  return true;
};"###)
}
#[cfg(feature = "type-guards")]
#[test]
fn no_verifier() {
    use serde_json;

    #[derive(Serialize, TypeScriptify)]
    #[ts(guard = false)]
    struct S {
        a: i32,
        b: i64,
        vals: Vec<String>,
    }

    let verify = S::type_script_guard();
    assert!(verify.is_none(), "expecting None")
}
#[cfg(feature = "type-guards")]
#[test]
fn verify_typescript_enum() {
    use serde_json;
    #[derive(Serialize, TypeScriptify)]
    enum TyEnum {
        Red,
        Green,
        Blue,
    }
    let verify_typescript_enum = prettier(&TyEnum::type_script_guard().unwrap());
    assert_snapshot_matches!(verify_typescript_enum,
        @r###"export const isTyEnum = (obj: any): obj is TyEnum => {
  if (!(obj === "Red" || obj === "Green" || obj === "Blue")) return false;
  return true;
};"###
    )
}
#[cfg(feature = "type-guards")]
#[test]
fn verify_ts_as() {
    #[derive(TypeScriptify)]
    struct Sub {
        #[ts(ts_as = "Vec<u8>", array_check = "first")]
        b: i32,
    }

    assert_snapshot_matches!(
        prettier(&Sub::type_script_guard().unwrap()),
        @r###"export const isSub = (obj: any): obj is Sub => {
  if (obj == undefined) return false;
  if (obj.b === undefined) return false;
  {
    const val = obj.b;
    if (!Array.isArray(val)) return false;
    for (let x of val) {
      if (!(typeof x === "number")) return false;
      break;
    }
  }
  return true;
};"###

    );
}
#[cfg(feature = "type-guards")]
#[test]
fn verify_option() {
    #[derive(Serialize, TypeScriptify)]
    pub struct Maybe {
        maybe: Option<String>,
    }

    assert_snapshot_matches!(
    prettier(&Maybe::type_script_guard().unwrap()),
        @r###"export const isMaybe = (obj: any): obj is Maybe => {
  if (obj == undefined) return false;
  if (obj.maybe === undefined) return false;
  {
    const val = obj.maybe;
    if (!(val === null)) {
      if (!(typeof val === "string")) return false;
    }
  }
  return true;
};"###
    );
}