tools.aqua.stars.data.av.dataclasses.Segment.kt Maven / Gradle / Ivy
/*
* Copyright 2023-2024 The STARS Project Authors
* SPDX-License-Identifier: Apache-2.0
*
* 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 tools.aqua.stars.data.av.dataclasses
import tools.aqua.stars.core.types.SegmentType
/**
* Evaluation segment.
*
* @property mainInitList [TickData] of the [Segment].
* @property simulationRunId Identifier of the simulation run.
* @property segmentSource Source identifier.
*/
data class Segment(
val mainInitList: List,
val simulationRunId: String = "",
override val segmentSource: String,
) : SegmentType {
override val tickData: List = mainInitList.onEach { it.segment = this }
override val ticks: Map = tickData.associateBy { it.currentTick }
override val primaryEntityId: Int
get() {
val firstTick = tickData.first()
val ego = firstTick.egoVehicle
checkNotNull(ego) { "There is no primary entity for tick $firstTick" }
val egoId = ego.id
check(tickData.any { it.vehicles.count { v -> v.isEgo } == 1 }) {
"There is at least one tick with multiple primary entities in segment ${this.toString(egoId)}"
}
if (tickData.any { it.egoVehicle?.id != egoId })
error("The ego id changes in Segment ${this.toString(egoId)}")
return egoId
}
/** Cache for all vehicle IDs. */
private val vehicleIdsCache = mutableListOf()
/** All vehicle IDs of the segment. */
val vehicleIds: List
get() {
if (vehicleIdsCache.isEmpty()) {
vehicleIdsCache.addAll(
tickData
.flatMap { tickData -> tickData.entities.filterIsInstance().map { it.id } }
.distinct())
}
return vehicleIdsCache
}
/** Cache for all pedestrian IDs. */
private val pedestrianIdsCache = mutableListOf()
/** All pedestrian IDs of the segment. */
@Suppress("unused")
val pedestrianIds: List
get() {
if (pedestrianIdsCache.isEmpty()) {
pedestrianIdsCache.addAll(
tickData
.flatMap { tickData ->
tickData.entities.filterIsInstance().map { it.id }
}
.distinct())
}
return pedestrianIdsCache
}
/**
* Converts segment to String representation including [tickData] range and given [egoId].
*
* @param egoId Identifier of the ego vehicle to be included.
*/
fun toString(egoId: Int): String =
"Segment[(${tickData.first().currentTick}..${tickData.last().currentTick})] from $simulationRunId " +
"with ego $egoId"
override fun toString(): String = toString(this.primaryEntityId)
override fun equals(other: Any?): Boolean {
if (other is Segment) {
return simulationRunId == other.simulationRunId &&
segmentSource == other.segmentSource &&
primaryEntityId == other.primaryEntityId &&
tickData == other.tickData
}
return super.equals(other)
}
override fun hashCode(): Int = this.toString().hashCode()
}