Spring Boot Shiro在线会话管理
This commit is contained in:
parent
e50f57b0fe
commit
5c6e3c3f16
|
|
@ -0,0 +1,103 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
|
||||||
|
<groupId>com.springboot</groupId>
|
||||||
|
<artifactId>demo</artifactId>
|
||||||
|
<version>0.0.1-SNAPSHOT</version>
|
||||||
|
<packaging>jar</packaging>
|
||||||
|
|
||||||
|
<name>demo</name>
|
||||||
|
<description>Demo project for Spring Boot</description>
|
||||||
|
|
||||||
|
<parent>
|
||||||
|
<groupId>org.springframework.boot</groupId>
|
||||||
|
<artifactId>spring-boot-starter-parent</artifactId>
|
||||||
|
<version>1.5.9.RELEASE</version>
|
||||||
|
<relativePath/> <!-- lookup parent from repository -->
|
||||||
|
</parent>
|
||||||
|
|
||||||
|
<properties>
|
||||||
|
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||||
|
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
|
||||||
|
<java.version>1.7</java.version>
|
||||||
|
<thymeleaf.version>3.0.2.RELEASE</thymeleaf.version>
|
||||||
|
<thymeleaf-layout-dialect.version>2.0.1</thymeleaf-layout-dialect.version>
|
||||||
|
</properties>
|
||||||
|
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.springframework.boot</groupId>
|
||||||
|
<artifactId>spring-boot-starter-web</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.mybatis.spring.boot</groupId>
|
||||||
|
<artifactId>mybatis-spring-boot-starter</artifactId>
|
||||||
|
<version>1.3.1</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.springframework.boot</groupId>
|
||||||
|
<artifactId>spring-boot-starter-thymeleaf</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<!-- shiro-spring -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.shiro</groupId>
|
||||||
|
<artifactId>shiro-spring</artifactId>
|
||||||
|
<version>1.4.0</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<!-- shiro ehcache -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.apache.shiro</groupId>
|
||||||
|
<artifactId>shiro-ehcache</artifactId>
|
||||||
|
<version>1.3.2</version>
|
||||||
|
</dependency>
|
||||||
|
<!-- ehchache -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.springframework.boot</groupId>
|
||||||
|
<artifactId>spring-boot-starter-cache</artifactId>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>net.sf.ehcache</groupId>
|
||||||
|
<artifactId>ehcache</artifactId>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<!-- https://mvnrepository.com/artifact/com.github.theborakompanioni/thymeleaf-extras-shiro -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.github.theborakompanioni</groupId>
|
||||||
|
<artifactId>thymeleaf-extras-shiro</artifactId>
|
||||||
|
<version>2.0.0</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
|
||||||
|
<!-- oracle驱动 -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.oracle</groupId>
|
||||||
|
<artifactId>ojdbc6</artifactId>
|
||||||
|
<version>6.0</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<!-- druid数据源驱动 -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.alibaba</groupId>
|
||||||
|
<artifactId>druid-spring-boot-starter</artifactId>
|
||||||
|
<version>1.1.6</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
</dependencies>
|
||||||
|
|
||||||
|
<build>
|
||||||
|
<plugins>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.springframework.boot</groupId>
|
||||||
|
<artifactId>spring-boot-maven-plugin</artifactId>
|
||||||
|
</plugin>
|
||||||
|
</plugins>
|
||||||
|
</build>
|
||||||
|
|
||||||
|
|
||||||
|
</project>
|
||||||
|
|
@ -0,0 +1,13 @@
|
||||||
|
package com.springboot;
|
||||||
|
|
||||||
|
import org.springframework.boot.SpringApplication;
|
||||||
|
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||||
|
|
||||||
|
|
||||||
|
@SpringBootApplication
|
||||||
|
public class Application {
|
||||||
|
|
||||||
|
public static void main(String[] args) {
|
||||||
|
SpringApplication.run(Application.class,args);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,133 @@
|
||||||
|
package com.springboot.config;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
|
||||||
|
import org.apache.shiro.cache.ehcache.EhCacheManager;
|
||||||
|
import org.apache.shiro.codec.Base64;
|
||||||
|
import org.apache.shiro.mgt.SecurityManager;
|
||||||
|
import org.apache.shiro.session.SessionListener;
|
||||||
|
import org.apache.shiro.session.mgt.SessionManager;
|
||||||
|
import org.apache.shiro.session.mgt.eis.MemorySessionDAO;
|
||||||
|
import org.apache.shiro.session.mgt.eis.SessionDAO;
|
||||||
|
import org.apache.shiro.spring.LifecycleBeanPostProcessor;
|
||||||
|
import org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor;
|
||||||
|
import org.apache.shiro.spring.web.ShiroFilterFactoryBean;
|
||||||
|
import org.apache.shiro.web.mgt.CookieRememberMeManager;
|
||||||
|
import org.apache.shiro.web.mgt.DefaultWebSecurityManager;
|
||||||
|
import org.apache.shiro.web.servlet.SimpleCookie;
|
||||||
|
import org.apache.shiro.web.session.mgt.DefaultWebSessionManager;
|
||||||
|
import org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.context.annotation.DependsOn;
|
||||||
|
|
||||||
|
import com.springboot.listener.ShiroSessionListener;
|
||||||
|
import com.springboot.shiro.ShiroRealm;
|
||||||
|
|
||||||
|
import at.pollux.thymeleaf.shiro.dialect.ShiroDialect;
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
public class ShiroConfig {
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public EhCacheManager getEhCacheManager() {
|
||||||
|
EhCacheManager em = new EhCacheManager();
|
||||||
|
em.setCacheManagerConfigFile("classpath:config/shiro-ehcache.xml");
|
||||||
|
return em;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public ShiroFilterFactoryBean shiroFilterFactoryBean(SecurityManager securityManager) {
|
||||||
|
ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean();
|
||||||
|
shiroFilterFactoryBean.setSecurityManager(securityManager);
|
||||||
|
shiroFilterFactoryBean.setLoginUrl("/login");
|
||||||
|
shiroFilterFactoryBean.setSuccessUrl("/index");
|
||||||
|
shiroFilterFactoryBean.setUnauthorizedUrl("/403");
|
||||||
|
LinkedHashMap<String, String> filterChainDefinitionMap = new LinkedHashMap<>();
|
||||||
|
|
||||||
|
filterChainDefinitionMap.put("/css/**", "anon");
|
||||||
|
filterChainDefinitionMap.put("/js/**", "anon");
|
||||||
|
filterChainDefinitionMap.put("/fonts/**", "anon");
|
||||||
|
filterChainDefinitionMap.put("/img/**", "anon");
|
||||||
|
filterChainDefinitionMap.put("/druid/**", "anon");
|
||||||
|
filterChainDefinitionMap.put("/logout", "logout");
|
||||||
|
filterChainDefinitionMap.put("/", "anon");
|
||||||
|
filterChainDefinitionMap.put("/**", "user");
|
||||||
|
|
||||||
|
shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);
|
||||||
|
|
||||||
|
return shiroFilterFactoryBean;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public SecurityManager securityManager(){
|
||||||
|
DefaultWebSecurityManager securityManager = new DefaultWebSecurityManager();
|
||||||
|
securityManager.setRealm(shiroRealm());
|
||||||
|
securityManager.setRememberMeManager(rememberMeManager());
|
||||||
|
securityManager.setCacheManager(getEhCacheManager());
|
||||||
|
securityManager.setSessionManager(sessionManager());
|
||||||
|
return securityManager;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean(name = "lifecycleBeanPostProcessor")
|
||||||
|
public LifecycleBeanPostProcessor lifecycleBeanPostProcessor() {
|
||||||
|
return new LifecycleBeanPostProcessor();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public ShiroRealm shiroRealm(){
|
||||||
|
ShiroRealm shiroRealm = new ShiroRealm();
|
||||||
|
return shiroRealm;
|
||||||
|
}
|
||||||
|
|
||||||
|
public SimpleCookie rememberMeCookie() {
|
||||||
|
SimpleCookie cookie = new SimpleCookie("rememberMe");
|
||||||
|
cookie.setMaxAge(86400);
|
||||||
|
return cookie;
|
||||||
|
}
|
||||||
|
|
||||||
|
public CookieRememberMeManager rememberMeManager() {
|
||||||
|
CookieRememberMeManager cookieRememberMeManager = new CookieRememberMeManager();
|
||||||
|
cookieRememberMeManager.setCookie(rememberMeCookie());
|
||||||
|
cookieRememberMeManager.setCipherKey(Base64.decode("4AvVhmFLUs0KTA3Kprsdag=="));
|
||||||
|
return cookieRememberMeManager;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@DependsOn({"lifecycleBeanPostProcessor"})
|
||||||
|
public DefaultAdvisorAutoProxyCreator advisorAutoProxyCreator() {
|
||||||
|
DefaultAdvisorAutoProxyCreator advisorAutoProxyCreator = new DefaultAdvisorAutoProxyCreator();
|
||||||
|
advisorAutoProxyCreator.setProxyTargetClass(true);
|
||||||
|
return advisorAutoProxyCreator;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public AuthorizationAttributeSourceAdvisor authorizationAttributeSourceAdvisor(SecurityManager securityManager) {
|
||||||
|
AuthorizationAttributeSourceAdvisor authorizationAttributeSourceAdvisor = new AuthorizationAttributeSourceAdvisor();
|
||||||
|
authorizationAttributeSourceAdvisor.setSecurityManager(securityManager);
|
||||||
|
return authorizationAttributeSourceAdvisor;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public ShiroDialect shiroDialect() {
|
||||||
|
return new ShiroDialect();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public SessionDAO sessionDAO() {
|
||||||
|
MemorySessionDAO sessionDAO = new MemorySessionDAO();
|
||||||
|
return sessionDAO;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public SessionManager sessionManager() {
|
||||||
|
DefaultWebSessionManager sessionManager = new DefaultWebSessionManager();
|
||||||
|
Collection<SessionListener> listeners = new ArrayList<SessionListener>();
|
||||||
|
listeners.add(new ShiroSessionListener());
|
||||||
|
sessionManager.setSessionListeners(listeners);
|
||||||
|
sessionManager.setSessionDAO(sessionDAO());
|
||||||
|
return sessionManager;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,65 @@
|
||||||
|
package com.springboot.controller;
|
||||||
|
|
||||||
|
import org.apache.shiro.SecurityUtils;
|
||||||
|
import org.apache.shiro.authc.AuthenticationException;
|
||||||
|
import org.apache.shiro.authc.IncorrectCredentialsException;
|
||||||
|
import org.apache.shiro.authc.LockedAccountException;
|
||||||
|
import org.apache.shiro.authc.UnknownAccountException;
|
||||||
|
import org.apache.shiro.authc.UsernamePasswordToken;
|
||||||
|
import org.apache.shiro.subject.Subject;
|
||||||
|
import org.springframework.stereotype.Controller;
|
||||||
|
import org.springframework.ui.Model;
|
||||||
|
import org.springframework.web.bind.annotation.GetMapping;
|
||||||
|
import org.springframework.web.bind.annotation.PostMapping;
|
||||||
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
|
import org.springframework.web.bind.annotation.ResponseBody;
|
||||||
|
|
||||||
|
import com.springboot.pojo.ResponseBo;
|
||||||
|
import com.springboot.pojo.User;
|
||||||
|
import com.springboot.util.MD5Utils;
|
||||||
|
|
||||||
|
@Controller
|
||||||
|
public class LoginController {
|
||||||
|
|
||||||
|
@GetMapping("/login")
|
||||||
|
public String login() {
|
||||||
|
return "login";
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping("/login")
|
||||||
|
@ResponseBody
|
||||||
|
public ResponseBo login(String username, String password, Boolean rememberMe) {
|
||||||
|
password = MD5Utils.encrypt(username, password);
|
||||||
|
UsernamePasswordToken token = new UsernamePasswordToken(username, password, rememberMe);
|
||||||
|
Subject subject = SecurityUtils.getSubject();
|
||||||
|
try {
|
||||||
|
subject.login(token);
|
||||||
|
return ResponseBo.ok();
|
||||||
|
} catch (UnknownAccountException e) {
|
||||||
|
return ResponseBo.error(e.getMessage());
|
||||||
|
} catch (IncorrectCredentialsException e) {
|
||||||
|
return ResponseBo.error(e.getMessage());
|
||||||
|
} catch (LockedAccountException e) {
|
||||||
|
return ResponseBo.error(e.getMessage());
|
||||||
|
} catch (AuthenticationException e) {
|
||||||
|
return ResponseBo.error("认证失败!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@RequestMapping("/")
|
||||||
|
public String redirectIndex() {
|
||||||
|
return "redirect:/index";
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping("/403")
|
||||||
|
public String forbid() {
|
||||||
|
return "403";
|
||||||
|
}
|
||||||
|
|
||||||
|
@RequestMapping("/index")
|
||||||
|
public String index(Model model) {
|
||||||
|
User user = (User) SecurityUtils.getSubject().getPrincipal();
|
||||||
|
model.addAttribute("user", user);
|
||||||
|
return "index";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,45 @@
|
||||||
|
package com.springboot.controller;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.stereotype.Controller;
|
||||||
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
|
import org.springframework.web.bind.annotation.ResponseBody;
|
||||||
|
|
||||||
|
import com.springboot.pojo.ResponseBo;
|
||||||
|
import com.springboot.pojo.UserOnline;
|
||||||
|
import com.springboot.service.SessionService;
|
||||||
|
|
||||||
|
|
||||||
|
@Controller
|
||||||
|
@RequestMapping("/online")
|
||||||
|
public class SessionController {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
SessionService sessionService;
|
||||||
|
|
||||||
|
@RequestMapping("index")
|
||||||
|
public String online() {
|
||||||
|
return "online";
|
||||||
|
}
|
||||||
|
|
||||||
|
@ResponseBody
|
||||||
|
@RequestMapping("list")
|
||||||
|
public List<UserOnline> list() {
|
||||||
|
return sessionService.list();
|
||||||
|
}
|
||||||
|
|
||||||
|
@ResponseBody
|
||||||
|
@RequestMapping("forceLogout")
|
||||||
|
public ResponseBo forceLogout(String id) {
|
||||||
|
try {
|
||||||
|
sessionService.forceLogout(id);
|
||||||
|
return ResponseBo.ok();
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
return ResponseBo.error("踢出用户失败");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,33 @@
|
||||||
|
package com.springboot.controller;
|
||||||
|
|
||||||
|
import org.apache.shiro.authz.annotation.RequiresPermissions;
|
||||||
|
import org.springframework.stereotype.Controller;
|
||||||
|
import org.springframework.ui.Model;
|
||||||
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
|
|
||||||
|
@Controller
|
||||||
|
@RequestMapping("/user")
|
||||||
|
public class UserController {
|
||||||
|
|
||||||
|
|
||||||
|
@RequiresPermissions("user:user")
|
||||||
|
@RequestMapping("list")
|
||||||
|
public String userList(Model model) {
|
||||||
|
model.addAttribute("value", "获取用户信息");
|
||||||
|
return "user";
|
||||||
|
}
|
||||||
|
|
||||||
|
@RequiresPermissions("user:add")
|
||||||
|
@RequestMapping("add")
|
||||||
|
public String userAdd(Model model) {
|
||||||
|
model.addAttribute("value", "新增用户");
|
||||||
|
return "user";
|
||||||
|
}
|
||||||
|
|
||||||
|
@RequiresPermissions("user:delete")
|
||||||
|
@RequestMapping("delete")
|
||||||
|
public String userDelete(Model model) {
|
||||||
|
model.addAttribute("value", "删除用户");
|
||||||
|
return "user";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,10 @@
|
||||||
|
package com.springboot.dao;
|
||||||
|
|
||||||
|
import org.apache.ibatis.annotations.Mapper;
|
||||||
|
|
||||||
|
import com.springboot.pojo.User;
|
||||||
|
|
||||||
|
@Mapper
|
||||||
|
public interface UserMapper {
|
||||||
|
User findByUserName(String userName);
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,11 @@
|
||||||
|
package com.springboot.dao;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import org.apache.ibatis.annotations.Mapper;
|
||||||
|
import com.springboot.pojo.Permission;
|
||||||
|
|
||||||
|
@Mapper
|
||||||
|
public interface UserPermissionMapper {
|
||||||
|
|
||||||
|
List<Permission> findByUserName(String userName);
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,12 @@
|
||||||
|
package com.springboot.dao;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.apache.ibatis.annotations.Mapper;
|
||||||
|
import com.springboot.pojo.Role;
|
||||||
|
|
||||||
|
@Mapper
|
||||||
|
public interface UserRoleMapper {
|
||||||
|
|
||||||
|
List<Role> findByUserName(String userName);
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,24 @@
|
||||||
|
package com.springboot.handler;
|
||||||
|
|
||||||
|
import org.apache.shiro.authz.AuthorizationException;
|
||||||
|
import org.apache.shiro.session.ExpiredSessionException;
|
||||||
|
import org.springframework.core.Ordered;
|
||||||
|
import org.springframework.core.annotation.Order;
|
||||||
|
import org.springframework.web.bind.annotation.ControllerAdvice;
|
||||||
|
import org.springframework.web.bind.annotation.ExceptionHandler;
|
||||||
|
|
||||||
|
|
||||||
|
@ControllerAdvice
|
||||||
|
@Order(value = Ordered.HIGHEST_PRECEDENCE)
|
||||||
|
public class GlobalExceptionHandler {
|
||||||
|
|
||||||
|
@ExceptionHandler(value = AuthorizationException.class)
|
||||||
|
public String handleAuthorizationException() {
|
||||||
|
return "403";
|
||||||
|
}
|
||||||
|
|
||||||
|
@ExceptionHandler(value = ExpiredSessionException.class )
|
||||||
|
public String handleExpiredSessionException() {
|
||||||
|
return "login";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,27 @@
|
||||||
|
package com.springboot.listener;
|
||||||
|
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
|
import org.apache.shiro.session.Session;
|
||||||
|
import org.apache.shiro.session.SessionListener;
|
||||||
|
|
||||||
|
public class ShiroSessionListener implements SessionListener{
|
||||||
|
|
||||||
|
private final AtomicInteger sessionCount = new AtomicInteger(0);
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onStart(Session session) {
|
||||||
|
sessionCount.incrementAndGet();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onStop(Session session) {
|
||||||
|
sessionCount.decrementAndGet();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onExpiration(Session session) {
|
||||||
|
sessionCount.decrementAndGet();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,31 @@
|
||||||
|
package com.springboot.pojo;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
|
||||||
|
public class Permission implements Serializable{
|
||||||
|
|
||||||
|
private static final long serialVersionUID = 7160557680614732403L;
|
||||||
|
private Integer id;
|
||||||
|
private String url;
|
||||||
|
private String name;
|
||||||
|
public Integer getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
public void setId(Integer id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
public String getUrl() {
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
public void setUrl(String url) {
|
||||||
|
this.url = url;
|
||||||
|
}
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
public void setName(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,50 @@
|
||||||
|
package com.springboot.pojo;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class ResponseBo extends HashMap<String, Object>{
|
||||||
|
private static final long serialVersionUID = 1L;
|
||||||
|
|
||||||
|
public ResponseBo() {
|
||||||
|
put("code", 0);
|
||||||
|
put("msg", "操作成功");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ResponseBo error() {
|
||||||
|
return error(1, "操作失败");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ResponseBo error(String msg) {
|
||||||
|
return error(500, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ResponseBo error(int code, String msg) {
|
||||||
|
ResponseBo ResponseBo = new ResponseBo();
|
||||||
|
ResponseBo.put("code", code);
|
||||||
|
ResponseBo.put("msg", msg);
|
||||||
|
return ResponseBo;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ResponseBo ok(String msg) {
|
||||||
|
ResponseBo ResponseBo = new ResponseBo();
|
||||||
|
ResponseBo.put("msg", msg);
|
||||||
|
return ResponseBo;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ResponseBo ok(Map<String, Object> map) {
|
||||||
|
ResponseBo ResponseBo = new ResponseBo();
|
||||||
|
ResponseBo.putAll(map);
|
||||||
|
return ResponseBo;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ResponseBo ok() {
|
||||||
|
return new ResponseBo();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ResponseBo put(String key, Object value) {
|
||||||
|
super.put(key, value);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,32 @@
|
||||||
|
package com.springboot.pojo;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
|
||||||
|
public class Role implements Serializable{
|
||||||
|
|
||||||
|
private static final long serialVersionUID = -227437593919820521L;
|
||||||
|
private Integer id;
|
||||||
|
private String name;
|
||||||
|
private String memo;
|
||||||
|
public Integer getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
public void setId(Integer id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
public void setName(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
public String getMemo() {
|
||||||
|
return memo;
|
||||||
|
}
|
||||||
|
public void setMemo(String memo) {
|
||||||
|
this.memo = memo;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,46 @@
|
||||||
|
package com.springboot.pojo;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.Date;
|
||||||
|
|
||||||
|
public class User implements Serializable{
|
||||||
|
|
||||||
|
private static final long serialVersionUID = -5440372534300871944L;
|
||||||
|
|
||||||
|
private Integer id;
|
||||||
|
private String userName;
|
||||||
|
private String password;
|
||||||
|
private Date createTime;
|
||||||
|
private String status;
|
||||||
|
public Integer getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
public void setId(Integer id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
public String getUserName() {
|
||||||
|
return userName;
|
||||||
|
}
|
||||||
|
public void setUserName(String userName) {
|
||||||
|
this.userName = userName;
|
||||||
|
}
|
||||||
|
public String getPassword() {
|
||||||
|
return password;
|
||||||
|
}
|
||||||
|
public void setPassword(String password) {
|
||||||
|
this.password = password;
|
||||||
|
}
|
||||||
|
public Date getCreateTime() {
|
||||||
|
return createTime;
|
||||||
|
}
|
||||||
|
public void setCreateTime(Date createTime) {
|
||||||
|
this.createTime = createTime;
|
||||||
|
}
|
||||||
|
public String getStatus() {
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
public void setStatus(String status) {
|
||||||
|
this.status = status;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,103 @@
|
||||||
|
package com.springboot.pojo;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.Date;
|
||||||
|
|
||||||
|
public class UserOnline implements Serializable{
|
||||||
|
|
||||||
|
private static final long serialVersionUID = 3828664348416633856L;
|
||||||
|
|
||||||
|
// session id
|
||||||
|
private String id;
|
||||||
|
// 用户id
|
||||||
|
private String userId;
|
||||||
|
// 用户名称
|
||||||
|
private String username;
|
||||||
|
// 用户主机地址
|
||||||
|
private String host;
|
||||||
|
// 用户登录时系统IP
|
||||||
|
private String systemHost;
|
||||||
|
// 状态
|
||||||
|
private String status;
|
||||||
|
// session创建时间
|
||||||
|
private Date startTimestamp;
|
||||||
|
// session最后访问时间
|
||||||
|
private Date lastAccessTime;
|
||||||
|
// 超时时间
|
||||||
|
private Long timeout;
|
||||||
|
|
||||||
|
|
||||||
|
public String getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(String id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getUserId() {
|
||||||
|
return userId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setUserId(String userId) {
|
||||||
|
this.userId = userId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getUsername() {
|
||||||
|
return username;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setUsername(String username) {
|
||||||
|
this.username = username;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getHost() {
|
||||||
|
return host;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setHost(String host) {
|
||||||
|
this.host = host;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSystemHost() {
|
||||||
|
return systemHost;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSystemHost(String systemHost) {
|
||||||
|
this.systemHost = systemHost;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getStatus() {
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setStatus(String status) {
|
||||||
|
this.status = status;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Date getStartTimestamp() {
|
||||||
|
return startTimestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setStartTimestamp(Date startTimestamp) {
|
||||||
|
this.startTimestamp = startTimestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Date getLastAccessTime() {
|
||||||
|
return lastAccessTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setLastAccessTime(Date lastAccessTime) {
|
||||||
|
this.lastAccessTime = lastAccessTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getTimeout() {
|
||||||
|
return timeout;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setTimeout(Long timeout) {
|
||||||
|
this.timeout = timeout;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,11 @@
|
||||||
|
package com.springboot.service;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import com.springboot.pojo.UserOnline;
|
||||||
|
|
||||||
|
public interface SessionService {
|
||||||
|
|
||||||
|
List<UserOnline> list();
|
||||||
|
boolean forceLogout(String sessionId);
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,64 @@
|
||||||
|
package com.springboot.service.impl;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.apache.shiro.session.Session;
|
||||||
|
import org.apache.shiro.session.mgt.eis.SessionDAO;
|
||||||
|
import org.apache.shiro.subject.SimplePrincipalCollection;
|
||||||
|
import org.apache.shiro.subject.support.DefaultSubjectContext;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
|
import com.springboot.pojo.User;
|
||||||
|
import com.springboot.pojo.UserOnline;
|
||||||
|
import com.springboot.service.SessionService;
|
||||||
|
|
||||||
|
@Service("sessionService")
|
||||||
|
public class SessionServiceImpl implements SessionService {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private SessionDAO sessionDAO;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<UserOnline> list() {
|
||||||
|
List<UserOnline> list = new ArrayList<>();
|
||||||
|
Collection<Session> sessions = sessionDAO.getActiveSessions();
|
||||||
|
for (Session session : sessions) {
|
||||||
|
UserOnline userOnline = new UserOnline();
|
||||||
|
User user = new User();
|
||||||
|
SimplePrincipalCollection principalCollection = new SimplePrincipalCollection();
|
||||||
|
if (session.getAttribute(DefaultSubjectContext.PRINCIPALS_SESSION_KEY) == null) {
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
principalCollection = (SimplePrincipalCollection) session
|
||||||
|
.getAttribute(DefaultSubjectContext.PRINCIPALS_SESSION_KEY);
|
||||||
|
user = (User) principalCollection.getPrimaryPrincipal();
|
||||||
|
userOnline.setUsername(user.getUserName());
|
||||||
|
userOnline.setUserId(user.getId().toString());
|
||||||
|
}
|
||||||
|
userOnline.setId((String) session.getId());
|
||||||
|
userOnline.setHost(session.getHost());
|
||||||
|
userOnline.setStartTimestamp(session.getStartTimestamp());
|
||||||
|
userOnline.setLastAccessTime(session.getLastAccessTime());
|
||||||
|
Long timeout = session.getTimeout();
|
||||||
|
if (timeout == 0l) {
|
||||||
|
userOnline.setStatus("离线");
|
||||||
|
} else {
|
||||||
|
userOnline.setStatus("在线");
|
||||||
|
}
|
||||||
|
userOnline.setTimeout(timeout);
|
||||||
|
list.add(userOnline);
|
||||||
|
}
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean forceLogout(String sessionId) {
|
||||||
|
Session session = sessionDAO.readSession(sessionId);
|
||||||
|
session.setTimeout(0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,90 @@
|
||||||
|
package com.springboot.shiro;
|
||||||
|
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
import org.apache.shiro.SecurityUtils;
|
||||||
|
import org.apache.shiro.authc.AuthenticationException;
|
||||||
|
import org.apache.shiro.authc.AuthenticationInfo;
|
||||||
|
import org.apache.shiro.authc.AuthenticationToken;
|
||||||
|
import org.apache.shiro.authc.IncorrectCredentialsException;
|
||||||
|
import org.apache.shiro.authc.LockedAccountException;
|
||||||
|
import org.apache.shiro.authc.SimpleAuthenticationInfo;
|
||||||
|
import org.apache.shiro.authc.UnknownAccountException;
|
||||||
|
import org.apache.shiro.authz.AuthorizationInfo;
|
||||||
|
import org.apache.shiro.authz.SimpleAuthorizationInfo;
|
||||||
|
import org.apache.shiro.realm.AuthorizingRealm;
|
||||||
|
import org.apache.shiro.subject.PrincipalCollection;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
|
||||||
|
import com.springboot.dao.UserMapper;
|
||||||
|
import com.springboot.dao.UserPermissionMapper;
|
||||||
|
import com.springboot.dao.UserRoleMapper;
|
||||||
|
import com.springboot.pojo.Permission;
|
||||||
|
import com.springboot.pojo.Role;
|
||||||
|
import com.springboot.pojo.User;
|
||||||
|
|
||||||
|
public class ShiroRealm extends AuthorizingRealm {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private UserMapper userMapper;
|
||||||
|
@Autowired
|
||||||
|
private UserRoleMapper userRoleMapper;
|
||||||
|
@Autowired
|
||||||
|
private UserPermissionMapper userPermissionMapper;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取用户角色和权限
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principal) {
|
||||||
|
User user = (User) SecurityUtils.getSubject().getPrincipal();
|
||||||
|
String userName = user.getUserName();
|
||||||
|
|
||||||
|
System.out.println("用户" + userName + "获取权限-----ShiroRealm.doGetAuthorizationInfo");
|
||||||
|
SimpleAuthorizationInfo simpleAuthorizationInfo = new SimpleAuthorizationInfo();
|
||||||
|
|
||||||
|
// 获取用户角色集
|
||||||
|
List<Role> roleList = userRoleMapper.findByUserName(userName);
|
||||||
|
Set<String> roleSet = new HashSet<String>();
|
||||||
|
for (Role r : roleList) {
|
||||||
|
roleSet.add(r.getName());
|
||||||
|
}
|
||||||
|
simpleAuthorizationInfo.setRoles(roleSet);
|
||||||
|
|
||||||
|
// 获取用户权限集
|
||||||
|
List<Permission> permissionList = userPermissionMapper.findByUserName(userName);
|
||||||
|
Set<String> permissionSet = new HashSet<String>();
|
||||||
|
for (Permission p : permissionList) {
|
||||||
|
permissionSet.add(p.getName());
|
||||||
|
}
|
||||||
|
simpleAuthorizationInfo.setStringPermissions(permissionSet);
|
||||||
|
return simpleAuthorizationInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 登录认证
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken token) throws AuthenticationException {
|
||||||
|
String userName = (String) token.getPrincipal();
|
||||||
|
String password = new String((char[]) token.getCredentials());
|
||||||
|
|
||||||
|
System.out.println("用户" + userName + "认证-----ShiroRealm.doGetAuthenticationInfo");
|
||||||
|
User user = userMapper.findByUserName(userName);
|
||||||
|
|
||||||
|
if (user == null) {
|
||||||
|
throw new UnknownAccountException("用户名或密码错误!");
|
||||||
|
}
|
||||||
|
if (!password.equals(user.getPassword())) {
|
||||||
|
throw new IncorrectCredentialsException("用户名或密码错误!");
|
||||||
|
}
|
||||||
|
if (user.getStatus().equals("0")) {
|
||||||
|
throw new LockedAccountException("账号已被锁定,请联系管理员!");
|
||||||
|
}
|
||||||
|
SimpleAuthenticationInfo info = new SimpleAuthenticationInfo(user, password, getName());
|
||||||
|
return info;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,28 @@
|
||||||
|
package com.springboot.util;
|
||||||
|
|
||||||
|
import org.apache.shiro.crypto.hash.SimpleHash;
|
||||||
|
import org.apache.shiro.util.ByteSource;
|
||||||
|
|
||||||
|
public class MD5Utils {
|
||||||
|
private static final String SALT = "mrbird";
|
||||||
|
|
||||||
|
private static final String ALGORITH_NAME = "md5";
|
||||||
|
|
||||||
|
private static final int HASH_ITERATIONS = 2;
|
||||||
|
|
||||||
|
public static String encrypt(String pswd) {
|
||||||
|
String newPassword = new SimpleHash(ALGORITH_NAME, pswd, ByteSource.Util.bytes(SALT), HASH_ITERATIONS).toHex();
|
||||||
|
return newPassword;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String encrypt(String username, String pswd) {
|
||||||
|
String newPassword = new SimpleHash(ALGORITH_NAME, pswd, ByteSource.Util.bytes(username + SALT),
|
||||||
|
HASH_ITERATIONS).toHex();
|
||||||
|
return newPassword;
|
||||||
|
}
|
||||||
|
public static void main(String[] args) {
|
||||||
|
|
||||||
|
System.out.println(MD5Utils.encrypt("tester", "123456"));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,77 @@
|
||||||
|
server:
|
||||||
|
context-path: /web
|
||||||
|
|
||||||
|
spring:
|
||||||
|
datasource:
|
||||||
|
druid:
|
||||||
|
# 数据库访问配置, 使用druid数据源
|
||||||
|
type: com.alibaba.druid.pool.DruidDataSource
|
||||||
|
driver-class-name: oracle.jdbc.driver.OracleDriver
|
||||||
|
url: jdbc:oracle:thin:@localhost:1521:ORCL
|
||||||
|
username: scott
|
||||||
|
password: 6742530
|
||||||
|
# 连接池配置
|
||||||
|
initial-size: 5
|
||||||
|
min-idle: 5
|
||||||
|
max-active: 20
|
||||||
|
# 连接等待超时时间
|
||||||
|
max-wait: 30000
|
||||||
|
# 配置检测可以关闭的空闲连接间隔时间
|
||||||
|
time-between-eviction-runs-millis: 60000
|
||||||
|
# 配置连接在池中的最小生存时间
|
||||||
|
min-evictable-idle-time-millis: 300000
|
||||||
|
validation-query: select '1' from dual
|
||||||
|
test-while-idle: true
|
||||||
|
test-on-borrow: false
|
||||||
|
test-on-return: false
|
||||||
|
# 打开PSCache,并且指定每个连接上PSCache的大小
|
||||||
|
pool-prepared-statements: true
|
||||||
|
max-open-prepared-statements: 20
|
||||||
|
max-pool-prepared-statement-per-connection-size: 20
|
||||||
|
# 配置监控统计拦截的filters, 去掉后监控界面sql无法统计, 'wall'用于防火墙
|
||||||
|
filters: stat
|
||||||
|
# Spring监控AOP切入点,如x.y.z.service.*,配置多个英文逗号分隔
|
||||||
|
aop-patterns: com.springboot.servie.*
|
||||||
|
|
||||||
|
|
||||||
|
# WebStatFilter配置
|
||||||
|
web-stat-filter:
|
||||||
|
enabled: true
|
||||||
|
# 添加过滤规则
|
||||||
|
url-pattern: /*
|
||||||
|
# 忽略过滤的格式
|
||||||
|
exclusions: '*.js,*.gif,*.jpg,*.png,*.css,*.ico,/druid/*'
|
||||||
|
|
||||||
|
# StatViewServlet配置
|
||||||
|
stat-view-servlet:
|
||||||
|
enabled: true
|
||||||
|
# 访问路径为/druid时,跳转到StatViewServlet
|
||||||
|
url-pattern: /druid/*
|
||||||
|
# 是否能够重置数据
|
||||||
|
reset-enable: false
|
||||||
|
# 需要账号密码才能访问控制台
|
||||||
|
login-username: druid
|
||||||
|
login-password: druid123
|
||||||
|
# IP白名单
|
||||||
|
# allow: 127.0.0.1
|
||||||
|
# IP黑名单(共同存在时,deny优先于allow)
|
||||||
|
# deny: 192.168.1.218
|
||||||
|
|
||||||
|
# 配置StatFilter
|
||||||
|
filter:
|
||||||
|
stat:
|
||||||
|
log-slow-sql: true
|
||||||
|
|
||||||
|
thymeleaf:
|
||||||
|
cache: false
|
||||||
|
|
||||||
|
|
||||||
|
mybatis:
|
||||||
|
# type-aliases扫描路径
|
||||||
|
type-aliases-package: com.springboot.pojo
|
||||||
|
# mapper xml实现扫描路径
|
||||||
|
mapper-locations: classpath:mapper/*.xml
|
||||||
|
property:
|
||||||
|
order: BEFORE
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -0,0 +1,24 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"
|
||||||
|
updateCheck="false">
|
||||||
|
<diskStore path="java.io.tmpdir/Tmp_EhCache" />
|
||||||
|
<defaultCache
|
||||||
|
maxElementsInMemory="10000"
|
||||||
|
eternal="false"
|
||||||
|
timeToIdleSeconds="120"
|
||||||
|
timeToLiveSeconds="120"
|
||||||
|
overflowToDisk="false"
|
||||||
|
diskPersistent="false"
|
||||||
|
diskExpiryThreadIntervalSeconds="120" />
|
||||||
|
|
||||||
|
<!-- 登录记录缓存锁定1小时 -->
|
||||||
|
<cache
|
||||||
|
name="passwordRetryCache"
|
||||||
|
maxEntriesLocalHeap="2000"
|
||||||
|
eternal="false"
|
||||||
|
timeToIdleSeconds="3600"
|
||||||
|
timeToLiveSeconds="0"
|
||||||
|
overflowToDisk="false"
|
||||||
|
statistics="true" />
|
||||||
|
</ehcache>
|
||||||
|
|
@ -0,0 +1,80 @@
|
||||||
|
-- ----------------------------
|
||||||
|
-- Table structure for T_PERMISSION
|
||||||
|
-- ----------------------------
|
||||||
|
CREATE TABLE T_PERMISSION (
|
||||||
|
ID NUMBER(10) NOT NULL ,
|
||||||
|
URL VARCHAR2(256 BYTE) NULL ,
|
||||||
|
NAME VARCHAR2(64 BYTE) NULL
|
||||||
|
);
|
||||||
|
COMMENT ON COLUMN T_PERMISSION.URL IS 'url地址';
|
||||||
|
COMMENT ON COLUMN T_PERMISSION.NAME IS 'url描述';
|
||||||
|
-- ----------------------------
|
||||||
|
-- Records of T_PERMISSION
|
||||||
|
-- ----------------------------
|
||||||
|
INSERT INTO T_PERMISSION VALUES ('1', '/user', 'user:user');
|
||||||
|
INSERT INTO T_PERMISSION VALUES ('2', '/user/add', 'user:add');
|
||||||
|
INSERT INTO T_PERMISSION VALUES ('3', '/user/delete', 'user:delete');
|
||||||
|
-- ----------------------------
|
||||||
|
-- Table structure for T_ROLE
|
||||||
|
-- ----------------------------
|
||||||
|
CREATE TABLE T_ROLE (
|
||||||
|
ID NUMBER NOT NULL ,
|
||||||
|
NAME VARCHAR2(32 BYTE) NULL ,
|
||||||
|
MEMO VARCHAR2(32 BYTE) NULL
|
||||||
|
);
|
||||||
|
COMMENT ON COLUMN T_ROLE.NAME IS '角色名称';
|
||||||
|
COMMENT ON COLUMN T_ROLE.MEMO IS '角色描述';
|
||||||
|
-- ----------------------------
|
||||||
|
-- Records of T_ROLE
|
||||||
|
-- ----------------------------
|
||||||
|
INSERT INTO T_ROLE VALUES ('1', 'admin', '超级管理员');
|
||||||
|
INSERT INTO T_ROLE VALUES ('2', 'test', '测试账户');
|
||||||
|
-- ----------------------------
|
||||||
|
-- Table structure for T_ROLE_PERMISSION
|
||||||
|
-- ----------------------------
|
||||||
|
CREATE TABLE T_ROLE_PERMISSION (
|
||||||
|
RID NUMBER(10) NULL ,
|
||||||
|
PID NUMBER(10) NULL
|
||||||
|
);
|
||||||
|
COMMENT ON COLUMN T_ROLE_PERMISSION.RID IS '角色id';
|
||||||
|
COMMENT ON COLUMN T_ROLE_PERMISSION.PID IS '权限id';
|
||||||
|
-- ----------------------------
|
||||||
|
-- Records of T_ROLE_PERMISSION
|
||||||
|
-- ----------------------------
|
||||||
|
INSERT INTO T_ROLE_PERMISSION VALUES ('1', '2');
|
||||||
|
INSERT INTO T_ROLE_PERMISSION VALUES ('1', '3');
|
||||||
|
INSERT INTO T_ROLE_PERMISSION VALUES ('2', '1');
|
||||||
|
INSERT INTO T_ROLE_PERMISSION VALUES ('1', '1');
|
||||||
|
-- ----------------------------
|
||||||
|
-- Table structure for T_USER
|
||||||
|
-- ----------------------------
|
||||||
|
CREATE TABLE T_USER (
|
||||||
|
ID NUMBER NOT NULL ,
|
||||||
|
USERNAME VARCHAR2(20 BYTE) NOT NULL ,
|
||||||
|
PASSWD VARCHAR2(128 BYTE) NOT NULL ,
|
||||||
|
CREATE_TIME DATE NULL ,
|
||||||
|
STATUS CHAR(1 BYTE) NOT NULL
|
||||||
|
);
|
||||||
|
COMMENT ON COLUMN T_USER.USERNAME IS '用户名';
|
||||||
|
COMMENT ON COLUMN T_USER.PASSWD IS '密码';
|
||||||
|
COMMENT ON COLUMN T_USER.CREATE_TIME IS '创建时间';
|
||||||
|
COMMENT ON COLUMN T_USER.STATUS IS '是否有效 1:有效 0:锁定';
|
||||||
|
-- ----------------------------
|
||||||
|
-- Records of T_USER
|
||||||
|
-- ----------------------------
|
||||||
|
INSERT INTO T_USER VALUES ('2', 'tester', '243e29429b340192700677d48c09d992', TO_DATE('2017-12-11 17:20:21', 'YYYY-MM-DD HH24:MI:SS'), '1');
|
||||||
|
INSERT INTO T_USER VALUES ('1', 'mrbird', '42ee25d1e43e9f57119a00d0a39e5250', TO_DATE('2017-12-11 10:52:48', 'YYYY-MM-DD HH24:MI:SS'), '1');
|
||||||
|
-- ----------------------------
|
||||||
|
-- Table structure for T_USER_ROLE
|
||||||
|
-- ----------------------------
|
||||||
|
CREATE TABLE T_USER_ROLE (
|
||||||
|
USER_ID NUMBER(10) NULL ,
|
||||||
|
RID NUMBER(10) NULL
|
||||||
|
);
|
||||||
|
COMMENT ON COLUMN T_USER_ROLE.USER_ID IS '用户id';
|
||||||
|
COMMENT ON COLUMN T_USER_ROLE.RID IS '角色id';
|
||||||
|
-- ----------------------------
|
||||||
|
-- Records of T_USER_ROLE
|
||||||
|
-- ----------------------------
|
||||||
|
INSERT INTO T_USER_ROLE VALUES ('1', '1');
|
||||||
|
INSERT INTO T_USER_ROLE VALUES ('2', '2');
|
||||||
|
|
@ -0,0 +1,17 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
|
||||||
|
<mapper namespace="com.springboot.dao.UserMapper">
|
||||||
|
|
||||||
|
<resultMap type="com.springboot.pojo.User" id="User">
|
||||||
|
<id column="id" property="id" javaType="java.lang.Integer" jdbcType="NUMERIC"/>
|
||||||
|
<id column="username" property="userName" javaType="java.lang.String" jdbcType="VARCHAR"/>
|
||||||
|
<id column="passwd" property="password" javaType="java.lang.String" jdbcType="VARCHAR"/>
|
||||||
|
<id column="create_time" property="createTime" javaType="java.util.Date" jdbcType="DATE"/>
|
||||||
|
<id column="status" property="status" javaType="java.lang.String" jdbcType="VARCHAR"/>
|
||||||
|
</resultMap>
|
||||||
|
|
||||||
|
<select id="findByUserName" resultMap="User">
|
||||||
|
select * from t_user where username = #{userName}
|
||||||
|
</select>
|
||||||
|
|
||||||
|
</mapper>
|
||||||
|
|
@ -0,0 +1,19 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
|
||||||
|
<mapper namespace="com.springboot.dao.UserPermissionMapper">
|
||||||
|
|
||||||
|
<resultMap type="com.springboot.pojo.Permission" id="permission">
|
||||||
|
<id column="id" property="id" javaType="java.lang.Integer" jdbcType="NUMERIC"/>
|
||||||
|
<id column="url" property="url" javaType="java.lang.String" jdbcType="VARCHAR"/>
|
||||||
|
<id column="name" property="name" javaType="java.lang.String" jdbcType="VARCHAR"/>
|
||||||
|
</resultMap>
|
||||||
|
|
||||||
|
<select id="findByUserName" resultMap="permission">
|
||||||
|
select p.id,p.url,p.name from t_role r
|
||||||
|
left join t_user_role ur on(r.id = ur.rid)
|
||||||
|
left join t_user u on(u.id = ur.user_id)
|
||||||
|
left join t_role_permission rp on(rp.rid = r.id)
|
||||||
|
left join t_permission p on(p.id = rp.pid )
|
||||||
|
where u.username = #{userName}
|
||||||
|
</select>
|
||||||
|
</mapper>
|
||||||
|
|
@ -0,0 +1,17 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
|
||||||
|
<mapper namespace="com.springboot.dao.UserRoleMapper">
|
||||||
|
|
||||||
|
<resultMap type="com.springboot.pojo.Role" id="role">
|
||||||
|
<id column="id" property="id" javaType="java.lang.Integer" jdbcType="NUMERIC"/>
|
||||||
|
<id column="name" property="name" javaType="java.lang.String" jdbcType="VARCHAR"/>
|
||||||
|
<id column="memo" property="memo" javaType="java.lang.String" jdbcType="VARCHAR"/>
|
||||||
|
</resultMap>
|
||||||
|
|
||||||
|
<select id="findByUserName" resultMap="role">
|
||||||
|
select r.id,r.name,r.memo from t_role r
|
||||||
|
left join t_user_role ur on(r.id = ur.rid)
|
||||||
|
left join t_user u on(u.id = ur.user_id)
|
||||||
|
where u.username = #{userName}
|
||||||
|
</select>
|
||||||
|
</mapper>
|
||||||
|
|
@ -0,0 +1,106 @@
|
||||||
|
.login-page {
|
||||||
|
width: 360px;
|
||||||
|
padding: 8% 0 0;
|
||||||
|
margin: auto;
|
||||||
|
}
|
||||||
|
.form {
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
background: #ffffff;
|
||||||
|
max-width: 360px;
|
||||||
|
margin: 0 auto 100px;
|
||||||
|
padding: 45px;
|
||||||
|
text-align: center;
|
||||||
|
box-shadow: 0 0 20px 0 rgba(0, 0, 0, 0.2), 0 5px 5px 0 rgba(0, 0, 0, 0.24);
|
||||||
|
}
|
||||||
|
.form input {
|
||||||
|
outline: 0;
|
||||||
|
background: #f2f2f2;
|
||||||
|
width: 100%;
|
||||||
|
border: 0;
|
||||||
|
margin: 0 0 15px;
|
||||||
|
padding: 15px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
.form button {
|
||||||
|
text-transform: uppercase;
|
||||||
|
outline: 0;
|
||||||
|
background: #4caf50;
|
||||||
|
width: 100%;
|
||||||
|
border: 0;
|
||||||
|
padding: 15px;
|
||||||
|
color: #ffffff;
|
||||||
|
font-size: 14px;
|
||||||
|
-webkit-transition: all 0.3 ease;
|
||||||
|
transition: all 0.3 ease;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.form button:hover,
|
||||||
|
.form button:active,
|
||||||
|
.form button:focus {
|
||||||
|
background: #43a047;
|
||||||
|
}
|
||||||
|
.form .message {
|
||||||
|
margin: 15px 0 0;
|
||||||
|
color: #b3b3b3;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
.form .message a {
|
||||||
|
color: #4caf50;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
.form .register-form {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
.form p {
|
||||||
|
text-align: left;
|
||||||
|
margin: 0;
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
.form p input {
|
||||||
|
width: auto;
|
||||||
|
margin-right: 10px;
|
||||||
|
}
|
||||||
|
.container {
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
max-width: 300px;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
.container:before,
|
||||||
|
.container:after {
|
||||||
|
content: "";
|
||||||
|
display: block;
|
||||||
|
clear: both;
|
||||||
|
}
|
||||||
|
.container .info {
|
||||||
|
margin: 50px auto;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
.container .info h1 {
|
||||||
|
margin: 0 0 15px;
|
||||||
|
padding: 0;
|
||||||
|
font-size: 36px;
|
||||||
|
font-weight: 300;
|
||||||
|
color: #1a1a1a;
|
||||||
|
}
|
||||||
|
.container .info span {
|
||||||
|
color: #4d4d4d;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
.container .info span a {
|
||||||
|
color: #000000;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
.container .info span .fa {
|
||||||
|
color: #ef3b3a;
|
||||||
|
}
|
||||||
|
body {
|
||||||
|
background: #76b852; /* fallback for old browsers */
|
||||||
|
background: -webkit-linear-gradient(right, #76b852, #8dc26f);
|
||||||
|
background: -moz-linear-gradient(right, #76b852, #8dc26f);
|
||||||
|
background: -o-linear-gradient(right, #76b852, #8dc26f);
|
||||||
|
background: linear-gradient(to left, #76b852, #8dc26f);
|
||||||
|
font-family: Lato,"PingFang SC","Microsoft YaHei",sans-serif;
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,22 @@
|
||||||
|
Date.prototype.Format = function (fmt) {
|
||||||
|
var o = {
|
||||||
|
"M+": this.getMonth() + 1,
|
||||||
|
"d+": this.getDate(),
|
||||||
|
"h+": this.getHours(),
|
||||||
|
"m+": this.getMinutes(),
|
||||||
|
"s+": this.getSeconds(),
|
||||||
|
"q+": Math.floor((this.getMonth() + 3) / 3),
|
||||||
|
"S": this.getMilliseconds()
|
||||||
|
};
|
||||||
|
if (/(y+)/.test(fmt)){
|
||||||
|
fmt = fmt.replace(RegExp.$1, (this.getFullYear() + "")
|
||||||
|
.substr(4 - RegExp.$1.length));
|
||||||
|
}
|
||||||
|
for (var k in o){
|
||||||
|
if (new RegExp("(" + k + ")").test(fmt)){
|
||||||
|
fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ?
|
||||||
|
(o[k]) : (("00" + o[k]).substr(("" + o[k]).length)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return fmt;
|
||||||
|
}
|
||||||
4
17.Spring-Boot-Shiro-Session/src/main/resources/static/js/jquery-1.11.1.min.js
vendored
Normal file
4
17.Spring-Boot-Shiro-Session/src/main/resources/static/js/jquery-1.11.1.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
|
|
@ -0,0 +1,11 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html xmlns:th="http://www.thymeleaf.org">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>暂无权限</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<p>您没有权限访问该资源!!</p>
|
||||||
|
<a th:href="@{/index}">返回</a>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
|
@ -0,0 +1,28 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html
|
||||||
|
xmlns:th="http://www.thymeleaf.org"
|
||||||
|
xmlns:shiro="http://www.pollix.at/thymeleaf/shiro" >
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>首页</title>
|
||||||
|
</head>
|
||||||
|
<style>
|
||||||
|
div {
|
||||||
|
border: 1px dashed #ddd;
|
||||||
|
padding: 10px;
|
||||||
|
margin: 10px 10px 10px 0px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<body>
|
||||||
|
<p>你好![[${user.userName}]]</p>
|
||||||
|
<p shiro:hasRole="admin">你的角色为超级管理员</p>
|
||||||
|
<p shiro:hasRole="test">你的角色为测试账户</p>
|
||||||
|
<div>
|
||||||
|
<a shiro:hasPermission="user:user" th:href="@{/user/list}">获取用户信息</a>
|
||||||
|
<a shiro:hasPermission="user:add" th:href="@{/user/add}">新增用户</a>
|
||||||
|
<a shiro:hasPermission="user:delete" th:href="@{/user/delete}">删除用户</a>
|
||||||
|
</div>
|
||||||
|
<a shiro:hasRole="admin" th:href="@{/online/index}">在线用户管理</a>
|
||||||
|
<a th:href="@{/logout}">注销</a>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
|
@ -0,0 +1,40 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html xmlns:th="http://www.thymeleaf.org">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>登录</title>
|
||||||
|
<link rel="stylesheet" th:href="@{/css/login.css}" type="text/css">
|
||||||
|
<script th:src="@{/js/jquery-1.11.1.min.js}"></script>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="login-page">
|
||||||
|
<div class="form">
|
||||||
|
<input type="text" placeholder="用户名" name="username" required="required"/>
|
||||||
|
<input type="password" placeholder="密码" name="password" required="required"/>
|
||||||
|
<p><input type="checkbox" name="rememberMe" />记住我</p>
|
||||||
|
<button onclick="login()">登录</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
<script th:inline="javascript">
|
||||||
|
var ctx = [[@{/}]];
|
||||||
|
function login() {
|
||||||
|
var username = $("input[name='username']").val();
|
||||||
|
var password = $("input[name='password']").val();
|
||||||
|
var rememberMe =$("input[name='rememberMe']").is(':checked');
|
||||||
|
$.ajax({
|
||||||
|
type: "post",
|
||||||
|
url: ctx + "login",
|
||||||
|
data: {"username": username,"password": password,"rememberMe": rememberMe},
|
||||||
|
dataType: "json",
|
||||||
|
success: function (r) {
|
||||||
|
if (r.code == 0) {
|
||||||
|
location.href = ctx + 'index';
|
||||||
|
} else {
|
||||||
|
alert(r.msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</html>
|
||||||
|
|
@ -0,0 +1,94 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html xmlns:th="http://www.thymeleaf.org">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>在线用户管理</title>
|
||||||
|
<script th:src="@{/js/jquery-1.11.1.min.js}"></script>
|
||||||
|
<script th:src="@{/js/dateFormat.js}"></script>
|
||||||
|
</head>
|
||||||
|
<style>
|
||||||
|
table {
|
||||||
|
margin: 20px 40px 20px 0px;
|
||||||
|
width: 100%;
|
||||||
|
border-collapse: collapse;
|
||||||
|
border-spacing: 0;
|
||||||
|
table-layout: automatic;
|
||||||
|
word-wrap: break-all
|
||||||
|
}
|
||||||
|
table>tbody>tr:nth-of-type(odd) {
|
||||||
|
background-color: #F7F7F7
|
||||||
|
}
|
||||||
|
|
||||||
|
th, td {
|
||||||
|
padding: 8px;
|
||||||
|
text-align: left;
|
||||||
|
vertical-align: middle;
|
||||||
|
font-weight: normal;
|
||||||
|
font-size: 12px;
|
||||||
|
border-bottom: 1px solid #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
th {
|
||||||
|
padding-bottom: 10px;
|
||||||
|
color: #fff;
|
||||||
|
font-weight: 700;
|
||||||
|
background: rgba(66, 185, 131, .9)
|
||||||
|
}
|
||||||
|
|
||||||
|
td {
|
||||||
|
border-bottom-width: 1px
|
||||||
|
}
|
||||||
|
|
||||||
|
</style>
|
||||||
|
<body>
|
||||||
|
<h3>在线用户数:<span id="onlineCount"></span></h3>
|
||||||
|
<table>
|
||||||
|
<tr>
|
||||||
|
<th>序号</th>
|
||||||
|
<th>用户名称</th>
|
||||||
|
<th>登录时间</th>
|
||||||
|
<th>最后访问时间</th>
|
||||||
|
<th>主机</th>
|
||||||
|
<th>状态</th>
|
||||||
|
<th>操作</th>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
<a th:href="@{/index}">返回</a>
|
||||||
|
</body>
|
||||||
|
<script th:inline="javascript">
|
||||||
|
var ctx = [[@{/}]];
|
||||||
|
$.get(ctx + "online/list", {}, function(r){
|
||||||
|
console.log(r);
|
||||||
|
var length = r.length;
|
||||||
|
$("#onlineCount").text(length);
|
||||||
|
var html = "";
|
||||||
|
for(var i = 0; i < length; i++){
|
||||||
|
html += "<tr>"
|
||||||
|
+ "<td>" + (i+1) + "</td>"
|
||||||
|
+ "<td>" + r[i].username + "</td>"
|
||||||
|
+ "<td>" + new Date(r[i].startTimestamp).Format("yyyy-MM-dd hh:mm:ss") + "</td>"
|
||||||
|
+ "<td>" + new Date(r[i].lastAccessTime).Format("yyyy-MM-dd hh:mm:ss") + "</td>"
|
||||||
|
+ "<td>" + r[i].host + "</td>"
|
||||||
|
+ "<td>" + r[i].status + "</td>"
|
||||||
|
+ "<td><a href='#' onclick='offline(\"" + r[i].id + "\",\"" + r[i].status +"\")'>下线</a></td>"
|
||||||
|
+ "</tr>";
|
||||||
|
}
|
||||||
|
$("table").append(html);
|
||||||
|
},"json");
|
||||||
|
|
||||||
|
function offline(id,status){
|
||||||
|
if(status == "离线"){
|
||||||
|
alert("该用户已是离线状态!!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$.get(ctx + "online/forceLogout", {"id": id}, function(r){
|
||||||
|
if (r.code == 0) {
|
||||||
|
alert('该用户已强制下线!');
|
||||||
|
location.href = ctx + 'online/index';
|
||||||
|
} else {
|
||||||
|
alert(r.msg);
|
||||||
|
}
|
||||||
|
},"json");
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</html>
|
||||||
|
|
@ -0,0 +1,11 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html xmlns:th="http://www.thymeleaf.org">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>[[${value}]]</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<p>[[${value}]]</p>
|
||||||
|
<a th:href="@{/index}">返回</a>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
Loading…
Reference in New Issue