Crates.io Build Status docs.rs

typescript-type-def

Generate TypeScript type definitions for Rust types.

This crate allows you to produce a TypeScript module containing type definitions which describe the JSON serialization of Rust types. The intended use is to define TypeScript types for data that is serialized from Rust types as JSON using serde_json so it can be safely used from TypeScript without needing to maintain a parallel set of type definitions.

One example of where this crate is useful is when working on a web application project with a Rust backend and a TypeScript frontend. If the data used to communicate between the two is defined in Rust and uses serde_json to encode/decode it for transmission across the network, you can use this crate to automatically generate a TypeScript definition file for those types in order to use them safely in your frontend code. This process can even be completely automated if you use this crate in a build script for your server to write the definition file to your TypeScript source code directory.

You can also use the Typescript type info generated by this library to write Typescript code that uses these type definitions. Rust types that implement [TypeDef] have an associated constant [TypeDef::INFO] which has a method TypeInfo::write_ref_expr that can be used for this purpose.

Features

Examples

Simple example: ```rust use serde::Serialize; use typescripttypedef::{ writedefinitionfile, DefinitionFileOptions, TypeDef, };

[derive(Serialize, TypeDef)]

struct Foo { a: usize, b: String, }

let tsmodule = { let mut buf = Vec::new(); let options = DefinitionFileOptions::default(); writedefinitionfile::<_, Foo>(&mut buf, options).unwrap(); String::fromutf8(buf).unwrap() }; asserteq!( tsmodule, r#"// AUTO-GENERATED by typescript-type-def

export default types; export namespace types{ export type Usize=number; export type Foo={"a":types.Usize;"b":string;}; } "# );

let foo = Foo { a: 123, b: "hello".toowned(), }; let json = serdejson::tostring(&foo).unwrap(); // This JSON matches the TypeScript type definition above asserteq!(json, r#"{"a":123,"b":"hello"}"#); ```

When working with a large codebase consisting of many types, a useful pattern is to declare an "API" type alias which lists all the types you want to make definitions for. For example: ```rust use serde::Serialize; use typescripttypedef::{writedefinitionfile, TypeDef};

[derive(Serialize, TypeDef)]

struct Foo { a: String, }

[derive(Serialize, TypeDef)]

struct Bar { a: String, }

[derive(Serialize, TypeDef)]

struct Baz { a: Qux, }

[derive(Serialize, TypeDef)]

struct Qux { a: String, }

// This type lists all the top-level types we want to make definitions for. // You don't need to list every type in your API here, only ones that // wouldn't be referenced otherwise. Note that Qux is not mentioned, but // is still emitted because it is a dependency of Baz. type Api = (Foo, Bar, Baz);

let tsmodule = { let mut buf = Vec::new(); writedefinitionfile::<_, Api>(&mut buf, Default::default()).unwrap(); String::fromutf8(buf).unwrap() }; asserteq!( tsmodule, r#"// AUTO-GENERATED by typescript-type-def

export default types; export namespace types{ export type Foo={"a":string;}; export type Bar={"a":string;}; export type Qux={"a":string;}; export type Baz={"a":types.Qux;}; } "# ); ```

Alternatively, you can use [write_definition_file_from_type_infos] with a list of TypeInfo references created at runtime to create a definition file. For example: ```rust use serde::Serialize; use typescripttypedef::{writedefinitionfilefromtype_infos, TypeDef};

[derive(Serialize, TypeDef)]

struct Foo { a: String, }

[derive(Serialize, TypeDef)]

struct Bar { a: String, }

[derive(Serialize, TypeDef)]

struct Baz { a: Qux, }

[derive(Serialize, TypeDef)]

struct Qux { a: String, }

// This list contains type info for all the top-level types we want to make // definitions for. // You don't need to list every type in your API here, only ones that // wouldn't be referenced otherwise. Note that Qux is not mentioned, but // is still emitted because it is a dependency of Baz. let api = vec![ &Foo::INFO, &Bar::INFO, &Baz::INFO, ];

let tsmodule = { let mut buf = Vec::new(); writedefinitionfilefromtypeinfos( &mut buf, Default::default(), &api, ) .unwrap(); String::fromutf8(buf).unwrap() }; asserteq!( ts_module, r#"// AUTO-GENERATED by typescript-type-def

export default types; export namespace types{ export type Foo={"a":string;}; export type Bar={"a":string;}; export type Qux={"a":string;}; export type Baz={"a":types.Qux;}; } "# ); ```