tsify-next 0.5.6

Tsify-next is a library for generating TypeScript definitions from rust code.
Documentation
#![allow(dead_code)]

use indoc::indoc;
use pretty_assertions::assert_eq;
use tsify_next::Tsify;

#[test]
fn test_rename() {
    /// Comment for RenamedStruct
    #[derive(Tsify)]
    struct RenamedStruct {
        /// Comment for X
        #[serde(rename = "X")]
        x: i32,
        /// Comment for Y
        #[serde(rename = "Y")]
        y: i32,
    }

    assert_eq!(
        RenamedStruct::DECL,
        indoc! {"
            /**
             * Comment for RenamedStruct
             */
            export interface RenamedStruct {
                /**
                 * Comment for X
                 */
                X: number;
                /**
                 * Comment for Y
                 */
                Y: number;
            }"
        }
    );

    /// Comment for RenamedEnum
    #[derive(Tsify)]
    enum RenamedEnum {
        /// Comment for X
        #[serde(rename = "X")]
        A(bool),
        /// Comment for Y
        #[serde(rename = "Y")]
        B(i64),
        /// Comment for Z
        #[serde(rename = "Z")]
        C(String),
        /// Comment for D
        #[serde(skip)]
        D(i32),
    }

    let expected = indoc! {r#"
        /**
         * Comment for RenamedEnum
         */
        export type RenamedEnum = { X: boolean } | { Y: number } | { Z: string };"#

    };

    assert_eq!(RenamedEnum::DECL, expected);
}

#[test]
fn test_rename_all() {
    /// Comment for Enum
    #[allow(clippy::enum_variant_names)]
    #[derive(Tsify)]
    #[serde(rename_all = "snake_case")]
    #[tsify(namespace)]
    enum Enum {
        /// Comment for snake_case
        SnakeCase { foo: bool, foo_bar: bool },
        /// Comment for camel_case
        #[serde(rename_all = "camelCase")]
        CamelCase { foo: bool, foo_bar: bool },
        /// Comment for kebab_case
        #[serde(rename_all = "kebab-case")]
        KebabCase { foo: bool, foo_bar: bool },
        /// Comment for screaming_snake_case
        #[serde(rename_all = "SCREAMING_SNAKE_CASE")]
        ScreamingSnakeCase { foo: bool, foo_bar: bool },
    }

    /// Comment for PascalCase
    #[derive(Tsify)]
    #[serde(rename_all = "PascalCase")]
    struct PascalCase {
        /// Comment for Foo
        foo: bool,
        /// Comment for FooBar
        foo_bar: bool,
    }

    /// Comment for ScreamingKebab
    #[derive(Tsify)]
    #[serde(rename_all = "SCREAMING-KEBAB-CASE")]
    struct ScreamingKebab {
        /// Comment for FOO
        foo: bool,
        /// Comment for FOO-BAR
        foo_bar: bool,
    }

    let expected = indoc! {r#"
        /**
         * Comment for Enum
         */
        declare namespace Enum {
            /**
             * Comment for snake_case
             */
            export type snake_case = { snake_case: { foo: boolean; foo_bar: boolean } };
            /**
             * Comment for camel_case
             */
            export type camel_case = { camel_case: { foo: boolean; fooBar: boolean } };
            /**
             * Comment for kebab_case
             */
            export type kebab_case = { kebab_case: { foo: boolean; "foo-bar": boolean } };
            /**
             * Comment for screaming_snake_case
             */
            export type screaming_snake_case = { screaming_snake_case: { FOO: boolean; FOO_BAR: boolean } };
        }

        /**
         * Comment for Enum
         */
        export type Enum = { snake_case: { foo: boolean; foo_bar: boolean } } | { camel_case: { foo: boolean; fooBar: boolean } } | { kebab_case: { foo: boolean; "foo-bar": boolean } } | { screaming_snake_case: { FOO: boolean; FOO_BAR: boolean } };"#
    };

    assert_eq!(Enum::DECL, expected);

    assert_eq!(
        PascalCase::DECL,
        indoc! {"
            /**
             * Comment for PascalCase
             */
            export interface PascalCase {
                /**
                 * Comment for Foo
                 */
                Foo: boolean;
                /**
                 * Comment for FooBar
                 */
                FooBar: boolean;
            }"
        }
    );

    assert_eq!(
        ScreamingKebab::DECL,
        indoc! {r#"
            /**
             * Comment for ScreamingKebab
             */
            export interface ScreamingKebab {
                /**
                 * Comment for FOO
                 */
                FOO: boolean;
                /**
                 * Comment for FOO-BAR
                 */
                "FOO-BAR": boolean;
            }"#
        }
    );
}

#[test]
fn test_quote_non_identifiers() {
    #[derive(Tsify)]
    struct NonIdentifierRenameStruct {
        #[serde(rename = "1")]
        x: i32,
        #[serde(rename = "1x")]
        y: i32,
        #[serde(rename = "-")]
        z: i32,
        #[serde(rename = " ")]
        w: i32,
        #[serde(rename = "#")]
        q: i32,
        #[serde(rename = "should_not_quote")]
        p: i32,
        #[serde(rename = "should$not$quote")]
        r: i32,
    }

    assert_eq!(
        NonIdentifierRenameStruct::DECL,
        indoc! {"
            export interface NonIdentifierRenameStruct {
                \"1\": number;
                \"1x\": number;
                \"-\": number;
                \" \": number;
                \"#\": number;
                should_not_quote: number;
                should$not$quote: number;
            }"
        }
    );

    #[derive(Tsify)]
    enum NonIdentifierRenameEnum {
        #[serde(rename = "hello-world")]
        A(bool),
        #[serde(rename = "hel#&*world")]
        B(i64),
        #[serde(rename = "hello world")]
        C(String),
        #[serde(rename = "")]
        D(i32),
        #[serde(rename = "should_not_quote")]
        E(String),
    }

    let expected = indoc! {r#"
        export type NonIdentifierRenameEnum = { "hello-world": boolean } | { "hel#&*world": number } | { "hello world": string } | { "": number } | { should_not_quote: string };"#

    };

    assert_eq!(NonIdentifierRenameEnum::DECL, expected);
}