Skip to content

Latest commit

 

History

History
241 lines (180 loc) · 12.9 KB

README.md

File metadata and controls

241 lines (180 loc) · 12.9 KB

Populace-core

This page provides information on how to use and customise the core graph walking and populating features of Populace

Graph Walking

Put simply, you can use GraphWalker to walk the fields and elements of an object graph.

  • Fields An instance of a class exposes the fields declared by the class definition.
  • Elements Where a field is a container type e.g. List, Map, etc, then the field can have child elements.

You provide Visitors that will be called for each field/element that is encountered. You can optionally control what fields and elements are included / excluded from the walk by installing Filters, and you can control what fields and elements are exposed from a specific type by installing Inspectors.

Filters

Filters can be installed to stop specific fields or elements from being walked, (or visited). Field and Element filters are installed when building a walker instance, as shown below:

GraphWalker walker = GraphWalker.newBuilder()
    .withFieldFilter(field -> field.getName().contains("includeMe")
    .withElementFilter(element -> element.getValue() != null)
    .build();

FieldFilters defines some common field filters e.g. to exclude certain types of fields. By default the walker excludes static and transient fields, though this can be overridden.

Filters can be combined using the standard Predicate logical operators, e.g. AND, OR and NOT.

Instance tracking

Populace comes with a special type of field and element filter that can be used to avoid circular-references, which would otherwise cause a stack-overflow, and to stop the same instance, which may be set on multiple fields or elements, being visited more than once.

The InstanceTracker is not installed by default, but can be installed as shown below.

Note: The instance tracker is stateful and mutable. It is therefore one of the few parts of Populace that is not thread-safe. Also, multiple calls to walk on the same walker will carry over the list of visited instances from one walk to the next. This is generally not desired. If not desired, ensure you call clear on the instance tracker between each walk.

InstanceTracker tracker = new InstanceTracker();
GraphWalker.Builder builder = GraphWalker.newBuilder();
GraphWalker walker = builder
    .withFieldFilter(builder.getFieldFilter().and(tracker.getFieldFilter()))
    .withElementFilter(builder.getElementFilter().and(tracker.getElementFilter()))
    .build();

Inspectors

Inspectors are similar to Filters in some ways. Where as Filters, given the list of fields and elements a type exposed, controls which fields and elements should be walked, Inspectors actual define the set of exposed fields/elements.

Inspectors should expose the fields and elements of a type and should be re-usable by others who use the same type. Avoid implementing inspectors to expose only a subset. On the other hand, filters, or combinations of filters, can be used to customise a walker to a specific use-case.

Populace defines inspector to cover most common types. (If you feel there is a inspector missing for some common type then please raise a ticket, or even better a PR). You may need to write your own inspectors, especially if you have custom container types that are not derived from the standard List, Set or Map interfaces.

Inspectors can be installed to handle a specific type, but they can also be installed so that they handle any types derived from a particular super-type, or any types that belong to a specific package. Populace also supports default inspectors for array and non-array types. See Registering Customisations for an explanation of the different levels. Inspectors are installed when building a walker:

final GraphWalker.Builder builder = GraphWalker.newBuilder();
final GraphWalker walker = builder.withInspectors(builder.inspectors()
        .withPackageInspector("java.util", TerminalInspector.INSTANCE)  // Don't look inside java.until types
        .build())
    .build();

Visitors

The visitors you pass to the walk function will be called back as each non-filtered field and element is visited. The visitor can mutate the value of the field or element if required, using the setValue method.

Visitors are are passed to the walk method:

walker.walk(typeToWalk, fieldVisitor, elementVisitor);

Graph Populating

Graph population builds on the graph walking functionality, adding the ability to populate and/or mutate existing fields and elements as they are encountered.

You can control this process by installing Mutators, which are responsible for populating/mutating values, and InstanceFactories, which are responsible for providing instances of types where needed.

Mutators

Mutators allow you to install custom code to control how a type is mutated, giving you complete control over how your object graphs are populated.

Populace comes complete with standard Mutators that can be composed to achieve most tasks, and you are free to implement your own as needed. By default, a new populator comes pre-configured with a sensible set of mutators that will handle most common types, though these can be overridden if needed.

Mutators can be installed to mutate specific types, subtypes of some known super-type, and any type belonging to a package, plus default mutators for array and non-array types can also be configured. See Registering Customisations for an explanation of the different levels. See below for an example:

GraphPopulator.Builder builder = GraphPopulator.newBuilder();
GraphPopulator populator = builder
    .withMutators(builder.mutators()
        .withSpecificMutator(String.class, new CustomStringMutator())
        .build())
    .build();

Instance Factories

Instance Factories allow you to install custom code to control how a type is created. Populace needs this to allow fields that are currently null, or container types that are currently empty, to be populated.

Populace comes complete with factories for standard types and any type that can be constructed using its default constructor, for any other type you will need to implement and register an instance factory.

Instance factories can be installed for specific types, subtypes of some super-type, any type belonging to a package, and default factories for array and non-array types can also be configured. See Registering Customisations for an explanation of the different levels.

In addition, a special type of instance factory, called a NullObjectStrategy, can be installed to handle any null Object s i.e. fields or elements where no type information is available, i.e. they are either of type Object or their type information can't be resolved, and the current value is null. In such situations Populace does not have enough information to populate the element, but you may chose to handle this however you like by installing a custom NullObjectStrategy.

In addition, a special type of instance factory, called a NullObjectStrategy, can be installed to handle any null objects encountered i.e. null fields or elements where no type information is available because either their compile-type type is Object or an unresolvable TypeVariable. In such situations Populace does not have enough information to populate the element, but you may chose to handle this however you like by installing a custom NullObjectStrategy. By default, Populace logs a warning.

Instance factories can be installed as shown below:

GraphPopulator.Builder builder = GraphPopulator.newBuilder();
GraphPopulator populator = builder
   .withInstanceFactories(builder.instanceFactories()
        .withSpecificFactory(MyType.class, new MyTypeFactory())
        .withNullObjectStrategy(new CustomNullObjectStrategy())
        .build())
   .build();

Registering Customisations

Many of the customisations available in Populace, such as Mutators, Inspectors etc, are registered at one of several levels. These levels are, from most specific to least, listed below:

  • specific types i.e. you can register a handler from a specific class e.g. String, or a specific parameterised type e.g. MyType<Integer>, or a specific array type e.g. long[]. Such handlers will be used in preference to any other handler.
  • super types i.e. you can register a handler for a super type such as List and it will be used to handle any derived types that do not have a more specific handler installed.
  • packages i.e. you can register a handler to handle any types that belong to a specific package e.g. java.lang.util. The handler will be used for any type that belongs to the package, including child packages, that do not have a more specific handler installed.
  • default this is the default handler to be use if the type is not an array type and no other, more specific, handler is installed.
  • array default arrays are treated slightly differently. The default array handler will be used where no specific handler is registered for an array type.

To understand what type informamtion will be used at runtime to look up suitable handlers you will first need to understand how Populace determines runtime type information.

Runtime-type resolution

Populace determines the runtime type of a field or element by making at both the compile-time and runtime type information available. Fields and elements with a null value are treated differently to non-null values, as non-null values have additional runtime type information.

Note to make use of runtime time information the fields Populace is walking need to be accessible. It is strongly advised that you install the SetAccessibleFieldVisitor to ensure fields are accessible. However, this is optional as Populace does support running with a security manager that would not allow the field to be made accessible. In such a situation Populace will only use compile-time type information.

Null fields

When the value of the field is null the only information available is the compile type type of the field.

If the compile time type of the field is a normal class, e.g. long or List, then Populace will use this type information when looking up configured customisations.

If the compile time type of the field is a parameterised type, e.g. Map<String,Integer>, then Populace will use the full parameterised type when looking up customisations. In addition, if any of the type arguments are type variables or wildcards, e.g. consider the field 'example' in the following code snippet below, Populace will use all available type information in parent/containing classes to resolve the type variables and any bounds to concrete types.

class SomeClass<T> {
    private final List<T> example;
}

If the compile time type of the field is a type variable, e.g. consider field 'example' in the code snippet below, then Populace will again use all type information available in parent/containing classes to resolve this to a concrete type.

class SomeClass<T> {
    private final T example;
}

Non-null fields

When the value of the field is not null, and the field is accessible, then Populace has additional type information available to it, and it will use this information to resolve the type of the field to a more specific type, if possible.

Because Populace can only make use of the fields runtime type information if the field is accessible it is strongly advised that you install the SetAccessibleFieldVisitor, to ensure all fields are accessible, unless there is a good reason not to. Without SetAccessibleFieldVisitor installed, Populace will not be able to use runtime type information. In which case, the logic for determining type information is the same as if the field was null.

If the runtime type of the field is a normal class, then the more specific runtime type will be used.

If the runtime type of the field is a type that has type arguments, then Populace will parameterise the runtime type using all available type information in the field's compile-time type and parent/container type information, e.g. if a field type is List<String> and the field value is an instance of ArrayList, then Populace will resolve the type to ArrayList<String>. Populace will also attempt to resolve all type variables and bounds.

Null elements

Similar to Null Fields, null elements have only the compile time type information of their container. Populace will attempt to resolve this type information as much as possible, before using it to look up customisations.

Non-null elements

As per Non-null Fields, non-null elements will also make use of the runtime type information available to build a more specific type to use to look up customisations.