Address em-http-request warnings about verify_peer
#524
Description
EventMachine supports making TCP connections via the
EM.connect
method. Once the connection is established, the
client can initiate a TLS session over the socket by calling
EM::Connection#start_tls
. This method does not verify the
server's identity by default, but it does accept an option named verify_peer
.
If this is set, the connection will invoke
EM::Connection#ssl_verify_peer
, a method supplied by the
caller which performs certificate validation. That is, EventMachine does not
implement such a validation routine itself; it requires the caller to supply
one.
Faye makes use of two libraries that use this interface:
em-http-request and faye-websocket. The latter is
maintained by the Faye project and the former by a third party.
In June 2020 a security advisory was published relating to the
use of of this EventMachine interface in em-http-request. This followed an
issue reporting the problem and a pull request to fix it.
The em-http-request maintainers decided to address this problem by publishing a
patch-version release, version 1.1.6, making the following changes:
- Importing an implementation of
ssl_verify_peer
from
the Faraday library (by copying the code, not by linking to this
package) - Sending
verify_peer: true
to EventMachine, only if instructed by the caller - Logging a warning if the caller does not ask for peer verification
This course of action was chosen in order to alert existing users to insecure
behaviour, and giving them an easy remediation for it; enabling peer
verification by default may break some existing legitimate clients, for example
clients talking to trusted servers using self-signed certificates.
It is worth noting a couple of things about the ssl_verify_peer
implementation
imported into em-http-request from Faraday. First, Faraday is a common interface
over several different Ruby HTTP clients, and em-http-request is one of its
supported adapters. Faraday enables verify_peer
by default, and therefore
patches em-http-request at runtime to support this. So, this patch exists
because Faraday wants peer verification to be default behaviour, and they
patched em-http-request to make it possible.
Second, the implementation that em-http-request has now imported is the very
first version of this code committed to Faraday's codebase in July 2013,
released in version 0.9.0 in January 2014. This code has been modified a few
times since, most recently in October 2019 before the release of Faraday 1.0.0
in January 2020.
It is not clear why or how this version of the ssl_verify_peer
code was
selected. A cursory inspection of the imported code against the latest version
in Faraday, and reading the commit history of this file,
indicates the code is doing essentially the same thing in terms of calls to the
OpenSSL module, but has been refactored to clarify control flow and to comply
with linter warnings. There has been one bug fix made to the
original code to use the hostname from the request URI, not the connection
hostname, to verify the certificate (these may differ if the request is sent via
a proxy). The code imported into em-http-request does not include this fix.
The Faye project needs to decide how to address this problem. Users that have
installed the latest em-http-request are now seeing warnings in their logs that
verify_peer
is not set. No other functional difference has taken place in
these programs; they were already not doing peer verification, what's changed is
that their authors now know about it.
There are a few things to take into consideration here. The least-effort change
Faye could make would be to send verify_peer: true
to em-http-request, causing
it to verifiy server certificates and removing the warning from users' logs.
This would have the effect that the initial request to the server would now be
verified. However, since faye-websocket suffers from the same non-verification
problem, the Faye client would be making an unverified connection if it
subsequently switched to using WebSocket. This suggests we should have
feature parity between these two network transports so that Faye as a whole can
make a consistent guarantee to users; enabling verification for normal HTTP but
not for WebSocket, and taking no further measures, would give users a false
sense of security.
This prompts the question of what changes faye-websocket should make to gain
parity. In the Node.js version of Faye, we rely on the default behaviour of the
https
and tls
modules, which is to verify certificates unless explicitly
told otherwise. However, sometimes we want to connect to servers whose
certificates are not recognised by the usual system certificate authority (CA),
for example servers with self-signed certificates, even if only for testing
purposes. Node provides a mechanism for this: rather than simply turning
verification off with the rejectUnauthorized: false
option, one can use the
ca
option to pass one's own set of certificate authorities, rather than using
the system one. Thus, a client can still trust a server it knows the certificate
of, without opting out of verification entirely and exposing itself to
person-in-the-middle attacks.
The ssl_verify_peer
implementation in em-http-request does not support this;
it only enables the default paths for OpenSSL::X509::Store
and does not let
the user set their own. This means that to talk to a server with a self-signed
cert one would have to leave verify_peer
unset. (Note that explicitly setting
verify_peer: false
rather than merely leaving it unset still causes the
warning to be logged.)
So, even if faye-websocket were to add the ability to set custom CAs, Faye would
not be able offer the same ability -- we're limited by the functionality offered
by em-http-request. That said, I would still prefer for faye-websocket in Ruby
to work like its Node counterpart, and for Faye to get as close as we can to
that. i.e. it should set verify_peer
by default, allow it to be unset, and
provide a way to supply your own CAs, which should be preferred over unsetting
verify_peer
.
This leaves two remaining problems: implementing the required functionality in
faye-websocket, and supporting verify_peer
in Faye itself. If both
em-http-request and faye-websocket support setting verify_peer
, then Faye can
make use of this functionality. em-http-request defaults to not setting this
option so we should think about the effect if we were to set it by default. If
faye-websocket defaults to enabling verify_peer
, and Faye doesn't have any
defined behaviour other than to forward the caller's options to each library,
then we'll get different behaviour depending on whether we're using HTTP or
WebSocket, which is undesirable. Whichever default we choose, a later release of
em-http-request might change its behaviour. So either way, Faye needs to have an
explicit policy about the default value of verify_peer
to send to these
libraries if this option is not set by the caller.
At present (in version 1.3.0) Faye does not support the caller setting any TLS
options that are sent to em-http-request. The only TLS-related option it
currently sends is sni_hostname
, which it gets from the request URL. As
clients currently do not have any control over the setting of this option, they
will be getting the same behaviour as if they'd set verify_peer: false
.
Therefore, defaulting to verify_peer: true
may cause a change in behaviour
that could constitute a breaking change. There are broadly two categories of
situation in which a Faye client would break if verify_peer
were enabled:
- The client is not talking to the server it thinks it is, because verification
is turned off and the client is being attacked - The client is intentionally talking to a server that the system CAs would not
recognised, for example a server with a self-signed cert
I would consider the first situation a bug: this is not intentional usage and
the client would benefit from being alerted to the fact it's talking to an
untrusted server. Setting verify_peer
would mean fixing the bug and is
therefore not a breaking change.
The second situation is intentional and supported usage, and if we set
verify_peer
by default then these clients will stop working. We need to
provide these clients with a way to start working again, and would therefore
need to expose the verify_peer
option so they can switch it off. This is not
an ideal solution but is the only one open to us given the current functionality
exposed in em-http-request. Exposing this option would constitute a new feature,
and breaking these clients in the first place may constitute a breaking change.
If Faye defaults to verify_peer: false
, then no existing clients would change
their behaviour (which may regarded as a bug per the above), and we would have
to expose the ability to enable verify_peer
. This gives us the option of
introducing the option as a new feature without a breaking change, but this
would still be at least a minor release, not a patch.
As regards the implementation, I have opened a PR against
faye-websocket containing an implementation to add support for verify_peer
and
cert_authority_file
. Details and discussion of the implementation should
happen on that PR rather than this issue. I am opening this issue to discuss the
overall strategy and how everything needs to fit together in Faye.