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

io.gravitee.gateway.services.sync.process.repository.DefaultSyncManager Maven / Gradle / Ivy

/*
 * Copyright © 2015 The Gravitee team (http://gravitee.io)
 *
 * Licensed 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 io.gravitee.gateway.services.sync.process.repository;

import static java.util.concurrent.TimeUnit.MILLISECONDS;

import io.gravitee.common.http.MediaType;
import io.gravitee.common.service.AbstractService;
import io.gravitee.common.utils.RxHelper;
import io.gravitee.gateway.services.sync.SyncManager;
import io.gravitee.gateway.services.sync.process.distributed.DistributedSynchronizer;
import io.gravitee.gateway.services.sync.process.distributed.service.DistributedSyncService;
import io.gravitee.gateway.services.sync.process.repository.handler.SyncHandler;
import io.gravitee.node.api.Node;
import io.reactivex.rxjava3.core.Completable;
import io.reactivex.rxjava3.core.Flowable;
import io.reactivex.rxjava3.core.Single;
import io.reactivex.rxjava3.disposables.Disposable;
import io.vertx.ext.web.Route;
import io.vertx.ext.web.Router;
import java.time.Instant;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReference;
import lombok.extern.slf4j.Slf4j;

/**
 * @author Guillaume LAMIRAND (guillaume.lamirand at graviteesource.com)
 * @author GraviteeSource Team
 */
@Slf4j
public class DefaultSyncManager extends AbstractService implements SyncManager {

    private static final String PATH = "/sync";
    /**
     * Add 30s delay before and after to avoid problem with "out of sync" clocks.
     */
    public static final int TIMEFRAME_DELAY = 30000;
    public static final int RETRY_DELAY_MS = 3_000;

    private final Router router;
    private final List synchronizers;
    private final List distributedSynchronizers;
    private final int retryAttempt;
    private final DistributedSyncService distributedSyncService;
    private final int delay;
    private final TimeUnit unit;
    private final List environments;

    private final AtomicLong syncCounter = new AtomicLong(0);
    private final AtomicBoolean initialSync = new AtomicBoolean(false);
    private final AtomicLong totalSyncOnErrorCounter = new AtomicLong(0);
    private final AtomicBoolean lastSyncOnError = new AtomicBoolean(false);

    private final AtomicReference lastSyncErrorMessage = new AtomicReference<>();

    private final AtomicBoolean isClusterPrimaryNode = new AtomicBoolean(true);
    private long nextFromTime = -1;
    private Disposable refreshDisposable;
    private Route routeHandler;

    public DefaultSyncManager(
        final Router router,
        final Node node,
        final List synchronizers,
        final List distributedSynchronizers,
        final DistributedSyncService distributedSyncService,
        final int delay,
        final TimeUnit unit,
        final int retryAttempt
    ) {
        this.router = router;
        this.synchronizers = synchronizers;
        this.distributedSynchronizers = distributedSynchronizers;
        this.distributedSyncService = distributedSyncService;
        this.delay = delay;
        this.unit = unit;
        this.retryAttempt = retryAttempt;
        this.environments = new ArrayList<>((Set) node.metadata().get(Node.META_ENVIRONMENTS));
    }

    @Override
    protected void doStart() {
        log.info("Starting sync manager");
        log.debug("Associate a new HTTP handler on {}", PATH);
        SyncHandler syncHandler = new SyncHandler(this);
        routeHandler = router.get(PATH).produces(MediaType.APPLICATION_JSON).handler(syncHandler);

        // Sort synchronizer by order
        synchronizers.sort(Comparator.comparingInt(RepositorySynchronizer::order));

        if (distributedSyncService.isEnabled()) {
            distributedSyncService.validate();
            if (distributedSynchronizers != null) {
                distributedSynchronizers.sort(Comparator.comparingInt(DistributedSynchronizer::order));
            }
            isClusterPrimaryNode.set(distributedSyncService.isPrimaryNode());
        }

        // force synchronization and then schedule next ones
        synchronize()
            .andThen(
                Completable.fromRunnable(() -> {
                    log.info("Sync service has been scheduled with delay [{}{}]", delay, unit.name());
                    refreshDisposable =
                        Flowable
                            .generate(
                                () -> 0L,
                                (state, emitter) -> {
                                    emitter.onNext(state);
                                    return state + 1;
                                }
                            )
                            .delay(delay, unit)
                            .rebatchRequests(1)
                            .concatMapCompletable(interval -> synchronize())
                            .subscribe();
                })
            )
            .blockingSubscribe();
    }

