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");

Этот пакет, очевидно, должен быть экспортирован другим пакетом.

Другие вопросы по тегам