diff --git a/README.md b/README.md
index 1b931f0f..4a1401a8 100644
--- a/README.md
+++ b/README.md
@@ -81,6 +81,7 @@ export type User = { user_id: number, first_name: string, last_name: string, };
| format | Enables formatting of the generated TypeScript bindings.
Currently, this unfortunately adds quite a few dependencies. |
| no-serde-warnings | By default, warnings are printed during build if unsupported serde attributes are encountered.
Enabling this feature silences these warnings. |
| import-esm | When enabled,`import` statements in the generated file will have the `.js` extension in the end of the path to conform to the ES Modules spec.
Example: `import { MyStruct } from "./my_struct.js"` |
+| import-deno | When enabled,`import` statements in the generated file will have the `.ts` extension in the end of the path to conform to the Deno spec.
Example: `import { MyStruct } from "./my_struct.ts"` |
| serde-json-impl | Implement `TS` for types from *serde_json* |
| chrono-impl | Implement `TS` for types from *chrono* |
| bigdecimal-impl | Implement `TS` for types from *bigdecimal* |
@@ -92,8 +93,8 @@ export type User = { user_id: number, first_name: string, last_name: string, };
| ordered-float-impl | Implement `TS` for types from *ordered_float* |
| heapless-impl | Implement `TS` for types from *heapless* |
| semver-impl | Implement `TS` for types from *semver* |
-| smol_str-impl | Implement `TS` for types from *smol_str* |
-| tokio-impl | Implement `TS` for types from *tokio* |
+| smol_str-impl | Implement `TS` for types from *smol_str* |
+| tokio-impl | Implement `TS` for types from *tokio* |
diff --git a/ts-rs/Cargo.toml b/ts-rs/Cargo.toml
index 85d00d48..0c82bf6e 100644
--- a/ts-rs/Cargo.toml
+++ b/ts-rs/Cargo.toml
@@ -35,6 +35,7 @@ smol_str-impl = ["smol_str"]
serde-json-impl = ["serde_json"]
no-serde-warnings = ["ts-rs-macros/no-serde-warnings"]
import-esm = []
+import-deno = []
tokio-impl = ["tokio"]
[dev-dependencies]
diff --git a/ts-rs/src/export.rs b/ts-rs/src/export.rs
index 3640c96e..9a0bfe66 100644
--- a/ts-rs/src/export.rs
+++ b/ts-rs/src/export.rs
@@ -309,13 +309,21 @@ fn generate_imports(
let dep_path = out_dir.as_ref().join(dep.output_path);
let rel_path = import_path(&path, &dep_path)?;
+ // Determine the appropriate extension to trim based on the active feature
+ let trimmed_rel_path = if cfg!(feature = "import-deno") {
+ rel_path.trim_end_matches(".ts")
+ } else if cfg!(feature = "import-esm") {
+ rel_path.trim_end_matches(".js")
+ } else {
+ &rel_path
+ };
+
+ // Check if the file is importing itself
let is_same_file = path
- .file_name()
+ .file_stem()
.and_then(std::ffi::OsStr::to_str)
- .map(|x| x.trim_end_matches(".ts"))
- .map(|x| format!("./{x}"))
- .map(|x| x == rel_path.trim_end_matches(".js"))
- .unwrap_or(false);
+ .map(|stem| format!("./{stem}"))
+ .map_or(false, |formatted_stem| formatted_stem == trimmed_rel_path);
if is_same_file {
continue;
@@ -347,6 +355,10 @@ fn import_path(from: &Path, import: &Path) -> Result {
str_path
};
+ if cfg!(feature = "import-deno") {
+ return Ok(path);
+ }
+
let path_without_extension = path.trim_end_matches(".ts");
Ok(if cfg!(feature = "import-esm") {
diff --git a/ts-rs/src/lib.rs b/ts-rs/src/lib.rs
index b2bbc073..cfa429a6 100644
--- a/ts-rs/src/lib.rs
+++ b/ts-rs/src/lib.rs
@@ -79,6 +79,7 @@
//! | format | Enables formatting of the generated TypeScript bindings.
Currently, this unfortunately adds quite a few dependencies. |
//! | no-serde-warnings | By default, warnings are printed during build if unsupported serde attributes are encountered.
Enabling this feature silences these warnings. |
//! | import-esm | When enabled,`import` statements in the generated file will have the `.js` extension in the end of the path to conform to the ES Modules spec.
Example: `import { MyStruct } from "./my_struct.js"` |
+//! | import-deno | When enabled,`import` statements in the generated file will have the `.ts` extension in the end of the path to conform to the Deno spec.
Example: `import { MyStruct } from "./my_struct.ts"` |
//! | serde-json-impl | Implement `TS` for types from *serde_json* |
//! | chrono-impl | Implement `TS` for types from *chrono* |
//! | bigdecimal-impl | Implement `TS` for types from *bigdecimal* |
@@ -90,8 +91,8 @@
//! | ordered-float-impl | Implement `TS` for types from *ordered_float* |
//! | heapless-impl | Implement `TS` for types from *heapless* |
//! | semver-impl | Implement `TS` for types from *semver* |
-//! | smol_str-impl | Implement `TS` for types from *smol_str* |
-//! | tokio-impl | Implement `TS` for types from *tokio* |
+//! | smol_str-impl | Implement `TS` for types from *smol_str* |
+//! | tokio-impl | Implement `TS` for types from *tokio* |
//!
//!
//!
diff --git a/ts-rs/tests/integration/imports.rs b/ts-rs/tests/integration/imports.rs
index bd7e5f81..87baae5d 100644
--- a/ts-rs/tests/integration/imports.rs
+++ b/ts-rs/tests/integration/imports.rs
@@ -28,42 +28,44 @@ fn test_def() {
TestEnum::export_all().unwrap();
let text = std::fs::read_to_string(TestEnum::default_output_path().unwrap()).unwrap();
- let expected = match (cfg!(feature = "format"), cfg!(feature = "import-esm")) {
- (true, true) => concat!(
- "// This file was generated by [ts-rs](https://github.com/Aleph-Alpha/ts-rs). Do not edit this file manually.\n",
- "import type { TestTypeA } from \"./ts_rs_test_type_a.js\";\n",
- "import type { TestTypeB } from \"./ts_rs_test_type_b.js\";\n",
- "\n",
- "export type TestEnum = { \"C\": { value: TestTypeB } } | {\n",
- " \"A1\": { value: TestTypeA };\n",
- "} | { \"A2\": { value: TestTypeA } };\n",
- ),
- (true, false) => concat!(
+ let extension = if cfg!(feature = "import-deno") {
+ ".ts"
+ } else if cfg!(feature = "import-esm") {
+ ".js"
+ } else {
+ ""
+ };
+
+ let import_a = format!(
+ "import type {{ TestTypeA }} from \"./ts_rs_test_type_a{}\";\n",
+ extension
+ );
+
+ let import_b = format!(
+ "import type {{ TestTypeB }} from \"./ts_rs_test_type_b{}\";\n",
+ extension
+ );
+
+ let expected = if cfg!(feature = "format") {
+ vec!(
"// This file was generated by [ts-rs](https://github.com/Aleph-Alpha/ts-rs). Do not edit this file manually.\n",
- "import type { TestTypeA } from \"./ts_rs_test_type_a\";\n",
- "import type { TestTypeB } from \"./ts_rs_test_type_b\";\n",
+ &import_a,
+ &import_b,
"\n",
"export type TestEnum = { \"C\": { value: TestTypeB } } | {\n",
" \"A1\": { value: TestTypeA };\n",
"} | { \"A2\": { value: TestTypeA } };\n",
- ),
- (false, true) => concat!(
+ )
+ } else {
+ vec!(
"// This file was generated by [ts-rs](https://github.com/Aleph-Alpha/ts-rs). Do not edit this file manually.\n",
- "import type { TestTypeA } from \"./ts_rs_test_type_a.js\";\n",
- "import type { TestTypeB } from \"./ts_rs_test_type_b.js\";\n",
+ &import_a,
+ &import_b,
"\n",
"export type TestEnum = { \"C\": { value: TestTypeB, } } | { \"A1\": { value: TestTypeA, } } | { \"A2\": { value: TestTypeA, } };",
"\n",
- ),
- (false, false) => concat!(
- "// This file was generated by [ts-rs](https://github.com/Aleph-Alpha/ts-rs). Do not edit this file manually.\n",
- "import type { TestTypeA } from \"./ts_rs_test_type_a\";\n",
- "import type { TestTypeB } from \"./ts_rs_test_type_b\";\n",
- "\n",
- "export type TestEnum = { \"C\": { value: TestTypeB, } } | { \"A1\": { value: TestTypeA, } } | { \"A2\": { value: TestTypeA, } };",
- "\n",
- ),
- };
+ )
+ }.join("").to_string();
assert_eq!(text, expected);
}
diff --git a/ts-rs/tests/integration/issue_168.rs b/ts-rs/tests/integration/issue_168.rs
index 732ba9be..190671e0 100644
--- a/ts-rs/tests/integration/issue_168.rs
+++ b/ts-rs/tests/integration/issue_168.rs
@@ -32,7 +32,7 @@ struct Baz {
}
#[test]
-#[cfg(not(feature = "import-esm"))]
+#[cfg(not(any(feature = "import-esm", feature = "import-deno")))]
fn issue_168() {
assert_eq!(
FooInlined::export_to_string().unwrap(),
diff --git a/ts-rs/tests/integration/issue_232.rs b/ts-rs/tests/integration/issue_232.rs
index 5cb6f684..582d105e 100644
--- a/ts-rs/tests/integration/issue_232.rs
+++ b/ts-rs/tests/integration/issue_232.rs
@@ -42,7 +42,7 @@ enum Enum {
}
#[test]
-#[cfg(not(feature = "import-esm"))]
+#[cfg(not(any(feature = "import-esm", feature = "import-deno")))]
fn issue_232() {
println!("{}", StateInlinedVec::export_to_string().unwrap());
assert_eq!(