/* Copyright (c) 2013-2014 Boundless and others. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Distribution License v1.0 * which accompanies this distribution, and is available at * https://www.eclipse.org/org/documents/edl-v10.html * * Contributors: * Juan Marin (Boundless) - initial implementation */ package org.locationtech.geogig.geotools.cli.porcelain; import java.io.File; import java.io.FileWriter; import java.io.IOException; import java.util.List; import javax.annotation.Nullable; import org.geotools.data.DataStore; import org.geotools.data.memory.MemoryDataStore; import org.geotools.data.simple.SimpleFeatureSource; import org.geotools.data.simple.SimpleFeatureStore; import org.geotools.feature.FeatureCollection; import org.geotools.geojson.feature.FeatureJSON; import org.locationtech.geogig.api.GeoGIG; import org.locationtech.geogig.api.NodeRef; import org.locationtech.geogig.api.ObjectId; import org.locationtech.geogig.api.RevFeatureType; import org.locationtech.geogig.api.RevObject; import org.locationtech.geogig.api.RevObject.TYPE; import org.locationtech.geogig.api.RevTree; import org.locationtech.geogig.api.plumbing.FindTreeChild; import org.locationtech.geogig.api.plumbing.ResolveObjectType; import org.locationtech.geogig.api.plumbing.ResolveTreeish; import org.locationtech.geogig.api.plumbing.RevObjectParse; import org.locationtech.geogig.api.plumbing.RevParse; import org.locationtech.geogig.cli.CLICommand; import org.locationtech.geogig.cli.CommandFailedException; import org.locationtech.geogig.cli.GeogigCLI; import org.locationtech.geogig.cli.InvalidParameterException; import org.locationtech.geogig.geotools.plumbing.ExportOp; import org.locationtech.geogig.geotools.plumbing.GeoToolsOpException; import org.opengis.feature.simple.SimpleFeatureType; import com.beust.jcommander.Parameter; import com.beust.jcommander.Parameters; import com.google.common.base.Optional; /** * Exports features from a feature type into a GeoJSON file. * * @see ExportOp */ @Parameters(commandNames = "export", commandDescription = "Export GeoJSON") public class GeoJsonExport extends AbstractGeoJsonCommand implements CLICommand { @Parameter(description = "<path> <geojson>", arity = 2) public List<String> args; @Parameter(names = { "--overwrite", "-o" }, description = "Overwrite output file") public boolean overwrite; @Parameter(names = { "--defaulttype" }, description = "Export only features with the tree default feature type if several types are found") public boolean defaultType; @Parameter(names = { "--alter" }, description = "Export all features if several types are found, altering them to adapt to the output feature type") public boolean alter; @Parameter(names = { "--featuretype" }, description = "Export only features with the specified feature type if several types are found") @Nullable public String sFeatureTypeId; /** * Executes the export command using the provided options. */ @Override protected void runInternal(GeogigCLI cli) throws InvalidParameterException, CommandFailedException, IOException { if (args.isEmpty()) { printUsage(cli); throw new CommandFailedException(); } String path = args.get(0); String geojson = args.get(1); File file = new File(geojson); if (file.exists() && !overwrite) { throw new CommandFailedException( "The selected GeoJSON file already exists. Use -o to overwrite"); } SimpleFeatureType outputFeatureType; ObjectId featureTypeId; if (sFeatureTypeId != null) { // Check the feature type id string is a correct id Optional<ObjectId> id = cli.getGeogig().command(RevParse.class) .setRefSpec(sFeatureTypeId).call(); checkParameter(id.isPresent(), "Invalid feature type reference", sFeatureTypeId); TYPE type = cli.getGeogig().command(ResolveObjectType.class).setObjectId(id.get()) .call(); checkParameter(type.equals(TYPE.FEATURETYPE), "Provided reference does not resolve to a feature type: ", sFeatureTypeId); outputFeatureType = (SimpleFeatureType) cli.getGeogig().command(RevObjectParse.class) .setObjectId(id.get()).call(RevFeatureType.class).get().type(); featureTypeId = id.get(); } else { try { outputFeatureType = getFeatureType(path, cli); featureTypeId = null; } catch (GeoToolsOpException e) { cli.getConsole().println("No features to export."); return; } } DataStore dataStore = new MemoryDataStore(outputFeatureType); final String typeName = dataStore.getTypeNames()[0]; final SimpleFeatureSource featureSource = dataStore.getFeatureSource(typeName); if (!(featureSource instanceof SimpleFeatureStore)) { throw new CommandFailedException("Could not create feature store."); } final SimpleFeatureStore featureStore = (SimpleFeatureStore) featureSource; ExportOp op = cli.getGeogig().command(ExportOp.class).setFeatureStore(featureStore) .setPath(path).setFilterFeatureTypeId(featureTypeId).setAlter(alter); op.setTransactional(false); if (defaultType) { op.exportDefaultFeatureType(); } FileWriter writer = null; try { op.setProgressListener(cli.getProgressListener()).call(); FeatureJSON fjson = new FeatureJSON(); @SuppressWarnings("rawtypes") FeatureCollection fc = featureSource.getFeatures(); writer = new FileWriter(file); fjson.writeFeatureCollection(fc, writer); } catch (IllegalArgumentException iae) { throw new org.locationtech.geogig.cli.InvalidParameterException(iae.getMessage(), iae); } catch (GeoToolsOpException e) { file.delete(); switch (e.statusCode) { case MIXED_FEATURE_TYPES: throw new CommandFailedException( "Error: The selected tree contains mixed feature types. Use --defaulttype or --featuretype <feature_type_ref> to export.", e); default: throw new CommandFailedException("Could not export. Error:" + e.statusCode.name(), e); } } finally { writer.flush(); writer.close(); } cli.getConsole().println(path + " exported successfully to " + geojson); } private SimpleFeatureType getFeatureType(String path, GeogigCLI cli) { checkParameter(path != null, "No path specified."); String refspec; if (path.contains(":")) { refspec = path; } else { refspec = "WORK_HEAD:" + path; } checkParameter(!refspec.endsWith(":"), "No path specified."); final GeoGIG geogig = cli.getGeogig(); Optional<ObjectId> rootTreeId = geogig.command(ResolveTreeish.class) .setTreeish(refspec.split(":")[0]).call(); checkParameter(rootTreeId.isPresent(), "Couldn't resolve '" + refspec + "' to a treeish object"); RevTree rootTree = geogig.getRepository().getTree(rootTreeId.get()); Optional<NodeRef> featureTypeTree = geogig.command(FindTreeChild.class) .setChildPath(refspec.split(":")[1]).setParent(rootTree).setIndex(true).call(); checkParameter(featureTypeTree.isPresent(), "pathspec '" + refspec.split(":")[1] + "' did not match any valid path"); Optional<RevObject> revObject = cli.getGeogig().command(RevObjectParse.class) .setObjectId(featureTypeTree.get().getMetadataId()).call(); if (revObject.isPresent() && revObject.get() instanceof RevFeatureType) { RevFeatureType revFeatureType = (RevFeatureType) revObject.get(); if (revFeatureType.type() instanceof SimpleFeatureType) { return (SimpleFeatureType) revFeatureType.type(); } else { throw new InvalidParameterException( "Cannot find feature type for the specified path"); } } else { throw new InvalidParameterException("Cannot find feature type for the specified path"); } } }