1 /* 2 * Copyright (C) 2012 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.internal.util; 18 19 import android.os.FileUtils; 20 import android.util.Slog; 21 22 import java.io.BufferedInputStream; 23 import java.io.BufferedOutputStream; 24 import java.io.File; 25 import java.io.FileInputStream; 26 import java.io.FileOutputStream; 27 import java.io.IOException; 28 import java.io.InputStream; 29 import java.io.OutputStream; 30 import java.util.zip.ZipEntry; 31 import java.util.zip.ZipOutputStream; 32 33 import libcore.io.IoUtils; 34 import libcore.io.Streams; 35 36 /** 37 * Utility that rotates files over time, similar to {@code logrotate}. There is 38 * a single "active" file, which is periodically rotated into historical files, 39 * and eventually deleted entirely. Files are stored under a specific directory 40 * with a well-known prefix. 41 * <p> 42 * Instead of manipulating files directly, users implement interfaces that 43 * perform operations on {@link InputStream} and {@link OutputStream}. This 44 * enables atomic rewriting of file contents in 45 * {@link #rewriteActive(Rewriter, long)}. 46 * <p> 47 * Users must periodically call {@link #maybeRotate(long)} to perform actual 48 * rotation. Not inherently thread safe. 49 */ 50 public class FileRotator { 51 private static final String TAG = "FileRotator"; 52 private static final boolean LOGD = false; 53 54 private final File mBasePath; 55 private final String mPrefix; 56 private final long mRotateAgeMillis; 57 private final long mDeleteAgeMillis; 58 59 private static final String SUFFIX_BACKUP = ".backup"; 60 private static final String SUFFIX_NO_BACKUP = ".no_backup"; 61 62 // TODO: provide method to append to active file 63 64 /** 65 * External class that reads data from a given {@link InputStream}. May be 66 * called multiple times when reading rotated data. 67 */ 68 public interface Reader { read(InputStream in)69 public void read(InputStream in) throws IOException; 70 } 71 72 /** 73 * External class that writes data to a given {@link OutputStream}. 74 */ 75 public interface Writer { write(OutputStream out)76 public void write(OutputStream out) throws IOException; 77 } 78 79 /** 80 * External class that reads existing data from given {@link InputStream}, 81 * then writes any modified data to {@link OutputStream}. 82 */ 83 public interface Rewriter extends Reader, Writer { reset()84 public void reset(); shouldWrite()85 public boolean shouldWrite(); 86 } 87 88 /** 89 * Create a file rotator. 90 * 91 * @param basePath Directory under which all files will be placed. 92 * @param prefix Filename prefix used to identify this rotator. 93 * @param rotateAgeMillis Age in milliseconds beyond which an active file 94 * may be rotated into a historical file. 95 * @param deleteAgeMillis Age in milliseconds beyond which a rotated file 96 * may be deleted. 97 */ FileRotator(File basePath, String prefix, long rotateAgeMillis, long deleteAgeMillis)98 public FileRotator(File basePath, String prefix, long rotateAgeMillis, long deleteAgeMillis) { 99 mBasePath = Preconditions.checkNotNull(basePath); 100 mPrefix = Preconditions.checkNotNull(prefix); 101 mRotateAgeMillis = rotateAgeMillis; 102 mDeleteAgeMillis = deleteAgeMillis; 103 104 // ensure that base path exists 105 mBasePath.mkdirs(); 106 107 // recover any backup files 108 for (String name : mBasePath.list()) { 109 if (!name.startsWith(mPrefix)) continue; 110 111 if (name.endsWith(SUFFIX_BACKUP)) { 112 if (LOGD) Slog.d(TAG, "recovering " + name); 113 114 final File backupFile = new File(mBasePath, name); 115 final File file = new File( 116 mBasePath, name.substring(0, name.length() - SUFFIX_BACKUP.length())); 117 118 // write failed with backup; recover last file 119 backupFile.renameTo(file); 120 121 } else if (name.endsWith(SUFFIX_NO_BACKUP)) { 122 if (LOGD) Slog.d(TAG, "recovering " + name); 123 124 final File noBackupFile = new File(mBasePath, name); 125 final File file = new File( 126 mBasePath, name.substring(0, name.length() - SUFFIX_NO_BACKUP.length())); 127 128 // write failed without backup; delete both 129 noBackupFile.delete(); 130 file.delete(); 131 } 132 } 133 } 134 135 /** 136 * Delete all files managed by this rotator. 137 */ deleteAll()138 public void deleteAll() { 139 final FileInfo info = new FileInfo(mPrefix); 140 for (String name : mBasePath.list()) { 141 if (info.parse(name)) { 142 // delete each file that matches parser 143 new File(mBasePath, name).delete(); 144 } 145 } 146 } 147 148 /** 149 * Dump all files managed by this rotator for debugging purposes. 150 */ dumpAll(OutputStream os)151 public void dumpAll(OutputStream os) throws IOException { 152 final ZipOutputStream zos = new ZipOutputStream(os); 153 try { 154 final FileInfo info = new FileInfo(mPrefix); 155 for (String name : mBasePath.list()) { 156 if (info.parse(name)) { 157 final ZipEntry entry = new ZipEntry(name); 158 zos.putNextEntry(entry); 159 160 final File file = new File(mBasePath, name); 161 final FileInputStream is = new FileInputStream(file); 162 try { 163 Streams.copy(is, zos); 164 } finally { 165 IoUtils.closeQuietly(is); 166 } 167 168 zos.closeEntry(); 169 } 170 } 171 } finally { 172 IoUtils.closeQuietly(zos); 173 } 174 } 175 176 /** 177 * Process currently active file, first reading any existing data, then 178 * writing modified data. Maintains a backup during write, which is restored 179 * if the write fails. 180 */ rewriteActive(Rewriter rewriter, long currentTimeMillis)181 public void rewriteActive(Rewriter rewriter, long currentTimeMillis) 182 throws IOException { 183 final String activeName = getActiveName(currentTimeMillis); 184 rewriteSingle(rewriter, activeName); 185 } 186 187 @Deprecated combineActive(final Reader reader, final Writer writer, long currentTimeMillis)188 public void combineActive(final Reader reader, final Writer writer, long currentTimeMillis) 189 throws IOException { 190 rewriteActive(new Rewriter() { 191 @Override 192 public void reset() { 193 // ignored 194 } 195 196 @Override 197 public void read(InputStream in) throws IOException { 198 reader.read(in); 199 } 200 201 @Override 202 public boolean shouldWrite() { 203 return true; 204 } 205 206 @Override 207 public void write(OutputStream out) throws IOException { 208 writer.write(out); 209 } 210 }, currentTimeMillis); 211 } 212 213 /** 214 * Process all files managed by this rotator, usually to rewrite historical 215 * data. Each file is processed atomically. 216 */ rewriteAll(Rewriter rewriter)217 public void rewriteAll(Rewriter rewriter) throws IOException { 218 final FileInfo info = new FileInfo(mPrefix); 219 for (String name : mBasePath.list()) { 220 if (!info.parse(name)) continue; 221 222 // process each file that matches parser 223 rewriteSingle(rewriter, name); 224 } 225 } 226 227 /** 228 * Process a single file atomically, first reading any existing data, then 229 * writing modified data. Maintains a backup during write, which is restored 230 * if the write fails. 231 */ rewriteSingle(Rewriter rewriter, String name)232 private void rewriteSingle(Rewriter rewriter, String name) throws IOException { 233 if (LOGD) Slog.d(TAG, "rewriting " + name); 234 235 final File file = new File(mBasePath, name); 236 final File backupFile; 237 238 rewriter.reset(); 239 240 if (file.exists()) { 241 // read existing data 242 readFile(file, rewriter); 243 244 // skip when rewriter has nothing to write 245 if (!rewriter.shouldWrite()) return; 246 247 // backup existing data during write 248 backupFile = new File(mBasePath, name + SUFFIX_BACKUP); 249 file.renameTo(backupFile); 250 251 try { 252 writeFile(file, rewriter); 253 254 // write success, delete backup 255 backupFile.delete(); 256 } catch (Throwable t) { 257 // write failed, delete file and restore backup 258 file.delete(); 259 backupFile.renameTo(file); 260 throw rethrowAsIoException(t); 261 } 262 263 } else { 264 // create empty backup during write 265 backupFile = new File(mBasePath, name + SUFFIX_NO_BACKUP); 266 backupFile.createNewFile(); 267 268 try { 269 writeFile(file, rewriter); 270 271 // write success, delete empty backup 272 backupFile.delete(); 273 } catch (Throwable t) { 274 // write failed, delete file and empty backup 275 file.delete(); 276 backupFile.delete(); 277 throw rethrowAsIoException(t); 278 } 279 } 280 } 281 282 /** 283 * Read any rotated data that overlap the requested time range. 284 */ readMatching(Reader reader, long matchStartMillis, long matchEndMillis)285 public void readMatching(Reader reader, long matchStartMillis, long matchEndMillis) 286 throws IOException { 287 final FileInfo info = new FileInfo(mPrefix); 288 for (String name : mBasePath.list()) { 289 if (!info.parse(name)) continue; 290 291 // read file when it overlaps 292 if (info.startMillis <= matchEndMillis && matchStartMillis <= info.endMillis) { 293 if (LOGD) Slog.d(TAG, "reading matching " + name); 294 295 final File file = new File(mBasePath, name); 296 readFile(file, reader); 297 } 298 } 299 } 300 301 /** 302 * Return the currently active file, which may not exist yet. 303 */ getActiveName(long currentTimeMillis)304 private String getActiveName(long currentTimeMillis) { 305 String oldestActiveName = null; 306 long oldestActiveStart = Long.MAX_VALUE; 307 308 final FileInfo info = new FileInfo(mPrefix); 309 for (String name : mBasePath.list()) { 310 if (!info.parse(name)) continue; 311 312 // pick the oldest active file which covers current time 313 if (info.isActive() && info.startMillis < currentTimeMillis 314 && info.startMillis < oldestActiveStart) { 315 oldestActiveName = name; 316 oldestActiveStart = info.startMillis; 317 } 318 } 319 320 if (oldestActiveName != null) { 321 return oldestActiveName; 322 } else { 323 // no active file found above; create one starting now 324 info.startMillis = currentTimeMillis; 325 info.endMillis = Long.MAX_VALUE; 326 return info.build(); 327 } 328 } 329 330 /** 331 * Examine all files managed by this rotator, renaming or deleting if their 332 * age matches the configured thresholds. 333 */ maybeRotate(long currentTimeMillis)334 public void maybeRotate(long currentTimeMillis) { 335 final long rotateBefore = currentTimeMillis - mRotateAgeMillis; 336 final long deleteBefore = currentTimeMillis - mDeleteAgeMillis; 337 338 final FileInfo info = new FileInfo(mPrefix); 339 String[] baseFiles = mBasePath.list(); 340 if (baseFiles == null) { 341 return; 342 } 343 344 for (String name : baseFiles) { 345 if (!info.parse(name)) continue; 346 347 if (info.isActive()) { 348 if (info.startMillis <= rotateBefore) { 349 // found active file; rotate if old enough 350 if (LOGD) Slog.d(TAG, "rotating " + name); 351 352 info.endMillis = currentTimeMillis; 353 354 final File file = new File(mBasePath, name); 355 final File destFile = new File(mBasePath, info.build()); 356 file.renameTo(destFile); 357 } 358 } else if (info.endMillis <= deleteBefore) { 359 // found rotated file; delete if old enough 360 if (LOGD) Slog.d(TAG, "deleting " + name); 361 362 final File file = new File(mBasePath, name); 363 file.delete(); 364 } 365 } 366 } 367 readFile(File file, Reader reader)368 private static void readFile(File file, Reader reader) throws IOException { 369 final FileInputStream fis = new FileInputStream(file); 370 final BufferedInputStream bis = new BufferedInputStream(fis); 371 try { 372 reader.read(bis); 373 } finally { 374 IoUtils.closeQuietly(bis); 375 } 376 } 377 writeFile(File file, Writer writer)378 private static void writeFile(File file, Writer writer) throws IOException { 379 final FileOutputStream fos = new FileOutputStream(file); 380 final BufferedOutputStream bos = new BufferedOutputStream(fos); 381 try { 382 writer.write(bos); 383 bos.flush(); 384 } finally { 385 FileUtils.sync(fos); 386 IoUtils.closeQuietly(bos); 387 } 388 } 389 rethrowAsIoException(Throwable t)390 private static IOException rethrowAsIoException(Throwable t) throws IOException { 391 if (t instanceof IOException) { 392 throw (IOException) t; 393 } else { 394 throw new IOException(t.getMessage(), t); 395 } 396 } 397 398 /** 399 * Details for a rotated file, either parsed from an existing filename, or 400 * ready to be built into a new filename. 401 */ 402 private static class FileInfo { 403 public final String prefix; 404 405 public long startMillis; 406 public long endMillis; 407 FileInfo(String prefix)408 public FileInfo(String prefix) { 409 this.prefix = Preconditions.checkNotNull(prefix); 410 } 411 412 /** 413 * Attempt parsing the given filename. 414 * 415 * @return Whether parsing was successful. 416 */ parse(String name)417 public boolean parse(String name) { 418 startMillis = endMillis = -1; 419 420 final int dotIndex = name.lastIndexOf('.'); 421 final int dashIndex = name.lastIndexOf('-'); 422 423 // skip when missing time section 424 if (dotIndex == -1 || dashIndex == -1) return false; 425 426 // skip when prefix doesn't match 427 if (!prefix.equals(name.substring(0, dotIndex))) return false; 428 429 try { 430 startMillis = Long.parseLong(name.substring(dotIndex + 1, dashIndex)); 431 432 if (name.length() - dashIndex == 1) { 433 endMillis = Long.MAX_VALUE; 434 } else { 435 endMillis = Long.parseLong(name.substring(dashIndex + 1)); 436 } 437 438 return true; 439 } catch (NumberFormatException e) { 440 return false; 441 } 442 } 443 444 /** 445 * Build current state into filename. 446 */ build()447 public String build() { 448 final StringBuilder name = new StringBuilder(); 449 name.append(prefix).append('.').append(startMillis).append('-'); 450 if (endMillis != Long.MAX_VALUE) { 451 name.append(endMillis); 452 } 453 return name.toString(); 454 } 455 456 /** 457 * Test if current file is active (no end timestamp). 458 */ isActive()459 public boolean isActive() { 460 return endMillis == Long.MAX_VALUE; 461 } 462 } 463 } 464