GenerateProperties.jl

Suite of macros to automatically generate getters & setters for object properties.
Author Kiruse
Popularity
1 Star
Updated Last
3 Years Ago
Started In
April 2021

GenerateProperties.jl

Suite of macros to automatically generate getters & setters for object properties, including corresponding hasproperty and propertynames methods.

Directly accessing properties of an object rather than defining an API can reduce namespace clutter (although one should definitely consider both approaches). Julia supports directly overriding default behavior of getproperty and setproperty!, but these methods are monolithic - i.e. you must differentiate between requested properties based on the property name. For large collections of virtual properties, this naturally separates properties' getters from their setters, reducing code clarity. These also do not add virtual properties to hasproperty and propertynames methods.

The @generate_properties macro provided by this library deals with all of these issues, albeit with some restrictions.

Usage

GenerateProperties exposes three macros: @generate_properties, @get, and @set, where the latter two do not have any use outside the scope of a @generate_properties block.

The library's syntax is rather simple:

@generate_propertyes <type> begin
    @get <property> = <code_body>
    @set <property> = <code_body>
end

Where you may, of course, order @get and @set declarations arbitrarily. As mentioned before, the @get and @set macros only function in tandem with the @generate_properties macro. Outside of one such they simply return nothing.

The syntax is designed to be reminiscent of "function assignment" syntax of Vanilla Julia. For examples, see below.

Note that code other than @get/@set will be silently swallowed.

Virtual Properties

GenerateProperties allows defining purely virtual properties. By example:

mutable struct MyStruct
    size::NTuple{2, Float64}
end

@generate_properties MyStruct begin
    @get width = self.size[1]
    @set width = self.size = (value, self.size[2])
    
    @get height = self.size[2]
    @set height = self.size = (self.size[1], value)
end

let inst = MyStruct((24, 25))
    inst.width  # 24
    inst.height # 25
    inst.size   # (24, 25)
    
    inst.width = 42
    inst.width # 42
    inst.size  # (42, 25)
    
    hasproperty(inst, :width)  # true
    hasproperty(inst, :height) # true
    hasproperty(inst, :size)   # true
end

Based on the enclosed @get and @set definitions, @generate_property automatically accumulates property names - both concrete fields and virtual properties - and automatically generates the corresponding property-related methods for your type.

@get and @set are designed to appear like property assignments - although they are really function bodies. Within code_body, one may refer to the current instance through the implicitly defined self::<type> argument. Within the code_body of a setter, one may access the implicitly defined value argument, which is the value to be assigned to the property.

Field Getters & Setters

Because self bypasses getproperty/setproperty!, you may override concrete fields as well. In doing so, omitting either a getter or setter will retain default Vanilla behavior. For example:

struct MyStruct
    size::NTuple{2, Float64}
    dirty::Bool
end
MyStruct(size = (0, 0)) = MyStruct(size, false)

@generate_properties MyStruct begin
    @set size = (self.dirty = true; self.size = value)
end

let inst = MyStruct()
    inst.size  # (0, 0)
    inst.dirty # false
    
    inst.size = (24, 25)
    inst.size  # (24, 25)
    inst.dirty # true
end

Library Restrictions

GenerateProperties does (currently) not support hybrid usage, i.e. manually defining getproperty/setproperty! while using @generate_property. You must commit to either approach.

Using self within a getter/setter bypasses getproperty & setproperty! and directly calls getfield and setfield! instead - otherwise this could cause a stack overflow.

Required Packages

No packages found.