Hi Piotr,

Thank you very much for your detailed and quick help.

However, to tell the truth, I’m a bit confused. I’ve been waiting for a long 
time for Log4j to finally work according to the JPMS rules. But in your 
message, you talk about compile-time and, as I understood, the use of some 
plugin-processor (from your project pom):

          <annotationProcessorPaths>
            <path>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-plugin-processor</artifactId>
              <version>3.0.0-beta3</version>
            </path>
          </annotationProcessorPaths>

Doesn’t all this completely contradict JPMS? When we talk about JPMS, we’re 
talking about modules, layers, and services. For example, if we need some 
plugin, then we:
a) create a module with the implementation of this plugin b) add the created 
module to the appropriate layer.

This is very convenient and works perfectly (even when classes have 
annotations). For instance, I have a project where console commands can be 
added by user. I created:

public interface CommandFactory {
    Set<Class<? extends Command>> getSupportedCommands();
    <T extends Command> T command(Class<? extends Command> klass);
}

And any module that has commands implements this interface and declares 
`provides ... with ...` in module-info, along with `opens` for the package 
containing the commands. As a result, the command manager gets the command 
classes, can analyze them, and create a command if needed.

By the way, I noticed something seemed off when I saw code duplication in your 
project:

@Configurable(elementType = Appender.ELEMENT_TYPE, printObject = true)
@Plugin(ConsoleAppender.PLUGIN_NAME)
public final class ConsoleAppender...

and

PluginEntry.builder()
  .setKey("console")
.setClassName("org.apache.logging.log4j.core.appender.ConsoleAppender")
  .setName("Console")
  .setNamespace("Core")
  .setElementType("appender")
  .setPrintable(true)
  .get(),

Or am I mistaken (which is always possible) and misunderstood everything?

Best regards, Pavel

On 11/28/24 19:45, Piotr P. Karwasz wrote:
Hi Pavel,

On 28.11.2024 15:47, PavelTurk wrote:
Hello everyone.

I am trying to create my own appender using JPMS service in 3.0.0-beta3. This is my code:

Apparently two small bugs made their way to `3.0.0-beta3`([1] and [2]), which 
make compilation under JPMS challenging, but not impossible.

You can find a complete example of a custom appender on my repo:

https://github.com/copernik-eu/bug-reproducibility/tree/main/log4j-core-3.0.0-beta3-jpms

[1] https://github.com/apache/logging-log4j2/issues/3250
[2] https://github.com/apache/logging-log4j2/issues/3251
import org.apache.logging.log4j.plugins.model.PluginEntry;
import org.apache.logging.log4j.plugins.model.PluginService;

public class PluginProvider extends PluginService {

This is usually not needed, you just need to have `log4j-plugin-processor` on 
the annotation processor path. If you disabled the automatic annotation 
processor detection, you also need to add 
`org.apache.logging.log4j.plugin.processor.PluginProcessor` to the list of 
annotation processors.

The plugin processor generates a class named `<pluginPackage>.plugins.Log4jPlugins`, 
where `<pluginPackage>`:

* Can be provided explicitly using `-ApluginPackage=...`,

* If it is not provided explicitly, it is the common prefix of all your Log4j 
Core plugins.

If you just have `com.example.FooAppender`, the generated `PluginService` will 
be: `com.example.plugins.Log4jPlugins`.

[1] https://logging.apache.org/log4j/3.x/components.html#log4j-plugin-processor

Result:
Exception in thread "main" 
org.apache.logging.log4j.plugins.di.spi.ReflectionException
    at 
org.apache.logging.log4j.plugins@3.0.0.beta3/org.apache.logging.log4j.plugins.di.spi.ReflectionAgent.invokeMethod(ReflectionAgent.java:84)
    at 
org.apache.logging.log4j.plugins@3.0.0.beta3/org.apache.logging.log4j.plugins.di.DefaultInstanceFactory.lambda$registerBundleMethod$13(DefaultInstanceFactory.java:306)

Do you have a message and a cause for this exception? If this is not issue 
#3250 I mentioned before, please share the entire stack trace.


So, java dies in main class where I initializer my logger.
Could anyone say: 1) how to fix it 2) How to use custom Namespace for my 
appender in configuration?
For example, if I do PluginEntry.builder().setNamespace("FooNS")?

The purpose of namespaces is to distinguish between plugins that have the same name, but have 
different functions. For example the `<Replace>` configuration element is in the 
"Core" namespace, while the `%replace` pattern is in the `Converter` space. In 
general you shouldn't be concerned by namespaces:

* The plugins that correspond to XML elements in the configuration file MUST be in the 
"Core" namespace.

Piotr


---------------------------------------------------------------------
To unsubscribe, e-mail: log4j-user-unsubscr...@logging.apache.org
For additional commands, e-mail: log4j-user-h...@logging.apache.org



---------------------------------------------------------------------
To unsubscribe, e-mail: log4j-user-unsubscr...@logging.apache.org
For additional commands, e-mail: log4j-user-h...@logging.apache.org

Reply via email to