package btools.server.request;
import java.io.BufferedWriter;
import java.io.File;
import java.io.StringWriter;
import java.util.Map;
import btools.router.OsmTrack;
import btools.router.RoutingContext;
import btools.server.ServiceContext;
/**
* URL query parameter handler for web and standalone server. Supports all
* BRouter features without restrictions.
*
* Parameters:
*
* lonlats = lon,lat|... (unlimited list of lon,lat waypoints separated by |)
* nogos = lon,lat,radius,weight|... (optional, list of lon, lat, radius in meters, weight (optional) separated by |)
* polylines = lon,lat,lon,lat,...,weight|... (unlimited list of lon,lat and weight (optional), lists separated by |)
* polygons = lon,lat,lon,lat,...,weight|... (unlimited list of lon,lat and weight (optional), lists separated by |)
* profile = profile file name without .brf
* alternativeidx = [0|1|2|3] (optional, default 0)
* format = [kml|gpx|geojson] (optional, default gpx)
* trackname = name used for filename and format specific trackname (optional, default brouter)
* exportWaypoints = 1 to export them (optional, default is no export)
* pois = lon,lat,name|... (optional)
* timode = turnInstructionMode [0=none, 1=auto-choose, 2=locus-style, 3=osmand-style, 4=comment-style, 5=gpsies-style, 6=orux-style, 7=locus-old-style] default 0
* heading = angle (optional to give a route a start direction)
* profile:xxx = parameter in profile (optional)
* straight = idx1,idx2,.. (optional, minimum one value, index of a direct routing point in the waypoint list)
*
* Example URLs:
* {@code http://localhost:17777/brouter?lonlats=8.799297,49.565883|8.811764,49.563606&nogos=&profile=trekking&alternativeidx=0&format=gpx}
* {@code http://localhost:17777/brouter?lonlats=1.1,1.2|2.1,2.2|3.1,3.2|4.1,4.2&nogos=-1.1,-1.2,1|-2.1,-2.2,2&profile=shortest&alternativeidx=1&format=kml&trackname=Ride&pois=1.1,2.1,Barner Bar}
*/
public class ServerHandler extends RequestHandler {
private RoutingContext rc;
public ServerHandler(ServiceContext serviceContext, Map params) {
super(serviceContext, params);
}
@Override
public RoutingContext readRoutingContext() {
rc = new RoutingContext();
rc.memoryclass = 128;
String profile = params.get("profile");
// when custom profile replace prefix with directory path
if (profile.startsWith(ProfileUploadHandler.CUSTOM_PREFIX)) {
String customProfile = profile.substring(ProfileUploadHandler.CUSTOM_PREFIX.length());
profile = new File(serviceContext.customProfileDir, customProfile).getPath();
} else if (profile.startsWith(ProfileUploadHandler.SHARED_PREFIX)) {
String customProfile = profile.substring(ProfileUploadHandler.SHARED_PREFIX.length());
profile = new File(serviceContext.sharedProfileDir, customProfile).getPath();
}
rc.localFunction = profile;
return rc;
}
@Override
public String formatTrack(OsmTrack track) {
String result;
// optional, may be null
String format = params.get("format");
String trackName = getTrackName();
if (trackName != null) {
track.name = trackName;
}
String exportWaypointsStr = params.get("exportWaypoints");
if (exportWaypointsStr != null && Integer.parseInt(exportWaypointsStr) != 0) {
track.exportWaypoints = true;
}
if (format == null || "gpx".equals(format)) {
result = track.formatAsGpx();
} else if ("kml".equals(format)) {
result = track.formatAsKml();
} else if ("geojson".equals(format)) {
result = track.formatAsGeoJson();
} else if ("csv".equals(format)) {
try {
StringWriter sw = new StringWriter();
BufferedWriter bw = new BufferedWriter(sw);
track.writeMessages(bw, rc);
return sw.toString();
} catch (Exception ex) {
return "Error: " + ex.getMessage();
}
} else {
System.out.println("unknown track format '" + format + "', using default");
result = track.formatAsGpx();
}
return result;
}
@Override
public String getMimeType() {
// default
String result = "text/plain";
// optional, may be null
String format = params.get("format");
if (format != null) {
if ("gpx".equals(format)) {
result = "application/gpx+xml";
} else if ("kml".equals(format)) {
result = "application/vnd.google-earth.kml+xml";
} else if ("geojson".equals(format)) {
result = "application/vnd.geo+json";
} else if ("csv".equals(format)) {
result = "text/tab-separated-values";
}
}
return result;
}
@Override
public String getFileName() {
String fileName = null;
String format = params.get("format");
String trackName = getTrackName();
if (format != null) {
fileName = (trackName == null ? "brouter" : trackName) + "." + format;
}
return fileName;
}
private String getTrackName() {
return params.get("trackname") == null ? null : params.get("trackname").replaceAll("[^a-zA-Z0-9 \\._\\-]+", "");
}
}