mirror of
https://github.com/sigoden/dufs.git
synced 2026-04-09 17:13:02 +03:00
Compare commits
11 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
c1c6dbc356 | ||
|
|
e29cf4c752 | ||
|
|
7f062b6705 | ||
|
|
ea8b9e9cce | ||
|
|
d2270be8fb | ||
|
|
2d0dfed456 | ||
|
|
4058a2db72 | ||
|
|
069cb64889 | ||
|
|
c85ea06785 | ||
|
|
68139c6263 | ||
|
|
deb6365a28 |
25
CHANGELOG.md
25
CHANGELOG.md
@@ -2,6 +2,31 @@
|
|||||||
|
|
||||||
All notable changes to this project will be documented in this file.
|
All notable changes to this project will be documented in this file.
|
||||||
|
|
||||||
|
## [0.21.0] - 2022-06-21
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
- Escape name contains html escape code ([#65](https://github.com/sigoden/dufs/issues/65))
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
- Use custom logger with timestamp in rfc3339 ([#67](https://github.com/sigoden/dufs/issues/67))
|
||||||
|
|
||||||
|
### Refactor
|
||||||
|
|
||||||
|
- Split css/js from index.html ([#68](https://github.com/sigoden/dufs/issues/68))
|
||||||
|
|
||||||
|
## [0.20.0] - 2022-06-20
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
- DecodeURI searching string ([#61](https://github.com/sigoden/dufs/issues/61))
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
- Added basic auth ([#60](https://github.com/sigoden/dufs/issues/60))
|
||||||
|
- Add option --allow-search ([#62](https://github.com/sigoden/dufs/issues/62))
|
||||||
|
|
||||||
## [0.19.0] - 2022-06-19
|
## [0.19.0] - 2022-06-19
|
||||||
|
|
||||||
### Features
|
### Features
|
||||||
|
|||||||
33
Cargo.lock
generated
33
Cargo.lock
generated
@@ -423,6 +423,7 @@ dependencies = [
|
|||||||
"bitflags",
|
"bitflags",
|
||||||
"clap_lex",
|
"clap_lex",
|
||||||
"indexmap",
|
"indexmap",
|
||||||
|
"terminal_size",
|
||||||
"textwrap",
|
"textwrap",
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -571,7 +572,7 @@ checksum = "fea41bba32d969b513997752735605054bc0dfa92b4c56bf1189f2e174be7a10"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "dufs"
|
name = "dufs"
|
||||||
version = "0.19.0"
|
version = "0.21.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"assert_cmd",
|
"assert_cmd",
|
||||||
"assert_fs",
|
"assert_fs",
|
||||||
@@ -582,7 +583,6 @@ dependencies = [
|
|||||||
"chrono",
|
"chrono",
|
||||||
"clap",
|
"clap",
|
||||||
"diqwest",
|
"diqwest",
|
||||||
"env_logger",
|
|
||||||
"futures",
|
"futures",
|
||||||
"get_if_addrs",
|
"get_if_addrs",
|
||||||
"headers",
|
"headers",
|
||||||
@@ -628,16 +628,6 @@ dependencies = [
|
|||||||
"cfg-if",
|
"cfg-if",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "env_logger"
|
|
||||||
version = "0.9.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "0b2cf0344971ee6c64c31be0d530793fba457d322dfec2810c453d0ef228f9c3"
|
|
||||||
dependencies = [
|
|
||||||
"humantime",
|
|
||||||
"log",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "event-listener"
|
name = "event-listener"
|
||||||
version = "2.5.2"
|
version = "2.5.2"
|
||||||
@@ -1032,12 +1022,6 @@ version = "1.0.2"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c4a1e36c821dbe04574f602848a19f742f4fb3c98d40449f11bcad18d6b17421"
|
checksum = "c4a1e36c821dbe04574f602848a19f742f4fb3c98d40449f11bcad18d6b17421"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "humantime"
|
|
||||||
version = "2.1.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "hyper"
|
name = "hyper"
|
||||||
version = "0.14.19"
|
version = "0.14.19"
|
||||||
@@ -2146,6 +2130,16 @@ dependencies = [
|
|||||||
"utf-8",
|
"utf-8",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "terminal_size"
|
||||||
|
version = "0.1.17"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "633c1a546cee861a1a6d0dc69ebeca693bf4296661ba7852b9d21d159e0506df"
|
||||||
|
dependencies = [
|
||||||
|
"libc",
|
||||||
|
"winapi 0.3.9",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "termtree"
|
name = "termtree"
|
||||||
version = "0.2.4"
|
version = "0.2.4"
|
||||||
@@ -2157,6 +2151,9 @@ name = "textwrap"
|
|||||||
version = "0.15.0"
|
version = "0.15.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "b1141d4d61095b28419e22cb0bbf02755f5e54e0526f97f1e3d1d160e60885fb"
|
checksum = "b1141d4d61095b28419e22cb0bbf02755f5e54e0526f97f1e3d1d160e60885fb"
|
||||||
|
dependencies = [
|
||||||
|
"terminal_size",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "thiserror"
|
name = "thiserror"
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "dufs"
|
name = "dufs"
|
||||||
version = "0.19.0"
|
version = "0.21.0"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
authors = ["sigoden <sigoden@gmail.com>"]
|
authors = ["sigoden <sigoden@gmail.com>"]
|
||||||
description = "Dufs is a simple file server."
|
description = "Dufs is a distinctive utility file server"
|
||||||
license = "MIT OR Apache-2.0"
|
license = "MIT OR Apache-2.0"
|
||||||
homepage = "https://github.com/sigoden/dufs"
|
homepage = "https://github.com/sigoden/dufs"
|
||||||
repository = "https://github.com/sigoden/dufs"
|
repository = "https://github.com/sigoden/dufs"
|
||||||
@@ -11,7 +11,7 @@ categories = ["command-line-utilities", "web-programming::http-server"]
|
|||||||
keywords = ["static", "file", "server", "webdav", "cli"]
|
keywords = ["static", "file", "server", "webdav", "cli"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
clap = { version = "3", default-features = false, features = ["std"] }
|
clap = { version = "3", default-features = false, features = ["std", "wrap_help"] }
|
||||||
chrono = "0.4"
|
chrono = "0.4"
|
||||||
tokio = { version = "1", features = ["rt-multi-thread", "macros", "fs", "io-util", "signal"]}
|
tokio = { version = "1", features = ["rt-multi-thread", "macros", "fs", "io-util", "signal"]}
|
||||||
tokio-rustls = "0.23"
|
tokio-rustls = "0.23"
|
||||||
@@ -34,7 +34,6 @@ lazy_static = "1.4"
|
|||||||
uuid = { version = "1.1", features = ["v4", "fast-rng"] }
|
uuid = { version = "1.1", features = ["v4", "fast-rng"] }
|
||||||
urlencoding = "2.1"
|
urlencoding = "2.1"
|
||||||
xml-rs = "0.8"
|
xml-rs = "0.8"
|
||||||
env_logger = { version = "0.9", default-features = false, features = ["humantime"] }
|
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
socket2 = "0.4"
|
socket2 = "0.4"
|
||||||
async-stream = "0.3"
|
async-stream = "0.3"
|
||||||
|
|||||||
103
README.md
103
README.md
@@ -3,7 +3,7 @@
|
|||||||
[](https://github.com/sigoden/dufs/actions/workflows/ci.yaml)
|
[](https://github.com/sigoden/dufs/actions/workflows/ci.yaml)
|
||||||
[](https://crates.io/crates/dufs)
|
[](https://crates.io/crates/dufs)
|
||||||
|
|
||||||
Dufs is a simple file server. Support static serve, search, upload, webdav...
|
Dufs is a distinctive utility file server that supports static serving, uploading, searching, accessing control, webdav...
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
@@ -40,27 +40,29 @@ Download from [Github Releases](https://github.com/sigoden/dufs/releases), unzip
|
|||||||
## CLI
|
## CLI
|
||||||
|
|
||||||
```
|
```
|
||||||
Dufs is a simple file server. - https://github.com/sigoden/dufs
|
Dufs is a distinctive utility file server - https://github.com/sigoden/dufs
|
||||||
|
|
||||||
USAGE:
|
USAGE:
|
||||||
dufs [OPTIONS] [--] [path]
|
dufs [OPTIONS] [--] [path]
|
||||||
|
|
||||||
ARGS:
|
ARGS:
|
||||||
<path> Path to a root directory for serving files [default: .]
|
<path> Specific path to serve [default: .]
|
||||||
|
|
||||||
OPTIONS:
|
OPTIONS:
|
||||||
-b, --bind <addr>... Specify bind address
|
-b, --bind <addr>... Specify bind address
|
||||||
-p, --port <port> Specify port to listen on [default: 5000]
|
-p, --port <port> Specify port to listen on [default: 5000]
|
||||||
--path-prefix <path> Specify an url path prefix
|
--path-prefix <path> Specify an path prefix
|
||||||
-a, --auth <rule>... Add auth for path
|
-a, --auth <rule>... Add auth for path
|
||||||
|
--auth-method <value> Select auth method [default: digest] [possible values: basic, digest]
|
||||||
-A, --allow-all Allow all operations
|
-A, --allow-all Allow all operations
|
||||||
--allow-upload Allow upload files/folders
|
--allow-upload Allow upload files/folders
|
||||||
--allow-delete Allow delete files/folders
|
--allow-delete Allow delete files/folders
|
||||||
|
--allow-search Allow search files/folders
|
||||||
--allow-symlink Allow symlink to files/folders outside root directory
|
--allow-symlink Allow symlink to files/folders outside root directory
|
||||||
--enable-cors Enable CORS, sets `Access-Control-Allow-Origin: *`
|
--enable-cors Enable CORS, sets `Access-Control-Allow-Origin: *`
|
||||||
--render-index Render index.html when requesting a directory
|
--render-index Serve index.html when requesting a directory, returns 404 if not found index.html
|
||||||
--render-try-index Render index.html if it exists when requesting a directory
|
--render-try-index Serve index.html when requesting a directory, returns file listing if not found index.html
|
||||||
--render-spa Render for single-page application
|
--render-spa Serve SPA(Single Page Application)
|
||||||
--tls-cert <path> Path to an SSL/TLS certificate to serve with HTTPS
|
--tls-cert <path> Path to an SSL/TLS certificate to serve with HTTPS
|
||||||
--tls-key <path> Path to the SSL/TLS certificate's private key
|
--tls-key <path> Path to the SSL/TLS certificate's private key
|
||||||
-h, --help Print help information
|
-h, --help Print help information
|
||||||
@@ -69,30 +71,60 @@ OPTIONS:
|
|||||||
|
|
||||||
## Examples
|
## Examples
|
||||||
|
|
||||||
Serve current working directory, no upload/delete
|
Serve current working directory
|
||||||
|
|
||||||
```
|
```
|
||||||
dufs
|
dufs
|
||||||
```
|
```
|
||||||
|
|
||||||
Allow upload/delete
|
Explicitly allow all operations including upload/delete
|
||||||
|
|
||||||
```
|
```
|
||||||
dufs -A
|
dufs -A
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Only allow upload operation
|
||||||
|
|
||||||
|
```
|
||||||
|
dufs --allow-upload
|
||||||
|
```
|
||||||
|
|
||||||
|
Serve a directory
|
||||||
|
|
||||||
|
```
|
||||||
|
dufs Downloads
|
||||||
|
```
|
||||||
|
|
||||||
|
Serve a single file
|
||||||
|
|
||||||
|
```
|
||||||
|
dufs linux-distro.iso
|
||||||
|
```
|
||||||
|
|
||||||
|
Serve index.html when requesting a directory
|
||||||
|
|
||||||
|
```
|
||||||
|
dufs --render-index
|
||||||
|
```
|
||||||
|
|
||||||
|
Serve SPA(Single Page Application)
|
||||||
|
|
||||||
|
```
|
||||||
|
dufs --render-spa
|
||||||
|
```
|
||||||
|
|
||||||
|
Require username/password
|
||||||
|
|
||||||
|
```
|
||||||
|
dufs -a /@admin:123
|
||||||
|
```
|
||||||
|
|
||||||
Listen on a specific port
|
Listen on a specific port
|
||||||
|
|
||||||
```
|
```
|
||||||
dufs -p 80
|
dufs -p 80
|
||||||
```
|
```
|
||||||
|
|
||||||
For a single page application (SPA)
|
|
||||||
|
|
||||||
```
|
|
||||||
dufs --render-spa
|
|
||||||
```
|
|
||||||
|
|
||||||
Use https
|
Use https
|
||||||
|
|
||||||
```
|
```
|
||||||
@@ -124,36 +156,9 @@ Delete a file/folder
|
|||||||
curl -X DELETE http://127.0.0.1:5000/path-to-file
|
curl -X DELETE http://127.0.0.1:5000/path-to-file
|
||||||
```
|
```
|
||||||
|
|
||||||
## Details
|
## Access Control
|
||||||
|
|
||||||
<details>
|
Dufs supports path level access control. You can control who can do what on which path with `--auth`/`-a`.
|
||||||
<summary>
|
|
||||||
|
|
||||||
#### 1. Control render logic
|
|
||||||
|
|
||||||
</summary>
|
|
||||||
|
|
||||||
|
|
||||||
The default render logic is:
|
|
||||||
|
|
||||||
- If request for a folder, rendering the directory listing.
|
|
||||||
- If request for a file, rendering the file.
|
|
||||||
- If request target does not exist, returns 404.
|
|
||||||
|
|
||||||
The `--render-*` options change the render logic:
|
|
||||||
|
|
||||||
- `--render-index`: If request for a folder, rendering index.html in the folder. If the index.html file does not exist, return 404.
|
|
||||||
- `--render-try-index`: Like `--render-index`, rendering the directory listing if the index.html file does not exist, other than return 404.
|
|
||||||
- `--render-spa`: If request target does not exist, rendering `/index.html`
|
|
||||||
|
|
||||||
</details>
|
|
||||||
|
|
||||||
<details>
|
|
||||||
<summary>
|
|
||||||
|
|
||||||
#### 2. Path level access control
|
|
||||||
|
|
||||||
</summary>
|
|
||||||
|
|
||||||
```
|
```
|
||||||
dufs -a <path>@<readwrite>[@<readonly>]
|
dufs -a <path>@<readwrite>[@<readonly>]
|
||||||
@@ -163,7 +168,7 @@ dufs -a <path>@<readwrite>[@<readonly>]
|
|||||||
- `<readwrite>`: Account with readwrite permission, required
|
- `<readwrite>`: Account with readwrite permission, required
|
||||||
- `<readonly>`: Account with readonly permission, optional
|
- `<readonly>`: Account with readonly permission, optional
|
||||||
|
|
||||||
> `*` as `<readonly>` means `<path>` is public, everyone can access/download it.
|
> `<readonly>` can be `*` means `<path>` is public, everyone can access/download it.
|
||||||
|
|
||||||
For example:
|
For example:
|
||||||
|
|
||||||
@@ -174,14 +179,6 @@ dufs -a /@admin:pass@* -a /ui@designer:pass1 -A
|
|||||||
- Account `admin:pass` can upload/delete/download any files/folders.
|
- Account `admin:pass` can upload/delete/download any files/folders.
|
||||||
- Account `designer:pass1` can upload/delete/download any files/folders in the `ui` folder.
|
- Account `designer:pass1` can upload/delete/download any files/folders in the `ui` folder.
|
||||||
|
|
||||||
Curl with digest auth:
|
|
||||||
|
|
||||||
```
|
|
||||||
curl --digest -u designer:pass1 http://127.0.0.1:5000/ui/path-to-file
|
|
||||||
```
|
|
||||||
|
|
||||||
</details>
|
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
Copyright (c) 2022 dufs-developers.
|
Copyright (c) 2022 dufs-developers.
|
||||||
|
|||||||
@@ -1,11 +1,16 @@
|
|||||||
html {
|
html {
|
||||||
font-family: -apple-system,BlinkMacSystemFont,Helvetica,Arial,sans-serif;
|
font-family: -apple-system,BlinkMacSystemFont,Roboto,Helvetica,Arial,sans-serif;
|
||||||
line-height: 1.5;
|
line-height: 1.5;
|
||||||
color: #24292e;
|
color: #24292e;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
/* prevent premature breadcrumb wrapping on mobile */
|
||||||
|
min-width: 500px;
|
||||||
|
}
|
||||||
|
|
||||||
.hidden {
|
.hidden {
|
||||||
display: none;
|
display: none !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.head {
|
.head {
|
||||||
@@ -49,6 +54,11 @@ html {
|
|||||||
margin-right: 10px;
|
margin-right: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.toolbox > div {
|
||||||
|
/* vertically align with breadcrumb text */
|
||||||
|
height: 1.1rem;
|
||||||
|
}
|
||||||
|
|
||||||
.searchbar {
|
.searchbar {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: nowrap;
|
flex-wrap: nowrap;
|
||||||
@@ -116,11 +126,6 @@ html {
|
|||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
.uploaders-table .cell-name,
|
|
||||||
.paths-table .cell-name {
|
|
||||||
width: 500px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.uploaders-table .cell-status {
|
.uploaders-table .cell-status {
|
||||||
width: 80px;
|
width: 80px;
|
||||||
padding-left: 0.6em;
|
padding-left: 0.6em;
|
||||||
@@ -143,7 +148,6 @@ html {
|
|||||||
padding-left: 0.6em;
|
padding-left: 0.6em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
.path svg {
|
.path svg {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
fill: rgba(3,47,98,0.5);
|
fill: rgba(3,47,98,0.5);
|
||||||
@@ -163,7 +167,7 @@ html {
|
|||||||
display: block;
|
display: block;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
max-width: calc(100vw - 375px);
|
max-width: calc(100vw - 375px);
|
||||||
min-width: 400px;
|
min-width: 200px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.path a:hover {
|
.path a:hover {
|
||||||
@@ -182,6 +186,12 @@ html {
|
|||||||
padding-right: 1em;
|
padding-right: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@media (min-width: 768px) {
|
||||||
|
.path a {
|
||||||
|
min-width: 400px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* dark theme */
|
/* dark theme */
|
||||||
@media (prefers-color-scheme: dark) {
|
@media (prefers-color-scheme: dark) {
|
||||||
body {
|
body {
|
||||||
@@ -200,7 +210,8 @@ html {
|
|||||||
}
|
}
|
||||||
|
|
||||||
svg,
|
svg,
|
||||||
.path svg {
|
.path svg,
|
||||||
|
.breadcrumb svg {
|
||||||
fill: #fff;
|
fill: #fff;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -22,7 +22,7 @@
|
|||||||
<input type="file" id="file" name="file" multiple>
|
<input type="file" id="file" name="file" multiple>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<form class="searchbar">
|
<form class="searchbar hidden">
|
||||||
<div class="icon">
|
<div class="icon">
|
||||||
<svg width="16" height="16" fill="currentColor" viewBox="0 0 16 16"><path d="M11.742 10.344a6.5 6.5 0 1 0-1.397 1.398h-.001c.03.04.062.078.098.115l3.85 3.85a1 1 0 0 0 1.415-1.414l-3.85-3.85a1.007 1.007 0 0 0-.115-.1zM12 6.5a5.5 5.5 0 1 1-11 0 5.5 5.5 0 0 1 11 0z"/></svg>
|
<svg width="16" height="16" fill="currentColor" viewBox="0 0 16 16"><path d="M11.742 10.344a6.5 6.5 0 1 0-1.397 1.398h-.001c.03.04.062.078.098.115l3.85 3.85a1 1 0 0 0 1.415-1.414l-3.85-3.85a1.007 1.007 0 0 0-.115-.1zM12 6.5a5.5 5.5 0 1 1-11 0 5.5 5.5 0 0 1 11 0z"/></svg>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -64,14 +64,16 @@ class Uploader {
|
|||||||
|
|
||||||
upload() {
|
upload() {
|
||||||
const { file, idx, name } = this;
|
const { file, idx, name } = this;
|
||||||
let url = getUrl(name);
|
const url = getUrl(name);
|
||||||
|
const encodedUrl = encodedStr(url);
|
||||||
|
const encodedName = encodedStr(name);
|
||||||
$uploadersTable.insertAdjacentHTML("beforeend", `
|
$uploadersTable.insertAdjacentHTML("beforeend", `
|
||||||
<tr id="upload${idx}" class="uploader">
|
<tr id="upload${idx}" class="uploader">
|
||||||
<td class="path cell-icon">
|
<td class="path cell-icon">
|
||||||
${getSvg(file.path_type)}
|
${getSvg(file.path_type)}
|
||||||
</td>
|
</td>
|
||||||
<td class="path cell-name">
|
<td class="path cell-name">
|
||||||
<a href="${url}">${name}</a>
|
<a href="${encodedUrl}">${encodedName}</a>
|
||||||
</td>
|
</td>
|
||||||
<td class="cell-status upload-status" id="uploadStatus${idx}"></td>
|
<td class="cell-status upload-status" id="uploadStatus${idx}"></td>
|
||||||
</tr>`);
|
</tr>`);
|
||||||
@@ -141,12 +143,14 @@ function addBreadcrumb(href, uri_prefix) {
|
|||||||
}
|
}
|
||||||
path += encodeURI(name);
|
path += encodeURI(name);
|
||||||
}
|
}
|
||||||
|
const encodedPath = encodedStr(path);
|
||||||
|
const encodedName = encodedStr(name);
|
||||||
if (i === 0) {
|
if (i === 0) {
|
||||||
$breadcrumb.insertAdjacentHTML("beforeend", `<a href="${path}"><svg width="16" height="16" viewBox="0 0 16 16"><path d="M6.5 14.5v-3.505c0-.245.25-.495.5-.495h2c.25 0 .5.25.5.5v3.5a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5v-7a.5.5 0 0 0-.146-.354L13 5.793V2.5a.5.5 0 0 0-.5-.5h-1a.5.5 0 0 0-.5.5v1.293L8.354 1.146a.5.5 0 0 0-.708 0l-6 6A.5.5 0 0 0 1.5 7.5v7a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5z"/></svg></a>`);
|
$breadcrumb.insertAdjacentHTML("beforeend", `<a href="${encodedPath}"><svg width="16" height="16" viewBox="0 0 16 16"><path d="M6.5 14.5v-3.505c0-.245.25-.495.5-.495h2c.25 0 .5.25.5.5v3.5a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5v-7a.5.5 0 0 0-.146-.354L13 5.793V2.5a.5.5 0 0 0-.5-.5h-1a.5.5 0 0 0-.5.5v1.293L8.354 1.146a.5.5 0 0 0-.708 0l-6 6A.5.5 0 0 0 1.5 7.5v7a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5z"/></svg></a>`);
|
||||||
} else if (i === len - 1) {
|
} else if (i === len - 1) {
|
||||||
$breadcrumb.insertAdjacentHTML("beforeend", `<b>${name}</b>`);
|
$breadcrumb.insertAdjacentHTML("beforeend", `<b>${encodedName}</b>`);
|
||||||
} else {
|
} else {
|
||||||
$breadcrumb.insertAdjacentHTML("beforeend", `<a href="${path}">${name}</a>`);
|
$breadcrumb.insertAdjacentHTML("beforeend", `<a href="${encodedPath}">${encodedName}</a>`);
|
||||||
}
|
}
|
||||||
if (i !== len - 1) {
|
if (i !== len - 1) {
|
||||||
$breadcrumb.insertAdjacentHTML("beforeend", `<span class="separator">/</span>`);
|
$breadcrumb.insertAdjacentHTML("beforeend", `<span class="separator">/</span>`);
|
||||||
@@ -160,28 +164,31 @@ function addBreadcrumb(href, uri_prefix) {
|
|||||||
* @param {number} index
|
* @param {number} index
|
||||||
*/
|
*/
|
||||||
function addPath(file, index) {
|
function addPath(file, index) {
|
||||||
|
const encodedName = encodedStr(file.name);
|
||||||
let url = getUrl(file.name)
|
let url = getUrl(file.name)
|
||||||
|
let encodedUrl = encodedStr(url);
|
||||||
let actionDelete = "";
|
let actionDelete = "";
|
||||||
let actionDownload = "";
|
let actionDownload = "";
|
||||||
if (file.path_type.endsWith("Dir")) {
|
if (file.path_type.endsWith("Dir")) {
|
||||||
url += "/";
|
url += "/";
|
||||||
|
encodedUrl += "/";
|
||||||
actionDownload = `
|
actionDownload = `
|
||||||
<div class="action-btn">
|
<div class="action-btn">
|
||||||
<a href="${url}?zip" title="Download folder as a .zip file">
|
<a href="${encodedUrl}?zip" title="Download folder as a .zip file">
|
||||||
<svg width="16" height="16" viewBox="0 0 16 16"><path d="M.5 9.9a.5.5 0 0 1 .5.5v2.5a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-2.5a.5.5 0 0 1 1 0v2.5a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2v-2.5a.5.5 0 0 1 .5-.5z"/><path d="M7.646 11.854a.5.5 0 0 0 .708 0l3-3a.5.5 0 0 0-.708-.708L8.5 10.293V1.5a.5.5 0 0 0-1 0v8.793L5.354 8.146a.5.5 0 1 0-.708.708l3 3z"/></svg>
|
<svg width="16" height="16" viewBox="0 0 16 16"><path d="M.5 9.9a.5.5 0 0 1 .5.5v2.5a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-2.5a.5.5 0 0 1 1 0v2.5a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2v-2.5a.5.5 0 0 1 .5-.5z"/><path d="M7.646 11.854a.5.5 0 0 0 .708 0l3-3a.5.5 0 0 0-.708-.708L8.5 10.293V1.5a.5.5 0 0 0-1 0v8.793L5.354 8.146a.5.5 0 1 0-.708.708l3 3z"/></svg>
|
||||||
</a>
|
</a>
|
||||||
</div>`;
|
</div>`;
|
||||||
} else {
|
} else {
|
||||||
actionDownload = `
|
actionDownload = `
|
||||||
<div class="action-btn" >
|
<div class="action-btn" >
|
||||||
<a href="${url}" title="Download file" download>
|
<a href="${encodedUrl}" title="Download file" download>
|
||||||
<svg width="16" height="16" viewBox="0 0 16 16"><path d="M.5 9.9a.5.5 0 0 1 .5.5v2.5a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-2.5a.5.5 0 0 1 1 0v2.5a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2v-2.5a.5.5 0 0 1 .5-.5z"/><path d="M7.646 11.854a.5.5 0 0 0 .708 0l3-3a.5.5 0 0 0-.708-.708L8.5 10.293V1.5a.5.5 0 0 0-1 0v8.793L5.354 8.146a.5.5 0 1 0-.708.708l3 3z"/></svg>
|
<svg width="16" height="16" viewBox="0 0 16 16"><path d="M.5 9.9a.5.5 0 0 1 .5.5v2.5a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-2.5a.5.5 0 0 1 1 0v2.5a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2v-2.5a.5.5 0 0 1 .5-.5z"/><path d="M7.646 11.854a.5.5 0 0 0 .708 0l3-3a.5.5 0 0 0-.708-.708L8.5 10.293V1.5a.5.5 0 0 0-1 0v8.793L5.354 8.146a.5.5 0 1 0-.708.708l3 3z"/></svg>
|
||||||
</a>
|
</a>
|
||||||
</div>`;
|
</div>`;
|
||||||
}
|
}
|
||||||
if (DATA.allow_delete) {
|
if (DATA.allow_delete) {
|
||||||
actionDelete = `
|
actionDelete = `
|
||||||
<div onclick="deletePath(${index})" class="action-btn" id="deleteBtn${index}" title="Delete ${file.name}">
|
<div onclick="deletePath(${index})" class="action-btn" id="deleteBtn${index}" title="Delete ${encodedName}">
|
||||||
<svg width="16" height="16" fill="currentColor"viewBox="0 0 16 16"><path d="M6.854 7.146a.5.5 0 1 0-.708.708L7.293 9l-1.147 1.146a.5.5 0 0 0 .708.708L8 9.707l1.146 1.147a.5.5 0 0 0 .708-.708L8.707 9l1.147-1.146a.5.5 0 0 0-.708-.708L8 8.293 6.854 7.146z"/><path d="M14 14V4.5L9.5 0H4a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2zM9.5 3A1.5 1.5 0 0 0 11 4.5h2V14a1 1 0 0 1-1 1H4a1 1 0 0 1-1-1V2a1 1 0 0 1 1-1h5.5v2z"/></svg>
|
<svg width="16" height="16" fill="currentColor"viewBox="0 0 16 16"><path d="M6.854 7.146a.5.5 0 1 0-.708.708L7.293 9l-1.147 1.146a.5.5 0 0 0 .708.708L8 9.707l1.146 1.147a.5.5 0 0 0 .708-.708L8.707 9l1.147-1.146a.5.5 0 0 0-.708-.708L8 8.293 6.854 7.146z"/><path d="M14 14V4.5L9.5 0H4a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2zM9.5 3A1.5 1.5 0 0 0 11 4.5h2V14a1 1 0 0 1-1 1H4a1 1 0 0 1-1-1V2a1 1 0 0 1 1-1h5.5v2z"/></svg>
|
||||||
</div>`;
|
</div>`;
|
||||||
}
|
}
|
||||||
@@ -197,7 +204,7 @@ function addPath(file, index) {
|
|||||||
${getSvg(file.path_type)}
|
${getSvg(file.path_type)}
|
||||||
</td>
|
</td>
|
||||||
<td class="path cell-name">
|
<td class="path cell-name">
|
||||||
<a href="${url}" title="${file.name}">${file.name}</a>
|
<a href="${encodedUrl}">${encodedName}</a>
|
||||||
</td>
|
</td>
|
||||||
<td class="cell-mtime">${formatMtime(file.mtime)}</td>
|
<td class="cell-mtime">${formatMtime(file.mtime)}</td>
|
||||||
<td class="cell-size">${formatSize(file.size).join(" ")}</td>
|
<td class="cell-size">${formatSize(file.size).join(" ")}</td>
|
||||||
@@ -333,15 +340,26 @@ function formatPercent(precent) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function encodedStr(rawStr) {
|
||||||
|
return rawStr.replace(/[\u00A0-\u9999<>\&]/g, function(i) {
|
||||||
|
return '&#'+i.charCodeAt(0)+';';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function ready() {
|
function ready() {
|
||||||
|
document.title = `Index of ${DATA.href} - Dufs`;
|
||||||
$pathsTable = document.querySelector(".paths-table")
|
$pathsTable = document.querySelector(".paths-table")
|
||||||
$pathsTableBody = document.querySelector(".paths-table tbody");
|
$pathsTableBody = document.querySelector(".paths-table tbody");
|
||||||
$uploadersTable = document.querySelector(".uploaders-table");
|
$uploadersTable = document.querySelector(".uploaders-table");
|
||||||
$emptyFolder = document.querySelector(".empty-folder");
|
$emptyFolder = document.querySelector(".empty-folder");
|
||||||
|
|
||||||
|
if (DATA.allow_search) {
|
||||||
|
document.querySelector(".searchbar").classList.remove("hidden");
|
||||||
if (params.q) {
|
if (params.q) {
|
||||||
document.getElementById('search').value = params.q;
|
document.getElementById('search').value = params.q;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
addBreadcrumb(DATA.href, DATA.uri_prefix);
|
addBreadcrumb(DATA.href, DATA.uri_prefix);
|
||||||
if (Array.isArray(DATA.paths)) {
|
if (Array.isArray(DATA.paths)) {
|
||||||
|
|||||||
35
src/args.rs
35
src/args.rs
@@ -5,6 +5,7 @@ use std::net::IpAddr;
|
|||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
use crate::auth::AccessControl;
|
use crate::auth::AccessControl;
|
||||||
|
use crate::auth::AuthMethod;
|
||||||
use crate::tls::{load_certs, load_private_key};
|
use crate::tls::{load_certs, load_private_key};
|
||||||
use crate::BoxResult;
|
use crate::BoxResult;
|
||||||
|
|
||||||
@@ -39,13 +40,13 @@ fn app() -> Command<'static> {
|
|||||||
Arg::new("path")
|
Arg::new("path")
|
||||||
.default_value(".")
|
.default_value(".")
|
||||||
.allow_invalid_utf8(true)
|
.allow_invalid_utf8(true)
|
||||||
.help("Path to a root directory for serving files"),
|
.help("Specific path to serve"),
|
||||||
)
|
)
|
||||||
.arg(
|
.arg(
|
||||||
Arg::new("path-prefix")
|
Arg::new("path-prefix")
|
||||||
.long("path-prefix")
|
.long("path-prefix")
|
||||||
.value_name("path")
|
.value_name("path")
|
||||||
.help("Specify an url path prefix"),
|
.help("Specify an path prefix"),
|
||||||
)
|
)
|
||||||
.arg(
|
.arg(
|
||||||
Arg::new("auth")
|
Arg::new("auth")
|
||||||
@@ -56,6 +57,14 @@ fn app() -> Command<'static> {
|
|||||||
.multiple_occurrences(true)
|
.multiple_occurrences(true)
|
||||||
.value_name("rule"),
|
.value_name("rule"),
|
||||||
)
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("auth-method")
|
||||||
|
.long("auth-method")
|
||||||
|
.help("Select auth method")
|
||||||
|
.possible_values(["basic", "digest"])
|
||||||
|
.default_value("digest")
|
||||||
|
.value_name("value"),
|
||||||
|
)
|
||||||
.arg(
|
.arg(
|
||||||
Arg::new("allow-all")
|
Arg::new("allow-all")
|
||||||
.short('A')
|
.short('A')
|
||||||
@@ -72,6 +81,11 @@ fn app() -> Command<'static> {
|
|||||||
.long("allow-delete")
|
.long("allow-delete")
|
||||||
.help("Allow delete files/folders"),
|
.help("Allow delete files/folders"),
|
||||||
)
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::new("allow-search")
|
||||||
|
.long("allow-search")
|
||||||
|
.help("Allow search files/folders"),
|
||||||
|
)
|
||||||
.arg(
|
.arg(
|
||||||
Arg::new("allow-symlink")
|
Arg::new("allow-symlink")
|
||||||
.long("allow-symlink")
|
.long("allow-symlink")
|
||||||
@@ -85,17 +99,17 @@ fn app() -> Command<'static> {
|
|||||||
.arg(
|
.arg(
|
||||||
Arg::new("render-index")
|
Arg::new("render-index")
|
||||||
.long("render-index")
|
.long("render-index")
|
||||||
.help("Render index.html when requesting a directory"),
|
.help("Serve index.html when requesting a directory, returns 404 if not found index.html"),
|
||||||
)
|
)
|
||||||
.arg(
|
.arg(
|
||||||
Arg::new("render-try-index")
|
Arg::new("render-try-index")
|
||||||
.long("render-try-index")
|
.long("render-try-index")
|
||||||
.help("Render index.html if it exists when requesting a directory"),
|
.help("Serve index.html when requesting a directory, returns file listing if not found index.html"),
|
||||||
)
|
)
|
||||||
.arg(
|
.arg(
|
||||||
Arg::new("render-spa")
|
Arg::new("render-spa")
|
||||||
.long("render-spa")
|
.long("render-spa")
|
||||||
.help("Render for single-page application"),
|
.help("Serve SPA(Single Page Application)"),
|
||||||
)
|
)
|
||||||
.arg(
|
.arg(
|
||||||
Arg::new("tls-cert")
|
Arg::new("tls-cert")
|
||||||
@@ -115,7 +129,7 @@ pub fn matches() -> ArgMatches {
|
|||||||
app().get_matches()
|
app().get_matches()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug)]
|
||||||
pub struct Args {
|
pub struct Args {
|
||||||
pub addrs: Vec<IpAddr>,
|
pub addrs: Vec<IpAddr>,
|
||||||
pub port: u16,
|
pub port: u16,
|
||||||
@@ -123,9 +137,11 @@ pub struct Args {
|
|||||||
pub path_is_file: bool,
|
pub path_is_file: bool,
|
||||||
pub path_prefix: String,
|
pub path_prefix: String,
|
||||||
pub uri_prefix: String,
|
pub uri_prefix: String,
|
||||||
|
pub auth_method: AuthMethod,
|
||||||
pub auth: AccessControl,
|
pub auth: AccessControl,
|
||||||
pub allow_upload: bool,
|
pub allow_upload: bool,
|
||||||
pub allow_delete: bool,
|
pub allow_delete: bool,
|
||||||
|
pub allow_search: bool,
|
||||||
pub allow_symlink: bool,
|
pub allow_symlink: bool,
|
||||||
pub render_index: bool,
|
pub render_index: bool,
|
||||||
pub render_spa: bool,
|
pub render_spa: bool,
|
||||||
@@ -162,9 +178,14 @@ impl Args {
|
|||||||
.values_of("auth")
|
.values_of("auth")
|
||||||
.map(|v| v.collect())
|
.map(|v| v.collect())
|
||||||
.unwrap_or_default();
|
.unwrap_or_default();
|
||||||
|
let auth_method = match matches.value_of("auth-method").unwrap() {
|
||||||
|
"basic" => AuthMethod::Basic,
|
||||||
|
_ => AuthMethod::Digest,
|
||||||
|
};
|
||||||
let auth = AccessControl::new(&auth, &uri_prefix)?;
|
let auth = AccessControl::new(&auth, &uri_prefix)?;
|
||||||
let allow_upload = matches.is_present("allow-all") || matches.is_present("allow-upload");
|
let allow_upload = matches.is_present("allow-all") || matches.is_present("allow-upload");
|
||||||
let allow_delete = matches.is_present("allow-all") || matches.is_present("allow-delete");
|
let allow_delete = matches.is_present("allow-all") || matches.is_present("allow-delete");
|
||||||
|
let allow_search = matches.is_present("allow-all") || matches.is_present("allow-search");
|
||||||
let allow_symlink = matches.is_present("allow-all") || matches.is_present("allow-symlink");
|
let allow_symlink = matches.is_present("allow-all") || matches.is_present("allow-symlink");
|
||||||
let render_index = matches.is_present("render-index");
|
let render_index = matches.is_present("render-index");
|
||||||
let render_try_index = matches.is_present("render-try-index");
|
let render_try_index = matches.is_present("render-try-index");
|
||||||
@@ -185,10 +206,12 @@ impl Args {
|
|||||||
path_is_file,
|
path_is_file,
|
||||||
path_prefix,
|
path_prefix,
|
||||||
uri_prefix,
|
uri_prefix,
|
||||||
|
auth_method,
|
||||||
auth,
|
auth,
|
||||||
enable_cors,
|
enable_cors,
|
||||||
allow_delete,
|
allow_delete,
|
||||||
allow_upload,
|
allow_upload,
|
||||||
|
allow_search,
|
||||||
allow_symlink,
|
allow_symlink,
|
||||||
render_index,
|
render_index,
|
||||||
render_try_index,
|
render_try_index,
|
||||||
|
|||||||
65
src/auth.rs
65
src/auth.rs
@@ -22,12 +22,12 @@ lazy_static! {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug)]
|
||||||
pub struct AccessControl {
|
pub struct AccessControl {
|
||||||
rules: HashMap<String, PathControl>,
|
rules: HashMap<String, PathControl>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug)]
|
||||||
pub struct PathControl {
|
pub struct PathControl {
|
||||||
readwrite: Account,
|
readwrite: Account,
|
||||||
readonly: Option<Account>,
|
readonly: Option<Account>,
|
||||||
@@ -76,6 +76,7 @@ impl AccessControl {
|
|||||||
path: &str,
|
path: &str,
|
||||||
method: &Method,
|
method: &Method,
|
||||||
authorization: Option<&HeaderValue>,
|
authorization: Option<&HeaderValue>,
|
||||||
|
auth_method: AuthMethod,
|
||||||
) -> GuardType {
|
) -> GuardType {
|
||||||
if self.rules.is_empty() {
|
if self.rules.is_empty() {
|
||||||
return GuardType::ReadWrite;
|
return GuardType::ReadWrite;
|
||||||
@@ -86,7 +87,10 @@ impl AccessControl {
|
|||||||
controls.push(control);
|
controls.push(control);
|
||||||
if let Some(authorization) = authorization {
|
if let Some(authorization) = authorization {
|
||||||
let Account { user, pass } = &control.readwrite;
|
let Account { user, pass } = &control.readwrite;
|
||||||
if valid_digest(authorization, method.as_str(), user, pass).is_some() {
|
if auth_method
|
||||||
|
.validate(authorization, method.as_str(), user, pass)
|
||||||
|
.is_some()
|
||||||
|
{
|
||||||
return GuardType::ReadWrite;
|
return GuardType::ReadWrite;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -99,7 +103,10 @@ impl AccessControl {
|
|||||||
}
|
}
|
||||||
if let Some(authorization) = authorization {
|
if let Some(authorization) = authorization {
|
||||||
if let Some(Account { user, pass }) = &control.readonly {
|
if let Some(Account { user, pass }) = &control.readonly {
|
||||||
if valid_digest(authorization, method.as_str(), user, pass).is_some() {
|
if auth_method
|
||||||
|
.validate(authorization, method.as_str(), user, pass)
|
||||||
|
.is_some()
|
||||||
|
{
|
||||||
return GuardType::ReadOnly;
|
return GuardType::ReadOnly;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -167,7 +174,19 @@ impl Account {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn generate_www_auth(stale: bool) -> String {
|
#[derive(Debug, Clone)]
|
||||||
|
pub enum AuthMethod {
|
||||||
|
Basic,
|
||||||
|
Digest,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AuthMethod {
|
||||||
|
pub fn www_auth(&self, stale: bool) -> String {
|
||||||
|
match self {
|
||||||
|
AuthMethod::Basic => {
|
||||||
|
format!("Basic realm=\"{}\"", REALM)
|
||||||
|
}
|
||||||
|
AuthMethod::Digest => {
|
||||||
let str_stale = if stale { "stale=true," } else { "" };
|
let str_stale = if stale { "stale=true," } else { "" };
|
||||||
format!(
|
format!(
|
||||||
"Digest realm=\"{}\",nonce=\"{}\",{}qop=\"auth\"",
|
"Digest realm=\"{}\",nonce=\"{}\",{}qop=\"auth\"",
|
||||||
@@ -175,14 +194,39 @@ pub fn generate_www_auth(stale: bool) -> String {
|
|||||||
create_nonce(),
|
create_nonce(),
|
||||||
str_stale
|
str_stale
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
pub fn valid_digest(
|
}
|
||||||
|
pub fn validate(
|
||||||
|
&self,
|
||||||
authorization: &HeaderValue,
|
authorization: &HeaderValue,
|
||||||
method: &str,
|
method: &str,
|
||||||
auth_user: &str,
|
auth_user: &str,
|
||||||
auth_pass: &str,
|
auth_pass: &str,
|
||||||
) -> Option<()> {
|
) -> Option<()> {
|
||||||
|
match self {
|
||||||
|
AuthMethod::Basic => {
|
||||||
|
let value: Vec<u8> =
|
||||||
|
base64::decode(strip_prefix(authorization.as_bytes(), b"Basic ").unwrap())
|
||||||
|
.unwrap();
|
||||||
|
let parts: Vec<&str> = std::str::from_utf8(&value).unwrap().split(':').collect();
|
||||||
|
|
||||||
|
if parts[0] != auth_user {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut h = Context::new();
|
||||||
|
h.consume(format!("{}:{}:{}", parts[0], REALM, parts[1]).as_bytes());
|
||||||
|
|
||||||
|
let http_pass = format!("{:x}", h.compute());
|
||||||
|
|
||||||
|
if http_pass == auth_pass {
|
||||||
|
return Some(());
|
||||||
|
}
|
||||||
|
|
||||||
|
None
|
||||||
|
}
|
||||||
|
AuthMethod::Digest => {
|
||||||
let digest_value = strip_prefix(authorization.as_bytes(), b"Digest ")?;
|
let digest_value = strip_prefix(authorization.as_bytes(), b"Digest ")?;
|
||||||
let user_vals = to_headermap(digest_value).ok()?;
|
let user_vals = to_headermap(digest_value).ok()?;
|
||||||
if let (Some(username), Some(nonce), Some(user_response)) = (
|
if let (Some(username), Some(nonce), Some(user_response)) = (
|
||||||
@@ -248,6 +292,9 @@ pub fn valid_digest(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
None
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Check if a nonce is still valid.
|
/// Check if a nonce is still valid.
|
||||||
|
|||||||
30
src/logger.rs
Normal file
30
src/logger.rs
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
use chrono::{Local, SecondsFormat};
|
||||||
|
use log::{Level, Metadata, Record};
|
||||||
|
use log::{LevelFilter, SetLoggerError};
|
||||||
|
|
||||||
|
struct SimpleLogger;
|
||||||
|
|
||||||
|
impl log::Log for SimpleLogger {
|
||||||
|
fn enabled(&self, metadata: &Metadata) -> bool {
|
||||||
|
metadata.level() <= Level::Info
|
||||||
|
}
|
||||||
|
|
||||||
|
fn log(&self, record: &Record) {
|
||||||
|
if self.enabled(record.metadata()) {
|
||||||
|
let timestamp = Local::now().to_rfc3339_opts(SecondsFormat::Secs, true);
|
||||||
|
if record.level() < Level::Info {
|
||||||
|
eprintln!("{} {} - {}", timestamp, record.level(), record.args());
|
||||||
|
} else {
|
||||||
|
println!("{} {} - {}", timestamp, record.level(), record.args());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn flush(&self) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
static LOGGER: SimpleLogger = SimpleLogger;
|
||||||
|
|
||||||
|
pub fn init() -> Result<(), SetLoggerError> {
|
||||||
|
log::set_logger(&LOGGER).map(|()| log::set_max_level(LevelFilter::Info))
|
||||||
|
}
|
||||||
15
src/main.rs
15
src/main.rs
@@ -1,5 +1,6 @@
|
|||||||
mod args;
|
mod args;
|
||||||
mod auth;
|
mod auth;
|
||||||
|
mod logger;
|
||||||
mod server;
|
mod server;
|
||||||
mod streamer;
|
mod streamer;
|
||||||
mod tls;
|
mod tls;
|
||||||
@@ -12,9 +13,8 @@ use crate::args::{matches, Args};
|
|||||||
use crate::server::{Request, Server};
|
use crate::server::{Request, Server};
|
||||||
use crate::tls::{TlsAcceptor, TlsStream};
|
use crate::tls::{TlsAcceptor, TlsStream};
|
||||||
|
|
||||||
use std::io::Write;
|
|
||||||
use std::net::{IpAddr, SocketAddr, TcpListener as StdTcpListener};
|
use std::net::{IpAddr, SocketAddr, TcpListener as StdTcpListener};
|
||||||
use std::{env, sync::Arc};
|
use std::sync::Arc;
|
||||||
|
|
||||||
use futures::future::join_all;
|
use futures::future::join_all;
|
||||||
use tokio::net::TcpListener;
|
use tokio::net::TcpListener;
|
||||||
@@ -32,16 +32,7 @@ async fn main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async fn run() -> BoxResult<()> {
|
async fn run() -> BoxResult<()> {
|
||||||
if env::var("RUST_LOG").is_err() {
|
logger::init().map_err(|e| format!("Failed to init logger, {}", e))?;
|
||||||
env::set_var("RUST_LOG", "info")
|
|
||||||
}
|
|
||||||
env_logger::builder()
|
|
||||||
.format(|buf, record| {
|
|
||||||
let timestamp = buf.timestamp_millis();
|
|
||||||
writeln!(buf, "[{} {}] {}", timestamp, record.level(), record.args())
|
|
||||||
})
|
|
||||||
.init();
|
|
||||||
|
|
||||||
let args = Args::parse(matches())?;
|
let args = Args::parse(matches())?;
|
||||||
let args = Arc::new(args);
|
let args = Arc::new(args);
|
||||||
let handles = serve(args.clone())?;
|
let handles = serve(args.clone())?;
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
use crate::auth::generate_www_auth;
|
|
||||||
use crate::streamer::Streamer;
|
use crate::streamer::Streamer;
|
||||||
use crate::utils::{decode_uri, encode_uri};
|
use crate::utils::{decode_uri, encode_uri};
|
||||||
use crate::{Args, BoxResult};
|
use crate::{Args, BoxResult};
|
||||||
@@ -45,11 +44,16 @@ const BUF_SIZE: usize = 65536;
|
|||||||
|
|
||||||
pub struct Server {
|
pub struct Server {
|
||||||
args: Arc<Args>,
|
args: Arc<Args>,
|
||||||
|
assets_prefix: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Server {
|
impl Server {
|
||||||
pub fn new(args: Arc<Args>) -> Self {
|
pub fn new(args: Arc<Args>) -> Self {
|
||||||
Self { args }
|
let assets_prefix = format!("{}__dufs_v{}_", args.uri_prefix, env!("CARGO_PKG_VERSION"));
|
||||||
|
Self {
|
||||||
|
args,
|
||||||
|
assets_prefix,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn call(
|
pub async fn call(
|
||||||
@@ -59,12 +63,15 @@ impl Server {
|
|||||||
) -> Result<Response, hyper::Error> {
|
) -> Result<Response, hyper::Error> {
|
||||||
let method = req.method().clone();
|
let method = req.method().clone();
|
||||||
let uri = req.uri().clone();
|
let uri = req.uri().clone();
|
||||||
|
let assets_prefix = self.assets_prefix.clone();
|
||||||
let enable_cors = self.args.enable_cors;
|
let enable_cors = self.args.enable_cors;
|
||||||
|
|
||||||
let mut res = match self.handle(req).await {
|
let mut res = match self.handle(req).await {
|
||||||
Ok(res) => {
|
Ok(res) => {
|
||||||
let status = res.status().as_u16();
|
let status = res.status().as_u16();
|
||||||
|
if !uri.path().starts_with(&assets_prefix) {
|
||||||
info!(r#"{} "{} {}" - {}"#, addr.ip(), method, uri, status,);
|
info!(r#"{} "{} {}" - {}"#, addr.ip(), method, uri, status,);
|
||||||
|
}
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
@@ -90,13 +97,17 @@ impl Server {
|
|||||||
let headers = req.headers();
|
let headers = req.headers();
|
||||||
let method = req.method().clone();
|
let method = req.method().clone();
|
||||||
|
|
||||||
if req_path == "/favicon.ico" && method == Method::GET {
|
if method == Method::GET && self.handle_embed_assets(req_path, &mut res).await? {
|
||||||
self.handle_send_favicon(headers, &mut res).await?;
|
|
||||||
return Ok(res);
|
return Ok(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
let authorization = headers.get(AUTHORIZATION);
|
let authorization = headers.get(AUTHORIZATION);
|
||||||
let guard_type = self.args.auth.guard(req_path, &method, authorization);
|
let guard_type = self.args.auth.guard(
|
||||||
|
req_path,
|
||||||
|
&method,
|
||||||
|
authorization,
|
||||||
|
self.args.auth_method.clone(),
|
||||||
|
);
|
||||||
if guard_type.is_reject() {
|
if guard_type.is_reject() {
|
||||||
self.auth_reject(&mut res);
|
self.auth_reject(&mut res);
|
||||||
return Ok(res);
|
return Ok(res);
|
||||||
@@ -129,6 +140,7 @@ impl Server {
|
|||||||
|
|
||||||
let allow_upload = self.args.allow_upload;
|
let allow_upload = self.args.allow_upload;
|
||||||
let allow_delete = self.args.allow_delete;
|
let allow_delete = self.args.allow_delete;
|
||||||
|
let allow_search = self.args.allow_search;
|
||||||
let render_index = self.args.render_index;
|
let render_index = self.args.render_index;
|
||||||
let render_spa = self.args.render_spa;
|
let render_spa = self.args.render_spa;
|
||||||
let render_try_index = self.args.render_try_index;
|
let render_try_index = self.args.render_try_index;
|
||||||
@@ -148,8 +160,9 @@ impl Server {
|
|||||||
.await?;
|
.await?;
|
||||||
} else if query == "zip" {
|
} else if query == "zip" {
|
||||||
self.handle_zip_dir(path, head_only, &mut res).await?;
|
self.handle_zip_dir(path, head_only, &mut res).await?;
|
||||||
} else if let Some(q) = query.strip_prefix("q=") {
|
} else if allow_search && query.starts_with("q=") {
|
||||||
self.handle_query_dir(path, q, head_only, &mut res).await?;
|
let q = decode_uri(&query[2..]).unwrap_or_default();
|
||||||
|
self.handle_query_dir(path, &q, head_only, &mut res).await?;
|
||||||
} else {
|
} else {
|
||||||
self.handle_ls_dir(path, true, head_only, &mut res).await?;
|
self.handle_ls_dir(path, true, head_only, &mut res).await?;
|
||||||
}
|
}
|
||||||
@@ -412,23 +425,38 @@ impl Server {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn handle_send_favicon(
|
async fn handle_embed_assets(&self, req_path: &str, res: &mut Response) -> BoxResult<bool> {
|
||||||
&self,
|
if let Some(name) = req_path.strip_prefix(&self.assets_prefix) {
|
||||||
headers: &HeaderMap<HeaderValue>,
|
match name {
|
||||||
res: &mut Response,
|
"index.js" => {
|
||||||
) -> BoxResult<()> {
|
*res.body_mut() = Body::from(INDEX_JS);
|
||||||
let path = self.args.path.join("favicon.ico");
|
res.headers_mut().insert(
|
||||||
let meta = fs::metadata(&path).await.ok();
|
"content-type",
|
||||||
let is_file = meta.map(|v| v.is_file()).unwrap_or_default();
|
HeaderValue::from_static("application/javascript"),
|
||||||
if is_file {
|
);
|
||||||
self.handle_send_file(path.as_path(), headers, false, res)
|
}
|
||||||
.await?;
|
"index.css" => {
|
||||||
} else {
|
*res.body_mut() = Body::from(INDEX_CSS);
|
||||||
|
res.headers_mut()
|
||||||
|
.insert("content-type", HeaderValue::from_static("text/css"));
|
||||||
|
}
|
||||||
|
"favicon.ico" => {
|
||||||
*res.body_mut() = Body::from(FAVICON_ICO);
|
*res.body_mut() = Body::from(FAVICON_ICO);
|
||||||
res.headers_mut()
|
res.headers_mut()
|
||||||
.insert("content-type", HeaderValue::from_static("image/x-icon"));
|
.insert("content-type", HeaderValue::from_static("image/x-icon"));
|
||||||
}
|
}
|
||||||
Ok(())
|
_ => {
|
||||||
|
return Ok(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
res.headers_mut().insert(
|
||||||
|
"cache-control",
|
||||||
|
HeaderValue::from_static("max-age=2592000, public"),
|
||||||
|
);
|
||||||
|
Ok(true)
|
||||||
|
} else {
|
||||||
|
Ok(false)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn handle_send_file(
|
async fn handle_send_file(
|
||||||
@@ -686,26 +714,30 @@ impl Server {
|
|||||||
paths.sort_unstable();
|
paths.sort_unstable();
|
||||||
let href = format!("/{}", normalize_path(path.strip_prefix(&self.args.path)?));
|
let href = format!("/{}", normalize_path(path.strip_prefix(&self.args.path)?));
|
||||||
let data = IndexData {
|
let data = IndexData {
|
||||||
href: href.clone(),
|
href,
|
||||||
uri_prefix: self.args.uri_prefix.clone(),
|
uri_prefix: self.args.uri_prefix.clone(),
|
||||||
paths,
|
paths,
|
||||||
allow_upload: self.args.allow_upload,
|
allow_upload: self.args.allow_upload,
|
||||||
allow_delete: self.args.allow_delete,
|
allow_delete: self.args.allow_delete,
|
||||||
|
allow_search: self.args.allow_search,
|
||||||
dir_exists: exist,
|
dir_exists: exist,
|
||||||
};
|
};
|
||||||
let data = serde_json::to_string(&data).unwrap();
|
let data = serde_json::to_string(&data).unwrap();
|
||||||
|
let asset_js = format!("{}index.js", self.assets_prefix);
|
||||||
|
let asset_css = format!("{}index.css", self.assets_prefix);
|
||||||
|
let asset_ico = format!("{}favicon.ico", self.assets_prefix);
|
||||||
let output = INDEX_HTML.replace(
|
let output = INDEX_HTML.replace(
|
||||||
"__SLOT__",
|
"__SLOT__",
|
||||||
&format!(
|
&format!(
|
||||||
r#"
|
r#"
|
||||||
<title>Index of {} - Dufs</title>
|
<link rel="icon" type="image/x-icon" href="{}">
|
||||||
<style>{}</style>
|
<link rel="stylesheet" href="{}">
|
||||||
<script>
|
<script>
|
||||||
const DATA =
|
DATA = {}
|
||||||
{}
|
</script>
|
||||||
{}</script>
|
<script src="{}"></script>
|
||||||
"#,
|
"#,
|
||||||
href, INDEX_CSS, data, INDEX_JS
|
asset_ico, asset_css, data, asset_js
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
res.headers_mut()
|
res.headers_mut()
|
||||||
@@ -720,7 +752,7 @@ const DATA =
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn auth_reject(&self, res: &mut Response) {
|
fn auth_reject(&self, res: &mut Response) {
|
||||||
let value = generate_www_auth(false);
|
let value = self.args.auth_method.www_auth(false);
|
||||||
set_webdav_headers(res);
|
set_webdav_headers(res);
|
||||||
res.headers_mut().typed_insert(Connection::close());
|
res.headers_mut().typed_insert(Connection::close());
|
||||||
res.headers_mut()
|
res.headers_mut()
|
||||||
@@ -819,6 +851,7 @@ struct IndexData {
|
|||||||
paths: Vec<PathItem>,
|
paths: Vec<PathItem>,
|
||||||
allow_upload: bool,
|
allow_upload: bool,
|
||||||
allow_delete: bool,
|
allow_delete: bool,
|
||||||
|
allow_search: bool,
|
||||||
dir_exists: bool,
|
dir_exists: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -59,3 +59,15 @@ fn allow_upload_delete_can_override(#[with(&["-A"])] server: TestServer) -> Resu
|
|||||||
assert_eq!(resp.status(), 201);
|
assert_eq!(resp.status(), 201);
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[rstest]
|
||||||
|
fn allow_search(#[with(&["--allow-search"])] server: TestServer) -> Result<(), Error> {
|
||||||
|
let resp = reqwest::blocking::get(format!("{}?q={}", server.url(), "test.html"))?;
|
||||||
|
assert_eq!(resp.status(), 200);
|
||||||
|
let paths = utils::retrive_index_paths(&resp.text()?);
|
||||||
|
assert!(!paths.is_empty());
|
||||||
|
for p in paths {
|
||||||
|
assert!(p.contains(&"test.html"));
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|||||||
@@ -36,7 +36,6 @@ fn path_prefix_propfind(
|
|||||||
#[case("index.html")]
|
#[case("index.html")]
|
||||||
fn serve_single_file(tmpdir: TempDir, port: u16, #[case] file: &str) -> Result<(), Error> {
|
fn serve_single_file(tmpdir: TempDir, port: u16, #[case] file: &str) -> Result<(), Error> {
|
||||||
let mut child = Command::cargo_bin("dufs")?
|
let mut child = Command::cargo_bin("dufs")?
|
||||||
.env("RUST_LOG", "false")
|
|
||||||
.arg(tmpdir.path().join(file))
|
.arg(tmpdir.path().join(file))
|
||||||
.arg("-p")
|
.arg("-p")
|
||||||
.arg(port.to_string())
|
.arg(port.to_string())
|
||||||
|
|||||||
61
tests/assets.rs
Normal file
61
tests/assets.rs
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
mod fixtures;
|
||||||
|
mod utils;
|
||||||
|
|
||||||
|
use fixtures::{server, Error, TestServer};
|
||||||
|
use rstest::rstest;
|
||||||
|
|
||||||
|
#[rstest]
|
||||||
|
fn assets(server: TestServer) -> Result<(), Error> {
|
||||||
|
let ver = env!("CARGO_PKG_VERSION");
|
||||||
|
let resp = reqwest::blocking::get(server.url())?;
|
||||||
|
let index_js = format!("/__dufs_v{}_index.js", ver);
|
||||||
|
let index_css = format!("/__dufs_v{}_index.css", ver);
|
||||||
|
let favicon_ico = format!("/__dufs_v{}_favicon.ico", ver);
|
||||||
|
let text = resp.text()?;
|
||||||
|
assert!(text.contains(&format!(r#"href="{}""#, index_css)));
|
||||||
|
assert!(text.contains(&format!(r#"href="{}""#, favicon_ico)));
|
||||||
|
assert!(text.contains(&format!(r#"src="{}""#, index_js)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[rstest]
|
||||||
|
fn asset_js(server: TestServer) -> Result<(), Error> {
|
||||||
|
let url = format!(
|
||||||
|
"{}__dufs_v{}_index.js",
|
||||||
|
server.url(),
|
||||||
|
env!("CARGO_PKG_VERSION")
|
||||||
|
);
|
||||||
|
let resp = reqwest::blocking::get(url)?;
|
||||||
|
assert_eq!(resp.status(), 200);
|
||||||
|
assert_eq!(
|
||||||
|
resp.headers().get("content-type").unwrap(),
|
||||||
|
"application/javascript"
|
||||||
|
);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[rstest]
|
||||||
|
fn asset_css(server: TestServer) -> Result<(), Error> {
|
||||||
|
let url = format!(
|
||||||
|
"{}__dufs_v{}_index.css",
|
||||||
|
server.url(),
|
||||||
|
env!("CARGO_PKG_VERSION")
|
||||||
|
);
|
||||||
|
let resp = reqwest::blocking::get(url)?;
|
||||||
|
assert_eq!(resp.status(), 200);
|
||||||
|
assert_eq!(resp.headers().get("content-type").unwrap(), "text/css");
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[rstest]
|
||||||
|
fn asset_ico(server: TestServer) -> Result<(), Error> {
|
||||||
|
let url = format!(
|
||||||
|
"{}__dufs_v{}_favicon.ico",
|
||||||
|
server.url(),
|
||||||
|
env!("CARGO_PKG_VERSION")
|
||||||
|
);
|
||||||
|
let resp = reqwest::blocking::get(url)?;
|
||||||
|
assert_eq!(resp.status(), 200);
|
||||||
|
assert_eq!(resp.headers().get("content-type").unwrap(), "image/x-icon");
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
@@ -80,3 +80,18 @@ fn auth_nest_share(
|
|||||||
assert_eq!(resp.status(), 200);
|
assert_eq!(resp.status(), 200);
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[rstest]
|
||||||
|
fn auth_basic(
|
||||||
|
#[with(&["--auth", "/@user:pass", "--auth-method", "basic", "-A"])] server: TestServer,
|
||||||
|
) -> Result<(), Error> {
|
||||||
|
let url = format!("{}file1", server.url());
|
||||||
|
let resp = fetch!(b"PUT", &url).body(b"abc".to_vec()).send()?;
|
||||||
|
assert_eq!(resp.status(), 401);
|
||||||
|
let resp = fetch!(b"PUT", &url)
|
||||||
|
.body(b"abc".to_vec())
|
||||||
|
.basic_auth("user", Some("pass"))
|
||||||
|
.send()?;
|
||||||
|
assert_eq!(resp.status(), 201);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|||||||
@@ -6,14 +6,13 @@ use assert_cmd::prelude::*;
|
|||||||
use assert_fs::fixture::TempDir;
|
use assert_fs::fixture::TempDir;
|
||||||
use regex::Regex;
|
use regex::Regex;
|
||||||
use rstest::rstest;
|
use rstest::rstest;
|
||||||
use std::io::{BufRead, BufReader};
|
use std::io::Read;
|
||||||
use std::process::{Command, Stdio};
|
use std::process::{Command, Stdio};
|
||||||
|
|
||||||
#[rstest]
|
#[rstest]
|
||||||
#[case(&["-b", "20.205.243.166"])]
|
#[case(&["-b", "20.205.243.166"])]
|
||||||
fn bind_fails(tmpdir: TempDir, port: u16, #[case] args: &[&str]) -> Result<(), Error> {
|
fn bind_fails(tmpdir: TempDir, port: u16, #[case] args: &[&str]) -> Result<(), Error> {
|
||||||
Command::cargo_bin("dufs")?
|
Command::cargo_bin("dufs")?
|
||||||
.env("RUST_LOG", "false")
|
|
||||||
.arg(tmpdir.path())
|
.arg(tmpdir.path())
|
||||||
.arg("-p")
|
.arg("-p")
|
||||||
.arg(port.to_string())
|
.arg(port.to_string())
|
||||||
@@ -51,7 +50,6 @@ fn bind_ipv4_ipv6(
|
|||||||
#[case(&["--path-prefix", "/prefix"])]
|
#[case(&["--path-prefix", "/prefix"])]
|
||||||
fn validate_printed_urls(tmpdir: TempDir, port: u16, #[case] args: &[&str]) -> Result<(), Error> {
|
fn validate_printed_urls(tmpdir: TempDir, port: u16, #[case] args: &[&str]) -> Result<(), Error> {
|
||||||
let mut child = Command::cargo_bin("dufs")?
|
let mut child = Command::cargo_bin("dufs")?
|
||||||
.env("RUST_LOG", "false")
|
|
||||||
.arg(tmpdir.path())
|
.arg(tmpdir.path())
|
||||||
.arg("-p")
|
.arg("-p")
|
||||||
.arg(port.to_string())
|
.arg(port.to_string())
|
||||||
@@ -61,22 +59,23 @@ fn validate_printed_urls(tmpdir: TempDir, port: u16, #[case] args: &[&str]) -> R
|
|||||||
|
|
||||||
wait_for_port(port);
|
wait_for_port(port);
|
||||||
|
|
||||||
// WARN assumes urls list is terminated by an empty line
|
let stdout = child.stdout.as_mut().expect("Failed to get stdout");
|
||||||
let url_lines = BufReader::new(child.stdout.take().unwrap())
|
let mut buf = [0; 1000];
|
||||||
|
let buf_len = stdout.read(&mut buf)?;
|
||||||
|
let output = std::str::from_utf8(&buf[0..buf_len])?;
|
||||||
|
let url_lines = output
|
||||||
.lines()
|
.lines()
|
||||||
.map(|line| line.expect("Error reading stdout"))
|
|
||||||
.take_while(|line| !line.is_empty()) /* non-empty lines */
|
.take_while(|line| !line.is_empty()) /* non-empty lines */
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>()
|
||||||
let url_lines = url_lines.join("\n");
|
.join("\n");
|
||||||
|
|
||||||
let urls = Regex::new(r"http://[a-zA-Z0-9\.\[\]:/]+")
|
let urls = Regex::new(r"http://[a-zA-Z0-9\.\[\]:/]+")
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.captures_iter(url_lines.as_str())
|
.captures_iter(url_lines.as_str())
|
||||||
.map(|caps| caps.get(0).unwrap().as_str())
|
.filter_map(|caps| caps.get(0).map(|v| v.as_str()))
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
assert!(!urls.is_empty());
|
assert!(!urls.is_empty());
|
||||||
|
|
||||||
for url in urls {
|
for url in urls {
|
||||||
reqwest::blocking::get(url)?.error_for_status()?;
|
reqwest::blocking::get(url)?.error_for_status()?;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,25 +0,0 @@
|
|||||||
mod fixtures;
|
|
||||||
mod utils;
|
|
||||||
|
|
||||||
use fixtures::{server, Error, TestServer};
|
|
||||||
use rstest::rstest;
|
|
||||||
|
|
||||||
#[rstest]
|
|
||||||
fn default_favicon(server: TestServer) -> Result<(), Error> {
|
|
||||||
let resp = reqwest::blocking::get(format!("{}favicon.ico", server.url()))?;
|
|
||||||
assert_eq!(resp.status(), 200);
|
|
||||||
assert_eq!(resp.headers().get("content-type").unwrap(), "image/x-icon");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[rstest]
|
|
||||||
fn exist_favicon(#[with(&["-A"])] server: TestServer) -> Result<(), Error> {
|
|
||||||
let url = format!("{}favicon.ico", server.url());
|
|
||||||
let data = b"abc";
|
|
||||||
let resp = fetch!(b"PUT", &url).body(data.to_vec()).send()?;
|
|
||||||
assert_eq!(resp.status(), 201);
|
|
||||||
let resp = reqwest::blocking::get(url)?;
|
|
||||||
assert_eq!(resp.status(), 200);
|
|
||||||
assert_eq!(resp.bytes()?, data.to_vec());
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
@@ -13,21 +13,7 @@ pub type Error = Box<dyn std::error::Error>;
|
|||||||
|
|
||||||
/// File names for testing purpose
|
/// File names for testing purpose
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
pub static FILES: &[&str] = &[
|
pub static FILES: &[&str] = &["test.txt", "test.html", "index.html", "😀.bin"];
|
||||||
"test.txt",
|
|
||||||
"test.html",
|
|
||||||
"index.html",
|
|
||||||
"test.mkv",
|
|
||||||
#[cfg(not(windows))]
|
|
||||||
"test \" \' & < >.csv",
|
|
||||||
"😀.data",
|
|
||||||
"⎙.mp4",
|
|
||||||
"#[]{}()@!$&'`+,;= %20.test",
|
|
||||||
#[cfg(unix)]
|
|
||||||
":?#[]{}<>()@!$&'`|*+,;= %20.test",
|
|
||||||
#[cfg(not(windows))]
|
|
||||||
"foo\\bar.test",
|
|
||||||
];
|
|
||||||
|
|
||||||
/// Directory names for testing diretory don't exist
|
/// Directory names for testing diretory don't exist
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
@@ -41,10 +27,6 @@ pub static DIR_NO_INDEX: &str = "dir-no-index/";
|
|||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
pub static DIRECTORIES: &[&str] = &["dira/", "dirb/", "dirc/", DIR_NO_INDEX];
|
pub static DIRECTORIES: &[&str] = &["dira/", "dirb/", "dirc/", DIR_NO_INDEX];
|
||||||
|
|
||||||
/// Name of a deeply nested file
|
|
||||||
#[allow(dead_code)]
|
|
||||||
pub static DEEPLY_NESTED_FILE: &str = "very/deeply/nested/test.rs";
|
|
||||||
|
|
||||||
/// Test fixture which creates a temporary directory with a few files and directories inside.
|
/// Test fixture which creates a temporary directory with a few files and directories inside.
|
||||||
/// The directories also contain files.
|
/// The directories also contain files.
|
||||||
#[fixture]
|
#[fixture]
|
||||||
@@ -69,10 +51,6 @@ pub fn tmpdir() -> TempDir {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
tmpdir
|
|
||||||
.child(&DEEPLY_NESTED_FILE)
|
|
||||||
.write_str("File in a deeply nested directory.")
|
|
||||||
.expect("Couldn't write to file");
|
|
||||||
tmpdir
|
tmpdir
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -96,7 +74,6 @@ where
|
|||||||
let tmpdir = tmpdir();
|
let tmpdir = tmpdir();
|
||||||
let child = Command::cargo_bin("dufs")
|
let child = Command::cargo_bin("dufs")
|
||||||
.expect("Couldn't find test binary")
|
.expect("Couldn't find test binary")
|
||||||
.env("RUST_LOG", "false")
|
|
||||||
.arg(tmpdir.path())
|
.arg(tmpdir.path())
|
||||||
.arg("-p")
|
.arg("-p")
|
||||||
.arg(port.to_string())
|
.arg(port.to_string())
|
||||||
@@ -124,7 +101,6 @@ where
|
|||||||
let tmpdir = tmpdir();
|
let tmpdir = tmpdir();
|
||||||
let child = Command::cargo_bin("dufs")
|
let child = Command::cargo_bin("dufs")
|
||||||
.expect("Couldn't find test binary")
|
.expect("Couldn't find test binary")
|
||||||
.env("RUST_LOG", "false")
|
|
||||||
.arg(tmpdir.path())
|
.arg(tmpdir.path())
|
||||||
.arg("-p")
|
.arg("-p")
|
||||||
.arg(port.to_string())
|
.arg(port.to_string())
|
||||||
|
|||||||
@@ -63,7 +63,7 @@ fn head_dir_zip(server: TestServer) -> Result<(), Error> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[rstest]
|
#[rstest]
|
||||||
fn get_dir_search(server: TestServer) -> Result<(), Error> {
|
fn get_dir_search(#[with(&["-A"])] server: TestServer) -> Result<(), Error> {
|
||||||
let resp = reqwest::blocking::get(format!("{}?q={}", server.url(), "test.html"))?;
|
let resp = reqwest::blocking::get(format!("{}?q={}", server.url(), "test.html"))?;
|
||||||
assert_eq!(resp.status(), 200);
|
assert_eq!(resp.status(), 200);
|
||||||
let paths = utils::retrive_index_paths(&resp.text()?);
|
let paths = utils::retrive_index_paths(&resp.text()?);
|
||||||
@@ -75,7 +75,19 @@ fn get_dir_search(server: TestServer) -> Result<(), Error> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[rstest]
|
#[rstest]
|
||||||
fn head_dir_search(server: TestServer) -> Result<(), Error> {
|
fn get_dir_search2(#[with(&["-A"])] server: TestServer) -> Result<(), Error> {
|
||||||
|
let resp = reqwest::blocking::get(format!("{}?q={}", server.url(), "😀.bin"))?;
|
||||||
|
assert_eq!(resp.status(), 200);
|
||||||
|
let paths = utils::retrive_index_paths(&resp.text()?);
|
||||||
|
assert!(!paths.is_empty());
|
||||||
|
for p in paths {
|
||||||
|
assert!(p.contains(&"😀.bin"));
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[rstest]
|
||||||
|
fn head_dir_search(#[with(&["-A"])] server: TestServer) -> Result<(), Error> {
|
||||||
let resp = fetch!(b"HEAD", format!("{}?q={}", server.url(), "test.html")).send()?;
|
let resp = fetch!(b"HEAD", format!("{}?q={}", server.url(), "test.html")).send()?;
|
||||||
assert_eq!(resp.status(), 200);
|
assert_eq!(resp.status(), 200);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
|
|||||||
@@ -37,12 +37,8 @@ pub fn encode_uri(v: &str) -> String {
|
|||||||
|
|
||||||
fn retrive_index_paths_impl(index: &str) -> Option<HashSet<String>> {
|
fn retrive_index_paths_impl(index: &str) -> Option<HashSet<String>> {
|
||||||
let lines: Vec<&str> = index.lines().collect();
|
let lines: Vec<&str> = index.lines().collect();
|
||||||
let (i, _) = lines
|
let line = lines.iter().find(|v| v.contains("DATA ="))?;
|
||||||
.iter()
|
let value: Value = line[7..].parse().ok()?;
|
||||||
.enumerate()
|
|
||||||
.find(|(_, v)| v.contains("const DATA"))?;
|
|
||||||
let line = lines.get(i + 1)?;
|
|
||||||
let value: Value = line.parse().ok()?;
|
|
||||||
let paths = value
|
let paths = value
|
||||||
.get("paths")?
|
.get("paths")?
|
||||||
.as_array()?
|
.as_array()?
|
||||||
|
|||||||
Reference in New Issue
Block a user