BOS项目第08天

8 BOS项目第08天

对权限涉及的5张表,进行CRUD的操作

8.1 初始化权限数据

  • 权限功能需要一张权限表记录数据


    image.png
  • 重要字段

    • page:访问路径
    • generatemenu:是否生成菜单
    • zindex:优先级,实现排序
    • pid:父级目录的id
  • 打开auth_function.sql,复制到Navicat

8.2 功能权限管理列表

8.2.1 数据列表展示

  • 写Dao,Service,Action,Jsp代码
    • model中的是generatemenu,需要修改jsp中对应的字段generateMenu
    • url改为:${pageContext.request.contextPath}/functionAction_pageQuery.action
    • 加上分页代码:
pageList:[5,10,15],
pageSize:10,
pagination:true,
  • exclude三个属性
    • 排除Function模型中两个会造成循环引用的属性:"function", "functions"
    • 权限与角色是多对多关系,权限表不用直接显示角色属性,所以要exclude属性roles
  • 解决分页的Bug
    • 因为Function和BaseAction都有参数page,page参数会优先传递给Model,分页无法获取参数,一直默认为第一页.解决方案就是手动给PageBean赋值


      image.png

8.2.2 功能权限信息添加

  • 把jsp中的id换成关键字,id不需要自己输入
  • 是否生成关菜单,用select表示,1用生成表示,0用不生成表示,注意,这里的generateMenu也要改成小写
{
                  field : 'generatemenu',
                  title : '是否生成菜单',
                  formatter : function(data, row, index){
                      if(data=="1"){
                          return "生成";
                      }else{
                          return "不生成";
                      }
                  },
                  width : 200
              },
  • 父功能点:parentFunction.id改成function.id,要与模型对应

  • 实现save业务链

    • Jsp:在表单上提供action属性:${pageContext.request.contextPath}/functionAction_save.action
    • save按钮的点击事件:$("#functionForm").submit();
  • 显示父功能点,后台返回Json,textField展示name

    • JSP部分
      <input name="function.id" class="easyui-combobox" data-options="valueField:'id',textField:'name',url:'${pageContext.request.contextPath}/functionAction_listJson.action'"/>
    • Action部分
public void listJson(){
        List<Function> functions = functionService.findAll();
        responseJson(functions, new String[]{"function", "functions"});
    }
  • 是否生成

8.3 角色管理

8.3.1 添加角色功能

image.png

8.3.2 修改授权数据的来源

  • 分析授权数据的获取源码
    • 本质就是将基础功能的树,再次显示
    • 权限表获取数据:${pageContext.request.contextPath}/functionAction_listJson.action
    • 后台返回的json,要提供pId属性
    • Fuction代码
     private String pId;

    public String getpId() {
        if(function != null){
            return function.getId();
        }
        return "0";
    }
  • 获取授权数据,勾选的所有id
var treeObj = $.fn.zTree.getZTreeObj("functionTree");
var nodes = treeObj.getCheckedNodes(true);
var ids = new Array();
for(var i = 0; i < nodes.length; i++){
    var id = nodes[i].id;
    ids.push(id);
}
//拼接所有id
var idsStr = ids.join(",");

8.3.4 数据提交与后台处理

  1. 表单提交
    • JSP将显示id换成关键字
    • 添加一个隐藏的input,名字为ids,用于提交参数
    • URL:"${pageContext.request.contextPath}/roleAction_save.action"
  2. 后台处理
    • DAO,Service
    • 添加一个RoleAction来处理表单提交
    • 中间表由Hibernate自动维护,只需写好映射即可
@Override
    public void save(Role role, String functionIds) {
        //1.保存角色
        roleDao.save(role);

        //2.添加角色,权限,中间表ids
        //2.1拆分id
        String[] functionIdsArr = functionIds.split(",");
        for (String functionId : functionIdsArr) {
            //把id封装成Function模型
            Function function = new Function();
            function.setId(functionId);

            //把function存在Role里面去
            role.getFunctions().add(function);//内部执行insert语句
        }
    }

8.3.5 角色管理列表实现

  • pageQuery方法,简单实现
    url : '${pageContext.request.contextPath}/roleAction_pageQuery.action?page=1&rows=20',

