Skip to content
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

*: disable CommonName auth for gRPC-gateway #10366

Merged
merged 4 commits into from
Jan 8, 2019
Merged

Conversation

hexfusion
Copy link
Contributor

@hexfusion hexfusion commented Jan 2, 2019

When client-cert-auth is enabled gRPC-gateway proxy request to etcd server will use the CN in client server cert if populated. We should not use this for authentication. In general gRPC-gateway does not support CN authentication and we also should document as such.

The check provided in this PR is based on headers send from gRPC-Gateway

https://github.com/grpc-ecosystem/grpc-gateway/blob/8976e602c518589cb6d40aca52a7dd8aef83706a/runtime/context.go#L183#

/cc @gyuho @mitake

@hexfusion hexfusion added the WIP label Jan 2, 2019
auth/store.go Outdated Show resolved Hide resolved
@gyuho
Copy link
Contributor

gyuho commented Jan 2, 2019

SerialNumber is unique by CA

It's still possible to set the same serial number by providing the template? https://godoc.org/crypto/x509#Certificate

not want to use the CommonName in this client server certificate

Would this affect peer cert validation? We added to 3.3 that we can validate joining peer certs by its common name when etcd --peer-cert-allowed-cn flag is specified.

auth/store.go Outdated
continue
}
ai = &AuthInfo{
Username: chains[0].Subject.CommonName,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@mitake I remember we had conversation about this chains[0], but can't remember what the reasoning was :0

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@gyuho I'll share the conversation after finding it, it is very subtle so I'm also not fully sure about the reasoning...

@hexfusion
Copy link
Contributor Author

It's still possible to set the same serial number by providing the template? https://godoc.org/crypto/x509#Certificate

I will add testing around this to see if it is possible to alter this on the client side. Might need to use finger print or other option if so.

Would this affect peer cert validation? We added to 3.3 that we can validate joining peer certs by its common name when etcd --peer-cert-allowed-cn flag is specified.

This logic should not effect the peer certs as the check is only on client server certs --key-file --cert-file. But I will add tests around this to verify.

@hexfusion hexfusion force-pushed the fx_cn branch 2 times, most recently from c0ecbd4 to 8b9c450 Compare January 3, 2019 00:03
@mitake
Copy link
Contributor

mitake commented Jan 3, 2019

@hexfusion this change will let etcd return errors to clients of gateway even if they provides valid certificate. I think it introduces confusing and hard to diagnose situation (e.g. users might think "my clients must have valid certificate but etcd returns errors"). Letting etcd rise errors as fast as possible would be more user friendly. How do you think? @gyuho @xiang90 @jpbetz

Probably returning an error at the handler layer will be helpful like this: 28265e7

@hexfusion
Copy link
Contributor Author

@mitake thank you for the note and I agree my initial approach was too broad. gRPC-gateway should not include CN in request. Although CN can be included from client request there is no way to forward this CN in a safe way. For example we could include it in ctx metadata but this feels poor solution to support. Since we know when GW makes request and this request is code generated with identifiable headers. We can unset it's CN value only then. This solution in my opinion will allow users to continue with etcd usage as expected for etcd server. gRPC requests will not be directly effected. We can then add note that CN auth is not supported for gRPC-gateway. This also allows users to use authentication with gRPC-gateway while keeping client-cert-auth support for gRPC.

I think this along with your idea of allowing gRPC-gateway to be disabled all together will be a good solution for folks. Do you agree?

@hexfusion hexfusion changed the title *: unset client server CommonName if client-cert-auth is enabled *: unset CommonName for gRPC-gateway request Jan 3, 2019
embed/serve.go Outdated Show resolved Hide resolved
@codecov-io
Copy link

codecov-io commented Jan 3, 2019

Codecov Report

Merging #10366 into master will increase coverage by 0.17%.
The diff coverage is 28.57%.

Impacted file tree graph

@@            Coverage Diff             @@
##           master   #10366      +/-   ##
==========================================
+ Coverage   71.68%   71.85%   +0.17%     
==========================================
  Files         392      392              
  Lines       36421    36664     +243     
==========================================
+ Hits        26107    26345     +238     
+ Misses       8495     8491       -4     
- Partials     1819     1828       +9
Impacted Files Coverage Δ
auth/store.go 74.67% <28.57%> (-0.86%) ⬇️
clientv3/concurrency/election.go 79.68% <0%> (-2.35%) ⬇️
etcdserver/api/rafthttp/msgappv2_codec.go 69.56% <0%> (-1.74%) ⬇️
etcdserver/api/v3rpc/watch.go 82.67% <0%> (-1.64%) ⬇️
etcdctl/ctlv3/command/printer_simple.go 72.48% <0%> (-1.35%) ⬇️
clientv3/retry_interceptor.go 65.46% <0%> (-0.33%) ⬇️
etcdserver/api/v3rpc/lease.go 69.31% <0%> (ø) ⬆️
proxy/grpcproxy/watcher.go 89.79% <0%> (ø) ⬆️
etcdserver/server.go 74.82% <0%> (+0.28%) ⬆️
etcdserver/v3_server.go 79.4% <0%> (+0.45%) ⬆️
... and 15 more

Continue to review full report at Codecov.

Legend - Click here to learn more
Δ = absolute <relative> (impact), ø = not affected, ? = missing data
Powered by Codecov. Last update 8945fec...5e92ca0. Read the comment docs.

@hexfusion
Copy link
Contributor Author

I will improve the tests and would like to break out auth to its own file for curlv3 tests but this I believe covers the issue for now.

auth/store.go Outdated Show resolved Hide resolved
@hexfusion hexfusion changed the title *: unset CommonName for gRPC-gateway request *: disable CommonName auth for gRPC-gateway Jan 4, 2019
@mitake
Copy link
Contributor

mitake commented Jan 5, 2019

LGTM

@hexfusion hexfusion removed the WIP label Jan 7, 2019
@hexfusion
Copy link
Contributor Author

@gyuho PTAL, looks like we should backport this to 3.2 as well?

auth/store.go Outdated
// header. The proxy uses etcd client server certificate. If the certificate
// has a CommonName we should never use this for authentication.
if gw := md["grpcgateway-accept"]; len(gw) > 0 {
return nil
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Add a warning as.lg.Warn as @mitake suggested?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

fixed

Copy link
Contributor

@gyuho gyuho left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

lgtm thanks @hexfusion !

@hexfusion hexfusion merged commit 83c051b into etcd-io:master Jan 8, 2019
@hexfusion hexfusion deleted the fx_cn branch January 8, 2019 17:31
@hexfusion
Copy link
Contributor Author

@gyuho @mitake thank you for reviews and input.

@nelljerram
Copy link

@hexfusion I have just come across this change, in an upgrade from etcd 3.3.10 to 3.3.13.

You wrote above:

This also allows users to use authentication with gRPC-gateway while keeping client-cert-auth support for gRPC

How should a gRPC-gateway client authenticate now, instead of by using the Common Name field? (I tried looking in the etcd docs for this, but could not find it.)

@hexfusion
Copy link
Contributor Author

hexfusion commented Aug 6, 2019

@nelljerram
Copy link

@hexfusion @gyuho I'm sorry for adding again to this merged PR thread, but I do not yet understand what the security issue was that motivated this change, and I would be most grateful if you could help me understand that.

The CVE (https://nvd.nist.gov/vuln/detail/CVE-2018-16886) says:

If an etcd client server TLS certificate contains a Common Name (CN) which matches a valid RBAC username, a remote attacker may authenticate as that user with any valid (trusted) client certificate in a REST API request to the gRPC-gateway.

Why is that a problem? The certificate still has to be signed by a CA that the etcd server trusts. Doesn't that mean that the etcd server can trust the CN ?

In the PR comments above, you write:

gRPC-gateway should not include CN in request.

Please can you explain more? Does this refer to the gRPC request that the gateway builds, and that is equivalent to the received HTTP/JSON request? Why should that not include the CN?

Although CN can be included from client request there is no way to forward this CN in a safe way.

Can you say more about the safety issue here? If I understand correctly, the gateway is in the same process (etcd) as the gRPC server, so there is not even a network hop here; so what could be unsafe?

@DifferentialOrange
Copy link

DifferentialOrange commented Jun 21, 2024

Sending an error with warning-like message "CommonName of client sending a request against gateway will be ignored" sounds as the opposite of user friendliness for me as a user. Common Name is not being ignored, the request is fully rejected if I have a Common Name in my certificate. And I don't get why having a Common Name is such a crime. (To be honest, I also don't get why username is not preferred to a Common Name if username/password explicitly provided in a client request.)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Development

Successfully merging this pull request may close these issues.

6 participants