/* * Copyright © 2014 Keith Packard * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. */ package org.altusmetrum.altoslib_12; import java.io.*; import java.net.*; import java.util.*; public class AltosMapStore { String url; public File file; LinkedList listeners = new LinkedList(); int status; private static File map_file(AltosLatLon center, int zoom, int maptype, int px_size, int scale) { double lat = center.lat; double lon = center.lon; char chlat = lat < 0 ? 'S' : 'N'; char chlon = lon < 0 ? 'W' : 'E'; if (lat < 0) lat = -lat; if (lon < 0) lon = -lon; String maptype_string = String.format("%s-", AltosMap.maptype_names[maptype]); String format_string; if (maptype == AltosMap.maptype_hybrid || maptype == AltosMap.maptype_satellite || maptype == AltosMap.maptype_terrain) format_string = "jpg"; else format_string = "png"; return new File(AltosPreferences.mapdir(), String.format("map-%c%.6f,%c%.6f-%s%d%s.%s", chlat, lat, chlon, lon, maptype_string, zoom, scale == 1 ? "" : String.format("-%d", scale), format_string)); } private static String map_url(AltosLatLon center, int zoom, int maptype, int px_size, int scale) { String format_string; int z = zoom; if (maptype == AltosMap.maptype_hybrid || maptype == AltosMap.maptype_satellite || maptype == AltosMap.maptype_terrain) format_string = "jpg"; else format_string = "png32"; for (int s = 1; s < scale; s <<= 1) z--; if (AltosVersion.has_google_maps_api_key()) return String.format("http://maps.google.com/maps/api/staticmap?center=%.6f,%.6f&zoom=%d&size=%dx%d&scale=%d&sensor=false&maptype=%s&format=%s&key=%s", center.lat, center.lon, z, px_size/scale, px_size/scale, scale, AltosMap.maptype_names[maptype], format_string, AltosVersion.google_maps_api_key); else return String.format("http://maps.google.com/maps/api/staticmap?center=%.6f,%.6f&zoom=%d&size=%dx%d&scale=%d&sensor=false&maptype=%s&format=%s", center.lat, center.lon, z, px_size/scale, px_size/scale, AltosMap.maptype_names[maptype], format_string); } public int status() { return status; } public synchronized void add_listener(AltosMapStoreListener listener) { if (!listeners.contains(listener)) listeners.add(listener); listener.notify_store(this, status); } public synchronized void remove_listener(AltosMapStoreListener listener) { listeners.remove(listener); } private synchronized void notify_listeners(int status) { this.status = status; for (AltosMapStoreListener listener : listeners) listener.notify_store(this, status); } static Object forbidden_lock = new Object(); static long forbidden_time; static boolean forbidden_set; public static int forbidden_response; private int fetch_url() { URL u; try { u = new URL(url); } catch (java.net.MalformedURLException e) { return AltosMapTile.bad_request; } byte[] data; URLConnection uc = null; try { uc = u.openConnection(); String type = uc.getContentType(); int contentLength = uc.getContentLength(); if (uc instanceof HttpURLConnection) { int response = ((HttpURLConnection) uc).getResponseCode(); switch (response) { case HttpURLConnection.HTTP_FORBIDDEN: case HttpURLConnection.HTTP_PAYMENT_REQUIRED: case HttpURLConnection.HTTP_UNAUTHORIZED: synchronized (forbidden_lock) { forbidden_time = System.nanoTime(); forbidden_set = true; forbidden_response = response; return AltosMapTile.forbidden; } } } InputStream in = new BufferedInputStream(uc.getInputStream()); int bytesRead = 0; int offset = 0; data = new byte[contentLength]; while (offset < contentLength) { bytesRead = in.read(data, offset, data.length - offset); if (bytesRead == -1) break; offset += bytesRead; } in.close(); if (offset != contentLength) return AltosMapTile.failed; } catch (IOException e) { return AltosMapTile.failed; } try { FileOutputStream out = new FileOutputStream(file); out.write(data); out.flush(); out.close(); } catch (FileNotFoundException e) { return AltosMapTile.bad_request; } catch (IOException e) { if (file.exists()) file.delete(); return AltosMapTile.bad_request; } return AltosMapTile.fetched; } static Object fetch_lock = new Object(); static final long forbidden_interval = 60l * 1000l * 1000l * 1000l; static final long google_maps_ratelimit_ms = 1200; static Object fetcher_lock = new Object(); static LinkedList waiting = new LinkedList(); static LinkedList running = new LinkedList(); static final int concurrent_fetchers = 128; static void start_fetchers() { while (!waiting.isEmpty() && running.size() < concurrent_fetchers) { AltosMapStore s = waiting.remove(); running.add(s); Thread lt = s.make_fetcher_thread(); lt.start(); } } void finish_fetcher() { synchronized(fetcher_lock) { running.remove(this); start_fetchers(); } } void add_fetcher() { synchronized(fetcher_lock) { waiting.add(this); start_fetchers(); } } class fetcher implements Runnable { public void run() { try { if (file.exists()) { notify_listeners(AltosMapTile.fetched); return; } synchronized(forbidden_lock) { if (forbidden_set && (System.nanoTime() - forbidden_time) < forbidden_interval) { notify_listeners(AltosMapTile.forbidden); return; } } int new_status; if (!AltosVersion.has_google_maps_api_key()) { synchronized (fetch_lock) { long startTime = System.nanoTime(); new_status = fetch_url(); if (new_status == AltosMapTile.fetched) { long duration_ms = (System.nanoTime() - startTime) / 1000000; if (duration_ms < google_maps_ratelimit_ms) { try { Thread.sleep(google_maps_ratelimit_ms - duration_ms); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } } } } } else { new_status = fetch_url(); } notify_listeners(new_status); } finally { finish_fetcher(); } } } private Thread make_fetcher_thread() { return new Thread(new fetcher()); } private void fetch() { add_fetcher(); } private AltosMapStore (String url, File file) { this.url = url; this.file = file; if (file.exists()) status = AltosMapTile.fetched; else { status = AltosMapTile.fetching; fetch(); } } public int hashCode() { return url.hashCode(); } public boolean equals(Object o) { if (o == null) return false; if (!(o instanceof AltosMapStore)) return false; AltosMapStore other = (AltosMapStore) o; return url.equals(other.url); } static HashMap stores = new HashMap(); public static AltosMapStore get(AltosLatLon center, int zoom, int maptype, int px_size, int scale) { String url = map_url(center, zoom, maptype, px_size, scale); AltosMapStore store; synchronized(stores) { if (stores.containsKey(url)) { store = stores.get(url); } else { store = new AltosMapStore(url, map_file(center, zoom, maptype, px_size, scale)); stores.put(url, store); } } return store; } }