jueves, 4 de febrero de 2010

Arquitectura de Software. Definiciones

El término "Arquitectura de software" esta ampliamente difundido pero no siempre correctamente definido. Lo mismo sucede con muchos otros términos asociados a la arquitectura de software. Aquí se presentan algunas definiciones de estos términos a fin de aclarar un poco el panorama:
Según IEEE 1471:

Arquitectura: Es la organización fundamental de un sistema comprendido en sus componentes, en las relaciones entre ellos y con el ambiente, y los principio que guían su diseño y evolución.

Descripcion de arquitectura: Es un conjunto de productos que documentan la arquitectura.

Vista de arquitectura: Es la representación de un sistema o parte de un sistema desde una perspectiva en particular.

"Viewpoint" arquitectural: Es un template que describe como crear y usar una vista de arquitectura. Una vista es el resultado de aplicar una viewpoint a un sistema en particular.

System Architecture: Es un conjunto de entidades, sus propiedades y relaciones entre ellas, que definen la estructura de un sistema.

Software Architecture: Es un conjunto de componentes de software, subsistemas, relaciones, interacciones y propiedades de cada uno de estos elementos, y un conjunto de principios que constituyen las propiedades y limitaciones fundamentales de un sistema de software.

Software architecting: Refiere al análisis, diseño, documentación, revisión, aprobación, y otras actividades relacionadas con la definición y management de una arquitectura de software.

Arquitectura de Referencia: Refiere a una definición de una arquitectura para un dominio en particular. Describe en alto nivel un conjunto de elementos involucrados en aplicaciones pertenecientes al dominio. Estos elementos deben intencionalmente ser dejados en alto nivel para poder aplicar a un gran número de sistemas. Las arquitecturas de referencia ahorran mucho tiempo de trabajo de arquitectos de software durante el diseño de una nueva aplicación del mismo dominio, además de proveer un lenguaje común entre arquitectos y desarrolladores, lo cual facilita la comunicación.

Atributos de la arquitectura de software.
  • Cultural adaptability: Soporte a múltiples lenguajes y culturas

  • Security: Previene accesos no autorizados

  • Data integrity: El sistema no corrompe datos ni suministra datos inconsistentes.

  • Maintainability:
    • Portability: ¿Puede el software ser portado a otras plataformas?
    • Changeability: Habilidad para agregar nuevas funciones y modificar funcionalidad existente
      • Fragility: pequeños cambios tienden a romper funcionalidad existentes
      • Rigidity: el software es difícil de modificar incluso cambios simples.
      • Duplication: el software con duplicación es mas difícil de mantener porque es grande y porque el cambio no es localizado.
    • Understandability: ¿Puede el software ser entendido de manera que sea facil realizar cambios?
    • Debugging: ¿El software soporta debugging?

  • Testability: ¿El software puede ser testeado eficientemente?

  • Usability: Es la medida de eficacia para la interface humana del software

  • Operational:
    • Availability: Porcentaje de tiempo en el que el sistema esta funcionando.
    • Manageability: Habilidad para inspeccionar y manejar componentes en ejecución
    • Upgradeability: ¿Puede el software ser actualizado mientras ejecuta?
    • Reliability: Habilidad para ejecutar funciones requeridas en un periodo de tiempo especificado.
    • Recoverability: Tiempo requerido para recuperarse de una falla.

  • Performance
    • Response: ¿Es la respuesta suficientemente rápida para escenarios de uso normal y extremo?
    • Scalability: La capacidad del sistema puede ser incrementada si es necesario?
    • Capacity/Throughput: Manejar grandes cargas y aún mantener la respuesta

  • Safety: El sistema no crea riesgos en el mundo real.
  • martes, 20 de octubre de 2009

    Pasar un Array a un Stored Procedure en Oracle

    Durante estos días tuve que hacer la prueba de pasar un array como parámetro a un Stored Procedure en un Oracle 8i desde Java. Al principio pense que era una tarea simple, pero luego encontre que no es tan así dado que me tope con 3 problemas que trataré de describir aquí.

    Pero primero dejo dos URL que tienen información de como hacer esto:
    * Muy buen foro
    * Documentación Oficial de Oracle

    Como mencione, al principio pense que usando JDBC podría lograr esto sin tener problemas pero me encontre con algunas limitaciones de JDBC y el tratamiento de los Array, así que tuve que trabajar directamente usando clases propietarias de Oracle. Quedando un código:


    DriverManager.registerDriver(new OracleDriver());

    String url = "jdbc:oracle:thin:@10.65.72.52:1521:BNPAIS";
    Connection conn = DriverManager.getConnection(url, "*******", "*******");

    String[] array = {"hola", "mundo"};

    ArrayDescriptor descriptor =
    ArrayDescriptor.createDescriptor( "T_LISTAVARCHAR2", conn );
    ARRAY array_to_pass =
    new ARRAY( descriptor, conn, array);

    CallableStatement ps = conn.prepareCall("{call TEST.TEST_ARRAY(?,?)}");
    ps.setArray(1, array_to_pass);
    ps.registerOutParameter(2, OracleTypes.VARCHAR);

    ps.execute();

    System.out.println( ps.getString(2) );

    conn.close();


    Ya en la base de datos, primero declaré el tipo de dato ARRAY que use para la prueba (un ARRAY de VARCHAR). Aquí tenia dos opciones, o bien declararlo dentro del package donde iba a tener mi SP o a nivel del schema.
    CREATE OR REPLACE TYPE t_listavarchar2 IS TABLE OF varchar2(300)


    Opte por la primer opción y enseguida me encontre con el sigueinte error:
    java.sql.SQLException: invalid name pattern:


    Basicamente, no me encontraba el tipo de dato declarado.:
    ArrayDescriptor descriptor = ArrayDescriptor.createDescriptor( "T_LISTAVARCHAR2", conn );


    Buscando la solución encontré 2 soluciones posibles, una (la mas simple) es declarar el tipo de datos a nivel del esquema y la otra es crear un sinónimo publico para el tipo (T_LISTAVARCHAR2) con los accesos y permisos correspondientes al usuario que intente acceder.

    Una vez solucionado este problema, me encontre por un par de errores de NoClassDefFoundError por falta de algunos JARs. Para solucionarlo además de classes12.jar puse en mi classpath: nls_charset12.jar

    Además aquí dejo otro link con información valiosa sobre los jar a colocar en nuestro classpath.

    La verdad que no fue complicado, pero tampoco fue tan simple como esperaba.

    lunes, 5 de octubre de 2009

    Typical Web Architecture.

    In computer security, a demilitarized zone, named after the military usage of the term and normally abbreviated to DMZ; also known as a Data Management Zone or Demarcation Zone or Perimeter Network, is a physical or logical subnetwork that contains and exposes an organization's external services to a larger, untrusted network, usually the Internet. The purpose of a DMZ is to add an additional layer of security to an organization's Local Area Network (LAN); an external attacker only has access to equipment in the DMZ, rather than the whole of the network.

    See more: DMZ - Wikipedia

    A DMZ is typically used to locate servers that need to be accessed from outside, like e-mail servers, Web and DNS.

    Here is a Dual firewalls DMZ typical for enterprise web applications:



    Typically, the DMZ is located between two firewalls and connects these.
    The first firewall (also called the "front-end" firewall) must be configured to allow both traffic destined both to the DMZ as well as to the internal network. The second firewall (also called "back-end" firewall) allows only traffic from the DMZ to the internal network. The first firewall handles a much larger amount of traffic than the second firewall.

    The application server and database are protected with this schema

    viernes, 20 de abril de 2007

    OT-Rules

    Execute business validations using OT Rules


    Abstract
    OT Rules is a simple rule engine that focuses on execution, validation and business rules composition. One of the main advantages of using OT Rules is that you can define business validations and keep them isolated from business logic so you can add, remove, change or compose these rules and new rules in a simple, flexible way without affecting the business logic. In this article, we will apply these validations, enable, disable and reuse them in different methods and add new validations without changing the business logic methods.

    How to create a rule.
    Rule creation is a simple process, you just have to create a new class that implements or extends some of the following interfaces or classes:

    # net.sf.opentranquera.rules.Rule: Base Interface for all the rules.
    # net.sf.opentranquera.rules.BusinessRule: Business rule specific Interface
    # net.sf.opentranquera.rules.CompositeRule: Composite rule specific Interface
    # net.sf.opentranquera.rules.AbstractRule: Can be used in different rule types
    # net.sf.opentranquera.rules.AbstractBusinessRule: Useful for creating business rules
    # Extend some of the existing rules in OT Rules so enhancing its behavior.

    In our example, we will extend net.sf.opentranquera.rules.AbstractBusinessRule in order to create different business validation rules.

    Example application
    Let's consider that we have to develop an application that makes an electronic fund transfer between different accounts. We can identify several services:
    # Fund transfer from one account to other
    # Get the balance of an existing account.

    We can identify here business validations that have to be considered when executing these services, for instance, when we are executing an electronic fund transfer or when we get an account balance we must validate that the accounts are valid or exist. To implement these validations, we are going to use OT Rules because:
    * Validations can change without changing the business logic
    * We could require to disable the validations in some particular environments
    * We could add new rules as the business change or evolve
    * We need flexibility to compose validations
    * We need to reuse the business validation along the project and not rewrite code.

    Note: in our example, for simplicity sake, we are going to use an in-memory java.util.Map . the configuration of the example application will be based on SpringFramework.

    Creating the rules.
    We have identified the following business rules:
    # DifferentAccountsRule: checks that both accounts source and destination are different.
    # AccountsExistsRule: Validates that both accounts exist. Uses AccountExistsRule to check each account.

    This is the rules source code:

    public class DifferentAccountRule extends AbstractBusinessRule {

    /*
    * @see net.sf.opentranquera.rules.Rule#evaluate()
    */
    public RuleResult evaluate() {
    Transfer transfer = (Transfer) this.getEvaluatedObject();
    if( transfer.getCreditAccount().equals(transfer.getDebitAccount()) )
    return this.getError("The accounts are equal");
    return this.getSuccess();
    }

    }

    public class AccountsExistsRule extends AbstractBusinessRule {

    private BusinessRule rule;

    /*
    * @see net.sf.opentranquera.rules.Rule#evaluate()
    */
    public RuleResult evaluate() {
    Transfer transfer = (Transfer) this.getEvaluatedObject();
    AbstractCompositeRuleResult result = new AbstractCompositeRuleResult(true) {
    };

    this.rule.setEvaluatedObject(transfer.getDebitAccount());
    RuleResult r1 = this.rule.evaluate();

    this.rule.setEvaluatedObject(transfer.getCreditAccount());
    RuleResult r2 = this.rule.evaluate();

    result.addResult( r1 );
    result.addResult( r2 );
    result.setSuccessful(r1.isSuccessful() && r2.isSuccessful());

    return result;
    }

    public void setRule(BusinessRule rule) {
    this.rule = rule;
    }
    }

    public class AccountExistsRule extends AbstractBusinessRule {

    private AccountDao dao;

    /*
    * @see net.sf.opentranquera.rules.Rule#evaluate()
    */
    public RuleResult evaluate() {
    String account = (String)this.getEvaluatedObject();

    // Verify that the account exists.
    if(this.dao.getAccount(account) == null)
    return super.getError("The account " + account + "does not exist.");
    return this.getSuccess();
    }

    public void setDao(AccountDao dao) {
    this.dao = dao;
    }
    }

    Now, we have to configure the rules and the service being used, in this example, SpringFramework (there is also another configuration format using XML, provided out-of-the-box by OT Rules):

    <bean id="service" class="net.sf.opentranquera.samples.rules.TransferServiceImpl">
    <property name="evaluator" ref="ruleEvaluator"/>
    </bean>

    <bean id="ruleEvaluator" class="net.sf.opentranquera.rules.RuleEvaluator">
    <property name="rules">
    <map>
    <entry key="transfer"><ref local="transferRule"/></entry>
    <entry key="balance"><ref local="accountExistsRule"/></entry>
    </map>
    </property>
    </bean>

    <bean id="transferRule" class="net.sf.opentranquera.rules.logical.AndRule">
    <constructor-arg index="0">
    <list>
    <bean class="net.sf.opentranquera.samples.rules.DifferentAccountRule"/>
    <bean class="net.sf.opentranquera.samples.rules.AccountsExistsRule">
    <property name="rule" ref="accountExistsRule"/>
    </bean>
    </list>
    </constructor-arg>
    <constructor-arg index="1" value="true"/>
    </bean>

    <bean id="accountExistsRule" class="net.sf.opentranquera.samples.rules.AccountExistsRule">
    <property name="dao" ref="dao"/>
    </bean>

    <bean id="dao" class="net.sf.opentranquera.samples.rules.AccountDao"/>

    As we can see iin the example, an evaluator gets injected in the service. The evaluator is net.sf.opentranquera.rules.RuleEvaluator typed. The ruleEvaluator is configured like a Spring-Bean, being injected the different rules that it can evaluate, in the example two: Transfer and Balance.
    Each rule can be, like the transfer rule, a CompositeRule, that contains other rules inside. Now, from the service implementation or with an interceptor, you can call the evaluate in the RuleEvaluator to execute the rules. This example is inside the method, but using an interceptor is recommended:

    public boolean transfer(Transfer transfer) throws TransferException {
    this.evaluator.setEvaluatedObject(transfer);
    RuleResult result = this.evaluator.evaluateRule("transfer");
    if(!result.isSuccessful())
    throw new TransferException(result.getMessages());

    // TODO logic ..
    return true;
    }

    Add a new validation using composition.
    Now, the features of OT Rules come to light while adding and modifying new rules and execute business validations without changing the source code (in this case, the service implementation). We are going to add a new business rule that checks if there is enough funds in the source account. First of all, we create the new java rule class:

    public class DebitAccountRule extends AbstractBusinessRule {

    /*
    * @see net.sf.opentranquera.rules.Rule#evaluate()
    */
    public RuleResult evaluate() {
    Transfer transfer = (Transfer) this.getEvaluatedObject();
    // Get the balance of the account
    // Checks if ther is enough funds to execute the transfer.

    return this.getSuccess();
    }

    }

    In this case, we have an mock implementation just to prove how OT rules works and is configured. Now we add the new rule in the configuration file:

    <bean id="transferRule" class="net.sf.opentranquera.rules.logical.AndRule">
    <constructor-arg index="0">
    <list>
    <bean class="net.sf.opentranquera.samples.rules.DifferentAccountRule"/>
    <bean class="net.sf.opentranquera.samples.rules.AccountsExistsRule">
    <property name="rule" ref="accountExistsRule"/>
    </bean>
    <bean class="net.sf.opentranquera.samples.rules.DebitAccountRule"/>
    </list>
    </constructor-arg>
    <constructor-arg index="1" value="true"/>
    </bean>

    That's all what you have to do in order to add a new validation rule to the method execution. As we can see, nothing is changed in the TransferServiceImpl source code.

    Rules reuse
    OT Rules allows to reuse the rules in different objects or RuleEvaluators, for instance in the previous example we are going to reuse the rule net.sf.opentranquera.samples.rules.AccountExistsRule, in the ruleEvaluator bean asigning the execution of the rule "balance" and in transferRule as one of the included rules:

    <bean id="ruleEvaluator" class="net.sf.opentranquera.rules.RuleEvaluator">
    <property name="rules">
    <map>
    <entry key="transfer"><ref local="transferRule"/></entry>
    <entry key="balance"><ref local="accountExistsRule"/></entry>
    </map>
    </property>
    </bean>
    <bean id="transferRule" class="net.sf.opentranquera.rules.logical.AndRule">
    <constructor-arg index="0">
    <list>
    <bean class="net.sf.opentranquera.samples.rules.DifferentAccountRule"/>
    <bean class="net.sf.opentranquera.samples.rules.AccountsExistsRule">
    <property name="rule" ref="accountExistsRule"/>
    </bean>
    <bean class="net.sf.opentranquera.samples.rules.DebitAccountRule"/>
    </list>
    </constructor-arg>
    <constructor-arg index="1" value="true"/>
    </bean>


    More Rules
    OT Rules has a built-in rule set to make the development easier:
    * AndRule: Executes the AND operation between several rules. Can use a short circuit if needed.
    * OrRule: Executes the OR operation between several rules. Also can use a short circuit if needed.
    * XorRule: Executes XOR between several rules.
    * NotRule: Applies NOT to the result of the rule execution
    * IfRule: Executes one or other rule using as a condition another rule.

    Also a set of RuleResults are provided, as follows:
    * SimpleResult: Basic functionality for a RuleResult.
    * TrueResult: Is a RuleResult that returns true.
    * FalseResult: Is a RuleResult that returns false.

    Finally, as mentioned earlier, OT Rules can be configured using different mechanisms:
    * Spring: as used in this example
    * API: You can create rule with java code, using a FluentAPI that enables the creation of rules in the java programming language, for example:

    FluentRule fir = new FluentRule("test");
    fir.rule(new TrueRule()).and(new TrueRule()).or(new FalseRule()).not();

    RuleEvaluator evaluator = FluentInterfaceRuleEvaluatorBuilder.createRuleEvaluator(fir);
    RuleResult result = evaluator.evaluateRule("test");


    * XML: A XML format that has an Eclipse plugin to edit. Here is an example:

    <rules>
    <rule name="test.single.rule"
    ruleClass="net.sf.opentranquera.rules.HelloWorldRule" />

    <rule name="test.param.rule"
    ruleClass="net.sf.opentranquera.rules.WordLengthRule">
    <param name="length" value="11" />
    </rule>

    <rule name="test.and.rule"
    ruleClass="net.sf.opentranquera.rules.logical.AndRule">
    <rule ruleClass="net.sf.opentranquera.rules.HelloWorldRule" />
    <rule ruleClass="net.sf.opentranquera.rules.WordLengthRule">
    <param name="length" value="14" />
    </rule>
    </rule>

    <rule name="test.not.rule"
    ruleClass="net.sf.opentranquera.rules.logical.NotRule">
    <rule ruleClass="net.sf.opentranquera.rules.support.FalseRule" />
    </rule>

    <rule name="test.negate.rule"
    ruleClass="net.sf.opentranquera.rules.support.TrueRule"
    modifier="negate" />

    <rule name="test.and.ref"
    ruleClass="net.sf.opentranquera.rules.logical.OrRule"
    shortCircuit="false">
    <rule ref="test.single.rule" />
    <rule ref="test.param.rule" />
    </rule>
    </rules>


    Summary:
    In this article, we introduced OT Rules and walked through a simple example using OT Rules to execute business validations. We've also explained the different built-in handy rules and how to configure the rule set to perform validations.

    lunes, 9 de abril de 2007

    OT-Rules y Spring

    Existen dos formas de utilizar reglas de OT-Rules dentro de una aplicación Spring:
    1. Configurando directamente un RuleEvaluator como Spring-Bean.
    2. Utilizar RuleEvaluatorFactoryBean

    Ambas tiene su beneficios que pasaré a explicar.
    La primera tiene la enorme ventaja de que se permite inyectar dependencias a las rules directamente (Servicios, DAOs, collaborations, etc).


    <bean class="net.sf.opentranquera.rules.RuleEvaluator" id="ruleEvaluator">
    <property name="rules">
    <map>
    <entry key="ventas">
    <ref local="andRule"/>
    </entry>
    </map>
    </property>
    </bean>

    La declaración de la regla es:

    <bean class="net.sf.opentranquera.rules.logical.AndRule" id="andRule">
    <constructor-arg index="0">
    <list>
    <bean class="ar.com.eds.mcd.mcventas.process.rules.VentasNetasRule"/>
    <bean class="ar.com.eds.mcd.mcventas.process.rules.VentasBrutasRule"/>
    ...

    Luego, puedo inyectar el RuleEvaluator en cualquier bean de spring:

    <bean class="ar.com.eds.mcd.mcventas.services.VentasServiceImpl" id="ventasService">
    <property name="ruleEvaluator" ref="ruleEvaluator"/>
    ... demas dependencias

    y hacer uso de este:

    RuleResult result = this.ruleEvaluator.evaluateRule("ventas");
    if(result.isSuccessful() ) {
    ...
    }

    Ahora bien, como dijimos configurando Rules de esta forma obtenemos la ventaja de no solo inyectar el RuleEvaluator en diferentes objetos de Spring sino que podemos inyectar diferentes objetos de Spring en nuestras rules.
    Pero que sucede si nosotros ya tenemos configuradas las rules utilizando el archivo propietario de OT-Rules (rules.xml) o deseamos utilizar el plugin de eclipse que nos facilita la creación y administración de rules?
    Bueno, en este caso lo que debemos hacer es utilizar el FactoryBean que provee OT-Bridges de la siguiente forma:

    <bean id="ruleEvaluator" class="net.sf.opentranquera.spring.rules.RuleEvaluatorFactoryBean">
    <property name="rules" value="rules.xml"/>
    </bean>

    Luego podemos inyectar el RuleEvaluator en cualquier objeto de spring.

    viernes, 30 de marzo de 2007

    OSCache y Tiles

    En el proyecto que estoy trabajando actualmente existe cierto contenido que se genera dinamicamente que, a su vez, puede ser cacheado (por ejemplo en menu) y para realizar esta operación utilice oscache. Me encnotre con un problema a la hora de ponerlo a funcionar con tiles.
    Cuando puse el tag <cache:cache> en mi layout.jsp (define el layout de mi pagina llamando a <tiles:insert>), surgio:

    Can't insert page '/layout.jsp' : Illegal to flush within a custom tag 

    Ahora bien, hice la prueba rapida de poner el valor de flush en false y funciona:

    <cache:cache>
    <tiles:insert attribute="menu" flush="false"/>
    </cache:cache>

    Por otro lado lo también funciona es utilizar el tag <cache:cache> dentro de las paginas jsp que tiles incluyo, por ejemplo en la pagina definida como body. En este caso si oscache realiza su trabajo correctamente y cache el contenido definido.

    martes, 19 de diciembre de 2006

    OT-Rules. Manejando cambios funcionales

    A medida que pasa el tiempo, en el desarrollo de aplicaciones de software, muchas cosas cambian, el equipo de trabajo cambia, las fechas de entrega y también los requerimientos iniciales ya sea porque el usuario final no sabia que quería o porque el funcional no entendio nada o simplemente por el hecho de que todo cambia (evoluciona).
    Dada esta necesidad de cambio (casi constante) en los requerimientos de una aplicación se hace necesario hacer que el software que construimos sea flexible y fácil de modificar.

    Como se dijo en el post anterior, OT-Rules permite escribir ciertas validaciones (o lógica de negocio) en una forma flexible, donde agregar nuevas reglas, modificar y eliminar reglas existentes y cambiar el comportamiento de como se debe ejecutar estas reglas se hace muy facil y simple.

    Aquí utilizaremos el código de ejemplo utilizado para el post anterior y por supuesto, agregaremos algunas cositas.
    Categorización de reglas.
    OT-Rules contiene una categorización de las reglas que este provee:
    * Reglas de soporte: Son aquellas que facilitan el desarrollo por ejemplo (net.sf.opentranquera.rules.support.TrueRule o net.sf.opentranquera.rules.support.FalseRule).
    * Reglas lógicas: Ejecutan acciones lógicas como AND, OR, XOR, NOT.
    * Reglas condicionales: Ejecutan condiciones, hasta ahora solo net.sf.opentranquera.rules.conditional.IfRule.
    * Reglas iterativas: Reglas que iteran e invoquan a otras reglas (FOR, WHILE)

    Short circuit
    Todas las reglas lógicas tienen el atributo short circuit que debería ser setteado en la creación/construcción de la regla, el cual permite definir el modo de ejecución que tendra ésta y trabaja de la misma manera que lo hace el short circuit de la clausula if de Java.
    Veamoslo en un ejemplo usando un AND:

    a && b -> short circuit = true: Si "a" se evalua en false (con lo cual el AND daria false independientemente de lo que "b") no se evalua el resultado de "b".
    a & b -> short circuit = false: Si "a" se evalua en false (con lo cual el AND daria false independientemente de lo que "b") igualmente se evalua el resultado de "b".

    Es decir, si tengo una regla de tipo AND que esta compuesta por otras tres reglas, como se puede ver en la regla llamada "transferRule" (del ejemplo anterior), solo se evaluara en true si todas se evaluan en true individualmente.

    <bean id="transferRule" class="net.sf.opentranquera.rules.logical.AndRule">
    <constructor-arg index="0">
    <list>
    <bean class="net.sf.opentranquera.samples.rules.DifferentAccountRule"/>
    <bean class="net.sf.opentranquera.samples.rules.AccountsExistsRule">
    <property name="rule" ref="accountExistsRule"/>
    </bean>
    </list>
    </constructor-arg>
    <constructor-arg index="1" value="true"/>
    </bean>

    El primer argumento del constructor es un List de Rules y el segundo argumento es un boolean que índica si la regla es short circuit (por defecto "short circuit = true").
    Ahora bien, si "DifferentAccountRule = true", se evalua AccountsExistsRule, si esta es true, entonces la evaluación de la regla transferRule sera true. Pero sí "DifferentAccountRule = false", AccountsExistsRule no sera evaluada (ejecutada) y el resultado sera false.
    De la misma forma que lo hacemos con la regla AND, lo podemos hacer con las reglas OR, XOR.
    Entonces, podemos finalizar diciendo que en el caso de que sea necesario cambiar este tipo de comportamiento de una regla lógica simplemente se modifica el valor del atributo short circuit en la configuración (sea cual sea), es así de fácil.

    La regla NOT.
    Existe una regla lógica que permite negar el resultado de la evaluación de otro regla (net.sf.opentranquera.rules.logical.NotRule).
    Esta regla se crea a partír de otra y la envuelve (se podría decír que la decora utilizando el pattern Decorator), luego al evaluarla (llamar a su método evaluate()) se evalua la regla contenida para finalmente negar su resultado (manteniendo todos sus mensajes).

    Veamos algunos ejemplos:
    Usando el XML de configuración de OR-Rules:

    <rule name="test.not.rule" ruleClass="net.sf.opentranquera.rules.logical.NotRule">
    <rule ruleClass="net.sf.opentranquera.rules.support.FalseRule"/>
    </rule>

    Creo una NotRule incluyendo una FalseRule. El resultado de esto es la negación de la evaluación de FalseRule (que siempre da false), es decir me devuelve el resultado true.

    <rule name="test.negate.rule" ruleClass="net.sf.opentranquera.rules.support.FalseRule"
    modifier="negate"/>

    Aquí­, se utiliza el atributo "modifier", esto indica que la rule "test.negate.rule" que es del tipo FalseRule debe ser negada luego de ejecutar. Es decír, el resultado sera true.
    Ahora veremos esta mismo configuración usando SpringFramework:

    <bean id="transferRule" class="net.sf.opentranquera.rules.logical.NotRule">
    <constructor-arg index="0">
    <bean class="net.sf.opentranquera.rules.support.FalseRule"/>
    </constructor-arg>
    </bean>

    Simple no?

    Cambiar lógica de evaluación.
    Llamo lógica de evaluación a aquella que no concierne a la regla en si, sino a como esta se evalua o a como se ejecutan las diferentes reglas que contiene, por ejemplo, si utilizo un regla AND para componer la ejecución de varias reglas, digo que la lógica del AND (ya sea con short circuit o no) es mi lógica de evaluación, mientras que las reglas ejecutan la lógica de negocio/validación que quiero evaluar.
    OT-Rules provee varias clases (reglas) que me permiten modificar esta lógica de evaluación independientemente de lo que hacen las reglas en sí mismas, de forma tal que los programadores se concentren en resolver la problematica particular de negocio y luego las componen utilizando alguna clase de OT-Rules (o alguna creada por ellos dado que OT-Rules es flexible en este aspecto) para decidir como van a ser evaluadas y ejecutadas.
    Ahora bien, si en lugar de tener las reglas escritar con OT-Rules las tendrias escritas directamente sobre el código del método del servicio y tendriamos la necesidad de modificar la forma en que se evaluan las reglas y no las reglas en sí­. Esto seria muy problematico, dado que tendriamos que modificar gran parte de nuestro código ya testeado.
    Pero si tenemos las reglas separadas en clases (incluso con sus test unitarios) para luego decirles por configuración como se evaluan, podría no ser tan problematico. De hecho, utilizando OT-Rules sería muy simple. En el caso recien mencionado, simplemente tendriamos que cambiar la AND rule por la nueva forma de evalución requerida (OR, XOR, etc).

    En el ejemplo anterior utilizabamos una AND rule para armar la regla "transferRule", pero si ahora necesitaramos que la evaluacion de esta rule esta basada en una lógica de OR, simplemente cambiamos la clase ANDRule, por la ORRule quedando de la siguiente manera:

    <bean id="transferRule" class="net.sf.opentranquera.rules.logical.OrRule">
    <constructor-arg index="0">
    <list>
    <bean class="net.sf.opentranquera.samples.rules.DifferentAccountRule"/>
    <bean class="net.sf.opentranquera.samples.rules.AccountsExistsRule">
    <property name="rule" ref="accountExistsRule"/>
    </bean>
    </list>
    </constructor-arg>
    <constructor-arg index="1" value="true"/>
    </bean>

    La de misma forma que cambiamos un AND por un OR podemos, incluso, componer reglas lógicas, por ejemplo, dentro de una regla OR tener un regla NOT que contiene una regla AND que a su vez contiene otras reglas (que pueden ser reglas de negocio o reglas compuestas AND, OR, XOR, etc.).

    Reuso de reglas (otra vez).
    Veamos lo recien comentado con un ejemplo.
    Recordemos del articulo anterior que teniamos la regla "transferRule" que evaluaba que al realizar una transferencia, las cuentas sean distintas, que existan y que la cuenta débito tenga el dinero disponible para realizarla.
    Ahora agregaremos un nuevo método para el servicio TransferService que realice una transferencia pero entre un mismo banco a modo de ejemplo. Para este caso es válida la regla (compuesta) "transferRule" y ádemas agregaremos la siguiente lógica de validación:
    # Como dijimos debe pasar la regla transferRule con lo cual estariamos haciendo reuso de la misma.
    # Verificar que la transferencia tenga como cuenta débito y cuenta crédito el mismo banco o bancos del mismo grupo empresarial.
    Veamos ahora como queda el nuevo método y la configuración de OT-Rules

    public boolean bankTransfer(Transfer transfer) throws TransferException {
    this.evaluator.setEvaluatedObject(transfer);
    RuleResult result = this.evaluator.evaluateRule("bankTransfer");
    if(!result.isSuccessful())
    throw new TransferException(result.getMessages());

    // TODO logic ...
    return true;
    }


    <bean id="bankTransferRule" class="net.sf.opentranquera.rules.logical.AndRule">
    <constructor-arg index="0">
    <list>
    <ref local="transferRule"/>
    <bean class="net.sf.opentranquera.rules.logical.OrRule">
    <constructor-arg index="0">
    <list>
    <bean class="net.sf.opentranquera.samples.rules.SameBankRule"/>
    <bean class="net.sf.opentranquera.samples.rules.SameBankGroupRule"/>
    </list>
    </constructor-arg>
    <constructor-arg index="1" value="true"/>
    </bean>
    </list>
    </constructor-arg>
    <constructor-arg index="1" value="false"/>
    </bean>

    Como podemos ver la nueva regla (bankTransferRule) tiene una lógica de evaluación del tipo AND que contiene la ya creada y declarada "transferRule" junto con una nueva regla del tipo OR que a su vez se compone de otras dos reglas, una que verifica que las cuentas sean del mismo banco y la otra que las cuentas sean del mismo grupo empresarial.
    Al evaluarse esta regla se dispararían las siguientes acciones:
    # Se evalua "transferRule", es decir se evalua la regla AND junto con sus compuestos.
    # Si la regla anterior da true, se evalua la regla OR. Esta regla compuesta ejecuta SameBankRule y SameBankGroupRule, si alguna de ellas da true, esta regla devuelve un resultado en true.
    # Finalmente si ambas reglas se evaluaron en true, se retorna un resultado true.

    Nota: Se podria hacer echo toda esta valiadcion en una sola regla (o de otras multiples formas utilizando diferentes tipos de composiciones), sin embargo lo hice así a modo de ejemplo.

    Usar fluent interface como mecanismo de configuración.
    OT-Rules tiene dos mecanismo de configuración por código:
    1) Creando las clases directamente utilizando el operador new: Como lo hacen los test case

    XorRule xorRule = new XorRule();
    xorRule.addRule(new FalseRule());
    xorRule.addRule(new TrueRule());

    xorRule.evaluate();

    2) Utilzando al API fluida: Evita la necesidad de conocer las clases, es mas declarativa e intuitiva

    FluentRule fir = new FluentRule("test");
    fir.rule(new FalseRule()).xor(new TrueRule()).or(new FalseRule()).not();

    RuleEvaluator evaluator = FluentInterfaceRuleEvaluatorBuilder.createRuleEvaluator(fir);
    RuleResult result = evaluator.evaluateRule("test");

    Más allá de que podemos reemplazar toda la configuración del ejemplo anterior usando esta API, para este tutorial solamente reemplazaremos la regla "transferRule" y para ello haremos uso del componente de OTF llamado bridge (OT-Bridge sirve para integrar las frameworks de OTF con otras frameworks).
    No modificaremos nada de código, todo es configuración (en realidad agregaremos una clase que tiene la codificación usando la API fluida).

    La declaración de la regla en el rulesContext.xml queda asi:

    <bean id="transferRule" class="net.sf.opentranquera.spring.rules.FluentRuleInterfaceFactoryBean">
    <property name="config">
    <bean class="net.sf.opentranquera.samples.rules.FluentConfigImpl"/>
    </property>
    </bean>

    Usando el FactoryBean FluentRuleInterfaceFactoryBean podemos configurar una rule usando la API fluida. En la propiedad "config" se inyecta la clase que implemente de net.sf.opentranquera.spring.rules.FluentConfig que contiene la codificación/configuración permitente.

    public class FluentConfigImpl implements FluentConfig, BeanFactoryAware {

    private BeanFactory bf;

    public FluentRule getFluent() {
    // create the rules
    DifferentAccountRule diffAccount = new DifferentAccountRule();
    AccountsExistsRule accountExists = new AccountsExistsRule();
    DebitAccountRule debitAccount = new DebitAccountRule();

    // get object from Spring
    AccountExistsRule accountExist = (AccountExistsRule)this.bf.getBean("accountExistsRule");
    accountExists.setRule(accountExist);

    FluentRule rule = new FluentRule("transferRule");
    rule.rule(diffAccount).and(accountExists).and(debitAccount);
    return rule;

    }

    public void setBeanFactory(BeanFactory bf) throws BeansException {
    this.bf = bf;
    }

    }

    Ahora, cuando se pide evaluar a la regla "transferRule" esta se obtendra ahora desde la configuración de la API fluida (y gracias los mecanismos internos de OT-Rules y OT-Bridge) permitiendo que sea transparente a la configuracion de la aplicacion.

    Conclusión.
    Hemos visto mas en detalle aspectos particulares de OT-Rules, como ser la categorización de reglas que propone y la regla NOT, como asi tambien el uso del atributo short circuit. Aprendimos que es la lógica de evaluación de una regla (según OT-Rules). Por ultimo conocimos como configurar el engine utilizando directamente código, mas presisamente con el uso de una API fluida.