    @Override
    protected void doStop() throws Exception {
        // Stop disposable
        if (refreshDisposable != null) {
            refreshDisposable.dispose();
        }
        if (routeHandler != null) {
            routeHandler.remove();
        }
        super.doStop();
    }

    private Completable synchronize() {
        return Completable
            .defer(() -> {
                log.debug("Running synchronization process...");
                return distributedSyncService.ready();
            })
            .andThen(
                Single.defer(() -> {
                    if (
                        distributedSyncService.isEnabled() &&
                        distributedSyncService.isPrimaryNode() &&
                        (nextFromTime == -1 || isClusterPrimaryNode.compareAndSet(false, true))
                    ) {
                        return distributedSyncService
                            .state()
                            .map(distributedSyncState -> {
                                log.debug("Retrieving distributed sync state");
                                nextFromTime = distributedSyncState.getFrom();
                                return distributedSyncState.getTo();
                            })
                            .switchIfEmpty(Single.just(System.currentTimeMillis()));
                    }
                    return Single.just(System.currentTimeMillis());
                })
            )
            .flatMapCompletable(nextToTime -> {
                log.debug("Synchronization #{} started at {}", syncCounter.incrementAndGet(), Instant.now());
                log.debug(
                    "Events from {} to {} would be synchronized.",
                    Instant.ofEpochMilli(nextFromTime - TIMEFRAME_DELAY),
                    Instant.ofEpochMilli(nextToTime + TIMEFRAME_DELAY)
                );
                Completable synchronizationCompletable;
                if (distributedSyncService.isEnabled() && !distributedSyncService.isPrimaryNode()) {
                    log.debug("Distributed synchronizers will be used as distributed sync is enabled, and current node is secondary.");
                    if (distributedSynchronizers != null) {
                        synchronizationCompletable =
                            Flowable
                                .fromIterable(distributedSynchronizers)
                                .concatMapCompletable(synchronizer ->
                                    synchronizer
                                        .synchronize(nextFromTime, nextToTime)
                                        .compose(upstream -> retrySynchronizer(upstream, synchronizer.getClass().getSimpleName()))
                                );
                    } else {
                        synchronizationCompletable = Completable.complete();
                    }
                } else {
                    synchronizationCompletable =
                        Flowable
                            .fromIterable(synchronizers)
                            .concatMapCompletable(synchronizer ->
                                synchronizer
                                    .synchronize(nextFromTime, nextToTime, environments)
                                    .compose(upstream -> retrySynchronizer(upstream, synchronizer.getClass().getSimpleName()))
                            );
                }
                return synchronizationCompletable
                    .andThen(distributedSyncService.storeState(nextFromTime, nextToTime))
                    .doOnComplete(() -> {
                        lastSyncOnError.set(false);
                        lastSyncErrorMessage.set(null);
                        if (nextFromTime == -1) {
                            initialSync.set(true);
                        }
                        nextFromTime = nextToTime;

                        log.debug(
                            "Synchronization #{} ended at {} (took {}ms}",
                            syncCounter.get(),
                            Instant.now().toString(),
                            System.currentTimeMillis() - nextToTime
                        );
                    });
            })
            .onErrorResumeNext(throwable ->
                Completable.fromRunnable(() -> {
                    // This condition is only to avoid printing the stacktrace on every loop
                    if (!lastSyncOnError.get()) {
                        log.error("Synchronization process has failed", throwable);
                    } else {
                        log.error("Synchronization process is still failing.");
                    }
                    lastSyncOnError.set(true);
                    lastSyncErrorMessage.set(throwable.getMessage());
                })
            );
    }

    private Completable retrySynchronizer(final Completable upstream, final String synchronizerClazz) {
        return upstream
            .doOnError(throwable -> log.warn("An error occurs while executing synchronizer {}, retrying...", synchronizerClazz, throwable))
            .compose(RxHelper.retry(retryAttempt, RETRY_DELAY_MS, MILLISECONDS))
            .doOnError(throwable -> log.error("Latest attempt of synchronizer {} has failed", synchronizerClazz, throwable));
    }

    public long nextSyncTime() {
        return nextFromTime;
    }

    public long syncCounter() {
        return syncCounter.longValue();
    }

    public boolean syncDone() {
        return initialSync.get();
    }

    public long totalSyncOnError() {
        return totalSyncOnErrorCounter.get();
    }

    public boolean lastSyncOnError() {
        return lastSyncOnError.get();
    }

    public String lastSyncErrorMessage() {
        return lastSyncErrorMessage.get();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy