All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.swisspush.gateleen.routing.RuleFactory Maven / Gradle / Ivy

There is a newer version: 2.1.13
Show newest version
package org.swisspush.gateleen.routing;

import io.vertx.core.buffer.Buffer;
import io.vertx.core.http.HttpClientOptions;
import io.vertx.core.json.JsonArray;
import io.vertx.core.json.JsonObject;
import io.vertx.core.net.ProxyOptions;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.swisspush.gateleen.core.http.HeaderFunction;
import org.swisspush.gateleen.core.http.HeaderFunctions;
import org.swisspush.gateleen.core.util.StringUtils;
import org.swisspush.gateleen.core.validation.ValidationResult;
import org.swisspush.gateleen.validation.RegexpValidator;
import org.swisspush.gateleen.validation.ValidationException;
import org.swisspush.gateleen.validation.Validator;

import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * RuleFactory is used to create rule objects from their text representation.
 */
public class RuleFactory {

    private Logger log = LoggerFactory.getLogger(RuleFactory.class);
    private Pattern urlParsePattern = Pattern.compile("^(?https?)://(?[^/:]+)(:(?\\$?[0-9]+))?(?/.*)$");
    private final Map properties;
    private String routingRulesSchema;

    public RuleFactory(Map properties, String routingRulesSchema) {
        this.properties = properties;
        this.routingRulesSchema = routingRulesSchema;
    }

    public List parseRules(Buffer buffer, int routeMultiplier) throws ValidationException {
        String replacedConfig;
        try {
            replacedConfig = StringUtils.replaceWildcardConfigs(buffer.toString("UTF-8"), properties);
        } catch (Exception e) {
            throw new ValidationException(e);
        }
        ValidationResult validationResult = Validator.validateStatic(Buffer.buffer(replacedConfig), routingRulesSchema, log);
        if (validationResult.isSuccess()) {
            return createRules(new JsonObject(replacedConfig), routeMultiplier);
        } else {
            throw new ValidationException(validationResult);
        }
    }

    public List createRules(JsonObject rules, int routeMultiplier) throws ValidationException {
        Set metricNames = new HashSet<>();
        List result = new ArrayList<>();
        for (String urlPattern : rules.fieldNames()) {
            log.debug("Creating a new rule-object for URL pattern {}", urlPattern);

            Rule ruleObj = new Rule();
            ruleObj.setUrlPattern(urlPattern);
            JsonObject rule = rules.getJsonObject(urlPattern);

            String headersFilter = rule.getString("headersFilter");
            if(headersFilter != null) {
                Pattern headersFilterPattern = RegexpValidator.throwIfPatternInvalid(headersFilter);
                ruleObj.setHeadersFilterPattern(headersFilterPattern);
            }

            String targetUrl = rule.getString("url");
            String path = rule.getString("path");

            prepareUrl(urlPattern, ruleObj, targetUrl, path);

            String metricName = rule.getString("metricName");
            if (metricName != null) {
                if(metricNames.contains(metricName)){
                    throw new ValidationException("Property 'metricName' must be unique. There are multiple rules with metricName '" + metricName + "'");
                } else {
                    metricNames.add(metricName);
                    ruleObj.setMetricName(metricName);
                }
            }

            ruleObj.setTimeout(1000 * rule.getInteger(Rule.CONNECTION_TIMEOUT_SEC_PROPERTY_NAME, Rule.CONNECTION_TIMEOUT_SEC_DEFAULT_VALUE));
            ruleObj.setKeepAliveTimeout(rule.getInteger("keepAliveTimeout", HttpClientOptions.DEFAULT_KEEP_ALIVE_TIMEOUT));
            ruleObj.setPoolSize(rule.getInteger(Rule.CONNECTION_POOL_SIZE_PROPERTY_NAME, Rule.CONNECTION_POOL_SIZE_DEFAULT_VALUE));

            int originalPoolSize = ruleObj.getPoolSize();
            int appliedPoolSize = evaluatePoolSize(originalPoolSize, routeMultiplier);

            ruleObj.setPoolSize(appliedPoolSize);
            log.debug("Original pool size is {}, applied size is {}", originalPoolSize, appliedPoolSize);

            ruleObj.setMaxWaitQueueSize(rule.getInteger(Rule.MAX_WAIT_QUEUE_SIZE_PROPERTY_NAME, Rule.MAX_WAIT_QUEUE_SIZE_DEFAULT_VALUE));
            ruleObj.setKeepAlive(rule.getBoolean("keepAlive", true));
            ruleObj.setExpandOnBackend(rule.getBoolean("expandOnBackend", false));
            ruleObj.setDeltaOnBackend(rule.getBoolean("deltaOnBackend", false));
            ruleObj.setStorageExpand(rule.getBoolean("storageExpand", false));
            ruleObj.setLogExpiry(rule.getInteger("logExpiry", 4 * 3600));

            JsonArray methods = rule.getJsonArray("methods");
            if (methods != null) {
                String[] methodsArray = new String[methods.size()];
                for (int i=0; i < methods.size(); i++) {
                    methodsArray[i] = methods.getString(i);
                }
                ruleObj.setMethods(methodsArray);
            }

            JsonArray profile = rule.getJsonArray("profile");
            if (profile != null) {
                String[] profileArray = new String[profile.size()];
                for (int i=0; i < profile.size(); i++) {
                    profileArray[i] = profile.getString(i);
                }
                ruleObj.setProfile(profileArray);
                log.debug("The profile-array is set. Those profile-information will be sent: {}", Arrays.toString(ruleObj.getProfile()));
            } else {
                log.debug("The profile-array is not set. So won't send any profile-information when using this rule.");
            }

            setStorage(ruleObj, rule, path);
            setTranslateStatus(ruleObj, rule);
            setStaticHeaders(ruleObj, rule);
            setProxyOptions(ruleObj, rule);
            setAuthentication(ruleObj, rule);

            result.add(ruleObj);
        }
        return result;
    }

