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

org.killbill.billing.util.config.tenant.MultiTenantConfigBase Maven / Gradle / Ivy

There is a newer version: 0.24.11
Show newest version
/*
 * Copyright 2014-2016 Groupon, Inc
 * Copyright 2014-2016 The Billing Project, LLC
 *
 * The Billing Project licenses this file to you under the Apache License, version 2.0
 * (the "License"); you may not use this file except in compliance with the
 * License.  You may obtain a copy of the License at:
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the
 * License for the specific language governing permissions and limitations
 * under the License.
 */

package org.killbill.billing.util.config.tenant;

import java.lang.reflect.Method;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import java.util.stream.Collectors;

import org.killbill.billing.callcontext.InternalTenantContext;
import org.killbill.billing.events.BusInternalEvent.BusInternalEventType;
import org.killbill.billing.util.config.definition.KillbillConfig;
import org.skife.config.Config;
import org.skife.config.Separator;
import org.skife.config.TimeSpan;

public abstract class MultiTenantConfigBase implements KillbillConfig {

    private final Map methodsCache = new HashMap<>();
    protected final KillbillConfig staticConfig;

    protected final CacheConfig cacheConfig;

    private static final Function INT_CONVERTER = Integer::valueOf;

    private static final Function TIME_SPAN_CONVERTER = TimeSpan::new;

    private static final Function BUS_EVENT_TYPE_CONVERTER = BusInternalEventType::valueOf;


    public MultiTenantConfigBase(final KillbillConfig staticConfig, final CacheConfig cacheConfig) {
        this.staticConfig = staticConfig;
        this.cacheConfig = cacheConfig;
    }

    //
    // The conversion methods are rather limited (but this is all we need).
    // Ideally we could reuse the bully/Coercer from skife package, but those are kept private.
    //
    protected List convertToListString(final String value, final String methodName) {
        final Method method = getConfigStaticMethodWithChecking(methodName);
        final List tokens = getTokens(method, value);
        return List.copyOf(tokens);
    }

    protected List convertToListTimeSpan(final String value, final String methodName) {
        final Method method = getConfigStaticMethodWithChecking(methodName);
        return getTokens(method, value).stream()
                .map(TIME_SPAN_CONVERTER)
                .collect(Collectors.toUnmodifiableList());
    }

    protected List convertToListBusInternalEventType(final String value, final String methodName) {
        final Method method = getConfigStaticMethodWithChecking(methodName);
        return getTokens(method, value).stream()
                .map(BUS_EVENT_TYPE_CONVERTER)
                .collect(Collectors.toUnmodifiableList());
    }

    protected List convertToListInteger(final String value, final String methodName) {
        final Method method = getConfigStaticMethodWithChecking(methodName);
        return getTokens(method, value).stream()
                .map(INT_CONVERTER)
                .collect(Collectors.toUnmodifiableList());
    }

    protected String getStringTenantConfig(final String methodName, final InternalTenantContext tenantContext) {
        // That means we want to default to static config value
        if (tenantContext == null) {
            return null;
        }
        final Method method = getConfigStaticMethodWithChecking(methodName);
        return getCachedValue(method.getAnnotation(Config.class), tenantContext);
    }

    private String getCachedValue(final Config annotation, final InternalTenantContext tenantContext) {
        final PerTenantConfig perTenantConfig = cacheConfig.getPerTenantConfig(tenantContext);
        for (final String propertyName : annotation.value()) {
            final String result = perTenantConfig.get(propertyName);
            if (result != null) {
                return result;
            }
        }
        return null;
    }

    private Method getConfigStaticMethodWithChecking(final String methodName) {
        final Method method = getConfigStaticMethod(methodName);
        if (!method.isAnnotationPresent(Config.class)) {
            throw new RuntimeException("Missing @Config annotation to skife config method " + method.getName());
        }
        return method;
    }

    private List getTokens(final Method method, final String value) {
        final Separator separator = method.getAnnotation(Separator.class);
        if (value == null || value.isEmpty()) {
            return Collections.emptyList();
        } else {
            return List.of(value.split(separator == null ? Separator.DEFAULT : separator.value()));
        }
    }

    protected Method getConfigStaticMethod(final String methodName) {
        Method method = methodsCache.get(methodName);
        if (method == null) {
            synchronized (methodsCache) {
                method = methodsCache.get(methodName);
                if (method == null) {
                    try {
                        method = getConfigClass().getMethod(methodName, InternalTenantContext.class);
                        methodsCache.put(methodName, method);
                    } catch (final NoSuchMethodException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        }
        return method;
    }

    protected abstract Class getConfigClass();
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy