前言
在插件开发中,我们经常需要把一些配置信息保存到服务器中,以便重启服务器的时候能够记录上一次保存的信息,这一点和设计器配置信息的保存和读取差不多。
实现方式
要保存信息到服务器上,一般来说,我们可以把数据存储到数据库或者xml文件中,这里介绍的是把数据保存到xml文件中。
关联接口
主要使用接口XMLFileManagerProvider。
实现过程
考虑到FineReport是支持本地设计和远程设计的,所以配置文件在读写的时候也需要考虑是否是远程环境,此外,还需要考虑集群环境中,配置文件的读取。
首先,需要继承类
com.fr.file.XMLFileManager
这个类用于读取WEB-INF/resources下的xml文件,继承了这个类,我们就不用去考虑是远程设计还是本地设计了,FineReport的读写资源的API已经为我们处理好了这一切。
其次,为了在集群环境中也能正确的读取配置文件,需要给实现类抽象出来一个接口,假设我们的配置文件类的名字叫ExpirationConfigManager,那么我们需要抽象出来一个接口叫ExpirationConfigManagerProvider
public interface ExpirationConfigManagerProvider extends RemoteXMLFileManagerProvider, java.io.Serializable, FCloneable {
int getValidMonthCount();
void setValidMonthCount(int validMonthCount);
int getWarningDayCount();
void setWarningDayCount(int count);
}
在ExpirationConfigManagerProvider接口中,我们需要提供对外调用的所有方法接口。
一般来说,配置信息类都以一个单例的形式存在。下面来看一看这个实现示例:
public class ExpirationConfigManager extends XMLFileManager implements ExpirationConfigManagerProvider {
private static final String XML_TAG = "ExpirationConfigManager";
private static ExpirationConfigManagerProvider manager = null;
public synchronized static ExpirationConfigManager getInstance() {
return (ExpirationConfigManager) getProviderInstance();
}
public synchronized static ExpirationConfigManagerProvider getProviderInstance() {
if (manager == null) {
if (isClusterMember()) {
return manager;
}
manager.readXMLFile();
}
return manager;
}
private synchronized static boolean isClusterMember() {
switch (BaseClusterHelper.getClusterState()) {
case LEADER:
manager = new ExpirationConfigManager();
RPC.registerSkeleton(manager);
return false;
case MEMBER:
String ip = BaseClusterHelper.getMainServiceIP();
manager = (ExpirationConfigManagerProvider) RPC.getProxy(ExpirationConfigManager.class, ip);
return true;
default:
manager = new ExpirationConfigManager();
break;
}
return false;
}
static {
GeneralContext.addEnvChangedListener(new EnvChangedListener() {
public void envChanged() {
ExpirationConfigManager.envChanged();
}
});
}
private synchronized static void envChanged() {
manager = null;
}
private int validMonthCount = 1;
private int warningDayCount = 6;
public int getValidMonthCount() {
return validMonthCount;
}
public void setValidMonthCount(int validMonthCount) {
this.validMonthCount = validMonthCount;
}
@Override
public int getWarningDayCount() {
return warningDayCount;
}
@Override
public void setWarningDayCount(int count) {
warningDayCount = count;
}
@Override
public String fileName() {
return "expiration.xml";
}
@Override
public void readXML(XMLableReader reader) {
if (reader.isAttr()) {
validMonthCount = reader.getAttrAsInt("validMonthCount", 1);
warningDayCount = reader.getAttrAsInt("warningDayCount", 6);
}
}
@Override
public void writeXML(XMLPrintWriter writer) {
writer.startTAG(XML_TAG);
writer.attr("validMonthCount", validMonthCount);
writer.attr("warningDayCount", warningDayCount);
writer.end();
}
@Override
public boolean writeResource() throws Exception {
return GeneralContext.getEnvProvider().writeResource(ExpirationConfigManager.getProviderInstance());
}
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
调用方式
int count = ExpirationConfigManager.getProviderInstance().getWarningDayCount();
保存信息
通过下面的方式,就可以把内存中信息保存到xml文件下了:
ExpirationConfigManagerProvider configManager = ExpirationConfigManager.getProviderInstance();
configManager.setWarningDayCount(warningCount);
configManager.setValidMonthCount(month);
FRContext.getCurrentEnv().writeResource(configManager);
特别注意
为了远程设计的时候,配置文件能够正确的保存,需要实现readFromInputStream方法,以ExpirationConfigManager为例:
// 服务器端新建一个对象
ExpirationConfigManager manager = new ExpirationConfigManager();
// 从客户端传过来的inputstream中读取对象属性
XMLTools.readInputStreamXML(manager, input);
// 赋值给当前服务器端对象
expirationConfigManager= manager;
// 服务器端保存到本地xml中
GeneralContext.getEnvProvider().writeResource(datasourceManager);
注册到插件中
<extra-core>
<XMLFileManagerProvider class="com.fr.plugin.demo.ExpirationConfigManager">
</extra-core>
其他
插件开发的时候,通常在插件包中会包含一个默认的配置文件,需要在插件安装的时候将这个配置文件移动到web应用目录之下,可以通过修改plugin.xml中的节点
<extra-core>
<XMLFileManagerProvider class="com.fr.plugin.demo.ExpirationConfigManager">
</extra-core>
<move-after-install>
<File name="myconfig.xml" dir="/resources" operation="0"/>
</move-after-install>
这样就会在插件安装的时候把插件包中的myconfig.xml文件移动到WEB-INF/resources目录下了,dir属性是一个相对路径,可以根据实际情况修改。