Merge
This commit is contained in:
commit
f17489b0d4
@ -22,7 +22,6 @@
|
|||||||
* CA 95054 USA or visit www.sun.com if you need additional information or
|
* CA 95054 USA or visit www.sun.com if you need additional information or
|
||||||
* have any questions.
|
* have any questions.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package com.sun.jmx.remote.internal;
|
package com.sun.jmx.remote.internal;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
@ -34,6 +33,7 @@ import java.util.List;
|
|||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
|
|
||||||
|
import java.security.AccessControlContext;
|
||||||
import java.security.AccessController;
|
import java.security.AccessController;
|
||||||
import java.security.PrivilegedAction;
|
import java.security.PrivilegedAction;
|
||||||
import javax.security.auth.Subject;
|
import javax.security.auth.Subject;
|
||||||
@ -54,6 +54,9 @@ import com.sun.jmx.remote.util.EnvHelp;
|
|||||||
|
|
||||||
|
|
||||||
public abstract class ClientNotifForwarder {
|
public abstract class ClientNotifForwarder {
|
||||||
|
|
||||||
|
private final AccessControlContext acc;
|
||||||
|
|
||||||
public ClientNotifForwarder(Map env) {
|
public ClientNotifForwarder(Map env) {
|
||||||
this(null, env);
|
this(null, env);
|
||||||
}
|
}
|
||||||
@ -87,6 +90,8 @@ public abstract class ClientNotifForwarder {
|
|||||||
this.command = command;
|
this.command = command;
|
||||||
if (thread == null) {
|
if (thread == null) {
|
||||||
thread = new Thread() {
|
thread = new Thread() {
|
||||||
|
|
||||||
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
while (true) {
|
while (true) {
|
||||||
Runnable r;
|
Runnable r;
|
||||||
@ -130,6 +135,7 @@ public abstract class ClientNotifForwarder {
|
|||||||
|
|
||||||
this.defaultClassLoader = defaultClassLoader;
|
this.defaultClassLoader = defaultClassLoader;
|
||||||
this.executor = ex;
|
this.executor = ex;
|
||||||
|
this.acc = AccessController.getContext();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -390,28 +396,85 @@ public abstract class ClientNotifForwarder {
|
|||||||
setState(TERMINATED);
|
setState(TERMINATED);
|
||||||
}
|
}
|
||||||
|
|
||||||
// -------------------------------------------------
|
|
||||||
// private classes
|
// -------------------------------------------------
|
||||||
// -------------------------------------------------
|
// private classes
|
||||||
|
// -------------------------------------------------
|
||||||
//
|
//
|
||||||
|
|
||||||
private class NotifFetcher implements Runnable {
|
private class NotifFetcher implements Runnable {
|
||||||
|
|
||||||
|
private volatile boolean alreadyLogged = false;
|
||||||
|
|
||||||
|
private void logOnce(String msg, SecurityException x) {
|
||||||
|
if (alreadyLogged) return;
|
||||||
|
// Log only once.
|
||||||
|
logger.config("setContextClassLoader",msg);
|
||||||
|
if (x != null) logger.fine("setContextClassLoader", x);
|
||||||
|
alreadyLogged = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set new context class loader, returns previous one.
|
||||||
|
private final ClassLoader setContextClassLoader(final ClassLoader loader) {
|
||||||
|
final AccessControlContext ctxt = ClientNotifForwarder.this.acc;
|
||||||
|
// if ctxt is null, log a config message and throw a
|
||||||
|
// SecurityException.
|
||||||
|
if (ctxt == null) {
|
||||||
|
logOnce("AccessControlContext must not be null.",null);
|
||||||
|
throw new SecurityException("AccessControlContext must not be null");
|
||||||
|
}
|
||||||
|
return AccessController.doPrivileged(
|
||||||
|
new PrivilegedAction<ClassLoader>() {
|
||||||
|
public ClassLoader run() {
|
||||||
|
try {
|
||||||
|
// get context class loader - may throw
|
||||||
|
// SecurityException - though unlikely.
|
||||||
|
final ClassLoader previous =
|
||||||
|
Thread.currentThread().getContextClassLoader();
|
||||||
|
|
||||||
|
// if nothing needs to be done, break here...
|
||||||
|
if (loader == previous) return previous;
|
||||||
|
|
||||||
|
// reset context class loader - may throw
|
||||||
|
// SecurityException
|
||||||
|
Thread.currentThread().setContextClassLoader(loader);
|
||||||
|
return previous;
|
||||||
|
} catch (SecurityException x) {
|
||||||
|
logOnce("Permission to set ContextClassLoader missing. " +
|
||||||
|
"Notifications will not be dispatched. " +
|
||||||
|
"Please check your Java policy configuration: " +
|
||||||
|
x, x);
|
||||||
|
throw x;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, ctxt);
|
||||||
|
}
|
||||||
|
|
||||||
public void run() {
|
public void run() {
|
||||||
|
final ClassLoader previous;
|
||||||
|
if (defaultClassLoader != null) {
|
||||||
|
previous = setContextClassLoader(defaultClassLoader);
|
||||||
|
} else {
|
||||||
|
previous = null;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
doRun();
|
||||||
|
} finally {
|
||||||
|
if (defaultClassLoader != null) {
|
||||||
|
setContextClassLoader(previous);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void doRun() {
|
||||||
synchronized (ClientNotifForwarder.this) {
|
synchronized (ClientNotifForwarder.this) {
|
||||||
currentFetchThread = Thread.currentThread();
|
currentFetchThread = Thread.currentThread();
|
||||||
|
|
||||||
if (state == STARTING)
|
if (state == STARTING) {
|
||||||
setState(STARTED);
|
setState(STARTED);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (defaultClassLoader != null) {
|
|
||||||
AccessController.doPrivileged(new PrivilegedAction<Void>() {
|
|
||||||
public Void run() {
|
|
||||||
Thread.currentThread().
|
|
||||||
setContextClassLoader(defaultClassLoader);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
NotificationResult nr = null;
|
NotificationResult nr = null;
|
||||||
if (!shouldStop() && (nr = fetchNotifs()) != null) {
|
if (!shouldStop() && (nr = fetchNotifs()) != null) {
|
||||||
@ -444,8 +507,9 @@ public abstract class ClientNotifForwarder {
|
|||||||
// check if an mbean unregistration notif
|
// check if an mbean unregistration notif
|
||||||
if (!listenerID.equals(mbeanRemovedNotifID)) {
|
if (!listenerID.equals(mbeanRemovedNotifID)) {
|
||||||
final ClientListenerInfo li = infoList.get(listenerID);
|
final ClientListenerInfo li = infoList.get(listenerID);
|
||||||
if (li != null)
|
if (li != null) {
|
||||||
listeners.put(listenerID,li);
|
listeners.put(listenerID, li);
|
||||||
|
}
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
final Notification notif = tn.getNotification();
|
final Notification notif = tn.getNotification();
|
||||||
@ -786,9 +850,7 @@ public abstract class ClientNotifForwarder {
|
|||||||
private long clientSequenceNumber = -1;
|
private long clientSequenceNumber = -1;
|
||||||
private final int maxNotifications;
|
private final int maxNotifications;
|
||||||
private final long timeout;
|
private final long timeout;
|
||||||
|
|
||||||
private Integer mbeanRemovedNotifID = null;
|
private Integer mbeanRemovedNotifID = null;
|
||||||
|
|
||||||
private Thread currentFetchThread;
|
private Thread currentFetchThread;
|
||||||
|
|
||||||
// state
|
// state
|
||||||
|
@ -596,7 +596,7 @@ public class CounterMonitor extends Monitor implements CounterMonitorMBean {
|
|||||||
* types sent by the counter monitor.
|
* types sent by the counter monitor.
|
||||||
*/
|
*/
|
||||||
public MBeanNotificationInfo[] getNotificationInfo() {
|
public MBeanNotificationInfo[] getNotificationInfo() {
|
||||||
return notifsInfo;
|
return notifsInfo.clone();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -478,7 +478,7 @@ public class GaugeMonitor extends Monitor implements GaugeMonitorMBean {
|
|||||||
* types sent by the gauge monitor.
|
* types sent by the gauge monitor.
|
||||||
*/
|
*/
|
||||||
public MBeanNotificationInfo[] getNotificationInfo() {
|
public MBeanNotificationInfo[] getNotificationInfo() {
|
||||||
return notifsInfo;
|
return notifsInfo.clone();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -32,6 +32,7 @@ import java.io.IOException;
|
|||||||
import java.security.AccessControlContext;
|
import java.security.AccessControlContext;
|
||||||
import java.security.AccessController;
|
import java.security.AccessController;
|
||||||
import java.security.PrivilegedAction;
|
import java.security.PrivilegedAction;
|
||||||
|
import java.security.ProtectionDomain;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.WeakHashMap;
|
import java.util.WeakHashMap;
|
||||||
@ -165,7 +166,10 @@ public abstract class Monitor
|
|||||||
/**
|
/**
|
||||||
* AccessControlContext of the Monitor.start() caller.
|
* AccessControlContext of the Monitor.start() caller.
|
||||||
*/
|
*/
|
||||||
private AccessControlContext acc;
|
private static final AccessControlContext noPermissionsACC =
|
||||||
|
new AccessControlContext(
|
||||||
|
new ProtectionDomain[] {new ProtectionDomain(null, null)});
|
||||||
|
private volatile AccessControlContext acc = noPermissionsACC;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Scheduler Service.
|
* Scheduler Service.
|
||||||
@ -744,7 +748,7 @@ public abstract class Monitor
|
|||||||
|
|
||||||
// Reset the AccessControlContext.
|
// Reset the AccessControlContext.
|
||||||
//
|
//
|
||||||
acc = null;
|
acc = noPermissionsACC;
|
||||||
|
|
||||||
// Reset the complex type attribute information
|
// Reset the complex type attribute information
|
||||||
// such that it is recalculated again.
|
// such that it is recalculated again.
|
||||||
@ -1560,10 +1564,12 @@ public abstract class Monitor
|
|||||||
|
|
||||||
public void run() {
|
public void run() {
|
||||||
final ScheduledFuture<?> sf;
|
final ScheduledFuture<?> sf;
|
||||||
|
final AccessControlContext ac;
|
||||||
synchronized (Monitor.this) {
|
synchronized (Monitor.this) {
|
||||||
sf = Monitor.this.schedulerFuture;
|
sf = Monitor.this.schedulerFuture;
|
||||||
|
ac = Monitor.this.acc;
|
||||||
}
|
}
|
||||||
AccessController.doPrivileged(new PrivilegedAction<Void>() {
|
PrivilegedAction<Void> action = new PrivilegedAction<Void>() {
|
||||||
public Void run() {
|
public Void run() {
|
||||||
if (Monitor.this.isActive()) {
|
if (Monitor.this.isActive()) {
|
||||||
final int an[] = alreadyNotifieds;
|
final int an[] = alreadyNotifieds;
|
||||||
@ -1576,7 +1582,11 @@ public abstract class Monitor
|
|||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}, Monitor.this.acc);
|
};
|
||||||
|
if (ac == null) {
|
||||||
|
throw new SecurityException("AccessControlContext cannot be null");
|
||||||
|
}
|
||||||
|
AccessController.doPrivileged(action, ac);
|
||||||
synchronized (Monitor.this) {
|
synchronized (Monitor.this) {
|
||||||
if (Monitor.this.isActive() &&
|
if (Monitor.this.isActive() &&
|
||||||
Monitor.this.schedulerFuture == sf) {
|
Monitor.this.schedulerFuture == sf) {
|
||||||
|
@ -184,6 +184,7 @@ public class StringMonitor extends Monitor implements StringMonitorMBean {
|
|||||||
* @return The derived gauge of the specified object.
|
* @return The derived gauge of the specified object.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public synchronized String getDerivedGauge(ObjectName object) {
|
public synchronized String getDerivedGauge(ObjectName object) {
|
||||||
return (String) super.getDerivedGauge(object);
|
return (String) super.getDerivedGauge(object);
|
||||||
}
|
}
|
||||||
@ -199,6 +200,7 @@ public class StringMonitor extends Monitor implements StringMonitorMBean {
|
|||||||
* @return The derived gauge timestamp of the specified object.
|
* @return The derived gauge timestamp of the specified object.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public synchronized long getDerivedGaugeTimeStamp(ObjectName object) {
|
public synchronized long getDerivedGaugeTimeStamp(ObjectName object) {
|
||||||
return super.getDerivedGaugeTimeStamp(object);
|
return super.getDerivedGaugeTimeStamp(object);
|
||||||
}
|
}
|
||||||
@ -341,8 +343,9 @@ public class StringMonitor extends Monitor implements StringMonitorMBean {
|
|||||||
* the Java class of the notification and the notification types sent by
|
* the Java class of the notification and the notification types sent by
|
||||||
* the string monitor.
|
* the string monitor.
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public MBeanNotificationInfo[] getNotificationInfo() {
|
public MBeanNotificationInfo[] getNotificationInfo() {
|
||||||
return notifsInfo;
|
return notifsInfo.clone();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -52,6 +52,9 @@ class Request {
|
|||||||
os = rawout;
|
os = rawout;
|
||||||
do {
|
do {
|
||||||
startLine = readLine();
|
startLine = readLine();
|
||||||
|
if (startLine == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
/* skip blank lines */
|
/* skip blank lines */
|
||||||
} while (startLine.equals (""));
|
} while (startLine.equals (""));
|
||||||
}
|
}
|
||||||
|
@ -441,6 +441,7 @@ class ServerImpl implements TimeSource {
|
|||||||
rawin = sslStreams.getInputStream();
|
rawin = sslStreams.getInputStream();
|
||||||
rawout = sslStreams.getOutputStream();
|
rawout = sslStreams.getOutputStream();
|
||||||
engine = sslStreams.getSSLEngine();
|
engine = sslStreams.getSSLEngine();
|
||||||
|
connection.sslStreams = sslStreams;
|
||||||
} else {
|
} else {
|
||||||
rawin = new BufferedInputStream(
|
rawin = new BufferedInputStream(
|
||||||
new Request.ReadStream (
|
new Request.ReadStream (
|
||||||
@ -450,6 +451,8 @@ class ServerImpl implements TimeSource {
|
|||||||
ServerImpl.this, chan
|
ServerImpl.this, chan
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
connection.raw = rawin;
|
||||||
|
connection.rawout = rawout;
|
||||||
}
|
}
|
||||||
Request req = new Request (rawin, rawout);
|
Request req = new Request (rawin, rawout);
|
||||||
requestLine = req.requestLine();
|
requestLine = req.requestLine();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user