8.4 用户管理模块

8.4.1 显示用户列表数据

  • UserAction中实现分页查询
  • 几个注意点:
    1. 之前的测试账户admin,要删掉,因为工资,生日等数据都为空,会造成转JSON异常
    2. Data是util包下的,而不是sql包下的
    3. User模型提供一个getBirthdayStr方法,对应birthdayStr属性,便于转换日期格式
    4. remark字段不显示,所以也应该exclude,roles也要exclude

8.4.2 添加用户

  • 效果图


    image.png
  • 首先需要获得角色数据,Action代码

public void listJson() throws IOException {
        List<Role> roles = roleService.findAll();
        responseJson(roles, new String[]{"users", "functions"});
    }
  • JS动态添加checkbox
//获取角色数据
        var url = "${pageContext.request.contextPath}/roleAction_listJson.action";
        $.post(url, function (data) {
            for (var i = 0;i<data.length;i++) {
                var name = data[i].name;
                var id = data[i].id;
                var inputTag = '<input type="checkbox" value="'+id+'" name="roleIds">' + name;
                $("#roleTd").append(inputTag);
            }
        });
  • 查看表单请求参数
image.png
  • userAction的添加方法实现
  • userService的实现
@Override
    public void save(User model, String[] roleIds) {
        //密码使用MD5
        String pwd = MD5Utils.text2md5(model.getPassword());
        model.setPassword(pwd);

        userDao.save(model);
        for (String roleId : roleIds) {
            Role role = new Role();
            role.setId(roleId);
            model.getRoles().add(role);
        }
    }
  • 用户列表中生日的显示,util包中Date格式的日期返回的是
默认返回的生日数据格式如图
{
            "birthday": {
                "date": 1,
                "day": 4,
                "hours": 0,
                "minutes": 0,
                "month": 5,
                "nanos": 0,
                "seconds": 0,
                "time": 1496246400000,
                "timezoneOffset": -480,
                "year": 117
}
  • 给User模型添加一个get方法,修改界面字段为birthdayStr,这再次说明了属性是get方法截取后的产物
public String getBirthdayStr(){
        if (birthday != null){
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
            return sdf.format(birthday);
        }else{
            return "未提交生日";
        }
    }
  • 把界面的电话字段改成显示角色
    1 在User模型中添加一个getRolesStr方法
public String getRolesStr(){
        String str = "";
        for (Role role : roles) {
            str += role.getName() + "、";
        }
        return str;
    }
  • 效果


    image.png

8.5 修改BOSRealm中的授权方法

8.5.1 授权的逻辑

  • 如果是admin,则拥有所有权限(Function所有的功能都能用)
  • 如果是非管理员,只能通过sql语句,用用户id查找权限
    1. 一个用户,可能有多个角色,一个角色又有多个权限,那么通过id查询权限,肯定有重复的权限数据,所以要DISTINCT,去除重复数据
    2. 推荐第二种写法,易于理解,三张主表都是多对多关系,使用左外连接
#根据用户ID,查找他所拥有的的权限function
SELECT f.id, f.name, f.page, f.code
FROM auth_function f
LEFT OUTER JOIN role_function rf
ON rf.function_id = f.id
LEFT OUTER JOIN auth_role r
ON rf.role_id = r.id
LEFT OUTER JOIN user_role ur
ON r.id = ur.role_id
WHERE ur.user_id = '40289f196c9f9da7016c9fa12f3a0000';

#第二种方式
SELECT DISTINCT f.id,f.name,f.page,f.code 
FROM 
auth_function f,
role_function rf,
auth_role r,
user_role ur
WHERE 
rf.function_id = f.id and 
rf.role_id = r.id and 
r.id = ur.role_id and
ur.user_id = '40289f196c9f9da7016c9fa12f3a0000';
  • 用户查找权限hql语句
@Override
    public List<Function> findListByUserId(String userId) {
        String hql = "SELECT DISTINCT f FROM Function f ";
        hql += "LEFT OUTER JOIN f.roles r ";
        hql += "LEFT OUTER JOIN r.users u ";
        hql += "WHERE u.id = ?";

        return hibernateTemplate.find(hql, userId);
    }
  • 给登录用户,授予权限,Realm代码
@Override
    protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principals) {
        //拿到用户
        User loginUser = (User) principals.getPrimaryPrincipal();

        //根据用户ID查权限
        List<Function> functions = null;
        //admin超级管理员
        if (loginUser.getUsername().equals("admin")){
            functions = functionDao.findAll();
        }else{
            functions = functionDao.findListByUserId(loginUser.getId());
        }

        //往shiro添加权限
        SimpleAuthorizationInfo info = new SimpleAuthorizationInfo();
        for (Function function : functions) {
            info.addStringPermission(function.getCode());
        }

        return info;
    }
  • 授权测试:JSP添加一个shiro标签


    image.png
  • 在数据库中添加一个admin


    image.png
  • 这样用hr登录,就没有作废功能,作废按钮不显示

  • 对数据库的一点补充

    1. 左外连接,就是左连接,此二者等价
    2. 还有全外连接,一般不用
    3. 多对多关系,本质就是通过中间表,做所有的连接,然后查询想要的数据,并去除重复


      1566109196(1).jpg

