8273402: Use derived NamingExceptions in com.sun.jndi.ldap.Connection#readReply

Reviewed-by: dfuchs
This commit is contained in:
Aleksei Efimov 2021-09-10 14:15:45 +00:00
parent 4e6de5f9de
commit c464f09056
3 changed files with 38 additions and 28 deletions

View File

@ -425,7 +425,7 @@ public final class Connection implements Runnable {
/** /**
* Reads a reply; waits until one is ready. * Reads a reply; waits until one is ready.
*/ */
BerDecoder readReply(LdapRequest ldr) throws IOException, NamingException { BerDecoder readReply(LdapRequest ldr) throws NamingException {
BerDecoder rber; BerDecoder rber;
// If socket closed, don't even try // If socket closed, don't even try
@ -436,7 +436,7 @@ public final class Connection implements Runnable {
} }
} }
NamingException namingException = null; IOException ioException = null;
try { try {
// if no timeout is set so we wait infinitely until // if no timeout is set so we wait infinitely until
// a response is received OR until the connection is closed or cancelled // a response is received OR until the connection is closed or cancelled
@ -445,25 +445,29 @@ public final class Connection implements Runnable {
} catch (InterruptedException ex) { } catch (InterruptedException ex) {
throw new InterruptedNamingException( throw new InterruptedNamingException(
"Interrupted during LDAP operation"); "Interrupted during LDAP operation");
} catch (CommunicationException ce) { } catch (IOException ioe) {
// Re-throw
throw ce;
} catch (NamingException ne) {
// Connection is timed out OR closed/cancelled // Connection is timed out OR closed/cancelled
namingException = ne; // getReplyBer throws IOException when the requests needs to be abandoned
ioException = ioe;
rber = null; rber = null;
} }
if (rber == null) { if (rber == null) {
abandonRequest(ldr, null); abandonRequest(ldr, null);
} }
// namingException can be not null in the following cases: // ioException can be not null in the following cases:
// a) The response is timed-out // a) The response is timed-out
// b) LDAP request connection has been closed or cancelled // b) LDAP request connection has been closed
// If the request has been cancelled - CommunicationException is
// thrown directly from LdapRequest.getReplyBer, since there is no
// need to abandon request.
// The exception message is initialized in LdapRequest::getReplyBer // The exception message is initialized in LdapRequest::getReplyBer
if (namingException != null) { if (ioException != null) {
// Re-throw NamingException after all cleanups are done // Throw CommunicationException after all cleanups are done
throw namingException; String message = ioException.getMessage();
var ce = new CommunicationException(message);
ce.initCause(ioException);
throw ce;
} }
return rber; return rber;
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 1999, 2020, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1999, 2021, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -103,17 +103,17 @@ final class LdapRequest {
* @param millis timeout, infinite if the value is negative * @param millis timeout, infinite if the value is negative
* @return BerDecoder if reply was read successfully * @return BerDecoder if reply was read successfully
* @throws CommunicationException request has been canceled and request does not need to be abandoned * @throws CommunicationException request has been canceled and request does not need to be abandoned
* @throws NamingException request has been closed or timed out. Request does need to be abandoned * @throws IOException request has been closed or timed out. Request does need to be abandoned
* @throws InterruptedException LDAP operation has been interrupted * @throws InterruptedException LDAP operation has been interrupted
*/ */
BerDecoder getReplyBer(long millis) throws NamingException, BerDecoder getReplyBer(long millis) throws IOException, CommunicationException,
InterruptedException { InterruptedException {
if (cancelled) { if (cancelled) {
throw new CommunicationException("Request: " + msgId + throw new CommunicationException("Request: " + msgId +
" cancelled"); " cancelled");
} }
if (isClosed()) { if (isClosed()) {
throw new NamingException(CLOSE_MSG); throw new IOException(CLOSE_MSG);
} }
BerDecoder result = millis > 0 ? BerDecoder result = millis > 0 ?
@ -126,11 +126,11 @@ final class LdapRequest {
// poll from 'replies' blocking queue ended-up with timeout // poll from 'replies' blocking queue ended-up with timeout
if (result == null) { if (result == null) {
throw new NamingException(String.format(TIMEOUT_MSG_FMT, millis)); throw new IOException(String.format(TIMEOUT_MSG_FMT, millis));
} }
// Unexpected EOF can be caused by connection closure or cancellation // Unexpected EOF can be caused by connection closure or cancellation
if (result == EOF) { if (result == EOF) {
throw new NamingException(CLOSE_MSG); throw new IOException(CLOSE_MSG);
} }
return result; return result;
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -23,12 +23,14 @@
/* /*
* @test * @test
* @bug 8062947 * @bug 8062947 8273402
* @summary Test that NamingException message text matches the failure reason * @summary Test that CommunicationException is thrown when connection is timed out or closed/cancelled,
* and it's text matches the failure reason.
* @library /test/lib lib * @library /test/lib lib
* @run testng NamingExceptionMessageTest * @run testng NamingExceptionMessageTest
*/ */
import javax.naming.CommunicationException;
import javax.naming.Context; import javax.naming.Context;
import javax.naming.NamingException; import javax.naming.NamingException;
import javax.naming.ServiceUnavailableException; import javax.naming.ServiceUnavailableException;
@ -55,9 +57,10 @@ public class NamingExceptionMessageTest {
ldapServer.start(); ldapServer.start();
ldapServer.awaitStartup(); ldapServer.awaitStartup();
var env = ldapServer.getInitialLdapCtxEnvironment(TIMEOUT_VALUE); var env = ldapServer.getInitialLdapCtxEnvironment(TIMEOUT_VALUE);
var namingException = Assert.expectThrows(NamingException.class, () -> new InitialDirContext(env)); var communicationException =
System.out.println("Got naming exception:" + namingException); Assert.expectThrows(CommunicationException.class, () -> new InitialDirContext(env));
Assert.assertEquals(namingException.getMessage(), EXPECTED_TIMEOUT_MESSAGE); System.out.println("Got CommunicationException:" + communicationException);
Assert.assertEquals(communicationException.getMessage(), EXPECTED_TIMEOUT_MESSAGE);
} }
} }
@ -74,9 +77,12 @@ public class NamingExceptionMessageTest {
// read-out of the reply message. For such cases test run is considered as successful. // read-out of the reply message. For such cases test run is considered as successful.
System.out.println("Got ServiceUnavailableException: Test PASSED"); System.out.println("Got ServiceUnavailableException: Test PASSED");
} else { } else {
// If exception is not ServiceUnavailableException - check the exception message // If exception is not ServiceUnavailableException, CommunicationException is expected
System.out.println("Got NamingException:" + namingException); Assert.assertTrue(namingException instanceof CommunicationException);
Assert.assertEquals(namingException.getMessage(), EXPECTED_CLOSURE_MESSAGE); var communicationException = (CommunicationException) namingException;
System.out.println("Got CommunicationException:" + communicationException);
// Check exception message
Assert.assertEquals(communicationException.getMessage(), EXPECTED_CLOSURE_MESSAGE);
} }
} }
} }