在上篇中介绍了从JAVA中获取Domino数据库中的文档信息,但有一个缺点,就是必须要有本地Notes,即限制了只能在Windows下使用,在有些情况下,可能需要在其他操作系统中通过JAVA获取Domino数据,本文即为介绍在JAVA中使用CORBA实现跨操作系统访问Domino。本文介绍的Domino版本应该要在R5以上,R5以下版本未经验证,无法确认是否可行。注意,本文中需要使用lotus.domino.corba包,网络上找到这个库源码非常不容易,作者也是花费了不少时间才从国外一个小网站上下载到,如果有需要,请联系作者。Corba访问Domino过程比较复杂,大致如下:
String ior = http://127.0.0.1/diiop_ior.txt;
Properties props = new Properties();
props.put("org.openorb.orb.core.ORB", "org.openorb.orb.core.ORB");
ORB orb = ORB.init(new String[]{}, props);
// 通过 IOR 得到 IObjectServer 对象
org.omg.CORBA.Object obj = orb.string_to_object(ior);
IObjectServer ios = IObjectServerHelper.narrow(obj);
// 通过 IObjectServer 获得 ISession
ProtocolVersion maxVersion = new ProtocolVersion(IBase.DOM_MAJOR_MINIMUM_VERSION, IBase.DOM_MINOR_MINIMUM_VERSION);
ProtocolVersion minVersion = new ProtocolVersion(IBase.DOM_MAJOR_VERSION,IBase.DOM_MINOR_VERSION);
SessionData sd = ios.createSession(maxVersion, minVersion, ((CrawlTask)job.getCrawltask()).getAccount() , ((CrawlTask)job.getCrawltask()).getPassword());
session = sd.sesObject;
通过账号和密码访问登录Domino后获得session,然后通过session获得数据库信息,如下:
// 使用 ISession
dbCache = session.getDatabase(sd.serverName, properties.getProperty("database"), false);
dCdata = dbCache.db.getAllDocuments();//iDatabase.search("1=1", null, 10);
IDatabaseHolder idatabaseholder = new IDatabaseHolder();
IntHolder intholder = new IntHolder();
IDocument document = dCdata.dcObject.getFirstDocMDB(idatabaseholder, intholder);
一样是或得到document,类型为IDocument,dCdata.dcObject即是文档列表,对此对象遍历即可
document = dCdata.dcObject.getNextDocMDB(document ,idatabaseholder, intholder);
对IDocument对象的遍历也和DIIOP也有一些差别:
ItemData[] id = fieldDoc.getData().items;
for (int i = 0; i < id.length; i++) {
MetaType metaType = new MetaType(id[i].name,String.valueOf(id[i].type)) ;
ItemValue item = id[i].values;
if (id[i].type == 1280) { //表示是文本
metaType.setValue(item.StringValue()) ;
} else if (id[i].type == 1024) { //表示是时间
DateTime dateTime = item.TimeObject() ;
if(dateTime!=null){
metaType.setValue(String.valueOf(dateTime.toString())) ;
}
} else if (id[i].type == 768) { //表示是整型
metaType.setValue(String.valueOf(String.valueOf(item.DoubleValue()))) ;
} else if (id[i].type == 1) { //表示是rtf域
//获得RTF域的附件信息
} else{
metaType.setValue(item.StringValue()) ;
}
}
类型代码和上篇中介绍的都是一样的,但RTF域的附件信息获取方式不太一样,需要对items[i].RTObject对象进行进一步处理,处理过程如下:
IRichTextItem rtf = items[i].RTObject ;
if(item.StringValue()!=null){
outputText.getMetadata().add(items[i].name, item.StringValue()) ;
}else if(rtf.getValueAsString()!=null){
outputText.getMetadata().add(items[i].name, rtf.getValueAsString()) ;
}
{
NameAndObject[] ed = rtf.getEmbeddedObjects() ;
if(ed.length>0){
for(NameAndObject nameObject : ed){
EmbeddedData eo = document.getAttachment(nameObject.name) ;
if(eo!=null && eo.embedObject!=null && eo.embedObject.getFileSize()>0){
BooleanHolder holder = new BooleanHolder(true);
IntHolder inholder = new IntHolder();
byte[] data = null ;
java.io.ByteArrayOutputStream output = new java.io.ByteArrayOutputStream() ;
try{
while((data = eo.embedObject.getFile(inholder, holder))!=null && data.length>0){
output.write(data) ;
}
}catch(Exception ex){
ex.printStackTrace();
}
java.io.ByteArrayInputStream input = new ByteArrayInputStream(output.toByteArray()) ;
//获得输入流,自定义后续处理过程
input.close() ;
}
}
}
}
最后是关闭session,调用 session.recycle()。