/* This method evaluates the Field using the first and second value and returns the result value. If the second operand type is a missing return a missing value. If it is a string, and the field is case insensitive, then convert the second operand to lower case, range through the fields of the first and compare, each field with the second. When equal, return the value. If the field is case sensitive, use the Field method to directly access the field and return it. For all other types, if the first operand expression is missing, return missing, else return null. */ func (this *Field) Apply(context Context, first, second value.Value) (value.Value, error) { switch second.Type() { case value.STRING: s := second.Actual().(string) v, ok := first.Field(s) if !ok && this.caseInsensitive { s = strings.ToLower(s) fields := first.Fields() for f, val := range fields { if s == strings.ToLower(f) { return value.NewValue(val), nil } } } return v, nil case value.MISSING: return value.MISSING_VALUE, nil default: if first.Type() == value.MISSING { return value.MISSING_VALUE, nil } else { return value.NULL_VALUE, nil } } }
/* Compute the Final. Compute the sum and the count. If these arent numbers throw an error. Compute the avg as sum/count. Check for divide by zero, and return a NULL value if true. */ func (this *Avg) ComputeFinal(cumulative value.Value, context Context) (value.Value, error) { if cumulative == value.NULL_VALUE { return cumulative, nil } sum, _ := cumulative.Field("sum") count, _ := cumulative.Field("count") if sum.Type() != value.NUMBER || count.Type() != value.NUMBER { return nil, fmt.Errorf("Missing or invalid sum or count in AVG: %v, %v.", sum.Actual(), count.Actual()) } if count.Actual().(float64) > 0.0 { return value.NewValue(sum.Actual().(float64) / count.Actual().(float64)), nil } else { return value.NULL_VALUE, nil } }
/* Aggregate input partial values into cumulative result number value for sum and count. If the partial results are not numbers, then return an error. */ func (this *Avg) cumulatePart(part, cumulative value.Value, context Context) (value.Value, error) { if part == value.NULL_VALUE { return cumulative, nil } else if cumulative == value.NULL_VALUE { return part, nil } psum, _ := part.Field("sum") pcount, _ := part.Field("count") csum, _ := cumulative.Field("sum") ccount, _ := cumulative.Field("count") if psum.Type() != value.NUMBER || pcount.Type() != value.NUMBER || csum.Type() != value.NUMBER || ccount.Type() != value.NUMBER { return nil, fmt.Errorf("Missing or invalid partial sum or count in AVG: %v, %v, %v, v.", psum.Actual(), pcount.Actual(), csum.Actual(), ccount.Actual()) } cumulative.SetField("sum", psum.Actual().(float64)+csum.Actual().(float64)) cumulative.SetField("count", pcount.Actual().(float64)+ccount.Actual().(float64)) return cumulative, nil }
/* Call the Field method using the item value input argument on the receiver. This returns a value. To evaluate an identifier, look into the current item, find a field whose name is the identifier, and return the value of that field within the current item. */ func (this *Identifier) Evaluate(item value.Value, context Context) (value.Value, error) { rv, _ := item.Field(this.identifier) return rv, nil }