Skip to content

Commit c740e1e

Browse files
committedJul 22, 2024
8333772: Incorrect Kerberos behavior when udp_preference_limit = 0
Reviewed-by: ssahoo, mullan
1 parent 0725eb1 commit c740e1e

File tree

4 files changed

+211
-32
lines changed

4 files changed

+211
-32
lines changed
 

‎src/java.security.jgss/share/classes/sun/security/krb5/KdcComm.java

+28-28
Original file line numberDiff line numberDiff line change
@@ -62,15 +62,18 @@ public final class KdcComm {
6262
// them can also be defined in a realm, which overrides value here.
6363

6464
/**
65-
* max retry time for a single KDC, default Krb5.KDC_RETRY_LIMIT (3)
65+
* max retry time for a single KDC, default Krb5.KDC_RETRY_LIMIT (3),
66+
* Must be > 0.
6667
*/
6768
private static int defaultKdcRetryLimit;
6869
/**
69-
* timeout requesting a ticket from KDC, in millisec, default 30 sec
70+
* timeout requesting a ticket from KDC, in millisec, default
71+
* Krb5.KDC_TIMEOUT (30000). Must be > 0.
7072
*/
7173
private static int defaultKdcTimeout;
7274
/**
73-
* max UDP packet size, default unlimited (-1)
75+
* max UDP packet size, default Krb5.KDC_DEFAULT_UDP_PREF_LIMIT (1465).
76+
* Must be >= 0 and <= Krb5.KDC_HARD_UDP_LIMIT (32700).
7477
*/
7578
private static int defaultUdpPrefLimit;
7679

@@ -146,9 +149,9 @@ public String run() {
146149
timeout = parseTimeString(temp);
147150

148151
temp = cfg.get("libdefaults", "max_retries");
149-
max_retries = parsePositiveIntString(temp);
152+
max_retries = parseNonNegativeIntString(temp);
150153
temp = cfg.get("libdefaults", "udp_preference_limit");
151-
udp_pref_limit = parsePositiveIntString(temp);
154+
udp_pref_limit = parseNonNegativeIntString(temp);
152155
} catch (Exception exc) {
153156
// ignore any exceptions; use default values
154157
if (DEBUG != null) {
@@ -157,7 +160,7 @@ public String run() {
157160
exc.getMessage());
158161
}
159162
}
160-
defaultKdcTimeout = timeout > 0 ? timeout : 30*1000; // 30 seconds
163+
defaultKdcTimeout = timeout > 0 ? timeout : Krb5.KDC_TIMEOUT;
161164
defaultKdcRetryLimit =
162165
max_retries > 0 ? max_retries : Krb5.KDC_RETRY_LIMIT;
163166

@@ -175,11 +178,11 @@ public String run() {
175178
/**
176179
* The instance fields
177180
*/
178-
private String realm;
181+
private final String realm;
179182

180183
public KdcComm(String realm) throws KrbException {
181184
if (realm == null) {
182-
realm = Config.getInstance().getDefaultRealm();
185+
realm = Config.getInstance().getDefaultRealm();
183186
if (realm == null) {
184187
throw new KrbException(Krb5.KRB_ERR_GENERIC,
185188
"Cannot find default realm");
@@ -191,11 +194,10 @@ public KdcComm(String realm) throws KrbException {
191194
public byte[] send(KrbKdcReq req)
192195
throws IOException, KrbException {
193196
int udpPrefLimit = getRealmSpecificValue(
194-
realm, "udp_preference_limit", defaultUdpPrefLimit);
197+
realm, "udp_preference_limit", defaultUdpPrefLimit, false);
195198

196199
byte[] obuf = req.encoding();
197-
boolean useTCP = (udpPrefLimit > 0 &&
198-
(obuf != null && obuf.length > udpPrefLimit));
200+
boolean useTCP = obuf != null && obuf.length > udpPrefLimit;
199201

200202
return send(req, useTCP);
201203
}
@@ -207,14 +209,6 @@ private byte[] send(KrbKdcReq req, boolean useTCP)
207209
return null;
208210
Config cfg = Config.getInstance();
209211

210-
if (realm == null) {
211-
realm = cfg.getDefaultRealm();
212-
if (realm == null) {
213-
throw new KrbException(Krb5.KRB_ERR_GENERIC,
214-
"Cannot find default realm");
215-
}
216-
}
217-
218212
String kdcList = cfg.getKDCList(realm);
219213
if (kdcList == null) {
220214
throw new KrbException("Cannot get kdc for realm " + realm);
@@ -296,9 +290,9 @@ private byte[] send(KrbKdcReq req, String tempKdc, boolean useTCP)
296290

297291
int port = Krb5.KDC_INET_DEFAULT_PORT;
298292
int retries = getRealmSpecificValue(
299-
realm, "max_retries", defaultKdcRetryLimit);
293+
realm, "max_retries", defaultKdcRetryLimit, true);
300294
int timeout = getRealmSpecificValue(
301-
realm, "kdc_timeout", defaultKdcTimeout);
295+
realm, "kdc_timeout", defaultKdcTimeout, true);
302296
if (badPolicy == BpType.TRY_LESS &&
303297
KdcAccessibility.isBad(tempKdc)) {
304298
if (retries > tryLessMaxRetries) {
@@ -339,7 +333,7 @@ private byte[] send(KrbKdcReq req, String tempKdc, boolean useTCP)
339333
}
340334
}
341335
if (portStr != null) {
342-
int tempPort = parsePositiveIntString(portStr);
336+
int tempPort = parseNonNegativeIntString(portStr);
343337
if (tempPort > 0)
344338
port = tempPort;
345339
}
@@ -444,10 +438,10 @@ private static int parseTimeString(String s) {
444438
return -1;
445439
}
446440
if (s.endsWith("s")) {
447-
int seconds = parsePositiveIntString(s.substring(0, s.length()-1));
441+
int seconds = parseNonNegativeIntString(s.substring(0, s.length()-1));
448442
return (seconds < 0) ? -1 : (seconds*1000);
449443
} else {
450-
return parsePositiveIntString(s);
444+
return parseNonNegativeIntString(s);
451445
}
452446
}
453447

@@ -461,9 +455,11 @@ private static int parseTimeString(String s) {
461455
* the global setting if null
462456
* @param key the key for the setting
463457
* @param defValue default value
458+
* @param mustBePositive true if value must be >0, false if value must be >=0
464459
* @return a value for the key
465460
*/
466-
private int getRealmSpecificValue(String realm, String key, int defValue) {
461+
private int getRealmSpecificValue(String realm, String key, int defValue,
462+
boolean mustBePositive) {
467463
int v = defValue;
468464

469465
if (realm == null) return v;
@@ -475,18 +471,22 @@ private int getRealmSpecificValue(String realm, String key, int defValue) {
475471
if (key.equals("kdc_timeout")) {
476472
temp = parseTimeString(value);
477473
} else {
478-
temp = parsePositiveIntString(value);
474+
temp = parseNonNegativeIntString(value);
479475
}
480476
} catch (Exception exc) {
481477
// Ignored, defValue will be picked up
482478
}
483479

484-
if (temp > 0) v = temp;
480+
if (mustBePositive) {
481+
if (temp > 0) v = temp;
482+
} else {
483+
if (temp >= 0) v = temp;
484+
}
485485

486486
return v;
487487
}
488488

489-
private static int parsePositiveIntString(String intString) {
489+
private static int parseNonNegativeIntString(String intString) {
490490
if (intString == null)
491491
return -1;
492492

‎src/java.security.jgss/share/classes/sun/security/krb5/internal/Krb5.java

+1
Original file line numberDiff line numberDiff line change
@@ -134,6 +134,7 @@ public class Krb5 {
134134
// number of retries before giving up
135135

136136
public static final int KDC_RETRY_LIMIT = 3;
137+
public static final int KDC_TIMEOUT = 30000;
137138
public static final int KDC_DEFAULT_UDP_PREF_LIMIT = 1465;
138139
public static final int KDC_HARD_UDP_LIMIT = 32700;
139140

‎test/jdk/sun/security/krb5/auto/KdcPolicy.java

+4-4
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@
3838

3939
/*
4040
* @test
41-
* @bug 8164656 8181461 8194486
41+
* @bug 8164656 8181461 8194486 8333772
4242
* @summary krb5.kdc.bad.policy test
4343
* @library /test/lib
4444
* @run main jdk.test.lib.FileInstaller TestHosts TestHosts
@@ -219,13 +219,13 @@ static void writeConf(int max, int to, int... ports) throws Exception {
219219
inDefaults += "udp_preference_limit = 10000\n";
220220
} else if (r.nextBoolean()) {
221221
inRealm += " udp_preference_limit = 10000\n";
222-
inDefaults += "udp_preference_limit = 1\n";
222+
inDefaults += "udp_preference_limit = 0\n";
223223
} // else no settings means UDP
224224
} else {
225225
if (r.nextBoolean()) {
226-
inDefaults += "udp_preference_limit = 1\n";
226+
inDefaults += "udp_preference_limit = 0\n";
227227
} else {
228-
inRealm += " udp_preference_limit = 1\n";
228+
inRealm += " udp_preference_limit = 0\n";
229229
inDefaults += "udp_preference_limit = 10000\n";
230230
}
231231
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,178 @@
1+
/*
2+
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
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 Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
import java.io.*;
25+
import java.nio.file.Files;
26+
import java.nio.file.Paths;
27+
import java.util.regex.Matcher;
28+
import java.util.regex.Pattern;
29+
30+
import jdk.test.lib.Asserts;
31+
import sun.security.krb5.Config;
32+
33+
/*
34+
* @test
35+
* @bug 8333772
36+
* @summary check krb5.conf reading on default and realm-specific values
37+
* @library /test/lib
38+
*/
39+
public class RealmSpecificValues {
40+
41+
static DebugMatcher cm = new DebugMatcher();
42+
43+
public static void main(String[] args) throws Exception {
44+
System.setProperty("sun.security.krb5.debug", "true");
45+
System.setProperty("java.security.krb5.conf", "alternative-krb5.conf");
46+
47+
// Defaults
48+
writeConf(-1, -1, -1, -1, -1, -1);
49+
test(true, 3, 30000);
50+
51+
// Below has settings. For each setting we provide 3 cases:
52+
// 1. Set in defaults, 2, set in realms, 3, both
53+
54+
// udp = 0 is useful
55+
writeConf(0, -1, -1, -1, -1, -1);
56+
test(false, 3, 30000);
57+
writeConf(-1, -1, -1, 0, -1, -1);
58+
test(false, 3, 30000);
59+
writeConf(1, -1, -1, 0, -1, -1);
60+
test(false, 3, 30000);
61+
62+
// max_retries = 0 is ignored
63+
writeConf(-1, 0, -1, -1, -1, -1);
64+
test(true, 3, 30000);
65+
writeConf(-1, -1, -1, -1, 0, -1);
66+
test(true, 3, 30000);
67+
writeConf(-1, 6, -1, -1, 0, -1); // Note: 0 is ignored, it does not reset to default
68+
test(true, 6, 30000);
69+
70+
// max_retries = 1 is useful
71+
writeConf(-1, 1, -1, -1, -1, -1);
72+
test(true, 1, 30000);
73+
writeConf(-1, -1, -1, -1, 1, -1);
74+
test(true, 1, 30000);
75+
writeConf(-1, 3, -1, -1, 1, -1);
76+
test(true, 1, 30000);
77+
78+
// timeout = 0 is ignored
79+
writeConf(-1, -1, 0, -1, -1, -1);
80+
test(true, 3, 30000);
81+
writeConf(-1, -1, -1, -1, -1, 0);
82+
test(true, 3, 30000);
83+
writeConf(-1, -1, 10000, -1, -1, 0);
84+
test(true, 3, 10000);
85+
86+
// timeout > 0 is useful
87+
writeConf(-1, -1, 10000, -1, -1, -1);
88+
test(true, 3, 10000);
89+
writeConf(-1, -1, -1, -1, -1, 10000);
90+
test(true, 3, 10000);
91+
writeConf(-1, -1, 20000, -1, -1, 10000);
92+
test(true, 3, 10000);
93+
}
94+
95+
static void writeConf(int limit, int retries, int timeout,
96+
int limitR, int retriesR, int timeoutR) throws Exception {
97+
98+
String inDefaults = "";
99+
if (limit >= 0) inDefaults += "udp_preference_limit = " + limit + "\n";
100+
if (retries >= 0) inDefaults += "max_retries = " + retries + "\n";
101+
if (timeout >= 0) inDefaults += "kdc_timeout = " + timeout + "\n";
102+
103+
String inRealm = "";
104+
if (limitR >= 0) inRealm += "udp_preference_limit = " + limitR + "\n";
105+
if (retriesR >= 0) inRealm += "max_retries = " + retriesR + "\n";
106+
if (timeoutR >= 0) inRealm += "kdc_timeout = " + timeoutR + "\n";
107+
108+
String conf = "[libdefaults]\n" +
109+
"default_realm = " + OneKDC.REALM + "\n" +
110+
inDefaults +
111+
"\n" +
112+
"[realms]\n" +
113+
OneKDC.REALM + " = {\n" +
114+
"kdc = " + OneKDC.KDCHOST + ":12345\n" +
115+
inRealm +
116+
"}\n";
117+
118+
Files.writeString(Paths.get("alternative-krb5.conf"), conf);
119+
}
120+
121+
static void test(boolean isUDP, int retries, int timeout) throws Exception {
122+
123+
PrintStream oldErr = System.err;
124+
ByteArrayOutputStream bo = new ByteArrayOutputStream();
125+
System.setErr(new PrintStream(bo));
126+
try {
127+
Config.refresh();
128+
Context.fromUserPass(OneKDC.USER, OneKDC.PASS, false);
129+
} catch (Exception e) {
130+
// will happen
131+
} finally {
132+
System.setErr(oldErr);
133+
}
134+
135+
String[] lines = new String(bo.toByteArray()).split("\n");
136+
for (String line: lines) {
137+
if (cm.match(line)) {
138+
System.out.println(line);
139+
Asserts.assertEQ(cm.isUDP(), isUDP);
140+
Asserts.assertEQ(cm.timeout(), timeout);
141+
Asserts.assertEQ(cm.retries(), retries);
142+
return;
143+
}
144+
}
145+
Asserts.fail("Should not reach here");
146+
}
147+
148+
/**
149+
* A helper class to match the krb5 debug output:
150+
* >>> KrbKdcReq send: kdc=kdc.rabbit.hole TCP:12345, timeout=30000,
151+
* number of retries =3, #bytes=141
152+
*/
153+
static class DebugMatcher {
154+
155+
static final Pattern re = Pattern.compile(
156+
">>> KrbKdcReq send: kdc=\\S+ (TCP|UDP):\\d+, " +
157+
"timeout=(\\d+), number of retries\\s*=(\\d+)");
158+
159+
Matcher matcher;
160+
161+
boolean match(String line) {
162+
matcher = re.matcher(line);
163+
return matcher.find();
164+
}
165+
166+
boolean isUDP() {
167+
return matcher.group(1).equals("UDP");
168+
}
169+
170+
int timeout() {
171+
return Integer.parseInt(matcher.group(2));
172+
}
173+
174+
int retries() {
175+
return Integer.parseInt(matcher.group(3));
176+
}
177+
}
178+
}

0 commit comments

Comments
 (0)
Please sign in to comment.