spring ldap简单的例子(四)
好了通过前面的文章已经可以用程序实现对LDAP的添加了,下面将继续进阶,通过程序读取Excel中的数据插入到LDAP中,说白了就是批量操作,看懂前面两个例子的基本也可以想来怎样进行导入,所以我这里稍加改动,我先把需求阐述一下,明白我们要做什么,以及为什么要用这篇文章中的这种方式来做。
简单分析:每个企业中都会有各种各样的部门,并且很多部门还会包含子部门,子部门还会继续包含子部门这样的,那么我们在导入部门节点的时候就必须要知道自己是属于哪一个节点的,只有知道了自己是属于哪一个节点的才能正确的导入组织结构。那么我们在拿到这份表格后,首先肯定是要自己对表格进行整理的,因为用户给你的表格你也不是拿过来就可以用的,通过你的整理,方便你程序对这个表格的操作。下面这张图片是对表格的加工:
这个表格中真正要导入的数据是:显示顺序、单位名称、单位简称、备注信息、单位编码,其他字段都是辅助字段。部门间的从属关系是我随便写的,
显示顺序,是用来标识在同一节点下的所有部门谁在上面,谁在下面,一般企业可能不会太在意这些,但是一些大的企业,尤其是央企,对这个是有要求的;
单位级别,是用来标识自己是几级单位,3级单位肯定会在某一个2级单位下,以此类推。
上级单位编码,这个一般来说你在收集数据的时候客户会提供给你,每个部门都知道自己的上级部门是谁,自己是直属还是隶属这个他们自己当然很清楚,在这个表里面,我们约定,是一级单位的就是没有上级单位,那么上级单位这一栏就为空,图中最下面的资金管理部是三级单位,那么他的上级单位就是投资管理部,再上一级单位就是市场经营部。
好了明白了这些前提,我们将这张表导入LDAP,
对于导入部门,我们使用实体来导入,为什么要使用实体呢,因为我们需要程序找出每个部门的上级部门,需要使用递归来完成,将没一个单位实例化为一个实体会很方便操作。但是使用实体的弊端是不够灵活,如果模板有改动,就需要改动对应的实体和属性添加部分。相对于人员来说,部门应该说还是可以使用实体的方式导入的,只要模板规定好了,变动一般不会太大。
好了建立实体,新添加一个包com.study.domain,实体命名为Ou.java
1 package com.study.domain; 2 3 public class Ou { 4 private String order;//显示顺序 5 private String orgName;//单位名称 6 private String simpOrgName;//单位简称 7 private String supOrgName;//上级单位 8 private String remark;//备注 9 private int orgGrade;//单位级别 10 private String orgNumber;//单位编码 11 private String supOrgNumber;//上级单位编码 12 13 14 //...get和set方法省略了,请自行添加 15 }
然后在修改接口OuDAO.java,添加导入部门时的方法定义:
package com.study.dao; import java.util.Map; import com.study.domain.Ou; public interface OuDAO { /** * 根据DN值插入单位 * @param map * @param supDn */ public void insertOu(Map<String,String> map,String supDn); /** * 通过单位实体和节点DN添加单位 * @param o * @param supDn */ public void insertOu(Ou o, String supDn); }
接下来在实现类中添加对该方法的实现,为了方便起见,我贴出的是全部的代码,
1 package com.study.dao.impl; 2 3 import java.util.Date; 4 import java.util.Map; 5 import javax.naming.directory.Attributes; 6 import javax.naming.directory.BasicAttribute; 7 import javax.naming.directory.BasicAttributes; 8 9 import org.apache.log4j.Logger; 10 import org.springframework.ldap.NameAlreadyBoundException; 11 import org.springframework.ldap.core.DistinguishedName; 12 import org.springframework.ldap.core.LdapTemplate; 13 14 import com.study.dao.OuDAO; 15 import com.study.domain.Ou; 16 17 public class OuDAOImpl implements OuDAO { 18 LdapTemplate ldapTemplate; 19 public void setLdapTemplate(LdapTemplate ldapTemplate){ 20 this.ldapTemplate=ldapTemplate; 21 } 22 Logger logger=Logger.getLogger(OuDAOImpl.class); 23 @Override 24 public void insertOu(Map<String, String> map, String supDn) { 25 Attributes ouAttributes=new BasicAttributes(); 26 BasicAttribute ouBasicAttribute=new BasicAttribute("objectclass"); 27 ouBasicAttribute.add("organizationalUnit"); 28 ouAttributes.put(ouBasicAttribute); 29 for(String str:map.keySet()){ 30 ouAttributes.put(str,map.get(str)); 31 } 32 DistinguishedName newContactDN=new DistinguishedName(supDn); 33 newContactDN.add("ou",map.get("ou")); 34 ldapTemplate.bind(newContactDN,null,ouAttributes); 35 36 } 37 @Override 38 public void insertOu(Ou o, String supDn) { 39 Attributes ouAttributes=new BasicAttributes(); 40 BasicAttribute ouBasicAttribute=new BasicAttribute("objectclass"); 41 ouBasicAttribute.add("organizationalUnit"); 42 ouAttributes.put(ouBasicAttribute); 43 ouAttributes.put("deptCode",o.getOrgNumber()); 44 ouAttributes.put("description",o.getRemark()); 45 //除了对象本身的属性外,手动为这个部门添加一个创建时间 46 ouAttributes.put("createTime",new Date(System.currentTimeMillis()).toLocaleString()); 47 DistinguishedName newContactDN=new DistinguishedName(supDn); 48 newContactDN.add("ou",o.getOrgName()); 49 try{ 50 ldapTemplate.bind(newContactDN,null,ouAttributes); 51 }catch(NameAlreadyBoundException exception){ 52 logger.warn(o.getOrgNumber()+"--"+exception.getExplanation()); 53 } 54 } 55 56 }
我在代码中象征性的添加进去了几个属性,对于低46行的这个属性cretatTime来说,我们首先要在core.schema中添加定义,添加定义如下:
attributetype ( 2.5.4.71 NAME 'createTime' DESC 'X.520(4th): pseudonym for the object' SUP name )
并且在objectclass中添加这个属性:
objectclass ( 2.5.6.5 NAME 'organizationalUnit' DESC 'RFC2256: an organizational unit' SUP top STRUCTURAL MUST ou MAY ( userPassword $ searchGuide $ seeAlso $ businessCategory $ x121Address $ registeredAddress $ destinationIndicator $ preferredDeliveryMethod $ telexNumber $ teletexTerminalIdentifier $ telephoneNumber $ internationaliSDNNumber $ facsimileTelephoneNumber $ street $ postOfficeBox $ postalCode $ postalAddress $ physicalDeliveryOfficeName $ st $ l $ description $ sslvpn $ incontroller $ deptcode $ createTime ) )
好,接下来是测试类了,在com.study.ou下新建类ImportOuForXls.java,代码如下:
1 package com.study.ou; 2 3 import java.io.FileInputStream; 4 import java.io.InputStream; 5 import java.util.ArrayList; 6 import java.util.Collections; 7 import java.util.Comparator; 8 import java.util.HashMap; 9 import java.util.List; 10 import java.util.Map; 11 12 import org.apache.poi.hssf.usermodel.HSSFWorkbook; 13 import org.apache.poi.ss.usermodel.Cell; 14 import org.apache.poi.ss.usermodel.Row; 15 import org.apache.poi.ss.usermodel.Sheet; 16 import org.springframework.beans.factory.BeanFactory; 17 import org.springframework.beans.factory.xml.XmlBeanFactory; 18 import org.springframework.core.io.ClassPathResource; 19 import org.springframework.core.io.Resource; 20 21 22 import com.study.dao.OuDAO; 23 import com.study.domain.Ou; 24 25 @SuppressWarnings("deprecation") 26 public class ImportOuForXls implements Comparator<Ou>{ 27 static Map<String,Ou> map=new HashMap<String,Ou>(); 28 Resource resource=new ClassPathResource("com/study/ou/ldap-ou.xml"); 29 BeanFactory factory=new XmlBeanFactory(resource); 30 OuDAO ldapOu=(OuDAO) factory.getBean("ldapContext"); 31 public void insertOuMethod(String src) throws Exception{ 32 InputStream ins=new FileInputStream(src); 33 //创建HSSFWork工作薄 34 HSSFWorkbook hssfWorkbook=new HSSFWorkbook(ins); 35 hssfWorkbook.setMissingCellPolicy(Row.CREATE_NULL_AS_BLANK); 36 List<Ou> list=new ArrayList<Ou>(); 37 38 //获得需要解析的sheet 39 Sheet sheet = sheet=hssfWorkbook.getSheetAt(0); 40 for(Row row:sheet){ 41 if(row.getRowNum()==0){ 42 continue; 43 } 44 //从第二行开始解析,row.getLastCellNum()有效列数 45 for(int a=0;a<row.getLastCellNum();a++){ 46 row.getCell(a).setCellType(Cell.CELL_TYPE_STRING); 47 } 48 Ou organ=new Ou(); 49 organ.setOrder(row.getCell(0).getStringCellValue()); 50 organ.setOrgName(row.getCell(1).getStringCellValue()); 51 organ.setSimpOrgName(row.getCell(2).getStringCellValue()); 52 organ.setSupOrgName(row.getCell(3).getStringCellValue()); 53 organ.setRemark(row.getCell(4).getStringCellValue()); 54 organ.setOrgGrade(Integer.parseInt(row.getCell(5).getStringCellValue())); 55 organ.setOrgNumber(row.getCell(6).getStringCellValue()); 56 organ.setSupOrgNumber(row.getCell(7).getStringCellValue().length()>1?row.getCell(7).getStringCellValue():"sup***"); 57 58 list.add(organ); 59 map.put(organ.getOrgNumber(), organ); 60 } 61 ImportOuForXls or=new ImportOuForXls(); 62 Collections.sort(list, or); 63 Map<String,Ou> supa=new HashMap<String,Ou>(); 64 List<Ou> remList=new ArrayList<Ou>(); 65 for(Ou o:list){ 66 //这里先找出所有的一级父节点,在这里可以完成一级节点的添加 67 if(o.getSupOrgNumber().equals("sup***")){ 68 supa.put(o.getOrgNumber(), o); 69 remList.add(o); 70 ldapOu.insertOu(o, ""); 71 System.out.println(o.getOrder()+","+o.getOrgGrade()+","+o.getOrgName()+","+o.getSupOrgName()); 72 } 73 } 74 list.removeAll(remList); 75 System.out.println(list.size()); 76 77 Collections.sort(list,or); 78 for(Ou o:list){ 79 getSupOu(o); 80 System.out.println(o.getOrgGrade()+","+o.getOrgName()+",父节点:"+supDnValue+",应当添加的DN值为:ou="+o.getOrgNumber()+","+supDnValue); 81 ldapOu.insertOu(o, (supDnValue.toString()).substring(0, supDnValue.length()-1)); 82 supDnValue.delete(0, supDnValue.length()); 83 } 84 } 85 static StringBuffer supDnValue=new StringBuffer(); //找出父节点 86 public void getSupOu(Ou o){ 87 88 if(!o.getSupOrgNumber().equals("sup***")){ 89 supDnValue.append("ou="); 90 supDnValue.append(map.get(o.getSupOrgNumber()).getOrgName()); 91 supDnValue.append(","); 92 getSupOu(map.get(o.getSupOrgNumber())); 93 } 94 } 95 public int compare(Ou o1, Ou o2) { 96 Ou a1=o1; 97 Ou a2=o2; 98 int flag=String.valueOf(a1.getOrgGrade()).compareTo(String.valueOf(a2.getOrgGrade())); 99 if(flag==0){ 100 return String.valueOf(a1.getOrgGrade()).compareTo(String.valueOf(a2.getOrgGrade())); 101 }else{ 102 return flag; 103 } 104 } 105 public static void main(String[] args) { 106 try { 107 new ImportOuForXls().insertOuMethod("ou.xls"); 108 } catch (Exception e) { 109 // TODO Auto-generated catch block 110 e.printStackTrace(); 111 } 112 } 113 }
通过POI读取表格中的数据,由于我们最后一列是上级单位,存在一种可能是这个单位本身就没有上级单位的,那么我们在这里使用一个三元表达式,将一级节点用"sup***"来标识,然后后面程序通过这个标识来判定当前节点是不是顶级节点。当前这个类实现了接口Comparator并且重写了方法 compare,这个方法通过表格中的单位级别将所有元素排序然后在65行的地方通过一个for循环与sup***标识结合找出所有的1级节点,并且执行添加,然后再将这个一级节点放入一个list中,到74行的时候原先会从list中移除所有的一级节点,要知道,我们在读取表格的时候是放了两份的,一份是存在map中,一份是存在list中,这个map可以看做暂时的一个数据库,后面的程序都是在操作这个map,包括找父节点这样的操作都是基于这个map的。
紧接着在第77行的时候再次对list进行排序,那么从这个list中读取节点的时候首先读取到的就是所有的2级节点了,然后使用一个增强for循环来遍历这个list,在78行这里,我们调用一个递归函数,这个递归函数会找出当前节点的父节点以及父节点的父节点,并且将它组装成一个符合LDAP规范的DN值,在LDAP中要插入一个节点,必须要知道当前节点的父节点的DN值。
以上面那个表格为例:
我要插入资金管理部,那么父节点的DN值就是:ou=tzglb,ou=scjy
是倒着写的,上面第86行的递归函数会组装出符合这样规范的DN值放入supDnValue,在第81行执行完添加后紧接着将这个supDnValue清空,进入下一次循环
执行添加的部分很简单了,相信都可以看懂了。
这个导入部门写的不够完美的地方呢就是添加的方法,里面的属性都是写死的,要是要更改属性的话就需要修改这里与部门的属性相对应。
那么在下一篇文章中呢,我会对这一部分进行优化,程序不用改动,通过你表格里的内容自动调整属性值,当然这个也是有前提的,守恒定律还是要遵循的,程序不动那就要表格动咯!
======================================
PS:
1、可能写的比较乱,凑合看了。
2、上面这个例子是可以跑通的。
3、如果有报错,仔细看报错的原因,我觉得这个应该说是比较容易找到的,spring ldap的报错还算是易懂了。
4、最后一点细节,表格的属性一定最好都是文本类型: