为JBoss AS7添加新模块

注:文本的内容主要探讨JBoss AS7的实现细节,涉及AS7的内核代码及分析,需要耗费大量时间与精力来学习理解。如果你只是基于JBoss进行应用开发或是日常运维管理,可以忽略本文,看看网站上其它和AS7使用及开发相关的内容。

在软件领域,模块化的设计思想变得越来越重要,它的目的是使得软件变得容易扩展。在这一方面,最典型的例子当属Linux内核:Linux Kernel通过模块化设计,允许开发人员撰写功能代码,并以驱动模块的方式加载进内核,使得系统扩展变得非常容易。JBoss AS7吸取了不少来自于操作系统方面的设计经验,通过"模块化"的设计,让开发人员可以通过搭积木的方式,向AS7中添加新的组件。

AS7有篇文档通过代码示例详细介绍了如何为AS7撰写新的模块,并加载进AS7核心:

https://docs.jboss.org/author/display/AS7/Extending+JBoss+AS+7


这篇文档中的语言比较简练,有些点说明的不够细致,因此撰写本文的 目的是针对这个例子做出一些补充说明帮助理解,因此,将上面这篇文档和本文合起来阅读才是完整的。

首先,阅读上面这篇文档需要一些准备知识:AS7使用JBoss Modules来实现模块化的设计,在核心中依赖JBoss Modules进行模块的定义,加载,生命周期的管理。因此,首先需要理解JBoss Modules,可以参考下面这篇文档:

http://bluedash.net/spaces/JBoss%20Module介绍


如果你对AS7的基本使用还不熟悉,可以参考这篇文档:

http://bluedash.net/spaces/JBoss%20AS%207%20快速上手


这些准备工作做完后,就可以开始阅读 _Extending JBoss AS 7_ 。跟着文档里面的示例一步一步实验,到最后调通示例并对整体思路有个比较清楚的把握,大概需要10几个小时的时间,所以请准备出三、四天左右的时间来看这个示例。

我把一些原文中阐述不够详尽的地方,补充说明如下:

示例说明及使用

Extending JBoss AS 7中的示例的目的是为AS7制作一个记录已部署项目数量的模块。当这个模块加载进AS7以后,AS7的日志会定时输出系统中已部署的项目数量。AS7的日志输出类似如下:

12:04:24,935 INFO  [stdout] (Thread-12) Current deployments deployed while war tracking active:
12:04:24,935 INFO  [stdout] (Thread-12) [my-app.war]
12:04:24,935 INFO  [stdout] (Thread-12) Cool: 0


这个模块的名字叫做Tracker。这个名字是在SubsystemExtension中定义的:

public class SubsystemExtension implements Extension {
...
public static final String SUBSYSTEM_NAME = "tracker";
}
                 

整个模块的加载入口点为SubsystemExtension的initialize方法:

public class SubsystemExtension implements Extension {
...
    @Override
    public void initialize(ExtensionContext context) {
        ...
        final ManagementResourceRegistration registration = subsystem.registerSubsystemModel(SubsystemProviders.SUBSYSTEM);
        ...
        //Add the type child
        ManagementResourceRegistration typeChild = registration.registerSubModel(PathElement.pathElement("type"), SubsystemProviders.TYPE_CHILD);
        ...
        subsystem.registerXMLElementWriter(parser);
    }
}


initialize中将模块中的各组件给装配到一起。两个ManagementResourceRegistration将组件注册进AS7;最后一行subsystem.registerXMLElementWriter(parser)负责tracker在AS7配置文件中配置内容的读取工作。以下是AS7中tracker相关的配置:

<extensions>
	<extension module="com.acme.corp.tracker"/>
</extensions>

<profile>
<subsystem xmlns="urn:com.acme.corp.tracker:1.0">
	<deployment-types>
		<deployment-type suffix="sar" tick="10000"/>
		<deployment-type suffix="war" tick="10000"/>
	</deployment-types>
</subsystem>
</profile>


源代码中的parser就是负责将模块的配置转化成xml。Parser定义在SubsystemExtension中的SubsystemParser:

private static class SubsystemParser implements XMLStreamConstants, XMLElementReader<List<ModelNode>>, XMLElementWriter<SubsystemMarshallingContext> {

	@Override
	public void writeContent(XMLExtendedStreamWriter writer, SubsystemMarshallingContext context) throws XMLStreamException {
		...
	}

	@Override
	public void readElement(XMLExtendedStreamReader reader, List<ModelNode> list) throws XMLStreamException {
		...
	}
}


此外,在整个模块的加载入口点,SubsystemExtension的initialize方法中,注册了SubsystemAdd:

public class SubsystemExtension implements Extension {
...
    @Override
    public void initialize(ExtensionContext context) {
        ...
        //We always need to add an 'add' operation
        registration.registerOperationHandler(ADD, SubsystemAdd.INSTANCE, SubsystemProviders.SUBSYSTEM_ADD, false);
    }
}


AS7内部定义了模块的生命周期,其中Add是指模块被添加时要做的事,因此SubsystemAdd就是在模块被添加进AS7内核时,要执行的工作:

class SubsystemAdd extends AbstractBoottimeAddStepHandler {
	@Override
	public void performBoottime(...) throws OperationFailedException {
 
			processorTarget.addDeploymentProcessor(SubsystemDeploymentProcessor.PHASE, SubsystemDeploymentProcessor.PRIORITY, new SubsystemDeploymentProcessor());
	}

}


注意到SubsystemAdd扩展的是AbstractBoottimeAddStepHandler,从名字就可以看出,AbstractBoottimeAddStepHandler的生命周期对应于模块启动时加载的这个环节。里面的performBoottime方法将将整个模块的主要功能,SubsystemDeploymentProcessor,添加进AS7内核。接下来我们可以看下SubsystemDeploymentProcessor:

public class SubsystemDeploymentProcessor implements DeploymentUnitProcessor {
    @Override
    public void deploy(...) throws DeploymentUnitProcessingException {
    	...
    }

    @Override
    public void undeploy(...) {
    	...
    }
}


SubsystemDeploymentProcessor实现的是DeploymentUnitProcessor,这是AS7提供的具多功能Processor模块当中的一种。开发者可根据自己的功能需要,选择不同的Processor来进行实现。这个DeploymentUnitProcessor从名字就可以看出,它是处理和工程部署有关的工作时,需要使用的Processor接口。

DeploymentUnitProcessor提供两个方法:deploy和undeploy,分别对应于当有新项目部署进AS7,或有既存项目从AS7中去除时,系统要做的事情。因此例子中的代码通过实现这两个方法,就完成了对AS7中部署项目的跟踪记录工作。

以上是整个例子的设计思路及脉络。如果要想完全理解这个例子中的全部细节,还需要拿出几天的时间来仔细阅读文档,并实际动手玩玩看这个例子。一旦掌握了AS7的模块化设计思路,你就可以慢慢体会体会到它的方便之处,并可以很方便地对AS7进行功能扩展和定制。

总的来讲,对AS7的模块开发,变得越来越像对Linux Kernel的驱动开发。理解了这种模块化的设计思路,对于自己架构应用时也很有助益。

你可能感兴趣的:(jboss,as7)