Eclipse RCP: как добавить внутренний класс при плетении
В моем приложении RCP я использую сервис WeavingHook для изменения стороннего класса во время загрузки. Я не могу вносить какие-либо изменения в сторонний код, поэтому моя ткацкая реализация должна работать с ним как есть.
У меня это работает, за исключением одного вопроса. Для одного из созданных мной модов я создал внутренний класс. Это должен быть внутренний класс, потому что он зависит от внутреннего класса, определенного в одном из суперклассов.
Теперь, когда сторонний код загружает измененный класс, у него нет файла для внутреннего класса в его classpath, который доступен только в плагине, где я реализую ткачество. Если сторонний плагин объявил политику друзей registered
Я мог бы добавить Eclipse-RegisterBuddy
директива к манифесту, но это не так.
Есть ли способ с помощью ткачества, чтобы позволить модифицированному классу иметь доступ к внутреннему анонимному классу, который был добавлен через бинарное переплетение?
Единственное решение, которое я могу придумать, - это скопировать файл внутреннего класса в каталог bin / стороннего кода, когда происходит двоичное переплетение. Похоже, взломать, так что я надеюсь, что есть лучший способ.
2 ответа
Вот решение, которое работает, копируя файл класса внутреннего класса в каталог bin / стороннего плагина, содержащего цель weave. Если кто-нибудь знает менее хакерский способ сделать это, пожалуйста, оставьте отдельный ответ.
Редактировать: сделать оригинальное решение более общим, обнаружив все внутренние файлы классов для создаваемого класса, и скопировав их в целевой модуль переплетения, если он там отсутствует или если размеры отличаются.
public class Activator extends Plugin {
...
public static final String CLASS_FILE_ROOT = "bin/";
/**
* Binary weaving of a class file may result in the addition or modification of
* inner class files in this plug-in which need to be placed on the classpath
* of the plug-in containing the weave target. This method compares inner class
* files for a given class in this plug-in with inner class files in the plug-in
* that contains the weave target. Any inner class files missing from the weave
* target or with a different size than the corresponding file in this plug-in
* are copied to the appropriate package directory in the weave target plug-in.
*
* @param bundle the bundle of the class being woven
* @param className the fully qualified name of the class being woven
* @return true if no errors were encountered in copying the inner class files
*/
private boolean checkInnerClassFileSync(Bundle bundle, String className) {
className = className.replaceAll("\\.", "/");
String classDir = CLASS_FILE_ROOT + className.substring(
0, FilenameUtils.indexOfLastSeparator(className));
for (String innerClass : getInnerClassNames(classDir, FilenameUtils.getName(className))) {
try {
URL srcUrl = getBundle().getEntry(classDir + "/" + innerClass);
File srcFile = new File(FileLocator.resolve(srcUrl).toURI());
URL destUrl = bundle.getEntry(classDir);
File destDir = new File(FileLocator.resolve(destUrl).toURI());
File destFile = FileUtils.getFile(destDir, innerClass);
if (srcFile.isFile() && (!destFile.exists() || destFile.length() != srcFile.length())) {
FileUtils.copyFile(srcFile, destFile);
}
} catch (IOException | URISyntaxException e) {
logger.log(Level.ERROR, "An error occurred while trying to copy an inner class file to the weave target bundle", e);
return false;
}
}
return true;
}
/**
* Get the class files representing inner classes defined in a specified class and
* found in a specified directory.
*
* @param dir a sub-directory containing a class file for <code>className</code>
* and possibly one or more class files representing inner classes defined in
* <code>className</code>
* @param className a class whose class file is found in <code>dir</code>
* @return class files names representing every inner class defined in
* <code>className</code> and found in <code>dir</code>.
*/
private String[] getInnerClassNames(String dir, String className) {
List<String> classNames = new ArrayList<String>();
try {
File classDir = new File(FileLocator.resolve(getBundle().getEntry(dir)).toURI());
for (String fName : classDir.list()) {
Pattern p = Pattern.compile("^" + className + "\\$.+$");
Matcher m = p.matcher(fName);
if (m.matches()) {
classNames.add(fName);
}
}
return classNames.toArray(new String[0]);
} catch (URISyntaxException | IOException e) {
logger.log(Level.ERROR, "An error occured while scanning for inner class files", e);
return classNames.toArray(new String[0]);
}
}
....
}
Вы можете динамически добавить Import-Package при создании класса через WeavingHook с getDynamicImports()
метод:
wovenClass.getDynamicImports().add("my.package.to.import");
Этот пакет, очевидно, должен быть экспортирован другим пакетом.