Compare commits
17 Commits
2c101abdbd
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
f44cfc6190
|
|||
|
126a86ec07
|
|||
|
227f78a767
|
|||
|
1ea1e65deb
|
|||
|
90707cc364
|
|||
|
6febf8ee22
|
|||
|
d79e96c498
|
|||
|
99705afa9e
|
|||
|
3f53c221c8
|
|||
|
725d33939e
|
|||
|
81ac1475ad
|
|||
|
d7b0127d20
|
|||
|
67b412a1a5
|
|||
|
6898297973
|
|||
|
45571c98fe
|
|||
|
acd2712ade
|
|||
|
ea2e43fe3d
|
Generated
+129
-21
@@ -76,15 +76,27 @@ checksum = "3c3610892ee6e0cbce8ae2700349fcf8f98adb0dbfbee85aec3c9179d29cc072"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"base64ct",
|
"base64ct",
|
||||||
"blake2",
|
"blake2",
|
||||||
"cpufeatures",
|
"cpufeatures 0.2.17",
|
||||||
"password-hash",
|
"password-hash",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "assert_cmd"
|
name = "arrayref"
|
||||||
version = "2.2.1"
|
version = "0.3.9"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "39bae1d3fa576f7c6519514180a72559268dd7d1fe104070956cb687bc6673bd"
|
checksum = "76a2e8124351fda1ef8aaaa3bbd7ebbcb486bbcd4225aca0aa0d84bb2db8fecb"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "arrayvec"
|
||||||
|
version = "0.7.6"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "assert_cmd"
|
||||||
|
version = "2.2.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "2aa3a22042e45de04255c7bf3626e239f450200fd0493c1e382263544b20aea6"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anstyle",
|
"anstyle",
|
||||||
"bstr",
|
"bstr",
|
||||||
@@ -103,9 +115,9 @@ checksum = "2af50177e190e07a26ab74f8b1efbfe2ef87da2116221318cb1c2e82baf7de06"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "bitflags"
|
name = "bitflags"
|
||||||
version = "2.11.1"
|
version = "2.13.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c4512299f36f043ab09a583e57bceb5a5aab7a73db1805848e8fef3c9e8c78b3"
|
checksum = "b4388bee8683e3d04af747c73422af53102d2bd24d9eadb6cbc100baef4b43f8"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "blake2"
|
name = "blake2"
|
||||||
@@ -116,6 +128,20 @@ dependencies = [
|
|||||||
"digest",
|
"digest",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "blake3"
|
||||||
|
version = "1.8.5"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "0aa83c34e62843d924f905e0f5c866eb1dd6545fc4d719e803d9ba6030371fce"
|
||||||
|
dependencies = [
|
||||||
|
"arrayref",
|
||||||
|
"arrayvec",
|
||||||
|
"cc",
|
||||||
|
"cfg-if",
|
||||||
|
"constant_time_eq",
|
||||||
|
"cpufeatures 0.3.0",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "block-buffer"
|
name = "block-buffer"
|
||||||
version = "0.10.4"
|
version = "0.10.4"
|
||||||
@@ -136,6 +162,16 @@ dependencies = [
|
|||||||
"serde",
|
"serde",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "cc"
|
||||||
|
version = "1.2.63"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "556e016178bb5662a08681bbe0f00f8e17631781a4dfc8c45e466e4b185ec27f"
|
||||||
|
dependencies = [
|
||||||
|
"find-msvc-tools",
|
||||||
|
"shlex",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "cfg-if"
|
name = "cfg-if"
|
||||||
version = "1.0.4"
|
version = "1.0.4"
|
||||||
@@ -150,7 +186,7 @@ checksum = "c3613f74bd2eac03dad61bd53dbe620703d4371614fe0bc3b9f04dd36fe4e818"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
"cipher",
|
"cipher",
|
||||||
"cpufeatures",
|
"cpufeatures 0.2.17",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -223,6 +259,12 @@ version = "1.0.5"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "1d07550c9036bf2ae0c684c4297d503f838287c83c53686d05370d0e139ae570"
|
checksum = "1d07550c9036bf2ae0c684c4297d503f838287c83c53686d05370d0e139ae570"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "constant_time_eq"
|
||||||
|
version = "0.4.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "3d52eff69cd5e647efe296129160853a42795992097e8af39800e1060caeea9b"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "cpufeatures"
|
name = "cpufeatures"
|
||||||
version = "0.2.17"
|
version = "0.2.17"
|
||||||
@@ -232,6 +274,15 @@ dependencies = [
|
|||||||
"libc",
|
"libc",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "cpufeatures"
|
||||||
|
version = "0.3.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "8b2a41393f66f16b0823bb79094d54ac5fbd34ab292ddafb9a0456ac9f87d201"
|
||||||
|
dependencies = [
|
||||||
|
"libc",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "crossbeam-channel"
|
name = "crossbeam-channel"
|
||||||
version = "0.5.15"
|
version = "0.5.15"
|
||||||
@@ -299,22 +350,31 @@ checksum = "9f1f227452a390804cdb637b74a86990f2a7d7ba4b7d5693aac9b4dd6defd8d6"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "fcry"
|
name = "fcry"
|
||||||
version = "0.10.0"
|
version = "0.12.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"argon2",
|
"argon2",
|
||||||
"assert_cmd",
|
"assert_cmd",
|
||||||
|
"blake3",
|
||||||
"chacha20poly1305",
|
"chacha20poly1305",
|
||||||
"clap",
|
"clap",
|
||||||
"crossbeam-channel",
|
"crossbeam-channel",
|
||||||
"getrandom 0.4.2",
|
"getrandom 0.4.2",
|
||||||
"libc",
|
"libc",
|
||||||
"rlimit",
|
"rlimit",
|
||||||
|
"same-file",
|
||||||
"secrets",
|
"secrets",
|
||||||
"tempfile",
|
"tempfile",
|
||||||
|
"unicode-normalization",
|
||||||
"windows-sys",
|
"windows-sys",
|
||||||
"zeroize",
|
"zeroize",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "find-msvc-tools"
|
||||||
|
version = "0.1.9"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "5baebc0774151f905a1a2cc41989300b1e6fbb29aff0ceffa1064fdd3088d582"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "foldhash"
|
name = "foldhash"
|
||||||
version = "0.1.5"
|
version = "0.1.5"
|
||||||
@@ -366,9 +426,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "hashbrown"
|
name = "hashbrown"
|
||||||
version = "0.17.0"
|
version = "0.17.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "4f467dd6dccf739c208452f8014c75c18bb8301b050ad1cfb27153803edb0f51"
|
checksum = "ed5909b6e89a2db4456e54cd5f673791d7eca6732202bbf2a9cc504fe2f9b84a"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "heck"
|
name = "heck"
|
||||||
@@ -389,7 +449,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "d466e9454f08e4a911e14806c24e16fba1b4c121d1ea474396f396069cf949d9"
|
checksum = "d466e9454f08e4a911e14806c24e16fba1b4c121d1ea474396f396069cf949d9"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"equivalent",
|
"equivalent",
|
||||||
"hashbrown 0.17.0",
|
"hashbrown 0.17.1",
|
||||||
"serde",
|
"serde",
|
||||||
"serde_core",
|
"serde_core",
|
||||||
]
|
]
|
||||||
@@ -435,15 +495,15 @@ checksum = "32a66949e030da00e8c7d4434b251670a91556f4144941d37452769c25d58a53"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "log"
|
name = "log"
|
||||||
version = "0.4.29"
|
version = "0.4.32"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "5e5032e24019045c762d3c0f28f5b6b8bbf38563a65908389bf7978758920897"
|
checksum = "953f07c43838f8e6f9758cab68bf5bed85465e7587ebe0b823f1bcd81978ad3a"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "memchr"
|
name = "memchr"
|
||||||
version = "2.8.0"
|
version = "2.8.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "f8ca58f447f06ed17d5fc4043ce1b10dd205e060fb3ce5b979b8ed8e59ff3f79"
|
checksum = "6b947ae49db0d222b1dbc6b113ce7248a3fc3a6ca21b696717bfc000ba4484d8"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "once_cell"
|
name = "once_cell"
|
||||||
@@ -496,7 +556,7 @@ version = "0.8.0"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "8159bd90725d2df49889a078b54f4f79e87f1f8a8444194cdca81d38f5393abf"
|
checksum = "8159bd90725d2df49889a078b54f4f79e87f1f8a8444194cdca81d38f5393abf"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cpufeatures",
|
"cpufeatures 0.2.17",
|
||||||
"opaque-debug",
|
"opaque-debug",
|
||||||
"universal-hash",
|
"universal-hash",
|
||||||
]
|
]
|
||||||
@@ -599,6 +659,15 @@ dependencies = [
|
|||||||
"windows-sys",
|
"windows-sys",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "same-file"
|
||||||
|
version = "1.0.6"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "93fc1dc3aaa9bfed95e02e6eadabb4baf7e3078b0bd1b4d7b6b0b68378900502"
|
||||||
|
dependencies = [
|
||||||
|
"winapi-util",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "secrets"
|
name = "secrets"
|
||||||
version = "1.3.0"
|
version = "1.3.0"
|
||||||
@@ -648,9 +717,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "serde_json"
|
name = "serde_json"
|
||||||
version = "1.0.149"
|
version = "1.0.150"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "83fc039473c5595ace860d8c4fafa220ff474b3fc6bfdb4293327f1a37e94d86"
|
checksum = "e8014e44b4736ed0538adeecded0fce2a272f22dc9578a7eb6b2d9993c74cfb9"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"itoa",
|
"itoa",
|
||||||
"memchr",
|
"memchr",
|
||||||
@@ -659,6 +728,12 @@ dependencies = [
|
|||||||
"zmij",
|
"zmij",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "shlex"
|
||||||
|
version = "2.0.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f8fadd59c855ef2080decdef8ff161eb6661b86933c9d82e5ba29dc602a55aba"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "strsim"
|
name = "strsim"
|
||||||
version = "0.11.1"
|
version = "0.11.1"
|
||||||
@@ -702,10 +777,25 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "8f50febec83f5ee1df3015341d8bd429f2d1cc62bcba7ea2076759d315084683"
|
checksum = "8f50febec83f5ee1df3015341d8bd429f2d1cc62bcba7ea2076759d315084683"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "typenum"
|
name = "tinyvec"
|
||||||
version = "1.20.0"
|
version = "1.11.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "40ce102ab67701b8526c123c1bab5cbe42d7040ccfd0f64af1a385808d2f43de"
|
checksum = "3e61e67053d25a4e82c844e8424039d9745781b3fc4f32b8d55ed50f5f667ef3"
|
||||||
|
dependencies = [
|
||||||
|
"tinyvec_macros",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "tinyvec_macros"
|
||||||
|
version = "0.1.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "typenum"
|
||||||
|
version = "1.20.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "b6f5e870be6c3b371b77fe0ee0bafb859fa4964b4404c27de1d380043c4dda20"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "unicode-ident"
|
name = "unicode-ident"
|
||||||
@@ -713,6 +803,15 @@ version = "1.0.24"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "e6e4313cd5fcd3dad5cafa179702e2b244f760991f45397d14d4ebf38247da75"
|
checksum = "e6e4313cd5fcd3dad5cafa179702e2b244f760991f45397d14d4ebf38247da75"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "unicode-normalization"
|
||||||
|
version = "0.1.25"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "5fd4f6878c9cb28d874b009da9e8d183b5abc80117c40bbd187a1fde336be6e8"
|
||||||
|
dependencies = [
|
||||||
|
"tinyvec",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "unicode-xid"
|
name = "unicode-xid"
|
||||||
version = "0.2.6"
|
version = "0.2.6"
|
||||||
@@ -830,6 +929,15 @@ version = "0.4.0"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
|
checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "winapi-util"
|
||||||
|
version = "0.1.11"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "c2a7b1c03c876122aa43f3020e6c3c3ee5c05081c9a00739faf7503aeba10d22"
|
||||||
|
dependencies = [
|
||||||
|
"windows-sys",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "winapi-x86_64-pc-windows-gnu"
|
name = "winapi-x86_64-pc-windows-gnu"
|
||||||
version = "0.4.0"
|
version = "0.4.0"
|
||||||
|
|||||||
+34
-16
@@ -1,37 +1,55 @@
|
|||||||
[package]
|
[package]
|
||||||
authors = ["ddidderr <ddidderr@paul.network>"]
|
|
||||||
edition = "2024"
|
|
||||||
name = "fcry"
|
name = "fcry"
|
||||||
version = "0.10.0"
|
version = "0.12.0"
|
||||||
|
edition = "2024"
|
||||||
|
license = "MIT-0"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
argon2 = "0.5"
|
argon2 = "0.5"
|
||||||
|
blake3 = "1"
|
||||||
chacha20poly1305 = "0.10"
|
chacha20poly1305 = "0.10"
|
||||||
clap = {version = "4", features = ["derive"]}
|
clap = { version = "4", features = ["derive"] }
|
||||||
crossbeam-channel = "0.5"
|
crossbeam-channel = "0.5"
|
||||||
getrandom = {version = "0.4"}
|
getrandom = { version = "0.4" }
|
||||||
protected-secrets = {package = "secrets", version = "1.3"}
|
protected-secrets = { package = "secrets", version = "1.3" }
|
||||||
zeroize = {version = "1", features = ["derive"]}
|
same-file = "1"
|
||||||
|
unicode-normalization = "0.1"
|
||||||
|
zeroize = { version = "1", features = ["derive"] }
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
assert_cmd = "2"
|
||||||
|
tempfile = "3"
|
||||||
|
|
||||||
[target.'cfg(unix)'.dependencies]
|
[target.'cfg(unix)'.dependencies]
|
||||||
libc = "0.2"
|
libc = "0.2"
|
||||||
rlimit = "0.11"
|
rlimit = "0.11"
|
||||||
|
|
||||||
[target.'cfg(windows)'.dependencies]
|
[target.'cfg(windows)'.dependencies]
|
||||||
windows-sys = {version = "0.61", features = [
|
windows-sys = {
|
||||||
"Win32_System_Console",
|
version = "0.61",
|
||||||
|
features = [
|
||||||
"Win32_Foundation",
|
"Win32_Foundation",
|
||||||
"Win32_Storage_FileSystem",
|
|
||||||
"Win32_Security",
|
"Win32_Security",
|
||||||
]}
|
"Win32_Storage_FileSystem",
|
||||||
|
"Win32_System_Console",
|
||||||
[dev-dependencies]
|
]
|
||||||
assert_cmd = "2"
|
}
|
||||||
tempfile = "3"
|
|
||||||
|
|
||||||
[profile.release]
|
[profile.release]
|
||||||
|
debug = true
|
||||||
|
strip = false
|
||||||
|
debug-assertions = true
|
||||||
|
overflow-checks = true
|
||||||
lto = false
|
lto = false
|
||||||
|
panic = "unwind"
|
||||||
|
incremental = true
|
||||||
|
|
||||||
|
[profile.production]
|
||||||
|
inherits = "release"
|
||||||
debug = false
|
debug = false
|
||||||
strip = true
|
strip = true
|
||||||
panic = "unwind"
|
debug-assertions = false
|
||||||
|
overflow-checks = false
|
||||||
|
lto = true
|
||||||
|
incremental = false
|
||||||
codegen-units = 1
|
codegen-units = 1
|
||||||
|
|||||||
@@ -0,0 +1,16 @@
|
|||||||
|
MIT No Attribution
|
||||||
|
|
||||||
|
Copyright 2026 fcry contributors
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy of this
|
||||||
|
software and associated documentation files (the "Software"), to deal in the Software
|
||||||
|
without restriction, including without limitation the rights to use, copy, modify,
|
||||||
|
merge, publish, distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
|
||||||
|
INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
||||||
|
HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
||||||
|
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
-675
@@ -1,675 +0,0 @@
|
|||||||
### GNU GENERAL PUBLIC LICENSE
|
|
||||||
|
|
||||||
Version 3, 29 June 2007
|
|
||||||
|
|
||||||
Copyright (C) 2007 Free Software Foundation, Inc.
|
|
||||||
<https://fsf.org/>
|
|
||||||
|
|
||||||
Everyone is permitted to copy and distribute verbatim copies of this
|
|
||||||
license document, but changing it is not allowed.
|
|
||||||
|
|
||||||
### Preamble
|
|
||||||
|
|
||||||
The GNU General Public License is a free, copyleft license for
|
|
||||||
software and other kinds of works.
|
|
||||||
|
|
||||||
The licenses for most software and other practical works are designed
|
|
||||||
to take away your freedom to share and change the works. By contrast,
|
|
||||||
the GNU General Public License is intended to guarantee your freedom
|
|
||||||
to share and change all versions of a program--to make sure it remains
|
|
||||||
free software for all its users. We, the Free Software Foundation, use
|
|
||||||
the GNU General Public License for most of our software; it applies
|
|
||||||
also to any other work released this way by its authors. You can apply
|
|
||||||
it to your programs, too.
|
|
||||||
|
|
||||||
When we speak of free software, we are referring to freedom, not
|
|
||||||
price. Our General Public Licenses are designed to make sure that you
|
|
||||||
have the freedom to distribute copies of free software (and charge for
|
|
||||||
them if you wish), that you receive source code or can get it if you
|
|
||||||
want it, that you can change the software or use pieces of it in new
|
|
||||||
free programs, and that you know you can do these things.
|
|
||||||
|
|
||||||
To protect your rights, we need to prevent others from denying you
|
|
||||||
these rights or asking you to surrender the rights. Therefore, you
|
|
||||||
have certain responsibilities if you distribute copies of the
|
|
||||||
software, or if you modify it: responsibilities to respect the freedom
|
|
||||||
of others.
|
|
||||||
|
|
||||||
For example, if you distribute copies of such a program, whether
|
|
||||||
gratis or for a fee, you must pass on to the recipients the same
|
|
||||||
freedoms that you received. You must make sure that they, too, receive
|
|
||||||
or can get the source code. And you must show them these terms so they
|
|
||||||
know their rights.
|
|
||||||
|
|
||||||
Developers that use the GNU GPL protect your rights with two steps:
|
|
||||||
(1) assert copyright on the software, and (2) offer you this License
|
|
||||||
giving you legal permission to copy, distribute and/or modify it.
|
|
||||||
|
|
||||||
For the developers' and authors' protection, the GPL clearly explains
|
|
||||||
that there is no warranty for this free software. For both users' and
|
|
||||||
authors' sake, the GPL requires that modified versions be marked as
|
|
||||||
changed, so that their problems will not be attributed erroneously to
|
|
||||||
authors of previous versions.
|
|
||||||
|
|
||||||
Some devices are designed to deny users access to install or run
|
|
||||||
modified versions of the software inside them, although the
|
|
||||||
manufacturer can do so. This is fundamentally incompatible with the
|
|
||||||
aim of protecting users' freedom to change the software. The
|
|
||||||
systematic pattern of such abuse occurs in the area of products for
|
|
||||||
individuals to use, which is precisely where it is most unacceptable.
|
|
||||||
Therefore, we have designed this version of the GPL to prohibit the
|
|
||||||
practice for those products. If such problems arise substantially in
|
|
||||||
other domains, we stand ready to extend this provision to those
|
|
||||||
domains in future versions of the GPL, as needed to protect the
|
|
||||||
freedom of users.
|
|
||||||
|
|
||||||
Finally, every program is threatened constantly by software patents.
|
|
||||||
States should not allow patents to restrict development and use of
|
|
||||||
software on general-purpose computers, but in those that do, we wish
|
|
||||||
to avoid the special danger that patents applied to a free program
|
|
||||||
could make it effectively proprietary. To prevent this, the GPL
|
|
||||||
assures that patents cannot be used to render the program non-free.
|
|
||||||
|
|
||||||
The precise terms and conditions for copying, distribution and
|
|
||||||
modification follow.
|
|
||||||
|
|
||||||
### TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
#### 0. Definitions.
|
|
||||||
|
|
||||||
"This License" refers to version 3 of the GNU General Public License.
|
|
||||||
|
|
||||||
"Copyright" also means copyright-like laws that apply to other kinds
|
|
||||||
of works, such as semiconductor masks.
|
|
||||||
|
|
||||||
"The Program" refers to any copyrightable work licensed under this
|
|
||||||
License. Each licensee is addressed as "you". "Licensees" and
|
|
||||||
"recipients" may be individuals or organizations.
|
|
||||||
|
|
||||||
To "modify" a work means to copy from or adapt all or part of the work
|
|
||||||
in a fashion requiring copyright permission, other than the making of
|
|
||||||
an exact copy. The resulting work is called a "modified version" of
|
|
||||||
the earlier work or a work "based on" the earlier work.
|
|
||||||
|
|
||||||
A "covered work" means either the unmodified Program or a work based
|
|
||||||
on the Program.
|
|
||||||
|
|
||||||
To "propagate" a work means to do anything with it that, without
|
|
||||||
permission, would make you directly or secondarily liable for
|
|
||||||
infringement under applicable copyright law, except executing it on a
|
|
||||||
computer or modifying a private copy. Propagation includes copying,
|
|
||||||
distribution (with or without modification), making available to the
|
|
||||||
public, and in some countries other activities as well.
|
|
||||||
|
|
||||||
To "convey" a work means any kind of propagation that enables other
|
|
||||||
parties to make or receive copies. Mere interaction with a user
|
|
||||||
through a computer network, with no transfer of a copy, is not
|
|
||||||
conveying.
|
|
||||||
|
|
||||||
An interactive user interface displays "Appropriate Legal Notices" to
|
|
||||||
the extent that it includes a convenient and prominently visible
|
|
||||||
feature that (1) displays an appropriate copyright notice, and (2)
|
|
||||||
tells the user that there is no warranty for the work (except to the
|
|
||||||
extent that warranties are provided), that licensees may convey the
|
|
||||||
work under this License, and how to view a copy of this License. If
|
|
||||||
the interface presents a list of user commands or options, such as a
|
|
||||||
menu, a prominent item in the list meets this criterion.
|
|
||||||
|
|
||||||
#### 1. Source Code.
|
|
||||||
|
|
||||||
The "source code" for a work means the preferred form of the work for
|
|
||||||
making modifications to it. "Object code" means any non-source form of
|
|
||||||
a work.
|
|
||||||
|
|
||||||
A "Standard Interface" means an interface that either is an official
|
|
||||||
standard defined by a recognized standards body, or, in the case of
|
|
||||||
interfaces specified for a particular programming language, one that
|
|
||||||
is widely used among developers working in that language.
|
|
||||||
|
|
||||||
The "System Libraries" of an executable work include anything, other
|
|
||||||
than the work as a whole, that (a) is included in the normal form of
|
|
||||||
packaging a Major Component, but which is not part of that Major
|
|
||||||
Component, and (b) serves only to enable use of the work with that
|
|
||||||
Major Component, or to implement a Standard Interface for which an
|
|
||||||
implementation is available to the public in source code form. A
|
|
||||||
"Major Component", in this context, means a major essential component
|
|
||||||
(kernel, window system, and so on) of the specific operating system
|
|
||||||
(if any) on which the executable work runs, or a compiler used to
|
|
||||||
produce the work, or an object code interpreter used to run it.
|
|
||||||
|
|
||||||
The "Corresponding Source" for a work in object code form means all
|
|
||||||
the source code needed to generate, install, and (for an executable
|
|
||||||
work) run the object code and to modify the work, including scripts to
|
|
||||||
control those activities. However, it does not include the work's
|
|
||||||
System Libraries, or general-purpose tools or generally available free
|
|
||||||
programs which are used unmodified in performing those activities but
|
|
||||||
which are not part of the work. For example, Corresponding Source
|
|
||||||
includes interface definition files associated with source files for
|
|
||||||
the work, and the source code for shared libraries and dynamically
|
|
||||||
linked subprograms that the work is specifically designed to require,
|
|
||||||
such as by intimate data communication or control flow between those
|
|
||||||
subprograms and other parts of the work.
|
|
||||||
|
|
||||||
The Corresponding Source need not include anything that users can
|
|
||||||
regenerate automatically from other parts of the Corresponding Source.
|
|
||||||
|
|
||||||
The Corresponding Source for a work in source code form is that same
|
|
||||||
work.
|
|
||||||
|
|
||||||
#### 2. Basic Permissions.
|
|
||||||
|
|
||||||
All rights granted under this License are granted for the term of
|
|
||||||
copyright on the Program, and are irrevocable provided the stated
|
|
||||||
conditions are met. This License explicitly affirms your unlimited
|
|
||||||
permission to run the unmodified Program. The output from running a
|
|
||||||
covered work is covered by this License only if the output, given its
|
|
||||||
content, constitutes a covered work. This License acknowledges your
|
|
||||||
rights of fair use or other equivalent, as provided by copyright law.
|
|
||||||
|
|
||||||
You may make, run and propagate covered works that you do not convey,
|
|
||||||
without conditions so long as your license otherwise remains in force.
|
|
||||||
You may convey covered works to others for the sole purpose of having
|
|
||||||
them make modifications exclusively for you, or provide you with
|
|
||||||
facilities for running those works, provided that you comply with the
|
|
||||||
terms of this License in conveying all material for which you do not
|
|
||||||
control copyright. Those thus making or running the covered works for
|
|
||||||
you must do so exclusively on your behalf, under your direction and
|
|
||||||
control, on terms that prohibit them from making any copies of your
|
|
||||||
copyrighted material outside their relationship with you.
|
|
||||||
|
|
||||||
Conveying under any other circumstances is permitted solely under the
|
|
||||||
conditions stated below. Sublicensing is not allowed; section 10 makes
|
|
||||||
it unnecessary.
|
|
||||||
|
|
||||||
#### 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
|
||||||
|
|
||||||
No covered work shall be deemed part of an effective technological
|
|
||||||
measure under any applicable law fulfilling obligations under article
|
|
||||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
|
||||||
similar laws prohibiting or restricting circumvention of such
|
|
||||||
measures.
|
|
||||||
|
|
||||||
When you convey a covered work, you waive any legal power to forbid
|
|
||||||
circumvention of technological measures to the extent such
|
|
||||||
circumvention is effected by exercising rights under this License with
|
|
||||||
respect to the covered work, and you disclaim any intention to limit
|
|
||||||
operation or modification of the work as a means of enforcing, against
|
|
||||||
the work's users, your or third parties' legal rights to forbid
|
|
||||||
circumvention of technological measures.
|
|
||||||
|
|
||||||
#### 4. Conveying Verbatim Copies.
|
|
||||||
|
|
||||||
You may convey verbatim copies of the Program's source code as you
|
|
||||||
receive it, in any medium, provided that you conspicuously and
|
|
||||||
appropriately publish on each copy an appropriate copyright notice;
|
|
||||||
keep intact all notices stating that this License and any
|
|
||||||
non-permissive terms added in accord with section 7 apply to the code;
|
|
||||||
keep intact all notices of the absence of any warranty; and give all
|
|
||||||
recipients a copy of this License along with the Program.
|
|
||||||
|
|
||||||
You may charge any price or no price for each copy that you convey,
|
|
||||||
and you may offer support or warranty protection for a fee.
|
|
||||||
|
|
||||||
#### 5. Conveying Modified Source Versions.
|
|
||||||
|
|
||||||
You may convey a work based on the Program, or the modifications to
|
|
||||||
produce it from the Program, in the form of source code under the
|
|
||||||
terms of section 4, provided that you also meet all of these
|
|
||||||
conditions:
|
|
||||||
|
|
||||||
- a) The work must carry prominent notices stating that you modified
|
|
||||||
it, and giving a relevant date.
|
|
||||||
- b) The work must carry prominent notices stating that it is
|
|
||||||
released under this License and any conditions added under
|
|
||||||
section 7. This requirement modifies the requirement in section 4
|
|
||||||
to "keep intact all notices".
|
|
||||||
- c) You must license the entire work, as a whole, under this
|
|
||||||
License to anyone who comes into possession of a copy. This
|
|
||||||
License will therefore apply, along with any applicable section 7
|
|
||||||
additional terms, to the whole of the work, and all its parts,
|
|
||||||
regardless of how they are packaged. This License gives no
|
|
||||||
permission to license the work in any other way, but it does not
|
|
||||||
invalidate such permission if you have separately received it.
|
|
||||||
- d) If the work has interactive user interfaces, each must display
|
|
||||||
Appropriate Legal Notices; however, if the Program has interactive
|
|
||||||
interfaces that do not display Appropriate Legal Notices, your
|
|
||||||
work need not make them do so.
|
|
||||||
|
|
||||||
A compilation of a covered work with other separate and independent
|
|
||||||
works, which are not by their nature extensions of the covered work,
|
|
||||||
and which are not combined with it such as to form a larger program,
|
|
||||||
in or on a volume of a storage or distribution medium, is called an
|
|
||||||
"aggregate" if the compilation and its resulting copyright are not
|
|
||||||
used to limit the access or legal rights of the compilation's users
|
|
||||||
beyond what the individual works permit. Inclusion of a covered work
|
|
||||||
in an aggregate does not cause this License to apply to the other
|
|
||||||
parts of the aggregate.
|
|
||||||
|
|
||||||
#### 6. Conveying Non-Source Forms.
|
|
||||||
|
|
||||||
You may convey a covered work in object code form under the terms of
|
|
||||||
sections 4 and 5, provided that you also convey the machine-readable
|
|
||||||
Corresponding Source under the terms of this License, in one of these
|
|
||||||
ways:
|
|
||||||
|
|
||||||
- a) Convey the object code in, or embodied in, a physical product
|
|
||||||
(including a physical distribution medium), accompanied by the
|
|
||||||
Corresponding Source fixed on a durable physical medium
|
|
||||||
customarily used for software interchange.
|
|
||||||
- b) Convey the object code in, or embodied in, a physical product
|
|
||||||
(including a physical distribution medium), accompanied by a
|
|
||||||
written offer, valid for at least three years and valid for as
|
|
||||||
long as you offer spare parts or customer support for that product
|
|
||||||
model, to give anyone who possesses the object code either (1) a
|
|
||||||
copy of the Corresponding Source for all the software in the
|
|
||||||
product that is covered by this License, on a durable physical
|
|
||||||
medium customarily used for software interchange, for a price no
|
|
||||||
more than your reasonable cost of physically performing this
|
|
||||||
conveying of source, or (2) access to copy the Corresponding
|
|
||||||
Source from a network server at no charge.
|
|
||||||
- c) Convey individual copies of the object code with a copy of the
|
|
||||||
written offer to provide the Corresponding Source. This
|
|
||||||
alternative is allowed only occasionally and noncommercially, and
|
|
||||||
only if you received the object code with such an offer, in accord
|
|
||||||
with subsection 6b.
|
|
||||||
- d) Convey the object code by offering access from a designated
|
|
||||||
place (gratis or for a charge), and offer equivalent access to the
|
|
||||||
Corresponding Source in the same way through the same place at no
|
|
||||||
further charge. You need not require recipients to copy the
|
|
||||||
Corresponding Source along with the object code. If the place to
|
|
||||||
copy the object code is a network server, the Corresponding Source
|
|
||||||
may be on a different server (operated by you or a third party)
|
|
||||||
that supports equivalent copying facilities, provided you maintain
|
|
||||||
clear directions next to the object code saying where to find the
|
|
||||||
Corresponding Source. Regardless of what server hosts the
|
|
||||||
Corresponding Source, you remain obligated to ensure that it is
|
|
||||||
available for as long as needed to satisfy these requirements.
|
|
||||||
- e) Convey the object code using peer-to-peer transmission,
|
|
||||||
provided you inform other peers where the object code and
|
|
||||||
Corresponding Source of the work are being offered to the general
|
|
||||||
public at no charge under subsection 6d.
|
|
||||||
|
|
||||||
A separable portion of the object code, whose source code is excluded
|
|
||||||
from the Corresponding Source as a System Library, need not be
|
|
||||||
included in conveying the object code work.
|
|
||||||
|
|
||||||
A "User Product" is either (1) a "consumer product", which means any
|
|
||||||
tangible personal property which is normally used for personal,
|
|
||||||
family, or household purposes, or (2) anything designed or sold for
|
|
||||||
incorporation into a dwelling. In determining whether a product is a
|
|
||||||
consumer product, doubtful cases shall be resolved in favor of
|
|
||||||
coverage. For a particular product received by a particular user,
|
|
||||||
"normally used" refers to a typical or common use of that class of
|
|
||||||
product, regardless of the status of the particular user or of the way
|
|
||||||
in which the particular user actually uses, or expects or is expected
|
|
||||||
to use, the product. A product is a consumer product regardless of
|
|
||||||
whether the product has substantial commercial, industrial or
|
|
||||||
non-consumer uses, unless such uses represent the only significant
|
|
||||||
mode of use of the product.
|
|
||||||
|
|
||||||
"Installation Information" for a User Product means any methods,
|
|
||||||
procedures, authorization keys, or other information required to
|
|
||||||
install and execute modified versions of a covered work in that User
|
|
||||||
Product from a modified version of its Corresponding Source. The
|
|
||||||
information must suffice to ensure that the continued functioning of
|
|
||||||
the modified object code is in no case prevented or interfered with
|
|
||||||
solely because modification has been made.
|
|
||||||
|
|
||||||
If you convey an object code work under this section in, or with, or
|
|
||||||
specifically for use in, a User Product, and the conveying occurs as
|
|
||||||
part of a transaction in which the right of possession and use of the
|
|
||||||
User Product is transferred to the recipient in perpetuity or for a
|
|
||||||
fixed term (regardless of how the transaction is characterized), the
|
|
||||||
Corresponding Source conveyed under this section must be accompanied
|
|
||||||
by the Installation Information. But this requirement does not apply
|
|
||||||
if neither you nor any third party retains the ability to install
|
|
||||||
modified object code on the User Product (for example, the work has
|
|
||||||
been installed in ROM).
|
|
||||||
|
|
||||||
The requirement to provide Installation Information does not include a
|
|
||||||
requirement to continue to provide support service, warranty, or
|
|
||||||
updates for a work that has been modified or installed by the
|
|
||||||
recipient, or for the User Product in which it has been modified or
|
|
||||||
installed. Access to a network may be denied when the modification
|
|
||||||
itself materially and adversely affects the operation of the network
|
|
||||||
or violates the rules and protocols for communication across the
|
|
||||||
network.
|
|
||||||
|
|
||||||
Corresponding Source conveyed, and Installation Information provided,
|
|
||||||
in accord with this section must be in a format that is publicly
|
|
||||||
documented (and with an implementation available to the public in
|
|
||||||
source code form), and must require no special password or key for
|
|
||||||
unpacking, reading or copying.
|
|
||||||
|
|
||||||
#### 7. Additional Terms.
|
|
||||||
|
|
||||||
"Additional permissions" are terms that supplement the terms of this
|
|
||||||
License by making exceptions from one or more of its conditions.
|
|
||||||
Additional permissions that are applicable to the entire Program shall
|
|
||||||
be treated as though they were included in this License, to the extent
|
|
||||||
that they are valid under applicable law. If additional permissions
|
|
||||||
apply only to part of the Program, that part may be used separately
|
|
||||||
under those permissions, but the entire Program remains governed by
|
|
||||||
this License without regard to the additional permissions.
|
|
||||||
|
|
||||||
When you convey a copy of a covered work, you may at your option
|
|
||||||
remove any additional permissions from that copy, or from any part of
|
|
||||||
it. (Additional permissions may be written to require their own
|
|
||||||
removal in certain cases when you modify the work.) You may place
|
|
||||||
additional permissions on material, added by you to a covered work,
|
|
||||||
for which you have or can give appropriate copyright permission.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, for material you
|
|
||||||
add to a covered work, you may (if authorized by the copyright holders
|
|
||||||
of that material) supplement the terms of this License with terms:
|
|
||||||
|
|
||||||
- a) Disclaiming warranty or limiting liability differently from the
|
|
||||||
terms of sections 15 and 16 of this License; or
|
|
||||||
- b) Requiring preservation of specified reasonable legal notices or
|
|
||||||
author attributions in that material or in the Appropriate Legal
|
|
||||||
Notices displayed by works containing it; or
|
|
||||||
- c) Prohibiting misrepresentation of the origin of that material,
|
|
||||||
or requiring that modified versions of such material be marked in
|
|
||||||
reasonable ways as different from the original version; or
|
|
||||||
- d) Limiting the use for publicity purposes of names of licensors
|
|
||||||
or authors of the material; or
|
|
||||||
- e) Declining to grant rights under trademark law for use of some
|
|
||||||
trade names, trademarks, or service marks; or
|
|
||||||
- f) Requiring indemnification of licensors and authors of that
|
|
||||||
material by anyone who conveys the material (or modified versions
|
|
||||||
of it) with contractual assumptions of liability to the recipient,
|
|
||||||
for any liability that these contractual assumptions directly
|
|
||||||
impose on those licensors and authors.
|
|
||||||
|
|
||||||
All other non-permissive additional terms are considered "further
|
|
||||||
restrictions" within the meaning of section 10. If the Program as you
|
|
||||||
received it, or any part of it, contains a notice stating that it is
|
|
||||||
governed by this License along with a term that is a further
|
|
||||||
restriction, you may remove that term. If a license document contains
|
|
||||||
a further restriction but permits relicensing or conveying under this
|
|
||||||
License, you may add to a covered work material governed by the terms
|
|
||||||
of that license document, provided that the further restriction does
|
|
||||||
not survive such relicensing or conveying.
|
|
||||||
|
|
||||||
If you add terms to a covered work in accord with this section, you
|
|
||||||
must place, in the relevant source files, a statement of the
|
|
||||||
additional terms that apply to those files, or a notice indicating
|
|
||||||
where to find the applicable terms.
|
|
||||||
|
|
||||||
Additional terms, permissive or non-permissive, may be stated in the
|
|
||||||
form of a separately written license, or stated as exceptions; the
|
|
||||||
above requirements apply either way.
|
|
||||||
|
|
||||||
#### 8. Termination.
|
|
||||||
|
|
||||||
You may not propagate or modify a covered work except as expressly
|
|
||||||
provided under this License. Any attempt otherwise to propagate or
|
|
||||||
modify it is void, and will automatically terminate your rights under
|
|
||||||
this License (including any patent licenses granted under the third
|
|
||||||
paragraph of section 11).
|
|
||||||
|
|
||||||
However, if you cease all violation of this License, then your license
|
|
||||||
from a particular copyright holder is reinstated (a) provisionally,
|
|
||||||
unless and until the copyright holder explicitly and finally
|
|
||||||
terminates your license, and (b) permanently, if the copyright holder
|
|
||||||
fails to notify you of the violation by some reasonable means prior to
|
|
||||||
60 days after the cessation.
|
|
||||||
|
|
||||||
Moreover, your license from a particular copyright holder is
|
|
||||||
reinstated permanently if the copyright holder notifies you of the
|
|
||||||
violation by some reasonable means, this is the first time you have
|
|
||||||
received notice of violation of this License (for any work) from that
|
|
||||||
copyright holder, and you cure the violation prior to 30 days after
|
|
||||||
your receipt of the notice.
|
|
||||||
|
|
||||||
Termination of your rights under this section does not terminate the
|
|
||||||
licenses of parties who have received copies or rights from you under
|
|
||||||
this License. If your rights have been terminated and not permanently
|
|
||||||
reinstated, you do not qualify to receive new licenses for the same
|
|
||||||
material under section 10.
|
|
||||||
|
|
||||||
#### 9. Acceptance Not Required for Having Copies.
|
|
||||||
|
|
||||||
You are not required to accept this License in order to receive or run
|
|
||||||
a copy of the Program. Ancillary propagation of a covered work
|
|
||||||
occurring solely as a consequence of using peer-to-peer transmission
|
|
||||||
to receive a copy likewise does not require acceptance. However,
|
|
||||||
nothing other than this License grants you permission to propagate or
|
|
||||||
modify any covered work. These actions infringe copyright if you do
|
|
||||||
not accept this License. Therefore, by modifying or propagating a
|
|
||||||
covered work, you indicate your acceptance of this License to do so.
|
|
||||||
|
|
||||||
#### 10. Automatic Licensing of Downstream Recipients.
|
|
||||||
|
|
||||||
Each time you convey a covered work, the recipient automatically
|
|
||||||
receives a license from the original licensors, to run, modify and
|
|
||||||
propagate that work, subject to this License. You are not responsible
|
|
||||||
for enforcing compliance by third parties with this License.
|
|
||||||
|
|
||||||
An "entity transaction" is a transaction transferring control of an
|
|
||||||
organization, or substantially all assets of one, or subdividing an
|
|
||||||
organization, or merging organizations. If propagation of a covered
|
|
||||||
work results from an entity transaction, each party to that
|
|
||||||
transaction who receives a copy of the work also receives whatever
|
|
||||||
licenses to the work the party's predecessor in interest had or could
|
|
||||||
give under the previous paragraph, plus a right to possession of the
|
|
||||||
Corresponding Source of the work from the predecessor in interest, if
|
|
||||||
the predecessor has it or can get it with reasonable efforts.
|
|
||||||
|
|
||||||
You may not impose any further restrictions on the exercise of the
|
|
||||||
rights granted or affirmed under this License. For example, you may
|
|
||||||
not impose a license fee, royalty, or other charge for exercise of
|
|
||||||
rights granted under this License, and you may not initiate litigation
|
|
||||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
|
||||||
any patent claim is infringed by making, using, selling, offering for
|
|
||||||
sale, or importing the Program or any portion of it.
|
|
||||||
|
|
||||||
#### 11. Patents.
|
|
||||||
|
|
||||||
A "contributor" is a copyright holder who authorizes use under this
|
|
||||||
License of the Program or a work on which the Program is based. The
|
|
||||||
work thus licensed is called the contributor's "contributor version".
|
|
||||||
|
|
||||||
A contributor's "essential patent claims" are all patent claims owned
|
|
||||||
or controlled by the contributor, whether already acquired or
|
|
||||||
hereafter acquired, that would be infringed by some manner, permitted
|
|
||||||
by this License, of making, using, or selling its contributor version,
|
|
||||||
but do not include claims that would be infringed only as a
|
|
||||||
consequence of further modification of the contributor version. For
|
|
||||||
purposes of this definition, "control" includes the right to grant
|
|
||||||
patent sublicenses in a manner consistent with the requirements of
|
|
||||||
this License.
|
|
||||||
|
|
||||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
|
||||||
patent license under the contributor's essential patent claims, to
|
|
||||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
|
||||||
propagate the contents of its contributor version.
|
|
||||||
|
|
||||||
In the following three paragraphs, a "patent license" is any express
|
|
||||||
agreement or commitment, however denominated, not to enforce a patent
|
|
||||||
(such as an express permission to practice a patent or covenant not to
|
|
||||||
sue for patent infringement). To "grant" such a patent license to a
|
|
||||||
party means to make such an agreement or commitment not to enforce a
|
|
||||||
patent against the party.
|
|
||||||
|
|
||||||
If you convey a covered work, knowingly relying on a patent license,
|
|
||||||
and the Corresponding Source of the work is not available for anyone
|
|
||||||
to copy, free of charge and under the terms of this License, through a
|
|
||||||
publicly available network server or other readily accessible means,
|
|
||||||
then you must either (1) cause the Corresponding Source to be so
|
|
||||||
available, or (2) arrange to deprive yourself of the benefit of the
|
|
||||||
patent license for this particular work, or (3) arrange, in a manner
|
|
||||||
consistent with the requirements of this License, to extend the patent
|
|
||||||
license to downstream recipients. "Knowingly relying" means you have
|
|
||||||
actual knowledge that, but for the patent license, your conveying the
|
|
||||||
covered work in a country, or your recipient's use of the covered work
|
|
||||||
in a country, would infringe one or more identifiable patents in that
|
|
||||||
country that you have reason to believe are valid.
|
|
||||||
|
|
||||||
If, pursuant to or in connection with a single transaction or
|
|
||||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
|
||||||
covered work, and grant a patent license to some of the parties
|
|
||||||
receiving the covered work authorizing them to use, propagate, modify
|
|
||||||
or convey a specific copy of the covered work, then the patent license
|
|
||||||
you grant is automatically extended to all recipients of the covered
|
|
||||||
work and works based on it.
|
|
||||||
|
|
||||||
A patent license is "discriminatory" if it does not include within the
|
|
||||||
scope of its coverage, prohibits the exercise of, or is conditioned on
|
|
||||||
the non-exercise of one or more of the rights that are specifically
|
|
||||||
granted under this License. You may not convey a covered work if you
|
|
||||||
are a party to an arrangement with a third party that is in the
|
|
||||||
business of distributing software, under which you make payment to the
|
|
||||||
third party based on the extent of your activity of conveying the
|
|
||||||
work, and under which the third party grants, to any of the parties
|
|
||||||
who would receive the covered work from you, a discriminatory patent
|
|
||||||
license (a) in connection with copies of the covered work conveyed by
|
|
||||||
you (or copies made from those copies), or (b) primarily for and in
|
|
||||||
connection with specific products or compilations that contain the
|
|
||||||
covered work, unless you entered into that arrangement, or that patent
|
|
||||||
license was granted, prior to 28 March 2007.
|
|
||||||
|
|
||||||
Nothing in this License shall be construed as excluding or limiting
|
|
||||||
any implied license or other defenses to infringement that may
|
|
||||||
otherwise be available to you under applicable patent law.
|
|
||||||
|
|
||||||
#### 12. No Surrender of Others' Freedom.
|
|
||||||
|
|
||||||
If conditions are imposed on you (whether by court order, agreement or
|
|
||||||
otherwise) that contradict the conditions of this License, they do not
|
|
||||||
excuse you from the conditions of this License. If you cannot convey a
|
|
||||||
covered work so as to satisfy simultaneously your obligations under
|
|
||||||
this License and any other pertinent obligations, then as a
|
|
||||||
consequence you may not convey it at all. For example, if you agree to
|
|
||||||
terms that obligate you to collect a royalty for further conveying
|
|
||||||
from those to whom you convey the Program, the only way you could
|
|
||||||
satisfy both those terms and this License would be to refrain entirely
|
|
||||||
from conveying the Program.
|
|
||||||
|
|
||||||
#### 13. Use with the GNU Affero General Public License.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, you have
|
|
||||||
permission to link or combine any covered work with a work licensed
|
|
||||||
under version 3 of the GNU Affero General Public License into a single
|
|
||||||
combined work, and to convey the resulting work. The terms of this
|
|
||||||
License will continue to apply to the part which is the covered work,
|
|
||||||
but the special requirements of the GNU Affero General Public License,
|
|
||||||
section 13, concerning interaction through a network will apply to the
|
|
||||||
combination as such.
|
|
||||||
|
|
||||||
#### 14. Revised Versions of this License.
|
|
||||||
|
|
||||||
The Free Software Foundation may publish revised and/or new versions
|
|
||||||
of the GNU General Public License from time to time. Such new versions
|
|
||||||
will be similar in spirit to the present version, but may differ in
|
|
||||||
detail to address new problems or concerns.
|
|
||||||
|
|
||||||
Each version is given a distinguishing version number. If the Program
|
|
||||||
specifies that a certain numbered version of the GNU General Public
|
|
||||||
License "or any later version" applies to it, you have the option of
|
|
||||||
following the terms and conditions either of that numbered version or
|
|
||||||
of any later version published by the Free Software Foundation. If the
|
|
||||||
Program does not specify a version number of the GNU General Public
|
|
||||||
License, you may choose any version ever published by the Free
|
|
||||||
Software Foundation.
|
|
||||||
|
|
||||||
If the Program specifies that a proxy can decide which future versions
|
|
||||||
of the GNU General Public License can be used, that proxy's public
|
|
||||||
statement of acceptance of a version permanently authorizes you to
|
|
||||||
choose that version for the Program.
|
|
||||||
|
|
||||||
Later license versions may give you additional or different
|
|
||||||
permissions. However, no additional obligations are imposed on any
|
|
||||||
author or copyright holder as a result of your choosing to follow a
|
|
||||||
later version.
|
|
||||||
|
|
||||||
#### 15. Disclaimer of Warranty.
|
|
||||||
|
|
||||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
|
||||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
|
||||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT
|
|
||||||
WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT
|
|
||||||
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
||||||
A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND
|
|
||||||
PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE
|
|
||||||
DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR
|
|
||||||
CORRECTION.
|
|
||||||
|
|
||||||
#### 16. Limitation of Liability.
|
|
||||||
|
|
||||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
|
||||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR
|
|
||||||
CONVEYS THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
|
||||||
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES
|
|
||||||
ARISING OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT
|
|
||||||
NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR
|
|
||||||
LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM
|
|
||||||
TO OPERATE WITH ANY OTHER PROGRAMS), EVEN IF SUCH HOLDER OR OTHER
|
|
||||||
PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
|
|
||||||
|
|
||||||
#### 17. Interpretation of Sections 15 and 16.
|
|
||||||
|
|
||||||
If the disclaimer of warranty and limitation of liability provided
|
|
||||||
above cannot be given local legal effect according to their terms,
|
|
||||||
reviewing courts shall apply local law that most closely approximates
|
|
||||||
an absolute waiver of all civil liability in connection with the
|
|
||||||
Program, unless a warranty or assumption of liability accompanies a
|
|
||||||
copy of the Program in return for a fee.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
### How to Apply These Terms to Your New Programs
|
|
||||||
|
|
||||||
If you develop a new program, and you want it to be of the greatest
|
|
||||||
possible use to the public, the best way to achieve this is to make it
|
|
||||||
free software which everyone can redistribute and change under these
|
|
||||||
terms.
|
|
||||||
|
|
||||||
To do so, attach the following notices to the program. It is safest to
|
|
||||||
attach them to the start of each source file to most effectively state
|
|
||||||
the exclusion of warranty; and each file should have at least the
|
|
||||||
"copyright" line and a pointer to where the full notice is found.
|
|
||||||
|
|
||||||
<one line to give the program's name and a brief idea of what it does.>
|
|
||||||
Copyright (C) <year> <name of author>
|
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
Also add information on how to contact you by electronic and paper
|
|
||||||
mail.
|
|
||||||
|
|
||||||
If the program does terminal interaction, make it output a short
|
|
||||||
notice like this when it starts in an interactive mode:
|
|
||||||
|
|
||||||
<program> Copyright (C) <year> <name of author>
|
|
||||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
|
||||||
This is free software, and you are welcome to redistribute it
|
|
||||||
under certain conditions; type `show c' for details.
|
|
||||||
|
|
||||||
The hypothetical commands \`show w' and \`show c' should show the
|
|
||||||
appropriate parts of the General Public License. Of course, your
|
|
||||||
program's commands might be different; for a GUI interface, you would
|
|
||||||
use an "about box".
|
|
||||||
|
|
||||||
You should also get your employer (if you work as a programmer) or
|
|
||||||
school, if any, to sign a "copyright disclaimer" for the program, if
|
|
||||||
necessary. For more information on this, and how to apply and follow
|
|
||||||
the GNU GPL, see <https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
The GNU General Public License does not permit incorporating your
|
|
||||||
program into proprietary programs. If your program is a subroutine
|
|
||||||
library, you may consider it more useful to permit linking proprietary
|
|
||||||
applications with the library. If this is what you want to do, use the
|
|
||||||
GNU Lesser General Public License instead of this License. But first,
|
|
||||||
please read <https://www.gnu.org/licenses/why-not-lgpl.html>.
|
|
||||||
@@ -1,12 +1,172 @@
|
|||||||
# fcry - [f]ile[cry]pt
|
# fcry - filecrypt
|
||||||
A file en-/decryption tool for easy use.
|
|
||||||
|
|
||||||
Currently `fcry` uses `ChaCha20Poly1305` ([RFC 8439](https://datatracker.ietf.org/doc/html/rfc8439)) as [AEAD](https://en.wikipedia.org/wiki/Authenticated_encryption) cipher provided by the [chacha20poly1305](https://docs.rs/chacha20poly1305/latest/chacha20poly1305/) crate.
|
`fcry` encrypts and decrypts files with an authenticated chunked format. New
|
||||||
|
files use XChaCha20-Poly1305 in a STREAM-style construction: a 19-byte random
|
||||||
|
nonce prefix, a 32-bit chunk counter, and a last-chunk bit form each 24-byte
|
||||||
|
XChaCha nonce. Every chunk authenticates the full file header as AEAD
|
||||||
|
associated data.
|
||||||
|
|
||||||
## Status
|
The tool is intended for local file encryption, scripted backups, and
|
||||||
Currently `fcry` is __not thoroughly tested__ and in __early stages of development__.
|
streaming-friendly decrypts. It is not a general archive format, does not hide
|
||||||
There is a chance, that something is broken as of now.
|
file size, and does not protect plaintext after another process has received
|
||||||
Encryption __seems__ to work, but due to a possible lack of understanding of some underlying methods
|
it.
|
||||||
(or misinterpretation) it could theoretically be not effective at all.
|
|
||||||
|
|
||||||
See [TODO.md](/ddidderr/fcry/src/branch/main/TODO.md) for further information.
|
## Usage
|
||||||
|
|
||||||
|
Encrypt with an interactive passphrase:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
fcry -i plain.bin -o plain.bin.fcry --passphrase
|
||||||
|
```
|
||||||
|
|
||||||
|
Decrypt with the same passphrase:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
fcry -d -i plain.bin.fcry -o plain.bin --passphrase
|
||||||
|
```
|
||||||
|
|
||||||
|
Use a raw 32-byte key file instead of a passphrase:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
fcry -i plain.bin -o plain.bin.fcry --key-file key.bin
|
||||||
|
fcry -d -i plain.bin.fcry -o plain.bin --key-file key.bin
|
||||||
|
```
|
||||||
|
|
||||||
|
For non-interactive passphrase use:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
FCRY_PASSWORD='correct horse battery staple' \
|
||||||
|
fcry -i plain.bin -o plain.bin.fcry --passphrase-env FCRY_PASSWORD
|
||||||
|
```
|
||||||
|
|
||||||
|
`--passphrase-env` is useful for automation, but the environment variable can
|
||||||
|
remain visible to the current process environment and platform tooling. Prefer
|
||||||
|
interactive entry or a protected key file when possible.
|
||||||
|
|
||||||
|
## Safety Properties
|
||||||
|
|
||||||
|
- File outputs are written to private, randomly named temporary files and are
|
||||||
|
renamed into place only after encryption or decryption succeeds. Existing
|
||||||
|
outputs require `--force`, except for the self-replacement case that is
|
||||||
|
handled through the temporary file.
|
||||||
|
- New passphrase encryptions use Argon2id by default with 1024 MiB of memory,
|
||||||
|
2 passes, and 4 lanes. Passphrases must be non-empty and at least 12 UTF-8
|
||||||
|
bytes unless `--allow-weak-kdf` is explicitly supplied for tests or legacy
|
||||||
|
interop.
|
||||||
|
- Decryption enforces a memory ceiling for Argon2id headers. The default cap is
|
||||||
|
the lower of 4096 MiB, the architecture limit, and available Linux memory
|
||||||
|
when that can be detected. Override it with `--max-argon-memory-mib` only for
|
||||||
|
files you trust.
|
||||||
|
- Chunk size is bounded to `1..=64 MiB`. Worker threads are capped at 256, and
|
||||||
|
the pipeline bounds in-flight chunk memory.
|
||||||
|
- v3 files carry a key commitment derived from the stretched key and committed
|
||||||
|
header fields. This gives a fast, clear wrong-key failure before chunk
|
||||||
|
processing and prevents stripping or downgrading the commitment without
|
||||||
|
authentication failure.
|
||||||
|
- On Unix, `fcry` makes a best-effort call to disable core dumps for the process
|
||||||
|
before handling secrets.
|
||||||
|
|
||||||
|
## Format
|
||||||
|
|
||||||
|
The current on-disk format version is v3.
|
||||||
|
|
||||||
|
```text
|
||||||
|
magic "fcry" 4 bytes
|
||||||
|
version u8 1
|
||||||
|
alg_id u8 1 (1 = XChaCha20-Poly1305)
|
||||||
|
flags u8 1
|
||||||
|
reserved u8 1 (must be 0)
|
||||||
|
chunk_size u32 LE 4
|
||||||
|
kdf_id u8 1 (0 = raw key, 1 = Argon2id)
|
||||||
|
kdf_params variable
|
||||||
|
nonce_prefix [u8; 19]
|
||||||
|
plaintext_length u64 LE only when flags bit 0 is set
|
||||||
|
key_commitment [u8; 32] only when flags bit 1 is set
|
||||||
|
ciphertext chunks each plaintext chunk plus a 16-byte Poly1305 tag
|
||||||
|
```
|
||||||
|
|
||||||
|
The encoded header is AEAD associated data for every chunk. Changing the chunk
|
||||||
|
size, KDF parameters, nonce prefix, committed plaintext length, key commitment,
|
||||||
|
or other header bytes causes authentication failure.
|
||||||
|
|
||||||
|
Version history:
|
||||||
|
|
||||||
|
- v1: no flags and no committed plaintext length.
|
||||||
|
- v2: adds the length-committed flag and optional `plaintext_length`.
|
||||||
|
- v3: requires the key-commitment flag and stores the 32-byte key commitment.
|
||||||
|
|
||||||
|
Regular file encryption commits `plaintext_length` in the header. Stdin
|
||||||
|
encryption cannot know the final length up front, so stdin-produced files do
|
||||||
|
not support random-access decrypt.
|
||||||
|
|
||||||
|
## Streaming And Ranges
|
||||||
|
|
||||||
|
Normal decrypt-to-stdout emits each plaintext chunk after that chunk has
|
||||||
|
authenticated. This means a truncated ciphertext can produce an authentic
|
||||||
|
prefix on stdout before the final truncation error is reported. That is
|
||||||
|
inherent to chunked streaming AE when bytes are released immediately.
|
||||||
|
|
||||||
|
Use `--buffer-verify` when decrypting to stdout if downstream consumers must
|
||||||
|
not see any plaintext until the whole file has authenticated:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
fcry -d -i plain.bin.fcry --passphrase --buffer-verify > plain.bin
|
||||||
|
```
|
||||||
|
|
||||||
|
`--buffer-verify` writes plaintext to a private temporary file first, verifies
|
||||||
|
the complete ciphertext, and copies to stdout only after success. File outputs
|
||||||
|
already get atomic temporary-file behavior, so `--buffer-verify` is only valid
|
||||||
|
for decrypt-to-stdout.
|
||||||
|
|
||||||
|
Random-access decrypt requires `--decrypt`, `--input-file`, `--offset`, and
|
||||||
|
`--length`, and the input must have a length-committed header:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
fcry -d -i plain.bin.fcry --passphrase --offset 1048576 --length 4096 > slice.bin
|
||||||
|
```
|
||||||
|
|
||||||
|
A successful range decrypt authenticates the requested chunks and header. It
|
||||||
|
does not prove that the rest of the file is present or untampered. Use a full
|
||||||
|
decrypt when you need whole-file integrity. `--length 0` is rejected because it
|
||||||
|
would authenticate no chunks.
|
||||||
|
|
||||||
|
## Threat Model
|
||||||
|
|
||||||
|
`fcry` aims to provide confidentiality and integrity for file contents against
|
||||||
|
an attacker who can read, copy, truncate, replace, or modify ciphertext files
|
||||||
|
after encryption. With passphrase mode, offline guessing is still possible; the
|
||||||
|
Argon2id parameters make each guess expensive but cannot make a weak passphrase
|
||||||
|
safe.
|
||||||
|
|
||||||
|
The format authenticates all header fields that affect decryption, including
|
||||||
|
KDF parameters, chunk size, nonce prefix, committed plaintext length, and key
|
||||||
|
commitment. Unknown header flags and unsupported algorithms are rejected.
|
||||||
|
|
||||||
|
The following are explicit non-goals:
|
||||||
|
|
||||||
|
- Hiding plaintext length or access patterns. `plaintext_length` is cleartext
|
||||||
|
for regular-file encryptions, and ciphertext length already reveals an
|
||||||
|
approximate plaintext size. There is no padding scheme.
|
||||||
|
- Preventing plaintext exposure after successful decrypt. Plaintext written to
|
||||||
|
stdout, files, pipes, shell history, terminals, swap, backups, or downstream
|
||||||
|
tools is outside `fcry`'s control.
|
||||||
|
- Protecting plaintext chunk buffers from every local memory-forensics route.
|
||||||
|
Keys and passphrases use protected/zeroizing storage where practical, and
|
||||||
|
chunk buffers are zeroized on drop, but decrypted plaintext necessarily exists
|
||||||
|
in ordinary process memory while being processed.
|
||||||
|
- Disabling Windows Error Reporting or minidumps. Unlike Unix core dumps, those
|
||||||
|
are controlled by per-machine Windows policy; `fcry` records this as an
|
||||||
|
operator/deployment responsibility rather than changing host-wide policy.
|
||||||
|
- Recovering from loss of the passphrase or raw key file. There is no escrow or
|
||||||
|
backdoor.
|
||||||
|
|
||||||
|
## Operational Notes
|
||||||
|
|
||||||
|
- Keep backups of important plaintext until you have verified the encrypted
|
||||||
|
file and your recovery path.
|
||||||
|
- Store raw key files with restrictive permissions. On Unix, `fcry` warns when
|
||||||
|
a key file is group/world accessible.
|
||||||
|
- Use `--allow-weak-kdf` only for tests or compatibility with old intentionally
|
||||||
|
weak files.
|
||||||
|
- Use `--temp-dir` when the default temporary-file location is not acceptable
|
||||||
|
for decrypt-to-stdout buffering or output staging.
|
||||||
|
|||||||
@@ -0,0 +1,33 @@
|
|||||||
|
set positional-arguments
|
||||||
|
|
||||||
|
run *args:
|
||||||
|
cargo run -- "$@"
|
||||||
|
|
||||||
|
build:
|
||||||
|
cargo build
|
||||||
|
|
||||||
|
build-release:
|
||||||
|
cargo build --release
|
||||||
|
|
||||||
|
build-production:
|
||||||
|
cargo build --profile production
|
||||||
|
|
||||||
|
fmt:
|
||||||
|
cargo +nightly fmt
|
||||||
|
tombi format
|
||||||
|
just --fmt
|
||||||
|
|
||||||
|
_fix:
|
||||||
|
cargo fix
|
||||||
|
cargo clippy --fix
|
||||||
|
|
||||||
|
fix: _fix fmt
|
||||||
|
|
||||||
|
clippy:
|
||||||
|
cargo clippy --workspace --all-targets --all-features -- -D warnings
|
||||||
|
|
||||||
|
test:
|
||||||
|
cargo test --workspace
|
||||||
|
|
||||||
|
clean:
|
||||||
|
cargo clean
|
||||||
+186
-37
@@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: MIT-0
|
||||||
|
|
||||||
use chacha20poly1305::{KeyInit, XChaCha20Poly1305, XNonce, aead::AeadInPlace};
|
use chacha20poly1305::{KeyInit, XChaCha20Poly1305, XNonce, aead::AeadInPlace};
|
||||||
use std::fs::File;
|
use std::fs::File;
|
||||||
@@ -7,12 +7,15 @@ use std::sync::Arc;
|
|||||||
|
|
||||||
use crate::error::*;
|
use crate::error::*;
|
||||||
use crate::header::{
|
use crate::header::{
|
||||||
AlgId, FLAG_LENGTH_COMMITTED, Header, KdfParams, NONCE_PREFIX_LEN, TAG_LEN, VERSION_CURRENT,
|
AlgId, FLAG_KEY_COMMITTED, FLAG_LENGTH_COMMITTED, Header, KdfParams, NONCE_PREFIX_LEN, TAG_LEN,
|
||||||
|
VERSION_CURRENT,
|
||||||
};
|
};
|
||||||
use crate::pipeline;
|
use crate::pipeline;
|
||||||
|
use crate::policy;
|
||||||
use crate::reader::{AheadReader, ReadInfoChunk};
|
use crate::reader::{AheadReader, ReadInfoChunk};
|
||||||
use crate::secrets::{SecretBytes32, SecretVec};
|
use crate::secrets::{SecretBytes32, SecretVec};
|
||||||
use crate::utils::*;
|
use crate::utils::*;
|
||||||
|
use zeroize::Zeroizing;
|
||||||
|
|
||||||
/// XChaCha20Poly1305 nonce: 24 bytes total. STREAM splits the trailing 5 bytes
|
/// XChaCha20Poly1305 nonce: 24 bytes total. STREAM splits the trailing 5 bytes
|
||||||
/// into a 4-byte big-endian counter and a 1-byte "last block" flag.
|
/// into a 4-byte big-endian counter and a 1-byte "last block" flag.
|
||||||
@@ -40,7 +43,7 @@ pub fn derive_key(
|
|||||||
match kdf {
|
match kdf {
|
||||||
KdfParams::Raw => {
|
KdfParams::Raw => {
|
||||||
let raw =
|
let raw =
|
||||||
raw_key.ok_or_else(|| FcryError::Format("raw kdf requires --raw-key".into()))?;
|
raw_key.ok_or_else(|| FcryError::Format("raw kdf requires --key-file".into()))?;
|
||||||
raw.with_array(|raw| out.with_mut_array(|out| out.copy_from_slice(raw)));
|
raw.with_array(|raw| out.with_mut_array(|out| out.copy_from_slice(raw)));
|
||||||
}
|
}
|
||||||
KdfParams::Argon2id {
|
KdfParams::Argon2id {
|
||||||
@@ -67,6 +70,32 @@ fn build_aead(key: &SecretBytes32) -> Arc<XChaCha20Poly1305> {
|
|||||||
Arc::new(key.with_array(|key| XChaCha20Poly1305::new(key.into())))
|
Arc::new(key.with_array(|key| XChaCha20Poly1305::new(key.into())))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn compute_key_commitment(key: &SecretBytes32, header: &Header) -> [u8; 32] {
|
||||||
|
key.with_array(|key| {
|
||||||
|
let mut hasher = blake3::Hasher::new_keyed(key);
|
||||||
|
hasher.update(b"fcry-kcv-v3");
|
||||||
|
hasher.update(&[0]);
|
||||||
|
hasher.update(&header.commitment_input_encoding());
|
||||||
|
*hasher.finalize().as_bytes()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn verify_key_commitment(header: &Header, key: &SecretBytes32) -> Result<(), FcryError> {
|
||||||
|
let Some(expected) = header.key_commitment else {
|
||||||
|
return Ok(());
|
||||||
|
};
|
||||||
|
let actual = compute_key_commitment(key, header);
|
||||||
|
let mut diff = 0u8;
|
||||||
|
for (a, b) in actual.iter().zip(expected.iter()) {
|
||||||
|
diff |= a ^ b;
|
||||||
|
}
|
||||||
|
if diff == 0 {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(FcryError::WrongKey)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Bump the per-chunk counter; surface a domain error on overflow rather than
|
/// Bump the per-chunk counter; surface a domain error on overflow rather than
|
||||||
/// panicking on debug or wrapping in release.
|
/// panicking on debug or wrapping in release.
|
||||||
pub(crate) fn bump_counter(counter: u32) -> Result<u32, FcryError> {
|
pub(crate) fn bump_counter(counter: u32) -> Result<u32, FcryError> {
|
||||||
@@ -75,6 +104,7 @@ pub(crate) fn bump_counter(counter: u32) -> Result<u32, FcryError> {
|
|||||||
.ok_or_else(|| FcryError::Format("STREAM counter overflow (input too large)".into()))
|
.ok_or_else(|| FcryError::Format("STREAM counter overflow (input too large)".into()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
pub fn encrypt<S: AsRef<str>>(
|
pub fn encrypt<S: AsRef<str>>(
|
||||||
input_file: Option<S>,
|
input_file: Option<S>,
|
||||||
output_file: Option<S>,
|
output_file: Option<S>,
|
||||||
@@ -83,11 +113,31 @@ pub fn encrypt<S: AsRef<str>>(
|
|||||||
kdf: KdfParams,
|
kdf: KdfParams,
|
||||||
threads: usize,
|
threads: usize,
|
||||||
) -> Result<(), FcryError> {
|
) -> Result<(), FcryError> {
|
||||||
let chunk_sz = chunk_size as usize;
|
encrypt_with_output_options(
|
||||||
|
input_file,
|
||||||
|
output_file,
|
||||||
|
key,
|
||||||
|
chunk_size,
|
||||||
|
kdf,
|
||||||
|
threads,
|
||||||
|
&OutSinkOptions::default(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn encrypt_with_output_options<S: AsRef<str>>(
|
||||||
|
input_file: Option<S>,
|
||||||
|
output_file: Option<S>,
|
||||||
|
key: &SecretBytes32,
|
||||||
|
chunk_size: u32,
|
||||||
|
kdf: KdfParams,
|
||||||
|
threads: usize,
|
||||||
|
output_options: &OutSinkOptions,
|
||||||
|
) -> Result<(), FcryError> {
|
||||||
|
let chunk_sz = policy::validate_chunk_size(chunk_size)?;
|
||||||
let input = open_input(input_file)?;
|
let input = open_input(input_file)?;
|
||||||
let plaintext_length = input.length;
|
let plaintext_length = input.length;
|
||||||
let mut f_plain = AheadReader::from(input.reader, chunk_sz);
|
let mut f_plain = AheadReader::from(input.reader, chunk_sz);
|
||||||
let mut f_encrypted = OutSink::open(output_file)?;
|
let mut f_encrypted = OutSink::open_with_options(output_file, output_options)?;
|
||||||
|
|
||||||
let mut nonce_prefix = [0u8; NONCE_PREFIX_LEN];
|
let mut nonce_prefix = [0u8; NONCE_PREFIX_LEN];
|
||||||
getrandom::fill(&mut nonce_prefix)?;
|
getrandom::fill(&mut nonce_prefix)?;
|
||||||
@@ -96,8 +146,8 @@ pub fn encrypt<S: AsRef<str>>(
|
|||||||
FLAG_LENGTH_COMMITTED
|
FLAG_LENGTH_COMMITTED
|
||||||
} else {
|
} else {
|
||||||
0
|
0
|
||||||
};
|
} | FLAG_KEY_COMMITTED;
|
||||||
let header = Header {
|
let mut header = Header {
|
||||||
version: VERSION_CURRENT,
|
version: VERSION_CURRENT,
|
||||||
alg: AlgId::XChaCha20Poly1305,
|
alg: AlgId::XChaCha20Poly1305,
|
||||||
flags,
|
flags,
|
||||||
@@ -105,7 +155,9 @@ pub fn encrypt<S: AsRef<str>>(
|
|||||||
kdf,
|
kdf,
|
||||||
nonce_prefix,
|
nonce_prefix,
|
||||||
plaintext_length,
|
plaintext_length,
|
||||||
|
key_commitment: None,
|
||||||
};
|
};
|
||||||
|
header.key_commitment = Some(compute_key_commitment(key, &header));
|
||||||
let aad = Arc::new(header.encode());
|
let aad = Arc::new(header.encode());
|
||||||
f_encrypted.write_all(&aad)?;
|
f_encrypted.write_all(&aad)?;
|
||||||
|
|
||||||
@@ -124,7 +176,7 @@ pub fn encrypt<S: AsRef<str>>(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut buf = vec![0u8; chunk_sz];
|
let mut buf = Zeroizing::new(vec![0u8; chunk_sz]);
|
||||||
let mut counter: u32 = 0;
|
let mut counter: u32 = 0;
|
||||||
let mut bytes_seen: u64 = 0;
|
let mut bytes_seen: u64 = 0;
|
||||||
|
|
||||||
@@ -132,18 +184,18 @@ pub fn encrypt<S: AsRef<str>>(
|
|||||||
match f_plain.read_ahead(&mut buf)? {
|
match f_plain.read_ahead(&mut buf)? {
|
||||||
ReadInfoChunk::Normal(_) => {
|
ReadInfoChunk::Normal(_) => {
|
||||||
let nonce = make_nonce(&nonce_prefix, counter, false);
|
let nonce = make_nonce(&nonce_prefix, counter, false);
|
||||||
aead.encrypt_in_place(&nonce, &aad, &mut buf)?;
|
aead.encrypt_in_place(&nonce, &aad, &mut *buf)?;
|
||||||
f_encrypted.write_all(&buf)?;
|
f_encrypted.write_all(&buf)?;
|
||||||
buf.truncate(chunk_sz);
|
buf.truncate(chunk_sz);
|
||||||
bytes_seen = bytes_seen.saturating_add(chunk_sz as u64);
|
bytes_seen = policy::checked_count_add(bytes_seen, chunk_sz, "bytes read")?;
|
||||||
counter = bump_counter(counter)?;
|
counter = bump_counter(counter)?;
|
||||||
}
|
}
|
||||||
ReadInfoChunk::Last(n) => {
|
ReadInfoChunk::Last(n) => {
|
||||||
buf.truncate(n);
|
buf.truncate(n);
|
||||||
let nonce = make_nonce(&nonce_prefix, counter, true);
|
let nonce = make_nonce(&nonce_prefix, counter, true);
|
||||||
aead.encrypt_in_place(&nonce, &aad, &mut buf)?;
|
aead.encrypt_in_place(&nonce, &aad, &mut *buf)?;
|
||||||
f_encrypted.write_all(&buf)?;
|
f_encrypted.write_all(&buf)?;
|
||||||
bytes_seen = bytes_seen.saturating_add(n as u64);
|
bytes_seen = policy::checked_count_add(bytes_seen, n, "bytes read")?;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
ReadInfoChunk::Empty => {
|
ReadInfoChunk::Empty => {
|
||||||
@@ -151,7 +203,7 @@ pub fn encrypt<S: AsRef<str>>(
|
|||||||
// authenticates the (empty) stream rather than silently producing nothing.
|
// authenticates the (empty) stream rather than silently producing nothing.
|
||||||
buf.clear();
|
buf.clear();
|
||||||
let nonce = make_nonce(&nonce_prefix, counter, true);
|
let nonce = make_nonce(&nonce_prefix, counter, true);
|
||||||
aead.encrypt_in_place(&nonce, &aad, &mut buf)?;
|
aead.encrypt_in_place(&nonce, &aad, &mut *buf)?;
|
||||||
f_encrypted.write_all(&buf)?;
|
f_encrypted.write_all(&buf)?;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -173,24 +225,65 @@ pub fn encrypt<S: AsRef<str>>(
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
pub fn decrypt<S: AsRef<str>>(
|
pub fn decrypt<S: AsRef<str>>(
|
||||||
input_file: Option<S>,
|
input_file: Option<S>,
|
||||||
output_file: Option<S>,
|
output_file: Option<S>,
|
||||||
raw_key: Option<&SecretBytes32>,
|
raw_key: Option<&SecretBytes32>,
|
||||||
passphrase: Option<&SecretVec>,
|
passphrase: Option<&SecretVec>,
|
||||||
threads: usize,
|
threads: usize,
|
||||||
|
) -> Result<(), FcryError> {
|
||||||
|
decrypt_with_argon_cap(
|
||||||
|
input_file,
|
||||||
|
output_file,
|
||||||
|
raw_key,
|
||||||
|
passphrase,
|
||||||
|
threads,
|
||||||
|
policy::default_argon_decrypt_cap_mib(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub fn decrypt_with_argon_cap<S: AsRef<str>>(
|
||||||
|
input_file: Option<S>,
|
||||||
|
output_file: Option<S>,
|
||||||
|
raw_key: Option<&SecretBytes32>,
|
||||||
|
passphrase: Option<&SecretVec>,
|
||||||
|
threads: usize,
|
||||||
|
max_argon_memory_mib: u32,
|
||||||
|
) -> Result<(), FcryError> {
|
||||||
|
decrypt_with_output_options(
|
||||||
|
input_file,
|
||||||
|
output_file,
|
||||||
|
raw_key,
|
||||||
|
passphrase,
|
||||||
|
threads,
|
||||||
|
max_argon_memory_mib,
|
||||||
|
&OutSinkOptions::default(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn decrypt_with_output_options<S: AsRef<str>>(
|
||||||
|
input_file: Option<S>,
|
||||||
|
output_file: Option<S>,
|
||||||
|
raw_key: Option<&SecretBytes32>,
|
||||||
|
passphrase: Option<&SecretVec>,
|
||||||
|
threads: usize,
|
||||||
|
max_argon_memory_mib: u32,
|
||||||
|
output_options: &OutSinkOptions,
|
||||||
) -> Result<(), FcryError> {
|
) -> Result<(), FcryError> {
|
||||||
let mut reader = open_input(input_file)?.reader;
|
let mut reader = open_input(input_file)?.reader;
|
||||||
let header = Header::read(&mut reader)?;
|
let header = Header::read_with_argon_cap(&mut reader, max_argon_memory_mib)?;
|
||||||
let aad = Arc::new(header.encode());
|
let aad = Arc::new(header.encode());
|
||||||
|
|
||||||
let key = derive_key(&header.kdf, raw_key, passphrase)?;
|
let key = derive_key(&header.kdf, raw_key, passphrase)?;
|
||||||
|
verify_key_commitment(&header, &key)?;
|
||||||
|
|
||||||
let chunk_sz = header.chunk_size as usize;
|
let chunk_sz = policy::validate_chunk_size(header.chunk_size)?;
|
||||||
let cipher_chunk = chunk_sz + TAG_LEN;
|
let cipher_chunk = policy::cipher_chunk_len(chunk_sz)?;
|
||||||
|
|
||||||
let mut f_encrypted = AheadReader::from(reader, cipher_chunk);
|
let mut f_encrypted = AheadReader::from(reader, cipher_chunk);
|
||||||
let mut f_plain = OutSink::open(output_file)?;
|
let mut f_plain = OutSink::open_with_options(output_file, output_options)?;
|
||||||
|
|
||||||
let aead = build_aead(&key);
|
let aead = build_aead(&key);
|
||||||
|
|
||||||
@@ -207,7 +300,7 @@ pub fn decrypt<S: AsRef<str>>(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut buf = vec![0u8; cipher_chunk];
|
let mut buf = Zeroizing::new(vec![0u8; cipher_chunk]);
|
||||||
let mut counter: u32 = 0;
|
let mut counter: u32 = 0;
|
||||||
let mut bytes_written: u64 = 0;
|
let mut bytes_written: u64 = 0;
|
||||||
|
|
||||||
@@ -215,18 +308,20 @@ pub fn decrypt<S: AsRef<str>>(
|
|||||||
match f_encrypted.read_ahead(&mut buf)? {
|
match f_encrypted.read_ahead(&mut buf)? {
|
||||||
ReadInfoChunk::Normal(_) => {
|
ReadInfoChunk::Normal(_) => {
|
||||||
let nonce = make_nonce(&header.nonce_prefix, counter, false);
|
let nonce = make_nonce(&header.nonce_prefix, counter, false);
|
||||||
aead.decrypt_in_place(&nonce, &aad, &mut buf)?;
|
aead.decrypt_in_place(&nonce, &aad, &mut *buf)?;
|
||||||
f_plain.write_all(&buf)?;
|
f_plain.write_all(&buf)?;
|
||||||
bytes_written = bytes_written.saturating_add(buf.len() as u64);
|
bytes_written =
|
||||||
|
policy::checked_count_add(bytes_written, buf.len(), "bytes written")?;
|
||||||
buf.resize(cipher_chunk, 0);
|
buf.resize(cipher_chunk, 0);
|
||||||
counter = bump_counter(counter)?;
|
counter = bump_counter(counter)?;
|
||||||
}
|
}
|
||||||
ReadInfoChunk::Last(n) => {
|
ReadInfoChunk::Last(n) => {
|
||||||
buf.truncate(n);
|
buf.truncate(n);
|
||||||
let nonce = make_nonce(&header.nonce_prefix, counter, true);
|
let nonce = make_nonce(&header.nonce_prefix, counter, true);
|
||||||
aead.decrypt_in_place(&nonce, &aad, &mut buf)?;
|
aead.decrypt_in_place(&nonce, &aad, &mut *buf)?;
|
||||||
f_plain.write_all(&buf)?;
|
f_plain.write_all(&buf)?;
|
||||||
bytes_written = bytes_written.saturating_add(buf.len() as u64);
|
bytes_written =
|
||||||
|
policy::checked_count_add(bytes_written, buf.len(), "bytes written")?;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
ReadInfoChunk::Empty => {
|
ReadInfoChunk::Empty => {
|
||||||
@@ -253,6 +348,7 @@ pub fn decrypt<S: AsRef<str>>(
|
|||||||
/// whose header has `FLAG_LENGTH_COMMITTED` set, so we know exactly where
|
/// whose header has `FLAG_LENGTH_COMMITTED` set, so we know exactly where
|
||||||
/// each ciphertext chunk lives and which chunk is the last (its nonce uses
|
/// each ciphertext chunk lives and which chunk is the last (its nonce uses
|
||||||
/// the STREAM last-block flag).
|
/// the STREAM last-block flag).
|
||||||
|
#[allow(dead_code)]
|
||||||
pub fn decrypt_range<S: AsRef<str>>(
|
pub fn decrypt_range<S: AsRef<str>>(
|
||||||
input_file: &str,
|
input_file: &str,
|
||||||
output_file: Option<S>,
|
output_file: Option<S>,
|
||||||
@@ -261,9 +357,56 @@ pub fn decrypt_range<S: AsRef<str>>(
|
|||||||
offset: u64,
|
offset: u64,
|
||||||
length: u64,
|
length: u64,
|
||||||
) -> Result<(), FcryError> {
|
) -> Result<(), FcryError> {
|
||||||
|
decrypt_range_with_argon_cap(
|
||||||
|
input_file,
|
||||||
|
output_file,
|
||||||
|
raw_key,
|
||||||
|
passphrase,
|
||||||
|
offset,
|
||||||
|
length,
|
||||||
|
policy::default_argon_decrypt_cap_mib(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub fn decrypt_range_with_argon_cap<S: AsRef<str>>(
|
||||||
|
input_file: &str,
|
||||||
|
output_file: Option<S>,
|
||||||
|
raw_key: Option<&SecretBytes32>,
|
||||||
|
passphrase: Option<&SecretVec>,
|
||||||
|
offset: u64,
|
||||||
|
length: u64,
|
||||||
|
max_argon_memory_mib: u32,
|
||||||
|
) -> Result<(), FcryError> {
|
||||||
|
decrypt_range_with_output_options(
|
||||||
|
input_file,
|
||||||
|
output_file,
|
||||||
|
raw_key,
|
||||||
|
passphrase,
|
||||||
|
offset,
|
||||||
|
length,
|
||||||
|
max_argon_memory_mib,
|
||||||
|
&OutSinkOptions::default(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::too_many_arguments)]
|
||||||
|
pub fn decrypt_range_with_output_options<S: AsRef<str>>(
|
||||||
|
input_file: &str,
|
||||||
|
output_file: Option<S>,
|
||||||
|
raw_key: Option<&SecretBytes32>,
|
||||||
|
passphrase: Option<&SecretVec>,
|
||||||
|
offset: u64,
|
||||||
|
length: u64,
|
||||||
|
max_argon_memory_mib: u32,
|
||||||
|
output_options: &OutSinkOptions,
|
||||||
|
) -> Result<(), FcryError> {
|
||||||
|
if length == 0 {
|
||||||
|
return Err(FcryError::Format("--length 0 is not allowed".into()));
|
||||||
|
}
|
||||||
let file = File::open(input_file)?;
|
let file = File::open(input_file)?;
|
||||||
let mut reader = BufReader::new(file);
|
let mut reader = BufReader::new(file);
|
||||||
let header = Header::read(&mut reader)?;
|
let header = Header::read_with_argon_cap(&mut reader, max_argon_memory_mib)?;
|
||||||
let aad = header.encode();
|
let aad = header.encode();
|
||||||
let header_len = aad.len() as u64;
|
let header_len = aad.len() as u64;
|
||||||
|
|
||||||
@@ -283,10 +426,13 @@ pub fn decrypt_range<S: AsRef<str>>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
let key = derive_key(&header.kdf, raw_key, passphrase)?;
|
let key = derive_key(&header.kdf, raw_key, passphrase)?;
|
||||||
|
verify_key_commitment(&header, &key)?;
|
||||||
let aead = build_aead(&key);
|
let aead = build_aead(&key);
|
||||||
|
|
||||||
let chunk_sz = header.chunk_size as u64;
|
let chunk_sz_usize = policy::validate_chunk_size(header.chunk_size)?;
|
||||||
let cipher_chunk = chunk_sz + TAG_LEN as u64;
|
let cipher_chunk_usize = policy::cipher_chunk_len(chunk_sz_usize)?;
|
||||||
|
let chunk_sz = chunk_sz_usize as u64;
|
||||||
|
let cipher_chunk = cipher_chunk_usize as u64;
|
||||||
|
|
||||||
// Layout invariants:
|
// Layout invariants:
|
||||||
// n_chunks = ceil(total / chunk_sz), but always ≥ 1 (the empty file
|
// n_chunks = ceil(total / chunk_sz), but always ≥ 1 (the empty file
|
||||||
@@ -297,23 +443,21 @@ pub fn decrypt_range<S: AsRef<str>>(
|
|||||||
(1u64, 0u64)
|
(1u64, 0u64)
|
||||||
} else {
|
} else {
|
||||||
let n = total.div_ceil(chunk_sz);
|
let n = total.div_ceil(chunk_sz);
|
||||||
let last = total - (n - 1) * chunk_sz;
|
let before_last = policy::checked_mul_u64(n - 1, chunk_sz, "last chunk offset")?;
|
||||||
|
let last = total
|
||||||
|
.checked_sub(before_last)
|
||||||
|
.ok_or_else(|| FcryError::Format("last chunk length underflow".into()))?;
|
||||||
(n, last)
|
(n, last)
|
||||||
};
|
};
|
||||||
let last_idx = n_chunks - 1;
|
let last_idx = n_chunks - 1;
|
||||||
|
|
||||||
let mut out = OutSink::open(output_file)?;
|
let mut out = OutSink::open_with_options(output_file, output_options)?;
|
||||||
|
|
||||||
if length == 0 {
|
|
||||||
out.commit()?;
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let start_chunk = offset / chunk_sz;
|
let start_chunk = offset / chunk_sz;
|
||||||
let end_chunk = (end - 1) / chunk_sz;
|
let end_chunk = (end - 1) / chunk_sz;
|
||||||
|
|
||||||
// Reusable buffer sized to a full chunk + tag.
|
// Reusable buffer sized to a full chunk + tag.
|
||||||
let mut buf = Vec::with_capacity(cipher_chunk as usize);
|
let mut buf = Zeroizing::new(Vec::with_capacity(cipher_chunk_usize));
|
||||||
|
|
||||||
let mut file = reader.into_inner();
|
let mut file = reader.into_inner();
|
||||||
|
|
||||||
@@ -329,19 +473,23 @@ pub fn decrypt_range<S: AsRef<str>>(
|
|||||||
let cipher_len_usz =
|
let cipher_len_usz =
|
||||||
usize::try_from(cipher_len).map_err(|_| FcryError::Format("chunk too big".into()))?;
|
usize::try_from(cipher_len).map_err(|_| FcryError::Format("chunk too big".into()))?;
|
||||||
|
|
||||||
let chunk_offset = header_len + i * cipher_chunk;
|
let chunk_offset = policy::checked_add_u64(
|
||||||
|
header_len,
|
||||||
|
policy::checked_mul_u64(i, cipher_chunk, "ciphertext chunk offset")?,
|
||||||
|
"ciphertext chunk offset",
|
||||||
|
)?;
|
||||||
file.seek(SeekFrom::Start(chunk_offset))?;
|
file.seek(SeekFrom::Start(chunk_offset))?;
|
||||||
buf.clear();
|
buf.clear();
|
||||||
buf.resize(cipher_len_usz, 0);
|
buf.resize(cipher_len_usz, 0);
|
||||||
file.read_exact(&mut buf)?;
|
file.read_exact(&mut buf)?;
|
||||||
|
|
||||||
let nonce = make_nonce(&header.nonce_prefix, i_u32, is_last);
|
let nonce = make_nonce(&header.nonce_prefix, i_u32, is_last);
|
||||||
aead.decrypt_in_place(&nonce, &aad, &mut buf)?;
|
aead.decrypt_in_place(&nonce, &aad, &mut *buf)?;
|
||||||
|
|
||||||
// `buf` is now plaintext for this chunk. Compute the chunk's plaintext
|
// `buf` is now plaintext for this chunk. Compute the chunk's plaintext
|
||||||
// window in absolute bytes and intersect with the requested range.
|
// window in absolute bytes and intersect with the requested range.
|
||||||
let chunk_start = i * chunk_sz;
|
let chunk_start = policy::checked_mul_u64(i, chunk_sz, "plaintext chunk offset")?;
|
||||||
let chunk_end = chunk_start + buf.len() as u64;
|
let chunk_end = policy::checked_count_add(chunk_start, buf.len(), "plaintext chunk end")?;
|
||||||
let lo = offset.max(chunk_start) - chunk_start;
|
let lo = offset.max(chunk_start) - chunk_start;
|
||||||
let hi = end.min(chunk_end) - chunk_start;
|
let hi = end.min(chunk_end) - chunk_start;
|
||||||
out.write_all(&buf[lo as usize..hi as usize])?;
|
out.write_all(&buf[lo as usize..hi as usize])?;
|
||||||
@@ -375,6 +523,7 @@ mod tests {
|
|||||||
kdf: KdfParams::Raw,
|
kdf: KdfParams::Raw,
|
||||||
nonce_prefix,
|
nonce_prefix,
|
||||||
plaintext_length: None,
|
plaintext_length: None,
|
||||||
|
key_commitment: None,
|
||||||
};
|
};
|
||||||
let aad = header.encode();
|
let aad = header.encode();
|
||||||
// First byte after MAGIC is the version — verify our fixture really
|
// First byte after MAGIC is the version — verify our fixture really
|
||||||
|
|||||||
+2
-1
@@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: MIT-0
|
||||||
|
|
||||||
use chacha20poly1305::aead;
|
use chacha20poly1305::aead;
|
||||||
use std::io;
|
use std::io;
|
||||||
@@ -12,6 +12,7 @@ pub enum FcryError {
|
|||||||
Format(String),
|
Format(String),
|
||||||
Kdf(String),
|
Kdf(String),
|
||||||
Passphrase(String),
|
Passphrase(String),
|
||||||
|
WrongKey,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<io::Error> for FcryError {
|
impl From<io::Error> for FcryError {
|
||||||
|
|||||||
+92
-11
@@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: MIT-0
|
||||||
|
|
||||||
//! On-disk file format for fcry.
|
//! On-disk file format for fcry.
|
||||||
//!
|
//!
|
||||||
@@ -14,6 +14,7 @@
|
|||||||
//! kdf_params variable (depends on kdf_id)
|
//! kdf_params variable (depends on kdf_id)
|
||||||
//! nonce_prefix [u8; 19] 19 (STREAM nonce prefix)
|
//! nonce_prefix [u8; 19] 19 (STREAM nonce prefix)
|
||||||
//! plaintext_length u64 LE 8 (only if version >= 2 and flags & 0x01)
|
//! plaintext_length u64 LE 8 (only if version >= 2 and flags & 0x01)
|
||||||
|
//! key_commitment [u8; 32] 32 (only if version >= 3 and flags & 0x02)
|
||||||
//! --- end of header ---
|
//! --- end of header ---
|
||||||
//! chunk[0..N] each chunk_size + 16 bytes,
|
//! chunk[0..N] each chunk_size + 16 bytes,
|
||||||
//! last may be shorter
|
//! last may be shorter
|
||||||
@@ -28,13 +29,16 @@
|
|||||||
//! * v2 — adds `FLAG_LENGTH_COMMITTED` (bit 0); when set, the total plaintext
|
//! * v2 — adds `FLAG_LENGTH_COMMITTED` (bit 0); when set, the total plaintext
|
||||||
//! length is appended after `nonce_prefix`. This enables random-access
|
//! length is appended after `nonce_prefix`. This enables random-access
|
||||||
//! decryption without scanning predecessors.
|
//! decryption without scanning predecessors.
|
||||||
|
//! * v3 — adds `FLAG_KEY_COMMITTED` (bit 1) and an authenticated key
|
||||||
|
//! commitment for fast wrong-key detection before chunk processing.
|
||||||
|
|
||||||
use std::io::Read;
|
use std::io::Read;
|
||||||
|
|
||||||
use crate::error::FcryError;
|
use crate::error::FcryError;
|
||||||
|
use crate::policy;
|
||||||
|
|
||||||
const MAGIC: [u8; 4] = *b"fcry";
|
const MAGIC: [u8; 4] = *b"fcry";
|
||||||
pub const VERSION_CURRENT: u8 = 2;
|
pub const VERSION_CURRENT: u8 = 3;
|
||||||
const VERSION_MIN: u8 = 1;
|
const VERSION_MIN: u8 = 1;
|
||||||
|
|
||||||
pub const NONCE_PREFIX_LEN: usize = 19;
|
pub const NONCE_PREFIX_LEN: usize = 19;
|
||||||
@@ -43,9 +47,11 @@ pub const TAG_LEN: usize = 16;
|
|||||||
/// Set in `flags` when the header carries an authenticated `plaintext_length`
|
/// Set in `flags` when the header carries an authenticated `plaintext_length`
|
||||||
/// field. Required for random-access decryption.
|
/// field. Required for random-access decryption.
|
||||||
pub const FLAG_LENGTH_COMMITTED: u8 = 0x01;
|
pub const FLAG_LENGTH_COMMITTED: u8 = 0x01;
|
||||||
|
pub const FLAG_KEY_COMMITTED: u8 = 0x02;
|
||||||
|
|
||||||
/// Mask of all flag bits this build understands. Unknown bits → reject.
|
/// Mask of all flag bits this build understands. Unknown bits → reject.
|
||||||
const FLAG_KNOWN_MASK: u8 = FLAG_LENGTH_COMMITTED;
|
const FLAG_KNOWN_MASK: u8 = FLAG_LENGTH_COMMITTED | FLAG_KEY_COMMITTED;
|
||||||
|
pub const KEY_COMMITMENT_LEN: usize = 32;
|
||||||
|
|
||||||
#[repr(u8)]
|
#[repr(u8)]
|
||||||
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
||||||
@@ -142,11 +148,13 @@ pub struct Header {
|
|||||||
pub nonce_prefix: [u8; NONCE_PREFIX_LEN],
|
pub nonce_prefix: [u8; NONCE_PREFIX_LEN],
|
||||||
/// Total plaintext byte count. `Some` iff `flags & FLAG_LENGTH_COMMITTED`.
|
/// Total plaintext byte count. `Some` iff `flags & FLAG_LENGTH_COMMITTED`.
|
||||||
pub plaintext_length: Option<u64>,
|
pub plaintext_length: Option<u64>,
|
||||||
|
/// v3 key commitment. `Some` iff `flags & FLAG_KEY_COMMITTED`.
|
||||||
|
pub key_commitment: Option<[u8; KEY_COMMITMENT_LEN]>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Header {
|
impl Header {
|
||||||
pub fn encode(&self) -> Vec<u8> {
|
fn encode_without_commitment(&self) -> Vec<u8> {
|
||||||
let mut out = Vec::with_capacity(72);
|
let mut out = Vec::with_capacity(104);
|
||||||
out.extend_from_slice(&MAGIC);
|
out.extend_from_slice(&MAGIC);
|
||||||
out.push(self.version);
|
out.push(self.version);
|
||||||
out.push(self.alg as u8);
|
out.push(self.alg as u8);
|
||||||
@@ -165,7 +173,30 @@ impl Header {
|
|||||||
out
|
out
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn encode(&self) -> Vec<u8> {
|
||||||
|
let mut out = self.encode_without_commitment();
|
||||||
|
if (self.flags & FLAG_KEY_COMMITTED) != 0 {
|
||||||
|
let commitment = self
|
||||||
|
.key_commitment
|
||||||
|
.expect("FLAG_KEY_COMMITTED set but key_commitment is None");
|
||||||
|
out.extend_from_slice(&commitment);
|
||||||
|
}
|
||||||
|
out
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn commitment_input_encoding(&self) -> Vec<u8> {
|
||||||
|
self.encode_without_commitment()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
pub fn read(r: &mut impl Read) -> Result<Self, FcryError> {
|
pub fn read(r: &mut impl Read) -> Result<Self, FcryError> {
|
||||||
|
Self::read_with_argon_cap(r, policy::default_argon_decrypt_cap_mib())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn read_with_argon_cap(
|
||||||
|
r: &mut impl Read,
|
||||||
|
max_argon_memory_mib: u32,
|
||||||
|
) -> Result<Self, FcryError> {
|
||||||
let mut magic = [0u8; 4];
|
let mut magic = [0u8; 4];
|
||||||
r.read_exact(&mut magic)?;
|
r.read_exact(&mut magic)?;
|
||||||
if magic != MAGIC {
|
if magic != MAGIC {
|
||||||
@@ -189,18 +220,25 @@ impl Header {
|
|||||||
if version < 2 && flags != 0 {
|
if version < 2 && flags != 0 {
|
||||||
return Err(FcryError::Format("v1 header must have flags == 0".into()));
|
return Err(FcryError::Format("v1 header must have flags == 0".into()));
|
||||||
}
|
}
|
||||||
|
if version < 3 && (flags & FLAG_KEY_COMMITTED) != 0 {
|
||||||
|
return Err(FcryError::Format(
|
||||||
|
"key commitment flag requires v3 header".into(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
if version >= 3 && (flags & FLAG_KEY_COMMITTED) == 0 {
|
||||||
|
return Err(FcryError::Format("v3 header must commit the key".into()));
|
||||||
|
}
|
||||||
let alg = AlgId::from_u8(alg_id)?;
|
let alg = AlgId::from_u8(alg_id)?;
|
||||||
|
|
||||||
let mut chunk_size_bytes = [0u8; 4];
|
let mut chunk_size_bytes = [0u8; 4];
|
||||||
r.read_exact(&mut chunk_size_bytes)?;
|
r.read_exact(&mut chunk_size_bytes)?;
|
||||||
let chunk_size = u32::from_le_bytes(chunk_size_bytes);
|
let chunk_size = u32::from_le_bytes(chunk_size_bytes);
|
||||||
if chunk_size == 0 {
|
policy::validate_chunk_size(chunk_size)?;
|
||||||
return Err(FcryError::Format("chunk_size must be > 0".into()));
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut kdf_id = [0u8; 1];
|
let mut kdf_id = [0u8; 1];
|
||||||
r.read_exact(&mut kdf_id)?;
|
r.read_exact(&mut kdf_id)?;
|
||||||
let kdf = KdfParams::read_from(kdf_id[0], r)?;
|
let kdf = KdfParams::read_from(kdf_id[0], r)?;
|
||||||
|
policy::validate_header_kdf(&kdf, max_argon_memory_mib)?;
|
||||||
|
|
||||||
let mut nonce_prefix = [0u8; NONCE_PREFIX_LEN];
|
let mut nonce_prefix = [0u8; NONCE_PREFIX_LEN];
|
||||||
r.read_exact(&mut nonce_prefix)?;
|
r.read_exact(&mut nonce_prefix)?;
|
||||||
@@ -213,6 +251,14 @@ impl Header {
|
|||||||
None
|
None
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let key_commitment = if (flags & FLAG_KEY_COMMITTED) != 0 {
|
||||||
|
let mut b = [0u8; KEY_COMMITMENT_LEN];
|
||||||
|
r.read_exact(&mut b)?;
|
||||||
|
Some(b)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
version,
|
version,
|
||||||
alg,
|
alg,
|
||||||
@@ -221,6 +267,7 @@ impl Header {
|
|||||||
kdf,
|
kdf,
|
||||||
nonce_prefix,
|
nonce_prefix,
|
||||||
plaintext_length,
|
plaintext_length,
|
||||||
|
key_commitment,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -235,11 +282,12 @@ mod tests {
|
|||||||
let h = Header {
|
let h = Header {
|
||||||
version: VERSION_CURRENT,
|
version: VERSION_CURRENT,
|
||||||
alg: AlgId::XChaCha20Poly1305,
|
alg: AlgId::XChaCha20Poly1305,
|
||||||
flags: 0,
|
flags: FLAG_KEY_COMMITTED,
|
||||||
chunk_size: 1024 * 1024,
|
chunk_size: 1024 * 1024,
|
||||||
kdf: KdfParams::Raw,
|
kdf: KdfParams::Raw,
|
||||||
nonce_prefix: [7u8; NONCE_PREFIX_LEN],
|
nonce_prefix: [7u8; NONCE_PREFIX_LEN],
|
||||||
plaintext_length: None,
|
plaintext_length: None,
|
||||||
|
key_commitment: Some([1u8; KEY_COMMITMENT_LEN]),
|
||||||
};
|
};
|
||||||
let bytes = h.encode();
|
let bytes = h.encode();
|
||||||
let mut cur = Cursor::new(&bytes);
|
let mut cur = Cursor::new(&bytes);
|
||||||
@@ -250,6 +298,7 @@ mod tests {
|
|||||||
assert_eq!(parsed.chunk_size, h.chunk_size);
|
assert_eq!(parsed.chunk_size, h.chunk_size);
|
||||||
assert_eq!(parsed.nonce_prefix, h.nonce_prefix);
|
assert_eq!(parsed.nonce_prefix, h.nonce_prefix);
|
||||||
assert_eq!(parsed.plaintext_length, None);
|
assert_eq!(parsed.plaintext_length, None);
|
||||||
|
assert_eq!(parsed.key_commitment, h.key_commitment);
|
||||||
assert_eq!(cur.position() as usize, bytes.len());
|
assert_eq!(cur.position() as usize, bytes.len());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -258,20 +307,49 @@ mod tests {
|
|||||||
let h = Header {
|
let h = Header {
|
||||||
version: VERSION_CURRENT,
|
version: VERSION_CURRENT,
|
||||||
alg: AlgId::XChaCha20Poly1305,
|
alg: AlgId::XChaCha20Poly1305,
|
||||||
flags: FLAG_LENGTH_COMMITTED,
|
flags: FLAG_LENGTH_COMMITTED | FLAG_KEY_COMMITTED,
|
||||||
chunk_size: 65536,
|
chunk_size: 65536,
|
||||||
kdf: KdfParams::Raw,
|
kdf: KdfParams::Raw,
|
||||||
nonce_prefix: [9u8; NONCE_PREFIX_LEN],
|
nonce_prefix: [9u8; NONCE_PREFIX_LEN],
|
||||||
plaintext_length: Some(123_456_789),
|
plaintext_length: Some(123_456_789),
|
||||||
|
key_commitment: Some([2u8; KEY_COMMITMENT_LEN]),
|
||||||
};
|
};
|
||||||
let bytes = h.encode();
|
let bytes = h.encode();
|
||||||
let mut cur = Cursor::new(&bytes);
|
let mut cur = Cursor::new(&bytes);
|
||||||
let parsed = Header::read(&mut cur).unwrap();
|
let parsed = Header::read(&mut cur).unwrap();
|
||||||
assert_eq!(parsed.flags, FLAG_LENGTH_COMMITTED);
|
assert_eq!(parsed.flags, FLAG_LENGTH_COMMITTED | FLAG_KEY_COMMITTED);
|
||||||
assert_eq!(parsed.plaintext_length, Some(123_456_789));
|
assert_eq!(parsed.plaintext_length, Some(123_456_789));
|
||||||
|
assert_eq!(parsed.key_commitment, h.key_commitment);
|
||||||
assert_eq!(cur.position() as usize, bytes.len());
|
assert_eq!(cur.position() as usize, bytes.len());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn v3_encoding_layout_stable() {
|
||||||
|
let h = Header {
|
||||||
|
version: VERSION_CURRENT,
|
||||||
|
alg: AlgId::XChaCha20Poly1305,
|
||||||
|
flags: FLAG_LENGTH_COMMITTED | FLAG_KEY_COMMITTED,
|
||||||
|
chunk_size: 0x0102_0304,
|
||||||
|
kdf: KdfParams::Raw,
|
||||||
|
nonce_prefix: [0x55u8; NONCE_PREFIX_LEN],
|
||||||
|
plaintext_length: Some(0x0807_0605_0403_0201),
|
||||||
|
key_commitment: Some([0xaau8; KEY_COMMITMENT_LEN]),
|
||||||
|
};
|
||||||
|
let commitment_input = h.commitment_input_encoding();
|
||||||
|
assert_eq!(commitment_input.len(), 40);
|
||||||
|
assert_eq!(&commitment_input[..4], b"fcry");
|
||||||
|
assert_eq!(commitment_input[4], 3);
|
||||||
|
assert_eq!(
|
||||||
|
&commitment_input[32..40],
|
||||||
|
&0x0807_0605_0403_0201u64.to_le_bytes()
|
||||||
|
);
|
||||||
|
|
||||||
|
let aad = h.encode();
|
||||||
|
assert_eq!(aad.len(), 72);
|
||||||
|
assert_eq!(&aad[..40], &commitment_input);
|
||||||
|
assert_eq!(&aad[40..], &[0xaau8; KEY_COMMITMENT_LEN]);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn rejects_bad_magic() {
|
fn rejects_bad_magic() {
|
||||||
let mut bytes = Header {
|
let mut bytes = Header {
|
||||||
@@ -282,6 +360,7 @@ mod tests {
|
|||||||
kdf: KdfParams::Raw,
|
kdf: KdfParams::Raw,
|
||||||
nonce_prefix: [0u8; NONCE_PREFIX_LEN],
|
nonce_prefix: [0u8; NONCE_PREFIX_LEN],
|
||||||
plaintext_length: None,
|
plaintext_length: None,
|
||||||
|
key_commitment: Some([3u8; KEY_COMMITMENT_LEN]),
|
||||||
}
|
}
|
||||||
.encode();
|
.encode();
|
||||||
bytes[0] ^= 1;
|
bytes[0] ^= 1;
|
||||||
@@ -301,6 +380,7 @@ mod tests {
|
|||||||
kdf: KdfParams::Raw,
|
kdf: KdfParams::Raw,
|
||||||
nonce_prefix: [0u8; NONCE_PREFIX_LEN],
|
nonce_prefix: [0u8; NONCE_PREFIX_LEN],
|
||||||
plaintext_length: None,
|
plaintext_length: None,
|
||||||
|
key_commitment: Some([4u8; KEY_COMMITMENT_LEN]),
|
||||||
}
|
}
|
||||||
.encode();
|
.encode();
|
||||||
// flags byte is at offset 6 (4 magic + version + alg)
|
// flags byte is at offset 6 (4 magic + version + alg)
|
||||||
@@ -328,6 +408,7 @@ mod tests {
|
|||||||
assert_eq!(parsed.flags, 0);
|
assert_eq!(parsed.flags, 0);
|
||||||
assert_eq!(parsed.chunk_size, 1024);
|
assert_eq!(parsed.chunk_size, 1024);
|
||||||
assert_eq!(parsed.plaintext_length, None);
|
assert_eq!(parsed.plaintext_length, None);
|
||||||
|
assert_eq!(parsed.key_commitment, None);
|
||||||
// Re-encoding must reproduce the original v1 bytes exactly so the
|
// Re-encoding must reproduce the original v1 bytes exactly so the
|
||||||
// recomputed AAD matches what the file was authenticated with.
|
// recomputed AAD matches what the file was authenticated with.
|
||||||
assert_eq!(parsed.encode(), bytes);
|
assert_eq!(parsed.encode(), bytes);
|
||||||
|
|||||||
+184
-44
@@ -1,9 +1,10 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: MIT-0
|
||||||
|
|
||||||
mod crypto;
|
mod crypto;
|
||||||
mod error;
|
mod error;
|
||||||
mod header;
|
mod header;
|
||||||
mod pipeline;
|
mod pipeline;
|
||||||
|
mod policy;
|
||||||
mod reader;
|
mod reader;
|
||||||
mod secrets;
|
mod secrets;
|
||||||
mod utils;
|
mod utils;
|
||||||
@@ -12,9 +13,13 @@ use crypto::*;
|
|||||||
use error::FcryError;
|
use error::FcryError;
|
||||||
use header::{ARGON2_SALT_LEN, KdfParams};
|
use header::{ARGON2_SALT_LEN, KdfParams};
|
||||||
use secrets::{SecretBytes32, SecretVec, read_passphrase_tty};
|
use secrets::{SecretBytes32, SecretVec, read_passphrase_tty};
|
||||||
use utils::DEFAULT_CHUNK_SIZE;
|
use utils::{DEFAULT_CHUNK_SIZE, OutSinkOptions};
|
||||||
|
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
|
use std::fs::File;
|
||||||
|
use std::io::Read;
|
||||||
|
use std::path::{Path, PathBuf};
|
||||||
|
use unicode_normalization::UnicodeNormalization;
|
||||||
use zeroize::Zeroizing;
|
use zeroize::Zeroizing;
|
||||||
|
|
||||||
/// fcry - [f]ile[cry]pt: A file en-/decryption tool for easy use
|
/// fcry - [f]ile[cry]pt: A file en-/decryption tool for easy use
|
||||||
@@ -34,13 +39,11 @@ struct Cli {
|
|||||||
#[clap(short, long)]
|
#[clap(short, long)]
|
||||||
output_file: Option<String>,
|
output_file: Option<String>,
|
||||||
|
|
||||||
/// The raw bytes of the crypto key. Has to be exactly 32 bytes.
|
/// Read the raw 32-byte crypto key from a file.
|
||||||
/// *** DANGEROUS: visible in process listings (ps/proc). Testing only. ***
|
#[clap(short = 'k', long, conflicts_with_all = ["passphrase", "passphrase_env"])]
|
||||||
#[clap(short, long, conflicts_with_all = ["passphrase", "passphrase_env"])]
|
key_file: Option<PathBuf>,
|
||||||
raw_key: Option<Zeroizing<String>>,
|
|
||||||
|
|
||||||
/// Read passphrase interactively (terminal). Implies argon2id KDF on encrypt.
|
/// Read passphrase interactively (terminal). Implies argon2id KDF on encrypt.
|
||||||
/// This is the default when no key source is specified.
|
|
||||||
#[clap(short, long)]
|
#[clap(short, long)]
|
||||||
passphrase: bool,
|
passphrase: bool,
|
||||||
|
|
||||||
@@ -54,22 +57,43 @@ struct Cli {
|
|||||||
chunk_size: u32,
|
chunk_size: u32,
|
||||||
|
|
||||||
/// Argon2id memory in MiB (encryption only). Default: 1024 (= 1 GiB).
|
/// Argon2id memory in MiB (encryption only). Default: 1024 (= 1 GiB).
|
||||||
#[clap(long, default_value_t = 1024)]
|
#[clap(long, default_value_t = policy::DEFAULT_ARGON_MEMORY_MIB)]
|
||||||
argon_memory: u32,
|
argon_memory: u32,
|
||||||
|
|
||||||
/// Argon2id passes / iterations (encryption only).
|
/// Argon2id passes / iterations (encryption only).
|
||||||
#[clap(long, default_value_t = 2)]
|
#[clap(long, default_value_t = policy::MIN_ARGON_PASSES)]
|
||||||
argon_passes: u32,
|
argon_passes: u32,
|
||||||
|
|
||||||
/// Argon2id parallelism / lanes (encryption only).
|
/// Argon2id parallelism / lanes (encryption only).
|
||||||
#[clap(long, default_value_t = 4)]
|
#[clap(long, default_value_t = policy::DEFAULT_ARGON_PARALLELISM)]
|
||||||
argon_parallelism: u32,
|
argon_parallelism: u32,
|
||||||
|
|
||||||
|
/// Permit intentionally weak passphrase/KDF parameters for tests or legacy interop.
|
||||||
|
#[clap(long)]
|
||||||
|
allow_weak_kdf: bool,
|
||||||
|
|
||||||
|
/// Maximum Argon2id memory accepted while decrypting, in MiB.
|
||||||
|
/// Overrides the dynamic default. Raising it can OOM constrained machines.
|
||||||
|
#[clap(long)]
|
||||||
|
max_argon_memory_mib: Option<u32>,
|
||||||
|
|
||||||
/// Number of worker threads for AEAD work. Defaults to the number of
|
/// Number of worker threads for AEAD work. Defaults to the number of
|
||||||
/// available CPUs. Set to 1 for fully serial encrypt/decrypt.
|
/// available CPUs. Set to 1 for fully serial encrypt/decrypt.
|
||||||
#[clap(short = 'j', long, value_parser = clap::value_parser!(u32).range(1..))]
|
#[clap(short = 'j', long, value_parser = clap::value_parser!(u32).range(1..))]
|
||||||
threads: Option<u32>,
|
threads: Option<u32>,
|
||||||
|
|
||||||
|
/// Replace an existing different output file after encryption/decryption succeeds.
|
||||||
|
#[clap(long)]
|
||||||
|
force: bool,
|
||||||
|
|
||||||
|
/// Directory for private temporary files.
|
||||||
|
#[clap(long)]
|
||||||
|
temp_dir: Option<PathBuf>,
|
||||||
|
|
||||||
|
/// For decrypt-to-stdout, verify the whole plaintext in a private temp file before emitting it.
|
||||||
|
#[clap(long, requires = "decrypt")]
|
||||||
|
buffer_verify: bool,
|
||||||
|
|
||||||
/// Random-access decrypt: byte offset of the slice to read.
|
/// Random-access decrypt: byte offset of the slice to read.
|
||||||
/// Requires `--decrypt`, an `--input-file` whose header has the
|
/// Requires `--decrypt`, an `--input-file` whose header has the
|
||||||
/// length-committed flag set, and `--length`.
|
/// length-committed flag set, and `--length`.
|
||||||
@@ -92,25 +116,76 @@ struct Cli {
|
|||||||
length: Option<u64>,
|
length: Option<u64>,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_raw_key(s: &str) -> Result<SecretBytes32, FcryError> {
|
fn read_key_file(path: &Path) -> Result<SecretBytes32, FcryError> {
|
||||||
let raw = s.as_bytes();
|
warn_if_key_file_world_readable(path);
|
||||||
if raw.len() != 32 {
|
let mut file = File::open(path)?;
|
||||||
|
let mut buf = Zeroizing::new([0u8; 33]);
|
||||||
|
let mut n = 0usize;
|
||||||
|
while n < buf.len() {
|
||||||
|
match file.read(&mut buf[n..]) {
|
||||||
|
Ok(0) => break,
|
||||||
|
Ok(read) => n += read,
|
||||||
|
Err(e) if e.kind() == std::io::ErrorKind::Interrupted => continue,
|
||||||
|
Err(e) => return Err(e.into()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if n < 32 {
|
||||||
return Err(FcryError::Format(format!(
|
return Err(FcryError::Format(format!(
|
||||||
"raw_key must be exactly 32 bytes, got {}",
|
"key file {} is too short: expected exactly 32 bytes, got {n}",
|
||||||
raw.len()
|
path.display()
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
if n > 32 {
|
||||||
|
return Err(FcryError::Format(format!(
|
||||||
|
"key file {} is too long: expected exactly 32 bytes; possible trailing newline",
|
||||||
|
path.display()
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
let mut extra = Zeroizing::new([0u8; 1]);
|
||||||
|
if file.read(&mut *extra)? != 0 {
|
||||||
|
return Err(FcryError::Format(format!(
|
||||||
|
"key file {} is too long: expected exactly 32 bytes; possible trailing newline",
|
||||||
|
path.display()
|
||||||
)));
|
)));
|
||||||
}
|
}
|
||||||
let mut key = SecretBytes32::zeroed();
|
let mut key = SecretBytes32::zeroed();
|
||||||
key.with_mut_array(|key| key.copy_from_slice(raw));
|
key.with_mut_array(|key| key.copy_from_slice(&buf[..32]));
|
||||||
Ok(key)
|
Ok(key)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(unix)]
|
||||||
|
fn warn_if_key_file_world_readable(path: &Path) {
|
||||||
|
use std::os::unix::fs::PermissionsExt;
|
||||||
|
if let Ok(meta) = std::fs::metadata(path) {
|
||||||
|
let mode = meta.permissions().mode();
|
||||||
|
if (mode & 0o077) != 0 {
|
||||||
|
eprintln!(
|
||||||
|
"Warning: key file {} is group/world accessible; consider chmod 600",
|
||||||
|
path.display()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(not(unix))]
|
||||||
|
fn warn_if_key_file_world_readable(_path: &Path) {}
|
||||||
|
|
||||||
/// Source of a passphrase: either the terminal or a named env var.
|
/// Source of a passphrase: either the terminal or a named env var.
|
||||||
enum PassphraseSource {
|
enum PassphraseSource {
|
||||||
Tty,
|
Tty,
|
||||||
EnvVar(String),
|
EnvVar(String),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn normalize_passphrase(pw: SecretVec) -> Result<SecretVec, FcryError> {
|
||||||
|
let normalized = pw.with_slice(|bytes| {
|
||||||
|
let s = std::str::from_utf8(bytes).map_err(|_| {
|
||||||
|
FcryError::Passphrase("passphrase must be valid UTF-8 after normalization".into())
|
||||||
|
})?;
|
||||||
|
Ok::<Zeroizing<String>, FcryError>(Zeroizing::new(s.nfc().collect::<String>()))
|
||||||
|
})?;
|
||||||
|
Ok(SecretVec::from_vec(normalized.as_bytes().to_vec()))
|
||||||
|
}
|
||||||
|
|
||||||
fn read_passphrase(src: &PassphraseSource, confirm: bool) -> Result<SecretVec, FcryError> {
|
fn read_passphrase(src: &PassphraseSource, confirm: bool) -> Result<SecretVec, FcryError> {
|
||||||
match src {
|
match src {
|
||||||
PassphraseSource::EnvVar(var) => {
|
PassphraseSource::EnvVar(var) => {
|
||||||
@@ -118,17 +193,22 @@ fn read_passphrase(src: &PassphraseSource, confirm: bool) -> Result<SecretVec, F
|
|||||||
// protected storage. The source Vec is zeroed after the copy.
|
// protected storage. The source Vec is zeroed after the copy.
|
||||||
// Note: a copy still exists in the process `environ` table; that is
|
// Note: a copy still exists in the process `environ` table; that is
|
||||||
// a known and accepted leak for the env-var path.
|
// a known and accepted leak for the env-var path.
|
||||||
let v = std::env::var(var).map_err(|_| {
|
let v = Zeroizing::new(std::env::var(var).map_err(|_| {
|
||||||
FcryError::Passphrase(format!("environment variable {var} not set or not unicode"))
|
FcryError::Passphrase(format!("environment variable {var} not set or not unicode"))
|
||||||
})?;
|
})?);
|
||||||
Ok(SecretVec::from_vec(v.into_bytes()))
|
let normalized = Zeroizing::new(v.as_str().nfc().collect::<String>());
|
||||||
|
Ok(SecretVec::from_vec(normalized.as_bytes().to_vec()))
|
||||||
}
|
}
|
||||||
PassphraseSource::Tty => {
|
PassphraseSource::Tty => {
|
||||||
let pw = read_passphrase_tty("Passphrase: ")
|
let pw = normalize_passphrase(
|
||||||
.map_err(|e| FcryError::Passphrase(e.to_string()))?;
|
read_passphrase_tty("Passphrase: ")
|
||||||
|
.map_err(|e| FcryError::Passphrase(e.to_string()))?,
|
||||||
|
)?;
|
||||||
if confirm {
|
if confirm {
|
||||||
let pw2 = read_passphrase_tty("Confirm passphrase: ")
|
let pw2 = normalize_passphrase(
|
||||||
.map_err(|e| FcryError::Passphrase(e.to_string()))?;
|
read_passphrase_tty("Confirm passphrase: ")
|
||||||
|
.map_err(|e| FcryError::Passphrase(e.to_string()))?,
|
||||||
|
)?;
|
||||||
if pw != pw2 {
|
if pw != pw2 {
|
||||||
return Err(FcryError::Passphrase("passphrases do not match".into()));
|
return Err(FcryError::Passphrase("passphrases do not match".into()));
|
||||||
}
|
}
|
||||||
@@ -156,16 +236,11 @@ fn disable_core_dumps() {
|
|||||||
fn run(mut cli: Cli) -> Result<(), FcryError> {
|
fn run(mut cli: Cli) -> Result<(), FcryError> {
|
||||||
// Move the secret-bearing fields out of `Cli` immediately so they don't
|
// Move the secret-bearing fields out of `Cli` immediately so they don't
|
||||||
// sit in the parsed struct for the rest of the function.
|
// sit in the parsed struct for the rest of the function.
|
||||||
let raw_key_str: Option<Zeroizing<String>> = cli.raw_key.take();
|
let key_file: Option<PathBuf> = cli.key_file.take();
|
||||||
let pw_src: Option<PassphraseSource> = if cli.passphrase {
|
let pw_src: Option<PassphraseSource> = if cli.passphrase {
|
||||||
Some(PassphraseSource::Tty)
|
Some(PassphraseSource::Tty)
|
||||||
} else if let Some(var) = cli.passphrase_env.take() {
|
|
||||||
Some(PassphraseSource::EnvVar(var))
|
|
||||||
} else if raw_key_str.is_none() {
|
|
||||||
// Default to interactive TTY passphrase when no key source is given.
|
|
||||||
Some(PassphraseSource::Tty)
|
|
||||||
} else {
|
} else {
|
||||||
None
|
cli.passphrase_env.take().map(PassphraseSource::EnvVar)
|
||||||
};
|
};
|
||||||
|
|
||||||
let decrypt_mode = cli.decrypt;
|
let decrypt_mode = cli.decrypt;
|
||||||
@@ -175,18 +250,54 @@ fn run(mut cli: Cli) -> Result<(), FcryError> {
|
|||||||
let argon_memory = cli.argon_memory;
|
let argon_memory = cli.argon_memory;
|
||||||
let argon_passes = cli.argon_passes;
|
let argon_passes = cli.argon_passes;
|
||||||
let argon_parallelism = cli.argon_parallelism;
|
let argon_parallelism = cli.argon_parallelism;
|
||||||
let threads = cli.threads.map(|n| n as usize).unwrap_or_else(|| {
|
let allow_weak_kdf = cli.allow_weak_kdf;
|
||||||
std::thread::available_parallelism()
|
let argon_cap = policy::resolve_argon_decrypt_cap(cli.max_argon_memory_mib)?;
|
||||||
.map(|n| n.get())
|
if argon_cap.overridden && argon_cap.effective_mib > argon_cap.default_mib {
|
||||||
.unwrap_or(1)
|
eprintln!(
|
||||||
});
|
"Warning: --max-argon-memory-mib raises the Argon2 decrypt trust ceiling from {} MiB to {} MiB; this can OOM constrained machines",
|
||||||
|
argon_cap.default_mib, argon_cap.effective_mib
|
||||||
|
);
|
||||||
|
}
|
||||||
|
let (threads, thread_warning) = policy::normalize_worker_threads(cli.threads);
|
||||||
|
if let Some(requested) = thread_warning {
|
||||||
|
eprintln!(
|
||||||
|
"Warning: requested {requested} worker threads; capped at {}",
|
||||||
|
policy::MAX_WORKER_THREADS
|
||||||
|
);
|
||||||
|
}
|
||||||
|
let force = cli.force;
|
||||||
|
let temp_dir = cli.temp_dir.take();
|
||||||
|
let buffer_verify = cli.buffer_verify;
|
||||||
let offset = cli.offset;
|
let offset = cli.offset;
|
||||||
let length = cli.length;
|
let length = cli.length;
|
||||||
drop(cli);
|
drop(cli);
|
||||||
|
|
||||||
|
if pw_src.is_none() && key_file.is_none() {
|
||||||
|
return Err(FcryError::Format(
|
||||||
|
"must provide one of --key-file, --passphrase, --passphrase-env".into(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
if buffer_verify && !decrypt_mode {
|
||||||
|
return Err(FcryError::Format(
|
||||||
|
"--buffer-verify is only valid for decrypt".into(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
if buffer_verify && output.is_some() {
|
||||||
|
return Err(FcryError::Format(
|
||||||
|
"--buffer-verify is only meaningful when decrypting to stdout".into(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
let output_options = OutSinkOptions {
|
||||||
|
force,
|
||||||
|
input_file: input.as_ref().map(PathBuf::from),
|
||||||
|
temp_dir,
|
||||||
|
buffer_verify_stdout: buffer_verify,
|
||||||
|
};
|
||||||
|
|
||||||
if decrypt_mode {
|
if decrypt_mode {
|
||||||
let raw_key = match raw_key_str.as_deref() {
|
let raw_key = match key_file.as_deref() {
|
||||||
Some(s) => Some(parse_raw_key(s)?),
|
Some(path) => Some(read_key_file(path)?),
|
||||||
None => None,
|
None => None,
|
||||||
};
|
};
|
||||||
let pw = match &pw_src {
|
let pw = match &pw_src {
|
||||||
@@ -202,10 +313,27 @@ fn run(mut cli: Cli) -> Result<(), FcryError> {
|
|||||||
"--offset/--length require --input-file (random-access needs a seekable file)".into(),
|
"--offset/--length require --input-file (random-access needs a seekable file)".into(),
|
||||||
)
|
)
|
||||||
})?;
|
})?;
|
||||||
decrypt_range(path, output, raw_key.as_ref(), pw.as_ref(), o, l)?;
|
decrypt_range_with_output_options(
|
||||||
|
path,
|
||||||
|
output,
|
||||||
|
raw_key.as_ref(),
|
||||||
|
pw.as_ref(),
|
||||||
|
o,
|
||||||
|
l,
|
||||||
|
argon_cap.effective_mib,
|
||||||
|
&output_options,
|
||||||
|
)?;
|
||||||
}
|
}
|
||||||
(None, None) => {
|
(None, None) => {
|
||||||
decrypt(input, output, raw_key.as_ref(), pw.as_ref(), threads)?;
|
decrypt_with_output_options(
|
||||||
|
input,
|
||||||
|
output,
|
||||||
|
raw_key.as_ref(),
|
||||||
|
pw.as_ref(),
|
||||||
|
threads,
|
||||||
|
argon_cap.effective_mib,
|
||||||
|
&output_options,
|
||||||
|
)?;
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
return Err(FcryError::Format(
|
return Err(FcryError::Format(
|
||||||
@@ -217,9 +345,12 @@ fn run(mut cli: Cli) -> Result<(), FcryError> {
|
|||||||
let (key, kdf) = if let Some(src) = &pw_src {
|
let (key, kdf) = if let Some(src) = &pw_src {
|
||||||
let mut salt = [0u8; ARGON2_SALT_LEN];
|
let mut salt = [0u8; ARGON2_SALT_LEN];
|
||||||
getrandom::fill(&mut salt)?;
|
getrandom::fill(&mut salt)?;
|
||||||
let m_cost_kib = argon_memory.checked_mul(1024).ok_or_else(|| {
|
let m_cost_kib = policy::validate_new_argon_params(
|
||||||
FcryError::Format("argon-memory too large (overflow when converting to KiB)".into())
|
argon_memory,
|
||||||
})?;
|
argon_passes,
|
||||||
|
argon_parallelism,
|
||||||
|
allow_weak_kdf,
|
||||||
|
)?;
|
||||||
let kdf = KdfParams::Argon2id {
|
let kdf = KdfParams::Argon2id {
|
||||||
salt,
|
salt,
|
||||||
m_cost: m_cost_kib,
|
m_cost: m_cost_kib,
|
||||||
@@ -227,13 +358,22 @@ fn run(mut cli: Cli) -> Result<(), FcryError> {
|
|||||||
p_cost: argon_parallelism,
|
p_cost: argon_parallelism,
|
||||||
};
|
};
|
||||||
let pw = read_passphrase(src, true)?;
|
let pw = read_passphrase(src, true)?;
|
||||||
|
policy::validate_new_passphrase(&pw, allow_weak_kdf)?;
|
||||||
let key = derive_key(&kdf, None, Some(&pw))?;
|
let key = derive_key(&kdf, None, Some(&pw))?;
|
||||||
(key, kdf)
|
(key, kdf)
|
||||||
} else {
|
} else {
|
||||||
let key = parse_raw_key(raw_key_str.as_deref().unwrap())?;
|
let key = read_key_file(key_file.as_deref().unwrap())?;
|
||||||
(key, KdfParams::Raw)
|
(key, KdfParams::Raw)
|
||||||
};
|
};
|
||||||
encrypt(input, output, &key, chunk_size, kdf, threads)?;
|
encrypt_with_output_options(
|
||||||
|
input,
|
||||||
|
output,
|
||||||
|
&key,
|
||||||
|
chunk_size,
|
||||||
|
kdf,
|
||||||
|
threads,
|
||||||
|
&output_options,
|
||||||
|
)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|||||||
+18
-16
@@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: MIT-0
|
||||||
|
|
||||||
//! Multi-threaded encrypt/decrypt pipeline.
|
//! Multi-threaded encrypt/decrypt pipeline.
|
||||||
//!
|
//!
|
||||||
@@ -42,31 +42,33 @@ use crossbeam_channel::{Receiver, RecvTimeoutError, Sender, bounded};
|
|||||||
use crate::crypto::{bump_counter, make_nonce};
|
use crate::crypto::{bump_counter, make_nonce};
|
||||||
use crate::error::FcryError;
|
use crate::error::FcryError;
|
||||||
use crate::header::NONCE_PREFIX_LEN;
|
use crate::header::NONCE_PREFIX_LEN;
|
||||||
|
use crate::policy;
|
||||||
use crate::reader::{AheadReader, ReadInfoChunk};
|
use crate::reader::{AheadReader, ReadInfoChunk};
|
||||||
use crate::utils::OutSink;
|
use crate::utils::OutSink;
|
||||||
|
use zeroize::Zeroizing;
|
||||||
|
|
||||||
struct Job {
|
struct Job {
|
||||||
counter: u32,
|
counter: u32,
|
||||||
last: bool,
|
last: bool,
|
||||||
buf: Vec<u8>,
|
buf: Zeroizing<Vec<u8>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Done {
|
struct Done {
|
||||||
counter: u32,
|
counter: u32,
|
||||||
buf: Vec<u8>,
|
buf: Zeroizing<Vec<u8>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Job-channel capacity: small multiples of worker count, enough to keep
|
/// Job-channel capacity: small multiples of worker count, enough to keep
|
||||||
/// workers fed without unbounded memory.
|
/// workers fed without unbounded memory.
|
||||||
fn channel_capacity(threads: usize) -> usize {
|
fn channel_capacity(threads: usize, in_flight: usize) -> usize {
|
||||||
(threads * 2).max(2)
|
policy::pipeline_channel_capacity(threads, in_flight)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Total in-flight chunk cap (jobs queued + at workers + in writer's reorder
|
/// Total in-flight chunk cap (jobs queued + at workers + in writer's reorder
|
||||||
/// buffer). Permit count; bounded above the job-channel capacity to absorb
|
/// buffer). Permit count; bounded above the job-channel capacity to absorb
|
||||||
/// reordering without blocking workers unnecessarily.
|
/// reordering without blocking workers unnecessarily.
|
||||||
fn in_flight_capacity(threads: usize) -> usize {
|
fn in_flight_capacity(threads: usize, chunk_len: usize) -> usize {
|
||||||
(threads * 4).max(4)
|
policy::pipeline_in_flight_capacity(threads, chunk_len)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
@@ -152,8 +154,8 @@ fn run_pipeline(
|
|||||||
threads: usize,
|
threads: usize,
|
||||||
is_encrypt: bool,
|
is_encrypt: bool,
|
||||||
) -> Result<(OutSink, u64), FcryError> {
|
) -> Result<(OutSink, u64), FcryError> {
|
||||||
let cap = channel_capacity(threads);
|
let in_flight = in_flight_capacity(threads, chunk_sz);
|
||||||
let in_flight = in_flight_capacity(threads);
|
let cap = channel_capacity(threads, in_flight);
|
||||||
let (jobs_tx, jobs_rx) = bounded::<Job>(cap);
|
let (jobs_tx, jobs_rx) = bounded::<Job>(cap);
|
||||||
let (done_tx, done_rx) = bounded::<Done>(cap);
|
let (done_tx, done_rx) = bounded::<Done>(cap);
|
||||||
|
|
||||||
@@ -193,7 +195,7 @@ fn run_pipeline(
|
|||||||
Err(RecvTimeoutError::Disconnected) => return Ok(bytes_seen),
|
Err(RecvTimeoutError::Disconnected) => return Ok(bytes_seen),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
let mut buf = vec![0u8; chunk_sz];
|
let mut buf = Zeroizing::new(vec![0u8; chunk_sz]);
|
||||||
match input.read_ahead(&mut buf)? {
|
match input.read_ahead(&mut buf)? {
|
||||||
ReadInfoChunk::Normal(_) => {
|
ReadInfoChunk::Normal(_) => {
|
||||||
if jobs_tx
|
if jobs_tx
|
||||||
@@ -206,7 +208,7 @@ fn run_pipeline(
|
|||||||
{
|
{
|
||||||
return Ok(bytes_seen);
|
return Ok(bytes_seen);
|
||||||
}
|
}
|
||||||
bytes_seen = bytes_seen.saturating_add(chunk_sz as u64);
|
bytes_seen = policy::checked_count_add(bytes_seen, chunk_sz, "bytes read")?;
|
||||||
counter = bump_counter(counter)?;
|
counter = bump_counter(counter)?;
|
||||||
}
|
}
|
||||||
ReadInfoChunk::Last(n) => {
|
ReadInfoChunk::Last(n) => {
|
||||||
@@ -216,7 +218,7 @@ fn run_pipeline(
|
|||||||
last: true,
|
last: true,
|
||||||
buf,
|
buf,
|
||||||
});
|
});
|
||||||
bytes_seen = bytes_seen.saturating_add(n as u64);
|
bytes_seen = policy::checked_count_add(bytes_seen, n, "bytes read")?;
|
||||||
return Ok(bytes_seen);
|
return Ok(bytes_seen);
|
||||||
}
|
}
|
||||||
ReadInfoChunk::Empty => {
|
ReadInfoChunk::Empty => {
|
||||||
@@ -261,9 +263,9 @@ fn run_pipeline(
|
|||||||
}
|
}
|
||||||
let nonce = make_nonce(&nonce_prefix, job.counter, job.last);
|
let nonce = make_nonce(&nonce_prefix, job.counter, job.last);
|
||||||
let res = if is_encrypt {
|
let res = if is_encrypt {
|
||||||
aead.encrypt_in_place(&nonce, aad.as_slice(), &mut job.buf)
|
aead.encrypt_in_place(&nonce, aad.as_slice(), &mut *job.buf)
|
||||||
} else {
|
} else {
|
||||||
aead.decrypt_in_place(&nonce, aad.as_slice(), &mut job.buf)
|
aead.decrypt_in_place(&nonce, aad.as_slice(), &mut *job.buf)
|
||||||
};
|
};
|
||||||
if let Err(e) = res {
|
if let Err(e) = res {
|
||||||
cancel.store(true, Ordering::Release);
|
cancel.store(true, Ordering::Release);
|
||||||
@@ -343,13 +345,13 @@ fn ordered_writer(
|
|||||||
permit_tx: Sender<()>,
|
permit_tx: Sender<()>,
|
||||||
) -> Result<(OutSink, u64), FcryError> {
|
) -> Result<(OutSink, u64), FcryError> {
|
||||||
let mut next: u32 = 0;
|
let mut next: u32 = 0;
|
||||||
let mut pending: BTreeMap<u32, Vec<u8>> = BTreeMap::new();
|
let mut pending: BTreeMap<u32, Zeroizing<Vec<u8>>> = BTreeMap::new();
|
||||||
let mut total: u64 = 0;
|
let mut total: u64 = 0;
|
||||||
for done in done_rx.iter() {
|
for done in done_rx.iter() {
|
||||||
pending.insert(done.counter, done.buf);
|
pending.insert(done.counter, done.buf);
|
||||||
while let Some(buf) = pending.remove(&next) {
|
while let Some(buf) = pending.remove(&next) {
|
||||||
output.write_all(&buf)?;
|
output.write_all(&buf)?;
|
||||||
total = total.saturating_add(buf.len() as u64);
|
total = policy::checked_count_add(total, buf.len(), "bytes written")?;
|
||||||
// `bump_counter` rejects overflow upstream; a wrap here would be
|
// `bump_counter` rejects overflow upstream; a wrap here would be
|
||||||
// a real bug, so use plain addition and let it panic in debug.
|
// a real bug, so use plain addition and let it panic in debug.
|
||||||
next += 1;
|
next += 1;
|
||||||
|
|||||||
+299
@@ -0,0 +1,299 @@
|
|||||||
|
// SPDX-License-Identifier: MIT-0
|
||||||
|
|
||||||
|
//! Central resource and format policy.
|
||||||
|
|
||||||
|
use std::fs;
|
||||||
|
|
||||||
|
use crate::error::FcryError;
|
||||||
|
use crate::header::{KdfParams, TAG_LEN};
|
||||||
|
use crate::secrets::SecretVec;
|
||||||
|
|
||||||
|
pub const MAX_CHUNK_SIZE: u32 = 64 * 1024 * 1024;
|
||||||
|
pub const DEFAULT_CHUNK_SIZE: u32 = 1024 * 1024;
|
||||||
|
|
||||||
|
pub const DEFAULT_ARGON_MEMORY_MIB: u32 = 1024;
|
||||||
|
pub const MIN_ARGON_MEMORY_MIB: u32 = 64;
|
||||||
|
pub const DEFAULT_ARGON_DECRYPT_CAP_MIB: u32 = 4096;
|
||||||
|
pub const MIN_ARGON_PASSES: u32 = 2;
|
||||||
|
pub const MAX_ARGON_PASSES: u32 = 64;
|
||||||
|
pub const DEFAULT_ARGON_PARALLELISM: u32 = 4;
|
||||||
|
pub const MAX_ARGON_PARALLELISM: u32 = 64;
|
||||||
|
pub const MIN_PASSPHRASE_BYTES: usize = 12;
|
||||||
|
|
||||||
|
pub const MAX_WORKER_THREADS: usize = 256;
|
||||||
|
pub const PIPELINE_IN_FLIGHT_BYTES: usize = 128 * 1024 * 1024;
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
||||||
|
pub struct ArgonDecryptCap {
|
||||||
|
pub default_mib: u32,
|
||||||
|
pub effective_mib: u32,
|
||||||
|
pub overridden: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn architecture_argon_cap_mib() -> u32 {
|
||||||
|
let usize_cap_mib = usize::MAX / 1024 / 1024;
|
||||||
|
let argon_m_cost_cap_mib = (u32::MAX / 1024) as usize;
|
||||||
|
usize_cap_mib
|
||||||
|
.min(argon_m_cost_cap_mib)
|
||||||
|
.min(u32::MAX as usize) as u32
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(target_os = "linux")]
|
||||||
|
fn available_memory_mib() -> Option<u32> {
|
||||||
|
let meminfo = fs::read_to_string("/proc/meminfo").ok()?;
|
||||||
|
for line in meminfo.lines() {
|
||||||
|
let Some(rest) = line.strip_prefix("MemAvailable:") else {
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
let kib = rest.split_whitespace().next()?.parse::<u64>().ok()?;
|
||||||
|
return u32::try_from(kib / 1024).ok();
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(not(target_os = "linux"))]
|
||||||
|
fn available_memory_mib() -> Option<u32> {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn default_argon_decrypt_cap_mib() -> u32 {
|
||||||
|
let mut cap = DEFAULT_ARGON_DECRYPT_CAP_MIB.min(architecture_argon_cap_mib());
|
||||||
|
if let Some(available) = available_memory_mib() {
|
||||||
|
cap = cap.min(available);
|
||||||
|
}
|
||||||
|
cap.max(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn resolve_argon_decrypt_cap(override_mib: Option<u32>) -> Result<ArgonDecryptCap, FcryError> {
|
||||||
|
let default_mib = default_argon_decrypt_cap_mib();
|
||||||
|
let Some(effective_mib) = override_mib else {
|
||||||
|
return Ok(ArgonDecryptCap {
|
||||||
|
default_mib,
|
||||||
|
effective_mib: default_mib,
|
||||||
|
overridden: false,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
if effective_mib == 0 {
|
||||||
|
return Err(FcryError::Format(
|
||||||
|
"--max-argon-memory-mib must be at least 1".into(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
let arch = architecture_argon_cap_mib();
|
||||||
|
if effective_mib > arch {
|
||||||
|
return Err(FcryError::Format(format!(
|
||||||
|
"--max-argon-memory-mib {effective_mib} exceeds this build's supported cap {arch}"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
Ok(ArgonDecryptCap {
|
||||||
|
default_mib,
|
||||||
|
effective_mib,
|
||||||
|
overridden: true,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn mib_to_kib(mib: u32, name: &str) -> Result<u32, FcryError> {
|
||||||
|
mib.checked_mul(1024).ok_or_else(|| {
|
||||||
|
FcryError::Format(format!("{name} too large (overflow converting MiB to KiB)"))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn validate_chunk_size(chunk_size: u32) -> Result<usize, FcryError> {
|
||||||
|
if chunk_size == 0 {
|
||||||
|
return Err(FcryError::Format("chunk_size must be > 0".into()));
|
||||||
|
}
|
||||||
|
if chunk_size > MAX_CHUNK_SIZE {
|
||||||
|
return Err(FcryError::Format(format!(
|
||||||
|
"chunk_size {chunk_size} exceeds maximum {MAX_CHUNK_SIZE}"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
usize::try_from(chunk_size)
|
||||||
|
.map_err(|_| FcryError::Format("chunk_size does not fit in usize".into()))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn cipher_chunk_len(plain_chunk_len: usize) -> Result<usize, FcryError> {
|
||||||
|
plain_chunk_len
|
||||||
|
.checked_add(TAG_LEN)
|
||||||
|
.ok_or_else(|| FcryError::Format("cipher chunk length overflow".into()))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn validate_new_argon_params(
|
||||||
|
memory_mib: u32,
|
||||||
|
passes: u32,
|
||||||
|
parallelism: u32,
|
||||||
|
allow_weak_kdf: bool,
|
||||||
|
) -> Result<u32, FcryError> {
|
||||||
|
if !allow_weak_kdf && memory_mib < MIN_ARGON_MEMORY_MIB {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon-memory must be at least {MIN_ARGON_MEMORY_MIB} MiB for new encryption (use --allow-weak-kdf only for tests/legacy interop)"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
if !allow_weak_kdf && passes < MIN_ARGON_PASSES {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon-passes must be at least {MIN_ARGON_PASSES} for new encryption (use --allow-weak-kdf only for tests/legacy interop)"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
validate_argon_common(memory_mib, passes, parallelism, "new encryption")?;
|
||||||
|
mib_to_kib(memory_mib, "argon-memory")
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn validate_new_passphrase(pw: &SecretVec, allow_weak_kdf: bool) -> Result<(), FcryError> {
|
||||||
|
let len = pw.len();
|
||||||
|
if len == 0 {
|
||||||
|
return Err(FcryError::Passphrase("passphrase must not be empty".into()));
|
||||||
|
}
|
||||||
|
if !allow_weak_kdf && len < MIN_PASSPHRASE_BYTES {
|
||||||
|
return Err(FcryError::Passphrase(format!(
|
||||||
|
"passphrase must be at least {MIN_PASSPHRASE_BYTES} UTF-8 bytes for new encryption (use --allow-weak-kdf only for tests/legacy interop)"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn validate_header_kdf(kdf: &KdfParams, max_argon_memory_mib: u32) -> Result<(), FcryError> {
|
||||||
|
match kdf {
|
||||||
|
KdfParams::Raw => Ok(()),
|
||||||
|
KdfParams::Argon2id {
|
||||||
|
m_cost,
|
||||||
|
t_cost,
|
||||||
|
p_cost,
|
||||||
|
..
|
||||||
|
} => {
|
||||||
|
let cap_kib = mib_to_kib(max_argon_memory_mib, "max argon memory")?;
|
||||||
|
if *m_cost == 0 {
|
||||||
|
return Err(FcryError::Format("argon2id memory cost must be > 0".into()));
|
||||||
|
}
|
||||||
|
if *t_cost == 0 {
|
||||||
|
return Err(FcryError::Format("argon2id passes must be > 0".into()));
|
||||||
|
}
|
||||||
|
if *p_cost == 0 {
|
||||||
|
return Err(FcryError::Format("argon2id parallelism must be > 0".into()));
|
||||||
|
}
|
||||||
|
if *m_cost > cap_kib {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon2id memory cost {} KiB exceeds configured decrypt cap {} MiB",
|
||||||
|
*m_cost, max_argon_memory_mib
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
if *t_cost > MAX_ARGON_PASSES {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon2id passes {} exceeds maximum {}",
|
||||||
|
*t_cost, MAX_ARGON_PASSES
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
if *p_cost > MAX_ARGON_PARALLELISM {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon2id parallelism {} exceeds maximum {}",
|
||||||
|
*p_cost, MAX_ARGON_PARALLELISM
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn validate_argon_common(
|
||||||
|
memory_mib: u32,
|
||||||
|
passes: u32,
|
||||||
|
parallelism: u32,
|
||||||
|
context: &str,
|
||||||
|
) -> Result<(), FcryError> {
|
||||||
|
if memory_mib == 0 {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon-memory must be > 0 for {context}"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
if passes == 0 {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon-passes must be > 0 for {context}"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
if passes > MAX_ARGON_PASSES {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon-passes {passes} exceeds maximum {MAX_ARGON_PASSES}"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
if parallelism == 0 {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon-parallelism must be > 0 for {context}"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
if parallelism > MAX_ARGON_PARALLELISM {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon-parallelism {parallelism} exceeds maximum {MAX_ARGON_PARALLELISM}"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
if memory_mib > architecture_argon_cap_mib() {
|
||||||
|
return Err(FcryError::Kdf(format!(
|
||||||
|
"argon-memory {memory_mib} exceeds this build's supported cap {}",
|
||||||
|
architecture_argon_cap_mib()
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn normalize_worker_threads(requested: Option<u32>) -> (usize, Option<u32>) {
|
||||||
|
let requested = requested.map(|n| n as usize).unwrap_or_else(|| {
|
||||||
|
std::thread::available_parallelism()
|
||||||
|
.map(|n| n.get())
|
||||||
|
.unwrap_or(1)
|
||||||
|
});
|
||||||
|
let capped = requested.clamp(1, MAX_WORKER_THREADS);
|
||||||
|
let warning = (requested > MAX_WORKER_THREADS).then_some(requested as u32);
|
||||||
|
(capped, warning)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn pipeline_in_flight_capacity(threads: usize, chunk_len: usize) -> usize {
|
||||||
|
let chunk_len = chunk_len.max(1);
|
||||||
|
let thread_cap = threads.saturating_mul(4).max(1);
|
||||||
|
let byte_cap = (PIPELINE_IN_FLIGHT_BYTES / chunk_len).max(1);
|
||||||
|
thread_cap.min(byte_cap)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn pipeline_channel_capacity(threads: usize, in_flight: usize) -> usize {
|
||||||
|
threads.saturating_mul(2).max(1).min(in_flight.max(1))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn checked_add_u64(a: u64, b: u64, what: &str) -> Result<u64, FcryError> {
|
||||||
|
a.checked_add(b)
|
||||||
|
.ok_or_else(|| FcryError::Format(format!("{what} overflow")))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn checked_mul_u64(a: u64, b: u64, what: &str) -> Result<u64, FcryError> {
|
||||||
|
a.checked_mul(b)
|
||||||
|
.ok_or_else(|| FcryError::Format(format!("{what} overflow")))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn checked_count_add(total: u64, delta: usize, what: &str) -> Result<u64, FcryError> {
|
||||||
|
checked_add_u64(total, delta as u64, what)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn chunk_size_bounds() {
|
||||||
|
assert!(validate_chunk_size(1).is_ok());
|
||||||
|
assert!(validate_chunk_size(MAX_CHUNK_SIZE).is_ok());
|
||||||
|
assert!(validate_chunk_size(0).is_err());
|
||||||
|
assert!(validate_chunk_size(MAX_CHUNK_SIZE + 1).is_err());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn argon_cap_override_replaces_default() {
|
||||||
|
let down = resolve_argon_decrypt_cap(Some(1)).unwrap();
|
||||||
|
assert_eq!(down.effective_mib, 1);
|
||||||
|
assert!(down.overridden);
|
||||||
|
let default = resolve_argon_decrypt_cap(None).unwrap();
|
||||||
|
assert_eq!(default.effective_mib, default.default_mib);
|
||||||
|
assert!(!default.overridden);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn pipeline_capacity_has_one_chunk_minimum() {
|
||||||
|
assert_eq!(
|
||||||
|
pipeline_in_flight_capacity(4, PIPELINE_IN_FLIGHT_BYTES * 2),
|
||||||
|
1
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
+6
-5
@@ -1,7 +1,8 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: MIT-0
|
||||||
|
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::io::{BufRead, Read};
|
use std::io::{BufRead, Read};
|
||||||
|
use zeroize::Zeroizing;
|
||||||
|
|
||||||
pub enum ReadInfoChunk {
|
pub enum ReadInfoChunk {
|
||||||
Normal(#[allow(dead_code)] usize),
|
Normal(#[allow(dead_code)] usize),
|
||||||
@@ -11,7 +12,7 @@ pub enum ReadInfoChunk {
|
|||||||
|
|
||||||
pub struct AheadReader {
|
pub struct AheadReader {
|
||||||
inner: Box<dyn BufRead + Send>,
|
inner: Box<dyn BufRead + Send>,
|
||||||
buf: Vec<u8>,
|
buf: Zeroizing<Vec<u8>>,
|
||||||
bufsz: usize,
|
bufsz: usize,
|
||||||
capacity: usize,
|
capacity: usize,
|
||||||
}
|
}
|
||||||
@@ -20,7 +21,7 @@ impl AheadReader {
|
|||||||
pub fn from(reader: Box<dyn BufRead + Send>, capacity: usize) -> Self {
|
pub fn from(reader: Box<dyn BufRead + Send>, capacity: usize) -> Self {
|
||||||
Self {
|
Self {
|
||||||
inner: reader,
|
inner: reader,
|
||||||
buf: vec![0; capacity],
|
buf: Zeroizing::new(vec![0; capacity]),
|
||||||
bufsz: 0,
|
bufsz: 0,
|
||||||
capacity,
|
capacity,
|
||||||
}
|
}
|
||||||
@@ -61,7 +62,7 @@ impl AheadReader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 2nd read directly into our internal buf
|
// 2nd read directly into our internal buf
|
||||||
let mut tmp = vec![0u8; self.capacity];
|
let mut tmp = Zeroizing::new(vec![0u8; self.capacity]);
|
||||||
let n2 = self.read_until_full(&mut tmp)?;
|
let n2 = self.read_until_full(&mut tmp)?;
|
||||||
self.buf = tmp;
|
self.buf = tmp;
|
||||||
self.bufsz = n2;
|
self.bufsz = n2;
|
||||||
@@ -78,7 +79,7 @@ impl AheadReader {
|
|||||||
let userbuf_sz = self.bufsz;
|
let userbuf_sz = self.bufsz;
|
||||||
|
|
||||||
// 2nd read directly into our internal buf
|
// 2nd read directly into our internal buf
|
||||||
let mut tmp = vec![0u8; self.capacity];
|
let mut tmp = Zeroizing::new(vec![0u8; self.capacity]);
|
||||||
let n2 = self.read_until_full(&mut tmp)?;
|
let n2 = self.read_until_full(&mut tmp)?;
|
||||||
self.buf = tmp;
|
self.buf = tmp;
|
||||||
self.bufsz = n2;
|
self.bufsz = n2;
|
||||||
|
|||||||
+57
-15
@@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: MIT-0
|
||||||
|
|
||||||
//! Secret-handling primitives.
|
//! Secret-handling primitives.
|
||||||
//!
|
//!
|
||||||
@@ -95,6 +95,10 @@ impl SecretVec {
|
|||||||
let inner = self.inner.borrow();
|
let inner = self.inner.borrow();
|
||||||
f(&inner[..self.len])
|
f(&inner[..self.len])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn len(&self) -> usize {
|
||||||
|
self.len
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PartialEq for SecretVec {
|
impl PartialEq for SecretVec {
|
||||||
@@ -192,13 +196,15 @@ mod imp {
|
|||||||
mod imp {
|
mod imp {
|
||||||
use super::{MAX_PASSPHRASE_LEN, SecretVec};
|
use super::{MAX_PASSPHRASE_LEN, SecretVec};
|
||||||
use std::fs::OpenOptions;
|
use std::fs::OpenOptions;
|
||||||
use std::io::{self, Read, Write};
|
use std::io::{self, Write};
|
||||||
use std::os::windows::io::AsRawHandle;
|
use std::os::windows::io::AsRawHandle;
|
||||||
|
use std::ptr;
|
||||||
use windows_sys::Win32::Foundation::HANDLE;
|
use windows_sys::Win32::Foundation::HANDLE;
|
||||||
use windows_sys::Win32::System::Console::{
|
use windows_sys::Win32::System::Console::{
|
||||||
ENABLE_ECHO_INPUT, ENABLE_LINE_INPUT, ENABLE_PROCESSED_INPUT, GetConsoleMode,
|
ENABLE_ECHO_INPUT, ENABLE_LINE_INPUT, ENABLE_PROCESSED_INPUT, GetConsoleMode, ReadConsoleW,
|
||||||
SetConsoleMode,
|
SetConsoleMode,
|
||||||
};
|
};
|
||||||
|
use zeroize::Zeroizing;
|
||||||
|
|
||||||
struct ConsoleModeGuard {
|
struct ConsoleModeGuard {
|
||||||
handle: HANDLE,
|
handle: HANDLE,
|
||||||
@@ -214,7 +220,7 @@ mod imp {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn read_passphrase_tty(prompt: &str) -> io::Result<SecretVec> {
|
pub fn read_passphrase_tty(prompt: &str) -> io::Result<SecretVec> {
|
||||||
let mut tty_in = OpenOptions::new().read(true).write(true).open("CONIN$")?;
|
let tty_in = OpenOptions::new().read(true).write(true).open("CONIN$")?;
|
||||||
let mut tty_out = OpenOptions::new().write(true).open("CONOUT$")?;
|
let mut tty_out = OpenOptions::new().write(true).open("CONOUT$")?;
|
||||||
|
|
||||||
let h_in = tty_in.as_raw_handle() as HANDLE;
|
let h_in = tty_in.as_raw_handle() as HANDLE;
|
||||||
@@ -236,18 +242,38 @@ mod imp {
|
|||||||
write!(tty_out, "{prompt}")?;
|
write!(tty_out, "{prompt}")?;
|
||||||
tty_out.flush()?;
|
tty_out.flush()?;
|
||||||
|
|
||||||
let mut buf = SecretVec::with_capacity(MAX_PASSPHRASE_LEN);
|
let mut wide = Zeroizing::new(Vec::<u16>::with_capacity(MAX_PASSPHRASE_LEN));
|
||||||
let mut byte = [0u8; 1];
|
|
||||||
loop {
|
loop {
|
||||||
match tty_in.read(&mut byte) {
|
let mut unit = 0u16;
|
||||||
Ok(0) => break,
|
let mut read = 0u32;
|
||||||
Ok(_) => match byte[0] {
|
let ok = unsafe {
|
||||||
b'\n' => break,
|
ReadConsoleW(
|
||||||
b'\r' => continue,
|
h_in,
|
||||||
b => buf.push(b)?,
|
(&mut unit as *mut u16).cast(),
|
||||||
},
|
1,
|
||||||
Err(e) if e.kind() == io::ErrorKind::Interrupted => continue,
|
&mut read,
|
||||||
Err(e) => return Err(e),
|
ptr::null_mut(),
|
||||||
|
)
|
||||||
|
};
|
||||||
|
if ok == 0 {
|
||||||
|
return Err(io::Error::last_os_error());
|
||||||
|
}
|
||||||
|
if read == 0 {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
const LF: u16 = b'\n' as u16;
|
||||||
|
const CR: u16 = b'\r' as u16;
|
||||||
|
match unit {
|
||||||
|
LF | CR => break,
|
||||||
|
u => {
|
||||||
|
if wide.len() >= MAX_PASSPHRASE_LEN {
|
||||||
|
return Err(io::Error::new(
|
||||||
|
io::ErrorKind::InvalidInput,
|
||||||
|
"secret buffer full",
|
||||||
|
));
|
||||||
|
}
|
||||||
|
wide.push(u);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -255,6 +281,22 @@ mod imp {
|
|||||||
let _ = writeln!(tty_out);
|
let _ = writeln!(tty_out);
|
||||||
let _ = tty_out.flush();
|
let _ = tty_out.flush();
|
||||||
|
|
||||||
|
let utf8 = Zeroizing::new(String::from_utf16(&wide).map_err(|_| {
|
||||||
|
io::Error::new(
|
||||||
|
io::ErrorKind::InvalidData,
|
||||||
|
"console passphrase is not valid UTF-16",
|
||||||
|
)
|
||||||
|
})?);
|
||||||
|
if utf8.len() > MAX_PASSPHRASE_LEN {
|
||||||
|
return Err(io::Error::new(
|
||||||
|
io::ErrorKind::InvalidInput,
|
||||||
|
"secret buffer full",
|
||||||
|
));
|
||||||
|
}
|
||||||
|
let mut buf = SecretVec::with_capacity(MAX_PASSPHRASE_LEN);
|
||||||
|
for b in utf8.as_bytes() {
|
||||||
|
buf.push(*b)?;
|
||||||
|
}
|
||||||
Ok(buf)
|
Ok(buf)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+208
-61
@@ -1,14 +1,16 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: MIT-0
|
||||||
|
|
||||||
use std::fs::{self, File};
|
use std::fs::{self, File, OpenOptions};
|
||||||
use std::io::{self, BufRead, BufReader, Write};
|
use std::io::{self, BufRead, BufReader, Seek, SeekFrom, Write};
|
||||||
use std::path::PathBuf;
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
|
use crate::policy;
|
||||||
|
|
||||||
/// Default plaintext chunk size: 1 MiB.
|
/// Default plaintext chunk size: 1 MiB.
|
||||||
///
|
///
|
||||||
/// Stored in the header per file, so callers may override via CLI without
|
/// Stored in the header per file, so callers may override via CLI without
|
||||||
/// breaking older files (the decryptor reads the size from the header).
|
/// breaking older files (the decryptor reads the size from the header).
|
||||||
pub const DEFAULT_CHUNK_SIZE: u32 = 1024 * 1024;
|
pub const DEFAULT_CHUNK_SIZE: u32 = policy::DEFAULT_CHUNK_SIZE;
|
||||||
|
|
||||||
/// Opened input.
|
/// Opened input.
|
||||||
///
|
///
|
||||||
@@ -45,63 +47,225 @@ pub(crate) fn open_input<S: AsRef<str>>(input_file: Option<S>) -> io::Result<Inp
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Default)]
|
||||||
|
pub struct OutSinkOptions {
|
||||||
|
pub force: bool,
|
||||||
|
pub input_file: Option<PathBuf>,
|
||||||
|
pub temp_dir: Option<PathBuf>,
|
||||||
|
pub buffer_verify_stdout: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) struct SecureTempFile {
|
||||||
|
path: PathBuf,
|
||||||
|
file: Option<File>,
|
||||||
|
remove_on_drop: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SecureTempFile {
|
||||||
|
fn create(dir: &Path, prefix: &str) -> io::Result<Self> {
|
||||||
|
fs::create_dir_all(dir)?;
|
||||||
|
for _ in 0..128 {
|
||||||
|
let mut rand = [0u8; 16];
|
||||||
|
getrandom::fill(&mut rand).map_err(io::Error::other)?;
|
||||||
|
let name = format!("{prefix}.{}.tmp", hex(&rand));
|
||||||
|
let path = dir.join(name);
|
||||||
|
let mut opts = OpenOptions::new();
|
||||||
|
opts.read(true).write(true).create_new(true);
|
||||||
|
#[cfg(unix)]
|
||||||
|
{
|
||||||
|
use std::os::unix::fs::OpenOptionsExt;
|
||||||
|
opts.mode(0o600);
|
||||||
|
}
|
||||||
|
match opts.open(&path) {
|
||||||
|
Ok(file) => {
|
||||||
|
return Ok(Self {
|
||||||
|
path,
|
||||||
|
file: Some(file),
|
||||||
|
remove_on_drop: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
Err(e) if e.kind() == io::ErrorKind::AlreadyExists => continue,
|
||||||
|
Err(e) => return Err(e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(io::Error::new(
|
||||||
|
io::ErrorKind::AlreadyExists,
|
||||||
|
"could not create a unique temporary file after 128 attempts",
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn file_mut(&mut self) -> &mut File {
|
||||||
|
self.file
|
||||||
|
.as_mut()
|
||||||
|
.expect("temporary file handle taken before commit")
|
||||||
|
}
|
||||||
|
|
||||||
|
fn sync_file(&mut self) -> io::Result<()> {
|
||||||
|
let file = self.file_mut();
|
||||||
|
file.flush()?;
|
||||||
|
file.sync_all()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn persist(mut self, final_path: &Path) -> io::Result<()> {
|
||||||
|
self.sync_file()?;
|
||||||
|
self.file.take();
|
||||||
|
#[cfg(windows)]
|
||||||
|
if final_path.exists() {
|
||||||
|
fs::remove_file(final_path)?;
|
||||||
|
}
|
||||||
|
fs::rename(&self.path, final_path)?;
|
||||||
|
self.remove_on_drop = false;
|
||||||
|
best_effort_fsync_parent(final_path);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn copy_to_stdout(mut self) -> io::Result<()> {
|
||||||
|
self.sync_file()?;
|
||||||
|
let mut file = self
|
||||||
|
.file
|
||||||
|
.take()
|
||||||
|
.expect("temporary file handle taken before stdout commit");
|
||||||
|
file.seek(SeekFrom::Start(0))?;
|
||||||
|
let mut stdout = io::stdout();
|
||||||
|
io::copy(&mut file, &mut stdout)?;
|
||||||
|
stdout.flush()?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for SecureTempFile {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
self.file.take();
|
||||||
|
if self.remove_on_drop {
|
||||||
|
let _ = fs::remove_file(&self.path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn hex(bytes: &[u8]) -> String {
|
||||||
|
const HEX: &[u8; 16] = b"0123456789abcdef";
|
||||||
|
let mut out = String::with_capacity(bytes.len() * 2);
|
||||||
|
for b in bytes {
|
||||||
|
out.push(HEX[(b >> 4) as usize] as char);
|
||||||
|
out.push(HEX[(b & 0x0f) as usize] as char);
|
||||||
|
}
|
||||||
|
out
|
||||||
|
}
|
||||||
|
|
||||||
|
fn best_effort_fsync_parent(path: &Path) {
|
||||||
|
let Some(parent) = path.parent() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
if let Ok(dir) = File::open(parent) {
|
||||||
|
let _ = dir.sync_all();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn temp_dir_for_target(final_path: &Path, explicit: Option<&Path>) -> PathBuf {
|
||||||
|
if let Some(dir) = explicit {
|
||||||
|
return dir.to_path_buf();
|
||||||
|
}
|
||||||
|
final_path
|
||||||
|
.parent()
|
||||||
|
.filter(|p| !p.as_os_str().is_empty())
|
||||||
|
.map(Path::to_path_buf)
|
||||||
|
.unwrap_or_else(|| PathBuf::from("."))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn temp_dir_for_stdout(explicit: Option<&Path>) -> PathBuf {
|
||||||
|
explicit
|
||||||
|
.map(Path::to_path_buf)
|
||||||
|
.unwrap_or_else(std::env::temp_dir)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn file_name_prefix(path: &Path) -> String {
|
||||||
|
path.file_name()
|
||||||
|
.and_then(|name| name.to_str())
|
||||||
|
.filter(|name| !name.is_empty())
|
||||||
|
.unwrap_or("fcry")
|
||||||
|
.to_owned()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn output_aliases_input(output: &Path, input: Option<&Path>) -> io::Result<bool> {
|
||||||
|
let Some(input) = input else {
|
||||||
|
return Ok(false);
|
||||||
|
};
|
||||||
|
match same_file::is_same_file(input, output) {
|
||||||
|
Ok(same) => Ok(same),
|
||||||
|
Err(e) if e.kind() == io::ErrorKind::NotFound => Ok(false),
|
||||||
|
Err(e) => Err(e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Output sink that supports atomic file replacement.
|
/// Output sink that supports atomic file replacement.
|
||||||
///
|
///
|
||||||
/// For file outputs: bytes are written to `<path>.tmp`. On `commit()`, the
|
/// For file outputs: bytes are written to a private, randomly named temp file.
|
||||||
/// temp file is renamed into place. If dropped without commit (panic, error,
|
/// On `commit()`, the temp file is fsynced and renamed into place. If dropped
|
||||||
/// process exit), the temp file is deleted so a partial/garbage file does
|
/// without commit (panic, error, process exit), the temp file is deleted so a
|
||||||
/// not replace any existing target.
|
/// partial/garbage file does not replace any existing target.
|
||||||
///
|
///
|
||||||
/// For stdout: behaves as a passthrough; `commit()` is a no-op.
|
/// For stdout: behaves as a passthrough; `commit()` is a no-op.
|
||||||
pub enum OutSink {
|
pub enum OutSink {
|
||||||
Stdout(io::Stdout),
|
Stdout(io::Stdout),
|
||||||
|
BufferVerify {
|
||||||
|
temp: SecureTempFile,
|
||||||
|
},
|
||||||
File {
|
File {
|
||||||
tmp_path: PathBuf,
|
|
||||||
final_path: PathBuf,
|
final_path: PathBuf,
|
||||||
file: Option<File>,
|
temp: SecureTempFile,
|
||||||
committed: bool,
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
impl OutSink {
|
impl OutSink {
|
||||||
|
#[allow(dead_code)]
|
||||||
pub fn open<S: AsRef<str>>(output_file: Option<S>) -> io::Result<Self> {
|
pub fn open<S: AsRef<str>>(output_file: Option<S>) -> io::Result<Self> {
|
||||||
|
Self::open_with_options(output_file, &OutSinkOptions::default())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn open_with_options<S: AsRef<str>>(
|
||||||
|
output_file: Option<S>,
|
||||||
|
options: &OutSinkOptions,
|
||||||
|
) -> io::Result<Self> {
|
||||||
match output_file {
|
match output_file {
|
||||||
|
None if options.buffer_verify_stdout => {
|
||||||
|
let dir = temp_dir_for_stdout(options.temp_dir.as_deref());
|
||||||
|
Ok(Self::BufferVerify {
|
||||||
|
temp: SecureTempFile::create(&dir, "fcry-buffer")?,
|
||||||
|
})
|
||||||
|
}
|
||||||
None => Ok(Self::Stdout(io::stdout())),
|
None => Ok(Self::Stdout(io::stdout())),
|
||||||
Some(f) => {
|
Some(f) => {
|
||||||
let final_path = PathBuf::from(f.as_ref());
|
let final_path = PathBuf::from(f.as_ref());
|
||||||
let mut tmp_path = final_path.clone();
|
if final_path.exists()
|
||||||
let name = tmp_path
|
&& !options.force
|
||||||
.file_name()
|
&& !output_aliases_input(&final_path, options.input_file.as_deref())?
|
||||||
.map(|n| n.to_os_string())
|
{
|
||||||
.unwrap_or_default();
|
return Err(io::Error::new(
|
||||||
let mut tmp_name = name;
|
io::ErrorKind::AlreadyExists,
|
||||||
tmp_name.push(".tmp");
|
format!(
|
||||||
tmp_path.set_file_name(tmp_name);
|
"output file {} already exists (use --force to replace it)",
|
||||||
let file = File::create(&tmp_path)?;
|
final_path.display()
|
||||||
Ok(Self::File {
|
),
|
||||||
tmp_path,
|
));
|
||||||
final_path,
|
}
|
||||||
file: Some(file),
|
let dir = temp_dir_for_target(&final_path, options.temp_dir.as_deref());
|
||||||
committed: false,
|
let prefix = file_name_prefix(&final_path);
|
||||||
})
|
let temp = SecureTempFile::create(&dir, &prefix)?;
|
||||||
|
Ok(Self::File { final_path, temp })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn commit(mut self) -> io::Result<()> {
|
pub fn commit(mut self) -> io::Result<()> {
|
||||||
if let Self::File {
|
match &mut self {
|
||||||
tmp_path,
|
Self::Stdout(s) => s.flush()?,
|
||||||
final_path,
|
Self::BufferVerify { .. } => {}
|
||||||
file,
|
Self::File { .. } => {}
|
||||||
committed,
|
}
|
||||||
} = &mut self
|
match self {
|
||||||
{
|
Self::Stdout(_) => {}
|
||||||
if let Some(mut f) = file.take() {
|
Self::BufferVerify { temp } => temp.copy_to_stdout()?,
|
||||||
f.flush()?;
|
Self::File { final_path, temp } => temp.persist(&final_path)?,
|
||||||
f.sync_all()?;
|
|
||||||
}
|
|
||||||
fs::rename(&*tmp_path, &*final_path)?;
|
|
||||||
*committed = true;
|
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@@ -111,33 +275,16 @@ impl Write for OutSink {
|
|||||||
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
||||||
match self {
|
match self {
|
||||||
Self::Stdout(s) => s.write(buf),
|
Self::Stdout(s) => s.write(buf),
|
||||||
Self::File { file, .. } => file.as_mut().expect("file taken before commit").write(buf),
|
Self::BufferVerify { temp } => temp.file_mut().write(buf),
|
||||||
|
Self::File { temp, .. } => temp.file_mut().write(buf),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn flush(&mut self) -> io::Result<()> {
|
fn flush(&mut self) -> io::Result<()> {
|
||||||
match self {
|
match self {
|
||||||
Self::Stdout(s) => s.flush(),
|
Self::Stdout(s) => s.flush(),
|
||||||
Self::File { file, .. } => match file.as_mut() {
|
Self::BufferVerify { temp } => temp.file_mut().flush(),
|
||||||
Some(f) => f.flush(),
|
Self::File { temp, .. } => temp.file_mut().flush(),
|
||||||
None => Ok(()),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Drop for OutSink {
|
|
||||||
fn drop(&mut self) {
|
|
||||||
if let Self::File {
|
|
||||||
tmp_path,
|
|
||||||
committed,
|
|
||||||
file,
|
|
||||||
..
|
|
||||||
} = self
|
|
||||||
&& !*committed
|
|
||||||
{
|
|
||||||
file.take(); // close the file before unlink
|
|
||||||
let _ = fs::remove_file(tmp_path);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+539
-49
@@ -1,4 +1,4 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: MIT-0
|
||||||
//
|
//
|
||||||
// Integration tests for the `fcry` binary.
|
// Integration tests for the `fcry` binary.
|
||||||
//
|
//
|
||||||
@@ -7,19 +7,28 @@
|
|||||||
// wrong key, truncation, bad magic).
|
// wrong key, truncation, bad magic).
|
||||||
|
|
||||||
use std::fs;
|
use std::fs;
|
||||||
use std::io::Write;
|
use std::io::{ErrorKind, Write};
|
||||||
use std::process::{Command, Stdio};
|
use std::process::{Command, Stdio};
|
||||||
|
|
||||||
use assert_cmd::cargo::CommandCargoExt;
|
use assert_cmd::cargo::CommandCargoExt;
|
||||||
use tempfile::TempDir;
|
use tempfile::TempDir;
|
||||||
|
|
||||||
const KEY: &[u8; 32] = b"0123456789abcdef0123456789abcdef";
|
const KEY: &[u8; 32] = b"0123456789abcdef0123456789abcdef";
|
||||||
const KEY_STR: &str = "0123456789abcdef0123456789abcdef";
|
|
||||||
|
|
||||||
fn fcry() -> Command {
|
fn fcry() -> Command {
|
||||||
Command::cargo_bin("fcry").unwrap()
|
Command::cargo_bin("fcry").unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn write_key_file(dir: &std::path::Path) -> std::path::PathBuf {
|
||||||
|
let key = dir.join("key.bin");
|
||||||
|
fs::write(&key, KEY).unwrap();
|
||||||
|
key
|
||||||
|
}
|
||||||
|
|
||||||
|
fn key_file_near(path: &std::path::Path) -> std::path::PathBuf {
|
||||||
|
write_key_file(path.parent().unwrap())
|
||||||
|
}
|
||||||
|
|
||||||
/// Deterministic pseudo-random plaintext of `n` bytes (xorshift, seedable).
|
/// Deterministic pseudo-random plaintext of `n` bytes (xorshift, seedable).
|
||||||
/// We avoid `/dev/urandom` so tests are reproducible on failure.
|
/// We avoid `/dev/urandom` so tests are reproducible on failure.
|
||||||
fn pseudo_random(seed: u64, n: usize) -> Vec<u8> {
|
fn pseudo_random(seed: u64, n: usize) -> Vec<u8> {
|
||||||
@@ -37,12 +46,13 @@ fn pseudo_random(seed: u64, n: usize) -> Vec<u8> {
|
|||||||
|
|
||||||
fn encrypt_file(plain: &std::path::Path, ct: &std::path::Path, chunk_size: Option<u32>) {
|
fn encrypt_file(plain: &std::path::Path, ct: &std::path::Path, chunk_size: Option<u32>) {
|
||||||
let mut cmd = fcry();
|
let mut cmd = fcry();
|
||||||
|
let key = key_file_near(ct);
|
||||||
cmd.arg("-i")
|
cmd.arg("-i")
|
||||||
.arg(plain)
|
.arg(plain)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(ct)
|
.arg(ct)
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR);
|
.arg(key);
|
||||||
if let Some(cs) = chunk_size {
|
if let Some(cs) = chunk_size {
|
||||||
cmd.arg("--chunk-size").arg(cs.to_string());
|
cmd.arg("--chunk-size").arg(cs.to_string());
|
||||||
}
|
}
|
||||||
@@ -55,14 +65,15 @@ fn encrypt_file(plain: &std::path::Path, ct: &std::path::Path, chunk_size: Optio
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn decrypt_file(ct: &std::path::Path, rt: &std::path::Path) {
|
fn decrypt_file(ct: &std::path::Path, rt: &std::path::Path) {
|
||||||
|
let key = key_file_near(ct);
|
||||||
let out = fcry()
|
let out = fcry()
|
||||||
.arg("-d")
|
.arg("-d")
|
||||||
.arg("-i")
|
.arg("-i")
|
||||||
.arg(ct)
|
.arg(ct)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(rt)
|
.arg(rt)
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(key)
|
||||||
.output()
|
.output()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
assert!(
|
assert!(
|
||||||
@@ -133,10 +144,12 @@ fn roundtrip_chunk_size_one_byte() {
|
|||||||
#[test]
|
#[test]
|
||||||
fn roundtrip_pipe_stdin_stdout() {
|
fn roundtrip_pipe_stdin_stdout() {
|
||||||
let data = pseudo_random(42, 200_000);
|
let data = pseudo_random(42, 200_000);
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let key = write_key_file(dir.path());
|
||||||
|
|
||||||
let mut enc = fcry()
|
let mut enc = fcry()
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(&key)
|
||||||
.stdin(Stdio::piped())
|
.stdin(Stdio::piped())
|
||||||
.stdout(Stdio::piped())
|
.stdout(Stdio::piped())
|
||||||
.stderr(Stdio::piped())
|
.stderr(Stdio::piped())
|
||||||
@@ -152,8 +165,8 @@ fn roundtrip_pipe_stdin_stdout() {
|
|||||||
|
|
||||||
let mut dec = fcry()
|
let mut dec = fcry()
|
||||||
.arg("-d")
|
.arg("-d")
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(&key)
|
||||||
.stdin(Stdio::piped())
|
.stdin(Stdio::piped())
|
||||||
.stdout(Stdio::piped())
|
.stdout(Stdio::piped())
|
||||||
.stderr(Stdio::piped())
|
.stderr(Stdio::piped())
|
||||||
@@ -182,19 +195,24 @@ fn rejects_wrong_key() {
|
|||||||
fs::write(&plain, pseudo_random(1, 1000)).unwrap();
|
fs::write(&plain, pseudo_random(1, 1000)).unwrap();
|
||||||
encrypt_file(&plain, &ct, None);
|
encrypt_file(&plain, &ct, None);
|
||||||
|
|
||||||
let wrong = "ffffffffffffffffffffffffffffffff";
|
let wrong = dir.path().join("wrong.key");
|
||||||
assert_ne!(wrong.as_bytes(), KEY);
|
fs::write(&wrong, b"ffffffffffffffffffffffffffffffff").unwrap();
|
||||||
let out = fcry()
|
let out = fcry()
|
||||||
.arg("-d")
|
.arg("-d")
|
||||||
.arg("-i")
|
.arg("-i")
|
||||||
.arg(&ct)
|
.arg(&ct)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(dir.path().join("rt.bin"))
|
.arg(dir.path().join("rt.bin"))
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(wrong)
|
.arg(wrong)
|
||||||
.output()
|
.output()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
assert!(!out.status.success(), "decrypt with wrong key should fail");
|
assert!(!out.status.success(), "decrypt with wrong key should fail");
|
||||||
|
assert!(
|
||||||
|
String::from_utf8_lossy(&out.stderr).contains("WrongKey"),
|
||||||
|
"expected distinct WrongKey error, got {}",
|
||||||
|
String::from_utf8_lossy(&out.stderr)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@@ -216,8 +234,8 @@ fn rejects_tampered_header() {
|
|||||||
.arg(&ct)
|
.arg(&ct)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(dir.path().join("rt.bin"))
|
.arg(dir.path().join("rt.bin"))
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(key_file_near(&ct))
|
||||||
.output()
|
.output()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
assert!(
|
assert!(
|
||||||
@@ -246,8 +264,8 @@ fn rejects_tampered_ciphertext() {
|
|||||||
.arg(&ct)
|
.arg(&ct)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(dir.path().join("rt.bin"))
|
.arg(dir.path().join("rt.bin"))
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(key_file_near(&ct))
|
||||||
.output()
|
.output()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
assert!(
|
assert!(
|
||||||
@@ -275,8 +293,8 @@ fn rejects_truncated_ciphertext() {
|
|||||||
.arg(&ct)
|
.arg(&ct)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(dir.path().join("rt.bin"))
|
.arg(dir.path().join("rt.bin"))
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(key_file_near(&ct))
|
||||||
.output()
|
.output()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
assert!(
|
assert!(
|
||||||
@@ -296,8 +314,8 @@ fn rejects_bad_magic() {
|
|||||||
.arg(&bogus)
|
.arg(&bogus)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(dir.path().join("rt.bin"))
|
.arg(dir.path().join("rt.bin"))
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(write_key_file(dir.path()))
|
||||||
.output()
|
.output()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
assert!(
|
assert!(
|
||||||
@@ -312,25 +330,145 @@ fn rejects_bad_magic() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn rejects_short_raw_key() {
|
fn rejects_short_key_file() {
|
||||||
let dir = TempDir::new().unwrap();
|
let dir = TempDir::new().unwrap();
|
||||||
let plain = dir.path().join("p.bin");
|
let plain = dir.path().join("p.bin");
|
||||||
|
let key = dir.path().join("short.key");
|
||||||
fs::write(&plain, b"hello").unwrap();
|
fs::write(&plain, b"hello").unwrap();
|
||||||
|
fs::write(&key, b"tooshort").unwrap();
|
||||||
let out = fcry()
|
let out = fcry()
|
||||||
.arg("-i")
|
.arg("-i")
|
||||||
.arg(&plain)
|
.arg(&plain)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(dir.path().join("c.bin"))
|
.arg(dir.path().join("c.bin"))
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg("tooshort")
|
.arg(&key)
|
||||||
.output()
|
.output()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
assert!(
|
assert!(
|
||||||
!out.status.success(),
|
!out.status.success(),
|
||||||
"encrypt with short raw_key should fail"
|
"encrypt with short key file should fail"
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn rejects_long_key_file_and_trailing_newline() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let plain = dir.path().join("p.bin");
|
||||||
|
let key = dir.path().join("long.key");
|
||||||
|
fs::write(&plain, b"hello").unwrap();
|
||||||
|
fs::write(&key, b"0123456789abcdef0123456789abcdef\n").unwrap();
|
||||||
|
let out = fcry()
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&plain)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(dir.path().join("c.bin"))
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(&key)
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(!out.status.success(), "long key file should fail");
|
||||||
|
assert!(
|
||||||
|
String::from_utf8_lossy(&out.stderr).contains("too long"),
|
||||||
|
"expected too-long error, got {}",
|
||||||
|
String::from_utf8_lossy(&out.stderr)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn non_utf8_key_file_roundtrips() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let key = dir.path().join("key.bin");
|
||||||
|
let plain = dir.path().join("plain.bin");
|
||||||
|
let ct = dir.path().join("ct.bin");
|
||||||
|
let rt = dir.path().join("rt.bin");
|
||||||
|
let key_bytes: Vec<u8> = (0..32u8).map(|b| b ^ 0x80).collect();
|
||||||
|
let data = pseudo_random(31, 8192);
|
||||||
|
fs::write(&key, key_bytes).unwrap();
|
||||||
|
fs::write(&plain, &data).unwrap();
|
||||||
|
|
||||||
|
let enc = fcry()
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&plain)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&ct)
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(&key)
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(
|
||||||
|
enc.status.success(),
|
||||||
|
"non-UTF-8 key encrypt failed: {}",
|
||||||
|
String::from_utf8_lossy(&enc.stderr)
|
||||||
|
);
|
||||||
|
let dec = fcry()
|
||||||
|
.arg("-d")
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&ct)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&rt)
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(&key)
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(
|
||||||
|
dec.status.success(),
|
||||||
|
"non-UTF-8 key decrypt failed: {}",
|
||||||
|
String::from_utf8_lossy(&dec.stderr)
|
||||||
|
);
|
||||||
|
assert_eq!(fs::read(&rt).unwrap(), data);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(unix)]
|
||||||
|
#[test]
|
||||||
|
fn split_fifo_key_file_read_roundtrips() {
|
||||||
|
use std::ffi::CString;
|
||||||
|
use std::fs::OpenOptions;
|
||||||
|
use std::os::unix::ffi::OsStrExt;
|
||||||
|
use std::thread;
|
||||||
|
use std::time::Duration;
|
||||||
|
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let fifo = dir.path().join("key.fifo");
|
||||||
|
let fifo_c = CString::new(fifo.as_os_str().as_bytes()).unwrap();
|
||||||
|
let rc = unsafe { libc::mkfifo(fifo_c.as_ptr(), 0o600) };
|
||||||
|
assert_eq!(rc, 0, "mkfifo failed: {}", std::io::Error::last_os_error());
|
||||||
|
|
||||||
|
let plain = dir.path().join("plain.bin");
|
||||||
|
let ct = dir.path().join("ct.bin");
|
||||||
|
let rt = dir.path().join("rt.bin");
|
||||||
|
let data = pseudo_random(33, 8192);
|
||||||
|
fs::write(&plain, &data).unwrap();
|
||||||
|
|
||||||
|
let fifo_writer = fifo.clone();
|
||||||
|
let writer = thread::spawn(move || {
|
||||||
|
let mut file = OpenOptions::new().write(true).open(&fifo_writer).unwrap();
|
||||||
|
file.write_all(&KEY[..8]).unwrap();
|
||||||
|
file.flush().unwrap();
|
||||||
|
thread::sleep(Duration::from_millis(50));
|
||||||
|
file.write_all(&KEY[8..]).unwrap();
|
||||||
|
});
|
||||||
|
|
||||||
|
let enc = fcry()
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&plain)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&ct)
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(&fifo)
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
writer.join().unwrap();
|
||||||
|
assert!(
|
||||||
|
enc.status.success(),
|
||||||
|
"split FIFO key encrypt failed: {}",
|
||||||
|
String::from_utf8_lossy(&enc.stderr)
|
||||||
|
);
|
||||||
|
|
||||||
|
decrypt_file(&ct, &rt);
|
||||||
|
assert_eq!(fs::read(&rt).unwrap(), data);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn roundtrip_passphrase_argon2id() {
|
fn roundtrip_passphrase_argon2id() {
|
||||||
let dir = TempDir::new().unwrap();
|
let dir = TempDir::new().unwrap();
|
||||||
@@ -352,6 +490,7 @@ fn roundtrip_passphrase_argon2id() {
|
|||||||
.arg("8")
|
.arg("8")
|
||||||
.arg("--argon-passes")
|
.arg("--argon-passes")
|
||||||
.arg("1")
|
.arg("1")
|
||||||
|
.arg("--allow-weak-kdf")
|
||||||
.env("FCRY_TEST_PW", "correct horse battery staple")
|
.env("FCRY_TEST_PW", "correct horse battery staple")
|
||||||
.output()
|
.output()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
@@ -394,6 +533,70 @@ fn roundtrip_passphrase_argon2id() {
|
|||||||
assert!(!bad.status.success(), "wrong passphrase should fail");
|
assert!(!bad.status.success(), "wrong passphrase should fail");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn weak_passphrase_kdf_rejected_without_override() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let plain = dir.path().join("p.bin");
|
||||||
|
fs::write(&plain, b"hello").unwrap();
|
||||||
|
let enc = fcry()
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&plain)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(dir.path().join("c.bin"))
|
||||||
|
.arg("--passphrase-env")
|
||||||
|
.arg("FCRY_TEST_PW")
|
||||||
|
.arg("--argon-memory")
|
||||||
|
.arg("8")
|
||||||
|
.arg("--argon-passes")
|
||||||
|
.arg("1")
|
||||||
|
.env("FCRY_TEST_PW", "short")
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(!enc.status.success(), "weak KDF/passphrase should fail");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn decrypt_argon_memory_cap_rejects_hostile_header() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let plain = dir.path().join("p.bin");
|
||||||
|
let ct = dir.path().join("c.bin");
|
||||||
|
fs::write(&plain, b"hello").unwrap();
|
||||||
|
let enc = fcry()
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&plain)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&ct)
|
||||||
|
.arg("--passphrase-env")
|
||||||
|
.arg("FCRY_TEST_PW")
|
||||||
|
.arg("--argon-memory")
|
||||||
|
.arg("8")
|
||||||
|
.arg("--argon-passes")
|
||||||
|
.arg("1")
|
||||||
|
.arg("--allow-weak-kdf")
|
||||||
|
.env("FCRY_TEST_PW", "correct horse battery staple")
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(enc.status.success());
|
||||||
|
|
||||||
|
let dec = fcry()
|
||||||
|
.arg("-d")
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&ct)
|
||||||
|
.arg("--passphrase-env")
|
||||||
|
.arg("FCRY_TEST_PW")
|
||||||
|
.arg("--max-argon-memory-mib")
|
||||||
|
.arg("1")
|
||||||
|
.env("FCRY_TEST_PW", "correct horse battery staple")
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(!dec.status.success(), "low decrypt cap should reject file");
|
||||||
|
assert!(
|
||||||
|
String::from_utf8_lossy(&dec.stderr).contains("decrypt cap"),
|
||||||
|
"expected cap error, got {}",
|
||||||
|
String::from_utf8_lossy(&dec.stderr)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn atomic_output_no_stale_tmp_on_failure() {
|
fn atomic_output_no_stale_tmp_on_failure() {
|
||||||
// A failed decrypt (wrong key) should not leave the output file behind.
|
// A failed decrypt (wrong key) should not leave the output file behind.
|
||||||
@@ -404,15 +607,16 @@ fn atomic_output_no_stale_tmp_on_failure() {
|
|||||||
fs::write(&plain, b"hello world").unwrap();
|
fs::write(&plain, b"hello world").unwrap();
|
||||||
encrypt_file(&plain, &ct, None);
|
encrypt_file(&plain, &ct, None);
|
||||||
|
|
||||||
let wrong = "ffffffffffffffffffffffffffffffff";
|
let wrong = dir.path().join("wrong.key");
|
||||||
|
fs::write(&wrong, b"ffffffffffffffffffffffffffffffff").unwrap();
|
||||||
let out = fcry()
|
let out = fcry()
|
||||||
.arg("-d")
|
.arg("-d")
|
||||||
.arg("-i")
|
.arg("-i")
|
||||||
.arg(&ct)
|
.arg(&ct)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(&rt)
|
.arg(&rt)
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(wrong)
|
.arg(&wrong)
|
||||||
.output()
|
.output()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
assert!(!out.status.success());
|
assert!(!out.status.success());
|
||||||
@@ -422,6 +626,131 @@ fn atomic_output_no_stale_tmp_on_failure() {
|
|||||||
assert!(!tmp.exists(), "temp file must be cleaned up");
|
assert!(!tmp.exists(), "temp file must be cleaned up");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn existing_output_refuses_without_force() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let plain = dir.path().join("p.bin");
|
||||||
|
let ct = dir.path().join("c.bin");
|
||||||
|
fs::write(&plain, b"hello").unwrap();
|
||||||
|
fs::write(&ct, b"existing").unwrap();
|
||||||
|
let out = fcry()
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&plain)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&ct)
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(write_key_file(dir.path()))
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(!out.status.success(), "existing output should refuse");
|
||||||
|
assert_eq!(fs::read(&ct).unwrap(), b"existing");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn force_replaces_only_after_success() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let plain = dir.path().join("p.bin");
|
||||||
|
let ct = dir.path().join("c.bin");
|
||||||
|
fs::write(&plain, b"hello").unwrap();
|
||||||
|
fs::write(&ct, b"existing").unwrap();
|
||||||
|
let out = fcry()
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&plain)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&ct)
|
||||||
|
.arg("--force")
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(write_key_file(dir.path()))
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(
|
||||||
|
out.status.success(),
|
||||||
|
"force encrypt failed: {}",
|
||||||
|
String::from_utf8_lossy(&out.stderr)
|
||||||
|
);
|
||||||
|
assert_ne!(fs::read(&ct).unwrap(), b"existing");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn in_place_replacement_roundtrips() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let path = dir.path().join("data.bin");
|
||||||
|
let original = pseudo_random(41, 50_000);
|
||||||
|
fs::write(&path, &original).unwrap();
|
||||||
|
|
||||||
|
let enc = fcry()
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&path)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&path)
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(write_key_file(dir.path()))
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(
|
||||||
|
enc.status.success(),
|
||||||
|
"in-place encrypt failed: {}",
|
||||||
|
String::from_utf8_lossy(&enc.stderr)
|
||||||
|
);
|
||||||
|
assert_ne!(fs::read(&path).unwrap(), original);
|
||||||
|
|
||||||
|
let dec = fcry()
|
||||||
|
.arg("-d")
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&path)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&path)
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(write_key_file(dir.path()))
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(
|
||||||
|
dec.status.success(),
|
||||||
|
"in-place decrypt failed: {}",
|
||||||
|
String::from_utf8_lossy(&dec.stderr)
|
||||||
|
);
|
||||||
|
assert_eq!(fs::read(&path).unwrap(), original);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn old_predictable_temp_name_input_is_not_truncated() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let input = dir.path().join("out.bin.tmp");
|
||||||
|
let output = dir.path().join("out.bin");
|
||||||
|
let original = pseudo_random(42, 1024);
|
||||||
|
fs::write(&input, &original).unwrap();
|
||||||
|
let out = fcry()
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&input)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&output)
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(write_key_file(dir.path()))
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(
|
||||||
|
out.status.success(),
|
||||||
|
"encrypt failed: {}",
|
||||||
|
String::from_utf8_lossy(&out.stderr)
|
||||||
|
);
|
||||||
|
assert_eq!(fs::read(&input).unwrap(), original);
|
||||||
|
assert!(output.exists());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(unix)]
|
||||||
|
#[test]
|
||||||
|
fn output_file_mode_is_0600() {
|
||||||
|
use std::os::unix::fs::PermissionsExt;
|
||||||
|
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let plain = dir.path().join("p.bin");
|
||||||
|
let ct = dir.path().join("c.bin");
|
||||||
|
fs::write(&plain, b"hello").unwrap();
|
||||||
|
encrypt_file(&plain, &ct, None);
|
||||||
|
let mode = fs::metadata(&ct).unwrap().permissions().mode() & 0o777;
|
||||||
|
assert_eq!(mode, 0o600);
|
||||||
|
}
|
||||||
|
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
// Multi-threaded pipeline + length-committed + random-access tests
|
// Multi-threaded pipeline + length-committed + random-access tests
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
@@ -433,12 +762,13 @@ fn encrypt_file_threads(
|
|||||||
threads: usize,
|
threads: usize,
|
||||||
) {
|
) {
|
||||||
let mut cmd = fcry();
|
let mut cmd = fcry();
|
||||||
|
let key = key_file_near(ct);
|
||||||
cmd.arg("-i")
|
cmd.arg("-i")
|
||||||
.arg(plain)
|
.arg(plain)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(ct)
|
.arg(ct)
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(key)
|
||||||
.arg("-j")
|
.arg("-j")
|
||||||
.arg(threads.to_string());
|
.arg(threads.to_string());
|
||||||
if let Some(cs) = chunk_size {
|
if let Some(cs) = chunk_size {
|
||||||
@@ -453,14 +783,15 @@ fn encrypt_file_threads(
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn decrypt_file_threads(ct: &std::path::Path, rt: &std::path::Path, threads: usize) {
|
fn decrypt_file_threads(ct: &std::path::Path, rt: &std::path::Path, threads: usize) {
|
||||||
|
let key = key_file_near(ct);
|
||||||
let out = fcry()
|
let out = fcry()
|
||||||
.arg("-d")
|
.arg("-d")
|
||||||
.arg("-i")
|
.arg("-i")
|
||||||
.arg(ct)
|
.arg(ct)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(rt)
|
.arg(rt)
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(key)
|
||||||
.arg("-j")
|
.arg("-j")
|
||||||
.arg(threads.to_string())
|
.arg(threads.to_string())
|
||||||
.output()
|
.output()
|
||||||
@@ -517,10 +848,12 @@ fn roundtrip_pipe_multi_threaded() {
|
|||||||
// length when we don't know the input size), but encrypt/decrypt must still
|
// length when we don't know the input size), but encrypt/decrypt must still
|
||||||
// round-trip cleanly across the pipeline.
|
// round-trip cleanly across the pipeline.
|
||||||
let data = pseudo_random(14, 200_000);
|
let data = pseudo_random(14, 200_000);
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let key = write_key_file(dir.path());
|
||||||
|
|
||||||
let mut enc = fcry()
|
let mut enc = fcry()
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(&key)
|
||||||
.arg("-j")
|
.arg("-j")
|
||||||
.arg("4")
|
.arg("4")
|
||||||
.stdin(Stdio::piped())
|
.stdin(Stdio::piped())
|
||||||
@@ -536,16 +869,17 @@ fn roundtrip_pipe_multi_threaded() {
|
|||||||
String::from_utf8_lossy(&enc_out.stderr)
|
String::from_utf8_lossy(&enc_out.stderr)
|
||||||
);
|
);
|
||||||
|
|
||||||
// flags byte at offset 6 must be 0 (no length committed for stdin input).
|
// flags byte at offset 6 must not set length commitment for stdin input.
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
enc_out.stdout[6], 0,
|
enc_out.stdout[6] & 0x01,
|
||||||
|
0,
|
||||||
"stdin-encrypted file unexpectedly committed length"
|
"stdin-encrypted file unexpectedly committed length"
|
||||||
);
|
);
|
||||||
|
|
||||||
let mut dec = fcry()
|
let mut dec = fcry()
|
||||||
.arg("-d")
|
.arg("-d")
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(&key)
|
||||||
.arg("-j")
|
.arg("-j")
|
||||||
.arg("4")
|
.arg("4")
|
||||||
.stdin(Stdio::piped())
|
.stdin(Stdio::piped())
|
||||||
@@ -567,6 +901,104 @@ fn roundtrip_pipe_multi_threaded() {
|
|||||||
assert_eq!(dec_out.stdout, data);
|
assert_eq!(dec_out.stdout, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn stdin_chunk_size_zero_fails_but_empty_valid_chunk_succeeds() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let key = write_key_file(dir.path());
|
||||||
|
let mut bad = fcry()
|
||||||
|
.arg("--chunk-size")
|
||||||
|
.arg("0")
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(&key)
|
||||||
|
.stdin(Stdio::piped())
|
||||||
|
.stdout(Stdio::piped())
|
||||||
|
.stderr(Stdio::piped())
|
||||||
|
.spawn()
|
||||||
|
.unwrap();
|
||||||
|
// Invalid options can make the child exit before it drains stdin.
|
||||||
|
if let Err(err) = bad.stdin.as_mut().unwrap().write_all(b"x") {
|
||||||
|
assert_eq!(
|
||||||
|
err.kind(),
|
||||||
|
ErrorKind::BrokenPipe,
|
||||||
|
"unexpected stdin write error for failing chunk-size 0 process: {err}"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
let bad_out = bad.wait_with_output().unwrap();
|
||||||
|
assert!(!bad_out.status.success(), "chunk-size 0 should fail");
|
||||||
|
|
||||||
|
let mut good = fcry()
|
||||||
|
.arg("--chunk-size")
|
||||||
|
.arg("1")
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(&key)
|
||||||
|
.stdin(Stdio::piped())
|
||||||
|
.stdout(Stdio::piped())
|
||||||
|
.stderr(Stdio::piped())
|
||||||
|
.spawn()
|
||||||
|
.unwrap();
|
||||||
|
drop(good.stdin.take());
|
||||||
|
let good_out = good.wait_with_output().unwrap();
|
||||||
|
assert!(
|
||||||
|
good_out.status.success(),
|
||||||
|
"empty stdin with valid chunk should succeed: {}",
|
||||||
|
String::from_utf8_lossy(&good_out.stderr)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn huge_thread_count_is_bounded() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let plain = dir.path().join("p.bin");
|
||||||
|
let ct = dir.path().join("c.bin");
|
||||||
|
fs::write(&plain, b"hello").unwrap();
|
||||||
|
let out = fcry()
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&plain)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&ct)
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(write_key_file(dir.path()))
|
||||||
|
.arg("-j")
|
||||||
|
.arg("1000000")
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(
|
||||||
|
out.status.success(),
|
||||||
|
"huge -j should be capped, got {}",
|
||||||
|
String::from_utf8_lossy(&out.stderr)
|
||||||
|
);
|
||||||
|
assert!(String::from_utf8_lossy(&out.stderr).contains("capped"));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn forged_huge_chunk_header_fails_before_allocation() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let forged = dir.path().join("forged.bin");
|
||||||
|
let mut bytes = Vec::new();
|
||||||
|
bytes.extend_from_slice(b"fcry");
|
||||||
|
bytes.push(3); // version
|
||||||
|
bytes.push(1); // alg
|
||||||
|
bytes.push(0x02); // key commitment flag
|
||||||
|
bytes.push(0); // reserved
|
||||||
|
bytes.extend_from_slice(&u32::MAX.to_le_bytes());
|
||||||
|
fs::write(&forged, bytes).unwrap();
|
||||||
|
|
||||||
|
let out = fcry()
|
||||||
|
.arg("-d")
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&forged)
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(write_key_file(dir.path()))
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(!out.status.success(), "huge chunk header should fail");
|
||||||
|
assert!(
|
||||||
|
String::from_utf8_lossy(&out.stderr).contains("chunk_size"),
|
||||||
|
"expected chunk_size error, got {}",
|
||||||
|
String::from_utf8_lossy(&out.stderr)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn file_input_commits_length() {
|
fn file_input_commits_length() {
|
||||||
// Encrypting from a regular file must auto-set FLAG_LENGTH_COMMITTED (bit 0
|
// Encrypting from a regular file must auto-set FLAG_LENGTH_COMMITTED (bit 0
|
||||||
@@ -580,8 +1012,39 @@ fn file_input_commits_length() {
|
|||||||
|
|
||||||
let bytes = fs::read(&ct).unwrap();
|
let bytes = fs::read(&ct).unwrap();
|
||||||
// Magic(4) + version(1) + alg(1) + flags(1) = byte 6
|
// Magic(4) + version(1) + alg(1) + flags(1) = byte 6
|
||||||
assert_eq!(bytes[4], 2, "version should be 2");
|
assert_eq!(bytes[4], 3, "version should be 3");
|
||||||
assert_eq!(bytes[6] & 0x01, 0x01, "length-committed flag should be set");
|
assert_eq!(bytes[6] & 0x01, 0x01, "length-committed flag should be set");
|
||||||
|
assert_eq!(bytes[6] & 0x02, 0x02, "key-committed flag should be set");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn v3_downgrade_or_commitment_stripping_fails_authentication() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let plain = dir.path().join("p.bin");
|
||||||
|
let ct = dir.path().join("c.bin");
|
||||||
|
let rt = dir.path().join("r.bin");
|
||||||
|
fs::write(&plain, pseudo_random(51, 1000)).unwrap();
|
||||||
|
encrypt_file(&plain, &ct, None);
|
||||||
|
|
||||||
|
let mut bytes = fs::read(&ct).unwrap();
|
||||||
|
bytes[4] = 2;
|
||||||
|
bytes[6] &= !0x02;
|
||||||
|
fs::write(&ct, bytes).unwrap();
|
||||||
|
|
||||||
|
let out = fcry()
|
||||||
|
.arg("-d")
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&ct)
|
||||||
|
.arg("-o")
|
||||||
|
.arg(&rt)
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(key_file_near(&ct))
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(
|
||||||
|
!out.status.success(),
|
||||||
|
"downgraded/stripped v3 header must fail authentication"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn encrypt_random_access_fixture(
|
fn encrypt_random_access_fixture(
|
||||||
@@ -608,8 +1071,8 @@ fn random_access_decrypt(
|
|||||||
.arg(ct)
|
.arg(ct)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(out)
|
.arg(out)
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(key_file_near(ct))
|
||||||
.arg("--offset")
|
.arg("--offset")
|
||||||
.arg(offset.to_string())
|
.arg(offset.to_string())
|
||||||
.arg("--length")
|
.arg("--length")
|
||||||
@@ -671,10 +1134,11 @@ fn random_access_rejects_stdin_encrypted() {
|
|||||||
let data = pseudo_random(18, 2000);
|
let data = pseudo_random(18, 2000);
|
||||||
let dir = TempDir::new().unwrap();
|
let dir = TempDir::new().unwrap();
|
||||||
let ct = dir.path().join("c.bin");
|
let ct = dir.path().join("c.bin");
|
||||||
|
let key = write_key_file(dir.path());
|
||||||
|
|
||||||
let mut enc = fcry()
|
let mut enc = fcry()
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(&key)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(&ct)
|
.arg(&ct)
|
||||||
.stdin(Stdio::piped())
|
.stdin(Stdio::piped())
|
||||||
@@ -693,14 +1157,13 @@ fn random_access_rejects_stdin_encrypted() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn random_access_zero_length() {
|
fn random_access_rejects_zero_length() {
|
||||||
let dir = TempDir::new().unwrap();
|
let dir = TempDir::new().unwrap();
|
||||||
let data = pseudo_random(19, 1000);
|
let data = pseudo_random(19, 1000);
|
||||||
let ct = encrypt_random_access_fixture(dir.path(), &data, 256);
|
let ct = encrypt_random_access_fixture(dir.path(), &data, 256);
|
||||||
let out = dir.path().join("empty.bin");
|
let out = dir.path().join("empty.bin");
|
||||||
let r = random_access_decrypt(&ct, &out, 500, 0);
|
let r = random_access_decrypt(&ct, &out, 500, 0);
|
||||||
assert!(r.status.success(), "zero-length slice should succeed");
|
assert!(!r.status.success(), "zero-length slice should fail");
|
||||||
assert_eq!(fs::read(&out).unwrap(), Vec::<u8>::new());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@@ -723,6 +1186,33 @@ fn random_access_tampered_length_fails() {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn buffer_verify_stdout_emits_nothing_on_truncated_ciphertext() {
|
||||||
|
let dir = TempDir::new().unwrap();
|
||||||
|
let plain = dir.path().join("p.bin");
|
||||||
|
let ct = dir.path().join("c.bin");
|
||||||
|
fs::write(&plain, pseudo_random(61, 3 * 1024 * 1024)).unwrap();
|
||||||
|
encrypt_file(&plain, &ct, Some(64 * 1024));
|
||||||
|
let mut bytes = fs::read(&ct).unwrap();
|
||||||
|
bytes.truncate(bytes.len() - 32);
|
||||||
|
fs::write(&ct, bytes).unwrap();
|
||||||
|
|
||||||
|
let out = fcry()
|
||||||
|
.arg("-d")
|
||||||
|
.arg("-i")
|
||||||
|
.arg(&ct)
|
||||||
|
.arg("--buffer-verify")
|
||||||
|
.arg("--key-file")
|
||||||
|
.arg(key_file_near(&ct))
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
assert!(!out.status.success(), "truncated decrypt should fail");
|
||||||
|
assert!(
|
||||||
|
out.stdout.is_empty(),
|
||||||
|
"buffer-verify must suppress partial stdout"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn rejects_zero_threads() {
|
fn rejects_zero_threads() {
|
||||||
// -j 0 is almost certainly a user mistake. Clap should reject it before
|
// -j 0 is almost certainly a user mistake. Clap should reject it before
|
||||||
@@ -735,8 +1225,8 @@ fn rejects_zero_threads() {
|
|||||||
.arg(&plain)
|
.arg(&plain)
|
||||||
.arg("-o")
|
.arg("-o")
|
||||||
.arg(dir.path().join("c.bin"))
|
.arg(dir.path().join("c.bin"))
|
||||||
.arg("--raw-key")
|
.arg("--key-file")
|
||||||
.arg(KEY_STR)
|
.arg(write_key_file(dir.path()))
|
||||||
.arg("-j")
|
.arg("-j")
|
||||||
.arg("0")
|
.arg("0")
|
||||||
.output()
|
.output()
|
||||||
|
|||||||
Reference in New Issue
Block a user