From 1149548d9a1fe797371d90af43f0cca3c55814e1 Mon Sep 17 00:00:00 2001 From: vmarc Date: Thu, 20 Jun 2024 10:45:38 +0200 Subject: [PATCH] #368 setup route analysis tool for development --- .../analysis/AnalysisStartConfiguration.scala | 38 ++++++++++- .../support/MigrateRouteRelationsTool.scala | 2 - .../next/support/RouteAnalysisTool.scala | 64 +++++++++++++++++++ .../kpn/core/tools/poi/PoiAnalyzerTool.scala | 2 +- .../location/LocationAnalyzerImpl.scala | 4 +- .../location/LocationStoreReader.scala | 11 +++- .../server/config/ServerConfiguration.scala | 5 ++ 7 files changed, 118 insertions(+), 8 deletions(-) create mode 100644 server/src/main/scala/kpn/core/tools/next/support/RouteAnalysisTool.scala diff --git a/server/src/main/scala/kpn/core/tools/analysis/AnalysisStartConfiguration.scala b/server/src/main/scala/kpn/core/tools/analysis/AnalysisStartConfiguration.scala index 7a15f1bd4..dfbe86dd2 100644 --- a/server/src/main/scala/kpn/core/tools/analysis/AnalysisStartConfiguration.scala +++ b/server/src/main/scala/kpn/core/tools/analysis/AnalysisStartConfiguration.scala @@ -4,6 +4,7 @@ import kpn.api.common.ReplicationId import kpn.api.common.changes.ChangeSet import kpn.api.custom.Timestamp import kpn.core.overpass.OverpassQueryExecutorRemoteImpl +import kpn.core.tools.config.Dirs import kpn.database.util.Mongo import kpn.server.analyzer.engine.analysis.location.LocationAnalyzerImpl import kpn.server.analyzer.engine.analysis.location.RouteLocatorImpl @@ -40,6 +41,11 @@ import kpn.server.analyzer.engine.tile.OldLinesTileCalculatorImpl import kpn.server.analyzer.engine.tile.OldNodeTileCalculatorImpl import kpn.server.analyzer.engine.tile.OldTileCalculatorImpl import kpn.server.analyzer.engine.tile.RouteTileCalculatorImpl +import kpn.server.analyzer.engine.tile.TileFileBuilderImpl +import kpn.server.analyzer.engine.tiles.TileAnalyzerImpl +import kpn.server.analyzer.engine.tiles.TileDataNodeBuilderImpl +import kpn.server.analyzer.engine.tiles.TileFileRepositoryImpl +import kpn.server.analyzer.engine.tiles.TilesBuilder import kpn.server.overpass.OverpassRepository import kpn.server.overpass.OverpassRepositoryImpl import kpn.server.repository.AnalysisRepository @@ -67,7 +73,7 @@ class AnalysisStartConfiguration(options: AnalysisStartToolOptions) { val nodeRepository: NodeRepository = new NodeRepositoryImpl(database) val analysisRepository: AnalysisRepository = new AnalysisRepositoryImpl(database) - private val locationAnalyzer = new LocationAnalyzerImpl(true) + private val locationAnalyzer = new LocationAnalyzerImpl(true, true) private val tileCalculator = new OldTileCalculatorImpl() @@ -155,4 +161,34 @@ class AnalysisStartConfiguration(options: AnalysisStartToolOptions) { ), ElementIds() ) + + private val tileDir = Dirs.root.getAbsolutePath + "/tiles" + private val tileDataNodeBuilder = new TileDataNodeBuilderImpl() + + val tileAnalyzer = { + val nodeRepository = new NodeRepositoryImpl(database) + val routeRepository = new RouteRepositoryImpl(database) + new TileAnalyzerImpl( + nodeRepository, + routeRepository, + tileDataNodeBuilder + ) + } + + val tilesBuilder: TilesBuilder = { + val tileCalculator = new OldTileCalculatorImpl() + val bitmapTileFileRepository = new TileFileRepositoryImpl(tileDir, "png") + val vectorTileFileRepository = new TileFileRepositoryImpl(tileDir, "mvt") + val tileFileBuilder = new TileFileBuilderImpl(bitmapTileFileRepository, vectorTileFileRepository) + val nodeTileCalculator = new OldNodeTileCalculatorImpl(tileCalculator) + val linesTileCalculator = new OldLinesTileCalculatorImpl(tileCalculator) + val routeTileCalculator = new RouteTileCalculatorImpl(linesTileCalculator) + new TilesBuilder( + bitmapTileFileRepository, + vectorTileFileRepository, + tileFileBuilder, + nodeTileCalculator, + routeTileCalculator + ) + } } diff --git a/server/src/main/scala/kpn/core/tools/next/support/MigrateRouteRelationsTool.scala b/server/src/main/scala/kpn/core/tools/next/support/MigrateRouteRelationsTool.scala index 5a311afda..7387cf4c7 100644 --- a/server/src/main/scala/kpn/core/tools/next/support/MigrateRouteRelationsTool.scala +++ b/server/src/main/scala/kpn/core/tools/next/support/MigrateRouteRelationsTool.scala @@ -6,8 +6,6 @@ import kpn.api.common.data.RelationMember import kpn.api.common.data.Way import kpn.api.common.data.WayMember import kpn.api.custom.Relation -import kpn.api.custom.Tags -import kpn.api.custom.Timestamp import kpn.core.tools.next.database.NextDatabase import kpn.core.tools.next.database.NextDatabaseImpl import kpn.core.tools.next.domain.NextRouteRelation diff --git a/server/src/main/scala/kpn/core/tools/next/support/RouteAnalysisTool.scala b/server/src/main/scala/kpn/core/tools/next/support/RouteAnalysisTool.scala new file mode 100644 index 000000000..9c06f9038 --- /dev/null +++ b/server/src/main/scala/kpn/core/tools/next/support/RouteAnalysisTool.scala @@ -0,0 +1,64 @@ +package kpn.core.tools.next.support + +import kpn.api.common.tiles.ZoomLevel +import kpn.api.custom.NetworkType +import kpn.api.custom.Relation +import kpn.core.tools.analysis.AnalysisStartConfiguration +import kpn.core.tools.analysis.AnalysisStartToolOptions +import kpn.core.util.Log + +object RouteAnalysisTool { + def main(args: Array[String]): Unit = { + val configuration = new AnalysisStartConfiguration(AnalysisStartToolOptions("kpn-next")) + new RouteAnalysisTool(configuration).analyze() + } +} + +class RouteAnalysisTool(config: AnalysisStartConfiguration) { + private val log = Log(classOf[RouteAnalysisTool]) + + def analyze(): Unit = { + log.info("Start") + analyzeRoutes(Seq(13844575L)) + buildTiles() + log.info(s"Done") + } + + private def analyzeRoutes(routeIds: Seq[Long]): Unit = { + val relations = config.overpassRepository.fullRelations(config.timestamp, routeIds) + relations.foreach { relation => + analyzeRoute(relation) + } + } + + private def analyzeRoute(relation: Relation): Unit = { + Log.context(s"route=${relation.id}") { + try { + config.masterRouteAnalyzer.analyze(relation) match { + case None => + case Some(routeAnalysis) => + config.routeRepository.save(routeAnalysis.route) + // TODO saveRouteChange(routeAnalysis) + } + } catch { + case e: Exception => + log.error(s"Error processing route ${relation.id}", e) + throw e + } + } + } + + private def buildTiles(): Unit = { + /* NetworkType.all */ Seq(NetworkType.hiking).foreach { networkType => + Log.context(networkType.name) { + log.info("Start tile analysis") + val tileAnalysis = config.tileAnalyzer.analysis(networkType) + (ZoomLevel.minZoom to ZoomLevel.vectorTileMaxZoom).foreach { z => + Log.context(s"$z") { + config.tilesBuilder.build(z, tileAnalysis) + } + } + } + } + } +} diff --git a/server/src/main/scala/kpn/core/tools/poi/PoiAnalyzerTool.scala b/server/src/main/scala/kpn/core/tools/poi/PoiAnalyzerTool.scala index 3b677e000..b706b2820 100644 --- a/server/src/main/scala/kpn/core/tools/poi/PoiAnalyzerTool.scala +++ b/server/src/main/scala/kpn/core/tools/poi/PoiAnalyzerTool.scala @@ -35,7 +35,7 @@ object PoiAnalyzerTool { new PoiLoaderImpl(overpassQueryExecutor) } val poiRepository = new PoiRepositoryImpl(poiDatabase) - val locationAnalyzer = new LocationAnalyzerImpl(true) + val locationAnalyzer = new LocationAnalyzerImpl(true, false) val poiScopeAnalyzer = new PoiScopeAnalyzerImpl(locationAnalyzer) val tileCalculator: OldTileCalculator = new OldTileCalculatorImpl() val masterPoiAnalyzer = new MasterPoiAnalyzerImpl() diff --git a/server/src/main/scala/kpn/server/analyzer/engine/analysis/location/LocationAnalyzerImpl.scala b/server/src/main/scala/kpn/server/analyzer/engine/analysis/location/LocationAnalyzerImpl.scala index d6081a0ee..b06b1e88c 100644 --- a/server/src/main/scala/kpn/server/analyzer/engine/analysis/location/LocationAnalyzerImpl.scala +++ b/server/src/main/scala/kpn/server/analyzer/engine/analysis/location/LocationAnalyzerImpl.scala @@ -7,12 +7,12 @@ import org.locationtech.jts.geom.Geometry import org.springframework.stereotype.Component @Component -class LocationAnalyzerImpl(analyzerEnabled: Boolean) extends LocationAnalyzerAbstract { +class LocationAnalyzerImpl(analyzerEnabled: Boolean, development: Boolean) extends LocationAnalyzerAbstract { private val log = Log(classOf[LocationAnalyzerImpl]) private val locationStore = if (analyzerEnabled) { - new LocationStoreReader().read() + new LocationStoreReader(development).read() } else { LocationStore(Seq.empty) diff --git a/server/src/main/scala/kpn/server/analyzer/engine/analysis/location/LocationStoreReader.scala b/server/src/main/scala/kpn/server/analyzer/engine/analysis/location/LocationStoreReader.scala index 6cc7a5701..da5553801 100644 --- a/server/src/main/scala/kpn/server/analyzer/engine/analysis/location/LocationStoreReader.scala +++ b/server/src/main/scala/kpn/server/analyzer/engine/analysis/location/LocationStoreReader.scala @@ -15,14 +15,21 @@ import org.locationtech.jts.geom.Polygon import java.io.File -class LocationStoreReader { +class LocationStoreReader(development: Boolean) { private val log = Log(classOf[LocationStoreReader]) private val root = s"${Dirs.root}/locations" def read(): LocationStore = { - val countries: Seq[LocationStoreCountry] = Country.all.map { country => + val locationCounties = if (development) { + log.warn("!!! Loading NL and BE locations only!!!") + Seq(Country.nl, Country.be) + } + else { + Country.all + } + val countries: Seq[LocationStoreCountry] = locationCounties.map { country => val locationStoreCountry = loadCountry(country) val filename = s"$root/${country.domain}/tree.json" val string = FileUtils.readFileToString(new File(filename), "UTF-8") diff --git a/server/src/main/scala/kpn/server/config/ServerConfiguration.scala b/server/src/main/scala/kpn/server/config/ServerConfiguration.scala index c5a972c93..cb6e588e2 100644 --- a/server/src/main/scala/kpn/server/config/ServerConfiguration.scala +++ b/server/src/main/scala/kpn/server/config/ServerConfiguration.scala @@ -77,6 +77,11 @@ class ServerConfiguration { value } + @Bean + def development(@Value("${app.development:false}") value: Boolean): Boolean = { + value + } + @Bean def changeSetInfoEngineEnabled(@Value("${app.change-set-info-engine-enabled:false}") value: Boolean): Boolean = { value