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

org.apache.flink.runtime.state.StateBackendLoader Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF 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.apache.flink.runtime.state;

import org.apache.flink.configuration.Configuration;
import org.apache.flink.configuration.IllegalConfigurationException;
import org.apache.flink.configuration.ReadableConfig;
import org.apache.flink.configuration.StateBackendOptions;
import org.apache.flink.configuration.StateChangelogOptions;
import org.apache.flink.runtime.state.changelog.ChangelogStateBackendHandle;
import org.apache.flink.runtime.state.delegate.DelegatingStateBackend;
import org.apache.flink.runtime.state.hashmap.HashMapStateBackend;
import org.apache.flink.runtime.state.hashmap.HashMapStateBackendFactory;
import org.apache.flink.util.DynamicCodeLoadingException;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;

import java.io.IOException;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.util.Collection;
import java.util.Optional;

import static org.apache.flink.util.Preconditions.checkNotNull;

/** This class contains utility methods to load state backends from configurations. */
public class StateBackendLoader {

    private static final Logger LOG = LoggerFactory.getLogger(StateBackendLoader.class);

    /** Used for Loading ChangelogStateBackend. */
    private static final String CHANGELOG_STATE_BACKEND =
            "org.apache.flink.state.changelog.ChangelogStateBackend";

    /** Used for Loading TempChangelogStateBackend. */
    private static final String DEACTIVATED_CHANGELOG_STATE_BACKEND =
            "org.apache.flink.state.changelog.DeactivatedChangelogStateBackend";

    /** Used for loading RocksDBStateBackend. */
    private static final String ROCKSDB_STATE_BACKEND_FACTORY =
            "org.apache.flink.contrib.streaming.state.EmbeddedRocksDBStateBackendFactory";

    /** Used for loading ForStStateBackend. */
    private static final String FORST_STATE_BACKEND_FACTORY =
            "org.apache.flink.state.forst.ForStStateBackendFactory";

    // ------------------------------------------------------------------------
    //  Configuration shortcut names
    // ------------------------------------------------------------------------
    /** The shortcut configuration name of the HashMap state backend. */
    public static final String HASHMAP_STATE_BACKEND_NAME = "hashmap";

    /** The shortcut configuration name for the RocksDB State Backend. */
    public static final String ROCKSDB_STATE_BACKEND_NAME = "rocksdb";

    public static final String FORST_STATE_BACKEND_NAME = "forst";

    // ------------------------------------------------------------------------
    //  Loading the state backend from a configuration
    // ------------------------------------------------------------------------

