/*
* Symphony - A modern community (forum/SNS/blog) platform written in Java.
* Copyright (C) 2012-2017, b3log.org & hacpai.com
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package org.b3log.symphony.processor.advice;
import org.b3log.latke.Keys;
import org.b3log.latke.ioc.inject.Inject;
import org.b3log.latke.ioc.inject.Named;
import org.b3log.latke.ioc.inject.Singleton;
import org.b3log.latke.logging.Level;
import org.b3log.latke.logging.Logger;
import org.b3log.latke.model.User;
import org.b3log.latke.service.LangPropsService;
import org.b3log.latke.servlet.DispatcherServlet;
import org.b3log.latke.servlet.HTTPRequestContext;
import org.b3log.latke.servlet.advice.BeforeRequestProcessAdvice;
import org.b3log.latke.servlet.advice.RequestProcessAdviceException;
import org.b3log.latke.servlet.handler.MatchResult;
import org.b3log.latke.servlet.handler.RequestDispatchHandler;
import org.b3log.latke.util.Stopwatchs;
import org.b3log.symphony.model.Permission;
import org.b3log.symphony.model.Role;
import org.b3log.symphony.service.RoleQueryService;
import org.b3log.symphony.util.Symphonys;
import org.json.JSONObject;
import javax.servlet.http.HttpServletRequest;
import java.lang.reflect.Method;
import java.util.*;
/**
* Permission check.
*
* @author <a href="http://88250.b3log.org">Liang Ding</a>
* @version 1.0.0.2, Dec 19, 2016
* @since 1.8.0
*/
@Named
@Singleton
public class PermissionCheck extends BeforeRequestProcessAdvice {
/**
* Logger.
*/
private static final Logger LOGGER = Logger.getLogger(PermissionCheck.class);
/**
* URL permission rules.
* <p>
* <"url:method", permissions>
* </p>
*/
private static final Map<String, Set<String>> URL_PERMISSION_RULES = new HashMap<>();
static {
// Loads permission URL rules
final String prefix = "permission.rule.url.";
final Set<String> keys = Symphonys.CFG.keySet();
for (final String key : keys) {
if (key.startsWith(prefix)) {
final String value = Symphonys.CFG.getString(key);
final Set<String> permissions = new HashSet<>(Arrays.asList(value.split(",")));
URL_PERMISSION_RULES.put(key, permissions);
}
}
}
/**
* Language service.
*/
@Inject
private LangPropsService langPropsService;
/**
* Role query service.
*/
@Inject
private RoleQueryService roleQueryService;
@Override
public void doAdvice(final HTTPRequestContext context, final Map<String, Object> args) throws RequestProcessAdviceException {
Stopwatchs.start("Check Permissions");
try {
final HttpServletRequest request = context.getRequest();
final JSONObject exception = new JSONObject();
exception.put(Keys.MSG, langPropsService.get("noPermissionLabel"));
exception.put(Keys.STATUS_CODE, false);
final String prefix = "permission.rule.url.";
final String requestURI = request.getRequestURI();
final String method = request.getMethod();
String rule = prefix;
final RequestDispatchHandler requestDispatchHandler
= (RequestDispatchHandler) DispatcherServlet.SYS_HANDLER.get(2 /* DispatcherServlet#L69 */);
try {
final Method doMatch = RequestDispatchHandler.class.getDeclaredMethod("doMatch",
String.class, String.class);
doMatch.setAccessible(true);
final MatchResult matchResult = (MatchResult) doMatch.invoke(requestDispatchHandler, requestURI, method);
rule += matchResult.getMatchedPattern() + "." + method;
} catch (final Exception e) {
LOGGER.log(Level.ERROR, "Match method failed", e);
throw new RequestProcessAdviceException(exception);
}
final Set<String> requisitePermissions = URL_PERMISSION_RULES.get(rule);
if (null == requisitePermissions) {
return;
}
final JSONObject user = (JSONObject) request.getAttribute(User.USER);
final String roleId = null != user ? user.optString(User.USER_ROLE) : Role.ROLE_ID_C_VISITOR;
final Set<String> grantPermissions = roleQueryService.getPermissions(roleId);
if (!Permission.hasPermission(requisitePermissions, grantPermissions)) {
throw new RequestProcessAdviceException(exception);
}
} finally {
Stopwatchs.end();
}
}
}