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

com.eed3si9n.jarjar.PackageRemapper Maven / Gradle / Ivy

There is a newer version: 1.14.1
Show newest version
/**
 * Copyright 2007 Google Inc.
 *
 * 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 com.eed3si9n.jarjar;

import org.objectweb.asm.*;
import org.objectweb.asm.commons.*;
import java.util.*;
import java.util.regex.Pattern;

class PackageRemapper extends TracingRemapper
{
    private static final String RESOURCE_SUFFIX = "RESOURCE";
    
    private static final Pattern ARRAY_FOR_NAME_PATTERN
        = Pattern.compile("\\[L[\\p{javaJavaIdentifierPart}\\.]+?;");

    private final List wildcards;
    private final List ruleList;
    private final Map typeCache = new HashMap();
    private final Map pathCache = new HashMap();
    private final Map valueCache = new HashMap();
    private final boolean verbose;
    private boolean modified = false;


    public PackageRemapper(List ruleList, boolean verbose) {
        this.verbose = verbose;
        this.ruleList = ruleList;
        wildcards = PatternElement.createWildcards(ruleList);
    }

    @Override
    public TracingRemapper copy() {
        return new PackageRemapper(this.ruleList, this.verbose);
    }

    // also used by KeepProcessor
    static boolean isArrayForName(String value) {
      return ARRAY_FOR_NAME_PATTERN.matcher(value).matches();
    }

    public String map(String key) {
        String s = typeCache.get(key);
        if (s == null) {
            s = replaceHelper(key);
            if (key.equals(s))
                s = null;
            typeCache.put(key, s);
        }
        return s;
    }

    public String mapPath(String path) {
        String s = pathCache.get(path);
        if (s == null) {
            s = path;
            int slash = s.lastIndexOf('/');
            String end;
            if (slash < 0) {
                end = s;
                s = RESOURCE_SUFFIX;
            } else {
                end = s.substring(slash + 1);
                s = s.substring(0, slash + 1) + RESOURCE_SUFFIX;
            }
            boolean absolute = s.startsWith("/");
            if (absolute) s = s.substring(1);
            
            s = replaceHelper(s);
            
            if (absolute) s = "/" + s;
            if (s.indexOf(RESOURCE_SUFFIX) < 0)
              return path;
            s = s.substring(0, s.length() - RESOURCE_SUFFIX.length()) + end;
            pathCache.put(path, s);
        }
        return s;
    }

    public Object mapValue(Object value) {
        if (value instanceof String) {
            String s = valueCache.get(value);
            if (s == null) {
                s = (String)value;
                if (isArrayForName(s)) {
                    String desc1 = s.replace('.', '/');
                    String desc2 = mapDesc(desc1);
                    if (!desc2.equals(desc1))
                        return desc2.replace('/', '.');
                } else {
                    s = mapPath(s);
                    if (s.equals(value)) {
                        boolean hasDot = s.indexOf('.') >= 0;
                        boolean hasSlash = s.indexOf('/') >= 0;
                        if (!(hasDot && hasSlash)) {
                            if (hasDot) {
                                s = replaceHelper(s.replace('.', '/')).replace('/', '.');
                            } else {
                                s = replaceHelper(s);
                            }
                        }
                    }
                }
                valueCache.put(value, s);
            }
            // TODO: add back class name to verbose message
            if (verbose && !s.equals(value))
                System.err.println("Changed \"" + value + "\" -> \"" + s + "\"");
            return s;
        } else {
            return super.mapValue(value);
        }
    }

    @Override
    public String mapInnerClassName(
            final String name, final String ownerName, final String innerName) {
        // The default method would try to be smart and find the inner class name
        // by looking at the $ sign. As the documentation calls out, this works in java
        // but not necessarily in other languages, including scala, where $ has multiple applications.
        // Since we actually don't care about remapping class names, only packages, we can ignore this
        return innerName;
    }

    private String replaceHelper(String value) {
        for (Wildcard wildcard : wildcards) {
            String test = wildcard.replace(value);
            if (test != null) {
                if (!test.equals(value))
                    this.modified = true;
                return test;
            }
        }
        return value;
    }

    @Override
    public boolean hasChanges() {
        return modified;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy