All Downloads are FREE. Search and download functionalities are using the official Maven repository.

proguard.io.ManifestRewriter Maven / Gradle / Ivy

Go to download

ProGuardCORE is a free library to read, analyze, modify, and write Java class files.

There is a newer version: 9.1.6
Show newest version
/*
 * ProGuardCORE -- library to process Java bytecode.
 *
 * Copyright (c) 2002-2020 Guardsquare NV
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package proguard.io;

import java.io.*;
import java.nio.charset.Charset;
import proguard.classfile.ClassPool;

/**
 * This {@link DataEntryReader} writes the manifest data entries that it reads to a given {@link
 * DataEntryWriter}, updating their contents based on the renamed classes in the given {@link
 * ClassPool}.
 *
 * @author Eric Lafortune
 */
public class ManifestRewriter extends DataEntryRewriter {
  /** Creates a new ManifestRewriter. */
  public ManifestRewriter(ClassPool classPool, Charset charset, DataEntryWriter dataEntryWriter) {
    super(classPool, charset, dataEntryWriter);
  }

  // Implementations for DataEntryRewriter.

  @Override
  protected void copyData(Reader reader, Writer writer) throws IOException {
    super.copyData(new SplitLineReader(reader), new SplitLineWriter(writer));
  }

  /**
   * This Reader reads manifest files, joining any split lines. It replaces the allowed CR/LF/CR+LF
   * alternatives by simple LF in the process.
   */
  private static class SplitLineReader extends FilterReader {
    private static final int NONE = -2;

    private int bufferedCharacter = NONE;

    public SplitLineReader(Reader reader) {
      super(reader);
    }

    // Implementations for Reader.

    @Override
    public int read() throws IOException {
      while (true) {
        // Get the buffered character or the first character.
        int c1 = bufferedCharacter != NONE ? bufferedCharacter : super.read();

        // Clear the buffered character.
        bufferedCharacter = NONE;

        // Return it if it's an ordinary character.
        if (c1 != '\n' && c1 != '\r') {
          return c1;
        }

        // It's a newline. Read the second character to see if it's a
        // continuation.
        int c2 = super.read();

        // Skip any corresponding, redundant \n or \r.
        if ((c2 == '\n' || c2 == '\r') && c1 != c2) {
          c2 = super.read();
        }

        // Isn't it a continuation after all?
        if (c2 != ' ') {
          // Buffer the second character and return a newline.
          bufferedCharacter = c2;
          return '\n';
        }

        // Just continue after the continuation characters.
      }
    }

    @Override
    public int read(char[] cbuf, int off, int len) throws IOException {
      // Delegate to reading a single character at a time.
      int count = 0;
      while (count < len) {
        int c = read();
        if (c == -1) {
          break;
        }

        cbuf[off + count++] = (char) c;
      }

      return count;
    }

    @Override
    public long skip(long n) throws IOException {
      // Delegate to reading a single character at a time.
      int count = 0;
      while (count < n) {
        int c = read();
        if (c == -1) {
          break;
        }

        count++;
      }

      return count;
    }
  }

  /** This Writer writes manifest files, splitting any long lines. */
  private static class SplitLineWriter extends FilterWriter {
    private int counter = 0;

    public SplitLineWriter(Writer writer) {
      super(writer);
    }

    // Implementations for Reader.

    @Override
    public void write(int c) throws IOException {
      // TODO: We should actually count the Utf-8 bytes, not the characters.
      if (c == '\n') {
        // Reset the character count.
        counter = 0;
      } else if (counter == 70) {
        // Insert a newline and a space.
        super.write('\n');
        super.write(' ');

        counter = 2;
      } else {
        counter++;
      }

      super.write(c);
    }

    @Override
    public void write(char[] cbuf, int off, int len) throws IOException {
      for (int count = 0; count < len; count++) {
        write(cbuf[off + count]);
      }
    }

    @Override
    public void write(String str, int off, int len) throws IOException {
      write(str.toCharArray(), off, len);
    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy