com.azure.cosmos.spark.CosmosPartitionPlanner.scala Maven / Gradle / Ivy
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.
package com.azure.cosmos.spark
import com.azure.cosmos.implementation.SparkBridgeImplementationInternal.extractContinuationTokensFromChangeFeedStateJson
import com.azure.cosmos.implementation.{SparkBridgeImplementationInternal, Strings}
import com.azure.cosmos.models.FeedRange
import com.azure.cosmos.spark.CosmosPredicates.{assertNotNull, assertNotNullOrEmpty, assertOnSparkDriver, requireNotNull}
import com.azure.cosmos.spark.CosmosTableSchemaInferrer.LsnAttributeName
import com.azure.cosmos.spark.diagnostics.BasicLoggingTrait
import com.azure.cosmos.{CosmosAsyncContainer, SparkBridgeInternal}
import com.fasterxml.jackson.databind.node.ObjectNode
import org.apache.spark.broadcast.Broadcast
import org.apache.spark.sql.connector.read.streaming.{ReadAllAvailable, ReadLimit, ReadMaxFiles, ReadMaxRows}
import reactor.core.scala.publisher.SMono.PimpJMono
import reactor.core.scala.publisher.{SFlux, SMono}
import java.time.Duration
import java.util
import java.util.concurrent.atomic.AtomicLong
import java.util.concurrent.{ConcurrentHashMap, ConcurrentMap}
import scala.collection.mutable
import scala.collection.mutable.ArrayBuffer
// scalastyle:off underscore.import
import scala.collection.JavaConverters._
// scalastyle:on underscore.import
private object CosmosPartitionPlanner extends BasicLoggingTrait {
val DefaultPartitionSizeInMB: Int = 5 * 1024 // 10 GB
def createInputPartitions
(
cosmosPartitioningConfig: CosmosPartitioningConfig,
container: CosmosAsyncContainer,
partitionMetadata: Array[PartitionMetadata],
defaultMinimalPartitionCount: Int,
defaultMaxPartitionSizeInMB: Int,
readLimit: ReadLimit,
isChangeFeed: Boolean
): Array[CosmosInputPartition] = {
TransientErrorsRetryPolicy.executeWithRetry(() =>
createInputPartitionsImpl(
cosmosPartitioningConfig,
container,
partitionMetadata,
defaultMinimalPartitionCount,
defaultMaxPartitionSizeInMB,
readLimit,
isChangeFeed))
}
private[this] def createInputPartitionsImpl
(
cosmosPartitioningConfig: CosmosPartitioningConfig,
container: CosmosAsyncContainer,
partitionMetadata: Array[PartitionMetadata],
defaultMinimalPartitionCount: Int,
defaultMaxPartitionSizeInMB: Int,
readLimit: ReadLimit,
isChangeFeed: Boolean
): Array[CosmosInputPartition] = {
assertOnSparkDriver()
//scalastyle:off multiple.string.literals
requireNotNull(cosmosPartitioningConfig, "cosmosPartitioningConfig")
require(defaultMaxPartitionSizeInMB >= 64, "Argument 'defaultMaxPartitionSizeInMB' must at least be 64")
require(defaultMinimalPartitionCount >= 1, "Argument 'defaultMinimalPartitionCount' must at least be 1")
//scalastyle:on multiple.string.literals
val inputPartitions = if (!isChangeFeed && cosmosPartitioningConfig.partitioningStrategy == PartitioningStrategies.Restrictive) {
partitionMetadata.map(metadata => CosmosInputPartition(metadata.feedRange, None))
} else {
val planningInfo = this.getPartitionPlanningInfo(partitionMetadata, readLimit)
cosmosPartitioningConfig.partitioningStrategy match {
case PartitioningStrategies.Restrictive =>
applyRestrictiveStrategy(planningInfo)
case PartitioningStrategies.Default =>
applyRestrictiveStrategy(planningInfo)
case PartitioningStrategies.Custom =>
applyCustomStrategy(
container,
planningInfo,
cosmosPartitioningConfig.targetedPartitionCount.get)
case PartitioningStrategies.Aggressive =>
applyStorageAlignedStrategy(
container,
planningInfo,
1 / defaultMaxPartitionSizeInMB.toDouble,
defaultMinimalPartitionCount
)
}
}
cosmosPartitioningConfig.feedRangeFiler match {
case Some(epkRangesInScope) => {
val overlappingPartitions = inputPartitions
.filter(inputPartition => {
epkRangesInScope.exists(epk => SparkBridgeImplementationInternal.doRangesOverlap(
epk,
inputPartition.feedRange))
})
val returnValue = new ArrayBuffer[CosmosInputPartition]()
overlappingPartitions.foreach(inputPartition => {
val overlappingEpkRanges = epkRangesInScope.filter(epk => SparkBridgeImplementationInternal.doRangesOverlap(
epk,
inputPartition.feedRange))
overlappingEpkRanges.foreach(overlappingRange => {
val finalRange = new NormalizedRange(
if (overlappingRange.min.compareTo(inputPartition.feedRange.min) > 0) {
overlappingRange.min
} else {
inputPartition.feedRange.min
},
if (overlappingRange.max.compareTo(inputPartition.feedRange.max) < 0) {
overlappingRange.max
} else {
inputPartition.feedRange.max
}
)
returnValue += new CosmosInputPartition(
finalRange, inputPartition.endLsn, inputPartition.continuationState)
})
})
returnValue.toArray
}
case None => inputPartitions
}
}
private[this] def getContinuationTokenLsnOfFirstItem(items: Iterable[ObjectNode]): Option[String] = {
getLsnOfFirstItem(items) match {
case Some(firstLsn) =>
Some(SparkBridgeImplementationInternal.toContinuationToken(firstLsn))
case None => None
}
}
private[spark] def getLsnOfFirstItem(items: Iterable[ObjectNode]): Option[Long] = {
items
.collectFirst({
case item: ObjectNode if item != null =>
val lsnNode = item.get(LsnAttributeName)
if (lsnNode != null && lsnNode.isNumber) {
Some(
// when grabbing the LSN from the item we need to use the item's LSN -1
// to ensure we would retrieve this item again
lsnNode.asLong() - 1)
} else {
None
}
})
.flatten
}
def createInitialOffset
(
container: CosmosAsyncContainer,
changeFeedConfig: CosmosChangeFeedConfig,
partitioningConfig: CosmosPartitioningConfig,
streamId: Option[String]
): String = {
TransientErrorsRetryPolicy.executeWithRetry(() =>
createInitialOffsetImpl(container, changeFeedConfig, partitioningConfig, streamId)
)
}
// scalastyle:off method.length
private[this] def createInitialOffsetImpl
(
container: CosmosAsyncContainer,
changeFeedConfig: CosmosChangeFeedConfig,
partitioningConfig: CosmosPartitioningConfig,
streamId: Option[String]
): String = {
assertOnSparkDriver()
val lastContinuationTokens: ConcurrentMap[FeedRange, String] = new ConcurrentHashMap[FeedRange, String]()
ContainerFeedRangesCache
.getFeedRanges(container)
.map(feedRangeList =>
partitioningConfig.feedRangeFiler match {
case Some(epkRangesInScope) => feedRangeList
.filter(feedRange => {
epkRangesInScope.exists(epk => SparkBridgeImplementationInternal.doRangesOverlap(
epk,
SparkBridgeImplementationInternal.toNormalizedRange(feedRange)))
})
case None => feedRangeList
})
.flatMapMany(feedRanges => SFlux.fromIterable(feedRanges))
.flatMap(feedRange => {
val requestOptions = changeFeedConfig.toRequestOptions(feedRange)
requestOptions.setMaxItemCount(1)
requestOptions.setMaxPrefetchPageCount(1)
requestOptions.setQuotaInfoEnabled(true)
container
.queryChangeFeed(requestOptions, classOf[ObjectNode])
.handle(r => {
val lsnFromItems = getContinuationTokenLsnOfFirstItem(r.getElements.asScala)
val continuation = if (lsnFromItems.isDefined) {
SparkBridgeImplementationInternal
.overrideLsnInChangeFeedContinuation(r.getContinuationToken,lsnFromItems.get)
} else {
r.getContinuationToken
}
if (!Strings.isNullOrWhiteSpace(continuation)) {
if (lastContinuationTokens.putIfAbsent(feedRange, continuation) == null && isDebugLogEnabled) {
val stateJson = SparkBridgeImplementationInternal.changeFeedContinuationToJson(continuation)
val range = SparkBridgeImplementationInternal.toNormalizedRange(feedRange)
logDebug(s"FeedRange '${range.min}-${range.max}': Set effective continuation '$stateJson")
}
}
})
.take(1)
.collectList()
.asScala
})
.asJava()
.collectList()
.block()
val offsetJsonBase64 = SparkBridgeImplementationInternal
.mergeChangeFeedContinuations(lastContinuationTokens.values().asScala)
//if (isDebugLogEnabled) {
val offsetJson = SparkBridgeImplementationInternal.changeFeedContinuationToJson(offsetJsonBase64)
// scala style rule flaky - even complaining on partial log messages
// scalastyle:off multiple.string.literals
logInfo(s"Initial offset of stream ${streamId.getOrElse("null")}: '$offsetJson'.")
// scalastyle:on multiple.string.literals
//}
offsetJsonBase64
}
// scalastyle:on method.length
// scalastyle:off parameter.number
def getLatestOffset
(
userConfig: Map[String, String],
startOffset: ChangeFeedOffset,
readLimit: ReadLimit,
maxStaleness: Duration,
clientConfiguration: CosmosClientConfiguration,
cosmosClientStateHandles: Broadcast[CosmosClientMetadataCachesSnapshots],
containerConfig: CosmosContainerConfig,
partitioningConfig: CosmosPartitioningConfig,
defaultParallelism: Int,
container: CosmosAsyncContainer
): ChangeFeedOffset = {
TransientErrorsRetryPolicy.executeWithRetry(() =>
getLatestOffsetImpl(
userConfig,
startOffset,
readLimit,
maxStaleness,
clientConfiguration,
cosmosClientStateHandles,
containerConfig,
partitioningConfig,
defaultParallelism,
container))
}
// scalastyle:on parameter.number
// scalastyle:off method.length
// scalastyle:off parameter.number
// Based on a start offset, calculate which is the next end offset
private[this] def getLatestOffsetImpl
(
userConfig: Map[String, String],
startOffset: ChangeFeedOffset,
readLimit: ReadLimit,
maxStaleness: Duration,
clientConfiguration: CosmosClientConfiguration,
cosmosClientStateHandles: Broadcast[CosmosClientMetadataCachesSnapshots],
containerConfig: CosmosContainerConfig,
partitioningConfig: CosmosPartitioningConfig,
defaultParallelism: Int,
container: CosmosAsyncContainer
): ChangeFeedOffset = {
assertOnSparkDriver()
assertNotNull(startOffset, "startOffset")
val latestPartitionMetadata = CosmosPartitionPlanner.getFilteredPartitionMetadata(
userConfig,
clientConfiguration,
Some(cosmosClientStateHandles),
containerConfig,
partitioningConfig,
true,
Some(maxStaleness)
)
val defaultMaxPartitionSizeInMB = DefaultPartitionSizeInMB
val defaultMinPartitionCount = 1 + (2 * defaultParallelism)
val orderedMetadataWithStartLsn = this.getOrderedPartitionMetadataWithStartLsn(
startOffset.changeFeedState,
latestPartitionMetadata)
val inputPartitions: Array[CosmosInputPartition] = CosmosPartitionPlanner.createInputPartitions(
partitioningConfig,
container,
orderedMetadataWithStartLsn,
defaultMinPartitionCount,
defaultMaxPartitionSizeInMB,
readLimit,
true
)
if (isDebugLogEnabled) {
val endOffsetDebug = new StringBuilder("EndOffSet using EndLsn: ")
for (range <- inputPartitions) {
endOffsetDebug ++= s"${range.feedRange.min}-${range.feedRange.max}: ${range.endLsn},"
}
logDebug(endOffsetDebug.toString)
}
val changeFeedStateJson = SparkBridgeImplementationInternal
.createChangeFeedStateJson(
startOffset.changeFeedState,
inputPartitions.map(m => (m.feedRange, m.endLsn.get)))
ChangeFeedOffset(changeFeedStateJson, Some(inputPartitions))
}
// scalastyle:on method.length
// scalastyle:on parameter.number
private[this] def getOrderedPartitionMetadataWithStartLsn
(
stateJson: String,
latestPartitionMetadata: Array[PartitionMetadata]
): Array[PartitionMetadata] = {
assert(!Strings.isNullOrWhiteSpace(stateJson), s"Argument 'stateJson' must not be null or empty.")
val orderedStartTokens = extractContinuationTokensFromChangeFeedStateJson(stateJson)
val orderedLatestTokens = latestPartitionMetadata
.sortBy(metadata => metadata.feedRange.min)
mergeStartAndLatestTokens(orderedStartTokens, orderedLatestTokens)
}
// scalastyle:off method.length
private[this] def mergeStartAndLatestTokens
(
startTokens: Array[(NormalizedRange, Long)],
latestTokens: Array[PartitionMetadata]
): Array[PartitionMetadata] = {
val orderedBoundaries = mutable.SortedSet[String]()
startTokens.foreach(token => {
orderedBoundaries += token._1.min
orderedBoundaries += token._1.max
})
latestTokens.foreach(metadata => {
orderedBoundaries += metadata.feedRange.min
orderedBoundaries += metadata.feedRange.max
})
var startTokensIndex = 0
var latestTokensIndex = 0
var orderedRangesIndex = 0
val orderedRanges: Array[NormalizedRange] = new Array[NormalizedRange](orderedBoundaries.size - 1)
orderedBoundaries
.tail
.foldLeft(None: Option[NormalizedRange])((previous: Option[NormalizedRange], current: String) => {
val range = previous match {
case Some(previousRange) =>
NormalizedRange(previousRange.max, current)
case None =>
NormalizedRange(orderedBoundaries.head, current)
}
orderedRanges(orderedRangesIndex) = range
orderedRangesIndex += 1
Some(range)
})
val result = new ArrayBuffer[PartitionMetadata]
orderedRanges
.foreach(range => {
val initialStartTokensIndex = startTokensIndex
val initialLatestTokensIndex = latestTokensIndex
while (startTokensIndex < startTokens.length &&
!SparkBridgeImplementationInternal.doRangesOverlap(range, startTokens(startTokensIndex)._1)) {
startTokensIndex += 1
}
while (startTokensIndex < startTokens.length &&
latestTokensIndex < latestTokens.length &&
!SparkBridgeImplementationInternal.doRangesOverlap(range, latestTokens(latestTokensIndex).feedRange)) {
latestTokensIndex += 1
}
if (startTokensIndex < startTokens.length &&
latestTokensIndex < latestTokens.length) {
val startLsn: Long = startTokens(startTokensIndex)._2
val newToken = latestTokens(latestTokensIndex).cloneForSubRange(range, startLsn)
result.append(newToken)
} else {
startTokensIndex = initialStartTokensIndex
latestTokensIndex = initialLatestTokensIndex
}
})
assert(result.size > 0)
result.toArray
}
// scalastyle:on method.length
private[this] def applyRestrictiveStrategy
(
partitionPlanningInfo: Array[PartitionPlanningInfo]
): Array[CosmosInputPartition] = {
partitionPlanningInfo.map(info =>
CosmosInputPartition(
info.feedRange,
info.endLsn))
}
private[this] def applyRestrictiveQueryStrategy
(
ranges: Array[NormalizedRange]
): Array[CosmosInputPartition] = {
ranges.map(range => CosmosInputPartition(range, None))
}
private[this] def applyStorageAlignedStrategy(
container: CosmosAsyncContainer,
planningInfo: Array[PartitionPlanningInfo],
splitCountMultiplier: Double,
defaultMinPartitionCount: Int
): Array[CosmosInputPartition] = {
assertNotNullOrEmpty(planningInfo, "planningInfo")
val totalScaleFactor = planningInfo.map(pi => pi.scaleFactor).sum
val effectiveSplitCountMultiplier =
if (splitCountMultiplier == 0 || totalScaleFactor == 0) {
1
} else {
splitCountMultiplier * math.max(
1,
defaultMinPartitionCount / (splitCountMultiplier * totalScaleFactor))
}
val inputPartitions =
new util.ArrayList[CosmosInputPartition](
(2 * totalScaleFactor * effectiveSplitCountMultiplier).toInt)
planningInfo.foreach(info => {
val numberOfSparkPartitions = math.min(
Int.MaxValue,
math.max(
1,
math.ceil(info.scaleFactor * effectiveSplitCountMultiplier).toInt))
SparkBridgeInternal
.trySplitFeedRange(container, info.feedRange, numberOfSparkPartitions)
.foreach(feedRange =>
inputPartitions.add(CosmosInputPartition(feedRange, info.endLsn)))
})
inputPartitions.asScala.toArray
}
private[this] def applyCustomStrategy(
container: CosmosAsyncContainer,
planningInfo: Array[PartitionPlanningInfo],
targetPartitionCount: Int
): Array[CosmosInputPartition] = {
val customPartitioningFactor = planningInfo
.map(pi => pi.scaleFactor)
.sum / targetPartitionCount
applyStorageAlignedStrategy(
container,
planningInfo,
customPartitioningFactor,
targetPartitionCount
)
}
def getPartitionPlanningInfo
(
partitionMetadata: Array[PartitionMetadata],
readLimit: ReadLimit
): Array[PartitionPlanningInfo] = {
assertOnSparkDriver()
assertNotNullOrEmpty(partitionMetadata, "partitionMetadata")
val partitionPlanningInfo =
new Array[PartitionPlanningInfo](partitionMetadata.length)
var index = 0
calculateEndLsn(partitionMetadata, readLimit)
.foreach(m => {
val storageSizeInMB: Double = m.totalDocumentSizeInKB / 1024.toDouble
val progressWeightFactor: Double = getChangeFeedProgressFactor(storageSizeInMB, m)
val scaleFactor = if (storageSizeInMB == 0) {
1
} else {
progressWeightFactor * storageSizeInMB
}
val planningInfo = PartitionPlanningInfo(
m.feedRange,
storageSizeInMB,
progressWeightFactor,
scaleFactor,
m.endLsn
)
partitionPlanningInfo(index) = planningInfo
index += 1
})
partitionPlanningInfo
}
private[spark] def calculateEndLsn
(
metadata: Array[PartitionMetadata],
readLimit: ReadLimit
): Array[PartitionMetadata] = {
val totalWeightedLsnGap = new AtomicLong(0)
metadata.foreach(m => {
totalWeightedLsnGap.addAndGet(m.getWeightedLsnGap)
})
metadata
// Update endLsn - which depends on read limit
.map(metadata => {
val endLsn = readLimit match {
case _: ReadAllAvailable => metadata.getAndValidateLatestLsn
case maxRowsLimit: ReadMaxRows =>
if (totalWeightedLsnGap.get <= maxRowsLimit.maxRows) {
val effectiveLatestLsn = metadata.getAndValidateLatestLsn
if (isDebugLogEnabled) {
val calculateDebugLine = s"calculateEndLsn (feedRange: ${metadata.feedRange}) - avg. Docs " +
s"per LSN: ${metadata.getAvgItemsPerLsn} documentCount ${metadata.documentCount} firstLsn " +
s"${metadata.firstLsn} latestLsn ${metadata.latestLsn} startLsn ${metadata.startLsn} weightedGap " +
s"${metadata.getWeightedLsnGap} effectiveEndLsn $effectiveLatestLsn maxRows ${maxRowsLimit.maxRows}"
logDebug(calculateDebugLine)
}
effectiveLatestLsn
} else {
// the weight of this feedRange compared to other feedRanges
val feedRangeWeightFactor = metadata.getWeightedLsnGap.toDouble / totalWeightedLsnGap.get
val allowedRate = (feedRangeWeightFactor * maxRowsLimit.maxRows() / metadata.getAvgItemsPerLsn)
.toLong
.max(1)
val effectiveLatestLsn = metadata.getAndValidateLatestLsn
val effectiveEndLsn = math.min(
effectiveLatestLsn,
metadata.startLsn + allowedRate)
if (isDebugLogEnabled) {
val calculateDebugLine = s"calculateEndLsn (feedRange: ${metadata.feedRange}) - avg. Docs/LSN: " +
s"${metadata.getAvgItemsPerLsn} feedRangeWeightFactor $feedRangeWeightFactor documentCount " +
s"${metadata.documentCount} firstLsn ${metadata.firstLsn} latestLsn ${metadata.latestLsn} " +
s"effectiveLatestLsn $effectiveLatestLsn startLsn " +
s"${metadata.startLsn} allowedRate $allowedRate weightedGap ${metadata.getWeightedLsnGap} " +
s"effectiveEndLsn $effectiveEndLsn maxRows $maxRowsLimit.maxRows"
logDebug(calculateDebugLine)
}
effectiveEndLsn
}
case _: ReadMaxFiles => throw new IllegalStateException("ReadLimitMaxFiles not supported by this source.")
}
metadata.withEndLsn(endLsn)
})
}
private[this] def getChangeFeedProgressFactor(
storageSizeInMB: Double,
metadata: PartitionMetadata): Double = {
val effectiveLatestLsn = metadata.getAndValidateLatestLsn
val effectiveEndLsn = metadata.endLsn.getOrElse(effectiveLatestLsn)
if (metadata.startLsn <= 0 || storageSizeInMB == 0) {
// No progress has been made so far - use one Spark partition per GB
1
} else if (effectiveEndLsn <= metadata.startLsn) {
// If progress has caught up with estimation already make sure we only use one Spark partition
// for the physical partition in Cosmos
1 / storageSizeInMB
} else {
// Use weight factor based on progress. This estimate assumes equal distribution of storage
// size per LSN - which is a "good enough" simplification
(effectiveEndLsn - metadata.startLsn) / effectiveLatestLsn.toDouble
}
}
private[this] def getFeedRanges
(
userConfig: Map[String, String],
cosmosClientConfig: CosmosClientConfiguration,
cosmosClientStateHandles: Option[Broadcast[CosmosClientMetadataCachesSnapshots]],
cosmosContainerConfig: CosmosContainerConfig
) = {
assertNotNull(cosmosClientConfig, "cosmosClientConfig")
assertNotNull(cosmosContainerConfig, "cosmosContainerConfig")
val cosmosClientMetadataCache =
cosmosClientStateHandles match {
case None => None
case _ => Some(cosmosClientStateHandles.get.value.cosmosClientMetadataCaches)
}
val calledFrom = s"CosmosPartitionPlanner.getFeedRanges(database ${cosmosContainerConfig.database}, container ${cosmosContainerConfig.container})"
Loan(
List[Option[CosmosClientCacheItem]](
Some(CosmosClientCache.apply(
cosmosClientConfig,
cosmosClientMetadataCache,
calledFrom)),
ThroughputControlHelper.getThroughputControlClientCacheItem(
userConfig,
calledFrom,
cosmosClientStateHandles,
cosmosClientConfig.sparkEnvironmentInfo)
))
.to(clientCacheItems => {
val container =
ThroughputControlHelper.getContainer(
userConfig,
cosmosContainerConfig,
clientCacheItems(0).get,
clientCacheItems(1))
ContainerFeedRangesCache
.getFeedRanges(container)
.map(feedRanges => feedRanges
.map(feedRange => SparkBridgeImplementationInternal.toNormalizedRange(feedRange))
.toArray)
})
}
def getFilteredPartitionMetadata(
userConfig: Map[String, String],
cosmosClientConfig: CosmosClientConfiguration,
cosmosClientStateHandles: Option[Broadcast[CosmosClientMetadataCachesSnapshots]],
cosmosContainerConfig: CosmosContainerConfig,
partitionConfig: CosmosPartitioningConfig,
isChangeFeed: Boolean,
maxStaleness: Option[Duration] = None
): Array[PartitionMetadata] = {
TransientErrorsRetryPolicy.executeWithRetry(() =>
getPartitionMetadataImpl(
userConfig,
cosmosClientConfig,
cosmosClientStateHandles,
cosmosContainerConfig,
partitionConfig,
isChangeFeed,
maxStaleness))
}
private[this] def getPartitionMetadataImpl(
userConfig: Map[String, String],
cosmosClientConfig: CosmosClientConfiguration,
cosmosClientStateHandles: Option[Broadcast[CosmosClientMetadataCachesSnapshots]],
cosmosContainerConfig: CosmosContainerConfig,
partitionConfig: CosmosPartitioningConfig,
isChangeFeed: Boolean,
maxStaleness: Option[Duration] = None
): Array[PartitionMetadata] = {
assertOnSparkDriver()
this
.getFeedRanges(
userConfig,
cosmosClientConfig,
cosmosClientStateHandles,
cosmosContainerConfig)
.map(feedRangeList =>
partitionConfig.feedRangeFiler match {
case Some(epkRangesInScope) => feedRangeList
.filter(feedRange => {
epkRangesInScope.exists(epk => SparkBridgeImplementationInternal.doRangesOverlap(
epk,
feedRange))
})
case None => feedRangeList
})
.flatMap(feedRanges => {
SFlux
.fromArray(feedRanges)
.flatMap(
normalizedRange =>
// for change feed we always need min and max LSN - even with `Restrictive` partitioning strategy
// for query we need metadata for any partitioning strategy but `Restrictive`
if (isChangeFeed || partitionConfig.partitioningStrategy != PartitioningStrategies.Restrictive) {
PartitionMetadataCache.apply(
userConfig,
cosmosClientConfig,
cosmosClientStateHandles,
cosmosContainerConfig,
normalizedRange,
maxStaleness
)
} else {
SMono.just(new PartitionMetadata(
userConfig,
cosmosClientConfig,
cosmosClientStateHandles,
cosmosContainerConfig,
normalizedRange,
0,
0,
None,
0,
0,
None,
new AtomicLong(0),
new AtomicLong(0)
))
})
.collectSeq()
})
.block()
.toArray
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy