700, "height"=>460, "image" => array( "width"=>200, "height"=>200, "top_offset"=>20, ), "image_text" => array( "width"=>400, "height"=>30, "top_offset"=>20, ), "graph" => array( "width"=>600, "height"=>160, "top_offset"=>30, ), "title" => array( "top_offset"=>50 ), "key" => array( "top_offset"=>50 ), "axes" => array( "x" => array( "height" => 20, ), "y" => array( "width" => 30, ), ), "grid" => array( "x" => array( "major" => array( "max" => 12, "min" => 4, ), "minor" => array( "max" => 48, "min" => 12, ), ), "y" => array( "major" => array( "max" => 8, "min" => 1, ), "minor" => array( "max" => 0, "min" => 0, ), ), ), ); $monitors = array(); $monitors_sql = "select * from Monitors order by Sequence asc"; //srand( 97981 ); foreach( dbFetchAll( $monitors_sql ) as $row ) { //if ( empty($row['WebColour']) ) //{ //$row['WebColour'] = sprintf( "#%02x%02x%02x", rand( 0, 255 ), rand( 0, 255), rand( 0, 255 ) ); //} $monitors[$row['Id']] = $row; } $range_sql = "select min(E.StartTime) as MinTime, max(E.EndTime) as MaxTime from Events as E inner join Monitors as M on (E.MonitorId = M.Id) where not isnull(E.StartTime) and not isnull(E.EndTime)"; $events_sql = "select E.Id,E.Name,E.StartTime,E.EndTime,E.Length,E.Frames,E.MaxScore,E.Cause,E.Notes,E.Archived,E.MonitorId from Events as E inner join Monitors as M on (E.MonitorId = M.Id) where not isnull(StartTime)"; if ( !empty($user['MonitorIds']) ) { $mon_filter_sql = " and M.Id in (".join( ",", preg_split( '/["\'\s]*,["\'\s]*/', $user['MonitorIds'] ) ).")"; $range_sql .= $mon_filter_sql; $events_sql .= $mon_filter_sql; } if ( isset($filter) ) $tree = parseFilterToTree( $filter ); else $tree = false; if ( isset($range) ) { $half_range = (int)($range/2); if ( isset($mid_time) ) { $mid_time_t = strtotime($mid_time); $min_time_t = $mid_time_t-$half_range; $max_time_t = $mid_time_t+$half_range; if ( !($range%1) ) { $max_time_t--; } $min_time = strftime( STRF_FMT_DATETIME_DB, $min_time_t ); $max_time = strftime( STRF_FMT_DATETIME_DB, $max_time_t ); } elseif ( isset($min_time) ) { $min_time_t = strtotime($min_time); $max_time_t = $min_time_t + $range; $mid_time_t = $min_time_t + $half_range; $mid_time = strftime( STRF_FMT_DATETIME_DB, $mid_time_t ); $max_time = strftime( STRF_FMT_DATETIME_DB, $max_time_t ); } elseif ( isset($max_time) ) { $max_time_t = strtotime($max_time); $min_time_t = $max_time_t - $range; $mid_time_t = $min_time_t + $half_range; $min_time = strftime( STRF_FMT_DATETIME_DB, $min_time_t ); $mid_time = strftime( STRF_FMT_DATETIME_DB, $mid_time_t ); } } elseif ( isset($min_time) && isset($max_time) ) { $min_time_t = strtotime($min_time); $max_time_t = strtotime($max_time); $range = ($max_time_t - $min_time_t) + 1; $half_range = (int)($range/2); $mid_time_t = $min_time_t + $half_range; $mid_time = strftime( STRF_FMT_DATETIME_DB, $mid_time_t ); } if ( isset($min_time) && isset($max_time) ) { $temp_min_time = $temp_max_time = $temp_expandable = false; extractDatetimeRange( $tree, $temp_min_time, $temp_max_time, $temp_expandable ); $filter_sql = parseTreeToSQL( $tree ); if ( $filter_sql ) { $filter_sql = " and $filter_sql"; $events_sql .= $filter_sql; } } else { //$filter_query = parseTreeToQuery( $tree ); //echo $filter_query; //echo '
'; $filter_sql = parseTreeToSQL( $tree ); $temp_min_time = $temp_max_time = $temp_expandable = false; extractDatetimeRange( $tree, $temp_min_time, $temp_max_time, $temp_expandable ); //echo $filter_sql; //echo '
'; if ( $filter_sql ) { $filter_sql = " and $filter_sql"; $range_sql .= $filter_sql; $events_sql .= $filter_sql; } if ( !isset($min_time) || !isset($max_time) ) { // Dynamically determine range $row = dbFetchOne( $range_sql ); if ( !isset($min_time) ) $min_time = $row['MinTime']; if ( !isset($max_time) ) $max_time = $row['MaxTime']; } if ( empty($min_time) ) $min_time = $temp_min_time; if ( empty($max_time) ) $max_time = $temp_max_time; if ( empty($max_time) ) $max_time = "now"; $min_time_t = strtotime($min_time); $max_time_t = strtotime($max_time); $range = ($max_time_t - $min_time_t) + 1; $half_range = (int)($range/2); $mid_time_t = $min_time_t + $half_range; $mid_time = strftime( STRF_FMT_DATETIME_DB, $mid_time_t ); } //echo "MnT: $temp_min_time, MxT: $temp_max_time, ExP: $temp_expandable
"; appendDatetimeRange( $tree, $min_time, $max_time ); $filter_query = parseTreeToQuery( $tree ); if ( $filter_query ) { $filter_query = '&'.$filter_query; } //echo $filter_query; //echo '
'; $scales = array( array( "name"=>"year", "factor"=>60*60*24*365, "align"=>1, "zoomout"=>2, "label"=>STRF_TL_AXIS_LABEL_YEAR ), array( "name"=>"month", "factor"=>60*60*24*30, "align"=>1, "zoomout"=>12, "label"=>STRF_TL_AXIS_LABEL_MONTH ), array( "name"=>"week", "factor"=>60*60*24*7, "align"=>1, "zoomout"=>4.25, "label"=>STRF_TL_AXIS_LABEL_WEEK, "label_check"=>"%W" ), array( "name"=>"day", "factor"=>60*60*24, "align"=>1, "zoomout"=>7, "label"=>STRF_TL_AXIS_LABEL_DAY ), array( "name"=>"hour", "factor"=>60*60, "align"=>1, "zoomout"=>24, "label"=>STRF_TL_AXIS_LABEL_HOUR, "label_check"=>"%H" ), array( "name"=>"minute10", "factor"=>60, "align"=>10, "zoomout"=>6, "label"=>STRF_TL_AXIS_LABEL_10MINUTE, "label_check"=>"%M" ), array( "name"=>"minute", "factor"=>60, "align"=>1, "zoomout"=>10, "label"=>STRF_TL_AXIS_LABEL_MINUTE, "label_check"=>"%M" ), array( "name"=>"second10", "factor"=>1, "align"=>10, "zoomout"=>6, "label"=>STRF_TL_AXIS_LABEL_10SECOND ), array( "name"=>"second", "factor"=>1, "align"=>1, "zoomout"=>10, "label"=>STRF_TL_AXIS_LABEL_SECOND ), ); $maj_x_scale = getDateScale( $scales, $range, $chart['grid']['x']['major']['min'], $chart['grid']['x']['major']['max'] ); //print_r( $maj_x_scale ); // Adjust the range etc for scale $min_time_t -= $min_time_t%($maj_x_scale['factor']*$maj_x_scale['align']); $min_time = strftime( STRF_FMT_DATETIME_DB, $min_time_t ); $max_time_t += (($maj_x_scale['factor']*$maj_x_scale['align'])-$max_time_t%($maj_x_scale['factor']*$maj_x_scale['align']))-1; if ( $max_time_t > time() ) $max_time_t = time(); $max_time = strftime( STRF_FMT_DATETIME_DB, $max_time_t ); $range = ($max_time_t - $min_time_t) + 1; $half_range = (int)($range/2); $mid_time_t = $min_time_t + $half_range; $mid_time = strftime( STRF_FMT_DATETIME_DB, $mid_time_t ); //echo "R:$range
"; //echo "MnT:$min_time
"; //echo "MnTt:$min_time_t
"; //echo "MdT:$mid_time
"; //echo "MdTt:$mid_time_t
"; //echo "MxT:$max_time
"; //echo "MxTt:$max_time_t
"; if ( isset($min_time) && isset($max_time) ) { $events_sql .= " and E.EndTime >= '$min_time' and E.StartTime <= '$max_time'"; } $events_sql .= " order by Id asc"; //echo "ESQL: $events_sql
"; $chart['data'] = array( "x" => array( "lo" => strtotime( $min_time ), "hi" => strtotime( $max_time ), ), "y" => array( "lo" => 0, "hi" => 0, ) ); $chart['data']['x']['range'] = ($chart['data']['x']['hi'] - $chart['data']['x']['lo']) + 1; $chart['data']['x']['density'] = $chart['data']['x']['range']/$chart['graph']['width']; $mon_event_slots = array(); $mon_frame_slots = array(); $monitor_ids = array(); foreach( dbFetchAll( $events_sql ) as $event ) { if ( !isset($monitor_ids[$event['MonitorId']]) ) $monitor_ids[$event['MonitorId']] = true; if ( !isset($mon_event_slots[$event['MonitorId']]) ) $mon_event_slots[$event['MonitorId']] = array(); if ( !isset($mon_frame_slots[$event['MonitorId']]) ) $mon_frame_slots[$event['MonitorId']] = array(); $curr_event_slots = &$mon_event_slots[$event['MonitorId']]; $curr_frame_slots = &$mon_frame_slots[$event['MonitorId']]; $start_time_t = strtotime($event['StartTime']); $start_index = $raw_start_index = (int)(($start_time_t - $chart['data']['x']['lo']) / $chart['data']['x']['density']); if ( $start_index < 0 ) $start_index = 0; if ( isset($event['EndTime']) ) { $end_time_t = strtotime($event['EndTime']); } else { $end_time_t = time(); } $end_index = $raw_end_index = (int)(($end_time_t - $chart['data']['x']['lo']) / $chart['data']['x']['density']); if ( $end_index >= $chart['graph']['width'] ) $end_index = $chart['graph']['width'] - 1; for ( $i = $start_index; $i <= $end_index; $i++ ) { if ( !isset($curr_event_slots[$i]) ) { if ( $raw_start_index == $raw_end_index ) { $offset = 1; } else { $offset = 1+ ($event['Frames']?((int)($event['Frames']*(($i-$raw_start_index)/($raw_end_index-$raw_start_index)))):0); } $curr_event_slots[$i] = array( "count"=>0, "width"=>1, "offset"=>$offset, "event"=>$event ); } else { $curr_event_slots[$i]['count']++; } } if ( $event['MaxScore'] > 0 ) { if ( $start_index == $end_index ) { $i = $start_index; if ( !isset($curr_frame_slots[$i]) ) { $curr_frame_slots[$i] = array( "count"=>1, "value"=>$event['MaxScore'], "event"=>$event ); } else { $curr_frame_slots[$i]['count']++; if ( $event['MaxScore'] > $curr_frame_slots[$i]['value'] ) { $curr_frame_slots[$i]['value'] = $event['MaxScore']; $curr_frame_slots[$i]['event'] = $event; } } if ( $event['MaxScore'] > $chart['data']['y']['hi'] ) { $chart['data']['y']['hi'] = $event['MaxScore']; } } else { $frames_sql = "select F.FrameId,F.Delta,unix_timestamp(F.TimeStamp) as TimeT,F.Score from Frames as F where F.EventId = '".$event['Id']."' and F.Score > 0"; foreach( dbFetchAll( $frames_sql ) as $frame ) { $frame_time_t = $frame['TimeT']; $frame_time_t = $start_time_t + $frame['Delta']; $frame_index = (int)(($frame_time_t - $chart['data']['x']['lo']) / $chart['data']['x']['density']); if ( $frame_index < 0 ) continue; if ( $frame_index >= $chart['graph']['width'] ) continue; if ( !isset($curr_frame_slots[$frame_index]) ) { $curr_frame_slots[$frame_index] = array( "count"=>1, "value"=>$frame['Score'], "event"=>$event, "frame"=>$frame ); } else { $curr_frame_slots[$frame_index]['count']++; if ( $frame['Score'] > $curr_frame_slots[$frame_index]['value'] ) { $curr_frame_slots[$frame_index]['value'] = $frame['Score']; $curr_frame_slots[$frame_index]['event'] = $event; $curr_frame_slots[$frame_index]['frame'] = $frame; } } if ( $frame['Score'] > $chart['data']['y']['hi'] ) { $chart['data']['y']['hi'] = $frame['Score']; } } } } } ksort($monitor_ids,SORT_NUMERIC); ksort($mon_event_slots,SORT_NUMERIC); ksort($mon_frame_slots,SORT_NUMERIC); // Add on missing frames $xcount = 0; foreach( array_keys($mon_frame_slots) as $monitor_id ) { unset( $curr_frame_slots ); $curr_frame_slots = &$mon_frame_slots[$monitor_id]; for ( $i = 0; $i < $chart['graph']['width']; $i++ ) { if ( isset($curr_frame_slots[$i]) ) { if ( !isset($curr_frame_slots[$i]['frame']) ) { $xcount++; $frames_sql = "select F.FrameId,F.Score from Frames as F where F.EventId = '".$curr_frame_slots[$i]['event']['Id']."' and F.Score > 0 order by F.FrameId limit 0,1"; $curr_frame_slots[$i]['frame'] = dbFetchOne( $frames_sql ); } } } } $chart['data']['y']['range'] = ($chart['data']['y']['hi'] - $chart['data']['y']['lo']) + 1; $chart['data']['y']['density'] = $chart['data']['y']['range']/$chart['graph']['height']; $maj_y_scale = getYScale( $chart['data']['y']['range'], $chart['grid']['y']['major']['min'], $chart['grid']['y']['major']['max'] ); //print_r( $maj_y_scale ); $max_width = 0; $max_height = 0; foreach ( array_keys($monitor_ids) as $monitor_id ) { if ( $max_width < $monitors[$monitor_id]['Width'] ) $max_width = $monitors[$monitor_id]['Width']; if ( $max_height < $monitors[$monitor_id]['Height'] ) $max_height = $monitors[$monitor_id]['Height']; } // Optimise boxes foreach( array_keys($mon_event_slots) as $monitor_id ) { unset( $curr_event_slots ); $curr_event_slots = &$mon_event_slots[$monitor_id]; for ( $i = 0; $i < $chart['graph']['width']; $i++ ) { if ( isset($curr_event_slots[$i]) ) { //if ( isset($curr_slot) && (($curr_slot['width'] < $min_event_width) || (($curr_slot['event']['Id'] == $curr_event_slots[$i]['event']['Id']) && ($curr_slot['frame']['FrameId'] == $curr_event_slots[$i]['frame']['FrameId'])) ) ) //if ( isset($curr_slot) && ($curr_slot['event']['Id'] == $curr_event_slots[$i]['event']['Id']) ) if ( isset($curr_slot) ) { if ( $curr_slot['event']['Id'] == $curr_event_slots[$i]['event']['Id'] ) { if ( $curr_slot['width'] < $max_event_width ) { // Merge slots for the same long event $curr_slot['width']++; unset( $curr_event_slots[$i] ); continue; } elseif ( $curr_slot['offset'] < $curr_event_slots[$i]['offset'] ) { // Split very long events $curr_event_slots[$i]['frame'] = array( 'FrameId'=>$curr_event_slots[$i]['offset'] ); } } elseif ( $curr_slot['width'] < $min_event_width ) { // Merge multiple small events $curr_slot['width']++; unset( $curr_event_slots[$i] ); continue; } } $curr_slot = &$curr_event_slots[$i]; //if ( isset($curr_slot) && ($curr_slot['width'] < $min_event_width || ($curr_slot['event']['Id'] == $curr_event_slots[$i]['event']['Id']) ) ) //{ //$curr_slot['width']++; //unset( $curr_event_slots[$i] ); //} //else //{ //$curr_slot = &$curr_event_slots[$i]; //} } else { unset( $curr_slot ); } } if ( isset( $curr_slot ) ) unset( $curr_slot ); } // Stack events $frame_slots = array(); $frame_monitor_ids = array_keys($mon_frame_slots); for ( $i = 0; $i < $chart['graph']['width']; $i++ ) { foreach ( $frame_monitor_ids as $frame_monitor_id ) { unset( $curr_frame_slots ); $curr_frame_slots = &$mon_frame_slots[$frame_monitor_id]; if ( isset($curr_frame_slots[$i]) ) { if ( !isset($frame_slots[$i]) ) { $frame_slots[$i] = array(); $frame_slots[$i][] = &$curr_frame_slots[$i]; } else { $slot_count = count($frame_slots[$i]); for ( $j = 0; $j < $slot_count; $j++ ) { if ( $curr_frame_slots[$i]['value'] > $frame_slots[$i][$j]['value'] ) { for ( $k = $slot_count; $k > $j; $k-- ) { $frame_slots[$i][$k] = $frame_slots[$i][$k-1]; } $frame_slots[$i][$j] = &$curr_frame_slots[$i]; break 2; } } $frame_slots[$i][] = &$curr_frame_slots[$i]; } } } } //print_r( $mon_event_slots ); //print_r( $mon_frame_slots ); //print_r( $chart ); preg_match( '/^(\d+)-(\d+)-(\d+) (\d+):(\d+)/', $min_time, $start_matches ); preg_match( '/^(\d+)-(\d+)-(\d+) (\d+):(\d+)/', $max_time, $end_matches ); if ( $start_matches[1] != $end_matches[1] ) { // Different years $title = strftime( STRF_TL_AXIS_RANGE_YEAR1, $chart['data']['x']['lo'] )." - ".strftime( STRF_TL_AXIS_RANGE_YEAR2, $chart['data']['x']['hi'] ); } elseif ( $start_matches[2] != $end_matches[2] ) { // Different months $title = strftime( STRF_TL_AXIS_RANGE_MONTH1, $chart['data']['x']['lo'] )." - ".strftime( STRF_TL_AXIS_RANGE_MONTH2, $chart['data']['x']['hi'] ); } elseif ( $start_matches[3] != $end_matches[3] ) { // Different dates $title = strftime( STRF_TL_AXIS_RANGE_DAY1, $chart['data']['x']['lo'] )." - ".strftime( STRF_TL_AXIS_RANGE_DAY2, $chart['data']['x']['hi'] ); } else { // Different times $title = strftime( STRF_TL_AXIS_RANGE_TIME1, $chart['data']['x']['lo'] )." - ".strftime( STRF_TL_AXIS_RANGE_TIME2, $chart['data']['x']['hi'] ); } function getDateScale( $scales, $range, $min_lines, $max_lines ) { foreach ( $scales as $scale ) { $align = isset($scale['align'])?$scale['align']:1; $scale_range = (int)($range/($scale['factor']*$align)); //echo "S:".$scale['name'].", A:$align, SR:$scale_range
"; if ( $scale_range >= $min_lines ) { $scale['range'] = $scale_range; break; } } if ( !isset($scale['range']) ) { $scale['range'] = (int)($range/($scale['factor']*$align)); } $scale['divisor'] = 1; while ( ($scale['range']/$scale['divisor']) > $max_lines ) { $scale['divisor']++; } $scale['lines'] = (int)($scale['range']/$scale['divisor']); return( $scale ); } function getYScale( $range, $min_lines, $max_lines ) { $scale['range'] = $range; $scale['divisor'] = 1; while ( $scale['range']/$scale['divisor'] > $max_lines ) { $scale['divisor']++; } $scale['lines'] = (int)(($scale['range']-1)/$scale['divisor'])+1; return( $scale ); } function drawXGrid( $chart, $scale, $label_class, $tick_class, $grid_class, $zoom_class=0 ) { global $zmSlangZoomIn; ob_start(); $label_count = 0; $last_tick = 0; unset( $last_label ); $label_check = isset($scale['label_check'])?$scale['label_check']:$scale['label']; for ( $i = 0; $i < $chart['graph']['width']; $i++ ) { $x = $i - 1; $time_offset = (int)($chart['data']['x']['lo'] + ($i * $chart['data']['x']['density'])); if ( $scale['align'] > 1 ) { $label = (int)(strftime( $label_check, $time_offset )/$scale['align']); } else { $label = strftime( $label_check, $time_offset ); } if ( !isset($last_label) || ($last_label != $label) ) { $label_count++; } if ( $label_count >= $scale['divisor'] ) { $label_count = 0; if ( isset($last_label) ) { if ( $label_class ) { ?>
"; $annotation .= $monitor['Name']. "
".$slot['event']['Name'].($slot_frame?("(".$slot_frame.")"):""). "
".strftime( STRF_FMT_DATETIME_SHORT, strtotime($slot['event']['StartTime']) ). " - ".$slot['event']['Length']."s". "
".htmlentities($slot['event']['Cause']). (!empty($slot['event']['Notes'])?("
".preg_replace( "/\n/", "
", htmlentities($slot['event']['Notes']))):""). (!empty($slot['event']['Archived'])?("
".$zmSlangArchived):""); if ( $slot['event']['Archived'] ) $annotation .= ""; if ( $slot_frame ) { $slot_frame -= $monitor['PreEventCount']; if ( $slot_frame < 1 ) $slot_frame = 1; } return( "\"loadEventImage( '".$image_path."', '".$annotation."', ".$slot['event']['Id'].", ".($slot_frame?$slot_frame:1).", ".(reScale( $monitor['Width'], $monitor['DefaultScale'], ZM_WEB_DEFAULT_SCALE )+$jws['event']['w']).", ".(reScale( $monitor['Height'], $monitor['DefaultScale'], ZM_WEB_DEFAULT_SCALE )+$jws['event']['h'])." );\"" ); } function getSlotViewEventBehaviour( $slot ) { global $monitors, $jws, $PHP_SELF; $monitor = &$monitors[$slot['event']['MonitorId']]; $slot_frame = isset($slot['frame'])?$slot['frame']['FrameId']:0; if ( $slot_frame ) { $slot_frame -= $monitor['PreEventCount']; if ( $slot_frame < 1 ) $slot_frame = 1; } return( "\"eventWindow( ".$slot['event']['Id'].", ".($slot_frame?$slot_frame:1).", ".(reScale( $monitor['Width'], $monitor['DefaultScale'], ZM_WEB_DEFAULT_SCALE )+$jws['event']['w']).", ".(reScale( $monitor['Height'], $monitor['DefaultScale'], ZM_WEB_DEFAULT_SCALE )+$jws['event']['h'])." );\"" ); } ?> <?= ZM_WEB_TITLE_PREFIX ?> - <?= $zmSlangTimeline ?>
Event Navigator
<   -   >
No Event
     
