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

org.apache.druid.indexing.common.actions.SegmentTransactionalInsertAction Maven / Gradle / Ivy

/*
 * 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.druid.indexing.common.actions;

import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.core.type.TypeReference;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableSet;
import org.apache.druid.indexing.common.LockGranularity;
import org.apache.druid.indexing.common.TaskLock;
import org.apache.druid.indexing.common.task.IndexTaskUtils;
import org.apache.druid.indexing.common.task.SegmentLockHelper;
import org.apache.druid.indexing.common.task.Task;
import org.apache.druid.indexing.overlord.CriticalAction;
import org.apache.druid.indexing.overlord.DataSourceMetadata;
import org.apache.druid.indexing.overlord.SegmentPublishResult;
import org.apache.druid.java.util.common.ISE;
import org.apache.druid.java.util.emitter.service.ServiceMetricEvent;
import org.apache.druid.query.DruidMetrics;
import org.apache.druid.timeline.DataSegment;
import org.joda.time.Interval;

import javax.annotation.Nullable;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;

/**
 * Insert segments into metadata storage. The segment versions must all be less than or equal to a lock held by
 * your task for the segment intervals.
 * 

* Word of warning: Very large "segments" sets can cause oversized audit log entries, which is bad because it means * that the task cannot actually complete. Callers should avoid this by avoiding inserting too many segments in the * same action. */ public class SegmentTransactionalInsertAction implements TaskAction { @Nullable private final Set segmentsToBeOverwritten; private final Set segments; @Nullable private final DataSourceMetadata startMetadata; @Nullable private final DataSourceMetadata endMetadata; public static SegmentTransactionalInsertAction overwriteAction( @Nullable Set segmentsToBeOverwritten, Set segmentsToPublish ) { return new SegmentTransactionalInsertAction(segmentsToBeOverwritten, segmentsToPublish, null, null); } public static SegmentTransactionalInsertAction appendAction( Set segments, @Nullable DataSourceMetadata startMetadata, @Nullable DataSourceMetadata endMetadata ) { return new SegmentTransactionalInsertAction(null, segments, startMetadata, endMetadata); } @JsonCreator private SegmentTransactionalInsertAction( @JsonProperty("segmentsToBeOverwritten") @Nullable Set segmentsToBeOverwritten, @JsonProperty("segments") Set segments, @JsonProperty("startMetadata") @Nullable DataSourceMetadata startMetadata, @JsonProperty("endMetadata") @Nullable DataSourceMetadata endMetadata ) { this.segmentsToBeOverwritten = segmentsToBeOverwritten; this.segments = ImmutableSet.copyOf(segments); this.startMetadata = startMetadata; this.endMetadata = endMetadata; } @JsonProperty @Nullable public Set getSegmentsToBeOverwritten() { return segmentsToBeOverwritten; } @JsonProperty public Set getSegments() { return segments; } @JsonProperty @Nullable public DataSourceMetadata getStartMetadata() { return startMetadata; } @JsonProperty @Nullable public DataSourceMetadata getEndMetadata() { return endMetadata; } @Override public TypeReference getReturnTypeReference() { return new TypeReference() { }; } /** * Performs some sanity checks and publishes the given segments. */ @Override public SegmentPublishResult perform(Task task, TaskActionToolbox toolbox) { final Set allSegments = new HashSet<>(segments); if (segmentsToBeOverwritten != null) { allSegments.addAll(segmentsToBeOverwritten); } TaskLocks.checkLockCoversSegments(task, toolbox.getTaskLockbox(), allSegments); if (segmentsToBeOverwritten != null && !segmentsToBeOverwritten.isEmpty()) { final List locks = toolbox.getTaskLockbox().findLocksForTask(task); // Let's do some sanity check that newSegments can overwrite oldSegments. if (locks.get(0).getGranularity() == LockGranularity.SEGMENT) { checkWithSegmentLock(); } } final SegmentPublishResult retVal; try { retVal = toolbox.getTaskLockbox().doInCriticalSection( task, allSegments.stream().map(DataSegment::getInterval).collect(Collectors.toList()), CriticalAction.builder() .onValidLocks( () -> toolbox.getIndexerMetadataStorageCoordinator().announceHistoricalSegments( segments, startMetadata, endMetadata ) ) .onInvalidLocks( () -> SegmentPublishResult.fail( "Invalid task locks. Maybe they are revoked by a higher priority task." + " Please check the overlord log for details." ) ) .build() ); } catch (Exception e) { throw new RuntimeException(e); } // Emit metrics final ServiceMetricEvent.Builder metricBuilder = new ServiceMetricEvent.Builder(); IndexTaskUtils.setTaskDimensions(metricBuilder, task); if (retVal.isSuccess()) { toolbox.getEmitter().emit(metricBuilder.build("segment/txn/success", 1)); } else { toolbox.getEmitter().emit(metricBuilder.build("segment/txn/failure", 1)); } // getSegments() should return an empty set if announceHistoricalSegments() failed for (DataSegment segment : retVal.getSegments()) { metricBuilder.setDimension(DruidMetrics.INTERVAL, segment.getInterval().toString()); toolbox.getEmitter().emit(metricBuilder.build("segment/added/bytes", segment.getSize())); } return retVal; } private void checkWithSegmentLock() { final Map> oldSegmentsMap = groupSegmentsByIntervalAndSort(segmentsToBeOverwritten); final Map> newSegmentsMap = groupSegmentsByIntervalAndSort(segments); oldSegmentsMap.values().forEach(SegmentLockHelper::verifyRootPartitionIsAdjacentAndAtomicUpdateGroupIsFull); newSegmentsMap.values().forEach(SegmentLockHelper::verifyRootPartitionIsAdjacentAndAtomicUpdateGroupIsFull); oldSegmentsMap.forEach((interval, oldSegmentsPerInterval) -> { final List newSegmentsPerInterval = Preconditions.checkNotNull( newSegmentsMap.get(interval), "segments of interval[%s]", interval ); // These lists are already sorted in groupSegmentsByIntervalAndSort(). final int oldStartRootPartitionId = oldSegmentsPerInterval.get(0).getStartRootPartitionId(); final int oldEndRootPartitionId = oldSegmentsPerInterval.get(oldSegmentsPerInterval.size() - 1) .getEndRootPartitionId(); final int newStartRootPartitionId = newSegmentsPerInterval.get(0).getStartRootPartitionId(); final int newEndRootPartitionId = newSegmentsPerInterval.get(newSegmentsPerInterval.size() - 1) .getEndRootPartitionId(); if (oldStartRootPartitionId != newStartRootPartitionId || oldEndRootPartitionId != newEndRootPartitionId) { throw new ISE( "Root partition range[%d, %d] of new segments doesn't match to root partition range[%d, %d] of old segments", newStartRootPartitionId, newEndRootPartitionId, oldStartRootPartitionId, oldEndRootPartitionId ); } newSegmentsPerInterval .forEach(eachNewSegment -> oldSegmentsPerInterval .forEach(eachOldSegment -> { if (eachNewSegment.getMinorVersion() <= eachOldSegment.getMinorVersion()) { throw new ISE( "New segment[%s] have a smaller minor version than old segment[%s]", eachNewSegment, eachOldSegment ); } })); }); } private static Map> groupSegmentsByIntervalAndSort(Set segments) { final Map> segmentsMap = new HashMap<>(); segments.forEach(segment -> segmentsMap.computeIfAbsent(segment.getInterval(), k -> new ArrayList<>()) .add(segment)); segmentsMap.values().forEach(segmentsPerInterval -> segmentsPerInterval.sort((s1, s2) -> { if (s1.getStartRootPartitionId() != s2.getStartRootPartitionId()) { return Integer.compare(s1.getStartRootPartitionId(), s2.getStartRootPartitionId()); } else { return Integer.compare(s1.getEndRootPartitionId(), s2.getEndRootPartitionId()); } })); return segmentsMap; } @Override public boolean isAudited() { return true; } @Override public String toString() { return "SegmentTransactionalInsertAction{" + "segmentsToBeOverwritten=" + segmentsToBeOverwritten + ", segments=" + segments + ", startMetadata=" + startMetadata + ", endMetadata=" + endMetadata + '}'; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy