YARN-6805. NPE in LinuxContainerExecutor due to null PrivilegedOperationException exit code. Contributed by Jason Lowe

(cherry picked from commit ebc048cc05)
This commit is contained in:
Jason Lowe 2017-07-13 17:44:47 -05:00
parent 6ce2b0e4d4
commit a77fb561ef
4 changed files with 111 additions and 17 deletions

View File

@ -275,6 +275,10 @@ public class LinuxContainerExecutor extends ContainerExecutor {
} }
} }
protected PrivilegedOperationExecutor getPrivilegedOperationExecutor() {
return PrivilegedOperationExecutor.getInstance(getConf());
}
@Override @Override
public void init() throws IOException { public void init() throws IOException {
Configuration conf = super.getConf(); Configuration conf = super.getConf();
@ -285,7 +289,7 @@ public class LinuxContainerExecutor extends ContainerExecutor {
PrivilegedOperation checkSetupOp = new PrivilegedOperation( PrivilegedOperation checkSetupOp = new PrivilegedOperation(
PrivilegedOperation.OperationType.CHECK_SETUP); PrivilegedOperation.OperationType.CHECK_SETUP);
PrivilegedOperationExecutor privilegedOperationExecutor = PrivilegedOperationExecutor privilegedOperationExecutor =
PrivilegedOperationExecutor.getInstance(conf); getPrivilegedOperationExecutor();
privilegedOperationExecutor.executePrivilegedOperation(checkSetupOp, privilegedOperationExecutor.executePrivilegedOperation(checkSetupOp,
false); false);
@ -382,7 +386,7 @@ public class LinuxContainerExecutor extends ContainerExecutor {
try { try {
Configuration conf = super.getConf(); Configuration conf = super.getConf();
PrivilegedOperationExecutor privilegedOperationExecutor = PrivilegedOperationExecutor privilegedOperationExecutor =
PrivilegedOperationExecutor.getInstance(conf); getPrivilegedOperationExecutor();
privilegedOperationExecutor.executePrivilegedOperation(prefixCommands, privilegedOperationExecutor.executePrivilegedOperation(prefixCommands,
initializeContainerOp, null, null, false, true); initializeContainerOp, null, null, false, true);
@ -549,8 +553,9 @@ public class LinuxContainerExecutor extends ContainerExecutor {
} }
builder.append("Stack trace: " builder.append("Stack trace: "
+ StringUtils.stringifyException(e) + "\n"); + StringUtils.stringifyException(e) + "\n");
if (!e.getOutput().isEmpty()) { String output = e.getOutput();
builder.append("Shell output: " + e.getOutput() + "\n"); if (output != null && !e.getOutput().isEmpty()) {
builder.append("Shell output: " + output + "\n");
} }
String diagnostics = builder.toString(); String diagnostics = builder.toString();
logOutput(diagnostics); logOutput(diagnostics);
@ -704,7 +709,7 @@ public class LinuxContainerExecutor extends ContainerExecutor {
try { try {
Configuration conf = super.getConf(); Configuration conf = super.getConf();
PrivilegedOperationExecutor privilegedOperationExecutor = PrivilegedOperationExecutor privilegedOperationExecutor =
PrivilegedOperationExecutor.getInstance(conf); getPrivilegedOperationExecutor();
privilegedOperationExecutor.executePrivilegedOperation(deleteAsUserOp, privilegedOperationExecutor.executePrivilegedOperation(deleteAsUserOp,
false); false);
@ -734,7 +739,7 @@ public class LinuxContainerExecutor extends ContainerExecutor {
try { try {
PrivilegedOperationExecutor privOpExecutor = PrivilegedOperationExecutor privOpExecutor =
PrivilegedOperationExecutor.getInstance(super.getConf()); getPrivilegedOperationExecutor();
String results = String results =
privOpExecutor.executePrivilegedOperation(listAsUserOp, true); privOpExecutor.executePrivilegedOperation(listAsUserOp, true);
@ -793,7 +798,7 @@ public class LinuxContainerExecutor extends ContainerExecutor {
mountCGroupsOp.appendArgs(cgroupKVs); mountCGroupsOp.appendArgs(cgroupKVs);
PrivilegedOperationExecutor privilegedOperationExecutor = PrivilegedOperationExecutor privilegedOperationExecutor =
PrivilegedOperationExecutor.getInstance(conf); getPrivilegedOperationExecutor();
privilegedOperationExecutor.executePrivilegedOperation(mountCGroupsOp, privilegedOperationExecutor.executePrivilegedOperation(mountCGroupsOp,
false); false);

View File

@ -24,7 +24,7 @@ import org.apache.hadoop.yarn.exceptions.YarnException;
public class PrivilegedOperationException extends YarnException { public class PrivilegedOperationException extends YarnException {
private static final long serialVersionUID = 1L; private static final long serialVersionUID = 1L;
private Integer exitCode; private int exitCode = -1;
private String output; private String output;
private String errorOutput; private String errorOutput;
@ -36,7 +36,7 @@ public class PrivilegedOperationException extends YarnException {
super(message); super(message);
} }
public PrivilegedOperationException(String message, Integer exitCode, public PrivilegedOperationException(String message, int exitCode,
String output, String errorOutput) { String output, String errorOutput) {
super(message); super(message);
this.exitCode = exitCode; this.exitCode = exitCode;
@ -48,8 +48,8 @@ public class PrivilegedOperationException extends YarnException {
super(cause); super(cause);
} }
public PrivilegedOperationException(Throwable cause, Integer exitCode, String public PrivilegedOperationException(Throwable cause, int exitCode,
output, String errorOutput) { String output, String errorOutput) {
super(cause); super(cause);
this.exitCode = exitCode; this.exitCode = exitCode;
this.output = output; this.output = output;
@ -59,7 +59,7 @@ public class PrivilegedOperationException extends YarnException {
super(message, cause); super(message, cause);
} }
public Integer getExitCode() { public int getExitCode() {
return exitCode; return exitCode;
} }

View File

@ -32,10 +32,10 @@ import org.apache.hadoop.yarn.exceptions.YarnException;
@InterfaceStability.Unstable @InterfaceStability.Unstable
public class ContainerExecutionException extends YarnException { public class ContainerExecutionException extends YarnException {
private static final long serialVersionUID = 1L; private static final long serialVersionUID = 1L;
private static final Integer EXIT_CODE_UNSET = -1; private static final int EXIT_CODE_UNSET = -1;
private static final String OUTPUT_UNSET = "<unknown>"; private static final String OUTPUT_UNSET = "<unknown>";
private Integer exitCode; private int exitCode;
private String output; private String output;
private String errorOutput; private String errorOutput;
@ -54,7 +54,7 @@ public class ContainerExecutionException extends YarnException {
} }
public ContainerExecutionException(String message, Integer exitCode, String public ContainerExecutionException(String message, int exitCode, String
output, String errorOutput) { output, String errorOutput) {
super(message); super(message);
this.exitCode = exitCode; this.exitCode = exitCode;
@ -62,7 +62,7 @@ public class ContainerExecutionException extends YarnException {
this.errorOutput = errorOutput; this.errorOutput = errorOutput;
} }
public ContainerExecutionException(Throwable cause, Integer exitCode, String public ContainerExecutionException(Throwable cause, int exitCode, String
output, String errorOutput) { output, String errorOutput) {
super(cause); super(cause);
this.exitCode = exitCode; this.exitCode = exitCode;
@ -70,7 +70,7 @@ public class ContainerExecutionException extends YarnException {
this.errorOutput = errorOutput; this.errorOutput = errorOutput;
} }
public Integer getExitCode() { public int getExitCode() {
return exitCode; return exitCode;
} }

View File

@ -23,7 +23,9 @@ import static org.junit.Assert.assertNotEquals;
import static org.junit.Assert.assertTrue; import static org.junit.Assert.assertTrue;
import static org.junit.Assume.assumeTrue; import static org.junit.Assume.assumeTrue;
import static org.mockito.Matchers.any; import static org.mockito.Matchers.any;
import static org.mockito.Matchers.anyBoolean;
import static org.mockito.Mockito.doAnswer; import static org.mockito.Mockito.doAnswer;
import static org.mockito.Mockito.doThrow;
import static org.mockito.Mockito.mock; import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.spy; import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.when; import static org.mockito.Mockito.when;
@ -40,6 +42,7 @@ import java.util.Arrays;
import java.util.HashMap; import java.util.HashMap;
import java.util.LinkedList; import java.util.LinkedList;
import java.util.List; import java.util.List;
import java.util.Map;
import org.apache.commons.logging.Log; import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory; import org.apache.commons.logging.LogFactory;
@ -47,6 +50,8 @@ import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.FileUtil; import org.apache.hadoop.fs.FileUtil;
import org.apache.hadoop.fs.Path; import org.apache.hadoop.fs.Path;
import org.apache.hadoop.util.StringUtils; import org.apache.hadoop.util.StringUtils;
import org.apache.hadoop.yarn.api.records.ApplicationAttemptId;
import org.apache.hadoop.yarn.api.records.ApplicationId;
import org.apache.hadoop.yarn.api.records.ContainerId; import org.apache.hadoop.yarn.api.records.ContainerId;
import org.apache.hadoop.yarn.api.records.ContainerLaunchContext; import org.apache.hadoop.yarn.api.records.ContainerLaunchContext;
import org.apache.hadoop.yarn.conf.YarnConfiguration; import org.apache.hadoop.yarn.conf.YarnConfiguration;
@ -54,6 +59,7 @@ import org.apache.hadoop.yarn.exceptions.ConfigurationException;
import org.apache.hadoop.yarn.server.nodemanager.containermanager.container.Container; import org.apache.hadoop.yarn.server.nodemanager.containermanager.container.Container;
import org.apache.hadoop.yarn.server.nodemanager.containermanager.container.ContainerDiagnosticsUpdateEvent; import org.apache.hadoop.yarn.server.nodemanager.containermanager.container.ContainerDiagnosticsUpdateEvent;
import org.apache.hadoop.yarn.server.nodemanager.containermanager.linux.privileged.PrivilegedOperation; import org.apache.hadoop.yarn.server.nodemanager.containermanager.linux.privileged.PrivilegedOperation;
import org.apache.hadoop.yarn.server.nodemanager.containermanager.linux.privileged.PrivilegedOperationException;
import org.apache.hadoop.yarn.server.nodemanager.containermanager.linux.privileged.PrivilegedOperationExecutor; import org.apache.hadoop.yarn.server.nodemanager.containermanager.linux.privileged.PrivilegedOperationExecutor;
import org.apache.hadoop.yarn.server.nodemanager.containermanager.linux.runtime.DefaultLinuxContainerRuntime; import org.apache.hadoop.yarn.server.nodemanager.containermanager.linux.runtime.DefaultLinuxContainerRuntime;
import org.apache.hadoop.yarn.server.nodemanager.containermanager.linux.runtime.LinuxContainerRuntime; import org.apache.hadoop.yarn.server.nodemanager.containermanager.linux.runtime.LinuxContainerRuntime;
@ -516,4 +522,87 @@ public class TestLinuxContainerExecutorWithMocks {
appSubmitter, cmd, "", baseDir0.toString(), baseDir1.toString()), appSubmitter, cmd, "", baseDir0.toString(), baseDir1.toString()),
readMockParams()); readMockParams());
} }
@Test
public void testNoExitCodeFromPrivilegedOperation() throws Exception {
Configuration conf = new Configuration();
final PrivilegedOperationExecutor spyPrivilegedExecutor =
spy(PrivilegedOperationExecutor.getInstance(conf));
doThrow(new PrivilegedOperationException("interrupted"))
.when(spyPrivilegedExecutor).executePrivilegedOperation(
any(List.class), any(PrivilegedOperation.class),
any(File.class), any(Map.class), anyBoolean(), anyBoolean());
LinuxContainerRuntime runtime = new DefaultLinuxContainerRuntime(
spyPrivilegedExecutor);
runtime.initialize(conf);
mockExec = new LinuxContainerExecutor(runtime);
mockExec.setConf(conf);
LinuxContainerExecutor lce = new LinuxContainerExecutor(runtime) {
@Override
protected PrivilegedOperationExecutor getPrivilegedOperationExecutor() {
return spyPrivilegedExecutor;
}
};
lce.setConf(conf);
InetSocketAddress address = InetSocketAddress.createUnresolved(
"localhost", 8040);
Path nmPrivateCTokensPath= new Path("file:///bin/nmPrivateCTokensPath");
LocalDirsHandlerService dirService = new LocalDirsHandlerService();
dirService.init(conf);
String appSubmitter = "nobody";
ApplicationId appId = ApplicationId.newInstance(1, 1);
ApplicationAttemptId attemptId = ApplicationAttemptId.newInstance(appId, 1);
ContainerId cid = ContainerId.newContainerId(attemptId, 1);
HashMap<String, String> env = new HashMap<>();
Container container = mock(Container.class);
ContainerLaunchContext context = mock(ContainerLaunchContext.class);
when(container.getContainerId()).thenReturn(cid);
when(container.getLaunchContext()).thenReturn(context);
when(context.getEnvironment()).thenReturn(env);
Path workDir = new Path("/tmp");
try {
lce.startLocalizer(new LocalizerStartContext.Builder()
.setNmPrivateContainerTokens(nmPrivateCTokensPath)
.setNmAddr(address)
.setUser(appSubmitter)
.setAppId(appId.toString())
.setLocId("12345")
.setDirsHandler(dirService)
.build());
Assert.fail("startLocalizer should have thrown an exception");
} catch (IOException e) {
assertTrue("Unexpected exception " + e,
e.getMessage().contains("exitCode"));
}
lce.activateContainer(cid, new Path(workDir, "pid.txt"));
lce.launchContainer(new ContainerStartContext.Builder()
.setContainer(container)
.setNmPrivateContainerScriptPath(new Path("file:///bin/echo"))
.setNmPrivateTokensPath(new Path("file:///dev/null"))
.setUser(appSubmitter)
.setAppId(appId.toString())
.setContainerWorkDir(workDir)
.setLocalDirs(dirsHandler.getLocalDirs())
.setLogDirs(dirsHandler.getLogDirs())
.setFilecacheDirs(new ArrayList<String>())
.setUserLocalDirs(new ArrayList<String>())
.setContainerLocalDirs(new ArrayList<String>())
.setContainerLogDirs(new ArrayList<String>())
.build());
lce.deleteAsUser(new DeletionAsUserContext.Builder()
.setUser(appSubmitter)
.setSubDir(new Path("/tmp/testdir"))
.build());
try {
lce.mountCgroups(new ArrayList<String>(), "hierarchy");
Assert.fail("mountCgroups should have thrown an exception");
} catch (IOException e) {
assertTrue("Unexpected exception " + e,
e.getMessage().contains("exit code"));
}
}
} }