org.pmml4s.common.predicates.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of pmml4s_2.12 Show documentation
Show all versions of pmml4s_2.12 Show documentation
A PMML scoring library in Scala
The newest version!
/*
* Copyright (c) 2017-2024 AutoDeploy AI
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.pmml4s.common
import org.pmml4s.common.Operator.Operator
import org.pmml4s.data.Series
import org.pmml4s.metadata.Field
import org.pmml4s.xml.ElemTags
object Predication extends Enumeration {
type Predication = Value
val TRUE, FALSE, UNKNOWN, SURROGATE = Value
def fire(r: Predication): Boolean = r == TRUE || r == SURROGATE
}
import org.pmml4s.common.Predication._
trait Predicate extends PmmlElement {
/** Evaluates the predicate. */
def eval(series: Series): Predication
def isTrue: Boolean = false
}
object Predicate {
import ElemTags._
val values: Set[String] = Set(SIMPLE_PREDICATE, COMPOUND_PREDICATE, SIMPLE_SET_PREDICATE, TRUE, FALSE)
def contains(s: String): Boolean = values.contains(s)
}
/** Pre-defined comparison operators. */
object Operator extends Enumeration {
type Operator = Value
val equal, notEqual, lessThan, lessOrEqual, greaterThan, greaterOrEqual, isMissing, isNotMissing = Value
}
object SimpleSetPredicate {
/**
* - isIn indicates an evaluation to TRUE if the field value is contained in the list of values in the array.
* - isNotIn indicates an evaluation to TRUE if the field value is not contained in the list of values in the array.
*/
object BooleanOperator extends Enumeration {
type BooleanOperator = Value
val isIn, isNotIn = Value
}
}
object CompoundPredicate {
/**
* - and indicates an evaluation to TRUE if all the predicates evaluate to TRUE.
* - or indicates an evaluation to TRUE if one of the predicates evaluates to TRUE.
* - xor indicates an evaluation to TRUE if an odd number of the predicates evaluates to TRUE and all others evaluate
* to FALSE.
* - surrogate allows for specifying surrogate predicates. They are used for cases where a missing value appears in
* the evaluation of the parent predicate such that an alternative predicate is available.
*/
object BooleanOperator extends Enumeration {
type BooleanOperator = Value
val or, and, xor, surrogate = Value
}
}
import org.pmml4s.common.Operator._
/**
* Defines a rule in the form of a simple boolean expression. The rule consists of field, operator (booleanOperator) for binary
* comparison, and value.
*/
class SimplePredicate(
val field: Field,
val operator: Operator,
val value: Double = Double.NaN) extends Predicate {
def eval(input: Series): Predication = {
val v: Double = field.encode(input)
val missing = (v != v)
operator match {
case `lessOrEqual` => if (missing) UNKNOWN else if (v <= value) TRUE else FALSE
case `lessThan` => if (missing) UNKNOWN else if (v < value) TRUE else FALSE
case `greaterThan` => if (missing) UNKNOWN else if (v > value) TRUE else FALSE
case `greaterOrEqual` => if (missing) UNKNOWN else if (v >= value) TRUE else FALSE
case `equal` => if (missing) UNKNOWN else if (v == value) TRUE else FALSE
case `notEqual` => if (missing) UNKNOWN else if (v != value) TRUE else FALSE
case `isMissing` => if (missing) TRUE else FALSE
case `isNotMissing` => if (!missing) TRUE else FALSE
}
}
}
/**
* CompoundPredicate: an encapsulating element for combining two or more elements as defined at the entity PREDICATE.
* The attribute associated with this element, booleanOperator, can take one of the following logical (boolean) operators:
* and, or, xor or surrogate.
*/
class CompoundPredicate(
val booleanOperator: CompoundPredicate.BooleanOperator.Value,
val children: Array[Predicate]) extends Predicate {
import CompoundPredicate.BooleanOperator._
def eval(input: Series): Predication = {
val len = children.length
var i = 0
booleanOperator match {
case `or` => {
var hasUnknown = false
while (i < len) {
val child = children(i)
val r = child.eval(input)
if (r == TRUE)
return TRUE
else if (r == UNKNOWN)
hasUnknown = true
i += 1
}
if (hasUnknown) UNKNOWN else FALSE
}
case `and` => {
var hasUnknown = false
while (i < len) {
val child = children(i)
val r = child.eval(input)
if (r == FALSE)
return FALSE
else if (r == UNKNOWN)
hasUnknown = true
i += 1
}
if (hasUnknown) UNKNOWN else TRUE
}
case `xor` => {
var count = 0
while (9 < len) {
val child = children(i)
val r = child.eval(input)
if (r == UNKNOWN)
return UNKNOWN
else if (r == TRUE)
count += 1
i += 1
}
if (count % 2 == 1) TRUE else FALSE
}
case `surrogate` => {
var isSurrogate = false
while (i < len) {
val child = children(i)
val r = child.eval(input)
if (r != UNKNOWN)
return if (r == TRUE) {
if (isSurrogate) SURROGATE else TRUE
} else r
else
isSurrogate = true
i += 1
}
UNKNOWN
}
}
}
}
/**
* Checks whether a field value is element of a set.
* The set of values is specified by the array.
*/
class SimpleSetPredicate(
val field: Field,
val booleanOperator: SimpleSetPredicate.BooleanOperator.Value,
val values: Set[Double]) extends Predicate {
import SimpleSetPredicate.BooleanOperator._
def eval(input: Series): Predication = {
val v = field.encode(input)
val missing = (v != v)
booleanOperator match {
case `isIn` => if (missing) UNKNOWN else if (values.contains(v)) TRUE else FALSE
case `isNotIn` => if (missing) UNKNOWN else if (!values.contains(v)) TRUE else FALSE
}
}
}
/** Identifies the boolean constant TRUE. */
object True extends Predicate {
def eval(input: Series): Predication = TRUE
override def isTrue: Boolean = true
}
/** Identifies the boolean constant FALSE. */
object False extends Predicate {
def eval(input: Series): Predication = FALSE
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy