|
|
@ -17,20 +17,27 @@
|
|
|
|
package org.springframework.boot.actuate.system;
|
|
|
|
package org.springframework.boot.actuate.system;
|
|
|
|
|
|
|
|
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.File;
|
|
|
|
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.util.concurrent.atomic.AtomicBoolean;
|
|
|
|
import java.util.concurrent.atomic.AtomicBoolean;
|
|
|
|
|
|
|
|
|
|
|
|
import org.apache.commons.logging.Log;
|
|
|
|
import org.apache.commons.logging.Log;
|
|
|
|
import org.apache.commons.logging.LogFactory;
|
|
|
|
import org.apache.commons.logging.LogFactory;
|
|
|
|
import org.springframework.boot.ApplicationPid;
|
|
|
|
import org.springframework.boot.ApplicationPid;
|
|
|
|
import org.springframework.boot.context.event.ApplicationStartedEvent;
|
|
|
|
import org.springframework.boot.bind.RelaxedPropertyResolver;
|
|
|
|
|
|
|
|
import org.springframework.boot.context.event.ApplicationEnvironmentPreparedEvent;
|
|
|
|
|
|
|
|
import org.springframework.boot.context.event.ApplicationPreparedEvent;
|
|
|
|
|
|
|
|
import org.springframework.boot.context.event.SpringApplicationEvent;
|
|
|
|
import org.springframework.context.ApplicationListener;
|
|
|
|
import org.springframework.context.ApplicationListener;
|
|
|
|
import org.springframework.core.Ordered;
|
|
|
|
import org.springframework.core.Ordered;
|
|
|
|
|
|
|
|
import org.springframework.core.env.Environment;
|
|
|
|
import org.springframework.util.Assert;
|
|
|
|
import org.springframework.util.Assert;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* An {@link ApplicationListener} that saves application PID into file. This application
|
|
|
|
* An {@link ApplicationListener} that saves application PID into file. This application
|
|
|
|
* listener will be triggered exactly once per JVM, and the file name can be overridden at
|
|
|
|
* listener will be triggered exactly once per JVM, and the file name can be overridden at
|
|
|
|
* runtime with a System property or environment variable named "PIDFILE" (or "pidfile").
|
|
|
|
* runtime with a System property or environment variable named "PIDFILE" (or "pidfile")
|
|
|
|
|
|
|
|
* or using a {@code spring.application.pidfile} property in the Spring
|
|
|
|
|
|
|
|
* {@link Environment}.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* @author Jakub Kubrynski
|
|
|
|
* @author Jakub Kubrynski
|
|
|
|
* @author Dave Syer
|
|
|
|
* @author Dave Syer
|
|
|
@ -38,13 +45,15 @@ import org.springframework.util.Assert;
|
|
|
|
* @since 1.2.0
|
|
|
|
* @since 1.2.0
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
public class ApplicationPidFileWriter implements
|
|
|
|
public class ApplicationPidFileWriter implements
|
|
|
|
ApplicationListener<ApplicationStartedEvent>, Ordered {
|
|
|
|
ApplicationListener<SpringApplicationEvent>, Ordered {
|
|
|
|
|
|
|
|
|
|
|
|
private static final Log logger = LogFactory.getLog(ApplicationPidFileWriter.class);
|
|
|
|
private static final Log logger = LogFactory.getLog(ApplicationPidFileWriter.class);
|
|
|
|
|
|
|
|
|
|
|
|
private static final String DEFAULT_FILE_NAME = "application.pid";
|
|
|
|
private static final String DEFAULT_FILE_NAME = "application.pid";
|
|
|
|
|
|
|
|
|
|
|
|
private static final String[] PROPERTY_VARIABLES = { "PIDFILE", "pidfile" };
|
|
|
|
private static final String[] SYSTEM_PROPERTY_VARIABLES = { "PIDFILE", "pidfile" };
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private static final String SPRING_PROPERTY = "spring.application.pidfile";
|
|
|
|
|
|
|
|
|
|
|
|
private static final AtomicBoolean created = new AtomicBoolean(false);
|
|
|
|
private static final AtomicBoolean created = new AtomicBoolean(false);
|
|
|
|
|
|
|
|
|
|
|
@ -52,12 +61,14 @@ public class ApplicationPidFileWriter implements
|
|
|
|
|
|
|
|
|
|
|
|
private final File file;
|
|
|
|
private final File file;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private Class<? extends SpringApplicationEvent> triggerEventType = ApplicationPreparedEvent.class;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Create a new {@link ApplicationPidFileWriter} instance using the filename
|
|
|
|
* Create a new {@link ApplicationPidFileWriter} instance using the filename
|
|
|
|
* 'application.pid'.
|
|
|
|
* 'application.pid'.
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
public ApplicationPidFileWriter() {
|
|
|
|
public ApplicationPidFileWriter() {
|
|
|
|
this.file = new File(DEFAULT_FILE_NAME);
|
|
|
|
this(new File(DEFAULT_FILE_NAME));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
@ -74,27 +85,65 @@ public class ApplicationPidFileWriter implements
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
public ApplicationPidFileWriter(File file) {
|
|
|
|
public ApplicationPidFileWriter(File file) {
|
|
|
|
Assert.notNull(file, "File must not be null");
|
|
|
|
Assert.notNull(file, "File must not be null");
|
|
|
|
String override = SystemProperties.get(PROPERTY_VARIABLES);
|
|
|
|
|
|
|
|
if (override != null) {
|
|
|
|
|
|
|
|
this.file = new File(override);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else {
|
|
|
|
|
|
|
|
this.file = file;
|
|
|
|
this.file = file;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Sets the type of application event that will trigger writing of the PID file.
|
|
|
|
|
|
|
|
* Defaults to {@link ApplicationPreparedEvent}. NOTE: If you use the
|
|
|
|
|
|
|
|
* {@link ApplicationPreparedEvent} to trigger the write, you will not be able to
|
|
|
|
|
|
|
|
* specify the PID filename in the Spring {@link Environment}.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
public void setTriggerEventType(
|
|
|
|
|
|
|
|
Class<? extends SpringApplicationEvent> triggerEventType) {
|
|
|
|
|
|
|
|
Assert.notNull(triggerEventType, "Trigger event type must not be null");
|
|
|
|
|
|
|
|
this.triggerEventType = triggerEventType;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
@Override
|
|
|
|
public void onApplicationEvent(ApplicationStartedEvent event) {
|
|
|
|
public void onApplicationEvent(SpringApplicationEvent event) {
|
|
|
|
|
|
|
|
if (this.triggerEventType.isInstance(event)) {
|
|
|
|
if (created.compareAndSet(false, true)) {
|
|
|
|
if (created.compareAndSet(false, true)) {
|
|
|
|
try {
|
|
|
|
try {
|
|
|
|
new ApplicationPid().write(this.file);
|
|
|
|
writePidFile(event);
|
|
|
|
this.file.deleteOnExit();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (Exception ex) {
|
|
|
|
catch (Exception ex) {
|
|
|
|
logger.warn(String.format("Cannot create pid file %s", this.file));
|
|
|
|
logger.warn(String.format("Cannot create pid file %s", this.file));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private void writePidFile(SpringApplicationEvent event) throws IOException {
|
|
|
|
|
|
|
|
File pidFile = this.file;
|
|
|
|
|
|
|
|
String override = SystemProperties.get(SYSTEM_PROPERTY_VARIABLES);
|
|
|
|
|
|
|
|
if (override != null) {
|
|
|
|
|
|
|
|
pidFile = new File(override);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else {
|
|
|
|
|
|
|
|
Environment environment = getEnvironment(event);
|
|
|
|
|
|
|
|
if (environment != null) {
|
|
|
|
|
|
|
|
override = new RelaxedPropertyResolver(environment)
|
|
|
|
|
|
|
|
.getProperty(SPRING_PROPERTY);
|
|
|
|
|
|
|
|
if (override != null) {
|
|
|
|
|
|
|
|
pidFile = new File(override);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
new ApplicationPid().write(pidFile);
|
|
|
|
|
|
|
|
pidFile.deleteOnExit();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private Environment getEnvironment(SpringApplicationEvent event) {
|
|
|
|
|
|
|
|
if (event instanceof ApplicationEnvironmentPreparedEvent) {
|
|
|
|
|
|
|
|
return ((ApplicationEnvironmentPreparedEvent) event).getEnvironment();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (event instanceof ApplicationPreparedEvent) {
|
|
|
|
|
|
|
|
return ((ApplicationPreparedEvent) event).getApplicationContext()
|
|
|
|
|
|
|
|
.getEnvironment();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void setOrder(int order) {
|
|
|
|
public void setOrder(int order) {
|
|
|
|
this.order = order;
|
|
|
|
this.order = order;
|
|
|
|