    /**
     * Loads the unwrapped state backend from the configuration, from the parameter 'state.backend',
     * as defined in {@link StateBackendOptions#STATE_BACKEND}.
     *
     * 

The state backends can be specified either via their shortcut name, or via the class name * of a {@link StateBackendFactory}. If a StateBackendFactory class name is specified, the * factory is instantiated (via its zero-argument constructor) and its {@link * StateBackendFactory#createFromConfig(ReadableConfig, ClassLoader)} method is called. * *

Recognized shortcut names are '{@value StateBackendLoader#HASHMAP_STATE_BACKEND_NAME}', * '{@value StateBackendLoader#ROCKSDB_STATE_BACKEND_NAME}' * * @param config The configuration to load the state backend from * @param classLoader The class loader that should be used to load the state backend * @param logger Optionally, a logger to log actions to (may be null) * @return The instantiated state backend. * @throws DynamicCodeLoadingException Thrown if a state backend factory is configured and the * factory class was not found or the factory could not be instantiated * @throws IllegalConfigurationException May be thrown by the StateBackendFactory when creating * / configuring the state backend in the factory * @throws IOException May be thrown by the StateBackendFactory when instantiating the state * backend */ @Nonnull public static StateBackend loadStateBackendFromConfig( ReadableConfig config, ClassLoader classLoader, @Nullable Logger logger) throws IllegalConfigurationException, DynamicCodeLoadingException, IOException { checkNotNull(config, "config"); checkNotNull(classLoader, "classLoader"); final String backendName = config.get(StateBackendOptions.STATE_BACKEND); // by default the factory class is the backend name String factoryClassName = backendName; switch (backendName.toLowerCase()) { case HASHMAP_STATE_BACKEND_NAME: HashMapStateBackend hashMapStateBackend = new HashMapStateBackendFactory().createFromConfig(config, classLoader); if (logger != null) { logger.info("State backend is set to heap memory {}", hashMapStateBackend); } return hashMapStateBackend; case ROCKSDB_STATE_BACKEND_NAME: factoryClassName = ROCKSDB_STATE_BACKEND_FACTORY; // fall through to the case that uses reflection to load the backend // that way we can keep RocksDB in a separate module break; case FORST_STATE_BACKEND_NAME: factoryClassName = FORST_STATE_BACKEND_FACTORY; // fall through to the case that uses reflection to load the backend // that way we can keep ForSt in a separate module break; } // The reflection loading path if (logger != null) { logger.info("Loading state backend via factory {}", factoryClassName); } StateBackendFactory factory; try { @SuppressWarnings("rawtypes") Class clazz = Class.forName(factoryClassName, false, classLoader) .asSubclass(StateBackendFactory.class); factory = clazz.newInstance(); } catch (ClassNotFoundException e) { throw new DynamicCodeLoadingException( "Cannot find configured state backend factory class: " + backendName, e); } catch (ClassCastException | InstantiationException | IllegalAccessException e) { throw new DynamicCodeLoadingException( "The class configured under '" + StateBackendOptions.STATE_BACKEND.key() + "' is not a valid state backend factory (" + backendName + ')', e); } return factory.createFromConfig(config, classLoader); } /** * Checks if an application-defined state backend is given, and if not, loads the state backend * from the configuration, from the parameter 'state.backend', as defined in {@link * StateBackendOptions#STATE_BACKEND}. If no state backend is configured, this instantiates the * default state backend (the {@link HashMapStateBackend}). * *

If an application-defined state backend is found, and the state backend is a {@link * ConfigurableStateBackend}, this methods calls {@link * ConfigurableStateBackend#configure(ReadableConfig, ClassLoader)} on the state backend. * *

Refer to {@link #loadStateBackendFromConfig(ReadableConfig, ClassLoader, Logger)} for * details on how the state backend is loaded from the configuration. * * @param jobConfig The job configuration to load the state backend from * @param clusterConfig The cluster configuration to load the state backend from * @param classLoader The class loader that should be used to load the state backend * @param logger Optionally, a logger to log actions to (may be null) * @return The instantiated state backend. * @throws DynamicCodeLoadingException Thrown if a state backend factory is configured and the * factory class was not found or the factory could not be instantiated * @throws IllegalConfigurationException May be thrown by the StateBackendFactory when creating * / configuring the state backend in the factory * @throws IOException May be thrown by the StateBackendFactory when instantiating the state * backend */ private static StateBackend loadFromApplicationOrConfigOrDefaultInternal( @Nullable StateBackend fromApplication, Configuration jobConfig, Configuration clusterConfig, ClassLoader classLoader, @Nullable Logger logger) throws IllegalConfigurationException, DynamicCodeLoadingException, IOException { checkNotNull(jobConfig, "jobConfig"); checkNotNull(clusterConfig, "clusterConfig"); checkNotNull(classLoader, "classLoader"); // Job level config can override the cluster level config. Configuration mergedConfig = new Configuration(clusterConfig); mergedConfig.addAll(jobConfig); final StateBackend backend; // In the FLINK-2.0, the state backend from application will be not supported anymore. // (1) the application defined state backend has precedence if (fromApplication != null) { // see if this is supposed to pick up additional configuration parameters if (fromApplication instanceof ConfigurableStateBackend) { // needs to pick up configuration if (logger != null) { logger.info( "Using job/cluster config to configure application-defined state backend: {}", fromApplication); } backend = ((ConfigurableStateBackend) fromApplication) // Use cluster config for backwards compatibility. .configure(clusterConfig, classLoader); } else { // keep as is! backend = fromApplication; } if (logger != null) { logger.info("Using application-defined state backend: {}", backend); } } else { // (2) check if the config defines a state backend backend = loadStateBackendFromConfig(mergedConfig, classLoader, logger); } return backend; } /** * This is the state backend loader that loads a {@link DelegatingStateBackend} wrapping the * state backend loaded from {@link * StateBackendLoader#loadFromApplicationOrConfigOrDefaultInternal} when delegation is enabled. * If delegation is not enabled, the underlying wrapped state backend is returned instead. * * @param fromApplication StateBackend defined from application * @param jobConfig The job level configuration to load the state backend from * @param clusterConfig The cluster level configuration to load the state backend from * @param classLoader The class loader that should be used to load the state backend * @param logger Optionally, a logger to log actions to (may be null) * @return The instantiated state backend. * @throws DynamicCodeLoadingException Thrown if a state backend (factory) is configured and the * (factory) class was not found or could not be instantiated * @throws IllegalConfigurationException May be thrown by the StateBackendFactory when creating * / configuring the state backend in the factory * @throws IOException May be thrown by the StateBackendFactory when instantiating the state * backend */ public static StateBackend fromApplicationOrConfigOrDefault( @Nullable StateBackend fromApplication, Configuration jobConfig, Configuration clusterConfig, ClassLoader classLoader, @Nullable Logger logger) throws IllegalConfigurationException, DynamicCodeLoadingException, IOException { StateBackend rootBackend = loadFromApplicationOrConfigOrDefaultInternal( fromApplication, jobConfig, clusterConfig, classLoader, logger); boolean enableChangeLog = jobConfig .getOptional(StateChangelogOptions.ENABLE_STATE_CHANGE_LOG) .orElse(clusterConfig.get(StateChangelogOptions.ENABLE_STATE_CHANGE_LOG)); StateBackend backend; if (enableChangeLog) { backend = wrapStateBackend(rootBackend, classLoader, CHANGELOG_STATE_BACKEND); LOG.info( "State backend loader loads {} to delegate {}", backend.getClass().getSimpleName(), rootBackend.getClass().getSimpleName()); } else { backend = rootBackend; LOG.info( "State backend loader loads the state backend as {}", backend.getClass().getSimpleName()); } return backend; } /** * Checks whether state backend uses managed memory, without having to deserialize or load the * state backend. * * @param config configuration to load the state backend from. * @param stateBackendFromApplicationUsesManagedMemory Whether the application-defined backend * uses Flink's managed memory. Empty if application has not defined a backend. * @param classLoader User code classloader. * @return Whether the state backend uses managed memory. */ public static boolean stateBackendFromApplicationOrConfigOrDefaultUseManagedMemory( Configuration config, Optional stateBackendFromApplicationUsesManagedMemory, ClassLoader classLoader) { checkNotNull(config, "config"); // (1) the application defined state backend has precedence if (stateBackendFromApplicationUsesManagedMemory.isPresent()) { return stateBackendFromApplicationUsesManagedMemory.get(); } // (2) check if the config defines a state backend try { final StateBackend fromConfig = loadStateBackendFromConfig(config, classLoader, LOG); return fromConfig.useManagedMemory(); } catch (IllegalConfigurationException | DynamicCodeLoadingException | IOException e) { LOG.warn( "Cannot decide whether state backend uses managed memory. Will reserve managed memory by default.", e); return true; } } /** * Load state backend which may wrap the original state backend for recovery. * * @param originalStateBackend StateBackend loaded from application or config. * @param classLoader User code classloader. * @param keyedStateHandles The state handles for restore. * @return Wrapped state backend for recovery. * @throws DynamicCodeLoadingException Thrown if keyed state handles of wrapped state backend * are found and the class was not found or could not be instantiated. */ public static StateBackend loadStateBackendFromKeyedStateHandles( StateBackend originalStateBackend, ClassLoader classLoader, Collection keyedStateHandles) throws DynamicCodeLoadingException { // Wrapping ChangelogStateBackend or ChangelogStateBackendHandle is not supported currently. if (!isChangelogStateBackend(originalStateBackend) && keyedStateHandles.stream() .anyMatch( stateHandle -> stateHandle instanceof ChangelogStateBackendHandle)) { return wrapStateBackend( originalStateBackend, classLoader, DEACTIVATED_CHANGELOG_STATE_BACKEND); } return originalStateBackend; } public static boolean isChangelogStateBackend(StateBackend backend) { return CHANGELOG_STATE_BACKEND.equals(backend.getClass().getName()); } private static StateBackend wrapStateBackend( StateBackend backend, ClassLoader classLoader, String className) throws DynamicCodeLoadingException { // ChangelogStateBackend resides in a separate module, load it using reflection try { Constructor constructor = Class.forName(className, false, classLoader) .asSubclass(DelegatingStateBackend.class) .getDeclaredConstructor(StateBackend.class); constructor.setAccessible(true); return constructor.newInstance(backend); } catch (ClassNotFoundException e) { throw new DynamicCodeLoadingException( "Cannot find DelegateStateBackend class: " + className, e); } catch (InstantiationException | IllegalAccessException | NoSuchMethodException | InvocationTargetException e) { throw new DynamicCodeLoadingException("Fail to initialize: " + className, e); } } // ------------------------------------------------------------------------ /** This class is not meant to be instantiated. */ private StateBackendLoader() {} }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy