2
|
1 |
/*
|
3462
|
2 |
* Copyright 2005-2009 Sun Microsystems, Inc. All Rights Reserved.
|
2
|
3 |
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
4 |
*
|
|
5 |
* This code is free software; you can redistribute it and/or modify it
|
|
6 |
* under the terms of the GNU General Public License version 2 only, as
|
|
7 |
* published by the Free Software Foundation.
|
|
8 |
*
|
|
9 |
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
10 |
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
11 |
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
12 |
* version 2 for more details (a copy is included in the LICENSE file that
|
|
13 |
* accompanied this code).
|
|
14 |
*
|
|
15 |
* You should have received a copy of the GNU General Public License version
|
|
16 |
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
17 |
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
18 |
*
|
|
19 |
* Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
|
20 |
* CA 95054 USA or visit www.sun.com if you need additional information or
|
|
21 |
* have any questions.
|
|
22 |
*/
|
|
23 |
|
|
24 |
/**
|
|
25 |
* @test
|
3462
|
26 |
* @bug 4635230 6365103 6366054 6824440
|
2
|
27 |
* @summary Basic unit tests for validating XML Signatures with JSR 105
|
|
28 |
* @compile -XDignore.symbol.file KeySelectors.java SignatureValidator.java
|
|
29 |
* X509KeySelector.java ValidationTests.java
|
|
30 |
* @run main ValidationTests
|
|
31 |
* @author Sean Mullan
|
|
32 |
*/
|
|
33 |
import java.io.File;
|
|
34 |
import java.io.FileInputStream;
|
|
35 |
import java.security.*;
|
|
36 |
import javax.xml.crypto.Data;
|
|
37 |
import javax.xml.crypto.KeySelector;
|
|
38 |
import javax.xml.crypto.OctetStreamData;
|
|
39 |
import javax.xml.crypto.URIDereferencer;
|
|
40 |
import javax.xml.crypto.URIReference;
|
|
41 |
import javax.xml.crypto.URIReferenceException;
|
|
42 |
import javax.xml.crypto.XMLCryptoContext;
|
3462
|
43 |
import javax.xml.crypto.dsig.XMLSignatureException;
|
2
|
44 |
import javax.xml.crypto.dsig.XMLSignatureFactory;
|
|
45 |
|
|
46 |
/**
|
|
47 |
* This is a testcase to validate all "merlin-xmldsig-twenty-three"
|
|
48 |
* testcases from Baltimore
|
|
49 |
*/
|
|
50 |
public class ValidationTests {
|
|
51 |
|
|
52 |
private static SignatureValidator validator;
|
|
53 |
private final static String DIR = System.getProperty("test.src", ".");
|
|
54 |
private final static String DATA_DIR =
|
|
55 |
DIR + System.getProperty("file.separator") + "data";
|
|
56 |
private final static String KEYSTORE =
|
|
57 |
DATA_DIR + System.getProperty("file.separator") + "certs" +
|
|
58 |
System.getProperty("file.separator") + "xmldsig.jks";
|
|
59 |
private final static String STYLESHEET =
|
|
60 |
"http://www.w3.org/TR/xml-stylesheet";
|
|
61 |
private final static String STYLESHEET_B64 =
|
|
62 |
"http://www.w3.org/Signature/2002/04/xml-stylesheet.b64";
|
|
63 |
|
|
64 |
private final static String[] FILES = {
|
|
65 |
"signature-enveloped-dsa.xml",
|
|
66 |
"signature-enveloping-b64-dsa.xml",
|
|
67 |
"signature-enveloping-dsa.xml",
|
|
68 |
"signature-enveloping-rsa.xml",
|
|
69 |
"signature-enveloping-hmac-sha1.xml",
|
|
70 |
"signature-external-dsa.xml",
|
|
71 |
"signature-external-b64-dsa.xml",
|
|
72 |
"signature-retrievalmethod-rawx509crt.xml",
|
|
73 |
"signature-keyname.xml",
|
|
74 |
"signature-x509-crt-crl.xml",
|
|
75 |
"signature-x509-crt.xml",
|
|
76 |
"signature-x509-is.xml",
|
|
77 |
"signature-x509-ski.xml",
|
|
78 |
"signature-x509-sn.xml",
|
|
79 |
// "signature.xml",
|
|
80 |
"exc-signature.xml",
|
|
81 |
"sign-spec.xml"
|
|
82 |
};
|
|
83 |
|
|
84 |
static KeySelector skks;
|
|
85 |
static {
|
|
86 |
try {
|
|
87 |
skks =
|
|
88 |
new KeySelectors.SecretKeySelector("secret".getBytes("ASCII"));
|
|
89 |
} catch (Exception e) {
|
|
90 |
//should not occur
|
|
91 |
}
|
|
92 |
}
|
|
93 |
private final static KeySelector SKKS = skks;
|
|
94 |
private final static KeySelector KVKS =
|
|
95 |
new KeySelectors.KeyValueKeySelector();
|
|
96 |
private final static KeySelector CKS =
|
|
97 |
new KeySelectors.CollectionKeySelector(new File(DATA_DIR));
|
|
98 |
private final static KeySelector RXKS =
|
|
99 |
new KeySelectors.RawX509KeySelector();
|
|
100 |
private final static KeySelector XKS = null;
|
|
101 |
private final static KeySelector[] KEY_SELECTORS = {
|
|
102 |
KVKS,
|
|
103 |
KVKS,
|
|
104 |
KVKS,
|
|
105 |
KVKS,
|
|
106 |
SKKS,
|
|
107 |
KVKS,
|
|
108 |
KVKS,
|
|
109 |
CKS,
|
|
110 |
CKS,
|
|
111 |
RXKS,
|
|
112 |
RXKS,
|
|
113 |
CKS,
|
|
114 |
CKS,
|
|
115 |
CKS,
|
|
116 |
// XKS,
|
|
117 |
KVKS,
|
|
118 |
RXKS
|
|
119 |
};
|
|
120 |
private static URIDereferencer httpUd = null;
|
|
121 |
|
|
122 |
public static void main(String args[]) throws Exception {
|
|
123 |
httpUd = new HttpURIDereferencer();
|
|
124 |
|
|
125 |
validator = new SignatureValidator(new File(DATA_DIR));
|
|
126 |
|
|
127 |
boolean atLeastOneFailed = false;
|
|
128 |
for (int i=0; i < FILES.length; i++) {
|
|
129 |
System.out.println("Validating " + FILES[i]);
|
|
130 |
if (test_signature(FILES[i], KEY_SELECTORS[i])) {
|
|
131 |
System.out.println("PASSED");
|
|
132 |
} else {
|
|
133 |
System.out.println("FAILED");
|
|
134 |
atLeastOneFailed = true;
|
|
135 |
}
|
|
136 |
}
|
|
137 |
// test with reference caching enabled
|
|
138 |
System.out.println("Validating sign-spec.xml with caching enabled");
|
|
139 |
if (test_signature("sign-spec.xml", RXKS, true)) {
|
|
140 |
System.out.println("PASSED");
|
|
141 |
} else {
|
|
142 |
System.out.println("FAILED");
|
|
143 |
atLeastOneFailed = true;
|
|
144 |
}
|
|
145 |
|
3462
|
146 |
System.out.println("Validating signature-enveloping-hmac-sha1-40.xml");
|
|
147 |
try {
|
|
148 |
test_signature("signature-enveloping-hmac-sha1-40.xml", SKKS, false);
|
|
149 |
System.out.println("FAILED");
|
|
150 |
atLeastOneFailed = true;
|
|
151 |
} catch (XMLSignatureException xse) {
|
|
152 |
System.out.println(xse.getMessage());
|
|
153 |
System.out.println("PASSED");
|
|
154 |
}
|
|
155 |
|
|
156 |
System.out.println("Validating signature-enveloping-hmac-sha1-trunclen-0-attack.xml");
|
|
157 |
try {
|
|
158 |
test_signature("signature-enveloping-hmac-sha1-trunclen-0-attack.xml", SKKS, false);
|
|
159 |
System.out.println("FAILED");
|
|
160 |
atLeastOneFailed = true;
|
|
161 |
} catch (XMLSignatureException xse) {
|
|
162 |
System.out.println(xse.getMessage());
|
|
163 |
System.out.println("PASSED");
|
|
164 |
}
|
|
165 |
|
|
166 |
System.out.println("Validating signature-enveloping-hmac-sha1-trunclen-8-attack.xml");
|
|
167 |
try {
|
|
168 |
test_signature("signature-enveloping-hmac-sha1-trunclen-8-attack.xml", SKKS, false);
|
|
169 |
System.out.println("FAILED");
|
|
170 |
atLeastOneFailed = true;
|
|
171 |
} catch (XMLSignatureException xse) {
|
|
172 |
System.out.println(xse.getMessage());
|
|
173 |
System.out.println("PASSED");
|
|
174 |
}
|
|
175 |
|
2
|
176 |
if (atLeastOneFailed) {
|
|
177 |
throw new Exception
|
|
178 |
("At least one signature did not validate as expected");
|
|
179 |
}
|
|
180 |
}
|
|
181 |
|
|
182 |
public static boolean test_signature(String file, KeySelector ks)
|
|
183 |
throws Exception {
|
|
184 |
return test_signature(file, ks, false);
|
|
185 |
}
|
|
186 |
|
|
187 |
public static boolean test_signature(String file, KeySelector ks,
|
|
188 |
boolean cache) throws Exception {
|
|
189 |
if (ks == null) {
|
|
190 |
KeyStore keystore = KeyStore.getInstance("JKS");
|
|
191 |
keystore.load
|
|
192 |
(new FileInputStream(KEYSTORE), "changeit".toCharArray());
|
|
193 |
ks = new X509KeySelector(keystore, false);
|
|
194 |
}
|
|
195 |
return validator.validate(file, ks, httpUd, cache);
|
|
196 |
}
|
|
197 |
|
|
198 |
/**
|
|
199 |
* This URIDereferencer returns locally cached copies of http content to
|
|
200 |
* avoid test failures due to network glitches, etc.
|
|
201 |
*/
|
|
202 |
private static class HttpURIDereferencer implements URIDereferencer {
|
|
203 |
private URIDereferencer defaultUd;
|
|
204 |
|
|
205 |
HttpURIDereferencer() {
|
|
206 |
defaultUd = XMLSignatureFactory.getInstance().getURIDereferencer();
|
|
207 |
}
|
|
208 |
|
|
209 |
public Data dereference(final URIReference ref, XMLCryptoContext ctx)
|
|
210 |
throws URIReferenceException {
|
|
211 |
String uri = ref.getURI();
|
|
212 |
if (uri.equals(STYLESHEET) || uri.equals(STYLESHEET_B64)) {
|
|
213 |
try {
|
|
214 |
FileInputStream fis = new FileInputStream(new File
|
|
215 |
(DATA_DIR, uri.substring(uri.lastIndexOf('/'))));
|
|
216 |
return new OctetStreamData(fis,ref.getURI(),ref.getType());
|
|
217 |
} catch (Exception e) { throw new URIReferenceException(e); }
|
|
218 |
}
|
|
219 |
|
|
220 |
// fallback on builtin deref
|
|
221 |
return defaultUd.dereference(ref, ctx);
|
|
222 |
}
|
|
223 |
}
|
|
224 |
}
|