core.pure.tds.tds.pure Maven / Gradle / Ivy
// Copyright 2020 Goldman Sachs
//
// 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.
import meta::pure::metamodel::path::*;
import meta::pure::tds::extensions::*;
Class meta::pure::tds::TabularDataSet
{
columns : TDSColumn[*];
columnByName(s:String[1]){$this.columns->filter(c|$c.name == $s)->first()}:TDSColumn[0..1];
rows : TDSRow[*];
}
Class meta::pure::tds::TDSColumn
{
<>
offset : Integer[0..1];
<>
name : String[1];
<>
type : DataType[0..1];
<>
enumMappingId : String[0..1];
<>
enumMapping: meta::pure::mapping::EnumerationMapping[0..1];
<>
documentation : String[0..1];
<>
sourceDataType: Any[0..1];
toString(){
'meta::pure::tds::TDSColumn('
+ 'name=' + $this.name
+ if($this.type->isNotEmpty(),
| ', ' + 'type=' + $this.type->makeString(),
| ''
)
+ if($this.offset->isNotEmpty(),
| ', ' + 'offset=' + $this.offset->makeString(),
| ''
)
+ if($this.documentation->isNotEmpty(),
| ', ' + 'documentation=' + $this.documentation->makeString(),
| ''
)
+ if($this.enumMappingId->isNotEmpty(),
| ', ' + 'enumMappingId=' + $this.enumMappingId->makeString(),
| ''
)
+ if($this.sourceDataType->isNotEmpty(),
| ', ' + 'sourceDataType=' + $this.sourceDataType->makeString(),
| ''
)
+ ')';
}:String[1];
}
Class meta::pure::tds::TDSRow
{
parent : TabularDataSet[0..1];
values : Any[*];
getString(colName:String[1]){$this.get($colName)->cast(@String)}:String[1];
getNullableString(colName:String[1]){ if($this.isNull($colName), | [] , | $this.getString($colName))}:String[0..1];
getNumber(colName:String[1]){$this.get($colName)->cast(@Number)}:Number[1];
getInteger(colName:String[1]){$this.get($colName)->cast(@Integer)}:Integer[1];
getFloat(colName:String[1]){$this.get($colName)->cast(@Number)->toFloat()}:Float[1];
getDecimal(colName:String[1]){$this.get($colName)->cast(@Number)->toDecimal()}:Decimal[1];
getDate(colName:String[1]){$this.get($colName)->cast(@Date)}:Date[1];
getDateTime(colName:String[1]){$this.get($colName)->cast(@DateTime)}:DateTime[1];
getStrictDate(colName:String[1]){$this.get($colName)->cast(@StrictDate)}:StrictDate[1];
getBoolean(colName:String[1]){$this.get($colName)->cast(@Boolean)}:Boolean[1];
getEnum(colName:String[1]){$this.get($colName)->cast(@Enum)}:Enum[1];
isNotNull(colName:String[1]){!$this.isNull($colName)}:Boolean[1];
get(colName:String[1])
{
let parent = $this.parent;
assertNotEmpty($parent, {| 'Unable to find index for the column:' +$colName+'. The TDS row has no parent, make sure the row has a parent.'});
let col = $this.parent->toOne().columnByName($colName);
assertNotEmpty($col, {| $colName+' is unknown (columns=['+$parent.columns.name->makeString(',')+'])!'});
$this.get($col->toOne());
}:Any[1];
isNull(colName:String[1]){
let parent = $this.parent;
assertNotEmpty($parent, {| 'Unable to find index for the column:' +$colName+'. The TDS row has no parent, make sure the row has a parent.'});
let col = $this.parent->toOne().columnByName($colName);
assertNotEmpty($col, {| $colName+' is unknown (columns=['+$parent.columns.name->makeString(',')+'])!'});
$this.isNull($col->toOne());
}:Boolean[1];
getString(col:TDSColumn[1]){$this.get($col)->cast(@String)}:String[1];
getNullableString(col:TDSColumn[1]){ if($this.isNull($col), | [] , | $this.getString($col))}:String[0..1];
getNumber(col:TDSColumn[1]){$this.get($col)->cast(@Number)}:Number[1];
getInteger(col:TDSColumn[1]){$this.get($col)->cast(@Integer)}:Integer[1];
getFloat(col:TDSColumn[1]){$this.get($col)->cast(@Float)}:Float[1];
isNull(col:TDSColumn[1]){$this.get($col) == ^TDSNull()}: Boolean[1];
isNotNull(col:TDSColumn[1]){!$this.isNull($col)}: Boolean[1];
get(col: TDSColumn[1]){
assertNotEmpty($col.offset, {| 'No index for the column:' +$col.name+'.'});
$this.values->at($col.offset->toOne())->toOne();
} : Any[1];
}
Class meta::pure::tds::TDSNull
{
//Used to make sure that two TDSNull instances are equal
<> key:Nil[0];
toString(){
'TDSNull'
}:String[1];
}
Class meta::pure::tds::AggregateValue
{
name : String[1];
mapFn : FunctionDefinition<{TDSRow[1]->T[0..1]}>[1];
aggregateFn : FunctionDefinition<{T[*]->U[0..1]}>[1];
}
Class meta::pure::tds::ColumnSpecification
{
name : String[1];
documentation : String[0..1];
}
Class {doc.doc = 'Normal Project column, also used for extending a TDS'}
meta::pure::tds::BasicColumnSpecification extends meta::pure::tds::ColumnSpecification
{
func : Function<{T[1]->Any[*]}>[1];
}
Enum meta::pure::tds::SortDirection
{
ASC, DESC
}
Class meta::pure::tds::SortInformation
{
column : String[1];
direction : SortDirection[1];
}
Class meta::pure::tds::ColumnSort
{
column:Function<{T[1]->Any[1]}>[1];
direction:SortDirection[1];
}
Class meta::pure::tds::Window
{
partition:Function<{T[1]->Any[1]}>[*];
}
Class meta::pure::tds::OlapOperation
{
//superclass for all operations
}
Class meta::pure::tds::OlapAggregation extends OlapOperation
{
mapFn : FunctionDefinition<{T[1]->Number[1]}>[1];
aggregateFn : FunctionDefinition<{Number[*]->Number[0..1]}>[1];
}
Class meta::pure::tds::OlapRank extends OlapOperation
{
rankFn: FunctionDefinition<{T[*]->Map[1]}>[1];
}
Class meta::pure::tds::WindowColumnSpecification extends ColumnSpecification
{
window:Window[0..1];
sortInfo: ColumnSort[0..1];
func: meta::pure::tds::OlapOperation[1];
}
function
{doc.doc = 'Helper function to create / define an aggreagation value (combination of column name and aggregation function)'}
meta::pure::tds::agg(name:String[1], mapFn:FunctionDefinition<{TDSRow[1]->T[0..1]}>[1], aggregateFn:FunctionDefinition<{T[*]->U[0..1]}>[1]):meta::pure::tds::AggregateValue[1]
{
^meta::pure::tds::AggregateValue(name = $name, mapFn=$mapFn, aggregateFn=$aggregateFn);
}
function meta::pure::tds::col(window: Window[1], func:FunctionDefinition<{T[*]->Map[1]}>[1], name: String[1]):WindowColumnSpecification[1]
{
let rank = ^OlapRank(rankFn = $func);
^WindowColumnSpecification
(
window = $window,
name = $name,
func = $rank
);
}
function meta::pure::tds::col(window: Window[1], sortInfo:ColumnSort[1], func:FunctionDefinition<{T[*]->Map[1]}>[1], name: String[1]):WindowColumnSpecification[1]
{
let rank = ^OlapRank(rankFn = $func);
^WindowColumnSpecification
(
window = $window,
sortInfo = $sortInfo,
name = $name,
func = $rank
);
}
function meta::pure::tds::col(window: Window[1], func: meta::pure::tds::OlapAggregation[1], name: String[1]):WindowColumnSpecification[1]
{
^WindowColumnSpecification
(
window = $window,
name = $name,
func = $func
)
}
function meta::pure::tds::col(window: Window[1],sortInfo:ColumnSort[1], func: meta::pure::tds::OlapAggregation[1], name: String[1]):WindowColumnSpecification[1]
{
^WindowColumnSpecification
(
window = $window,
sortInfo = $sortInfo,
name = $name,
func = $func
)
}
function meta::pure::tds::col(sortInfo:ColumnSort[1], func: meta::pure::tds::OlapAggregation[1], name: String[1]):WindowColumnSpecification[1]
{
^WindowColumnSpecification
(
sortInfo = $sortInfo,
name = $name,
func = $func
);
}
function meta::pure::tds::col(sortInfo:ColumnSort[1], func:FunctionDefinition<{T[*]->Map[1]}>[1], name: String[1]):WindowColumnSpecification[1]
{
let rank = ^OlapRank(rankFn = $func);
^WindowColumnSpecification
(
sortInfo = $sortInfo,
name = $name,
func = $rank
);
}
function
{doc.doc = 'Helper function to create a column specificantion based on the specified value function and column name'}
meta::pure::tds::col(func : Function<{T[1]->Any[*]}>[1], name : String[1]):meta::pure::tds::BasicColumnSpecification[1]
{
^BasicColumnSpecification
(
func = $func,
name = $name
)
}
function
{doc.doc = 'Helper function to create a column specificantion based on the specified value function, column name and documentation for the column'}
meta::pure::tds::col(func : Function<{T[1]->Any[*]}>[1], name : String[1], documentation : String[1]):meta::pure::tds::BasicColumnSpecification[1]
{
^BasicColumnSpecification
(
func = $func,
name = $name,
documentation = $documentation
)
}
function meta::pure::tds::tdsRows(tds:TabularDataSet[1]):TDSRow[*]
{
$tds.rows
}
function meta::pure::tds::projectWithColumnSubset(set:T[*], columnSpecifications:ColumnSpecification[*], columnList:String[*]):TabularDataSet[1]
{
$columnList->map( i| assert($columnSpecifications.name->contains($i) , 'SubSetColumnList should be in project List ' ) );
$set->project($columnSpecifications->filter(c|$c.name->in($columnList)));
}
function meta::pure::tds::projectWithColumnSubset(set:T[*], functions:Function<{T[1]->Any[*]}>[*], ids:String[*], columnList:String[*]):TabularDataSet[1]
{
$columnList->map( i| assert($ids->contains($i) , 'SubSetColumnList should be in project List ' ) );
$set->project( $functions->zip($ids)->filter(p:PairAny[*]}>,String>[1] |$p.second->in($columnList))->map(p|^BasicColumnSpecification(func = $p.first->cast(@Function<{Nil[1]->Any[*]}>), name = $p.second))->cast(@BasicColumnSpecification));
}
function
{doc.doc = 'Project the values from the columns specifications provided to create a new TDS'}
meta::pure::tds::project(set:T[*], columnSpecifications:ColumnSpecification[*]):TabularDataSet[1]
{
let simpleCols = $columnSpecifications->filter(c|$c->instanceOf(BasicColumnSpecification))->cast(@BasicColumnSpecification);
let functions = $simpleCols.func;
let res = ^TabularDataSet(
columns = $simpleCols->map(cs|let offset = $columnSpecifications->indexOf($cs);
^TDSColumn(offset = $offset,
name = $cs.name,
type = $cs.func->functionReturnType().rawType->toOne()->cast(@DataType),
documentation = $cs.documentation);
));
let newRows = $set->map(value|
$functions->fold(
{
path, list | let res = $path->eval($value);
let updated = if ($res->size() == 0,|^TDSNull(),|$res);
^List(values = $updated->map(navValue|$list.values->map(row|^$row(values+=$navValue))));
},
^List(values = ^TDSRow(values=[], parent=$res))
).values;
);
$res->mutateAdd('rows',$newRows);
$res;
}
function
{doc.doc = 'Project the values from the lambda functions specified to create a new TDS'}
meta::pure::tds::project(set:K[*], functions:Function<{K[1]->Any[*]}>[*], ids:String[*]):TabularDataSet[1]
{
$set->project($functions->zip($ids)->map(p|^BasicColumnSpecification(func = $p.first->cast(@Function<{Nil[1]->Any[*]}>), name = $p.second))->cast(@BasicColumnSpecification));
}
function
{doc.doc = 'Project the values from the property paths specified to create a new TDS'}
meta::pure::tds::project(set:T[*], paths:Path[*]):TabularDataSet[1]
{
$paths->validatePaths();
project($set, $paths, $paths->map(p|$p->buildColumnNameOutOfPath()));
}
function
{doc.doc = 'Sort the provided TDS based on the specified information (sorting is based on order of the sort information)'}
meta::pure::tds::sort(tds:TabularDataSet[1], sortInfo:SortInformation[*]):TabularDataSet[1]
{
^$tds(rows = $tds.rows->sort({a,b|$a->multipleColumnComp($b, $sortInfo)}));
}
function
{doc.doc = 'Sort the provided TDS based on the specified column & direction'}
meta::pure::tds::sort(tds:TabularDataSet[1], column:String[1], direction:SortDirection[1]):TabularDataSet[1]
{
$tds->sort(^SortInformation(column=$column, direction=$direction));
}
function
{doc.doc = 'Sort the provided TDS based on the specified column (with ascending direction)'}
meta::pure::tds::sort(tds:TabularDataSet[1], columns:String[*]):TabularDataSet[1]
{
$tds->sort($columns->map(c | ^SortInformation(column=$c, direction=SortDirection.ASC)));
}
function
{doc.doc = 'Reduce the number of rows in the provided TDS, taking the first set of rows based on the specified size'}
meta::pure::tds::limit(tds:TabularDataSet[1], size:Integer[1]):TabularDataSet[1]
{
let newTds = ^TabularDataSet(columns=$tds.columns);
$newTds->mutateAdd('rows', $tds.rows->limit($size)->map(r|^TDSRow(parent=$newTds,values=$r.values)));
}
function
{doc.doc = 'Reduce the number of rows in the provided TDS if a size is specified (otherwise returns original rows), taking the first set of rows based on the specified size'}
meta::pure::tds::limit(tds:TabularDataSet[1], size:Integer[0..1]):TabularDataSet[1]
{
if($size->isEmpty(),
| $tds,
| $tds->limit($size->toOne())
)
}
function
{doc.doc = 'Reduce the number of rows in the provided TDS, taking the first set of rows based on the specified size'}
meta::pure::tds::take(tds:TabularDataSet[1], size:Integer[1]):TabularDataSet[1]
{
let newTds = ^TabularDataSet(columns=$tds.columns);
$newTds->mutateAdd('rows', $tds.rows->take($size)->map(r|^TDSRow(parent=$newTds,values=$r.values)));
}
function
{doc.doc = 'Returns the n-th (n = pageNumber) page TDS after paginating as per pageSize'}
meta::pure::tds::paginated(tds:TabularDataSet[1], pageNumber:Integer[1], pageSize: Integer[1]):TabularDataSet[1]
{
let newTds = ^TabularDataSet(columns=$tds.columns);
$newTds->mutateAdd('rows', $tds.rows->paginated($pageNumber, $pageSize)->map(r|^TDSRow(parent=$newTds,values=$r.values)));
}
function
{doc.doc = 'Reduce the number of rows in the provided TDS, dropping the first set of rows based on the specified size'}
meta::pure::tds::drop(tds:TabularDataSet[1], size:Integer[1]):TabularDataSet[1]
{
let newTds = ^TabularDataSet(columns=$tds.columns);
$newTds->mutateAdd('rows', $tds.rows->drop($size)->map(r|^TDSRow(parent=$newTds,values=$r.values)));
}
function
{doc.doc = 'Reduce the number of rows in the provided TDS, selecting the set of rows in the specified range between start and stop'}
meta::pure::tds::slice(tds:TabularDataSet[1], start:Integer[1], stop:Integer[1]):TabularDataSet[1]
{
let newTds = ^TabularDataSet(columns=$tds.columns);
$newTds->mutateAdd('rows', $tds.rows->slice($start, $stop)->map(r|^TDSRow(parent=$newTds,values=$r.values)));
}
function
{doc.doc = 'Reduce the number of rows in the provided TDS, selecting the set of rows that match the provided filter function'}
meta::pure::tds::filter(tds:TabularDataSet[1], f:Function<{TDSRow[1]->Boolean[1]}>[1]):TabularDataSet[1]
{
let newTds = ^TabularDataSet(columns=$tds.columns);
$newTds->mutateAdd('rows', $tds.rows->filter($f)->map(r|^TDSRow(parent=$newTds,values=$r.values)));
}
function <> meta::pure::tds::multipleColumnComp(row:TDSRow[1], other:TDSRow[1], sortInfo:SortInformation[*]):Integer[1]
{
if ($sortInfo->isEmpty(),
| 0,
| let s = $sortInfo->head();
let rowVal = if($row.isNull($s.column->toOne()), |[], |$row.get($s.column->toOne()));
let otherVal = if($other.isNull($s.column->toOne()), |[], |$other.get($s.column->toOne()));
let res = $rowVal->match([
none: Any[0]|
$otherVal->match([
none: Any[0] | 0,
otherVal: Any[1] | 1
]),
val: Any[1]|
$otherVal->match([
none: Any[0] | -1,
otherVal: Any[1] | $val->compare($otherVal)
])
]);
if ($res == 0, |$row->multipleColumnComp($other, $sortInfo->tail()), |if ($s.direction == SortDirection.ASC, |$res,|-$res));
);
}
function
{doc.doc = 'Remove duplicate rows from the priovded TDS'}
meta::pure::tds::distinct(tds:TabularDataSet[1]):TabularDataSet[1]
{
let newTds = ^$tds(rows = []);
let distinctRows = $tds.rows->removeDuplicates({l, r | $l.values == $r.values})->map(r | ^$r(parent = $newTds));
$newTds->mutateAdd('rows', $distinctRows);
}
function
{doc.doc = 'Append the rows from the additional TDSs to the rows in the first TDS. N.B. The column must be the same for the TDSs'}
meta::pure::tds::concatenate(tds1:TabularDataSet[1], tdss:TabularDataSet[*]):TabularDataSet[1]
{
$tdss->forAll(tds2|
assert($tds1.columns->size() == $tds2.columns->size(), | 'Number of columns must match to concatenate, got ' + $tds1.columns.name->joinStrings('[', ',', ']') + ' and ' + $tds2.columns.name->joinStrings('[', ',', ']') );
assertEquals($tds1.columns.name, $tds2.columns.name, | 'Columns names must match to concatenate, got ' + $tds1.columns.name->joinStrings('[', ',', ']') + ' and ' + $tds2.columns.name->joinStrings('[', ',', ']') );
assert(zip($tds1.columns, $tds2.columns)->forAll(pair |
assertEquals($pair.first.name, $pair.second.name);
assertEquals($pair.first.type, $pair.second.type, | 'Column types must match to concatenate, mismatched for column ' + $pair.first.name + ' (' + $pair.first.type->makeString() + ' vs ' + $pair.second.type->makeString() + ')');
));
);
let tds = ^TabularDataSet(columns=$tds1.columns->map(col|^TDSColumn(offset=$col.offset, name=$col.name, type=$col.type)));
let newRows = $tds1.rows->concatenate($tdss.rows)->map(r|^TDSRow(parent=$tds,values=$r.values));
$tds->mutateAdd('rows', $newRows);
$tds;
}
function
{doc.doc = 'Append the rows from the second TDS to the rows in the first TDS. N.B. The column must be the same for the two input TDSs'}
meta::pure::tds::concatenate(tds1:TabularDataSet[1], tds2:TabularDataSet[1]):TabularDataSet[1]
{
$tds1->concatenate($tds2->toOneMany());
}
function meta::pure::tds::validatePaths(paths:Path[*]):Boolean[1]
{
let names = $paths->map(p|
let lastElement = $p.path->last()->cast(@PropertyPathElement)->toOne();
assert($lastElement.property->functionReturnType().rawType->toOne()->instanceOf(DataType), 'All the paths provided to project need to end with a DataType');
if($p.name == '',|$lastElement.property.name;,|$p.name);
);
assertEquals($names->size(), $names->removeDuplicates()->size(), 'The list of Paths provided to project can\'t contain duplicate names. Please use ! to provide an alias. Example:#/Person/lastName!newAlias#');
}
function
{doc.doc = 'Select the value for the specified column (based on its name) from each of the rows in the TDS provided'}
meta::pure::tds::columnValues(tds:TabularDataSet[1], pathAsString:String[1]):Any[*]
{
columnValues($tds, $tds.columns->map(c | $c.name)->indexOf($pathAsString));
}
function
{doc.doc = 'Select the value for the specified column (based on its index) from each of the rows in the TDS provided'}
meta::pure::tds::columnValues(tds:TabularDataSet[1], columnIndex:Integer[1]):Any[*]
{
$tds.rows->map(row | $row.values->at($columnIndex));
}
function
{doc.doc = 'Rename the specified column in the provided TDS to a new name'}
meta::pure::tds::renameColumn(tds:TabularDataSet[1], oldColumnName:String[1], newColumnName:String[1]):TabularDataSet[1]
{
$tds->renameColumns(pair($oldColumnName, $newColumnName));
}
function
{doc.doc = 'Rename the specified columns in the provided TDS to a new name'}
meta::pure::tds::renameColumns(tds:TabularDataSet[1], columnMappings : Pair[*]):TabularDataSet[1]
{
let columnMappingsMap = $columnMappings->newMap();
let newTds = ^TabularDataSet(columns=$tds.columns->map(c |
let newName = $columnMappingsMap->get($c.name);
if($newName->isNotEmpty(), | ^$c(name=$newName->toOne()), | $c);
));
$newTds->mutateAdd('rows', $tds.rows->map(r | ^TDSRow(parent=$newTds,values=$r.values)));
$newTds;
}
function
{doc.doc = 'Add additional new calculated columns to the provided TDS'}
meta::pure::tds::extend(tds:TabularDataSet[1], newColumnFunctions:BasicColumnSpecification[*]):TabularDataSet[1]
{
let newTds = ^TabularDataSet(
columns=$tds.columns->concatenate($newColumnFunctions->size()->range()->map(index |
let col = $newColumnFunctions->at($index);
^TDSColumn(offset=$tds.columns->size() + $index, name=$col.name, type=$col.func->functionReturnType().rawType->toOne()->cast(@DataType));
)));
//todo: remove this by making parent an association
$newTds->mutateAdd('rows', $tds.rows->map(r | ^TDSRow(parent=$newTds,
values=$r.values->concatenate(
$newColumnFunctions.func->map(f |
let value = $f->evaluate(^List(values=[$r]));
if($value->isEmpty(), |^TDSNull(), |$value);
)
)
)
)
);
$newTds;
}
function
{doc.doc = 'Project the specified calculated columns from the provided TDS. This is similar to extend, but rather than adding the columns it replaces all of the existing ones '}
meta::pure::tds::project(tds:TabularDataSet[1], columnFunctions:ColumnSpecification[*]):TabularDataSet[1]
{
let newColumnFunctions = $columnFunctions->filter(c|$c->instanceOf(BasicColumnSpecification))->cast(@BasicColumnSpecification);
let newTds = ^TabularDataSet(
columns=$newColumnFunctions->size()->range()->map(index |
let col = $newColumnFunctions->at($index);
^TDSColumn(offset=$index, name=$col.name, type=$col.func->functionReturnType().rawType->toOne()->cast(@DataType));
));
//todo: remove this by making parent an association
$newTds->mutateAdd('rows', $tds.rows->map(r | ^TDSRow(parent=$newTds,
values=$newColumnFunctions.func->map(f |
let value = $f->evaluate(^List(values=[$r]));
if($value->isEmpty(), |^TDSNull(), |$value);
)
)
)
);
$newTds;
}
function
{doc.doc = 'Restrict / reduce the columns from within the provided TDS'}
meta::pure::tds::restrict(tds:TabularDataSet[1], columnNames : String[*]):TabularDataSet[1]
{
assertEmpty($columnNames->removeAll($tds.columns.name), | 'Columns not found in input TDS: ' + $columnNames->removeAll($tds.columns.name)->joinStrings('[', ',', ']'));
let origCols = $columnNames->map(columnName|$tds.columns->filter(c|$c.name == $columnName)->toOne());
let newCols = $origCols->size()->range()->map(index |let origCol = $origCols->at($index); ^$origCol(offset=$index););
let newTds = ^TabularDataSet(
columns=$newCols
);
let restrictedRows = $tds.rows->map(r | ^TDSRow(values = $origCols.offset->map(offset|$r.values->at($offset)), parent=$newTds));
//todo: remove this by making parent an association
$newTds->mutateAdd('rows',$restrictedRows);
$newTds;
}
function
{doc.doc = 'Restrict / reduce columns in provided TDS and return TDS with distinct rows'}
meta::pure::tds::restrictDistinct(tds : TabularDataSet[1], columnNames : String[*]):TabularDataSet[1]
{
$tds->restrict($columnNames)->distinct()
}
function
{doc.doc = 'Aggregate the data within the provided TDS using the specified groupings and aggregation functions'}
meta::pure::tds::groupBy(tds:TabularDataSet[1], columns:String[*], aggValues:meta::pure::tds::AggregateValue[*]):TabularDataSet[1]
{
let groupByOutputCols = range($columns->size())->map(index|
let name = $columns->at($index);
let colType = $tds.columnByName($name)->toOne().type;
^TDSColumn(offset = $index,
type = $colType,
name = $name
);
);
let aggOutputCols = range($aggValues->size())->map(index|
let name = $aggValues->at($index).name;
let colType = $aggValues->at($index).aggregateFn->functionReturnType().rawType->toOne()->cast(@DataType);
^TDSColumn(offset = $index + $columns->size(),
type = $colType,
name = $name
);
);
let outputCols = $groupByOutputCols->concatenate($aggOutputCols);
let newTds = ^TabularDataSet(
columns = $outputCols);
let outputRows = $tds.rows->aggRows($columns,
$columns->map(name|agg($name, row|$row.get($name), values|$values->at(0)))->concatenate($aggValues),
$newTds
);
$newTds->mutateAdd('rows', $outputRows);
$newTds;
}
function <> meta::pure::tds::aggRows(rows : TDSRow[*], columnNames : String[*], aggValues:meta::pure::tds::AggregateValue[*], parent:TabularDataSet[1]) : TDSRow[*]
{
if ($columnNames->isEmpty(), {|
let values = $aggValues->map(aggValue|
let rowMapValues = $rows->map(row|$aggValue.mapFn->eval($row));
let value = $aggValue.aggregateFn->eval($rowMapValues);
if($value->isEmpty(), |^TDSNull(), |$value);
);
^TDSRow(values = $values, parent=$parent);
}, {|
let col = $columnNames->at(0);
let distinctColValues = $rows->map(row|$row.get($col))->distinct();
$distinctColValues->map(colValue|
$rows->filter(row|$row.get($col) == $colValue)->aggRows($columnNames->tail(), $aggValues, $parent)
);
});
}
function
{doc.doc = 'Helper function to create a default column name for the specified path'}
meta::pure::tds::buildColumnNameOutOfPath(p:Path[1]):String[1]
{
if($p.name=='',|$p.path->last()->cast(@PropertyPathElement).property.name->toOne(),|$p.name)->toOne()
}
function
{doc.doc = 'Helper function to create sort information with the specified column in descending order'}
meta::pure::tds::desc(column:String[1]):SortInformation[1]
{
^SortInformation(column=$column, direction=SortDirection.DESC)
}
function
{doc.doc = 'Helper function to create sort information with the specified column in ascending order'}
meta::pure::tds::asc(column:String[1]):SortInformation[1]
{
^SortInformation(column=$column, direction=SortDirection.ASC)
}
function meta::pure::tds::sortAsc(column:Function<{T[1]->Any[1]}>[1]): ColumnSort[1]
{
^ColumnSort
(
column=$column,
direction= SortDirection.ASC
)
}
function meta::pure::tds::window(functions:Function<{T[1]->Any[1]}>[*]):Window[1]
{
^Window
(
partition=$functions
)
}
function meta::pure::tds::sortDesc(column:Function<{T[1]->Any[1]}>[1]): ColumnSort[1]
{
^ColumnSort
(
column=$column,
direction= SortDirection.DESC
)
}
function meta::pure::tds::func(map:FunctionDefinition<{T[1]->Any[1]}>[1], agg:FunctionDefinition<{Number[*]->Number[0..1]}>[1]):OlapAggregation[1]
{
^OlapAggregation
(
mapFn= $map,
aggregateFn= $agg
)
}
function meta::pure::tds::extractOperation(o:OlapOperation[1]): FunctionDefinition[1]
{
$o->match([
a:OlapAggregation[1]| $a.aggregateFn,
r: OlapRank[1]| $r.rankFn,
t: TdsOlapAggregation[1]| $t.func,
tr: TdsOlapRank[1]| $tr.func
]
)
}
/*OLAP TDS protocol*/
function meta::pure::tds::olapGroupBy(tds:TabularDataSet[1], columns:String[*], sortBy:SortInformation[0..1],operation:meta::pure::tds::OlapOperation[1],columnName:String[1]):TabularDataSet[1]
{
fail('This Function is not meant for use outside the context of an execute()');
$tds;
}
function meta::pure::tds::olapGroupBy(tds:TabularDataSet[1], columns:String[*], operation:meta::pure::tds::OlapOperation[1],columnName:String[1]):TabularDataSet[1]
{
fail('This Function is not meant for use outside the context of an execute()');
$tds;
}
function meta::pure::tds::olapGroupBy(tds:TabularDataSet[1], operation:meta::pure::tds::OlapOperation[1],columnName:String[1]):TabularDataSet[1]
{
fail('This Function is not meant for use outside the context of an execute()');
$tds;
}
function meta::pure::tds::olapGroupBy(tds:TabularDataSet[1], sort:SortInformation[0..1], operation:meta::pure::tds::OlapOperation[1],columnName:String[1]):TabularDataSet[1]
{
fail('This Function is not meant for use outside the context of an execute()');
$tds;
}
function meta::pure::tds::olapGroupBy(tds:TabularDataSet[1], columns:String[*], sortBy:SortInformation[0..1],operation:FunctionDefinition<{TDSRow[*]->Map[1]}>[1],columnName:String[1]):TabularDataSet[1]
{
olapGroupBy($tds,$columns,$sortBy,func($operation),$columnName);
}
function meta::pure::tds::olapGroupBy(tds:TabularDataSet[1], columns:String[*], operation:FunctionDefinition<{TDSRow[*]->Map[1]}>[1],columnName:String[1]):TabularDataSet[1]
{
olapGroupBy($tds,$columns,func($operation),$columnName);
}
function meta::pure::tds::olapGroupBy(tds:TabularDataSet[1], operation:FunctionDefinition<{TDSRow[*]->Map[1]}>[1],columnName:String[1]):TabularDataSet[1]
{
olapGroupBy($tds,func($operation),$columnName);
}
function meta::pure::tds::olapGroupBy(tds:TabularDataSet[1], sortBy:SortInformation[0..1], operation:FunctionDefinition<{TDSRow[*]->Map[1]}>[1],columnName:String[1]):TabularDataSet[1]
{
olapGroupBy($tds,$sortBy,func($operation),$columnName);
}
function meta::pure::tds::func(col:String[1], f: FunctionDefinition<{Number[*]->Number[0..1]}>[1]): TdsOlapAggregation[1]
{
^TdsOlapAggregation(colName = $col, func = $f);
}
function meta::pure::tds::func(f: FunctionDefinition<{TDSRow[*]->Map[1]}>[1]): TdsOlapRank[1]
{
^TdsOlapRank(func = $f);
}
Class meta::pure::tds::TdsOlapAggregation extends meta::pure::tds::OlapOperation
{
colName : String[1] ;
func: FunctionDefinition<{Number[*]->Number[0..1]}>[1];
}
Class meta::pure::tds::TdsOlapRank extends meta::pure::tds::OlapOperation
{
func:FunctionDefinition<{T[*]->Map[1]}>[1];
}
// Multi In
function meta::pure::tds::tdsContains(object: T[1], functions: Function<{T[1]->Any[0..1]}>[*], tds: TabularDataSet[1]):Boolean[1]
{
assert($functions->isNotEmpty(), 'Functions cannot be empty');
assert($tds.columns->size() == $functions->size(), | 'Number of functions doesn\'t match TDS columns, Functions number: ' + $functions->size()->toString() + ', TDSColumns number: ' + $tds.columns->size()->toString());
$tds.rows->exists(r | $r.values == $functions->map(k | $k->eval($object)));
}
function meta::pure::tds::tdsContains(object: T[1], functions: Function<{T[1]->Any[0..1]}>[*], ids:String[*], tds: TabularDataSet[1], crossOperation: Function<{TDSRow[1],TDSRow[1]->Boolean[1]}>[1]):Boolean[1]
{
assert($functions->isNotEmpty(), 'Functions cannot be empty');
assert($tds.columns->size() == $functions->size(), | 'Number of functions doesn\'t match TDS columns, Functions number: ' + $functions->size()->toString() + ', TDSColumns number: ' + $tds.columns->size()->toString());
let projected = $object->project($functions, $ids);
($projected.rows->size() != 0) && $tds.rows->exists(r | $crossOperation->eval($projected.rows->at(0), $r));
}
function meta::pure::tds::groupBy(set:K[*], functions:meta::pure::metamodel::function::Function<{K[1]->Any[*]}>[*], aggValues:meta::pure::functions::collection::AggregateValue[*], ids:String[*]):TabularDataSet[1]
{
let grouped = $set->fold({value,map|let key = ^List(values=$functions->map(f | let val = $f->eval($value);
if($val->isEmpty(), | ^TDSNull(), | $val);
));
let currentList = $map->get($key);
$map->put($key, ^List(values=if($currentList->isEmpty(), |$value, |$currentList->toOne().values->concatenate($value))));
},^Map,List>());
let columnInfo = $ids->zip($functions->concatenate($aggValues.aggregateFn));
let columns = $columnInfo->map(info | ^TDSColumn(
name = $info.first,
offset = $columnInfo->indexOf($info),
type = $info.second->functionReturnType().rawType->toOne()->cast(@DataType)));
let tds = ^TabularDataSet(columns=$columns);
let rows = $grouped->keys()->map({key | let listObjs = $grouped->get($key);
let aggResults = $aggValues->map(a | let vals = $listObjs.values->map(o | $a.mapFn->eval($o));
if ($vals->isEmpty(), | ^TDSNull(), | $a.aggregateFn->eval($vals)->cast(@Any));
);
^TDSRow(values=$key.values->concatenate($aggResults), parent=$tds);
});
$tds->mutateAdd('rows', $rows);
$tds;
}
function meta::pure::tds::groupByWithWindowSubset(set:K[*], functions:meta::pure::metamodel::function::Function<{K[1]->Any[*]}>[*], aggValues:meta::pure::functions::collection::AggregateValue[*], ids:String[*] , subSelectIds:String[*] , subAggIds:String[*] ):TabularDataSet[1]
{
let newAggValues = $aggValues->filter( g| $subAggIds->map(i |$ids->indexOf($i))->contains($aggValues->indexOf($g) + $functions->size() ))->sort({a1, a2 | compare($subAggIds->indexOf($ids->at($aggValues->indexOf($a1) + $functions->size())) , $subAggIds->indexOf($ids->at($aggValues->indexOf($a2) + $functions->size()))) });
let newIds = $subSelectIds->concatenate($subAggIds);
let subSetfunctions = $functions->filter(f | $subSelectIds->map(i | $ids->indexOf($i))->contains($functions->indexOf($f)) )->sort({a1, a2 | compare($ids->indexOf($ids->at($functions->indexOf($a1))) , $ids->indexOf($ids->at($functions->indexOf($a2)))) });
meta::pure::tds::groupBy( $set, $subSetfunctions ,$newAggValues , $newIds );
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy