Skip to content

Commit 02469a4

Browse files
authored
Merge pull request #235 from Buckram123/symlinks-does-not-work-in-debug
Symbolic links in debug mode
2 parents ed8faec + 23c8cbf commit 02469a4

File tree

6 files changed

+45
-7
lines changed

6 files changed

+45
-7
lines changed

examples/public/symlinks/main.js

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
../main.js

impl/src/lib.rs

+10-1
Original file line numberDiff line numberDiff line change
@@ -141,7 +141,16 @@ fn dynamic(ident: &syn::Ident, folder_path: String, prefix: Option<&str>, includ
141141
let canonical_file_path = file_path.canonicalize().ok()?;
142142
if !canonical_file_path.starts_with(#canonical_folder_path) {
143143
// Tried to request a path that is not in the embedded folder
144-
return ::std::option::Option::None;
144+
145+
// TODO: Currently it allows "path_traversal_attack" for the symlink files
146+
// For it to be working properly we need to get absolute path first
147+
// and check that instead if it starts with `canonical_folder_path`
148+
// https://doc.rust-lang.org/std/path/fn.absolute.html (currently nightly)
149+
// Should be allowed only if it was a symlink
150+
let metadata = ::std::fs::symlink_metadata(file_path.as_path()).ok()?;
151+
if !metadata.is_symlink() {
152+
return ::std::option::Option::None;
153+
}
145154
}
146155

147156
if rust_embed::utils::is_path_included(&rel_file_path, INCLUDES, EXCLUDES) {

tests/include_exclude.rs

+16-2
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,8 @@ fn get_works() {
99
assert!(AllAssets::get("index.html").is_some(), "index.html should exist");
1010
assert!(AllAssets::get("gg.html").is_none(), "gg.html should not exist");
1111
assert!(AllAssets::get("images/llama.png").is_some(), "llama.png should exist");
12-
assert_eq!(AllAssets::iter().count(), 6);
12+
assert!(AllAssets::get("symlinks/main.js").is_some(), "main.js should exist");
13+
assert_eq!(AllAssets::iter().count(), 7);
1314
}
1415

1516
#[derive(RustEmbed)]
@@ -36,8 +37,9 @@ struct ExcludeSomeAssets;
3637
fn excluding_some_assets_works() {
3738
assert!(ExcludeSomeAssets::get("index.html").is_none(), "index.html should not exist");
3839
assert!(ExcludeSomeAssets::get("main.js").is_some(), "main.js should exist");
40+
assert!(ExcludeSomeAssets::get("symlinks/main.js").is_some(), "main.js symlink should exist");
3941
assert!(ExcludeSomeAssets::get("images/llama.png").is_none(), "llama.png should not exist");
40-
assert_eq!(ExcludeSomeAssets::iter().count(), 2);
42+
assert_eq!(ExcludeSomeAssets::iter().count(), 3);
4143
}
4244

4345
#[derive(RustEmbed)]
@@ -52,3 +54,15 @@ fn exclude_has_higher_priority() {
5254
assert!(ExcludePriorityAssets::get("images/llama.png").is_some(), "llama.png should exist");
5355
assert_eq!(ExcludePriorityAssets::iter().count(), 2);
5456
}
57+
58+
#[derive(RustEmbed)]
59+
#[folder = "examples/public/symlinks"]
60+
#[include = "main.js"]
61+
struct IncludeSymlink;
62+
63+
#[test]
64+
fn include_symlink() {
65+
assert_eq!(IncludeSymlink::iter().count(), 1);
66+
assert_eq!(IncludeSymlink::iter().next(), Some(std::borrow::Cow::Borrowed("main.js")));
67+
assert!(IncludeSymlink::get("main.js").is_some())
68+
}

tests/interpolated_path.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ fn iter_works() {
1919
assert!(Asset::get(file.as_ref()).is_some());
2020
num_files += 1;
2121
}
22-
assert_eq!(num_files, 6);
22+
assert_eq!(num_files, 7);
2323
}
2424

2525
#[test]
@@ -32,6 +32,6 @@ fn trait_works_generic_helper<E: rust_embed::RustEmbed>() {
3232
assert!(E::get(file.as_ref()).is_some());
3333
num_files += 1;
3434
}
35-
assert_eq!(num_files, 6);
35+
assert_eq!(num_files, 7);
3636
assert!(E::get("gg.html").is_none(), "gg.html should not exist");
3737
}

tests/lib.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ fn iter_works() {
2828
assert!(Asset::get(file.as_ref()).is_some());
2929
num_files += 1;
3030
}
31-
assert_eq!(num_files, 6);
31+
assert_eq!(num_files, 7);
3232
}
3333

3434
#[test]
@@ -41,6 +41,6 @@ fn trait_works_generic_helper<E: rust_embed::RustEmbed>() {
4141
assert!(E::get(file.as_ref()).is_some());
4242
num_files += 1;
4343
}
44-
assert_eq!(num_files, 6);
44+
assert_eq!(num_files, 7);
4545
assert!(E::get("gg.html").is_none(), "gg.html should not exist");
4646
}

tests/path_traversal_attack.rs

+14
Original file line numberDiff line numberDiff line change
@@ -11,3 +11,17 @@ struct Assets;
1111
fn path_traversal_attack_fails() {
1212
assert!(Assets::get("../basic.rs").is_none());
1313
}
14+
15+
#[derive(RustEmbed)]
16+
#[folder = "examples/axum-spa/"]
17+
struct AxumAssets;
18+
19+
// TODO:
20+
/// Prevent attempts to access symlinks outside of the embedded folder.
21+
/// This is mainly a concern when running in debug mode, since that loads from
22+
/// the file system at runtime.
23+
#[test]
24+
#[ignore = "see https://github.com/pyrossh/rust-embed/pull/235"]
25+
fn path_traversal_attack_symlink_fails() {
26+
assert!(Assets::get("../public/symlinks/main.js").is_none());
27+
}

0 commit comments

Comments
 (0)