The process of adding a new runtime to a WSO2 product is explained below. For the full list of capabilities available in this kernel version, see the features section in the root README.md file.
From Carbon 5 onwards, Carbon provides a pluggable interface to add runtimes to the existing server. Following are the instructions that you need to follow when adding a new runtime.
The following example illustrates how you can plug your own runtime and register it with the Carbon runtime framework. In this example, we will run through the steps for plugging the Tomcat runtime (which is currently available with the Carbon 4.2.0 release) to Carbon 5.
-
Create a simple maven project with the following dependencies.
<dependency> <groupId>org.wso2.carbon</groupId> <artifactId>org.wso2.carbon.core</artifactId> <version>5.0.0</version> </dependency>
You may also need to add other dependencies according to the requirement.
- Implement the runtime interface. See the example given below.
This code sample contains the implementation that was done for Carbon 4.2.0 components. In this code segment, the current Tomcat integration is re-factored to Carbon at http://svn.wso2.org/repos/wso2/carbon/kernel/branches/4.2.0/core/org.wso2.carbon.tomcat/4.2.0/
.
public class TomcatRuntime implements Runtime {
private static Log log = LogFactory.getLog(TomcatRuntime.class);
private static CarbonTomcatService carbonTomcatService;
private InputStream inputStream;
static ClassLoader bundleCtxtClassLoader;
private RuntimeState state = RuntimeState.PENDING;
/**
* initialization code goes here.i.e : configuring TomcatService instance using catalina-server.xml
*/
@Override
public void init() {
bundleCtxtClassLoader = Thread.currentThread().getContextClassLoader();
String carbonHome = System.getProperty("carbon.home");
String catalinaHome = new File(carbonHome).getAbsolutePath() + File.separator + "lib" +
File.separator + "tomcat";
String catalinaXML = new File(carbonHome).getAbsolutePath() + File.separator +
"repository" + File.separator + "conf" + File.separator +
"tomcat" + File.separator + "catalina-server.xml";
try {
inputStream = new FileInputStream(new File(catalinaXML));
} catch (FileNotFoundException e) {
log.error("could not locate the file catalina-server.xml", e);
}
//setting catalina.base system property. carbonTomcatService configurator refers this property while carbonTomcatService instance creation.
//you can override the property in wso2server.sh
if (System.getProperty("catalina.base") == null) {
System.setProperty("catalina.base", System.getProperty("carbon.home") + File.separator +
"lib" + File.separator + "tomcat");
}
carbonTomcatService = new CarbonTomcatService();
carbonTomcatService.configure(catalinaHome, inputStream);
state = RuntimeState.INACTIVE;
}
/**
* starting the a carbonTomcatService instance in a new thread. Otherwise activator gets blocked.
*/
@Override
public synchronized void start() {
new Thread(new Runnable() {
public void run() {
Thread.currentThread().setContextClassLoader(bundleCtxtClassLoader);
try {
carbonTomcatService.start();
state = RuntimeState.ACTIVE;
} catch (LifecycleException e) {
log.error("carbonTomcatService life-cycle exception", e);
}
}
}).start();
}
/**
* stopping the carbonTomcatService instance
*/
@Override
public void stop() {
try {
carbonTomcatService.stop();
state = RuntimeState.INACTIVE;
} catch (LifecycleException e) {
log.error("Error while stopping carbonTomcatService", e);
}
}
@Override
public void startMaintenance() {
// work specific to startMaintenance
state = RuntimeState.MAINTENANCE;
}
@Override
public void stopMaintenance() {
// work specific to stopMaintenance
state = RuntimeState.INACTIVE;
}
@Override
public Enum<RuntimeState> getState() {
return state;
}
@Override
public void setState(RuntimeState runtimeState) {
this.state = runtimeState;
}
/**
* we are not expecting others to access this service. The only use case would be activator.
* hence package private access modifier
*
* @return
*/
CarbonTomcatService getTomcatInstance() {
return carbonTomcatService;
}
}
-
Write a bundle activator for the above runtime as shown below.
public class TomcatRuntimeActivator implements BundleActivator { private static Log log = LogFactory.getLog(TomcatRuntimeActivator.class); private TomcatRuntime tomcatRuntime; private ServiceRegistration serviceRegistration; private ServiceListener serviceListener; public void start(BundleContext bundleContext) throws Exception { try { this.tomcatRuntime = new TomcatRuntime(); tomcatRuntime.init(); // we'll start the server once our transports get up // this is done by startup finalizer it will start all the runtimes // runtime will only serve request once the runtime is start() serviceRegistration = bundleContext.registerService(Runtime.class.getName(), tomcatRuntime, null); serviceRegistration = bundleContext.registerService(TomcatService.class.getName(), tomcatRuntime.getTomcatInstance(), null); if (log.isDebugEnabled()) { log.debug("Registering the JNDI stream handler..."); } //registering JNDI stream handler Hashtable<String, String[]> properties = new Hashtable<String, String[]>(); properties.put(URLConstants.URL_HANDLER_PROTOCOL, new String[]{"jndi"}); bundleContext.registerService(URLStreamHandlerService.class.getName(), new JNDIURLStreamHandlerService(), properties); } catch (Throwable t) { log.fatal("Error while starting Tomcat " + t.getMessage(), t); //do not throw because framework will keep trying. catching throwable is a bad thing, but //looks like we have no other option. } } public void stop(BundleContext bundleContext) throws Exception { this.tomcatRuntime.stop(); serviceRegistration.unregister(); tomcatRuntime = null; } }
-
Once the above points are addressed in your project, we need to add the Maven bundle plugin properties to generate the
component-level
metadata forscr
annotations and bundle info. An example of the POM file is given below. The packaging should be 'bundle' in thepom.xml
.<properties> <bundle.activator>org.wso2.carbon.tomcat.internal.TomcatRuntimeActivator</bundle.activator> <private.package> org.wso2.carbon.tomcat.internal.*, org.wso2.carbon.tomcat.jndi.* </private.package> <export.package> !org.wso2.carbon.tomcat.internal.*, org.wso2.carbon.tomcat.*, org.wso2.carbon.tomcat.api.*, org.wso2.carbon.tomcat.server.*; version="${project.version}" </export.package> <import.package> org.apache.tomcat.*;version="[1.7.0,2.0.0)", org.apache.catalina.*;version="[1.7.0,2.0.0)", org.apache.naming.*;version="[1.7.0,2.0.0)", *;resolution:=optional </import.package> </properties>
You can test the new runtime by following the steps given below.
- Build your component, which will generate an OSGi bundle.
- Now this bundle can be installed in a running Carbon server instance by adding the bundle to the dropins directory in the OSGi repository (which is
<CARBON_HOME>/osgi
). Once you have added your bundle to the dropins directory, it will be installed to the OSGi runtime by the Kernel launcher.
Find out more about how the dropins directory is used for deploying bundles.