    public static int evaluatePoolSize(int originalPoolSize, int routeMultiplier) {
        return ceilDiv(originalPoolSize, routeMultiplier);
    }

    private static int ceilDiv(int x, int y){
        return -Math.floorDiv(-x,y);
    }

    private void setStorage(Rule ruleObj, JsonObject rule, String path) throws ValidationException {
        String ruleStorage = rule.getString("storage");
        if (ruleStorage != null) {
            if (path == null) {
                throw new ValidationException("For storage routing, 'path' must be specified.");
            }
            ruleObj.setScheme("storage");
            ruleObj.setStorage(ruleStorage);
        }
    }

    private void setTranslateStatus(Rule ruleObj, JsonObject rule) {
        JsonObject translateStatus = rule.getJsonObject("translateStatus");
        if (translateStatus != null) {
            ruleObj.addTranslateStatus(new LinkedHashMap<>());
            for (String pattern : translateStatus.fieldNames()) {
                ruleObj.getTranslateStatus().put(Pattern.compile(pattern), translateStatus.getInteger(pattern));
            }
        }
    }

    private void setProxyOptions(Rule ruleObj, JsonObject rule) {
        JsonObject proxyOptions = rule.getJsonObject("proxyOptions");
        if(proxyOptions != null){
            ruleObj.setProxyOptions(new ProxyOptions(proxyOptions));
        }
    }

    private void setAuthentication(Rule ruleObj, JsonObject rule) throws ValidationException {
        JsonObject basicAuth = rule.getJsonObject("basicAuth");
        String oAuthId = rule.getString("oAuthId");

        if (basicAuth != null && oAuthId != null) {
            throw new ValidationException("Either 'basicAuth' or 'oAuthId' can be given, not both");
        }

        if (basicAuth != null) {
            ruleObj.setBasicAuthUsername(basicAuth.getString("username"));
            ruleObj.setBasicAuthPassword(basicAuth.getString("password"));
        } else if (oAuthId != null) {
            ruleObj.setOAuthId(oAuthId);
        }
    }

    @Deprecated
    private void setStaticHeaders(Rule ruleObj, JsonObject rule) {
        final JsonArray headers = rule.getJsonArray("headers");
        if (headers != null) {
            final HeaderFunction headerFunction = HeaderFunctions.parseFromJson(headers);
            ruleObj.setHeaderFunction(headerFunction);
            return;
        }

        // in previous Gateleen versions we only had the "staticHeaders" to unconditionally add headers with fix values
        // We now have a more dynamic concept of a "manipulator chain" - which is also configured different in JSON syntax
        // For backward compatibility we still parse the old "staticHeaders" - but now create a manipulator chain accordingly
        JsonObject staticHeaders = rule.getJsonObject("staticHeaders");
        if (staticHeaders != null) {
            log.warn("you use the deprecated \"staticHeaders\" syntax in your routing rule JSON ({}). Please migrate to the more flexible \"headers\" syntax", rule);
            ruleObj.setHeaderFunction(HeaderFunctions.parseStaticHeadersFromJson(staticHeaders));
        }
    }

    private void prepareUrl(String urlPattern, Rule ruleObj, String targetUrl, String path) throws ValidationException {
        if (targetUrl != null || path != null) {
            if (targetUrl != null && path != null) {
                throw new ValidationException("Either 'url' or 'path' must be given, not both");
            }
            if (targetUrl != null) {
                Matcher urlMatcher = urlParsePattern.matcher(targetUrl);
                if (!urlMatcher.matches()) {
                    throw new ValidationException("Invalid url for pattern " + urlPattern + ": " + targetUrl);
                }
                ruleObj.setHost(urlMatcher.group("host"));
                ruleObj.setScheme(urlMatcher.group("scheme"));
                ruleObj.setPort(ruleObj.getScheme().equals("https") ? 443 : 80);

                String portString = urlMatcher.group("port");
                if (portString != null) {
                    if(portString.startsWith("$")){
                        ruleObj.setPortWildcard(portString);
                    } else {
                        ruleObj.setPort(Integer.parseInt(portString));
                    }
                }

                ruleObj.setPath(urlMatcher.group("path"));
            } else {
                if (!path.startsWith("/")) {
                    throw new ValidationException("Illegal value for 'path', it must be a path starting with slash");
                }
                ruleObj.setPath(path);
                ruleObj.setHost("localhost");
                ruleObj.setScheme("local");
            }
        } else {
            // preventing a nullpointer
            ruleObj.setScheme("null");
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy