Rewrite AndroidSyncSettings to be significantly simpler.
[chromium-blink-merge.git] / net / quic / crypto / proof_test.cc
blob4407b93e9c2dcb3956e469f5bf3452b0811f57cc
1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "base/files/file_path.h"
6 #include "net/base/ip_endpoint.h"
7 #include "net/base/net_errors.h"
8 #include "net/base/test_completion_callback.h"
9 #include "net/base/test_data_directory.h"
10 #include "net/cert/cert_status_flags.h"
11 #include "net/cert/cert_verify_result.h"
12 #include "net/cert/x509_certificate.h"
13 #include "net/quic/crypto/proof_source.h"
14 #include "net/quic/crypto/proof_verifier.h"
15 #include "net/quic/test_tools/crypto_test_utils.h"
16 #include "net/test/cert_test_util.h"
17 #include "testing/gtest/include/gtest/gtest.h"
19 #if defined(OS_WIN)
20 #include "base/win/windows_version.h"
21 #endif
23 using std::string;
24 using std::vector;
26 namespace net {
27 namespace test {
28 namespace {
30 // TestProofVerifierCallback is a simple callback for a ProofVerifier that
31 // signals a TestCompletionCallback when called and stores the results from the
32 // ProofVerifier in pointers passed to the constructor.
33 class TestProofVerifierCallback : public ProofVerifierCallback {
34 public:
35 TestProofVerifierCallback(TestCompletionCallback* comp_callback,
36 bool* ok,
37 string* error_details)
38 : comp_callback_(comp_callback),
39 ok_(ok),
40 error_details_(error_details) {}
42 void Run(bool ok,
43 const string& error_details,
44 scoped_ptr<ProofVerifyDetails>* details) override {
45 *ok_ = ok;
46 *error_details_ = error_details;
48 comp_callback_->callback().Run(0);
51 private:
52 TestCompletionCallback* const comp_callback_;
53 bool* const ok_;
54 string* const error_details_;
57 // RunVerification runs |verifier->VerifyProof| and asserts that the result
58 // matches |expected_ok|.
59 void RunVerification(ProofVerifier* verifier,
60 const string& hostname,
61 const string& server_config,
62 const vector<string>& certs,
63 const string& proof,
64 bool expected_ok) {
65 scoped_ptr<ProofVerifyDetails> details;
66 TestCompletionCallback comp_callback;
67 bool ok;
68 string error_details;
69 scoped_ptr<ProofVerifyContext> verify_context(
70 CryptoTestUtils::ProofVerifyContextForTesting());
71 TestProofVerifierCallback* callback =
72 new TestProofVerifierCallback(&comp_callback, &ok, &error_details);
74 QuicAsyncStatus status = verifier->VerifyProof(
75 hostname, server_config, certs, proof, verify_context.get(),
76 &error_details, &details, callback);
78 switch (status) {
79 case QUIC_FAILURE:
80 delete callback;
81 ASSERT_FALSE(expected_ok);
82 ASSERT_NE("", error_details);
83 return;
84 case QUIC_SUCCESS:
85 delete callback;
86 ASSERT_TRUE(expected_ok);
87 ASSERT_EQ("", error_details);
88 return;
89 case QUIC_PENDING:
90 comp_callback.WaitForResult();
91 ASSERT_EQ(expected_ok, ok);
92 break;
96 // Reads the certificate named "quic_" + |file_name| in the test data directory.
97 // The certificate must be PEM encoded. Returns the DER-encoded certificate.
98 string LoadTestCert(const string& file_name) {
99 base::FilePath certs_dir = GetTestCertsDirectory();
100 scoped_refptr<X509Certificate> cert =
101 ImportCertFromFile(certs_dir, "quic_" + file_name);
102 CHECK_NE(static_cast<X509Certificate*>(nullptr), cert.get());
104 string der_bytes;
105 CHECK(X509Certificate::GetDEREncoded(cert->os_cert_handle(), &der_bytes));
106 return der_bytes;
109 } // namespace
111 // TODO(rtenneti): Enable testing of ProofVerifier.
112 TEST(ProofTest, DISABLED_Verify) {
113 scoped_ptr<ProofSource> source(CryptoTestUtils::ProofSourceForTesting());
114 scoped_ptr<ProofVerifier> verifier(
115 CryptoTestUtils::ProofVerifierForTesting());
117 const string server_config = "server config bytes";
118 const string hostname = "test.example.com";
119 const vector<string>* certs;
120 const vector<string>* first_certs;
121 string error_details, signature, first_signature;
122 IPEndPoint server_ip;
124 ASSERT_TRUE(source->GetProof(server_ip, hostname, server_config,
125 false /* no ECDSA */, &first_certs,
126 &first_signature));
127 ASSERT_TRUE(source->GetProof(server_ip, hostname, server_config,
128 false /* no ECDSA */, &certs, &signature));
130 // Check that the proof source is caching correctly:
131 ASSERT_EQ(first_certs, certs);
132 ASSERT_EQ(signature, first_signature);
134 RunVerification(
135 verifier.get(), hostname, server_config, *certs, signature, true);
137 RunVerification(
138 verifier.get(), "foo.com", server_config, *certs, signature, false);
140 RunVerification(
141 verifier.get(), server_config.substr(1, string::npos), server_config,
142 *certs, signature, false);
144 const string corrupt_signature = "1" + signature;
145 RunVerification(
146 verifier.get(), hostname, server_config, *certs, corrupt_signature,
147 false);
149 vector<string> wrong_certs;
150 for (size_t i = 1; i < certs->size(); i++) {
151 wrong_certs.push_back((*certs)[i]);
153 RunVerification(
154 verifier.get(), "foo.com", server_config, wrong_certs, corrupt_signature,
155 false);
158 // A known answer test that allows us to test ProofVerifier without a working
159 // ProofSource.
160 TEST(ProofTest, VerifyRSAKnownAnswerTest) {
161 // These sample signatures were generated by running the Proof.Verify test
162 // and dumping the bytes of the |signature| output of ProofSource::GetProof().
163 static const unsigned char signature_data_0[] = {
164 0x31, 0xd5, 0xfb, 0x40, 0x30, 0x75, 0xd2, 0x7d, 0x61, 0xf9, 0xd7, 0x54,
165 0x30, 0x06, 0xaf, 0x54, 0x0d, 0xb0, 0x0a, 0xda, 0x63, 0xca, 0x7e, 0x9e,
166 0xce, 0xba, 0x10, 0x05, 0x1b, 0xa6, 0x7f, 0xef, 0x2b, 0xa3, 0xff, 0x3c,
167 0xbb, 0x9a, 0xe4, 0xbf, 0xb8, 0x0c, 0xc1, 0xbd, 0xed, 0xc2, 0x90, 0x68,
168 0xeb, 0x45, 0x48, 0xea, 0x3c, 0x95, 0xf8, 0xa2, 0xb9, 0xe7, 0x62, 0x29,
169 0x00, 0xc3, 0x18, 0xb4, 0x16, 0x6f, 0x5e, 0xb0, 0xc1, 0x26, 0xc0, 0x4b,
170 0x84, 0xf5, 0x97, 0xfc, 0x17, 0xf9, 0x1c, 0x43, 0xb8, 0xf2, 0x3f, 0x38,
171 0x32, 0xad, 0x36, 0x52, 0x2c, 0x26, 0x92, 0x7a, 0xea, 0x2c, 0xa2, 0xf4,
172 0x28, 0x2f, 0x19, 0x4d, 0x1f, 0x11, 0x46, 0x82, 0xd0, 0xc4, 0x86, 0x56,
173 0x5c, 0x97, 0x9e, 0xc6, 0x37, 0x8e, 0xaf, 0x9d, 0x69, 0xe9, 0x4f, 0x5a,
174 0x6d, 0x70, 0x75, 0xc7, 0x41, 0x95, 0x68, 0x53, 0x94, 0xca, 0x31, 0x63,
175 0x61, 0x9f, 0xb8, 0x8c, 0x3b, 0x75, 0x36, 0x8b, 0x69, 0xa2, 0x35, 0xc0,
176 0x4b, 0x77, 0x55, 0x08, 0xc2, 0xb4, 0x56, 0xd2, 0x81, 0xce, 0x9e, 0x25,
177 0xdb, 0x50, 0x74, 0xb3, 0x8a, 0xd9, 0x20, 0x42, 0x3f, 0x85, 0x2d, 0xaa,
178 0xfd, 0x66, 0xfa, 0xd6, 0x95, 0x55, 0x6b, 0x63, 0x63, 0x04, 0xf8, 0x6c,
179 0x3e, 0x08, 0x22, 0x39, 0xb9, 0x9a, 0xe0, 0xd7, 0x01, 0xff, 0xeb, 0x8a,
180 0xb9, 0xe2, 0x34, 0xa5, 0xa0, 0x51, 0xe9, 0xbe, 0x15, 0x12, 0xbf, 0xbe,
181 0x64, 0x3d, 0x3f, 0x98, 0xce, 0xc1, 0xa6, 0x33, 0x32, 0xd3, 0x5c, 0xa8,
182 0x39, 0x93, 0xdc, 0x1c, 0xb9, 0xab, 0x3c, 0x80, 0x62, 0xb3, 0x76, 0x21,
183 0xdf, 0x47, 0x1e, 0xa9, 0x0e, 0x5e, 0x8a, 0xbe, 0x66, 0x5b, 0x7c, 0x21,
184 0xfa, 0x78, 0x2d, 0xd1, 0x1d, 0x5c, 0x35, 0x8a, 0x34, 0xb2, 0x1a, 0xc2,
185 0xc4, 0x4b, 0x53, 0x54,
187 static const unsigned char signature_data_1[] = {
188 0x01, 0x7b, 0x52, 0x35, 0xe3, 0x51, 0xdd, 0xf1, 0x67, 0x8d, 0x31, 0x5e,
189 0xa3, 0x75, 0x1f, 0x68, 0x6c, 0xdd, 0x41, 0x7a, 0x18, 0x25, 0xe0, 0x12,
190 0x6e, 0x84, 0x46, 0x5e, 0xb2, 0x98, 0xd7, 0x84, 0xe1, 0x62, 0xe0, 0xc1,
191 0xc4, 0xd7, 0x4f, 0x4f, 0x80, 0xc1, 0x92, 0xd6, 0x02, 0xaf, 0xca, 0x28,
192 0x9f, 0xe0, 0xf3, 0x74, 0xd7, 0xf1, 0x44, 0x67, 0x59, 0x27, 0xc8, 0xc2,
193 0x8b, 0xd4, 0xe5, 0x4a, 0x07, 0xfd, 0x00, 0xd6, 0x8a, 0xbf, 0x8b, 0xcd,
194 0x6a, 0xe0, 0x1d, 0xf6, 0x4b, 0x68, 0x0f, 0xcf, 0xb9, 0xd0, 0xa1, 0xbc,
195 0x2e, 0xcf, 0x7c, 0x03, 0x47, 0x11, 0xe4, 0x4c, 0xbc, 0x1b, 0x6b, 0xa5,
196 0x2a, 0x82, 0x86, 0xa4, 0x7f, 0x1d, 0x85, 0x64, 0x21, 0x10, 0xd2, 0xb2,
197 0xa0, 0x31, 0xa2, 0x78, 0xe6, 0xf2, 0xea, 0x96, 0x38, 0x8c, 0x9a, 0xe1,
198 0x01, 0xab, 0x8e, 0x95, 0x66, 0xc8, 0xe5, 0xcc, 0x80, 0xa3, 0xbd, 0x16,
199 0xa7, 0x79, 0x19, 0x39, 0x61, 0x3d, 0xff, 0x37, 0xca, 0x9f, 0x97, 0x05,
200 0xc7, 0xcb, 0xf0, 0xea, 0xaf, 0x64, 0x07, 0xc0, 0xed, 0x2a, 0x98, 0xa4,
201 0xaf, 0x04, 0x6f, 0xf2, 0xc9, 0xb2, 0x73, 0x9a, 0x56, 0x85, 0x43, 0x64,
202 0x5f, 0xaa, 0xb7, 0xff, 0x31, 0x4c, 0x2e, 0x6c, 0x17, 0xcf, 0xe5, 0xbe,
203 0x7f, 0x7e, 0xad, 0xf5, 0x6f, 0x84, 0x50, 0x20, 0x29, 0xb3, 0x57, 0xe7,
204 0xb1, 0xdc, 0x2c, 0x95, 0x48, 0xfe, 0xb0, 0xc1, 0x92, 0xda, 0xc5, 0x58,
205 0x95, 0xb0, 0x1a, 0x3a, 0x05, 0x71, 0x3c, 0x6d, 0x20, 0x01, 0x4c, 0xa9,
206 0xe4, 0x38, 0x08, 0x65, 0xb4, 0xbd, 0x86, 0x76, 0xbd, 0xad, 0x25, 0x06,
207 0x74, 0x0b, 0xca, 0x95, 0x27, 0x0c, 0x13, 0x08, 0x7e, 0x30, 0xcf, 0xf6,
208 0xb5, 0xc1, 0x2a, 0x08, 0xfc, 0x4b, 0xc6, 0xb5, 0x2f, 0x23, 0x27, 0x32,
209 0x89, 0xdb, 0x0e, 0x4a,
211 static const unsigned char signature_data_2[] = {
212 0x6d, 0x7d, 0x22, 0x8c, 0x85, 0xc4, 0x8a, 0x80, 0x05, 0xe4, 0x3c, 0xaf,
213 0x10, 0x3b, 0xe3, 0x51, 0xb1, 0x86, 0x52, 0x63, 0xb6, 0x17, 0x33, 0xbd,
214 0x1b, 0x1e, 0xc4, 0x50, 0x10, 0xfc, 0xcc, 0xea, 0x6b, 0x11, 0xeb, 0x6d,
215 0x5e, 0x00, 0xe7, 0xf3, 0x67, 0x99, 0x74, 0x53, 0x12, 0x8f, 0xe4, 0x3e,
216 0x20, 0x17, 0x8e, 0x83, 0xe6, 0xdc, 0x83, 0x91, 0x0e, 0xf3, 0x69, 0x22,
217 0x95, 0x14, 0xdf, 0xc1, 0xda, 0xb5, 0xdb, 0x6a, 0x1a, 0xb4, 0x4f, 0x26,
218 0xd0, 0x32, 0x1d, 0x73, 0x95, 0x1f, 0x39, 0x1d, 0x00, 0xcb, 0xc3, 0x92,
219 0x49, 0x53, 0xcb, 0x5c, 0x36, 0x70, 0x19, 0xd9, 0x64, 0x36, 0xda, 0xfb,
220 0x20, 0xe5, 0x47, 0xd9, 0x08, 0xc6, 0x5a, 0x9e, 0x87, 0x1a, 0xdb, 0x11,
221 0x7b, 0x17, 0xfc, 0x53, 0x7b, 0xc1, 0xa0, 0xc0, 0x33, 0xcf, 0x96, 0xba,
222 0x03, 0x79, 0x8e, 0xc6, 0x05, 0xd2, 0xb7, 0xa2, 0xe2, 0xc1, 0x67, 0xb7,
223 0x6a, 0xeb, 0xb1, 0x40, 0xbb, 0x7d, 0x57, 0xcb, 0xc2, 0x60, 0x9f, 0xf1,
224 0x72, 0xe5, 0xad, 0xce, 0x95, 0x45, 0x7c, 0xbc, 0x75, 0x81, 0x45, 0x19,
225 0xe1, 0xa7, 0x2f, 0x05, 0x52, 0xeb, 0xed, 0xdd, 0x19, 0xd9, 0x1a, 0xc9,
226 0x5a, 0x06, 0x8e, 0x29, 0x54, 0xb5, 0x4f, 0x80, 0xaa, 0x36, 0x36, 0xc0,
227 0xff, 0x64, 0xac, 0xe8, 0x0f, 0x99, 0x35, 0x5e, 0xc6, 0x72, 0x1f, 0x8c,
228 0xc4, 0x2b, 0x7d, 0xc1, 0xfb, 0xf0, 0x12, 0x61, 0xb1, 0x18, 0x65, 0xdd,
229 0xc2, 0x38, 0x92, 0xba, 0x84, 0xf8, 0xc8, 0x5e, 0x17, 0x63, 0xe0, 0x9c,
230 0x2c, 0xe6, 0x70, 0x71, 0xdc, 0xe5, 0xc1, 0xea, 0xb3, 0x9a, 0xb6, 0x91,
231 0xdc, 0xc5, 0x56, 0x84, 0x8a, 0x31, 0x31, 0x23, 0x61, 0x94, 0x7e, 0x01,
232 0x22, 0x49, 0xf3, 0xcb, 0x0e, 0x31, 0x03, 0x04, 0x1b, 0x14, 0x43, 0x7c,
233 0xad, 0x42, 0xe5, 0x55,
236 scoped_ptr<ProofVerifier> verifier(
237 CryptoTestUtils::ProofVerifierForTesting());
239 const string server_config = "server config bytes";
240 const string hostname = "test.example.com";
242 vector<string> certs(2);
243 certs[0] = LoadTestCert("test.example.com.crt");
244 certs[1] = LoadTestCert("intermediate.crt");
246 // Signatures are nondeterministic, so we test multiple signatures on the
247 // same server_config.
248 vector<string> signatures(3);
249 signatures[0].assign(reinterpret_cast<const char*>(signature_data_0),
250 sizeof(signature_data_0));
251 signatures[1].assign(reinterpret_cast<const char*>(signature_data_1),
252 sizeof(signature_data_1));
253 signatures[2].assign(reinterpret_cast<const char*>(signature_data_2),
254 sizeof(signature_data_2));
256 for (size_t i = 0; i < signatures.size(); i++) {
257 const string& signature = signatures[i];
259 RunVerification(
260 verifier.get(), hostname, server_config, certs, signature, true);
261 RunVerification(
262 verifier.get(), "foo.com", server_config, certs, signature, false);
263 RunVerification(
264 verifier.get(), hostname, server_config.substr(1, string::npos),
265 certs, signature, false);
267 const string corrupt_signature = "1" + signature;
268 RunVerification(
269 verifier.get(), hostname, server_config, certs, corrupt_signature,
270 false);
272 vector<string> wrong_certs;
273 for (size_t i = 1; i < certs.size(); i++) {
274 wrong_certs.push_back(certs[i]);
276 RunVerification(verifier.get(), hostname, server_config, wrong_certs,
277 signature, false);
281 // A known answer test that allows us to test ProofVerifier without a working
282 // ProofSource.
283 TEST(ProofTest, VerifyECDSAKnownAnswerTest) {
284 // Disable this test on platforms that do not support ECDSA certificates.
285 #if defined(OS_WIN)
286 if (base::win::GetVersion() < base::win::VERSION_VISTA)
287 return;
288 #endif
290 // These sample signatures were generated by running the Proof.Verify test
291 // (modified to use ECDSA for signing proofs) and dumping the bytes of the
292 // |signature| output of ProofSource::GetProof().
293 static const unsigned char signature_data_0[] = {
294 0x30, 0x45, 0x02, 0x21, 0x00, 0x89, 0xc4, 0x7d, 0x08, 0xd1, 0x49, 0x19,
295 0x6c, 0xd1, 0x7c, 0xb9, 0x25, 0xe0, 0xe3, 0xbd, 0x6a, 0x5c, 0xd7, 0xaa,
296 0x0c, 0xdc, 0x4f, 0x8e, 0xeb, 0xde, 0xbf, 0x32, 0xf8, 0xd1, 0x84, 0x95,
297 0x97, 0x02, 0x20, 0x29, 0x3d, 0x49, 0x22, 0x73, 0xed, 0x8b, 0xde, 0x3d,
298 0xc2, 0xa4, 0x20, 0xcc, 0xe7, 0xc8, 0x2a, 0x85, 0x20, 0x9b, 0x5b, 0xda,
299 0xcd, 0x58, 0x23, 0xbe, 0x89, 0x73, 0x31, 0x87, 0x51, 0xd1, 0x01,
301 static const unsigned char signature_data_1[] = {
302 0x30, 0x46, 0x02, 0x21, 0x00, 0xec, 0xdf, 0x69, 0xc8, 0x24, 0x59, 0x93,
303 0xda, 0x49, 0xee, 0x37, 0x28, 0xaf, 0xeb, 0x0e, 0x2f, 0x80, 0x17, 0x4b,
304 0x3b, 0xf6, 0x54, 0xcd, 0x3b, 0x86, 0xc5, 0x98, 0x0d, 0xff, 0xc6, 0xb1,
305 0xe7, 0x02, 0x21, 0x00, 0xe1, 0x36, 0x8c, 0xc0, 0xf4, 0x50, 0x5f, 0xba,
306 0xfb, 0xe2, 0xff, 0x1d, 0x5d, 0x64, 0xe4, 0x07, 0xbb, 0x5a, 0x4b, 0x19,
307 0xb6, 0x39, 0x7a, 0xc4, 0x12, 0xc6, 0xe5, 0x42, 0xc8, 0x78, 0x33, 0xcd,
309 static const unsigned char signature_data_2[] = {
310 0x30, 0x45, 0x02, 0x20, 0x09, 0x51, 0xe9, 0xde, 0xdb, 0x01, 0xfd, 0xb4,
311 0xd8, 0x20, 0xbb, 0xad, 0x41, 0xe3, 0xaa, 0xe7, 0xa3, 0xc3, 0x32, 0x10,
312 0x9d, 0xfa, 0x37, 0xce, 0x17, 0xd1, 0x29, 0xf9, 0xd4, 0x1d, 0x0d, 0x19,
313 0x02, 0x21, 0x00, 0xc6, 0x20, 0xd4, 0x28, 0xf9, 0x70, 0xb5, 0xb4, 0xff,
314 0x4a, 0x35, 0xba, 0xa0, 0xf2, 0x8e, 0x00, 0xf7, 0xcb, 0x43, 0xaf, 0x2d,
315 0x1f, 0xce, 0x92, 0x05, 0xca, 0x29, 0xfe, 0xd2, 0x8f, 0xd9, 0x31,
318 scoped_ptr<ProofVerifier> verifier(
319 CryptoTestUtils::ProofVerifierForTesting());
321 const string server_config = "server config bytes";
322 const string hostname = "test.example.com";
324 vector<string> certs(2);
325 certs[0] = LoadTestCert("test_ecc.example.com.crt");
326 certs[1] = LoadTestCert("intermediate.crt");
328 // Signatures are nondeterministic, so we test multiple signatures on the
329 // same server_config.
330 vector<string> signatures(3);
331 signatures[0].assign(reinterpret_cast<const char*>(signature_data_0),
332 sizeof(signature_data_0));
333 signatures[1].assign(reinterpret_cast<const char*>(signature_data_1),
334 sizeof(signature_data_1));
335 signatures[2].assign(reinterpret_cast<const char*>(signature_data_2),
336 sizeof(signature_data_2));
338 for (size_t i = 0; i < signatures.size(); i++) {
339 const string& signature = signatures[i];
341 RunVerification(
342 verifier.get(), hostname, server_config, certs, signature, true);
343 RunVerification(
344 verifier.get(), "foo.com", server_config, certs, signature, false);
345 RunVerification(
346 verifier.get(), hostname, server_config.substr(1, string::npos),
347 certs, signature, false);
349 // An ECDSA signature is DER-encoded. Corrupt the last byte so that the
350 // signature can still be DER-decoded correctly.
351 string corrupt_signature = signature;
352 corrupt_signature[corrupt_signature.size() - 1] += 1;
353 RunVerification(
354 verifier.get(), hostname, server_config, certs, corrupt_signature,
355 false);
357 // Prepending a "1" makes the DER invalid.
358 const string bad_der_signature1 = "1" + signature;
359 RunVerification(
360 verifier.get(), hostname, server_config, certs, bad_der_signature1,
361 false);
363 vector<string> wrong_certs;
364 for (size_t i = 1; i < certs.size(); i++) {
365 wrong_certs.push_back(certs[i]);
367 RunVerification(
368 verifier.get(), hostname, server_config, wrong_certs, signature,
369 false);
373 } // namespace test
374 } // namespace net