-
Notifications
You must be signed in to change notification settings - Fork 1
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Stabilize closing mechanism for EncWriter and DecWriter #17
Comments
UpdateCurrently the most promising approach is a quite interesting trait construction.
and a crate-internal non-exported trait in another module - e.g.:
Now, any external type can (in theory) implement the publicly exported
Further Interestingly, at the moment no external type can be used as inner writer for
Now, any type that implements the
Additionally we can provide a |
This commit implements the close-semantic for `EncWriter` and `DecWriter` such that both implement the internal `Close` trait but not the public/exported `Close` trait. Instead both provide a `close(mut self)` method that has the same semantics as the `close` method defined by (both) `Close` traits but takes ownership of the `EncWriter` / `DecWriter`. Further any type (`std::io::Write`) that should be wrapped with an `EncWriter` / `DecWriter` must implement the (internal) `Close` trait. Since this is not possible (the trait is private), we implement the internal `Close` trait for any type that implements the public/exported `Close` trait. For further details see issue: #17. Further this commit adds the `NopCloser` type that can wrap any `std::io::Write` and implements `Close`. This commit also adds some documentation. Update #17
This commit implements the close-semantic for `EncWriter` and `DecWriter` such that both implement the internal `Close` trait but not the public/exported `Close` trait. Instead both provide a `close(mut self)` method that has the same semantics as the `close` method defined by (both) `Close` traits but takes ownership of the `EncWriter` / `DecWriter`. Further any type (`std::io::Write`) that should be wrapped with an `EncWriter` / `DecWriter` must implement the (internal) `Close` trait. Since this is not possible (the trait is private), we implement the internal `Close` trait for any type that implements the public/exported `Close` trait. For further details see issue: #17. Further this commit adds the `NopCloser` type that can wrap any `std::io::Write` and implements `Close`. This commit also adds some documentation. Update #17
This commit implements the close-semantic for `EncWriter` and `DecWriter` such that both implement the internal `Close` trait but not the public/exported `Close` trait. Instead both provide a `close(mut self)` method that has the same semantics as the `close` method defined by (both) `Close` traits but takes ownership of the `EncWriter` / `DecWriter`. Further any type (`std::io::Write`) that should be wrapped with an `EncWriter` / `DecWriter` must implement the (internal) `Close` trait. Since this is not possible (the trait is private), we implement the internal `Close` trait for any type that implements the public/exported `Close` trait. For further details see issue: #17. Further this commit adds the `NopCloser` type that can wrap any `std::io::Write` and implements `Close`. This commit also adds some documentation. Update #17
This commit implements the close-semantic for `EncWriter` and `DecWriter` such that both implement the internal `Close` trait but not the public/exported `Close` trait. Instead both provide a `close(mut self)` method that has the same semantics as the `close` method defined by (both) `Close` traits but takes ownership of the `EncWriter` / `DecWriter`. Further any type (`std::io::Write`) that should be wrapped with an `EncWriter` / `DecWriter` must implement the (internal) `Close` trait. Since this is not possible (the trait is private), we implement the internal `Close` trait for any type that implements the public/exported `Close` trait. For further details see issue: #17. Further this commit adds the `NopCloser` type that can wrap any `std::io::Write` and implements `Close`. This commit also adds some documentation. Update #17
Background Information
The en/decryption of the last fragment differs from all previous fragments since the
first bit of the associated data is flipped from
0
to1
(i.e.aad[0] |= 0x80
). Now, readers andwriters must somehow detect whether the entire plaintext/ciphertext stream has been received -
we don't know whether the (ciphertext) data is authentic but we need to somehow detect the end-of-stream symbol to trigger the "special" logic for the last fragment.
For readers we can assume an EOF after
Ok(0)
. However, for writers the caller has to signal "somehow" that no more data will be written. Therefore, some "close"/"flush" mechanism is necessary.How to signal an end-of-stream for
EncWriter
andDecWriter
In general, there are two main options:
flush
method defined bystd::io::Write
.Optionally, a flush can happen in the destructor - again like
BufWriter
.close
method that signales the end-of-stream.Optionally, the close can happen (if not called explicitly) in the destructor.
Drop
behaviorInvoking the "close" mechanism (if not invoked explicitly) does not seem to be correct.
EncWriter
:If the
EncWriter
is "closed" implicitly (duringdrop
) it is not possible to handle any errorthat happens during writing the final fragment to the underlying writer - the only option would
be to panic. Not handling the error may cause silent data corruption since an incomplete
ciphertext cannot be distinguished from a maliciously truncated one, and therefore, cannot be
decrypted successfully & securely. However, panic'ing when writing to the underlying writer
fails introduces non-deterministic program failures. In general, the destructor should not
panic.
DecWriter
:The same is true for
DecWriter
. In addition, "closing" duringdrop
also fails if the ciphertextstream is not authentic. Not handling that error causes incomplete plaintexts such that an
attacker can mount truncation attacks against programs that use the implicit "close" - either
intentionally or accidentally. On the other side, panic'ing in that case gives an attacker the
ability to mount DoS attacks against such programs.
Therefore:
Now, given that the "close" mechanism must be invoked explicitly, not doing so is a logic bug. Therefore, I suspect that the "correct" behavior (even though unconventional and maybe controversial) is to panic when an
EncWriter
orDecWriter
gets dropped but hasn't been "closed" explicitly. More general, when not "closed" and no previouswrite
call failed."Close" mechanisms
Using
flush
Initially
flush
may seem to be an appropriate way to trigger the en/decryption of the last fragment. However, there are the following issues:flush
takes a (mut
) reference toself
and therefore it's possible to write any combinationof
write
andflush
calls - e.g.:BufWriter
anywrite
that happens afterflush
must failfor
EncWriter
orDecWriter
. Actually, trying to write more data to a "closed" writer is a logicbug.
To prevent this we would need a method takes ownership of the writer - e.g.
flush(mut self) -> io::Result<()>
orclose(mut self) -> io::Result<()>
.Alternatives
There are some alternatives to
flush
. However, there is the fundamental problem that we need to "close" a chain of internal writers sinceio::Write
should be composable:close(mut self) -> io::Result<()>
. However, it would not be possible to invokeit on any inner writer since it does not implement a
close
method.Close
trait with aclose(mut self) -> io::Result<()>
method. Still, this only worksfor the top level
EncWriter
/DecWriter
(if we write a customDrop
implementation).Close
trait with aclose(&mut self) -> io::Result<()>
method and aNopCloser
typethat wraps any type that does not implement
Close
. So far that's the most flexibleapproach since callers can provide custom implementations of
Close
such that we can havea chain of
close
calls. However, a caller can still e.g. write toEncWriter
/DecWriter
afterclose
. We could define a separateclose(mut self) -> io::Result<()>
method forEncWriter
and
DecWriter
such that specialization will select theclose
that takes ownership. However,callers can still do the wrong thing by calling
Close::close(w)
and than write tow
. TheClose
trait documentation can also help by indicating theClose
should be implemented butnever directly used. Anyway, even though the incorrect example may be artificial an ideal
solution would be to proof in the type system that such an error cannot be made by
callers.
The text was updated successfully, but these errors were encountered: