DEV Community

Charles Koffler
Charles Koffler

Posted on • Edited on

Clprolf — Official Documentation

Introduction

Clprolf("Clear PROgramming Language and Framework") is Java (or C#) framework.

Its goal is to make certain object-oriented programming best practices explicit, without introducing heavy architecture or a steep learning curve.

Clprolf is based on a simple idea:

A class should clearly express its primary role.

The framework helps to:

  • separate business logic from technical code,
  • limit architectural drift,
  • make inheritance more coherent,
  • improve system readability.

In Java (ArchUnit)

In the Java ecosystem, components are marked using annotations:

import org.clprolf.framework.ClAgent;

@ClAgent
public class CarImpl implements Car {
    // Agent logic...
}

Enter fullscreen mode Exit fullscreen mode

In C# (ArchUnitNET)

In the .NET ecosystem, the strict equivalent uses C# Attributes enclosed in brackets [...] placed directly above the class:

using Clprolf.ArchUnitNet.Attributes;

namespace MyApp.Agents {

    [ClAgent]
    public class Car : ICar
    {
        // Agent logic...
    }

}
Enter fullscreen mode Exit fullscreen mode

💡 Note for Framework Users (.NET vs Java)

  • Syntax: Java's @Annotation becomes [Attribute] in C#.

I) The Two Fundamental Principles

Clprolf is built around two central principles.


1. A class is either business-oriented or technical

Every class belongs to one of the following worlds:

Business / Domain World

The class represents a business or conceptual responsibility.

Examples:

  • order management,
  • business logic,
  • simulation,
  • functional orchestration.

These classes are declared using:

@ClAgent
Enter fullscreen mode Exit fullscreen mode

Technical World

The class performs a technical task:

  • database access,
  • networking,
  • file handling,
  • display,
  • infrastructure.

These classes are declared using:

@ClWorker
Enter fullscreen mode Exit fullscreen mode

2. Inheritance must preserve the domain

A class should only inherit from another class belonging to the same conceptual domain.

Otherwise:

composition should be used.

This principle prevents incoherent hierarchies and mixed responsibilities.


II) Class Types

Clprolf contains only three class types.


II.1) ClAgent

Represents a business or conceptual class.

An agent:

  • contains business logic,
  • orchestrates processes,
  • makes decisions,
  • avoids heavy technical code.

Note: entities and DTOs are typically classified as agents, since they represent domain data.

Example:

@ClAgent
public class OrderProcessor {

    private OrderRepository repository;

    public void process(Order order) {
        if(order.total() <= 0) {
            throw new Error();
        }
        repository.save(order);
    }
}
Enter fullscreen mode Exit fullscreen mode

II.2) ClWorker

Represents a technical class.

A technical class is primarily intended to support agent classes rather than be organized around a class domain.
Workers provide technical support and infrastructure services. They may coordinate or use low-level agent classes such as File, Connection, Random, Logger, or Parser, but unlike those classes, a worker is not organized around a class domain of its own.
Instead, it exists to support other components through technical mechanisms, infrastructure access, application startup, operating-system interaction, or similar responsibilities.

A worker:

  • provides technical support,
  • manages infrastructure and execution mechanisms,
  • contains technical code.

Example:

@ClWorker
public class OrderRepository {

    public void save(Order order) {

        // database access

    }
}
Enter fullscreen mode Exit fullscreen mode

II.3) ClDraft

An object without a defined role.

Used:

  • during prototyping,
  • during refactoring,
  • when the role is not yet clear.

Example:

@ClDraft
public class TemporaryManager {
}
Enter fullscreen mode Exit fullscreen mode

ClDraft enables a flexible approach similar to classical OOP.


II.4) Primary Domain and Technical Code

Clprolf encourages moving as much technical code as possible from agent classes into worker classes.

However, an agent may contain a reasonable amount of technical code when doing so improves simplicity or readability.

An agent always has a primary domain representing its central responsibility.

Secondary responsibilities may exist as long as they remain consistent with that primary domain.


II.5) Freedom of Interpretation and framework Recommendations

The choice between agent and worker is left to the developer.

Some responsibilities may be interpreted differently depending on the adopted architectural vision.

For example, a connection may be represented:

  • as an agent if viewed as a functional abstraction;
  • or as a worker if viewed as a purely technical mechanism.

However, in these cases, Clprolf recommends using an agent. For example, for a Connection class:

  • an agent to represent the connection,
  • and delegate technical code to one or more worker classes.

III) Inheritance

A class only inherits from another class within the same domain.


Valid Example

@ClAgent
public class Animal {
}

@ClAgent
public class Dog extends Animal {
}
Enter fullscreen mode Exit fullscreen mode

Discouraged Example

@ClWorker
public class ClientRepository {
}

@ClAgent
public class Dog extends ClientRepository {
}
Enter fullscreen mode Exit fullscreen mode

Here, the domains are incompatible.

Composition should be used instead.

Inheritance can be forced using @ClBypass above the class.


IV) Flexibility

Clprolf is flexible.

The developer therefore keeps their freedom:

  • mixing responsibilities when necessary,
  • progressive migration,
  • compatibility with existing code,
  • while always maintaining a primary domain.

The framework mainly acts as:

a structural guide.


V) Interfaces

In Clprolf, interfaces are viewed as:

abstract forms of inheritance.

They therefore participate in the structural continuity of the system.

ClFamily = primary family interface
ClTrait  = trait, shared capability between families
ClDraft = unrestricted interface
Enter fullscreen mode Exit fullscreen mode

In Clprolf, interfaces are not viewed as simple technical contracts.

Both extends and implements relationships are considered genuine conceptual inheritance relationships.


V.1) ClFamily

An interface representing an abstract family.

Used for:

  • polymorphism,
  • decoupling,
  • implementation variants.

Family interfaces also have a target role:

  • ClAgent
  • or ClWorker

Example

@ClAgent
@ClFamily
public interface Animal {

    void eat(int quantity);

}
Enter fullscreen mode Exit fullscreen mode

The hierarchy of ClFamily interfaces naturally reflects the hierarchy of concrete classes.

@ClAgent
@ClFamily
public interface Horse extends Animal {

    void jump(int height);

}
Enter fullscreen mode Exit fullscreen mode

Which may lead to:

@ClAgent
public class AnimalImpl implements Animal { (...) }
Enter fullscreen mode Exit fullscreen mode
@ClAgent
public class HorseImpl extends AnimalImpl implements Horse { (...) }
Enter fullscreen mode Exit fullscreen mode

V.2) ClTrait

An interface representing a shared capability across multiple ClFamily.

Traits also use a target role:

  • ClAgent
  • ClWorker

Note: a @ClTrait may be annotated with both @ClAgent and @ClWorker.

This exception is reserved for genuinely cross-cutting traits that can be used by both agents and workers.


Business Example

@ClAgent
@ClTrait
public interface Payable {
    void pay();
}
Enter fullscreen mode Exit fullscreen mode

Technical Example

@ClWorker
@ClTrait
public interface Persistable {
    void save();
}
Enter fullscreen mode Exit fullscreen mode

V.3) ClFree

A generic interface without a specific role.

Allows flexibility.


Example

@ClFree
public interface ExternalApi {
}
Enter fullscreen mode Exit fullscreen mode

V.4) Interface Usage

In Clprolf, ClFamily interfaces are the equivalent of pure abstract classes.

They are intended to be implemented by one or more future Clprolf classes.

They therefore possess a target role (agent or worker).

A class may implement only one primary ClFamily at a time, and the role of the class must match the target role of the interface.

Clprolf therefore adopts a simple interface implementation model, in the same way that Java uses single class inheritance. Indeed, a ClFamily is always the structural reflection of its implementation. This notably enables systematic loose coupling.

However, multiple interface implementation is not removed; it is simply moved to the Family implemented by the class.

This family interface may itself inherit from multiple Family and/or Trait` interfaces.

As a result, interfaces that would otherwise have been implemented directly by the class are grouped at the level of its primary Family.

Clprolf therefore preserves the expressive power of multiple interface inheritance while maintaining a simple and coherent structure for concrete classes.


Trait interfaces express a capability shared among multiple Family interfaces.

A Trait therefore represents a cross-cutting trait shared by several families.

Normally, a Trait may only be inherited by a Family, not directly by a class.

However, direct implementation of a Trait by a class remains tolerated in Clprolf, although discouraged.

text
Concrete Class
↓ implements
ClFamily
↓ inherits from
ClTrait

A Family may inherit from multiple Family and/or Trait.

A Trait may inherit only from other Trait, since a trait remains a trait.

Interface inheritance may still be forced using @ClInterfaceBypass (or @ClBypass when forcing inheritance across different target roles).


V.5) Note on Clprolf and the Interface Segregation Principle (ISP)

Clprolf inherently respects the ISP; it is simply a matter of adapting the design of your classes and interfaces using the appropriate families and traits:

`java
@ClAgent
@ClTrait
public interface Scanner {
void scan(Document doc);
}

@ClAgent
@ClTrait
public interface Fax {
void fax(Document doc);
}

@ClAgent
@ClTrait
public interface Printer {
void print(Document doc);
}

@ClAgent
@ClFamily
public interface OldPrinter extends Printer {

}

@ClAgent
@ClFamily
public interface ModernPrinter extends OldPrinter, Scanner, Fax {
}

@ClAgent
public class OldPrinterImpl implements OldPrinter {
// (...)
}

@ClAgent
public class ModernPrinterImpl implements ModernPrinter {
// (...)
}

VI) General Architecture

Clprolf naturally encourages a simple architecture.

`text
agent
↓ delegates to
worker
`

agent classes contain:

  • business rules,
  • decisions,
  • orchestration.

worker classes perform:

  • technical work,
  • system access,
  • machine operations.

An agent delegates technical code to one or more worker classes.

A worker serves the agent.


VIII) Purpose of the framework

Clprolf does not aim to replace classical OOP.

It aims to make certain important distinctions explicit:

  • business vs technical,
  • coherent inheritance vs composition,
  • primary responsibility of a class.

IX) ArchUnit Checker

An ArchUnit-based checker is available for the Clprolf Framework on GitHub. It is open-source and consists of two classes: ClprolfArchTest and ClprolfStrictArchTest. It validates semantic rules.
The rules in ClprolfStrictArchTest are optional. Likewise, it is easy to change the annotation names if you prefer a different vocabulary.

1. Java Version (ArchUnit)

Available on GitHub, the Java checker is open-source and centers around two main test classes:

  • ClprolfArchTest: Validates the standard and fundamental semantic rules of the framework.
  • ClprolfStrictArchTest: Gathers optional, more rigid constraints for demanding projects (such as forbidding a class from directly implementing a ClTrait). The checkers (both Java and .NET) also contain the definitions for Clprolf annotations (or attributes).

2. C# .NET Version (ArchUnitNET)

The port of the .NET extension is available and published on GitHub.

The Visual Studio solution (2022 to date) contains a project with the framework and ArchUnit rules, along with an xUnit project for the tests.
A third example project is also included. There are currently 8 mandatory tests and 4 strict tests.

Rules to Follow

clprolf_classes_must_not_mix_agent_and_worker:

A class cannot be annotated as both @ClAgent and @ClWorker.

agent_worker_inheritance_must_not_mix

A @ClWorker class cannot inherit from a @ClAgent class, and vice versa.

family_role_must_match_implementation

The target role of a @ClFamily interface must match the role of the implementing class (@ClAgent or @ClWorker). Bypassing is possible using @ClBypass.

(non-strict mode) trait_interface_role_must_match_direct_implementation

A class that directly implements a trait must have a compatible role (unless bypassed with @ClBypass). Forbidden in strict mode.

trait_interfaces_must_extend_only_trait_interfaces

@ClTrait interfaces can only inherit from other @ClTrait interfaces. Bypassing is possible using @ClInterfaceBypass.

clprolf_interfaces_must_have_target_role

@ClFamily interfaces must have exactly one target role: @ClAgent or @ClWorker.
@ClTrait interfaces must have at least one target role: @ClAgent, @ClWorker, or exceptionally both.

inheriting_interface_role_must_match_trait_interface_target_role

Interfaces (family or trait) that inherit from a trait must have a role compatible with that trait (unless bypassed with @ClBypass).

family_interface_target_role_must_match_inherited_family_interface

Family interfaces inherited by another family interface must have a compatible role, unless @ClBypass is used.

Stricter Rules:

optional_all_classes_should_have_clprolf_role

All classes must have a Clprolf role (@ClAgent, @ClWorker, or @ClDraft).

optional_all_interfaces_should_have_clprolf_role

All interfaces must have a Clprolf role (@ClFamily, @ClTrait, or @ClFree).

optional_class_should_not_implement_trait_directly

A class cannot directly implement a @ClTrait interface (unless @ClInterfaceBypass is used).

optional_class_must_implement_only_one_family_interface (OPTIONAL)

A Clprolf class can only implement a single @ClFamily interface. Bypassing is possible using @ClInterfaceBypass.

X) Clprolf and Existing Architectures

Clprolf is compatible with existing architectural approaches such as Domain-Driven Design (DDD), Model-View-Controller (MVC), Clean Architecture, Hexagonal Architecture, and others.

Rather than replacing these architectures, Clprolf acts as an additional layer between Object-Oriented Programming (OOP) and software architecture. Its purpose is to complement, clarify, and reinforce existing architectural principles by making class roles and inheritance relationships more explicit.

In this way, Clprolf helps improve architectural consistency while remaining fully compatible with established design practices.

XI) Summary

Clprolf introduces very few concepts.

Classes

`text
Agent
Worker
Draft
`

Interfaces

`text
Family
Trait
Free
`

Two Fundamental Rules

`text

  1. Separate business and technical concerns.
  2. Inherit only within the same domain. `

Top comments (0)