Skip to content

Latest commit

 

History

History
292 lines (209 loc) · 9.92 KB

README.adoc

File metadata and controls

292 lines (209 loc) · 9.92 KB

Pixie

Pixie replaces any scenario where you would use reflection to instantiate a Java object. Handles configuration, dependency injection and events.

Configuration

Pixie is bootstrapped from a plain properties file. Inside the properties is a complete definition of the system. Configuration files look similar to this:

jon=new://org.example.Person
jon.age = 46
jon.address=@home

home=new://org.example.Address
home.street=823 Roosevelt Street
home.city=River Falls
home.state=WI
home.zipcode=54022
home.color?=blue
home.country?=USA

Strict Validation for Accurate Configurations

Pixie enforces a strict validation mechanism where every property in the configuration file must correspond to an @Option or @Component in the respective class. This design ensures:

  • Error Prevention: Configuration typos or outdated options are caught immediately during component initialization, preventing unexpected behavior at runtime.

  • Clarity and Transparency: Users can trust that only explicitly defined options affect the behavior of the application, avoiding confusion about whether a setting is valid or not.

  • Ease of Maintenance: If an option is removed or renamed, the system fails fast, guiding users to correct their configuration rather than silently ignoring invalid entries.

Components

The sample configuration above references two classes that must be built:

  • org.example.Person

  • org.example.Address

Here is how they would be defined. First, let’s start with Address. Pixie would build this component first because it doesn’t have any references to any other components, so is the least likely to cause a circular reference.

public static class Address {
    private final String street;
    private final String city;
    private final State state;
    private final int zipcode;
    private final String country;

    public Address(@Option("street") final String street,
                   @Option("city") final String city,
                   @Option("state") final State state,
                   @Option("zipcode") final int zipcode,
                   @Option("country") @Default("USA") final String country) {
        this.street = street;
        this.city = city;
        this.state = state;
        this.zipcode = zipcode;
        this.country = country;
    }

    public String getStreet() {
        return street;
    }

    public String getCity() {
        return city;
    }

    public State getState() {
        return state;
    }

    public int getZipcode() {
        return zipcode;
    }

    public String getCountry() {
        return country;
    }
}

Notice all the fields are final. Pixie only supports constructor injection. This allows us to both create safe runtime objects whose fields are final, but it also gives us the ability to test our components without Pixie.

@Option

Also note that Pixie supports injecting configuration properties via @Option. The value of the annotation points to the particular property you want. It can be either a relative reference as shown above, or an absolute reference.

Options are first assumed to be relative, so given @Option("city") on a component whose name is jon, Pixie would imply jon. in the name and first look for this property:

  • jon.city

If this does not succeed, Pixie will simply look in the global configuration for:

  • city

Of course, many components may reference the same property city. Using jon.city would be the only way to target it specifically to the Person named jon

@Option types

Options may be String, int, float and other kinds of Java primitives. They also can be any object type that matches either of these rules:

  • Has a public constructor that takes String

  • Has a state method that takes String and returns an instance of itself

This allows Pixie to support Enums, classes like URI.create(String) and more.

@Default

In the above @Option("city"), if jon.city and city are not found, Pixie will look for an @Default annotation on the same parameter and use that for the default setting.

This allows Pixie configuration files to be small and only contain what the user has chosen to manage.

@Nullable

@Nullable is a qualifier used to indicate an optional parameter is null. This way, you don’t need to add a @Default annotation with a constant you will then have to test in the code.

For instance in @Option("city") @Nullable final String city, if city hasn’t been specified, city will be null.

Important
It is invalid to use both @Default and @Nullable.
Note
@Nullable can also be applied to component injections. Pixie will then look in the system instance if the component is available. If not, it won’t try to lazily create an instance to inject it.

@Name

@Name is used on a parameter to get the name of the current component injected.

public static class Person {

    private final String name;
    private final int age;

    public Person(@Name final String name,
                  @Option("age") final int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public int getAge() {
        return age;
    }
}

With the following configuration

alfred = new://Person
alfred.age = 20

john = new://Person
john.age = 52

nick = new://Person
nick.age = 75

With the Person class above and the configuration above, Pixie will instantiate 3 Person instances. The first instance will get alfred for the injected name, next instance will respectively receive john and nick.

@Component

One component may reference another component via annotating the respective constructor parameter with @Component this tells Pixie to look for a component of that specific name and type.

public static class Person {

    private final String name;
    private final int age;
    private final Address address;

    public Person(@Name final String name,
                  @Option("age") final int age,
                  @Component("address") final Address address) {
        this.name = name;
        this.age = age;
        this.address = address;
    }

    public String getName() {
        return name;
    }

    public int getAge() {
        return age;
    }

    public Address getAddress() {
        return address;
    }
}

In the original configuration example, our Person named jon is configured to need an address called @home.

jon.address=@home

The @ symbol tells Pixie that the configuration value points to another component named home.

If no Address component exists, Pixie will look to see if Address is a class that Pixie can build. If so, Pixie will attempt to create one on the fly hoping there are enough defaults and configuration to fully create the object. If not, Pixie will fail and the system will not start up.

No Property lookups

Pixie intentionally does not expose any getProperty style of methods that allow configuration values to be looked up. You must create a simple object with a constructor annotated with @Option and ask Pixie to create it.

This limitation is intentional so that configuration properties can only be referenced via strongly typed annotations, which means we can statically know the name and type every single available configuration property the system supports.

We don’t want to give up this advantage for the ease of doing string lookups.

This doesn’t cost us anything and in fact it adds considerably to properties management.

Let’s say we have good reason to create "global" properties. We’d normally feel compelled to prefix everything with pixie., however let’s imagine a comprimise where we use the module name as the prefix.

Say for example we have three modules:

  • pixie-system

  • pixie-core

  • pixie-openejb

Let’s now imagine this pattern as a very clever way to archive module-scoped properties.

system=new://pixie.org.tomitribe.SystemOptions
system.debug=true
system.licence=1234-2315123412-12316125
system.keystore=somepath.keys

core=new://com.tomitribe.pixie.core.CoreOptions
core.dateformat=YYYY-mm-dd
core.timeunit=NANOSECONDS
core.checkinterval=10 seconds

openejb=new://com.tomitribe.pixie.openejb.OpenEjbOptions
openejb.debug=true
openejb.database=MONITOR
openejb.entitymanager=INGORE

There are interesting points about the above pattern:

  • If a user specifies a property that doesn’t exist, an exception will be thrown. A common issue with normal properties is when the code that looked it up and acted upon it is deleted. There’s no indication to the user they may be attempting to use a "dead" property. Here, the user cannot be mislead by specifying a module property that does not exist.

  • Code remains clean. To reference the property in various places in the module you would need to get the respective "Options" class injected. If you do not have that module as a dependency, you cannot do this. In the above imaginary scenario, code in pixie-openejb can see OpenEjbOptions, CoreOptions and SystemOptions. However, code from pixie-system cannot see OpenEjbOptions or even CoreOptions.

  • You always know where to look. If a property doesn’t fit anywhere in particular, it goes into the module’s "Options" class. There’s no time wasted by over-thinking how to manage the property and where it belongs. Further, you can go to the Options class and do a "Find Usages" in the IDE to see who is using the property and how.

  • Easy refactoring. If you have more than one bit of code using the property and you wish to rename the property, there are no string usages of it to worry about. You can change its type or name very easily using regular refactoring features of the IDE. No string find-and-replace.

  • Easy Deprecation. It would be quite easy for us to add annotations to support deprecating properties in favor of new names. This could involve logging a warning to the user, updating the config and proceeding forward.