/*
* Nocturne
* Copyright (c) 2015-2016, Lapis <https://github.com/LapisBlue>
*
* The MIT License
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package blue.lapis.nocturne.util.helper;
import static blue.lapis.nocturne.util.Constants.CLASS_PATH_SEPARATOR_CHAR;
import static blue.lapis.nocturne.util.Constants.CLASS_PATH_SEPARATOR_PATTERN;
import static blue.lapis.nocturne.util.Constants.INNER_CLASS_SEPARATOR_CHAR;
import static blue.lapis.nocturne.util.Constants.INNER_CLASS_SEPARATOR_PATTERN;
import static blue.lapis.nocturne.util.Constants.Processing.CLASS_PREFIX;
import static blue.lapis.nocturne.util.Constants.Processing.CLASS_REGEX;
import static blue.lapis.nocturne.util.Constants.Processing.CLASS_SUFFIX;
import static blue.lapis.nocturne.util.Constants.Processing.DELIMITER;
import static blue.lapis.nocturne.util.Constants.Processing.MEMBER_PREFIX;
import static blue.lapis.nocturne.util.Constants.Processing.MEMBER_REGEX;
import static blue.lapis.nocturne.util.Constants.Processing.MEMBER_SUFFIX;
import blue.lapis.nocturne.Main;
import blue.lapis.nocturne.jar.model.attribute.MethodDescriptor;
import blue.lapis.nocturne.jar.model.attribute.Type;
import blue.lapis.nocturne.util.MemberType;
import java.util.ArrayList;
import java.util.List;
import java.util.regex.Matcher;
/**
* Static utility class for certain string manipulator functions.
*/
public final class StringHelper {
// class format is ^NOCTURNE+name^
// member format is %NOCTURNE+TYPE-name-descriptor%
public static String getProcessedName(String qualName, String descriptor, MemberType memberType) {
if (memberType == MemberType.CLASS) {
return CLASS_PREFIX + qualName + CLASS_SUFFIX;
}
return MEMBER_PREFIX + memberType.name() + DELIMITER + qualName + DELIMITER + descriptor + MEMBER_SUFFIX;
}
public static String getProcessedDescriptor(MemberType memberType, String desc) {
switch (memberType) {
case FIELD: {
if (desc.startsWith("L") && desc.endsWith(";")) {
String typeClass = desc.substring(1, desc.length() - 1);
if (Main.getLoadedJar().getClass(typeClass).isPresent()) {
return "L" + getProcessedName(typeClass, null, MemberType.CLASS) + ";";
}
}
break;
}
case METHOD: {
if (!desc.contains(MEMBER_PREFIX)) { // if this condition is true then it's already been processed
MethodDescriptor md = MethodDescriptor.fromString(desc);
List<Type> newParams = new ArrayList<>();
for (Type param : md.getParamTypes()) {
if (param.isPrimitive()) {
newParams.add(param);
} else {
String typeClass = param.getClassName();
if (Main.getLoadedJar().getClass(typeClass).isPresent()) {
newParams.add(new Type(getProcessedName(typeClass, null, MemberType.CLASS),
param.getArrayDimensions()));
} else {
newParams.add(param);
}
}
}
Type returnType = md.getReturnType();
if (!returnType.isPrimitive()) {
String typeClass = returnType.getClassName();
if (Main.getLoadedJar().getClass(typeClass).isPresent()) {
returnType = new Type(getProcessedName(typeClass, null, MemberType.CLASS),
returnType.getArrayDimensions());
}
}
Type[] newParamArr = new Type[newParams.size()];
newParams.toArray(newParamArr);
MethodDescriptor newMd = new MethodDescriptor(returnType, newParamArr);
return newMd.toString();
}
break;
}
default: {
throw new AssertionError();
}
}
return desc;
}
public static String getUnprocessedName(String processed) {
boolean clazz = processed.startsWith(CLASS_PREFIX);
Matcher matcher = (clazz ? CLASS_REGEX : MEMBER_REGEX).matcher(processed);
if (!matcher.find()) {
throw new IllegalArgumentException("String " + processed + " is not a processed member name");
}
return matcher.group(clazz ? 1 : 2);
}
public static String resolvePackageName(String qualifiedClassName) {
return qualifiedClassName.indexOf(CLASS_PATH_SEPARATOR_CHAR) != -1
? qualifiedClassName.substring(0, qualifiedClassName.lastIndexOf(CLASS_PATH_SEPARATOR_CHAR))
: "";
}
public static String unqualify(String qualified) {
String unqual = qualified;
if (unqual.contains(CLASS_PATH_SEPARATOR_CHAR + "")) {
String[] arr = CLASS_PATH_SEPARATOR_PATTERN.split(unqual);
unqual = arr[arr.length - 1];
}
if (unqual.contains(INNER_CLASS_SEPARATOR_CHAR + "")) {
String[] arr = INNER_CLASS_SEPARATOR_PATTERN.split(unqual);
unqual = arr[arr.length - 1];
}
return unqual;
}
public static boolean isJavaClassIdentifier(String str) {
str = CLASS_PATH_SEPARATOR_PATTERN.matcher(str).replaceAll("");
str = INNER_CLASS_SEPARATOR_PATTERN.matcher(str).replaceAll("");
return isJavaIdentifier(str);
}
public static boolean isJavaIdentifier(String str) {
if (str.length() == 0 || !Character.isJavaIdentifierStart(str.charAt(0))) {
return false;
}
for (int i = 1; i < str.length(); i++) {
char c = str.charAt(i);
if (!Character.isAlphabetic(c) && !Character.isDigit(c) && !Character.isJavaIdentifierPart(c)) {
return false;
}
}
return true;
}
}