/* * Copyright (c) 2010-2012 Sonatype, Inc. All rights reserved. * * This program is licensed to you under the Apache License Version 2.0, * and you may not use this file except in compliance with the Apache License Version 2.0. * You may obtain a copy of the Apache License Version 2.0 at http://www.apache.org/licenses/LICENSE-2.0. * * Unless required by applicable law or agreed to in writing, * software distributed under the Apache License Version 2.0 is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the Apache License Version 2.0 for the specific language governing permissions and limitations there under. */ package org.asynchttpclient.handler.resumable; import static java.nio.charset.StandardCharsets.UTF_8; import static org.asynchttpclient.util.MiscUtils.closeSilently; import java.io.File; import java.io.FileNotFoundException; import java.io.OutputStream; import java.nio.file.Files; import java.util.Map; import java.util.Scanner; import java.util.concurrent.ConcurrentHashMap; import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** * A {@link org.asynchttpclient.handler.resumable.ResumableAsyncHandler.ResumableProcessor} which use a properties file * to store the download index information. */ public class PropertiesBasedResumableProcessor implements ResumableAsyncHandler.ResumableProcessor { private final static Logger log = LoggerFactory.getLogger(PropertiesBasedResumableProcessor.class); private final static File TMP = new File(System.getProperty("java.io.tmpdir"), "ahc"); private final static String storeName = "ResumableAsyncHandler.properties"; private final ConcurrentHashMap<String, Long> properties = new ConcurrentHashMap<>(); /** * {@inheritDoc} */ @Override public void put(String url, long transferredBytes) { properties.put(url, transferredBytes); } /** * {@inheritDoc} */ @Override public void remove(String uri) { if (uri != null) { properties.remove(uri); } } /** * {@inheritDoc} */ @Override public void save(Map<String, Long> map) { log.debug("Saving current download state {}", properties.toString()); OutputStream os = null; try { if (!TMP.exists() && !TMP.mkdirs()) { throw new IllegalStateException("Unable to create directory: " + TMP.getAbsolutePath()); } File f = new File(TMP, storeName); if (!f.exists() && !f.createNewFile()) { throw new IllegalStateException("Unable to create temp file: " + f.getAbsolutePath()); } if (!f.canWrite()) { throw new IllegalStateException(); } os = Files.newOutputStream(f.toPath()); for (Map.Entry<String, Long> e : properties.entrySet()) { os.write(append(e).getBytes(UTF_8)); } os.flush(); } catch (Throwable e) { log.warn(e.getMessage(), e); } finally { closeSilently(os); } } private static String append(Map.Entry<String, Long> e) { return new StringBuilder(e.getKey()).append('=').append(e.getValue()).append('\n').toString(); } /** * {@inheritDoc} */ @Override public Map<String, Long> load() { Scanner scan = null; try { scan = new Scanner(new File(TMP, storeName), UTF_8.name()); scan.useDelimiter("[=\n]"); String key; String value; while (scan.hasNext()) { key = scan.next().trim(); value = scan.next().trim(); properties.put(key, Long.valueOf(value)); } log.debug("Loading previous download state {}", properties.toString()); } catch (FileNotFoundException ex) { log.debug("Missing {}", storeName); } catch (Throwable ex) { // Survive any exceptions log.warn(ex.getMessage(), ex); } finally { if (scan != null) scan.close(); } return properties; } }