Charles Dick 6ddf8a8795 refactor aggrow to make adding new sources of data easier
Reviewed By: michalgr

Differential Revision: D3961648

fbshipit-source-id: 3c77d3c1352fd89e12163eee393ffcebe09ea8e3
2016-10-25 07:13:51 -07:00

763 lines
91 KiB
JavaScript

/**
* Copyright (c) 2016-present, Facebook, Inc.
* All rights reserved.
*
* This source code is licensed under the BSD-style license found in the
* LICENSE file in the root directory of this source tree. An additional grant
* of patent rights can be found in the PATENTS file in the same directory.
*/
'use strict';
/*eslint no-bitwise: "off"*/
/*eslint no-console-disallow: "off"*/
// TODO: future features
// put in a module.exports
// filtering / search
// pivot around frames in the middle of a stack by callers / callees
// graphing?
function StringInterner(){// eslint-disable-line no-unused-vars
var strings=[];
var ids={};
return{
intern:function internString(s){
var find=ids[s];
if(find===undefined){
var id=strings.length;
ids[s]=id;
strings.push(s);
return id;
}else{
return find;
}
},
get:function getString(id){
return strings[id];
}};
}
function StackRegistry(){// eslint-disable-line no-unused-vars
return{
root:{id:0},
nodeCount:1,
maxDepth:-1,
stackIdMap:null,
insert:function insertNode(parent,frameId){
if(this.stackIdMap!==null){
throw'stacks already flattened';
}
var node=parent[frameId];
if(node===undefined){
node={id:this.nodeCount};
this.nodeCount++;
parent[frameId]=node;
}
return node;
},
get:function getStackArray(id){
return this.stackIdMap[id];
},
flatten:function flattenStacks(){
if(this.stackIdMap!==null){
return;
}
var stackFrameCount=0;
function countStacks(tree,depth){
var leaf=true;
for(var frameId in tree){
if(frameId!=='id'){
leaf=countStacks(tree[frameId],depth+1);
}
}
if(leaf){
stackFrameCount+=depth;
}
return false;
}
countStacks(this.root,0);
console.log('size needed to store stacks: '+(stackFrameCount*4).toString()+'B');
var stackIdMap=new Array(this.nodeCount);
var stackArray=new Int32Array(stackFrameCount);
var maxStackDepth=0;
stackFrameCount=0;
function flattenStacksImpl(tree,stack){
var childStack=void 0;
maxStackDepth=Math.max(maxStackDepth,stack.length);
for(var frameId in tree){
if(frameId!=='id'){
stack.push(Number(frameId));
childStack=flattenStacksImpl(tree[frameId],stack);
stack.pop();
}
}
var id=tree.id;
if(id<0||id>=stackIdMap.length||stackIdMap[id]!==undefined){
throw'invalid stack id!';
}
if(childStack!==undefined){
// each child must have our stack as a prefix, so just use that
stackIdMap[id]=childStack.subarray(0,stack.length);
}else{
var newStack=stackArray.subarray(stackFrameCount,stackFrameCount+stack.length);
stackFrameCount+=stack.length;
for(var i=0;i<stack.length;i++){
newStack[i]=stack[i];
}
stackIdMap[id]=newStack;
}
return stackIdMap[id];
}
flattenStacksImpl(this.root,[]);
this.root=null;
this.stackIdMap=stackIdMap;
this.maxDepth=maxStackDepth;
}};
}
function AggrowData(columns){// eslint-disable-line no-unused-vars
var columnCount=columns.length;
var columnConverter=columns.map(function(c){
switch(c.type){
case'int':// stores raw value
return function(i){return i;};
case'string':// stores interned id of string
return function(s){return c.strings.intern(s);};
case'stack':// stores id of stack node
return function(s){return s.id;};
default:
throw'unknown AggrowData column type';}
});
return{
data:new Int32Array(0),
columns:columns,
rowCount:0,
rowInserter:function rowInserter(numRows){
console.log(
'increasing row data from '+(this.data.length*4).toLocaleString()+' B to '+
(this.data.length*4+numRows*columnCount*4).toLocaleString()+' B');
var newData=new Int32Array(this.data.length+numRows*columnCount);
newData.set(this.data);
var currOffset=this.data.length;
var endOffset=newData.length;
this.data=newData;
this.rowCount=newData.length/columnCount;
return{
insertRow:function insertRow(){
if(currOffset>=endOffset){
throw'tried to insert data off end of added range';
}
if(arguments.length!==columnCount){
throw'expected data for '+columnCount.toString()+' columns, got'+
arguments.length.toString()+' columns';
}
for(var i=0;i<arguments.length;i++){
newData[currOffset+i]=columnConverter[i](arguments[i]);
}
currOffset+=columnCount;
},
done:function done(){
if(currOffset!==endOffset){
throw'unfilled rows';
}
}};
}};
}
function Aggrow(aggrowData){
var columns=aggrowData.columns;
var columnCount=columns.length;
var data=aggrowData.data;
function columnIndex(columnName,columnType){
var index=columns.findIndex(function(c){return c.name===columnName&&c.type===columnType;});
if(index<0){
throw'did not find data column '+columnName+' with type '+columnType;
}
return index;
}
for(var i=0;i<columns.length;i++){
if(columns[i].type==='stack'){
columns[i].stacks.flatten();
}
}
return{
expander:new AggrowExpander(aggrowData.rowCount),
addSumAggregator:function addSumAggregator(aggregatorName,columnName){
var index=columnIndex(columnName,'int');
return this.expander.addAggregator(
aggregatorName,
function aggregateSize(indices){
var size=0;
for(var _i=0;_i<indices.length;_i++){
var row=indices[_i];
size+=data[row*columnCount+index];
}
return size;
},
function(value){return value.toLocaleString();},
function(a,b){return b-a;});
},
addCountAggregator:function addCountAggregator(aggregatorName){
return this.expander.addAggregator(
aggregatorName,
function aggregateCount(indices){
return indices.length;
},
function(value){return value.toLocaleString();},
function(a,b){return b-a;});
},
addStringExpander:function addStringExpander(expanderName,columnName){
var index=columnIndex(columnName,'string');
var strings=columns[index].strings;
return this.expander.addFieldExpander(
expanderName,
function(row){return strings.get(data[row*columnCount+index]);},
function(rowA,rowB){return data[rowA*columnCount+index]-data[rowB*columnCount+index];});
},
addNumberExpander:function addNumberExpander(expanderName,columnName){
var index=columnIndex(columnName,'int');
return this.expander.addFieldExpander(
expanderName,
function(row){return data[row*columnCount+index].toLocaleString();},
function(rowA,rowB){return data[rowA*columnCount+index]-data[rowB*columnCount+index];});
},
addPointerExpander:function addPointerExpander(expanderName,columnName){
var index=columnIndex(columnName,'int');
return this.expander.addFieldExpander(
expanderName,
function(row){return'0x'+(data[row*columnCount+index]>>>0).toString();},
function(rowA,rowB){return data[rowA*columnCount+index]-data[rowB*columnCount+index];});
},
addStackExpander:function addStackExpander(expanderName,columnName,formatter){
// TODO: options for caller/callee, pivoting
var index=columnIndex(columnName,'stack');
var stacks=columns[index].stacks;
return this.expander.addCalleeStackExpander(
expanderName,
stacks.maxDepth,
function(row){return stacks.get(data[row*columnCount+index]);},
formatter);
}};
}
function AggrowExpander(numRows){// eslint-disable-line no-unused-vars
// expander ID definitions
var FIELD_EXPANDER_ID_MIN=0x0000;
var FIELD_EXPANDER_ID_MAX=0x7fff;
var STACK_EXPANDER_ID_MIN=0x8000;
var STACK_EXPANDER_ID_MAX=0xffff;
// used for row.expander which reference state.activeExpanders (with frame index masked in)
var INVALID_ACTIVE_EXPANDER=-1;
var ACTIVE_EXPANDER_MASK=0xffff;
var ACTIVE_EXPANDER_FRAME_SHIFT=16;
// aggregator ID definitions
var AGGREGATOR_ID_MAX=0xffff;
// active aggragators can have sort order changed in the reference
var ACTIVE_AGGREGATOR_MASK=0xffff;
var ACTIVE_AGGREGATOR_ASC_BIT=0x10000;
// tree node state definitions
var NODE_EXPANDED_BIT=0x0001;// this row is expanded
var NODE_REAGGREGATE_BIT=0x0002;// children need aggregates
var NODE_REORDER_BIT=0x0004;// children need to be sorted
var NODE_REPOSITION_BIT=0x0008;// children need position
var NODE_INDENT_SHIFT=16;
function calleeFrameIdGetter(stack,depth){
return stack[depth];
}
function callerFrameIdGetter(stack,depth){
return stack[stack.length-depth-1];
}
function createStackComparers(stackGetter,frameIdGetter,maxStackDepth){
var comparers=new Array(maxStackDepth);var _loop=function _loop(
depth){
var captureDepth=depth;// NB: to capture depth per loop iteration
comparers[depth]=function calleeStackComparer(rowA,rowB){
var a=stackGetter(rowA);
var b=stackGetter(rowB);
// NB: we put the stacks that are too short at the top,
// so they can be grouped into the '<exclusive>' bucket
if(a.length<=captureDepth&&b.length<=captureDepth){
return 0;
}else if(a.length<=captureDepth){
return-1;
}else if(b.length<=captureDepth){
return 1;
}
return frameIdGetter(a,captureDepth)-frameIdGetter(b,captureDepth);
};};for(var depth=0;depth<maxStackDepth;depth++){_loop(depth);
}
return comparers;
}
function createTreeNode(parent,label,indices,expander){
var indent=parent===null?0:(parent.state>>>NODE_INDENT_SHIFT)+1;
var state=NODE_REPOSITION_BIT|
NODE_REAGGREGATE_BIT|
NODE_REORDER_BIT|
indent<<NODE_INDENT_SHIFT;
return{
parent:parent,// null if root
children:null,// array of children nodes
label:label,// string to show in UI
indices:indices,// row indices under this node
aggregates:null,// result of aggregate on indices
expander:expander,// index into state.activeExpanders
top:0,// y position of top row (in rows)
height:1,// number of rows including children
state:state};
}
function noSortOrder(a,b){
return 0;
}
var indices=new Int32Array(numRows);
for(var i=0;i<numRows;i++){
indices[i]=i;
}
var state={
fieldExpanders:[],// tree expanders that expand on simple values
stackExpanders:[],// tree expanders that expand stacks
activeExpanders:[],// index into field or stack expanders, hierarchy of tree
aggregators:[],// all available aggregators, might not be used
activeAggregators:[],// index into aggregators, to actually compute
sorter:noSortOrder,// compare function that uses sortOrder to sort row.children
root:createTreeNode(null,'<root>',indices,INVALID_ACTIVE_EXPANDER)};
function evaluateAggregate(row){
var activeAggregators=state.activeAggregators;
var aggregates=new Array(activeAggregators.length);
for(var j=0;j<activeAggregators.length;j++){
var aggregator=state.aggregators[activeAggregators[j]];
aggregates[j]=aggregator.aggregator(row.indices);
}
row.aggregates=aggregates;
row.state|=NODE_REAGGREGATE_BIT;
}
function evaluateAggregates(row){
if((row.state&NODE_EXPANDED_BIT)!==0){
var children=row.children;
for(var _i2=0;_i2<children.length;_i2++){
evaluateAggregate(children[_i2]);
}
row.state|=NODE_REORDER_BIT;
}
row.state^=NODE_REAGGREGATE_BIT;
}
function evaluateOrder(row){
if((row.state&NODE_EXPANDED_BIT)!==0){
var children=row.children;
for(var _i3=0;_i3<children.length;_i3++){
var child=children[_i3];
child.state|=NODE_REORDER_BIT;
}
children.sort(state.sorter);
row.state|=NODE_REPOSITION_BIT;
}
row.state^=NODE_REORDER_BIT;
}
function evaluatePosition(row){
if((row.state&NODE_EXPANDED_BIT)!==0){
var children=row.children;
var childTop=row.top+1;
for(var _i4=0;_i4<children.length;_i4++){
var child=children[_i4];
if(child.top!==childTop){
child.top=childTop;
child.state|=NODE_REPOSITION_BIT;
}
childTop+=child.height;
}
}
row.state^=NODE_REPOSITION_BIT;
}
function getRowsImpl(row,top,height,result){
if((row.state&NODE_REAGGREGATE_BIT)!==0){
evaluateAggregates(row);
}
if((row.state&NODE_REORDER_BIT)!==0){
evaluateOrder(row);
}
if((row.state&NODE_REPOSITION_BIT)!==0){
evaluatePosition(row);
}
if(row.top>=top&&row.top<top+height){
if(result[row.top-top]!=null){
throw'getRows put more than one row at position '+row.top+' into result';
}
result[row.top-top]=row;
}
if((row.state&NODE_EXPANDED_BIT)!==0){
var children=row.children;
for(var _i5=0;_i5<children.length;_i5++){
var child=children[_i5];
if(child.top<top+height&&top<child.top+child.height){
getRowsImpl(child,top,height,result);
}
}
}
}
function updateHeight(row,heightChange){
while(row!==null){
row.height+=heightChange;
row.state|=NODE_REPOSITION_BIT;
row=row.parent;
}
}
function addChildrenWithFieldExpander(row,expander,nextActiveIndex){
var rowIndices=row.indices;
var comparer=expander.comparer;
rowIndices.sort(comparer);
var begin=0;
var end=1;
row.children=[];
while(end<rowIndices.length){
if(comparer(rowIndices[begin],rowIndices[end])!==0){
row.children.push(createTreeNode(
row,
expander.name+': '+expander.formatter(rowIndices[begin]),
rowIndices.subarray(begin,end),
nextActiveIndex));
begin=end;
}
end++;
}
row.children.push(createTreeNode(
row,
expander.name+': '+expander.formatter(rowIndices[begin]),
rowIndices.subarray(begin,end),
nextActiveIndex));
}
function addChildrenWithStackExpander(row,expander,activeIndex,depth,nextActiveIndex){
var rowIndices=row.indices;
var stackGetter=expander.stackGetter;
var frameIdGetter=expander.frameIdGetter;
var frameGetter=expander.frameGetter;
var comparer=expander.comparers[depth];
var expandNextFrame=activeIndex|depth+1<<ACTIVE_EXPANDER_FRAME_SHIFT;
rowIndices.sort(comparer);
var columnName='';
if(depth===0){
columnName=expander.name+': ';
}
// put all the too-short stacks under <exclusive>
var begin=0;
var beginStack=null;
row.children=[];
while(begin<rowIndices.length){
beginStack=stackGetter(rowIndices[begin]);
if(beginStack.length>depth){
break;
}
begin++;
}
if(begin>0){
row.children.push(createTreeNode(
row,
columnName+'<exclusive>',
rowIndices.subarray(0,begin),
nextActiveIndex));
}
// aggregate the rest under frames
if(begin<rowIndices.length){
var end=begin+1;
while(end<rowIndices.length){
var endStack=stackGetter(rowIndices[end]);
if(frameIdGetter(beginStack,depth)!==frameIdGetter(endStack,depth)){
row.children.push(createTreeNode(
row,
columnName+frameGetter(frameIdGetter(beginStack,depth)),
rowIndices.subarray(begin,end),
expandNextFrame));
begin=end;
beginStack=endStack;
}
end++;
}
row.children.push(createTreeNode(
row,
columnName+frameGetter(frameIdGetter(beginStack,depth)),
rowIndices.subarray(begin,end),
expandNextFrame));
}
}
function contractRow(row){
if((row.state&NODE_EXPANDED_BIT)===0){
throw'can not contract row, already contracted';
}
row.state^=NODE_EXPANDED_BIT;
var heightChange=1-row.height;
updateHeight(row,heightChange);
}
function pruneExpanders(row,oldExpander,newExpander){
row.state|=NODE_REPOSITION_BIT;
if(row.expander===oldExpander){
row.state|=NODE_REAGGREGATE_BIT|NODE_REORDER_BIT|NODE_REPOSITION_BIT;
if((row.state&NODE_EXPANDED_BIT)!==0){
contractRow(row);
}
row.children=null;
row.expander=newExpander;
}else{
row.state|=NODE_REPOSITION_BIT;
var children=row.children;
if(children!=null){
for(var _i6=0;_i6<children.length;_i6++){
var child=children[_i6];
pruneExpanders(child,oldExpander,newExpander);
}
}
}
}
return{
addFieldExpander:function addFieldExpander(name,formatter,comparer){
if(FIELD_EXPANDER_ID_MIN+state.fieldExpanders.length>=FIELD_EXPANDER_ID_MAX){
throw'too many field expanders!';
}
state.fieldExpanders.push({
name:name,// name for column
formatter:formatter,// row index -> display string
comparer:comparer});
return FIELD_EXPANDER_ID_MIN+state.fieldExpanders.length-1;
},
addCalleeStackExpander:function addCalleeStackExpander(name,maxStackDepth,stackGetter,frameGetter){
if(STACK_EXPANDER_ID_MIN+state.fieldExpanders.length>=STACK_EXPANDER_ID_MAX){
throw'too many stack expanders!';
}
state.stackExpanders.push({
name:name,// name for column
stackGetter:stackGetter,// row index -> stack array
comparers:createStackComparers(stackGetter,calleeFrameIdGetter,maxStackDepth),// depth -> comparer
frameIdGetter:calleeFrameIdGetter,// (stack, depth) -> string id
frameGetter:frameGetter});
return STACK_EXPANDER_ID_MIN+state.stackExpanders.length-1;
},
addCallerStackExpander:function addCallerStackExpander(name,maxStackDepth,stackGetter,frameGetter){
if(STACK_EXPANDER_ID_MIN+state.fieldExpanders.length>=STACK_EXPANDER_ID_MAX){
throw'too many stack expanders!';
}
state.stackExpanders.push({
name:name,
stackGetter:stackGetter,
comparers:createStackComparers(stackGetter,callerFrameIdGetter,maxStackDepth),
frameIdGetter:callerFrameIdGetter,
frameGetter:frameGetter});
return STACK_EXPANDER_ID_MIN+state.stackExpanders.length-1;
},
getExpanders:function getExpanders(){
var expanders=[];
for(var _i7=0;_i7<state.fieldExpanders.length;_i7++){
expanders.push(FIELD_EXPANDER_ID_MIN+_i7);
}
for(var _i8=0;_i8<state.stackExpanders.length;_i8++){
expanders.push(STACK_EXPANDER_ID_MIN+_i8);
}
return expanders;
},
getExpanderName:function getExpanderName(id){
if(id>=FIELD_EXPANDER_ID_MIN&&id<=FIELD_EXPANDER_ID_MAX){
return state.fieldExpanders[id-FIELD_EXPANDER_ID_MIN].name;
}else if(id>=STACK_EXPANDER_ID_MIN&&id<=STACK_EXPANDER_ID_MAX){
return state.stackExpanders[id-STACK_EXPANDER_ID_MIN].name;
}
throw'Unknown expander ID '+id.toString();
},
setActiveExpanders:function setActiveExpanders(ids){
for(var _i9=0;_i9<ids.length;_i9++){
var id=ids[_i9];
if(id>=FIELD_EXPANDER_ID_MIN&&id<=FIELD_EXPANDER_ID_MAX){
if(id-FIELD_EXPANDER_ID_MIN>=state.fieldExpanders.length){
throw'field expander for id '+id.toString()+' does not exist!';
}
}else if(id>=STACK_EXPANDER_ID_MIN&&id<=STACK_EXPANDER_ID_MAX){
if(id-STACK_EXPANDER_ID_MIN>=state.stackExpanders.length){
throw'stack expander for id '+id.toString()+' does not exist!';
}
}
}
for(var _i10=0;_i10<ids.length;_i10++){
if(state.activeExpanders.length<=_i10){
pruneExpanders(state.root,INVALID_ACTIVE_EXPANDER,_i10);
break;
}else if(ids[_i10]!==state.activeExpanders[_i10]){
pruneExpanders(state.root,_i10,_i10);
break;
}
}
// TODO: if ids is prefix of activeExpanders, we need to make an expander invalid
state.activeExpanders=ids.slice();
},
getActiveExpanders:function getActiveExpanders(){
return state.activeExpanders.slice();
},
addAggregator:function addAggregator(name,aggregator,formatter,sorter){
if(state.aggregators.length>=AGGREGATOR_ID_MAX){
throw'too many aggregators!';
}
state.aggregators.push({
name:name,// name for column
aggregator:aggregator,// index array -> aggregate value
formatter:formatter,// aggregate value -> display string
sorter:sorter});
return state.aggregators.length-1;
},
getAggregators:function getAggregators(){
var aggregators=[];
for(var _i11=0;_i11<state.aggregators.length;_i11++){
aggregators.push(_i11);
}
return aggregators;
},
getAggregatorName:function getAggregatorName(id){
return state.aggregators[id&ACTIVE_AGGREGATOR_MASK].name;
},
setActiveAggregators:function setActiveAggregators(ids){
for(var _i12=0;_i12<ids.length;_i12++){
var id=ids[_i12]&ACTIVE_AGGREGATOR_MASK;
if(id<0||id>state.aggregators.length){
throw'aggregator id '+id.toString()+' not valid';
}
}
state.activeAggregators=ids.slice();
// NB: evaluate root here because dirty bit is for children
// so someone has to start with root, and it might as well be right away
evaluateAggregate(state.root);
var sorter=noSortOrder;var _loop2=function _loop2(
_i13){
var ascending=(ids[_i13]&ACTIVE_AGGREGATOR_ASC_BIT)!==0;
var id=ids[_i13]&ACTIVE_AGGREGATOR_MASK;
var comparer=state.aggregators[id].sorter;
var captureSorter=sorter;
var captureIndex=_i13;
sorter=function sorter(a,b){
var c=comparer(a.aggregates[captureIndex],b.aggregates[captureIndex]);
if(c===0){
return captureSorter(a,b);
}
return ascending?-c:c;
};};for(var _i13=ids.length-1;_i13>=0;_i13--){_loop2(_i13);
}
state.sorter=sorter;
state.root.state|=NODE_REORDER_BIT;
},
getActiveAggregators:function getActiveAggregators(){
return state.activeAggregators.slice();
},
getRows:function getRows(top,height){
var result=new Array(height);
for(var _i14=0;_i14<height;_i14++){
result[_i14]=null;
}
getRowsImpl(state.root,top,height,result);
return result;
},
getRowLabel:function getRowLabel(row){
return row.label;
},
getRowIndent:function getRowIndent(row){
return row.state>>>NODE_INDENT_SHIFT;
},
getRowAggregate:function getRowAggregate(row,index){
var aggregator=state.aggregators[state.activeAggregators[index]];
return aggregator.formatter(row.aggregates[index]);
},
getHeight:function getHeight(){
return state.root.height;
},
canExpand:function canExpand(row){
return(row.state&NODE_EXPANDED_BIT)===0&&row.expander!==INVALID_ACTIVE_EXPANDER;
},
canContract:function canContract(row){
return(row.state&NODE_EXPANDED_BIT)!==0;
},
expand:function expand(row){
if((row.state&NODE_EXPANDED_BIT)!==0){
throw'can not expand row, already expanded';
}
if(row.height!==1){
throw'unexpanded row has height '+row.height.toString()+' != 1';
}
if(row.children===null){// first expand, generate children
var activeIndex=row.expander&ACTIVE_EXPANDER_MASK;
var nextActiveIndex=activeIndex+1;// NB: if next is stack, frame is 0
if(nextActiveIndex>=state.activeExpanders.length){
nextActiveIndex=INVALID_ACTIVE_EXPANDER;
}
if(activeIndex>=state.activeExpanders.length){
throw'invalid active expander index '+activeIndex.toString();
}
var exId=state.activeExpanders[activeIndex];
if(exId>=FIELD_EXPANDER_ID_MIN&&
exId<FIELD_EXPANDER_ID_MIN+state.fieldExpanders.length){
var expander=state.fieldExpanders[exId-FIELD_EXPANDER_ID_MIN];
addChildrenWithFieldExpander(row,expander,nextActiveIndex);
}else if(exId>=STACK_EXPANDER_ID_MIN&&
exId<STACK_EXPANDER_ID_MIN+state.stackExpanders.length){
var depth=row.expander>>>ACTIVE_EXPANDER_FRAME_SHIFT;
var _expander=state.stackExpanders[exId-STACK_EXPANDER_ID_MIN];
addChildrenWithStackExpander(row,_expander,activeIndex,depth,nextActiveIndex);
}else{
throw'state.activeIndex '+activeIndex.toString()+
' has invalid expander'+exId.toString();
}
}
row.state|=NODE_EXPANDED_BIT|
NODE_REAGGREGATE_BIT|NODE_REORDER_BIT|NODE_REPOSITION_BIT;
var heightChange=0;
for(var _i15=0;_i15<row.children.length;_i15++){
heightChange+=row.children[_i15].height;
}
updateHeight(row,heightChange);
// if children only contains one node, then expand it as well
if(row.children.length===1&&this.canExpand(row.children[0])){
this.expand(row.children[0]);
}
},
contract:function contract(row){
contractRow(row);
}};
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/aggrow.js"],"names":["StringInterner","strings","ids","intern","internString","s","find","undefined","id","length","push","get","getString","StackRegistry","root","nodeCount","maxDepth","stackIdMap","insert","insertNode","parent","frameId","node","getStackArray","flatten","flattenStacks","stackFrameCount","countStacks","tree","depth","leaf","console","log","toString","Array","stackArray","Int32Array","maxStackDepth","flattenStacksImpl","stack","childStack","Math","max","Number","pop","subarray","newStack","i","AggrowData","columns","columnCount","columnConverter","map","c","type","data","rowCount","rowInserter","numRows","toLocaleString","newData","set","currOffset","endOffset","insertRow","arguments","done","Aggrow","aggrowData","columnIndex","columnName","columnType","index","findIndex","name","stacks","expander","AggrowExpander","addSumAggregator","aggregatorName","addAggregator","aggregateSize","indices","size","row","value","a","b","addCountAggregator","aggregateCount","addStringExpander","expanderName","addFieldExpander","rowA","rowB","addNumberExpander","addPointerExpander","addStackExpander","formatter","addCalleeStackExpander","FIELD_EXPANDER_ID_MIN","FIELD_EXPANDER_ID_MAX","STACK_EXPANDER_ID_MIN","STACK_EXPANDER_ID_MAX","INVALID_ACTIVE_EXPANDER","ACTIVE_EXPANDER_MASK","ACTIVE_EXPANDER_FRAME_SHIFT","AGGREGATOR_ID_MAX","ACTIVE_AGGREGATOR_MASK","ACTIVE_AGGREGATOR_ASC_BIT","NODE_EXPANDED_BIT","NODE_REAGGREGATE_BIT","NODE_REORDER_BIT","NODE_REPOSITION_BIT","NODE_INDENT_SHIFT","calleeFrameIdGetter","callerFrameIdGetter","createStackComparers","stackGetter","frameIdGetter","comparers","captureDepth","calleeStackComparer","createTreeNode","label","indent","state","children","aggregates","top","height","noSortOrder","fieldExpanders","stackExpanders","activeExpanders","aggregators","activeAggregators","sorter","evaluateAggregate","j","aggregator","evaluateAggregates","evaluateOrder","child","sort","evaluatePosition","childTop","getRowsImpl","result","updateHeight","heightChange","addChildrenWithFieldExpander","nextActiveIndex","rowIndices","comparer","begin","end","addChildrenWithStackExpander","activeIndex","frameGetter","expandNextFrame","beginStack","endStack","contractRow","pruneExpanders","oldExpander","newExpander","addCallerStackExpander","getExpanders","expanders","getExpanderName","setActiveExpanders","slice","getActiveExpanders","getAggregators","getAggregatorName","setActiveAggregators","ascending","captureSorter","captureIndex","getActiveAggregators","getRows","getRowLabel","getRowIndent","getRowAggregate","getHeight","canExpand","canContract","expand","exId","contract"],"mappings":"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","file":"aggrow.js","sourcesContent":["/**\n * Copyright (c) 2016-present, Facebook, Inc.\n * All rights reserved.\n *\n * This source code is licensed under the BSD-style license found in the\n * LICENSE file in the root directory of this source tree. An additional grant\n * of patent rights can be found in the PATENTS file in the same directory.\n */\n'use strict';\n/*eslint no-bitwise: \"off\"*/\n/*eslint no-console-disallow: \"off\"*/\n\n// TODO: future features\n// put in a module.exports\n// filtering / search\n// pivot around frames in the middle of a stack by callers / callees\n// graphing?\n\nfunction StringInterner() { // eslint-disable-line no-unused-vars\n  const strings = [];\n  const ids = {};\n  return {\n    intern: function internString(s) {\n      const find = ids[s];\n      if (find === undefined) {\n        const id = strings.length;\n        ids[s] = id;\n        strings.push(s);\n        return id;\n      } else {\n        return find;\n      }\n    },\n    get: function getString(id) {\n      return strings[id];\n    },\n  };\n}\n\nfunction StackRegistry() { // eslint-disable-line no-unused-vars\n  return {\n    root: { id: 0 },\n    nodeCount: 1,\n    maxDepth: -1,\n    stackIdMap: null,\n    insert: function insertNode(parent, frameId) {\n      if (this.stackIdMap !== null) {\n        throw 'stacks already flattened';\n      }\n      let node = parent[frameId];\n      if (node === undefined) {\n        node = { id: this.nodeCount };\n        this.nodeCount++;\n        parent[frameId] = node;\n      }\n      return node;\n    },\n    get: function getStackArray(id) {\n      return this.stackIdMap[id];\n    },\n    flatten: function flattenStacks() {\n      if (this.stackIdMap !== null) {\n        return;\n      }\n      let stackFrameCount = 0;\n      function countStacks(tree, depth) {\n        let leaf = true;\n        for (const frameId in tree) {\n          if (frameId !== 'id') {\n            leaf = countStacks(tree[frameId], depth + 1);\n          }\n        }\n        if (leaf) {\n          stackFrameCount += depth;\n        }\n        return false;\n      }\n      countStacks(this.root, 0);\n      console.log('size needed to store stacks: ' + (stackFrameCount * 4).toString() + 'B');\n      const stackIdMap = new Array(this.nodeCount);\n      const stackArray = new Int32Array(stackFrameCount);\n      let maxStackDepth = 0;\n      stackFrameCount = 0;\n      function flattenStacksImpl(tree, stack) {\n        let childStack;\n        maxStackDepth = Math.max(maxStackDepth, stack.length);\n        for (const frameId in tree) {\n          if (frameId !== 'id') {\n            stack.push(Number(frameId));\n            childStack = flattenStacksImpl(tree[frameId], stack);\n            stack.pop();\n          }\n        }\n\n        const id = tree.id;\n        if (id < 0 || id >= stackIdMap.length || stackIdMap[id] !== undefined) {\n          throw 'invalid stack id!';\n        }\n\n        if (childStack !== undefined) {\n          // each child must have our stack as a prefix, so just use that\n          stackIdMap[id] = childStack.subarray(0, stack.length);\n        } else {\n          const newStack = stackArray.subarray(stackFrameCount, stackFrameCount + stack.length);\n          stackFrameCount += stack.length;\n          for (let i = 0; i < stack.length; i++) {\n            newStack[i] = stack[i];\n          }\n          stackIdMap[id] = newStack;\n        }\n        return stackIdMap[id];\n      }\n      flattenStacksImpl(this.root, []);\n      this.root = null;\n      this.stackIdMap = stackIdMap;\n      this.maxDepth = maxStackDepth;\n    },\n  };\n}\n\nfunction AggrowData(columns) { // eslint-disable-line no-unused-vars\n  const columnCount = columns.length;\n  const columnConverter = columns.map(c => {\n    switch (c.type) {\n      case 'int':     // stores raw value\n        return (i) => i;\n      case 'string':  // stores interned id of string\n        return (s) => c.strings.intern(s);\n      case 'stack':   // stores id of stack node\n        return (s) => s.id;\n      default:\n        throw 'unknown AggrowData column type';\n    }\n  });\n  return {\n    data: new Int32Array(0),\n    columns: columns,\n    rowCount: 0,\n    rowInserter: function rowInserter(numRows) {\n      console.log(\n        'increasing row data from ' + (this.data.length * 4).toLocaleString() + ' B to ' +\n        (this.data.length * 4 + numRows * columnCount * 4).toLocaleString() + ' B'\n      );\n      const newData = new Int32Array(this.data.length + numRows * columnCount);\n      newData.set(this.data);\n      let currOffset = this.data.length;\n      const endOffset = newData.length;\n      this.data = newData;\n      this.rowCount = newData.length / columnCount;\n      return {\n        insertRow: function insertRow() {\n          if (currOffset >= endOffset) {\n            throw 'tried to insert data off end of added range';\n          }\n          if (arguments.length !== columnCount) {\n            throw 'expected data for ' + columnCount.toString() + ' columns, got' +\n              arguments.length.toString() + ' columns';\n          }\n          for (let i = 0; i < arguments.length; i++) {\n            newData[currOffset + i] = columnConverter[i](arguments[i]);\n          }\n          currOffset += columnCount;\n        },\n        done: function done() {\n          if (currOffset !== endOffset) {\n            throw 'unfilled rows';\n          }\n        },\n      };\n    },\n  };\n}\n\nfunction Aggrow(aggrowData) {\n  const columns = aggrowData.columns;\n  const columnCount = columns.length;\n  const data = aggrowData.data;\n  function columnIndex(columnName, columnType) {\n    const index = columns.findIndex(c => c.name === columnName && c.type === columnType);\n    if (index < 0) {\n      throw 'did not find data column ' + columnName + ' with type ' + columnType;\n    }\n    return index;\n  }\n  for (let i = 0; i < columns.length; i++) {\n    if (columns[i].type === 'stack') {\n      columns[i].stacks.flatten();\n    }\n  }\n  return {\n    expander: new AggrowExpander(aggrowData.rowCount),\n    addSumAggregator: function addSumAggregator(aggregatorName, columnName) {\n      const index = columnIndex(columnName, 'int');\n      return this.expander.addAggregator(\n        aggregatorName,\n        function aggregateSize(indices) {\n          let size = 0;\n          for (let i = 0; i < indices.length; i++) {\n            const row = indices[i];\n            size += data[row * columnCount + index];\n          }\n          return size;\n        },\n        (value) => value.toLocaleString(),\n        (a, b) => b - a,\n      );\n    },\n    addCountAggregator: function addCountAggregator(aggregatorName) {\n      return this.expander.addAggregator(\n        aggregatorName,\n        function aggregateCount(indices) {\n          return indices.length;\n        },\n        (value) => value.toLocaleString(),\n        (a, b) => b - a,\n      );\n    },\n    addStringExpander: function addStringExpander(expanderName, columnName) {\n      const index = columnIndex(columnName, 'string');\n      const strings = columns[index].strings;\n      return this.expander.addFieldExpander(\n        expanderName,\n        (row) => strings.get(data[row * columnCount + index]),\n        (rowA, rowB) => data[rowA * columnCount + index] - data[rowB * columnCount + index],\n      );\n    },\n    addNumberExpander: function addNumberExpander(expanderName, columnName) {\n      const index = columnIndex(columnName, 'int');\n      return this.expander.addFieldExpander(\n        expanderName,\n        (row) => data[row * columnCount + index].toLocaleString(),\n        (rowA, rowB) => data[rowA * columnCount + index] - data[rowB * columnCount + index],\n      );\n    },\n    addPointerExpander: function addPointerExpander(expanderName, columnName) {\n      const index = columnIndex(columnName, 'int');\n      return this.expander.addFieldExpander(\n        expanderName,\n        (row) => '0x' + (data[row * columnCount + index] >>> 0).toString(),\n        (rowA, rowB) => data[rowA * columnCount + index] - data[rowB * columnCount + index],\n      );\n    },\n    addStackExpander: function addStackExpander(expanderName, columnName, formatter) {\n      // TODO: options for caller/callee, pivoting\n      const index = columnIndex(columnName, 'stack');\n      const stacks = columns[index].stacks;\n      return this.expander.addCalleeStackExpander(\n        expanderName,\n        stacks.maxDepth,\n        (row) => stacks.get(data[row * columnCount + index]),\n        formatter,\n      );\n    },\n  };\n}\n\nfunction AggrowExpander(numRows) { // eslint-disable-line no-unused-vars\n  // expander ID definitions\n  const FIELD_EXPANDER_ID_MIN       = 0x0000;\n  const FIELD_EXPANDER_ID_MAX       = 0x7fff;\n  const STACK_EXPANDER_ID_MIN       = 0x8000;\n  const STACK_EXPANDER_ID_MAX       = 0xffff;\n\n  // used for row.expander which reference state.activeExpanders (with frame index masked in)\n  const INVALID_ACTIVE_EXPANDER     = -1;\n  const ACTIVE_EXPANDER_MASK        = 0xffff;\n  const ACTIVE_EXPANDER_FRAME_SHIFT = 16;\n\n  // aggregator ID definitions\n  const AGGREGATOR_ID_MAX           = 0xffff;\n\n  // active aggragators can have sort order changed in the reference\n  const ACTIVE_AGGREGATOR_MASK      = 0xffff;\n  const ACTIVE_AGGREGATOR_ASC_BIT   = 0x10000;\n\n  // tree node state definitions\n  const NODE_EXPANDED_BIT           = 0x0001; // this row is expanded\n  const NODE_REAGGREGATE_BIT        = 0x0002; // children need aggregates\n  const NODE_REORDER_BIT            = 0x0004; // children need to be sorted\n  const NODE_REPOSITION_BIT         = 0x0008; // children need position\n  const NODE_INDENT_SHIFT           = 16;\n\n  function calleeFrameIdGetter(stack, depth) {\n    return stack[depth];\n  }\n\n  function callerFrameIdGetter(stack, depth) {\n    return stack[stack.length - depth - 1];\n  }\n\n  function createStackComparers(stackGetter, frameIdGetter, maxStackDepth) {\n    const comparers = new Array(maxStackDepth);\n    for (let depth = 0; depth < maxStackDepth; depth++) {\n      const captureDepth = depth; // NB: to capture depth per loop iteration\n      comparers[depth] = function calleeStackComparer(rowA, rowB) {\n        const a = stackGetter(rowA);\n        const b = stackGetter(rowB);\n        // NB: we put the stacks that are too short at the top,\n        // so they can be grouped into the '<exclusive>' bucket\n        if (a.length <= captureDepth && b.length <= captureDepth) {\n            return 0;\n        } else if (a.length <= captureDepth) {\n            return -1;\n        } else if (b.length <= captureDepth) {\n            return 1;\n        }\n        return frameIdGetter(a, captureDepth) - frameIdGetter(b, captureDepth);\n      };\n    }\n    return comparers;\n  }\n\n  function createTreeNode(parent, label, indices, expander) {\n    const indent = parent === null ? 0 : (parent.state >>> NODE_INDENT_SHIFT) + 1;\n    const state = NODE_REPOSITION_BIT |\n      NODE_REAGGREGATE_BIT |\n      NODE_REORDER_BIT |\n      (indent << NODE_INDENT_SHIFT);\n    return {\n      parent: parent,     // null if root\n      children: null,     // array of children nodes\n      label: label,       // string to show in UI\n      indices: indices,   // row indices under this node\n      aggregates: null,   // result of aggregate on indices\n      expander: expander, // index into state.activeExpanders\n      top: 0,             // y position of top row (in rows)\n      height: 1,          // number of rows including children\n      state: state,       // see NODE_* definitions above\n    };\n  }\n\n  function noSortOrder(a, b) {\n    return 0;\n  }\n\n  const indices = new Int32Array(numRows);\n  for (let i = 0; i < numRows; i++) {\n    indices[i] = i;\n  }\n\n  const state = {\n    fieldExpanders: [],     // tree expanders that expand on simple values\n    stackExpanders: [],     // tree expanders that expand stacks\n    activeExpanders: [],    // index into field or stack expanders, hierarchy of tree\n    aggregators: [],        // all available aggregators, might not be used\n    activeAggregators: [],  // index into aggregators, to actually compute\n    sorter: noSortOrder,    // compare function that uses sortOrder to sort row.children\n    root: createTreeNode(null, '<root>', indices, INVALID_ACTIVE_EXPANDER),\n  };\n\n  function evaluateAggregate(row) {\n    const activeAggregators = state.activeAggregators;\n    const aggregates = new Array(activeAggregators.length);\n    for (let j = 0; j < activeAggregators.length; j++) {\n      const aggregator = state.aggregators[activeAggregators[j]];\n      aggregates[j] = aggregator.aggregator(row.indices);\n    }\n    row.aggregates = aggregates;\n    row.state |= NODE_REAGGREGATE_BIT;\n  }\n\n  function evaluateAggregates(row) {\n    if ((row.state & NODE_EXPANDED_BIT) !== 0) {\n      const children = row.children;\n      for (let i = 0; i < children.length; i++) {\n        evaluateAggregate(children[i]);\n      }\n      row.state |= NODE_REORDER_BIT;\n    }\n    row.state ^= NODE_REAGGREGATE_BIT;\n  }\n\n  function evaluateOrder(row) {\n    if ((row.state & NODE_EXPANDED_BIT) !== 0) {\n      const children = row.children;\n      for (let i = 0; i < children.length; i++) {\n        const child = children[i];\n        child.state |= NODE_REORDER_BIT;\n      }\n      children.sort(state.sorter);\n      row.state |= NODE_REPOSITION_BIT;\n    }\n    row.state ^= NODE_REORDER_BIT;\n  }\n\n  function evaluatePosition(row) {\n    if ((row.state & NODE_EXPANDED_BIT) !== 0) {\n      const children = row.children;\n      let childTop = row.top + 1;\n      for (let i = 0; i < children.length; i++) {\n        const child = children[i];\n        if (child.top !== childTop) {\n          child.top = childTop;\n          child.state |= NODE_REPOSITION_BIT;\n        }\n        childTop += child.height;\n      }\n    }\n    row.state ^= NODE_REPOSITION_BIT;\n  }\n\n  function getRowsImpl(row, top, height, result) {\n    if ((row.state & NODE_REAGGREGATE_BIT) !== 0) {\n      evaluateAggregates(row);\n    }\n    if ((row.state & NODE_REORDER_BIT) !== 0) {\n      evaluateOrder(row);\n    }\n    if ((row.state & NODE_REPOSITION_BIT) !== 0) {\n      evaluatePosition(row);\n    }\n\n    if (row.top >= top && row.top < top + height) {\n      if (result[row.top - top] != null) {\n        throw 'getRows put more than one row at position ' + row.top + ' into result';\n      }\n      result[row.top - top] = row;\n    }\n    if ((row.state & NODE_EXPANDED_BIT) !== 0) {\n      const children = row.children;\n      for (let i = 0; i < children.length; i++) {\n        const child = children[i];\n        if (child.top < top + height && top < child.top + child.height) {\n          getRowsImpl(child, top, height, result);\n        }\n      }\n    }\n  }\n\n  function updateHeight(row, heightChange) {\n    while (row !== null) {\n      row.height += heightChange;\n      row.state |= NODE_REPOSITION_BIT;\n      row = row.parent;\n    }\n  }\n\n  function addChildrenWithFieldExpander(row, expander, nextActiveIndex) {\n    const rowIndices = row.indices;\n    const comparer = expander.comparer;\n    rowIndices.sort(comparer);\n    let begin = 0;\n    let end = 1;\n    row.children = [];\n    while (end < rowIndices.length) {\n      if (comparer(rowIndices[begin], rowIndices[end]) !== 0) {\n        row.children.push(createTreeNode(\n          row,\n          expander.name + ': ' + expander.formatter(rowIndices[begin]),\n          rowIndices.subarray(begin, end),\n          nextActiveIndex));\n        begin = end;\n      }\n      end++;\n    }\n    row.children.push(createTreeNode(\n      row,\n      expander.name + ': ' + expander.formatter(rowIndices[begin]),\n      rowIndices.subarray(begin, end),\n      nextActiveIndex));\n  }\n\n  function addChildrenWithStackExpander(row, expander, activeIndex, depth, nextActiveIndex) {\n    const rowIndices = row.indices;\n    const stackGetter = expander.stackGetter;\n    const frameIdGetter = expander.frameIdGetter;\n    const frameGetter = expander.frameGetter;\n    const comparer = expander.comparers[depth];\n    const expandNextFrame = activeIndex | ((depth + 1) << ACTIVE_EXPANDER_FRAME_SHIFT);\n    rowIndices.sort(comparer);\n    let columnName = '';\n    if (depth === 0) {\n      columnName = expander.name + ': ';\n    }\n\n    // put all the too-short stacks under <exclusive>\n    let begin = 0;\n    let beginStack = null;\n    row.children = [];\n    while (begin < rowIndices.length) {\n      beginStack = stackGetter(rowIndices[begin]);\n      if (beginStack.length > depth) {\n        break;\n      }\n      begin++;\n    }\n    if (begin > 0) {\n      row.children.push(createTreeNode(\n        row,\n        columnName + '<exclusive>',\n        rowIndices.subarray(0, begin),\n        nextActiveIndex));\n    }\n    // aggregate the rest under frames\n    if (begin < rowIndices.length) {\n      let end = begin + 1;\n      while (end < rowIndices.length) {\n        const endStack = stackGetter(rowIndices[end]);\n        if (frameIdGetter(beginStack, depth) !== frameIdGetter(endStack, depth)) {\n          row.children.push(createTreeNode(\n            row,\n            columnName + frameGetter(frameIdGetter(beginStack, depth)),\n            rowIndices.subarray(begin, end),\n            expandNextFrame));\n          begin = end;\n          beginStack = endStack;\n        }\n        end++;\n      }\n      row.children.push(createTreeNode(\n        row,\n        columnName + frameGetter(frameIdGetter(beginStack, depth)),\n        rowIndices.subarray(begin, end),\n        expandNextFrame));\n    }\n  }\n\n  function contractRow(row) {\n      if ((row.state & NODE_EXPANDED_BIT) === 0) {\n        throw 'can not contract row, already contracted';\n      }\n      row.state ^= NODE_EXPANDED_BIT;\n      const heightChange = 1 - row.height;\n      updateHeight(row, heightChange);\n  }\n\n  function pruneExpanders(row, oldExpander, newExpander) {\n    row.state |= NODE_REPOSITION_BIT;\n    if (row.expander === oldExpander) {\n      row.state |= NODE_REAGGREGATE_BIT | NODE_REORDER_BIT | NODE_REPOSITION_BIT;\n      if ((row.state & NODE_EXPANDED_BIT) !== 0) {\n        contractRow(row);\n      }\n      row.children = null;\n      row.expander = newExpander;\n    } else {\n      row.state |= NODE_REPOSITION_BIT;\n      const children = row.children;\n      if (children != null) {\n        for (let i = 0; i < children.length; i++) {\n          const child = children[i];\n          pruneExpanders(child, oldExpander, newExpander);\n        }\n      }\n    }\n  }\n\n  return {\n    addFieldExpander: function addFieldExpander(name, formatter, comparer) {\n      if (FIELD_EXPANDER_ID_MIN + state.fieldExpanders.length >= FIELD_EXPANDER_ID_MAX) {\n        throw 'too many field expanders!';\n      }\n      state.fieldExpanders.push({\n        name: name, // name for column\n        formatter: formatter, // row index -> display string\n        comparer: comparer, // compares by two row indices\n      });\n      return FIELD_EXPANDER_ID_MIN + state.fieldExpanders.length - 1;\n    },\n    addCalleeStackExpander: function addCalleeStackExpander(name, maxStackDepth, stackGetter, frameGetter) {\n      if (STACK_EXPANDER_ID_MIN + state.fieldExpanders.length >= STACK_EXPANDER_ID_MAX) {\n        throw 'too many stack expanders!';\n      }\n      state.stackExpanders.push({\n        name: name, // name for column\n        stackGetter: stackGetter, // row index -> stack array\n        comparers: createStackComparers(stackGetter, calleeFrameIdGetter, maxStackDepth),  // depth -> comparer\n        frameIdGetter: calleeFrameIdGetter, // (stack, depth) -> string id\n        frameGetter: frameGetter,\n      });\n      return STACK_EXPANDER_ID_MIN + state.stackExpanders.length - 1;\n    },\n    addCallerStackExpander: function addCallerStackExpander(name, maxStackDepth, stackGetter, frameGetter) {\n      if (STACK_EXPANDER_ID_MIN + state.fieldExpanders.length >= STACK_EXPANDER_ID_MAX) {\n        throw 'too many stack expanders!';\n      }\n      state.stackExpanders.push({\n        name: name,\n        stackGetter: stackGetter,\n        comparers: createStackComparers(stackGetter, callerFrameIdGetter, maxStackDepth),\n        frameIdGetter: callerFrameIdGetter,\n        frameGetter: frameGetter,\n      });\n      return STACK_EXPANDER_ID_MIN + state.stackExpanders.length - 1;\n    },\n    getExpanders: function getExpanders() {\n      const expanders = [];\n      for (let i = 0; i < state.fieldExpanders.length; i++) {\n        expanders.push(FIELD_EXPANDER_ID_MIN + i);\n      }\n      for (let i = 0; i < state.stackExpanders.length; i++) {\n        expanders.push(STACK_EXPANDER_ID_MIN + i);\n      }\n      return expanders;\n    },\n    getExpanderName: function getExpanderName(id) {\n      if (id >= FIELD_EXPANDER_ID_MIN && id <= FIELD_EXPANDER_ID_MAX) {\n        return state.fieldExpanders[id - FIELD_EXPANDER_ID_MIN].name;\n      } else if (id >= STACK_EXPANDER_ID_MIN && id <= STACK_EXPANDER_ID_MAX) {\n        return state.stackExpanders[id - STACK_EXPANDER_ID_MIN].name;\n      }\n      throw 'Unknown expander ID ' + id.toString();\n    },\n    setActiveExpanders: function setActiveExpanders(ids) {\n      for (let i = 0; i < ids.length; i++) {\n        const id = ids[i];\n        if (id >= FIELD_EXPANDER_ID_MIN && id <= FIELD_EXPANDER_ID_MAX) {\n          if (id - FIELD_EXPANDER_ID_MIN >= state.fieldExpanders.length) {\n            throw 'field expander for id ' + id.toString() + ' does not exist!';\n          }\n        } else if (id >= STACK_EXPANDER_ID_MIN && id <= STACK_EXPANDER_ID_MAX) {\n          if (id - STACK_EXPANDER_ID_MIN >= state.stackExpanders.length) {\n            throw 'stack expander for id ' + id.toString() + ' does not exist!';\n          }\n        }\n      }\n      for (let i = 0; i < ids.length; i++) {\n        if (state.activeExpanders.length <= i) {\n          pruneExpanders(state.root, INVALID_ACTIVE_EXPANDER, i);\n          break;\n        } else if (ids[i] !== state.activeExpanders[i]) {\n          pruneExpanders(state.root, i, i);\n          break;\n        }\n      }\n      // TODO: if ids is prefix of activeExpanders, we need to make an expander invalid\n      state.activeExpanders = ids.slice();\n    },\n    getActiveExpanders: function getActiveExpanders() {\n      return state.activeExpanders.slice();\n    },\n    addAggregator: function addAggregator(name, aggregator, formatter, sorter) {\n      if (state.aggregators.length >= AGGREGATOR_ID_MAX) {\n        throw 'too many aggregators!';\n      }\n      state.aggregators.push({\n        name: name,             // name for column\n        aggregator: aggregator, // index array -> aggregate value\n        formatter: formatter,   // aggregate value -> display string\n        sorter: sorter,         // compare two aggregate values\n      });\n      return state.aggregators.length - 1;\n    },\n    getAggregators: function getAggregators() {\n      const aggregators = [];\n      for (let i = 0; i < state.aggregators.length; i++) {\n        aggregators.push(i);\n      }\n      return aggregators;\n    },\n    getAggregatorName: function getAggregatorName(id) {\n      return state.aggregators[id & ACTIVE_AGGREGATOR_MASK].name;\n    },\n    setActiveAggregators: function setActiveAggregators(ids) {\n      for (let i = 0; i < ids.length; i++) {\n        const id = ids[i] & ACTIVE_AGGREGATOR_MASK;\n        if (id < 0 || id > state.aggregators.length) {\n          throw 'aggregator id ' + id.toString() + ' not valid';\n        }\n      }\n      state.activeAggregators = ids.slice();\n      // NB: evaluate root here because dirty bit is for children\n      // so someone has to start with root, and it might as well be right away\n      evaluateAggregate(state.root);\n      let sorter = noSortOrder;\n      for (let i = ids.length - 1; i >= 0; i--) {\n        const ascending = (ids[i] & ACTIVE_AGGREGATOR_ASC_BIT) !== 0;\n        const id = ids[i] & ACTIVE_AGGREGATOR_MASK;\n        const comparer = state.aggregators[id].sorter;\n        const captureSorter = sorter;\n        const captureIndex = i;\n        sorter = function (a, b) {\n          const c = comparer(a.aggregates[captureIndex], b.aggregates[captureIndex]);\n          if (c === 0) {\n            return captureSorter(a, b);\n          }\n          return ascending ? -c : c;\n        };\n      }\n      state.sorter = sorter;\n      state.root.state |= NODE_REORDER_BIT;\n    },\n    getActiveAggregators: function getActiveAggregators() {\n      return state.activeAggregators.slice();\n    },\n    getRows: function getRows(top, height) {\n      const result = new Array(height);\n      for (let i = 0; i < height; i++) {\n        result[i] = null;\n      }\n      getRowsImpl(state.root, top, height, result);\n      return result;\n    },\n    getRowLabel: function getRowLabel(row) {\n      return row.label;\n    },\n    getRowIndent: function getRowIndent(row) {\n      return row.state >>> NODE_INDENT_SHIFT;\n    },\n    getRowAggregate: function getRowAggregate(row, index) {\n      const aggregator = state.aggregators[state.activeAggregators[index]];\n      return aggregator.formatter(row.aggregates[index]);\n    },\n    getHeight: function getHeight() {\n      return state.root.height;\n    },\n    canExpand: function canExpand(row) {\n      return (row.state & NODE_EXPANDED_BIT) === 0 && (row.expander !== INVALID_ACTIVE_EXPANDER);\n    },\n    canContract: function canContract(row) {\n      return (row.state & NODE_EXPANDED_BIT) !== 0;\n    },\n    expand: function expand(row) {\n      if ((row.state & NODE_EXPANDED_BIT) !== 0) {\n        throw 'can not expand row, already expanded';\n      }\n      if (row.height !== 1) {\n        throw 'unexpanded row has height ' + row.height.toString() + ' != 1';\n      }\n      if (row.children === null) {  // first expand, generate children\n        const activeIndex = row.expander & ACTIVE_EXPANDER_MASK;\n        let nextActiveIndex = activeIndex + 1;  // NB: if next is stack, frame is 0\n        if (nextActiveIndex >= state.activeExpanders.length) {\n          nextActiveIndex = INVALID_ACTIVE_EXPANDER;\n        }\n        if (activeIndex >= state.activeExpanders.length) {\n          throw 'invalid active expander index ' + activeIndex.toString();\n        }\n        const exId = state.activeExpanders[activeIndex];\n        if (exId >= FIELD_EXPANDER_ID_MIN &&\n            exId < FIELD_EXPANDER_ID_MIN + state.fieldExpanders.length) {\n          const expander = state.fieldExpanders[exId - FIELD_EXPANDER_ID_MIN];\n          addChildrenWithFieldExpander(row, expander, nextActiveIndex);\n        } else if (exId >= STACK_EXPANDER_ID_MIN &&\n            exId < STACK_EXPANDER_ID_MIN + state.stackExpanders.length) {\n          const depth = row.expander >>> ACTIVE_EXPANDER_FRAME_SHIFT;\n          const expander = state.stackExpanders[exId - STACK_EXPANDER_ID_MIN];\n          addChildrenWithStackExpander(row, expander, activeIndex, depth, nextActiveIndex);\n        } else {\n          throw 'state.activeIndex ' + activeIndex.toString()\n            + ' has invalid expander' + exId.toString();\n        }\n      }\n      row.state |= NODE_EXPANDED_BIT\n        | NODE_REAGGREGATE_BIT | NODE_REORDER_BIT | NODE_REPOSITION_BIT;\n      let heightChange = 0;\n      for (let i = 0; i < row.children.length; i++) {\n        heightChange += row.children[i].height;\n      }\n      updateHeight(row, heightChange);\n      // if children only contains one node, then expand it as well\n      if (row.children.length === 1 && this.canExpand(row.children[0])) {\n        this.expand(row.children[0]);\n      }\n    },\n    contract: function contract(row) {\n      contractRow(row);\n    },\n  };\n}\n"]}
// @generated