8.6 使用ehcache缓存权限数据

  1. 导入ehcache的jar包,3个


    image.png
  2. 使用ehcache缓存权限数据

<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="../config/ehcache.xsd">

    <diskStore path="java.io.tmpdir"/>
    <defaultCache
            maxElementsInMemory="10000"
            eternal="false"
            timeToIdleSeconds="120"
            timeToLiveSeconds="120"
            overflowToDisk="true"
            maxElementsOnDisk="10000000"
            diskPersistent="false"
            diskExpiryThreadIntervalSeconds="120"
            memoryStoreEvictionPolicy="LRU"
    />
</ehcache>
  1. 在spring配置文件中添加授权缓存策略
<bean id="cacheManager" class="org.apache.shiro.cache.ehcache.EhCacheManager">
        <property name="cacheManagerConfigFile" value="classpath:ehcache.xml"></property>
    </bean>

    <!--2.配置shiro的安全管理者-->
    <bean id="securityManager" class="org.apache.shiro.web.mgt.DefaultWebSecurityManager">
        <!--配置realm对象-->
        <property name="realm" ref="realm"></property>
        <!--配置缓存管理者-->
        <property name="cacheManager" ref="cacheManager"></property>
    </bean>

    <!--3.配置一个realm对象,这个对象要继承AuthorizingRealm-->
    <bean id="realm" class="com.kdj.bos.web.realm.BOSRealm"></bean>

8.7 修改主界面菜单从数据库获取

  • 主界面菜单根据登录用户的角色不用来显示不同的菜单,管理员拥有全部功能
  • dao代码
@Override
    public List<Function> findMenuByUserId(String id) {
        String hql = "SELECT DISTINCT f From Function f ";
        hql += "LEFT OUTER JOIN f.roles r ";
        hql += "LEFT OUTER JOIN r.users u ";
        hql += "WHERE u.id = ? AND f.generatemenu = '1' ORDER BY f.zindex DESC";
        return (List<Function>) this.hibernateTemplate.find(hql,id);
    }

    @Override
    public List<Function> findAllMenu() {
        String hql = "From Function f where f.generatemenu = '1' ORDER BY f.zindex DESC";
        return (List<Function>) this.hibernateTemplate.find(hql);
    }
  • 前后端交互的本质:前端发送一个请求,后端返回一个json,就这么简单
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容

  • JSP总结(经典) day1 JSP 定义: 1)Java Server Page, Java EE 组件,本...
    java日记阅读 3,402评论 0 13
  • 一. Java基础部分.................................................
    wy_sure阅读 3,785评论 0 11
  • 静态网页 动态网页CGI asp(html+JavaScript+com组件) PHP jsp(html+jav...
    高永骁阅读 422评论 0 1
  • 1.学习内容 JSP技术入门和常用指令 JSP的内置对象&标签介绍 EL表达式&EL的内置对象 2.JSP技术入门...
    WendyVIV阅读 2,116评论 1 18
  • 一、jsp介绍 jsp作用jsp全称java server pages(java服务器页面),本质是一个servl...
    圣贤与无赖阅读 1,238评论 0 14