RegionChooserCommandLineInterface.java 3.73 KB
Newer Older
Eric Duminil's avatar
Eric Duminil committed
1
2
package eu.simstadt.regionchooser;

Eric Duminil's avatar
Eric Duminil committed
3
import java.nio.charset.StandardCharsets;
Eric Duminil's avatar
Eric Duminil committed
4
import java.nio.file.Files;
Eric Duminil's avatar
Eric Duminil committed
5
import java.nio.file.Path;
Eric Duminil's avatar
Eric Duminil committed
6
import java.nio.file.Paths;
Eric Duminil's avatar
Eric Duminil committed
7
import java.util.Scanner;
Eric Duminil's avatar
Eric Duminil committed
8
import java.util.concurrent.Callable;
Eric Duminil's avatar
Eric Duminil committed
9
10
11
import org.locationtech.jts.geom.Polygon;
import org.locationtech.jts.io.WKTReader;
import org.locationtech.jts.io.WKTWriter;
Eric Duminil's avatar
Eric Duminil committed
12
import org.osgeo.proj4j.CoordinateReferenceSystem;
Eric Duminil's avatar
Eric Duminil committed
13
14
import picocli.CommandLine;
import picocli.CommandLine.Command;
Eric Duminil's avatar
Eric Duminil committed
15
import picocli.CommandLine.Option;
Eric Duminil's avatar
Eric Duminil committed
16

Eric Duminil's avatar
Eric Duminil committed
17
18
19
20
21
// Example usage:
// --input /home/ricou/Desktop/CGSC_Repository/Würzburg.proj/LoD2_566_5516_2_BY.gml,/home/ricou/Desktop/CGSC_Repository/Würzburg.proj/LoD2_568_5516_2_BY.gml
// --output /home/ricou/Desktop/output.gml
// --wkt /home/ricou/Desktop/grombuhl.txt

Eric Duminil's avatar
Eric Duminil committed
22

Eric Duminil's avatar
Eric Duminil committed
23
@Command(name = "region_chooser", mixinStandardHelpOptions = true, version = "regionChooser x.x", description = "Extracts a region from one or more citygmls.", sortOptions = false)
Eric Duminil's avatar
Eric Duminil committed
24
class RegionChooserCommandLineInterface implements Callable<Integer>
Eric Duminil's avatar
Eric Duminil committed
25
{
Eric Duminil's avatar
Eric Duminil committed
26
27
	@Option(names = { "-i",
			"--input" }, required = true, split = ",", description = "Citygml files to extract from", paramLabel = "input.gml")
Eric Duminil's avatar
Eric Duminil committed
28
	Path[] citygmls;
Eric Duminil's avatar
Eric Duminil committed
29

Eric Duminil's avatar
Eric Duminil committed
30
31
	@Option(names = { "-o",
			"--output" }, required = true, description = "Output file", paramLabel = "output.gml")
Eric Duminil's avatar
Eric Duminil committed
32
	Path outputCityGML;
Eric Duminil's avatar
Eric Duminil committed
33

Eric Duminil's avatar
Eric Duminil committed
34
35
	@Option(names = { "-e", "--epsg" }, description = "EPSG id for coordinate reference system", paramLabel = "31467")
	Integer espgId;
Eric Duminil's avatar
Eric Duminil committed
36

Eric Duminil's avatar
Eric Duminil committed
37
38
39
	@Option(names = { "-l",
			"--local" }, description = "Are WKT coordinates in local CRS?", paramLabel = "local_coordinates?")
	boolean localCoordinates;
Eric Duminil's avatar
Eric Duminil committed
40

Eric Duminil's avatar
Eric Duminil committed
41
42
	@Option(names = { "-w",
			"--wkt" }, description = "File containing WKT polygon, or - for stdin", paramLabel = "polygon.wkt")
Eric Duminil's avatar
Eric Duminil committed
43
44
	String wktFile = "-";

Eric Duminil's avatar
Eric Duminil committed
45
	@Override
Eric Duminil's avatar
Eric Duminil committed
46
	public Integer call() throws Exception {
Eric Duminil's avatar
Eric Duminil committed
47
48
		//TODO: Move as much logic to utils as possible, and test it.
		//TODO: Use logger
Eric Duminil's avatar
Eric Duminil committed
49
		CoordinateReferenceSystem localCRS;
Eric Duminil's avatar
Eric Duminil committed
50

Eric Duminil's avatar
Eric Duminil committed
51
		if (espgId == null) {
Eric Duminil's avatar
Eric Duminil committed
52
			localCRS = RegionChooserUtils.crsFromCityGMLHeader(citygmls[0]);
Eric Duminil's avatar
Eric Duminil committed
53
		} else {
Eric Duminil's avatar
Eric Duminil committed
54
			localCRS = RegionChooserUtils.crsFromSrsName("EPSG:" + espgId);
Eric Duminil's avatar
Eric Duminil committed
55
		}
Eric Duminil's avatar
Eric Duminil committed
56
		System.out.println("Coordinate system is " + localCRS);
Eric Duminil's avatar
Eric Duminil committed
57

Eric Duminil's avatar
Eric Duminil committed
58
		String wktPolygon;
Eric Duminil's avatar
Eric Duminil committed
59

Eric Duminil's avatar
Eric Duminil committed
60
61
		if (wktFile.equals("-")) {
			if (System.in.available() == 0) {
Eric Duminil's avatar
Eric Duminil committed
62
				throw new IllegalArgumentException("Please provide \"POLYGON((x1 y1, x2 y2, ...))\" to standard input.");
Eric Duminil's avatar
Eric Duminil committed
63
			} else {
Eric Duminil's avatar
Eric Duminil committed
64
				wktPolygon = getInput();
Eric Duminil's avatar
Eric Duminil committed
65
			}
Eric Duminil's avatar
Eric Duminil committed
66
		} else {
Eric Duminil's avatar
Eric Duminil committed
67
68
69
70
			wktPolygon = new String(Files.readAllBytes(Paths.get(wktFile)), StandardCharsets.UTF_8);
			if (wktPolygon.isEmpty()) {
				throw new IllegalArgumentException("Please write \"POLYGON((x1 y1, x2 y2, ...))\" inside " + wktFile);
			}
Eric Duminil's avatar
Eric Duminil committed
71
		}
Eric Duminil's avatar
Eric Duminil committed
72

Eric Duminil's avatar
Eric Duminil committed
73
		if (!localCoordinates) {
Eric Duminil's avatar
Eric Duminil committed
74
75
			final WKTReader WKT_READER = new WKTReader();
			final WKTWriter WKT_WRITER = new WKTWriter();
Eric Duminil's avatar
Eric Duminil committed
76
			// WKT coordinates are in WGS84, so should be first converted to srsName
Eric Duminil's avatar
Eric Duminil committed
77
78
79
			Polygon wgs84Polygon = (Polygon) WKT_READER.read(wktPolygon);
			wktPolygon = WKT_WRITER
					.write(RegionChooserUtils.changePolygonCRS(wgs84Polygon, RegionChooserUtils.WGS84, localCRS));
Eric Duminil's avatar
Eric Duminil committed
80
81
		}

Eric Duminil's avatar
Eric Duminil committed
82
83
		System.out.println("WKT Polygon : " + wktPolygon);

Eric Duminil's avatar
Eric Duminil committed
84
		StringBuilder sb = RegionExtractor.selectRegionDirectlyFromCityGML(wktPolygon, localCRS.toString(), citygmls);
Eric Duminil's avatar
Eric Duminil committed
85

Eric Duminil's avatar
Eric Duminil committed
86
87
		//TODO: Check how many buildings are written. Warning if 0?

Eric Duminil's avatar
Eric Duminil committed
88
89
		RegionChooserUtils.writeStringBuilderToFile(sb, outputCityGML);

Eric Duminil's avatar
Eric Duminil committed
90
		return 0;
Eric Duminil's avatar
Eric Duminil committed
91
92
	}

Eric Duminil's avatar
Eric Duminil committed
93
	private static String getInput() {
Eric Duminil's avatar
Eric Duminil committed
94
95
96
		try (Scanner myObj = new Scanner(System.in)) {
			return myObj.nextLine();
		}
Eric Duminil's avatar
Eric Duminil committed
97
98
	}

Eric Duminil's avatar
Eric Duminil committed
99
100
101
102
103
	// this example implements Callable, so parsing, error handling and handling user
	// requests for usage help or version help can be done with one line of code.
	public static void main(String... args) {
		int exitCode = new CommandLine(new RegionChooserCommandLineInterface()).execute(args);
		System.exit(exitCode);
Eric Duminil's avatar
Eric Duminil committed
104
	}
Eric Duminil's avatar
Eric Duminil committed
105
}