$slots ) { foreach ( $slots as $slot ) { $slot_height = (int)($slot['value']/$chart['data']['y']['density']); if ( $slot_height <= 0 ) continue; if ( $mouseover ) { $behaviours = array( "onClick=".getSlotViewEventBehaviour( $slot ), "onMouseOver=".getSlotLoadImageBehaviour( $slot ), ); } else { $behaviours = array( "onClick=".getSlotLoadImageBehaviour( $slot ), ); } ?>
>
$slot ) { $slot_height = (int)($slot['value']/$chart['data']['y']['density']); if ( $slot_height <= 0 ) continue; if ( $mouseover ) { $behaviours = array( "onClick=".getSlotViewEventBehaviour( $slot ), "onMouseOver=".getSlotLoadImageBehaviour( $slot ), ); } else { $behaviours = array( "onClick=".getSlotLoadImageBehaviour( $slot ), ); } ?>
>
>
0 ) { $postfix_expr = array(); $postfix_stack = array(); $priorities = array( '<' => 1, '<=' => 1, '>' => 1, '>=' => 1, '=' => 2, '!=' => 2, '=~' => 2, '!~' => 2, '=[]' => 2, '![]' => 2, 'and' => 3, 'or' => 4, ); for ( $i = 0; $i <= count($filter['terms']); $i++ ) { if ( !empty($filter['terms'][$i]['cnj']) ) { while( true ) { if ( !count($postfix_stack) ) { $postfix_stack[] = array( 'type'=>"cnj", 'value'=>$filter['terms'][$i]['cnj'], 'sql_value'=>$filter['terms'][$i]['cnj']); break; } elseif ( $postfix_stack[count($postfix_stack)-1]['type'] == 'obr' ) { $postfix_stack[] = array( 'type'=>"cnj", 'value'=>$filter['terms'][$i]['cnj'], 'sql_value'=>$filter['terms'][$i]['cnj']); break; } elseif ( $priorities[$filter['terms'][$i]['cnj']] < $priorities[$postfix_stack[count($postfix_stack)-1]['value']] ) { $postfix_stack[] = array( 'type'=>"cnj", 'value'=>$filter['terms'][$i]['cnj'], 'sql_value'=>$filter['terms'][$i]['cnj']); break; } else { $postfix_expr[] = array_pop( $postfix_stack ); } } } if ( !empty($filter['terms'][$i]['obr']) ) { for ( $j = 0; $j < $filter['terms'][$i]['obr']; $j++ ) { $postfix_stack[] = array( 'type'=>"obr", 'value'=>$filter['terms'][$i]['obr']); } } if ( !empty($filter['terms'][$i]['attr']) ) { $dt_attr = false; switch ( $filter['terms'][$i]['attr']) { case 'MonitorName': $sql_value = 'M.'.preg_replace( '/^Monitor/', '', $filter['terms'][$i]['attr']); break; case 'Name': $sql_value = "E.Name"; break; case 'Cause': $sql_value = "E.Cause"; break; case 'DateTime': $sql_value = "E.StartTime"; $dt_attr = true; break; case 'Date': $sql_value = "to_days( E.StartTime )"; $dt_attr = true; break; case 'Time': $sql_value = "extract( hour_second from E.StartTime )"; break; case 'Weekday': $sql_value = "weekday( E.StartTime )"; break; case 'Id': case 'Name': case 'MonitorId': case 'Length': case 'Frames': case 'AlarmFrames': case 'TotScore': case 'AvgScore': case 'MaxScore': case 'Archived': $sql_value = "E.".$filter['terms'][$i]['attr']; break; case 'DiskPercent': $sql_value = getDiskPercent(); break; case 'DiskBlocks': $sql_value = getDiskBlocks(); break; default : $sql_value = $filter['terms'][$i]['attr']; break; } if ( $dt_attr ) { $postfix_expr[] = array( 'type'=>"attr", 'value'=>$filter['terms'][$i]['attr'], 'sql_value'=>$sql_value, 'dt_attr'=>true ); } else { $postfix_expr[] = array( 'type'=>"attr", 'value'=>$filter['terms'][$i]['attr'], 'sql_value'=>$sql_value ); } } if ( isset($filter['terms'][$i]['op']) ) { if ( empty($filter['terms'][$i]['op']) ) { $filter['terms'][$i]['op' ]= '='; } switch ( $filter['terms'][$i]['op' ]) { case '=' : case '!=' : case '>=' : case '>' : case '<' : case '<=' : $sql_value = $filter['terms'][$i]['op']; break; case '=~' : $sql_value = "regexp"; break; case '!~' : $sql_value = "not regexp"; break; case '=[]' : $sql_value = 'in ('; break; case '![]' : $sql_value = 'not in ('; break; } while( true ) { if ( !count($postfix_stack) ) { $postfix_stack[] = array( 'type'=>"op", 'value'=>$filter['terms'][$i]['op'], 'sql_value'=>$sql_value ); break; } elseif ( $postfix_stack[count($postfix_stack)-1]['type'] == 'obr' ) { $postfix_stack[] = array( 'type'=>"op", 'value'=>$filter['terms'][$i]['op'], 'sql_value'=>$sql_value ); break; } elseif ( $priorities[$filter['terms'][$i]['op']] < $priorities[$postfix_stack[count($postfix_stack)-1]['value']] ) { $postfix_stack[] = array( 'type'=>"op", 'value'=>$filter['terms'][$i]['op'], 'sql_value'=>$sql_value ); break; } else { $postfix_expr[] = array_pop( $postfix_stack ); } } } if ( isset($filter['terms'][$i]['val']) ) { $value_list = array(); foreach ( preg_split( '/["\'\s]*?,["\'\s]*?/', preg_replace( '/^["\']+?(.+)["\']+?$/', '$1', $filter['terms'][$i]['val' ]) ) as $value ) { switch ( $filter['terms'][$i]['attr']) { case 'MonitorName': case 'Name': case 'Cause': $value = "'$value'"; break; case 'DateTime': $value = "'".strftime( STRF_FMT_DATETIME_DB, strtotime( $value ) )."'"; break; case 'Date': $value = "to_days( '".strftime( STRF_FMT_DATETIME_DB, strtotime( $value ) )."' )"; break; case 'Time': $value = "extract( hour_second from '".strftime( STRF_FMT_DATETIME_DB, strtotime( $value ) )."' )"; break; case 'Weekday': $value = "weekday( '".strftime( STRF_FMT_DATETIME_DB, strtotime( $value ) )."' )"; break; } $value_list[] = $value; } $postfix_expr[] = array( 'type'=>"val", 'value'=>$filter['terms'][$i]['val'], 'sql_value'=>join( ',', $value_list ) ); } if ( !empty($filter['terms'][$i]['cbr']) ) { for ( $j = 0; $j < $filter['terms'][$i]['cbr']; $j++ ) { while ( count($postfix_stack) ) { $element = array_pop( $postfix_stack ); if ( $element['type'] == "obr" ) { $postfix_expr[count($postfix_expr)-1]['bracket'] = true; break; } $postfix_expr[] = $element; } } } } while ( count($postfix_stack) ) { $postfix_expr[] = array_pop( $postfix_stack ); } $expr_stack = array(); //foreach ( $postfix_expr as $element ) //{ //echo $element['value']." "; //} //echo "
"; foreach ( $postfix_expr as $element ) { if ( $element['type'] == 'attr' || $element['type'] == 'val' ) { $node = array( 'data'=>$element, 'count'=>0 ); $expr_stack[] = $node; } elseif ( $element['type'] == 'op' || $element['type'] == 'cnj' ) { $right = array_pop( $expr_stack ); $left = array_pop( $expr_stack ); $node = array( 'data'=>$element, 'count'=>2+$left['count']+$right['count'], 'right'=>$right, 'left'=>$left ); $expr_stack[] = $node; } else { die( "Unexpected element type '".$element['type']."', value '".$element['value']."'" ); } } if ( count($expr_stack) != 1 ) { die( "Expression stack has ".count($expr_stack)." elements" ); } $expr_tree = array_pop( $expr_stack ); return( $expr_tree ); } return( false ); } function _parseTreeToInfix( $node ) { $expression = ''; if ( isset($node) ) { if ( isset($node['left']) ) { if ( !empty($node['data']['bracket']) ) $expression .= '( '; $expression .= _parseTreeToInfix( $node['left'] ); } $expression .= $node['data']['value']." "; if ( isset($node['right']) ) { $expression .= _parseTreeToInfix( $node['right'] ); if ( !empty($node['data']['bracket']) ) $expression .= ') '; } } return( $expression ); } function parseTreeToInfix( $tree ) { return( _parseTreeToInfix( $tree ) ); } function _parseTreeToSQL( $node, $cbr=false ) { $expression = ''; if ( $node ) { if ( isset($node['left']) ) { if ( !empty($node['data']['bracket']) ) $expression .= '( '; $expression .= _parseTreeToSQL( $node['left'] ); } $in_expr = $node['data']['type'] == 'op' && ($node['data']['value'] == '=[]' || $node['data']['value'] == '![]'); $expression .= $node['data']['sql_value']; if ( !$in_expr ) $expression .= ' '; if ( $cbr ) $expression .= ') '; if ( isset($node['right']) ) { $expression .= _parseTreeToSQL( $node['right'], $in_expr ); if ( !empty($node['data']['bracket']) ) $expression .= ') '; } } return( $expression ); } function parseTreeToSQL( $tree ) { return( _parseTreeToSQL( $tree ) ); } function _parseTreeToFilter( $node, &$terms, &$level ) { $elements = array(); if ( $node ) { if ( isset($node['left']) ) { if ( !empty($node['data']['bracket']) ) $terms[$level]['obr'] = 1; _parseTreeToFilter( $node['left'], $terms, $level ); } if ( $node['data']['type'] == 'cnj' ) { $level++; } $terms[$level][$node['data']['type']] = $node['data']['value']; if ( isset($node['right']) ) { _parseTreeToFilter( $node['right'], $terms, $level ); if ( !empty($node['data']['bracket']) ) $terms[$level]['cbr'] = 1; } } } function parseTreeToFilter( $tree ) { $terms = array(); if ( isset($tree) ) { $level = 0; _parseTreeToFilter( $tree, $terms, $level ); } return( array( 'terms' => $terms ) ); } function parseTreeToQuery( $tree ) { $filter = parseTreeToFilter( $tree ); parseFilter( $filter ); return( $filter['query'] ); } function _drawTree( $node, $level ) { if ( isset($node['left']) ) { _drawTree( $node['left'], $level+1 ); } echo str_repeat( ".", $level*2 ).$node['data']['value']."
"; if ( isset($node['right']) ) { _drawTree( $node['right'], $level+1 ); } } function drawTree( $tree ) { _drawTree( $tree, 0 ); } function _extractDatetimeRange( &$node, &$min_time, &$max_time, &$expandable, $sub_or ) { $pruned = $left_pruned = $right_pruned = false; if ( $node ) { if ( isset($node['left']) && isset($node['right']) ) { if ( $node['data']['type'] == 'cnj' && $node['data']['value'] == 'or' ) { $sub_or = true; } elseif ( !empty($node['left']['data']['dt_attr']) ) { if ( $sub_or ) { $expandable = false; } elseif ( $node['data']['type'] == 'op' ) { if ( $node['data']['value'] == '>' || $node['data']['value'] == '>=' ) { if ( !$min_time || $min_time > $node['right']['data']['sql_value'] ) { $min_time = $node['right']['data']['value']; return( true ); } } if ( $node['data']['value'] == '<' || $node['data']['value'] == '<=' ) { if ( !$max_time || $max_time < $node['right']['data']['sql_value'] ) { $max_time = $node['right']['data']['value']; return( true ); } } } else { die( "Unexpected node type '".$node['data']['type']."'" ); } return( false ); } $left_pruned = _extractDatetimeRange( $node['left'], $min_time, $max_time, $expandable, $sub_or ); $right_pruned = _extractDatetimeRange( $node['right'], $min_time, $max_time, $expandable, $sub_or ); if ( $left_pruned && $right_pruned ) { $pruned = true; } elseif ( $left_pruned ) { $node = $node['right']; } elseif ( $right_pruned ) { $node = $node['left']; } } } return( $pruned ); } function extractDatetimeRange( &$tree, &$min_time, &$max_time, &$expandable ) { $min_time = ""; $max_time = ""; $expandable = true; _extractDateTimeRange( $tree, $min_time, $max_time, $expandable, false ); } function appendDatetimeRange( &$tree, $min_time, $max_time=false ) { $attr_node = array( 'data'=>array( 'type'=>'attr', 'value'=>'DateTime', 'sql_value'=>'E.StartTime', 'dt_attr'=>true ), 'count'=>0 ); $val_node = array( 'data'=>array( 'type'=>'val', 'value'=>$min_time, 'sql_value'=>$min_time ), 'count'=>0 ); $op_node = array( 'data'=>array( 'type'=>'op', 'value'=>'>=', 'sql_value'=>'>=' ), 'count'=>2, 'left'=>$attr_node, 'right'=>$val_node ); if ( isset($tree) ) { $cnj_node = array( 'data'=>array( 'type'=>'cnj', 'value'=>'and', 'sql_value'=>'and' ), 'count'=>2+$tree['count']+$op_node['count'], 'left'=>$tree, 'right'=>$op_node ); $tree = $cnj_node; } else { $tree = $op_node; } if ( $max_time ) { $attr_node = array( 'data'=>array( 'type'=>'attr', 'value'=>'DateTime', 'sql_value'=>'E.StartTime', 'dt_attr'=>true ), 'count'=>0 ); $val_node = array( 'data'=>array( 'type'=>'val', 'value'=>$max_time, 'sql_value'=>$max_time ), 'count'=>0 ); $op_node = array( 'data'=>array( 'type'=>'op', 'value'=>'<=', 'sql_value'=>'<=' ), 'count'=>2, 'left'=>$attr_node, 'right'=>$val_node ); $cnj_node = array( 'data'=>array( 'type'=>'cnj', 'value'=>'and', 'sql_value'=>'and' ), 'count'=>2+$tree['count']+$op_node['count'], 'left'=>$tree, 'right'=>$op_node ); $tree = $cnj_node; } } ?>