Logo Search packages:      
Sourcecode: nagios version File versions  Download package

trends.c

/**************************************************************************
 *
 * TRENDS.C -  Nagios State Trends CGI
 *
 * Copyright (c) 1999-2004 Ethan Galstad (nagios@nagios.org)
 * Last Modified: 05-19-2004
 *
 * License:
 * 
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 *************************************************************************/

#include "../common/config.h"
#include "../common/locations.h"
#include "../common/common.h"
#include "../common/objects.h"
#include "../common/comments.h"
#include "../common/statusdata.h"

#include "cgiutils.h"
#include "getcgi.h"
#include "auth.h"
#include "edata.h"

#include <gd.h>               /* Boutell's GD library function */
#include <gdfonts.h>          /* GD library small font definition */


/*#define DEBUG               1*/


extern char main_config_file[MAX_FILENAME_LENGTH];
extern char url_images_path[MAX_FILENAME_LENGTH];
extern char url_stylesheets_path[MAX_FILENAME_LENGTH];
extern char physical_images_path[MAX_FILENAME_LENGTH];

extern host *host_list;
extern service *service_list;

extern int     log_rotation_method;


/* archived state types */
#define AS_CURRENT_STATE        -1   /* special case for initial assumed state */
#define AS_NO_DATA            0
#define AS_PROGRAM_END        1
#define AS_PROGRAM_START      2
#define AS_HOST_UP            3
#define AS_HOST_DOWN          4
#define AS_HOST_UNREACHABLE   5
#define AS_SVC_OK       6
#define AS_SVC_UNKNOWN        7
#define AS_SVC_WARNING        8
#define AS_SVC_CRITICAL       9

/* display types */
#define DISPLAY_HOST_TRENDS   0
#define DISPLAY_SERVICE_TRENDS      1
#define DISPLAY_NO_TRENDS     2

/* input types */
#define GET_INPUT_NONE          0
#define GET_INPUT_TARGET_TYPE   1
#define GET_INPUT_HOST_TARGET   2
#define GET_INPUT_SERVICE_TARGET 3
#define GET_INPUT_OPTIONS       4

/* modes */
#define CREATE_HTML           0
#define CREATE_IMAGE          1

/* standard report times */
#define TIMEPERIOD_CUSTOM     0
#define TIMEPERIOD_TODAY      1
#define TIMEPERIOD_YESTERDAY  2
#define TIMEPERIOD_THISWEEK   3
#define TIMEPERIOD_LASTWEEK   4
#define TIMEPERIOD_THISMONTH  5
#define TIMEPERIOD_LASTMONTH  6
#define TIMEPERIOD_THISQUARTER      7
#define TIMEPERIOD_LASTQUARTER      8
#define TIMEPERIOD_THISYEAR   9
#define TIMEPERIOD_LASTYEAR   10
#define TIMEPERIOD_LAST24HOURS      11
#define TIMEPERIOD_LAST7DAYS  12

#define MIN_TIMESTAMP_SPACING 10

#define MAX_ARCHIVE_SPREAD    65
#define MAX_ARCHIVE           65
#define MAX_ARCHIVE_BACKTRACKS      60

authdata current_authdata;

typedef struct archived_state_struct{
      time_t  time_stamp;
      int     entry_type;
      int     processed_state;
      char    *state_info;
      struct archived_state_struct *next;
        }archived_state;


archived_state *as_list=NULL;

time_t t1;
time_t t2;

int start_second=0;
int start_minute=0;
int start_hour=0;
int start_day=1;
int start_month=1;
int start_year=2000;
int end_second=0;
int end_minute=0;
int end_hour=24;
int end_day=1;
int end_month=1;
int end_year=2000;

int display_type=DISPLAY_NO_TRENDS;
int mode=CREATE_HTML;
int input_type=GET_INPUT_NONE;
int timeperiod_type=TIMEPERIOD_LAST24HOURS;
int compute_time_from_parts=FALSE;

int display_popups=TRUE;
int use_map=TRUE;
int small_image=FALSE;
int embedded=FALSE;
int display_header=TRUE;

int assume_initial_states=TRUE;
int assume_state_retention=TRUE;

char *host_name="";
char *svc_description="";



void graph_all_trend_data(void);
void graph_trend_data(int,int,time_t,time_t,time_t,char *);
void draw_timestamps(void);
void draw_timestamp(int,time_t);
void draw_time_breakdowns(void);
void draw_horizontal_grid_lines(void);
void draw_dashed_line(int,int,int,int,int);

int convert_host_state_to_archived_state(int);
int convert_service_state_to_archived_state(int);
void add_archived_state(int,time_t,char *);
void free_archived_state_list(void);
void read_archived_state_data(void);
void scan_log_file_for_archived_state_data(char *);
void convert_timeperiod_to_times(int);
void compute_report_times(void);
void get_time_breakdown_string(unsigned long,unsigned long,char *,char *buffer,int);

void document_header(int);
void document_footer(void);
int process_cgivars(void);
void write_popup_code(void);

gdImagePtr trends_image=0;
int color_white=0;
int color_black=0;
int color_red=0;
int color_darkred=0;
int color_green=0;
int color_darkgreen=0;
int color_yellow=0;
int color_orange=0;
FILE *image_file=NULL;

int image_width=600;
int image_height=300;

#define HOST_DRAWING_WIDTH    498
#define HOST_DRAWING_HEIGHT   70
#define HOST_DRAWING_X_OFFSET 116
#define HOST_DRAWING_Y_OFFSET 55

#define SVC_DRAWING_WIDTH     498
#define SVC_DRAWING_HEIGHT    90
#define SVC_DRAWING_X_OFFSET  116
#define SVC_DRAWING_Y_OFFSET  55

#define SMALL_HOST_DRAWING_WIDTH    500
#define SMALL_HOST_DRAWING_HEIGHT   20
#define SMALL_HOST_DRAWING_X_OFFSET 0
#define SMALL_HOST_DRAWING_Y_OFFSET 0

#define SMALL_SVC_DRAWING_WIDTH     500
#define SMALL_SVC_DRAWING_HEIGHT    20
#define SMALL_SVC_DRAWING_X_OFFSET  0
#define SMALL_SVC_DRAWING_Y_OFFSET  0

int drawing_width=0;
int drawing_height=0;

int drawing_x_offset=0;
int drawing_y_offset=0;

int last_known_state=AS_NO_DATA;

int zoom_factor=4;
int backtrack_archives=2;
int earliest_archive=0;
time_t earliest_time;
time_t latest_time;
int earliest_state=AS_NO_DATA;
int latest_state=AS_NO_DATA;

int initial_assumed_state=AS_NO_DATA;

unsigned long time_up=0L;
unsigned long time_down=0L;
unsigned long time_unreachable=0L;
unsigned long time_ok=0L;
unsigned long time_warning=0L;
unsigned long time_unknown=0L;
unsigned long time_critical=0L;




int main(int argc, char **argv){
      int result=OK;
      char temp_buffer[MAX_INPUT_BUFFER];
      char image_template[MAX_INPUT_BUFFER];
      char start_time[MAX_INPUT_BUFFER];
      char end_time[MAX_INPUT_BUFFER];
      int string_width;
      int string_height;
      char start_timestring[MAX_INPUT_BUFFER];
      char end_timestring[MAX_INPUT_BUFFER];
      host *temp_host;
      service *temp_service;
      int is_authorized=TRUE;
      int found=FALSE;
      int days,hours,minutes,seconds;
      char *first_service=NULL;
      time_t t3;
      time_t current_time;
      struct tm *t;


      /* reset internal CGI variables */
      reset_cgi_vars();
      
      /* read the CGI configuration file */
      result=read_cgi_config_file(DEFAULT_CGI_CONFIG_FILE);
      if(result==ERROR){
            if(mode==CREATE_HTML){
                  document_header(FALSE);
                  cgi_config_file_error(DEFAULT_CGI_CONFIG_FILE);
                  document_footer();
                    }
            return ERROR;
              }

      /* read the main configuration file */
      result=read_main_config_file(main_config_file);
      if(result==ERROR){
            if(mode==CREATE_HTML){
                  document_header(FALSE);
                  main_config_file_error(main_config_file);
                  document_footer();
                    }
            return ERROR;
              }

      /* initialize time period to last 24 hours */
      time(&current_time);
      t2=current_time;
      t1=(time_t)(current_time-(60*60*24));

      /* default number of backtracked archives */
      switch(log_rotation_method){
      case LOG_ROTATION_MONTHLY:
            backtrack_archives=1;
            break;
      case LOG_ROTATION_WEEKLY:
            backtrack_archives=2;
            break;
      case LOG_ROTATION_DAILY:
            backtrack_archives=4;
            break;
      case LOG_ROTATION_HOURLY:
            backtrack_archives=8;
            break;
      default:
            backtrack_archives=2;
            break;
              }

      /* get the arguments passed in the URL */
      process_cgivars();

      /* get authentication information */
      get_authentication_information(&current_authdata);

      /* read all object configuration data */
      result=read_all_object_configuration_data(main_config_file,READ_ALL_OBJECT_DATA);
      if(result==ERROR){
            if(mode==CREATE_HTML){
                  document_header(FALSE);
                  object_data_error();
                  document_footer();
                    }
            return ERROR;
                }

      /* read all status data */
      result=read_all_status_data(DEFAULT_CGI_CONFIG_FILE,READ_ALL_STATUS_DATA);
      if(result==ERROR){
            if(mode==CREATE_HTML){
                  document_header(FALSE);
                  status_data_error();
                  document_footer();
                    }
            return ERROR;
                }

      document_header(TRUE);

      if(compute_time_from_parts==TRUE)
            compute_report_times();

      /* make sure times are sane, otherwise swap them */
      if(t2<t1){
            t3=t2;
            t2=t1;
            t1=t3;
              }

      /* don't let user create reports in the future */
      if(t2>current_time){
            t2=current_time;
            if(t1>t2)
                  t1=t2-(60*60*24);
              }
                  
      if(mode==CREATE_HTML && display_header==TRUE){

            /* begin top table */
            printf("<table border=0 width=100%% cellspacing=0 cellpadding=0>\n");
            printf("<tr>\n");

            /* left column of the first row */
            printf("<td align=left valign=top width=33%%>\n");

            if(display_type==DISPLAY_HOST_TRENDS)
                  snprintf(temp_buffer,sizeof(temp_buffer)-1,"Host State Trends");
            else if(display_type==DISPLAY_SERVICE_TRENDS)
                  snprintf(temp_buffer,sizeof(temp_buffer)-1,"Service State Trends");
            else
                  snprintf(temp_buffer,sizeof(temp_buffer)-1,"Host and Service State Trends");
            temp_buffer[sizeof(temp_buffer)-1]='\x0';
            display_info_table(temp_buffer,FALSE,&current_authdata);

            if(display_type!=DISPLAY_NO_TRENDS && input_type==GET_INPUT_NONE){

                  printf("<TABLE BORDER=1 CELLPADDING=0 CELLSPACING=0 CLASS='linkBox'>\n");
                  printf("<TR><TD CLASS='linkBox'>\n");

                  if(display_type==DISPLAY_HOST_TRENDS){
                        printf("<a href='%s?host=%s&t1=%lu&t2=%lu&assumestateretention=%s&assumeinitialstates=%s&initialassumedstate=%d&backtrack=%d&show_log_entries'>View Availability Report For This Host</a><BR>\n",AVAIL_CGI,url_encode(host_name),t1,t2,(assume_state_retention==TRUE)?"yes":"no",(assume_initial_states==TRUE)?"yes":"no",initial_assumed_state,backtrack_archives);
                        printf("<a href='%s?host=%s&t1=%lu&t2=%lu&assumestateretention=%s'>View Alert Histogram For This Host</a><BR>\n",HISTOGRAM_CGI,url_encode(host_name),t1,t2,(assume_state_retention==TRUE)?"yes":"no");
                        printf("<a href='%s?host=%s'>View Status Detail For This Host</a><BR>\n",STATUS_CGI,url_encode(host_name));
                        printf("<a href='%s?host=%s'>View Alert History For This Host</a><BR>\n",HISTORY_CGI,url_encode(host_name));
                        printf("<a href='%s?host=%s'>View Notifications For This Host</a><BR>\n",NOTIFICATIONS_CGI,url_encode(host_name));
                            }
                  else{
                        printf("<a href='%s?host=%s&t1=%lu&t2=%lu&assumestateretention=%s&assumeinitialstates=%s&initialassumedstate=%d&backtrack=%d'>View Trends For This Host</a><BR>\n",TRENDS_CGI,url_encode(host_name),t1,t2,(assume_state_retention==TRUE)?"yes":"no",(assume_initial_states==TRUE)?"yes":"no",initial_assumed_state,backtrack_archives);
                        printf("<a href='%s?host=%s",AVAIL_CGI,url_encode(host_name));
                        printf("&service=%s&t1=%lu&t2=%lu&assumestateretention=%s&assumeinitialstates=%s&backtrack=%d&show_log_entries'>View Availability Report For This Service</a><BR>\n",url_encode(svc_description),t1,t2,(assume_state_retention==TRUE)?"yes":"no",(assume_initial_states==TRUE)?"yes":"no",backtrack_archives);
                        printf("<a href='%s?host=%s",HISTOGRAM_CGI,url_encode(host_name));
                        printf("&service=%s&t1=%lu&t2=%lu&assumestateretention=%s'>View Alert Histogram For This Service</a><BR>\n",url_encode(svc_description),t1,t2,(assume_state_retention==TRUE)?"yes":"no");
                        printf("<A HREF='%s?host=%s&",HISTORY_CGI,url_encode(host_name));
                        printf("service=%s'>View Alert History This Service</A><BR>\n",url_encode(svc_description));
                        printf("<A HREF='%s?host=%s&",NOTIFICATIONS_CGI,url_encode(host_name));
                        printf("service=%s'>View Notifications This Service</A><BR>\n",url_encode(svc_description));
                            }

                  printf("</TD></TR>\n");
                  printf("</TABLE>\n");
                    }

            printf("</td>\n");

            /* center column of top row */
            printf("<td align=center valign=top width=33%%>\n");

            if(display_type!=DISPLAY_NO_TRENDS && input_type==GET_INPUT_NONE){

                  printf("<DIV ALIGN=CENTER CLASS='dataTitle'>\n");
                  if(display_type==DISPLAY_HOST_TRENDS)
                        printf("Host '%s'",host_name);
                  else if(display_type==DISPLAY_SERVICE_TRENDS)
                        printf("Service '%s' On Host '%s'",svc_description,host_name);
                  printf("</DIV>\n");

                  printf("<BR>\n");

                  printf("<IMG SRC='%s%s' BORDER=0 ALT='%s State Trends'>\n",url_images_path,TRENDS_ICON,(display_type==DISPLAY_HOST_TRENDS)?"Host":"Service");

                  printf("<BR CLEAR=ALL>\n");

                  get_time_string(&t1,start_timestring,sizeof(start_timestring)-1,SHORT_DATE_TIME);
                  get_time_string(&t2,end_timestring,sizeof(end_timestring)-1,SHORT_DATE_TIME);
                  printf("<div align=center class='reportRange'>%s to %s</div>\n",start_timestring,end_timestring);

                  get_time_breakdown((time_t)(t2-t1),&days,&hours,&minutes,&seconds);
                  printf("<div align=center class='reportDuration'>Duration: %dd %dh %dm %ds</div>\n",days,hours,minutes,seconds);
                    }

            printf("</td>\n");

            /* right hand column of top row */
            printf("<td align=right valign=bottom width=33%%>\n");

            printf("<table border=0 CLASS='optBox'>\n");

            if(display_type!=DISPLAY_NO_TRENDS && input_type==GET_INPUT_NONE){

                  printf("<form method=\"GET\" action=\"%s\">\n",TRENDS_CGI);
                  if(display_popups==FALSE)
                        printf("<input type='hidden' name='nopopups' value=''>\n");
                  if(use_map==FALSE)
                        printf("<input type='hidden' name='nomap' value=''>\n");
                  printf("<input type='hidden' name='t1' value='%lu'>\n",(unsigned long)t1);
                  printf("<input type='hidden' name='t2' value='%lu'>\n",(unsigned long)t2);
                  printf("<input type='hidden' name='host' value='%s'>\n",host_name);
                  if(display_type==DISPLAY_SERVICE_TRENDS)
                        printf("<input type='hidden' name='service' value='%s'>\n",svc_description);

                  printf("<tr><td CLASS='optBoxItem' valign=top align=left>Assume initial states:</td><td CLASS='optBoxItem' valign=top align=left>Assume state retention:</td></tr>\n");
                  printf("<tr><td CLASS='optBoxItem' valign=top align=left>\n");
                  printf("<select name='assumeinitialstates'>\n");
                  printf("<option value=yes %s>yes\n",(assume_initial_states==TRUE)?"selected":"");
                  printf("<option value=no %s>no\n",(assume_initial_states==TRUE)?"":"selected");
                  printf("</select>\n");
                  printf("</td><td CLASS='optBoxItem' valign=top align=left>\n");
                  printf("<select name='assumestateretention'>\n");
                  printf("<option value=yes %s>yes\n",(assume_state_retention==TRUE)?"selected":"");
                  printf("<option value=no %s>no\n",(assume_state_retention==TRUE)?"":"selected");
                  printf("</select>\n");
                  printf("</td></tr>\n");

                  printf("<tr><td CLASS='optBoxItem' valign=top align=left>First assumed state:</td><td CLASS='optBoxItem' valign=top align=left>Backtracked archives:</td></tr>\n");
                  printf("<tr><td CLASS='optBoxItem' valign=top align=left>");
                  printf("<select name='initialassumedstate'>\n");
                  printf("<option value=%d %s>Unspecified\n",AS_NO_DATA,(initial_assumed_state==AS_NO_DATA)?"SELECTED":"");
                  printf("<option value=%d %s>Current State\n",AS_CURRENT_STATE,(initial_assumed_state==AS_CURRENT_STATE)?"SELECTED":"");
                  if(display_type==DISPLAY_HOST_TRENDS){
                        printf("<option value=%d %s>Host Up\n",AS_HOST_UP,(initial_assumed_state==AS_HOST_UP)?"SELECTED":"");
                        printf("<option value=%d %s>Host Down\n",AS_HOST_DOWN,(initial_assumed_state==AS_HOST_DOWN)?"SELECTED":"");
                        printf("<option value=%d %s>Host Unreachable\n",AS_HOST_UNREACHABLE,(initial_assumed_state==AS_HOST_UNREACHABLE)?"SELECTED":"");
                          }
                  else{
                        printf("<option value=%d %s>Service Ok\n",AS_SVC_OK,(initial_assumed_state==AS_SVC_OK)?"SELECTED":"");
                        printf("<option value=%d %s>Service Warning\n",AS_SVC_WARNING,(initial_assumed_state==AS_SVC_WARNING)?"SELECTED":"");
                        printf("<option value=%d %s>Service Unknown\n",AS_SVC_UNKNOWN,(initial_assumed_state==AS_SVC_UNKNOWN)?"SELECTED":"");
                        printf("<option value=%d %s>Service Critical\n",AS_SVC_CRITICAL,(initial_assumed_state==AS_SVC_CRITICAL)?"SELECTED":"");
                          }
                  printf("</select>\n");
                  printf("</td><td CLASS='optBoxItem' valign=top align=left>\n");
                  printf("<input type='text' name='backtrack' size='2' maxlength='2' value='%d'>\n",backtrack_archives);
                  printf("</td></tr>\n");

                  printf("<tr><td CLASS='optBoxItem' valign=top align=left>Report period:</td><td CLASS='optBoxItem' valign=top align=left>Zoom factor:</td></tr>\n");
                  printf("<tr><td CLASS='optBoxItem' valign=top align=left>\n");
                  printf("<select name='timeperiod'>\n");
                  printf("<option SELECTED>[ Current time range ]\n");
                  printf("<option value=last24hours>Last 24 Hours\n");
                  printf("<option value=today>Today\n");
                  printf("<option value=yesterday>Yesterday\n");
                  printf("<option value=thisweek>This Week\n");
                  printf("<option value=last7days>Last 7 Days\n");
                  printf("<option value=lastweek>Last Week\n");
                  printf("<option value=thismonth>This Month\n");
                  printf("<option value=lastmonth>Last Month\n");
                  printf("<option value=thisyear>This Year\n");
                  printf("<option value=lastyear>Last Year\n");
                  printf("</select>\n");
                  printf("</td><td CLASS='optBoxItem' valign=top align=left>\n");
                  printf("<select name='zoom'>\n");
                  printf("<option value=%d selected>%d\n",zoom_factor,zoom_factor);
                  printf("<option value=+2>+2\n");
                  printf("<option value=+3>+3\n");
                  printf("<option value=+4>+4\n");
                  printf("<option value=-2>-2\n");
                  printf("<option value=-3>-3\n");
                  printf("<option value=-4>-4\n");
                  printf("</select>\n");
                  printf("</td></tr>\n");

                  printf("<tr><td CLASS='optBoxItem' valign=top align=left>\n");
                  printf("</td><td CLASS='optBoxItem' valign=top align=left>\n");
                  printf("<input type='submit' value='Update'>\n");
                  printf("</td></tr>\n");

                  printf("</form>\n");
                    }

            /* display context-sensitive help */
            printf("<tr><td></td><td align=right valign=bottom>\n");
            if(display_type!=DISPLAY_NO_TRENDS && input_type==GET_INPUT_NONE){
                  if(display_type==DISPLAY_HOST_TRENDS)
                        display_context_help(CONTEXTHELP_TRENDS_HOST);
                  else
                        display_context_help(CONTEXTHELP_TRENDS_SERVICE);
                    }
            else if(display_type==DISPLAY_NO_TRENDS || input_type!=GET_INPUT_NONE){
                  if(input_type==GET_INPUT_NONE)
                        display_context_help(CONTEXTHELP_TRENDS_MENU1);
                  else if(input_type==GET_INPUT_TARGET_TYPE)
                        display_context_help(CONTEXTHELP_TRENDS_MENU1);
                  else if(input_type==GET_INPUT_HOST_TARGET)
                        display_context_help(CONTEXTHELP_TRENDS_MENU2);
                  else if(input_type==GET_INPUT_SERVICE_TARGET)
                        display_context_help(CONTEXTHELP_TRENDS_MENU3);
                  else if(input_type==GET_INPUT_OPTIONS)
                        display_context_help(CONTEXTHELP_TRENDS_MENU4);
                    }
            printf("</td></tr>\n");

            printf("</table>\n");

            printf("</td>\n");

            /* end of top table */
            printf("</tr>\n");
            printf("</table>\n");
              }


#ifndef DEBUG
      /* check authorization... */
      if(display_type==DISPLAY_HOST_TRENDS){
            temp_host=find_host(host_name,NULL);
            if(temp_host==NULL || is_authorized_for_host(temp_host,&current_authdata)==FALSE)
                  is_authorized=FALSE;
              }
      else if(display_type==DISPLAY_SERVICE_TRENDS){
            temp_service=find_service(host_name,svc_description,NULL);
            if(temp_service==NULL || is_authorized_for_service(temp_service,&current_authdata)==FALSE)
                  is_authorized=FALSE;
              }
      if(is_authorized==FALSE){

            if(mode==CREATE_HTML)
                  printf("<P><DIV ALIGN=CENTER CLASS='errorMessage'>It appears as though you are not authorized to view information for the specified %s...</DIV></P>\n",(display_type==DISPLAY_HOST_TRENDS)?"host":"service");

            document_footer();
            free_memory();
            return ERROR;
              }
#endif


      /* set drawing parameters, etc */

      if(small_image==TRUE){
            image_height=20;
            image_width=500;
              }
      else{
            image_height=300;
            image_width=600;
              }

      if(display_type==DISPLAY_HOST_TRENDS){

            if(small_image==TRUE){
                  drawing_width=SMALL_HOST_DRAWING_WIDTH;
                  drawing_height=SMALL_HOST_DRAWING_HEIGHT;
                  drawing_x_offset=SMALL_HOST_DRAWING_X_OFFSET;
                  drawing_y_offset=SMALL_HOST_DRAWING_Y_OFFSET;
                    }
            else{
                  drawing_width=HOST_DRAWING_WIDTH;
                  drawing_height=HOST_DRAWING_HEIGHT;
                  drawing_x_offset=HOST_DRAWING_X_OFFSET;
                  drawing_y_offset=HOST_DRAWING_Y_OFFSET;
                    }
              }
      else if(display_type==DISPLAY_SERVICE_TRENDS){

            if(small_image==TRUE){
                  drawing_width=SMALL_SVC_DRAWING_WIDTH;
                  drawing_height=SMALL_SVC_DRAWING_HEIGHT;
                  drawing_x_offset=SMALL_SVC_DRAWING_X_OFFSET;
                  drawing_y_offset=SMALL_SVC_DRAWING_Y_OFFSET;
                    }
            else{
                  drawing_width=SVC_DRAWING_WIDTH;
                  drawing_height=SVC_DRAWING_HEIGHT;
                  drawing_x_offset=SVC_DRAWING_X_OFFSET;
                  drawing_y_offset=SVC_DRAWING_Y_OFFSET;
                    }
              }

      /* last known state should always be initially set to indeterminate! */
      last_known_state=AS_NO_DATA;


      /* initialize PNG image */
      if(display_type!=DISPLAY_NO_TRENDS && mode==CREATE_IMAGE){

            if(small_image==TRUE){
                  trends_image=gdImageCreate(image_width,image_height);
                  if(trends_image==NULL){
#ifdef DEBUG
                        printf("Error: Could not allocate memory for image\n");
#endif
                        return ERROR;
                              }
                    }

            else{

                  if(display_type==DISPLAY_HOST_TRENDS)
                        snprintf(image_template,sizeof(image_template)-1,"%s/trendshost.png",physical_images_path);
                  else
                        snprintf(image_template,sizeof(image_template)-1,"%s/trendssvc.png",physical_images_path);
                  image_template[sizeof(image_template)-1]='\x0';

                  /* allocate buffer for storing image */
                  trends_image=NULL;
                  image_file=fopen(image_template,"r");
                  if(image_file!=NULL){
                        trends_image=gdImageCreateFromPng(image_file);
                        fclose(image_file);
                          }
                  if(trends_image==NULL)
                        trends_image=gdImageCreate(image_width,image_height);
                  if(trends_image==NULL){
#ifdef DEBUG
                        printf("Error: Could not allocate memory for image\n");
#endif
                        return ERROR;
                              }
                    }

            /* allocate colors used for drawing */
            color_white=gdImageColorAllocate(trends_image,255,255,255);
            color_black=gdImageColorAllocate(trends_image,0,0,0);
            color_red=gdImageColorAllocate(trends_image,255,0,0);
            color_darkred=gdImageColorAllocate(trends_image,128,0,0);
            color_green=gdImageColorAllocate(trends_image,0,210,0);
            color_darkgreen=gdImageColorAllocate(trends_image,0,128,0);
            color_yellow=gdImageColorAllocate(trends_image,176,178,20);
            color_orange=gdImageColorAllocate(trends_image,255,100,25);

            /* set transparency index */
            gdImageColorTransparent(trends_image,color_white);

            /* make sure the graphic is interlaced */
            gdImageInterlace(trends_image,1);

            if(small_image==FALSE){

                  /* title */
                  snprintf(start_time,sizeof(start_time)-1,"%s",ctime(&t1));
                  start_time[sizeof(start_time)-1]='\x0';
                  start_time[strlen(start_time)-1]='\x0';
                  snprintf(end_time,sizeof(end_time)-1,"%s",ctime(&t2));
                  end_time[sizeof(end_time)-1]='\x0';
                  end_time[strlen(end_time)-1]='\x0';
                  
                  string_height=gdFontSmall->h;

                  if(display_type==DISPLAY_HOST_TRENDS)
                        snprintf(temp_buffer,sizeof(temp_buffer)-1,"State History For Host '%s'",host_name);
                  else
                        snprintf(temp_buffer,sizeof(temp_buffer)-1,"State History For Service '%s' On Host '%s'",svc_description,host_name);
                  temp_buffer[sizeof(temp_buffer)-1]='\x0';
                  string_width=gdFontSmall->w*strlen(temp_buffer);
                  gdImageString(trends_image,gdFontSmall,(drawing_width/2)-(string_width/2)+drawing_x_offset,string_height,(unsigned char *)temp_buffer,color_black);

                  snprintf(temp_buffer,sizeof(temp_buffer)-1,"%s to %s",start_time,end_time);
                  temp_buffer[sizeof(temp_buffer)-1]='\x0';
                  string_width=gdFontSmall->w*strlen(temp_buffer);
                  gdImageString(trends_image,gdFontSmall,(drawing_width/2)-(string_width/2)+drawing_x_offset,(string_height*2)+5,(unsigned char *)temp_buffer,color_black);


                  /* first time stamp */
                  snprintf(temp_buffer,sizeof(temp_buffer)-1,"%s",start_time);
                  temp_buffer[sizeof(temp_buffer)-1]='\x0';
                  string_width=gdFontSmall->w*strlen(temp_buffer);
                  gdImageStringUp(trends_image,gdFontSmall,drawing_x_offset-(string_height/2),drawing_y_offset+drawing_height+string_width+5,(unsigned char *)temp_buffer,color_black);
                    }
              }

      if(display_type!=DISPLAY_NO_TRENDS && input_type==GET_INPUT_NONE){


            if(mode==CREATE_IMAGE || (mode==CREATE_HTML && use_map==TRUE)){

                  /* read in all necessary archived state data */
                  read_archived_state_data();

                  /* graph archived state trend data */
                  graph_all_trend_data();
                    }

            /* print URL to image */
            if(mode==CREATE_HTML){

                  printf("<BR><BR>\n");
                  printf("<DIV ALIGN=CENTER>\n");
                  printf("<IMG SRC='%s?createimage&t1=%lu&t2=%lu",TRENDS_CGI,(unsigned long)t1,(unsigned long)t2);
                  printf("&assumeinitialstates=%s",(assume_initial_states==TRUE)?"yes":"no");
                  printf("&initialassumedstate=%d",initial_assumed_state);
                  printf("&assumestateretention=%s",(assume_state_retention==TRUE)?"yes":"no");
                  printf("&host=%s",url_encode(host_name));
                  if(display_type==DISPLAY_SERVICE_TRENDS)
                        printf("&service=%s",url_encode(svc_description));
                  if(backtrack_archives>0)
                        printf("&backtrack=%d",backtrack_archives);
                  printf("&zoom=%d",zoom_factor);
                  printf("' BORDER=0 name='trendsimage' useMap='#trendsmap' width=900>\n");
                  printf("</DIV>\n");
                    }

            if(mode==CREATE_IMAGE || (mode==CREATE_HTML && use_map==TRUE)){

                  /* draw timestamps */
                  draw_timestamps();

                  /* draw horizontal lines */
                  draw_horizontal_grid_lines();

                  /* draw state time breakdowns */
                  draw_time_breakdowns();
                    }

            if(mode==CREATE_IMAGE){

                  /* use STDOUT for writing the image data... */
                  image_file=stdout;

#ifndef DEBUG
                  /* write the image to file */
                  gdImagePng(trends_image,image_file);
#endif
#ifdef DEBUG
                  image_file=fopen("trends.png","w");
                  if(image_file==NULL)
                        printf("Could not open trends.png for writing!\n");
                  else{
                        gdImagePng(trends_image,image_file);
                        fclose(image_file);
                            }
#endif

                  /* free memory allocated to image */
                  gdImageDestroy(trends_image);
                      }
              }


      /* show user a selection of hosts and services to choose from... */
      if(display_type==DISPLAY_NO_TRENDS || input_type!=GET_INPUT_NONE){

            /* ask the user for what host they want a report for */
            if(input_type==GET_INPUT_HOST_TARGET){

                  printf("<P><DIV ALIGN=CENTER>\n");
                  printf("<DIV CLASS='reportSelectTitle'>Step 2: Select Host</DIV>\n");
                  printf("</DIV></P>\n");

                  printf("<P><DIV ALIGN=CENTER>\n");

                  printf("<TABLE BORDER=0 cellspacing=0 cellpadding=10>\n");
                  printf("<form method=\"GET\" action=\"%s\">\n",TRENDS_CGI);
                  printf("<input type='hidden' name='input' value='getoptions'>\n");

                  printf("<tr><td class='reportSelectSubTitle' valign=center>Host</td>\n");
                  printf("<td class='reportSelectItem' valign=center>\n");
                  printf("<select name='host'>\n");

                  for(temp_host=host_list;temp_host!=NULL;temp_host=temp_host->next){
                        if(is_authorized_for_host(temp_host,&current_authdata)==TRUE)
                              printf("<option value='%s'>%s\n",temp_host->name,temp_host->name);
                          }

                  printf("</select>\n");
                  printf("</td></tr>\n");

                  printf("<tr><td></td><td class='reportSelectItem'>\n");
                  printf("<input type='submit' value='Continue to Step 3'>\n");
                  printf("</td></tr>\n");

                  printf("</form>\n");
                  printf("</TABLE>\n");

                  printf("</DIV></P>\n");
                    }

            /* ask the user for what service they want a report for */
            else if(input_type==GET_INPUT_SERVICE_TARGET){

                  printf("<SCRIPT LANGUAGE='JavaScript'>\n");
                  printf("function gethostname(hostindex){\n");
                  printf("hostnames=[");

                  for(temp_service=service_list;temp_service!=NULL;temp_service=temp_service->next){
                        if(is_authorized_for_service(temp_service,&current_authdata)==TRUE){
                              if(found==TRUE)
                                    printf(",");
                              else
                                    first_service=temp_service->host_name;
                              printf(" \"%s\"",temp_service->host_name);
                              found=TRUE;
                                  }
                            }
            
                  printf(" ]\n");
                  printf("return hostnames[hostindex];\n");
                  printf("}\n");
                  printf("</SCRIPT>\n");


                  printf("<P><DIV ALIGN=CENTER>\n");
                  printf("<DIV CLASS='reportSelectTitle'>Step 2: Select Service</DIV>\n");
                  printf("</DIV></P>\n");

                  printf("<P><DIV ALIGN=CENTER>\n");
                  
                  printf("<TABLE BORDER=0 cellpadding=5>\n");
                  printf("<form method=\"GET\" action=\"%s\" name=\"serviceform\">\n",TRENDS_CGI);
                  printf("<input type='hidden' name='input' value='getoptions'>\n");
                  printf("<input type='hidden' name='host' value='%s'>\n",(first_service==NULL)?"unknown":first_service);

                  printf("<tr><td class='reportSelectSubTitle'>Service:</td>\n");
                  printf("<td class='reportSelectItem'>\n");
                  printf("<select name='service' onFocus='document.serviceform.host.value=gethostname(this.selectedIndex);' onChange='document.serviceform.host.value=gethostname(this.selectedIndex);'>\n");

                  for(temp_service=service_list;temp_service!=NULL;temp_service=temp_service->next){
                        if(is_authorized_for_service(temp_service,&current_authdata)==TRUE)
                              printf("<option value='%s'>%s;%s\n",temp_service->description,temp_service->host_name,temp_service->description);
                            }

                  printf("</select>\n");
                  printf("</td></tr>\n");

                  printf("<tr><td></td><td class='reportSelectItem'>\n");
                  printf("<input type='submit' value='Continue to Step 3'>\n");
                  printf("</td></tr>\n");

                  printf("</form>\n");
                  printf("</TABLE>\n");

                  printf("</DIV></P>\n");
                    }

            /* ask the user for report range and options */
            else if(input_type==GET_INPUT_OPTIONS){

                  time(&current_time);
                  t=localtime(&current_time);

                  start_day=1;
                  start_year=t->tm_year+1900;
                  end_day=t->tm_mday;
                  end_year=t->tm_year+1900;

                  printf("<P><DIV ALIGN=CENTER>\n");
                  printf("<DIV CLASS='reportSelectTitle'>Step 3: Select Report Options</DIV>\n");
                  printf("</DIV></P>\n");

                  printf("<P><DIV ALIGN=CENTER>\n");

                  printf("<TABLE BORDER=0 CELLPADDING=5>\n");
                  printf("<form method=\"GET\" action=\"%s\">\n",TRENDS_CGI);
                  printf("<input type='hidden' name='host' value='%s'>\n",host_name);
                  if(display_type==DISPLAY_SERVICE_TRENDS)
                        printf("<input type='hidden' name='service' value='%s'>\n",svc_description);

                  printf("<tr><td class='reportSelectSubTitle' align=right>Report period:</td>\n");
                  printf("<td class='reportSelectItem'>\n");
                  printf("<select name='timeperiod'>\n");
                  printf("<option value=last24hours>Last 24 Hours\n");
                  printf("<option value=today>Today\n");
                  printf("<option value=yesterday>Yesterday\n");
                  printf("<option value=thisweek>This Week\n");
                  printf("<option value=last7days SELECTED>Last 7 Days\n");
                  printf("<option value=lastweek>Last Week\n");
                  printf("<option value=thismonth>This Month\n");
                  printf("<option value=lastmonth>Last Month\n");
                  printf("<option value=thisyear>This Year\n");
                  printf("<option value=lastyear>Last Year\n");
                  printf("<option value=custom>* CUSTOM REPORT PERIOD *\n");
                  printf("</select>\n");
                  printf("</td></tr>\n");

                  printf("<tr><td colspan=2 valign=top calss='reportSelectSubTitle'><i>If Custom Report Period...</i></td></tr>\n");

                  printf("<tr>");
                  printf("<td valign=top class='reportSelectSubTitle'>Start Date (Inclusive):</td>\n");
                  printf("<td align=left valign=top class='reportSelectItem'>");
                  printf("<select name='smon'>\n");
                  printf("<option value='1' %s>January\n",(t->tm_mon==0)?"SELECTED":"");
                  printf("<option value='2' %s>February\n",(t->tm_mon==1)?"SELECTED":"");
                  printf("<option value='3' %s>March\n",(t->tm_mon==2)?"SELECTED":"");
                  printf("<option value='4' %s>April\n",(t->tm_mon==3)?"SELECTED":"");
                  printf("<option value='5' %s>May\n",(t->tm_mon==4)?"SELECTED":"");
                  printf("<option value='6' %s>June\n",(t->tm_mon==5)?"SELECTED":"");
                  printf("<option value='7' %s>July\n",(t->tm_mon==6)?"SELECTED":"");
                  printf("<option value='8' %s>August\n",(t->tm_mon==7)?"SELECTED":"");
                  printf("<option value='9' %s>September\n",(t->tm_mon==8)?"SELECTED":"");
                  printf("<option value='10' %s>October\n",(t->tm_mon==9)?"SELECTED":"");
                  printf("<option value='11' %s>November\n",(t->tm_mon==10)?"SELECTED":"");
                  printf("<option value='12' %s>December\n",(t->tm_mon==11)?"SELECTED":"");
                  printf("</select>\n ");
                  printf("<input type='text' size='2' maxlength='2' name='sday' value='%d'> ",start_day);
                  printf("<input type='text' size='4' maxlength='4' name='syear' value='%d'>",start_year);
                  printf("<input type='hidden' name='shour' value='0'>\n");
                  printf("<input type='hidden' name='smin' value='0'>\n");
                  printf("<input type='hidden' name='ssec' value='0'>\n");
                  printf("</td>\n");
                  printf("</tr>\n");

                  printf("<tr>");
                  printf("<td valign=top class='reportSelectSubTitle'>End Date (Inclusive):</td>\n");
                  printf("<td align=left valign=top class='reportSelectItem'>");
                  printf("<select name='emon'>\n");
                  printf("<option value='1' %s>January\n",(t->tm_mon==0)?"SELECTED":"");
                  printf("<option value='2' %s>February\n",(t->tm_mon==1)?"SELECTED":"");
                  printf("<option value='3' %s>March\n",(t->tm_mon==2)?"SELECTED":"");
                  printf("<option value='4' %s>April\n",(t->tm_mon==3)?"SELECTED":"");
                  printf("<option value='5' %s>May\n",(t->tm_mon==4)?"SELECTED":"");
                  printf("<option value='6' %s>June\n",(t->tm_mon==5)?"SELECTED":"");
                  printf("<option value='7' %s>July\n",(t->tm_mon==6)?"SELECTED":"");
                  printf("<option value='8' %s>August\n",(t->tm_mon==7)?"SELECTED":"");
                  printf("<option value='9' %s>September\n",(t->tm_mon==8)?"SELECTED":"");
                  printf("<option value='10' %s>October\n",(t->tm_mon==9)?"SELECTED":"");
                  printf("<option value='11' %s>November\n",(t->tm_mon==10)?"SELECTED":"");
                  printf("<option value='12' %s>December\n",(t->tm_mon==11)?"SELECTED":"");
                  printf("</select>\n ");
                  printf("<input type='text' size='2' maxlength='2' name='eday' value='%d'> ",end_day);
                  printf("<input type='text' size='4' maxlength='4' name='eyear' value='%d'>",end_year);
                  printf("<input type='hidden' name='ehour' value='24'>\n");
                  printf("<input type='hidden' name='emin' value='0'>\n");
                  printf("<input type='hidden' name='esec' value='0'>\n");
                  printf("</td>\n");
                  printf("</tr>\n");

                  printf("<tr><td colspan=2><br></td></tr>\n");

                  printf("<tr><td class='reportSelectSubTitle' align=right>Assume Initial States:</td>\n");
                  printf("<td class='reportSelectItem'>\n");
                  printf("<select name='assumeinitialstates'>\n");
                  printf("<option value=yes>Yes\n");
                  printf("<option value=no>No\n");
                  printf("</select>\n");
                  printf("</td></tr>\n");

                  printf("<tr><td class='reportSelectSubTitle' align=right>Assume State Retention:</td>\n");
                  printf("<td class='reportSelectItem'>\n");
                  printf("<select name='assumestateretention'>\n");
                  printf("<option value=yes>Yes\n");
                  printf("<option value=no>No\n");
                  printf("</select>\n");
                  printf("</td></tr>\n");

                  printf("<tr><td class='reportSelectSubTitle' align=right>First Assumed State:</td>\n");
                  printf("<td class='reportSelectItem'>\n");
                  printf("<select name='initialassumedstate'>\n");
                  printf("<option value=%d>Unspecified\n",AS_NO_DATA);
                  printf("<option value=%d>Current State\n",AS_CURRENT_STATE);
                  if(display_type==DISPLAY_HOST_TRENDS){
                        printf("<option value=%d>Host Up\n",AS_HOST_UP);
                        printf("<option value=%d>Host Down\n",AS_HOST_DOWN);
                        printf("<option value=%d>Host Unreachable\n",AS_HOST_UNREACHABLE);
                          }
                  else{
                        printf("<option value=%d>Service Ok\n",AS_SVC_OK);
                        printf("<option value=%d>Service Warning\n",AS_SVC_WARNING);
                        printf("<option value=%d>Service Unknown\n",AS_SVC_UNKNOWN);
                        printf("<option value=%d>Service Critical\n",AS_SVC_CRITICAL);
                          }
                  printf("</select>\n");
                  printf("</td></tr>\n");

                  printf("<tr><td class='reportSelectSubTitle' align=right>Backtracked Archives:</td>\n");
                  printf("<td class='reportSelectItem'>\n");
                  printf("<input type='text' name='backtrack' size='2' maxlength='2' value='%d'>\n",backtrack_archives);
                  printf("</td></tr>\n");

                  printf("<tr><td class='reportSelectSubTitle' align=right>Suppress image map:</td><td class='reportSelectItem'><input type='checkbox' name='nomap'></td></tr>");
                  printf("<tr><td class='reportSelectSubTitle' align=right>Suppress popups:</td><td class='reportSelectItem'><input type='checkbox' name='nopopups'></td></tr>\n");

                  printf("<tr><td></td><td class='reportSelectItem'><input type='submit' value='Create Report'></td></tr>\n");

                  printf("</form>\n");
                  printf("</TABLE>\n");

                  printf("</DIV></P>\n");

                  /*
                  printf("<P><DIV ALIGN=CENTER CLASS='helpfulHint'>\n");
                  printf("Note: Choosing the 'suppress image map' option will make the report run approximately twice as fast as it would otherwise, but it will prevent you from being able to zoom in on specific time periods.\n");
                  printf("</DIV></P>\n");
                  */
                    }

            /* as the user whether they want a graph for a host or service */
            else{
                  printf("<P><DIV ALIGN=CENTER>\n");
                  printf("<DIV CLASS='reportSelectTitle'>Step 1: Select Report Type</DIV>\n");
                  printf("</DIV></P>\n");

                  printf("<P><DIV ALIGN=CENTER>\n");

                  printf("<TABLE BORDER=0 cellpadding=5>\n");
                  printf("<form method=\"GET\" action=\"%s\">\n",TRENDS_CGI);

                  printf("<tr><td class='reportSelectSubTitle'><input type='radio' name='input' value='gethost'></td><td class='reportSelectItem'>Host</td></tr>\n");
                  printf("<tr><td class='reportSelectSubTitle'><input type='radio' name='input' value='getservice'></td><td class='reportSelectItem'>Service</td></tr>\n");

                  printf("<tr><td></td><td class='reportSelectItem'>\n");
                  printf("<input type='submit' value='Continue to Step 2'>\n");
                  printf("</td></tr>\n");

                  printf("</form>\n");
                  printf("</TABLE>\n");

                  printf("</DIV></P>\n");
                    }
            
                }

      document_footer();

      /* free memory allocated to the archived state data list */
      free_archived_state_list();

      /* free all other allocated memory */
      free_memory();

      return OK;
        }



void document_header(int use_stylesheet){
      char date_time[MAX_DATETIME_LENGTH];
      time_t current_time;
      time_t expire_time;

      if(mode==CREATE_HTML){
            printf("Cache-Control: no-store\n");
            printf("Pragma: no-cache\n");

            time(&current_time);
            get_time_string(&current_time,date_time,sizeof(date_time),HTTP_DATE_TIME);
            printf("Last-Modified: %s\n",date_time);

            expire_time=(time_t)0;
            get_time_string(&expire_time,date_time,sizeof(date_time),HTTP_DATE_TIME);
            printf("Expires: %s\n",date_time);

            printf("Content-type: text/html\n\n");

            if(embedded==TRUE)
                  return;

            printf("<html>\n");
            printf("<head>\n");
            printf("<title>\n");
            printf("Nagios Trends\n");
            printf("</title>\n");

            if(use_stylesheet==TRUE)
                  printf("<LINK REL='stylesheet' TYPE='text/css' HREF='%s%s'>\n",url_stylesheets_path,TRENDS_CSS);
      
            /* write JavaScript code for popup window */
            if(display_type!=DISPLAY_NO_TRENDS)
                  write_popup_code();

            printf("</head>\n");

            printf("<BODY CLASS='trends'>\n");

            /* include user SSI header */
            include_ssi_files(TRENDS_CGI,SSI_HEADER);

            printf("<div id=\"popup\" style=\"position:absolute; z-index:1; visibility: hidden\"></div>\n");
              }

      else{
            printf("Cache-Control: no-store\n");
            printf("Pragma: no-cache\n");

            time(&current_time);
            get_time_string(&current_time,date_time,sizeof(date_time),HTTP_DATE_TIME);
            printf("Last-Modified: %s\n",date_time);

            expire_time=(time_t)0L;
            get_time_string(&expire_time,date_time,sizeof(date_time),HTTP_DATE_TIME);
            printf("Expires: %s\n",date_time);

            printf("Content-Type: image/png\n\n");
              }

      return;
        }



void document_footer(void){

      if(embedded==TRUE)
            return;

      if(mode==CREATE_HTML){

            /* include user SSI footer */
            include_ssi_files(TRENDS_CGI,SSI_FOOTER);

            printf("</body>\n");
            printf("</html>\n");
              }

      return;
        }



int process_cgivars(void){
      char **variables;
      int error=FALSE;
      int x;

      variables=getcgivars();

      for(x=0;variables[x]!=NULL;x++){

            /* do some basic length checking on the variable identifier to prevent buffer overflows */
            if(strlen(variables[x])>=MAX_INPUT_BUFFER-1){
                  x++;
                  continue;
                    }

            /* we found the host argument */
            else if(!strcmp(variables[x],"host")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  host_name=(char *)malloc(strlen(variables[x])+1);
                  if(host_name==NULL)
                        host_name="";
                  else
                        strcpy(host_name,variables[x]);

                  display_type=DISPLAY_HOST_TRENDS;
                    }

            /* we found the node width argument */
            else if(!strcmp(variables[x],"service")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  svc_description=(char *)malloc(strlen(variables[x])+1);
                  if(svc_description==NULL)
                        svc_description="";
                  else
                        strcpy(svc_description,variables[x]);

                  display_type=DISPLAY_SERVICE_TRENDS;
                    }

            /* we found first time argument */
            else if(!strcmp(variables[x],"t1")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  t1=(time_t)strtoul(variables[x],NULL,10);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                    }

            /* we found first time argument */
            else if(!strcmp(variables[x],"t2")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  t2=(time_t)strtoul(variables[x],NULL,10);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                    }

            /* we found the image creation option */
            else if(!strcmp(variables[x],"createimage")){
                  mode=CREATE_IMAGE;
                    }

            /* we found the assume initial states option */
            else if(!strcmp(variables[x],"assumeinitialstates")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(!strcmp(variables[x],"yes"))
                        assume_initial_states=TRUE;
                  else
                        assume_initial_states=FALSE;
                    }

            /* we found the assume initial state ok option */
            else if(!strcmp(variables[x],"initialassumedstate")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  initial_assumed_state=atoi(variables[x]);
                    }

            /* we found the assume state retention option */
            else if(!strcmp(variables[x],"assumestateretention")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(!strcmp(variables[x],"yes"))
                        assume_state_retention=TRUE;
                  else
                        assume_state_retention=FALSE;
                    }

            /* we found the zoom factor argument */
            else if(!strcmp(variables[x],"zoom")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  zoom_factor=atoi(variables[x]);
                  if(zoom_factor==0)
                        zoom_factor=1;
                    }

            /* we found the backtrack archives argument */
            else if(!strcmp(variables[x],"backtrack")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  backtrack_archives=atoi(variables[x]);
                  if(backtrack_archives<0)
                        backtrack_archives=0;
                  if(backtrack_archives>MAX_ARCHIVE_BACKTRACKS)
                        backtrack_archives=MAX_ARCHIVE_BACKTRACKS;
                    }

            /* we found the standard timeperiod argument */
            else if(!strcmp(variables[x],"timeperiod")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(!strcmp(variables[x],"today"))
                        timeperiod_type=TIMEPERIOD_TODAY;
                  else if(!strcmp(variables[x],"yesterday"))
                        timeperiod_type=TIMEPERIOD_YESTERDAY;
                  else if(!strcmp(variables[x],"thisweek"))
                        timeperiod_type=TIMEPERIOD_THISWEEK;
                  else if(!strcmp(variables[x],"lastweek"))
                        timeperiod_type=TIMEPERIOD_LASTWEEK;
                  else if(!strcmp(variables[x],"thismonth"))
                        timeperiod_type=TIMEPERIOD_THISMONTH;
                  else if(!strcmp(variables[x],"lastmonth"))
                        timeperiod_type=TIMEPERIOD_LASTMONTH;
                  else if(!strcmp(variables[x],"thisquarter"))
                        timeperiod_type=TIMEPERIOD_THISQUARTER;
                  else if(!strcmp(variables[x],"lastquarter"))
                        timeperiod_type=TIMEPERIOD_LASTQUARTER;
                  else if(!strcmp(variables[x],"thisyear"))
                        timeperiod_type=TIMEPERIOD_THISYEAR;
                  else if(!strcmp(variables[x],"lastyear"))
                        timeperiod_type=TIMEPERIOD_LASTYEAR;
                  else if(!strcmp(variables[x],"last24hours"))
                        timeperiod_type=TIMEPERIOD_LAST24HOURS;
                  else if(!strcmp(variables[x],"last7days"))
                        timeperiod_type=TIMEPERIOD_LAST7DAYS;
                  else if(!strcmp(variables[x],"custom"))
                        timeperiod_type=TIMEPERIOD_CUSTOM;
                  else
                        continue;

                  convert_timeperiod_to_times(timeperiod_type);
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"smon")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  start_month=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"sday")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  start_day=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"syear")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  start_year=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"smin")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  start_minute=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"ssec")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  start_second=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"shour")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  start_hour=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }


            /* we found time argument */
            else if(!strcmp(variables[x],"emon")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  end_month=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"eday")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  end_day=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"eyear")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  end_year=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"emin")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  end_minute=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"esec")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  end_second=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found time argument */
            else if(!strcmp(variables[x],"ehour")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(timeperiod_type!=TIMEPERIOD_CUSTOM)
                        continue;

                  end_hour=atoi(variables[x]);
                  timeperiod_type=TIMEPERIOD_CUSTOM;
                  compute_time_from_parts=TRUE;
                    }

            /* we found the embed option */
            else if(!strcmp(variables[x],"embedded"))
                  embedded=TRUE;

            /* we found the noheader option */
            else if(!strcmp(variables[x],"noheader"))
                  display_header=FALSE;

            /* we found the nopopups option */
            else if(!strcmp(variables[x],"nopopups"))
                  display_popups=FALSE;

            /* we found the nomap option */
            else if(!strcmp(variables[x],"nomap")){
                  display_popups=FALSE;
                  use_map=FALSE;
                    }

            /* we found the input option */
            else if(!strcmp(variables[x],"input")){
                  x++;
                  if(variables[x]==NULL){
                        error=TRUE;
                        break;
                          }

                  if(!strcmp(variables[x],"gethost"))
                        input_type=GET_INPUT_HOST_TARGET;
                  else if(!strcmp(variables[x],"getservice"))
                        input_type=GET_INPUT_SERVICE_TARGET;
                  else if(!strcmp(variables[x],"getoptions"))
                        input_type=GET_INPUT_OPTIONS;
                  else
                        input_type=GET_INPUT_TARGET_TYPE;
                    }

            /* we found the small image option */
            else if(!strcmp(variables[x],"smallimage"))
                  small_image=TRUE;

              }

      /* free memory allocated to the CGI variables */
      free_cgivars(variables);

      return error;
        }



/* top level routine for graphic all trend data */
void graph_all_trend_data(void){
      archived_state *temp_as;
      archived_state *last_as;
      time_t a;
      time_t b;
      time_t current_time;
      int current_state=AS_NO_DATA;
      int graph_end_data=TRUE;
      int have_some_real_data=FALSE;
      hoststatus *hststatus=NULL;
      servicestatus *svcstatus=NULL;
      unsigned long wobble=300;
      int first_real_state=AS_NO_DATA;
      time_t initial_assumed_time;
      int error=FALSE;


      time(&current_time);

      /* if left hand of graph is after current time, we can't do anything at all.... */
      if(t1>current_time)
            return;

      /* find current state for host or service */
      if(display_type==DISPLAY_HOST_TRENDS)
            hststatus=find_hoststatus(host_name);
      else
            svcstatus=find_servicestatus(host_name,svc_description);


      /************************************/
      /* INSERT CURRENT STATE (IF WE CAN) */
      /************************************/

      /* if current time DOES NOT fall within graph bounds, so we can't do anything as far as assuming current state */
      /* the "wobble" value is necessary because when the CGI is called to do the PNG generation, t2 will actually be less that current_time by a bit */

      /* if we don't have any data, assume current state (if possible) */
      if(as_list==NULL && current_time>t1 && current_time<(t2+wobble)){

            /* we don't have any historical information, but the current time falls within the reporting period, so use */
            /* the current status of the host/service as the starting data */
            if(display_type==DISPLAY_HOST_TRENDS){
                  if(hststatus!=NULL){

                        if(hststatus->status==HOST_DOWN)
                              last_known_state=AS_HOST_DOWN;
                        else if(hststatus->status==HOST_UNREACHABLE)
                              last_known_state=AS_HOST_UNREACHABLE;
                        else
                              last_known_state=AS_HOST_UP;

                        /* add a dummy archived state item, so something can get graphed */
                        add_archived_state(last_known_state,t1,"Current Host State Assumed (Faked Log Entry)");

                        /* use the current state as the last known real state */
                        first_real_state=last_known_state;
                          }
                    }
            else{
                  if(svcstatus!=NULL){

                        if(svcstatus->status==SERVICE_OK || svcstatus->status==SERVICE_RECOVERY)
                              last_known_state=AS_SVC_OK;
                        else if(svcstatus->status==SERVICE_WARNING)
                              last_known_state=AS_SVC_WARNING;
                        else if(svcstatus->status==SERVICE_CRITICAL || svcstatus->status==SERVICE_HOST_DOWN || svcstatus->status==SERVICE_UNREACHABLE)
                              last_known_state=AS_SVC_CRITICAL;
                        else if(svcstatus->status==SERVICE_UNKNOWN)
                              last_known_state=AS_SVC_UNKNOWN;

                        /* add a dummy archived state item, so something can get graphed */
                        add_archived_state(last_known_state,t1,"Current Service State Assumed (Faked Log Entry)");

                        /* use the current state as the last known real state */
                        first_real_state=last_known_state;
                          }
                    }
              }


      /******************************************/
      /* INSERT FIRST ASSUMED STATE (IF WE CAN) */
      /******************************************/

      if(initial_assumed_state!=AS_NO_DATA){

            /* see if its okay to assume initial state for this subject */
            error=FALSE;
            if(display_type==DISPLAY_SERVICE_TRENDS){
                  if(initial_assumed_state!=AS_SVC_OK && initial_assumed_state!=AS_SVC_WARNING && initial_assumed_state!=AS_SVC_UNKNOWN && initial_assumed_state!=AS_SVC_CRITICAL && initial_assumed_state!=AS_CURRENT_STATE)
                        error=TRUE;
                  if(initial_assumed_state==AS_CURRENT_STATE && svcstatus==NULL)
                        error=TRUE;
                    }
            else{
                  if(initial_assumed_state!=AS_HOST_UP && initial_assumed_state!=AS_HOST_DOWN && initial_assumed_state!=AS_HOST_UNREACHABLE && initial_assumed_state!=AS_CURRENT_STATE)
                        error=TRUE;
                  if(initial_assumed_state==AS_CURRENT_STATE && hststatus==NULL)
                        error=TRUE;
                    }

            /* get the current state if applicable */
            if(initial_assumed_state==AS_CURRENT_STATE && error==FALSE){
                  if(display_type==DISPLAY_HOST_TRENDS){
                        switch(hststatus->status){
                        case HOST_DOWN:
                              initial_assumed_state=AS_HOST_DOWN;
                              break;
                        case HOST_UNREACHABLE:
                              initial_assumed_state=AS_HOST_UNREACHABLE;
                              break;
                        case HOST_UP:
                              initial_assumed_state=AS_HOST_UP;
                              break;
                        default:
                              error=TRUE;
                              break;
                                }
                          }
                  else{
                        switch(svcstatus->status){
                        case SERVICE_OK:
                              initial_assumed_state=AS_SVC_OK;
                              break;
                        case SERVICE_WARNING:
                              initial_assumed_state=AS_SVC_WARNING;
                              break;
                        case SERVICE_UNKNOWN:
                              initial_assumed_state=AS_SVC_UNKNOWN;
                              break;
                        case SERVICE_CRITICAL:
                              initial_assumed_state=AS_SVC_CRITICAL;
                              break;
                        default:
                              error=TRUE;
                              break;
                                }
                          }
                    }

            if(error==FALSE){

                  /* add this assumed state entry before any entries in the list and <= t1 */
                  if(as_list==NULL)
                        initial_assumed_time=t1;
                  else if(as_list->time_stamp>t1)
                        initial_assumed_time=t1;
                  else
                        initial_assumed_time=as_list->time_stamp-1;
                  
                  if(display_type==DISPLAY_HOST_TRENDS)
                        add_archived_state(initial_assumed_state,initial_assumed_time,"First Host State Assumed (Faked Log Entry)");
                  else
                        add_archived_state(initial_assumed_state,initial_assumed_time,"First Service State Assumed (Faked Log Entry)");
                    }
              }




      /**************************************/
      /* BAIL OUT IF WE DON'T HAVE ANYTHING */
      /**************************************/

      have_some_real_data=FALSE;
      for(temp_as=as_list;temp_as!=NULL;temp_as=temp_as->next){
            if(temp_as->entry_type!=AS_NO_DATA && temp_as->entry_type!=AS_PROGRAM_START && temp_as->entry_type!=AS_PROGRAM_END){
                  have_some_real_data=TRUE;
                  break;
                    }
              }
      if(have_some_real_data==FALSE)
            return;




      /* if we're creating the HTML, start map code... */
      if(mode==CREATE_HTML)
            printf("<MAP name='trendsmap'>\n");

      last_as=NULL;
      earliest_time=t2;
      latest_time=t1;



#ifdef DEBUG
      printf("--- BEGINNING/MIDDLE SECTION ---<BR>\n");
#endif

      /**********************************/
      /*    BEGINNING/MIDDLE SECTION    */
      /**********************************/

      for(temp_as=as_list;temp_as!=NULL;temp_as=temp_as->next){

            /* keep this as last known state if this is the first entry or if it occurs before the starting point of the graph */
            if((temp_as->time_stamp<=t1 || temp_as==as_list) && (temp_as->entry_type!=AS_NO_DATA && temp_as->entry_type!=AS_PROGRAM_END && temp_as->entry_type!=AS_PROGRAM_START)){
                  last_known_state=temp_as->entry_type;
#ifdef DEBUG
                  printf("SETTING LAST KNOWN STATE=%d<br>\n",last_known_state);
#endif
                    }

            /* skip this entry if it occurs before the starting point of the graph */
            if(temp_as->time_stamp<=t1){
#ifdef DEBUG
                  printf("SKIPPING PRE-EVENT: %d @ %lu<br>\n",temp_as->entry_type,temp_as->time_stamp);
#endif
                  last_as=temp_as;
                  continue;
                    }

            /* graph this span if we're not on the first item */
            if(last_as!=NULL){

                  a=last_as->time_stamp;
                  b=temp_as->time_stamp;

                  /* we've already passed the last time displayed in the graph */
                  if(a>t2)
                        break;

                  /* only graph this data if its on the graph */
                  else if(b>t1){
                        
                        /* clip last time if it exceeds graph limits */
                        if(b>t2)
                              b=t2;

                        /* clip first time if it preceeds graph limits */
                        if(a<t1)
                              a=t1;

                        /* save this time if its the earliest we've graphed */
                        if(a<earliest_time){
                              earliest_time=a;
                              earliest_state=last_as->entry_type;
                                }

                        /* save this time if its the latest we've graphed */
                        if(b>latest_time){
                              latest_time=b;
                              latest_state=last_as->entry_type;
                                }

                        /* compute availability times for this chunk */
                        graph_trend_data(last_as->entry_type,temp_as->entry_type,last_as->time_stamp,a,b,last_as->state_info);

                        /* return if we've reached the end of the graph limits */
                        if(b>=t2){
                              last_as=temp_as;
                              break;
                                }
                          }
                        }

            
            /* keep track of the last item */
            last_as=temp_as;
              }


#ifdef DEBUG
      printf("--- END SECTION ---<BR>\n");
#endif

      /**********************************/
      /*           END SECTION          */
      /**********************************/

      if(last_as!=NULL){

            /* don't process an entry that is beyond the limits of the graph */
            if(last_as->time_stamp<t2){

                  time(&current_time);
                  b=current_time;
                  if(b>t2)
                        b=t2;

                  a=last_as->time_stamp;
                  if(a<t1)
                        a=t1;

                  /* fake the current state (it doesn't really matter for graphing) */
                  if(display_type==DISPLAY_HOST_TRENDS)
                        current_state=AS_HOST_UP;
                  else
                        current_state=AS_SVC_OK;

                  /* compute availability times for last state */
                  graph_trend_data(last_as->entry_type,current_state,a,a,b,last_as->state_info);
                    }
              }



      /* if we're creating the HTML, close the map code */
      if(mode==CREATE_HTML)
            printf("</MAP>\n");

      return;
        }



/* graphs trend data */
void graph_trend_data(int first_state,int last_state,time_t real_start_time,time_t start_time,time_t end_time,char *state_info){
      int start_state;
      int end_state;
      int start_pixel=0;
      int end_pixel=0;
      int color_to_use=0;
      int height=0;
      double start_pixel_ratio;
      double end_pixel_ratio;
      char temp_buffer[MAX_INPUT_BUFFER];
      char state_string[MAX_INPUT_BUFFER];
      char end_timestring[MAX_INPUT_BUFFER];
      char start_timestring[MAX_INPUT_BUFFER];
      time_t center_time;
      time_t next_start_time;
      time_t next_end_time;
      int days=0;
      int hours=0;
      int minutes=0;
      int seconds=0;

      /* can't graph if we don't have data... */
      if(first_state==AS_NO_DATA || last_state==AS_NO_DATA)
            return;
      if(first_state==AS_PROGRAM_START && (last_state==AS_PROGRAM_END || last_state==AS_PROGRAM_START)){
            if(assume_initial_states==FALSE)
                  return;
              }
      if(first_state==AS_PROGRAM_END)
            return;

      /* special case if first entry was program start */
      if(first_state==AS_PROGRAM_START){
#ifdef DEBUG
            printf("First state=program start!\n");
#endif
            if(assume_initial_states==TRUE){
#ifdef DEBUG
                  printf("\tWe are assuming initial states...\n");
#endif
                  if(assume_state_retention==TRUE){
                        start_state=last_known_state;
#ifdef DEBUG
                        printf("\tWe are assuming state retention (%d)...\n",start_state);
#endif
                          }
                  else{
#ifdef DEBUG
                        printf("\tWe are NOT assuming state retention...\n");
#endif
                        if(display_type==DISPLAY_HOST_TRENDS)
                              start_state=AS_HOST_UP;
                        else
                              start_state=AS_SVC_OK;
                          }
                    }
            else{
#ifdef DEBUG
                  printf("We ARE NOT assuming initial states!\n");
#endif
                  return;
                    }
              }
      else{
            start_state=first_state;
            last_known_state=first_state;
              }

      /* special case if last entry was program stop */
      if(last_state==AS_PROGRAM_END)
            end_state=first_state;
      else
            end_state=last_state;

#ifdef DEBUG
      printf("Graphing state %d\n",start_state);
      printf("\tfrom %s",ctime(&start_time));
      printf("\tto %s",ctime(&end_time));
#endif

      if(start_time<t1)
            start_time=t1;
      if(end_time>t2)
            end_time=t2;
      if(end_time<t1 || start_time>t2)
            return;

      /* calculate the first and last pixels to use */
      if(start_time==t1)
            start_pixel=0;
      else{
            start_pixel_ratio=((double)(start_time-t1))/((double)(t2-t1));
            start_pixel=(int)(start_pixel_ratio*(drawing_width-1));
              }
      if(end_time==t1)
            end_pixel=0;
      else{
            end_pixel_ratio=((double)(end_time-t1))/((double)(t2-t1));
            end_pixel=(int)(end_pixel_ratio*(drawing_width-1));
              }

#ifdef DEBUG
      printf("\tPixel %d to %d\n\n",start_pixel,end_pixel);
#endif


      /* we're creating the image, so draw... */
      if(mode==CREATE_IMAGE){

            /* figure out the color to use for drawing */
            switch(start_state){
            case AS_HOST_UP:
                  color_to_use=color_green;
                  height=60;
                  break;
            case AS_HOST_DOWN:
                  color_to_use=color_red;
                  height=40;
                  break;
            case AS_HOST_UNREACHABLE:
                  color_to_use=color_darkred;
                  height=20;
                  break;
            case AS_SVC_OK:
                  color_to_use=color_green;
                  height=80;
                  break;
            case AS_SVC_WARNING:
                  color_to_use=color_yellow;
                  height=60;
                  break;
            case AS_SVC_UNKNOWN:
                  color_to_use=color_orange;
                  height=40;
                  break;
            case AS_SVC_CRITICAL:
                  color_to_use=color_red;
                  height=20;
                  break;
            default:
                  color_to_use=color_black;
                  height=0;
                  break;
                      }

            /* draw a rectangle */
            if(start_state!=AS_NO_DATA)
                  gdImageFilledRectangle(trends_image,start_pixel+drawing_x_offset,drawing_height-height+drawing_y_offset,end_pixel+drawing_x_offset,drawing_height+drawing_y_offset,color_to_use);
              }

      /* else we're creating the HTML, so write map area code... */
      else{


            /* figure out the the state string to use */
            switch(start_state){
            case AS_HOST_UP:
                  strcpy(state_string,"UP");
                  height=60;
                  break;
            case AS_HOST_DOWN:
                  strcpy(state_string,"DOWN");
                  height=40;
                  break;
            case AS_HOST_UNREACHABLE:
                  strcpy(state_string,"UNREACHABLE");
                  height=20;
                  break;
            case AS_SVC_OK:
                  strcpy(state_string,"OK");
                  height=80;
                  break;
            case AS_SVC_WARNING:
                  strcpy(state_string,"WARNING");
                  height=60;
                  break;
            case AS_SVC_UNKNOWN:
                  strcpy(state_string,"UNKNOWN");
                  height=40;
                  break;
            case AS_SVC_CRITICAL:
                  strcpy(state_string,"CRITICAL");
                  height=20;
                  break;
            default:
                  strcpy(state_string,"?");
                  height=5;
                  break;
                      }

            /* get the center of this time range */
            center_time=start_time+((end_time-start_time)/2);

            /* determine next start and end time range with zoom factor */
            if(zoom_factor>0){
                  next_start_time=center_time-(((t2-t1)/2)/zoom_factor);
                  next_end_time=center_time+(((t2-t1)/2)/zoom_factor);
                    }
            else{
                  next_start_time=center_time+(((t2-t1)/2)*zoom_factor);
                  next_end_time=center_time-(((t2-t1)/2)*zoom_factor);
                    }

            printf("<AREA shape='rect' ");

            printf("coords='%d,%d,%d,%d' ",drawing_x_offset+start_pixel,drawing_y_offset+(drawing_height-height),drawing_x_offset+end_pixel,drawing_y_offset+drawing_height);

            printf("href='%s?t1=%lu&t2=%lu&host=%s",TRENDS_CGI,(unsigned long)next_start_time,(unsigned long)next_end_time,url_encode(host_name));
            if(display_type==DISPLAY_SERVICE_TRENDS)
                  printf("&service=%s",url_encode(svc_description));
            printf("&assumeinitialstates=%s",(assume_initial_states==TRUE)?"on":"off");
            printf("&initialassumedstate=%d",initial_assumed_state);
            printf("&assumestateretention=%s",(assume_state_retention==TRUE)?"on":"off");
            if(backtrack_archives>0)
                  printf("&backtrack=%d",backtrack_archives);
            printf("&zoom=%d",zoom_factor);

            printf("' ");

            /* display popup text */
            if(display_popups==TRUE){

                  snprintf(start_timestring,sizeof(start_timestring)-1,"%s",ctime(&real_start_time));
                  start_timestring[sizeof(start_timestring)-1]='\x0';
                  start_timestring[strlen(start_timestring)-1]='\x0';

                  snprintf(end_timestring,sizeof(end_timestring)-1,"%s",ctime(&end_time));
                  end_timestring[sizeof(end_timestring)-1]='\x0';
                  end_timestring[strlen(end_timestring)-1]='\x0';

                  /* calculate total time in this state */
                  get_time_breakdown((time_t)(end_time-start_time),&days,&hours,&minutes,&seconds);

                  /* sanitize plugin output */
                  sanitize_plugin_output(state_info);

                  printf("onMouseOver='showPopup(\"");
                  snprintf(temp_buffer,sizeof(temp_buffer)-1,"<B><U>%s</U></B><BR><B>Time Range</B>: <I>%s</I> to <I>%s</I><BR><B>Duration</B>: <I>%dd %dh %dm %ds</I><BR><B>State Info</B>: <I>%s</I>",state_string,start_timestring,end_timestring,days,hours,minutes,seconds,(state_info==NULL)?"N/A":state_info);
                  temp_buffer[sizeof(temp_buffer)-1]='\x0';
                  printf("%s",temp_buffer);
                  printf("\",event)' onMouseOut='hidePopup()'");
                    }

            printf(">\n");

              }


      /* calculate time in this state */
      switch(start_state){
      case AS_HOST_UP:
            time_up+=(unsigned long)(end_time-start_time);
            break;
      case AS_HOST_DOWN:
            time_down+=(unsigned long)(end_time-start_time);
            break;
      case AS_HOST_UNREACHABLE:
            time_unreachable+=(unsigned long)(end_time-start_time);
            break;
      case AS_SVC_OK:
            time_ok+=(unsigned long)(end_time-start_time);
            break;
      case AS_SVC_WARNING:
            time_warning+=(unsigned long)(end_time-start_time);
            break;
      case AS_SVC_UNKNOWN:
            time_unknown+=(unsigned long)(end_time-start_time);
            break;
      case AS_SVC_CRITICAL:
            time_critical+=(unsigned long)(end_time-start_time);
            break;
      default:
            break;
                }

      return;
        }



/* convert current host state to archived state value */
int convert_host_state_to_archived_state(int current_status){

      if(current_status==HOST_UP)
            return AS_HOST_UP;
      if(current_status==HOST_DOWN)
            return AS_HOST_DOWN;
      if(current_status==HOST_UNREACHABLE)
            return AS_HOST_UNREACHABLE;

      return AS_NO_DATA;
        }


/* convert current service state to archived state value */
int convert_service_state_to_archived_state(int current_status){

      if(current_status==SERVICE_OK)
            return AS_SVC_OK;
      if(current_status==SERVICE_UNKNOWN)
            return AS_SVC_UNKNOWN;
      if(current_status==SERVICE_WARNING)
            return AS_SVC_WARNING;
      if(current_status==SERVICE_CRITICAL)
            return AS_SVC_CRITICAL;

      return AS_NO_DATA;
        }



/* adds an archived state entry */
void add_archived_state(int state_type, time_t time_stamp, char *state_info){
      archived_state *last_as=NULL;
      archived_state *temp_as=NULL;
      archived_state *new_as=NULL;

#ifdef DEBUG
      printf("Added state %d @ %s",state_type,ctime(&time_stamp));
#endif

      /* allocate memory for the new entry */
      new_as=(archived_state *)malloc(sizeof(archived_state));
      if(new_as==NULL)
            return;

      /* allocate memory fo the state info */
      if(state_info!=NULL){
            new_as->state_info=(char *)malloc(strlen(state_info)+1);
            if(new_as->state_info!=NULL)
                  strcpy(new_as->state_info,state_info);
              }
      else new_as->state_info=NULL;

      new_as->entry_type=state_type;
      new_as->processed_state=state_type;
      new_as->time_stamp=time_stamp;

      /* add the new entry to the list in memory, sorted by time */
      last_as=as_list;
      for(temp_as=as_list;temp_as!=NULL;temp_as=temp_as->next){
            if(new_as->time_stamp<temp_as->time_stamp){
                  new_as->next=temp_as;
                  if(temp_as==as_list)
                        as_list=new_as;
                  else
                        last_as->next=new_as;
                  break;
                    }
            else
                  last_as=temp_as;
              }
      if(as_list==NULL){
            new_as->next=NULL;
            as_list=new_as;
              }
      else if(temp_as==NULL){
            new_as->next=NULL;
            last_as->next=new_as;
              }

      return;
        }


/* frees memory allocated to the archived state list */
void free_archived_state_list(void){
      archived_state *this_as=NULL;
      archived_state *next_as=NULL;

      for(this_as=as_list;this_as!=NULL;){
            next_as=this_as->next;
            if(this_as->state_info!=NULL)
                  free(this_as->state_info);
            free(this_as);
            this_as=next_as;
              }

      as_list=NULL;

      return;
        }


/* reads log files for archived state data */
void read_archived_state_data(void){
      char filename[MAX_FILENAME_LENGTH];
      int newest_archive=0;
      int oldest_archive=0;
      int current_archive;

#ifdef DEBUG
      printf("Determining archives to use...\n");
#endif

      /* determine earliest archive to use */
      oldest_archive=determine_archive_to_use_from_time(t1);
      if(log_rotation_method!=LOG_ROTATION_NONE)
            oldest_archive+=backtrack_archives;

      /* determine most recent archive to use */
      newest_archive=determine_archive_to_use_from_time(t2);

      if(oldest_archive<newest_archive)
            oldest_archive=newest_archive;

#ifdef DEBUG
      printf("Oldest archive: %d\n",oldest_archive);
      printf("Newest archive: %d\n",newest_archive);
#endif

      /* read in all the necessary archived logs */
      for(current_archive=newest_archive;current_archive<=oldest_archive;current_archive++){

            /* get the name of the log file that contains this archive */
            get_log_archive_to_use(current_archive,filename,sizeof(filename)-1);

#ifdef DEBUG      
            printf("\tCurrent archive: %d (%s)\n",current_archive,filename);
#endif

            /* scan the log file for archived state data */
            scan_log_file_for_archived_state_data(filename);
              }

      return;
        }



/* grabs archives state data from a log file */
void scan_log_file_for_archived_state_data(char *filename){
      char input_buffer[MAX_INPUT_BUFFER];
      char input_buffer2[MAX_INPUT_BUFFER];
      char entry_host_name[MAX_INPUT_BUFFER];
      char entry_svc_description[MAX_INPUT_BUFFER];
      char *plugin_output;
      char *temp_buffer;
      time_t time_stamp;
      FILE *fp;

      /* print something so browser doesn't time out */
      if(mode==CREATE_HTML){
            printf(" ");
            fflush(NULL);
              }

      fp=fopen(filename,"r");
      if(fp==NULL){
#ifdef DEBUG
            printf("Could not open file '%s' for reading.\n",filename);
#endif
            return;
              }

#ifdef DEBUG
      printf("Scanning log file '%s' for archived state data...\n",filename);
#endif

      while(read_line(input_buffer,MAX_INPUT_BUFFER,fp)){

              if(feof(fp))
                    break;

            if(input_buffer==NULL)
                    continue;

            strcpy(input_buffer2,input_buffer);
            temp_buffer=strtok(input_buffer2,"]");
            time_stamp=(temp_buffer==NULL)?(time_t)0:(time_t)strtoul(temp_buffer+1,NULL,10);

            /* program starts/restarts */
            if(strstr(input_buffer," starting..."))
                  add_archived_state(AS_PROGRAM_START,time_stamp,"Program start");
            if(strstr(input_buffer," restarting..."))
                  add_archived_state(AS_PROGRAM_START,time_stamp,"Program restart");

            /* program stops */
            if(strstr(input_buffer," shutting down..."))
                  add_archived_state(AS_PROGRAM_END,time_stamp,"Normal program termination");
            if(strstr(input_buffer,"Bailing out"))
                  add_archived_state(AS_PROGRAM_END,time_stamp,"Abnormal program termination");

            if(display_type==DISPLAY_HOST_TRENDS){
                  if(strstr(input_buffer,"HOST ALERT:")){

                        strcpy(input_buffer2,input_buffer);

                        /* get host name */
                        temp_buffer=my_strtok(input_buffer2,"]");
                        temp_buffer=my_strtok(NULL,":");
                        temp_buffer=my_strtok(NULL,";");
                        strncpy(entry_host_name,(temp_buffer==NULL)?"":temp_buffer+1,sizeof(entry_host_name));
                        entry_host_name[sizeof(entry_host_name)-1]='\x0';

                        if(strcmp(host_name,entry_host_name))
                              continue;

                        /* skip soft states */
                        if(strstr(input_buffer,";SOFT;"))
                              continue;
                        
                        /* get the plugin output */
                        temp_buffer=my_strtok(NULL,";");
                        temp_buffer=my_strtok(NULL,";");
                        temp_buffer=my_strtok(NULL,";");
                        plugin_output=my_strtok(NULL,"\n");

                        if(strstr(input_buffer,";DOWN;"))
                              add_archived_state(AS_HOST_DOWN,time_stamp,plugin_output);
                        else if(strstr(input_buffer,";UNREACHABLE;"))
                              add_archived_state(AS_HOST_UNREACHABLE,time_stamp,plugin_output);
                        else if(strstr(input_buffer,";RECOVERY") || strstr(input_buffer,";UP;"))
                              add_archived_state(AS_HOST_UP,time_stamp,plugin_output);
                        else
                              add_archived_state(AS_NO_DATA,time_stamp,plugin_output);
                          }
                    }
            if(display_type==DISPLAY_SERVICE_TRENDS){
                  if(strstr(input_buffer,"SERVICE ALERT:")){

                        strcpy(input_buffer2,input_buffer);

                        /* get host name */
                        temp_buffer=my_strtok(input_buffer2,"]");
                        temp_buffer=my_strtok(NULL,":");
                        temp_buffer=my_strtok(NULL,";");
                        strncpy(entry_host_name,(temp_buffer==NULL)?"":temp_buffer+1,sizeof(entry_host_name));
                        entry_host_name[sizeof(entry_host_name)-1]='\x0';

                        if(strcmp(host_name,entry_host_name))
                              continue;
                        
                        /* get service description */
                        temp_buffer=my_strtok(NULL,";");
                        strncpy(entry_svc_description,(temp_buffer==NULL)?"":temp_buffer,sizeof(entry_svc_description));
                        entry_svc_description[sizeof(entry_svc_description)-1]='\x0';

                        if(strcmp(svc_description,entry_svc_description))
                              continue;

                        /* skip soft states */
                        if(strstr(input_buffer,";SOFT;"))
                              continue;

                        /* get the plugin output */
                        temp_buffer=my_strtok(NULL,";");
                        temp_buffer=my_strtok(NULL,";");
                        temp_buffer=my_strtok(NULL,";");
                        plugin_output=my_strtok(NULL,"\n");

                        if(strstr(input_buffer,";CRITICAL;"))
                              add_archived_state(AS_SVC_CRITICAL,time_stamp,plugin_output);
                        else if(strstr(input_buffer,";WARNING;"))
                              add_archived_state(AS_SVC_WARNING,time_stamp,plugin_output);
                        else if(strstr(input_buffer,";UNKNOWN;"))
                              add_archived_state(AS_SVC_UNKNOWN,time_stamp,plugin_output);
                        else if(strstr(input_buffer,";RECOVERY;") || strstr(input_buffer,";OK;"))
                              add_archived_state(AS_SVC_OK,time_stamp,plugin_output);
                        else
                              add_archived_state(AS_NO_DATA,time_stamp,plugin_output);

                          }
                    }
            
              }

      fclose(fp);
      
      return;
        }
      


/* write JavaScript code and layer for popup window */
void write_popup_code(void){
      char *border_color="#000000";
      char *background_color="#ffffcc";
      int border=1;
      int padding=3;
      int x_offset=3;
      int y_offset=3;

      printf("<SCRIPT LANGUAGE='JavaScript'>\n");
      printf("<!--\n");
      printf("// JavaScript popup based on code originally found at http://www.helpmaster.com/htmlhelp/javascript/popjbpopup.htm\n");
      printf("function showPopup(text, eventObj){\n");
      printf("if(!document.all && document.getElementById)\n");
      printf("{ document.all=document.getElementsByTagName(\"*\")}\n");
      printf("ieLayer = 'document.all[\\'popup\\']';\n");
      printf("nnLayer = 'document.layers[\\'popup\\']';\n");
      printf("moLayer = 'document.getElementById(\\'popup\\')';\n");

      printf("if(!(document.all||document.layers||document.documentElement)) return;\n");

      printf("if(document.all) { document.popup=eval(ieLayer); }\n");
      printf("else {\n");
      printf("  if (document.documentElement) document.popup=eval(moLayer);\n");
      printf("  else document.popup=eval(nnLayer);\n");
      printf("}\n");

      printf("var table = \"\";\n");

      printf("if (document.all||document.documentElement){\n");
      printf("table += \"<table bgcolor='%s' border=%d cellpadding=%d cellspacing=0>\";\n",background_color,border,padding);
      printf("table += \"<tr><td>\";\n");
      printf("table += \"<table cellspacing=0 cellpadding=%d>\";\n",padding);
      printf("table += \"<tr><td bgcolor='%s' class='popupText'>\" + text + \"</td></tr>\";\n",background_color);
      printf("table += \"</table></td></tr></table>\"\n");
      printf("document.popup.innerHTML = table;\n");
      printf("document.popup.style.left = (document.all ? eventObj.x : eventObj.layerX) + %d;\n",x_offset);
      printf("document.popup.style.top  = (document.all ? eventObj.y : eventObj.layerY) + %d;\n",y_offset);
      printf("document.popup.style.visibility = \"visible\";\n");
      printf("} \n");
 

      printf("else{\n");
      printf("table += \"<table cellpadding=%d border=%d cellspacing=0 bordercolor='%s'>\";\n",padding,border,border_color);
      printf("table += \"<tr><td bgcolor='%s' class='popupText'>\" + text + \"</td></tr></table>\";\n",background_color);
      printf("document.popup.document.open();\n");
      printf("document.popup.document.write(table);\n");
      printf("document.popup.document.close();\n");

      /* set x coordinate */
      printf("document.popup.left = eventObj.layerX + %d;\n",x_offset);
      
      /* make sure we don't overlap the right side of the screen */
      printf("if(document.popup.left + document.popup.document.width + %d > window.innerWidth) document.popup.left = window.innerWidth - document.popup.document.width - %d - 16;\n",x_offset,x_offset);
            
      /* set y coordinate */
      printf("document.popup.top  = eventObj.layerY + %d;\n",y_offset);
      
      /* make sure we don't overlap the bottom edge of the screen */
      printf("if(document.popup.top + document.popup.document.height + %d > window.innerHeight) document.popup.top = window.innerHeight - document.popup.document.height - %d - 16;\n",y_offset,y_offset);
            
      /* make the popup visible */
      printf("document.popup.visibility = \"visible\";\n");
      printf("}\n");
      printf("}\n");

      printf("function hidePopup(){ \n");
      printf("if (!(document.all || document.layers || document.documentElement)) return;\n");
      printf("if (document.popup == null){ }\n");
      printf("else if (document.all||document.documentElement) document.popup.style.visibility = \"hidden\";\n");
      printf("else document.popup.visibility = \"hidden\";\n");
      printf("document.popup = null;\n");
      printf("}\n");
      printf("//-->\n");

      printf("</SCRIPT>\n");

      return;
        }




/* write timestamps */
void draw_timestamps(void){
      int last_timestamp=0;
      archived_state *temp_as;
      double start_pixel_ratio;
      int start_pixel;

      if(mode!=CREATE_IMAGE)
            return;

      /* draw first timestamp */
      draw_timestamp(0,t1);
      last_timestamp=0;

      for(temp_as=as_list;temp_as!=NULL;temp_as=temp_as->next){

            if(temp_as->time_stamp<t1 || temp_as->time_stamp>t2)
                  continue;

            start_pixel_ratio=((double)(temp_as->time_stamp-t1))/((double)(t2-t1));
            start_pixel=(int)(start_pixel_ratio*(drawing_width-1));

            /* draw start timestamp if possible */
            if((start_pixel > last_timestamp+MIN_TIMESTAMP_SPACING) && (start_pixel < drawing_width-1-MIN_TIMESTAMP_SPACING)){
                  draw_timestamp(start_pixel,temp_as->time_stamp);
                  last_timestamp=start_pixel;
                    }
              }

      /* draw last timestamp */
      draw_timestamp(drawing_width-1,t2);

      return;
        }


/* write timestamp below graph */
void draw_timestamp(int ts_pixel, time_t ts_time){
      char temp_buffer[MAX_INPUT_BUFFER];
      int string_height;
      int string_width;

      snprintf(temp_buffer,sizeof(temp_buffer)-1,"%s",ctime(&ts_time));
      temp_buffer[sizeof(temp_buffer)-1]='\x0';
      temp_buffer[strlen(temp_buffer)-1]='\x0';

      string_height=gdFontSmall->h;
      string_width=gdFontSmall->w*strlen(temp_buffer);

      if(small_image==FALSE)
            gdImageStringUp(trends_image,gdFontSmall,ts_pixel+drawing_x_offset-(string_height/2),drawing_y_offset+drawing_height+string_width+5,(unsigned char *)temp_buffer,color_black);

      /* draw a dashed vertical line at this point */
      if(ts_pixel>0 && ts_pixel<(drawing_width-1))
            draw_dashed_line(ts_pixel+drawing_x_offset,drawing_y_offset,ts_pixel+drawing_x_offset,drawing_y_offset+drawing_height,color_black);

      return;
        }



/* draw total state times */
void draw_time_breakdowns(void){
      char temp_buffer[MAX_INPUT_BUFFER];
      unsigned long total_time=0L;
      unsigned long total_state_time;
      unsigned long time_indeterminate=0L;
      int string_height;

      if(mode==CREATE_HTML)
            return;

      if(small_image==TRUE)
            return;

      total_time=(unsigned long)(t2-t1);

      if(display_type==DISPLAY_HOST_TRENDS)
            total_state_time=time_up+time_down+time_unreachable;
      else
            total_state_time=time_ok+time_warning+time_unknown+time_critical;

      if(total_state_time>=total_time)
            time_indeterminate=0L;
      else
            time_indeterminate=total_time-total_state_time;

      string_height=gdFontSmall->h;

      if(display_type==DISPLAY_HOST_TRENDS){

            get_time_breakdown_string(total_time,time_up,"Up",&temp_buffer[0],sizeof(temp_buffer));
            gdImageString(trends_image,gdFontSmall,drawing_x_offset+drawing_width+20,drawing_y_offset+5,(unsigned char *)temp_buffer,color_darkgreen);
            gdImageString(trends_image,gdFontSmall,drawing_x_offset-10-(gdFontSmall->w*2),drawing_y_offset+5,"Up",color_darkgreen);

            get_time_breakdown_string(total_time,time_down,"Down",&temp_buffer[0],sizeof(temp_buffer));
            gdImageString(trends_image,gdFontSmall,drawing_x_offset+drawing_width+20,drawing_y_offset+25,(unsigned char *)temp_buffer,color_red);
            gdImageString(trends_image,gdFontSmall,drawing_x_offset-10-(gdFontSmall->w*4),drawing_y_offset+25,"Down",color_red);

            get_time_breakdown_string(total_time,time_unreachable,"Unreachable",&temp_buffer[0],sizeof(temp_buffer));
            gdImageString(trends_image,gdFontSmall,drawing_x_offset+drawing_width+20,drawing_y_offset+45,(unsigned char *)temp_buffer,color_darkred);
            gdImageString(trends_image,gdFontSmall,drawing_x_offset-10-(gdFontSmall->w*11),drawing_y_offset+45,"Unreachable",color_darkred);

            get_time_breakdown_string(total_time,time_indeterminate,"Indeterminate",&temp_buffer[0],sizeof(temp_buffer));
            gdImageString(trends_image,gdFontSmall,drawing_x_offset+drawing_width+20,drawing_y_offset+65,(unsigned char *)temp_buffer,color_black);
            gdImageString(trends_image,gdFontSmall,drawing_x_offset-10-(gdFontSmall->w*13),drawing_y_offset+65,"Indeterminate",color_black);
              }
      else{
            get_time_breakdown_string(total_time,time_ok,"Ok",&temp_buffer[0],sizeof(temp_buffer));
            gdImageString(trends_image,gdFontSmall,drawing_x_offset+drawing_width+20,drawing_y_offset+5,(unsigned char *)temp_buffer,color_darkgreen);
            gdImageString(trends_image,gdFontSmall,drawing_x_offset-10-(gdFontSmall->w*2),drawing_y_offset+5,"Ok",color_darkgreen);

            get_time_breakdown_string(total_time,time_warning,"Warning",&temp_buffer[0],sizeof(temp_buffer));
            gdImageString(trends_image,gdFontSmall,drawing_x_offset+drawing_width+20,drawing_y_offset+25,(unsigned char *)temp_buffer,color_yellow);
            gdImageString(trends_image,gdFontSmall,drawing_x_offset-10-(gdFontSmall->w*7),drawing_y_offset+25,"Warning",color_yellow);

            get_time_breakdown_string(total_time,time_unknown,"Unknown",&temp_buffer[0],sizeof(temp_buffer));
            gdImageString(trends_image,gdFontSmall,drawing_x_offset+drawing_width+20,drawing_y_offset+45,(unsigned char *)temp_buffer,color_orange);
            gdImageString(trends_image,gdFontSmall,drawing_x_offset-10-(gdFontSmall->w*7),drawing_y_offset+45,"Unknown",color_orange);

            get_time_breakdown_string(total_time,time_critical,"Critical",&temp_buffer[0],sizeof(temp_buffer));
            gdImageString(trends_image,gdFontSmall,drawing_x_offset+drawing_width+20,drawing_y_offset+65,(unsigned char *)temp_buffer,color_red);
            gdImageString(trends_image,gdFontSmall,drawing_x_offset-10-(gdFontSmall->w*8),drawing_y_offset+65,"Critical",color_red);

            get_time_breakdown_string(total_time,time_indeterminate,"Indeterminate",&temp_buffer[0],sizeof(temp_buffer));
            gdImageString(trends_image,gdFontSmall,drawing_x_offset+drawing_width+20,drawing_y_offset+85,(unsigned char *)temp_buffer,color_black);
            gdImageString(trends_image,gdFontSmall,drawing_x_offset-10-(gdFontSmall->w*13),drawing_y_offset+85,"Indeterminate",color_black);
              }

      return;
        }


void get_time_breakdown_string(unsigned long total_time, unsigned long state_time, char *state_string, char *buffer, int buffer_length){
      int days;
      int hours;
      int minutes;
      int seconds;
      double percent_time;

      get_time_breakdown(state_time,&days,&hours,&minutes,&seconds);
      if(total_time==0L)
            percent_time=0.0;
      else
            percent_time=((double)state_time/total_time)*100.0;
      snprintf(buffer,buffer_length-1,"%-13s: (%3.1f%%) %dd %dh %dm %ds",state_string,percent_time,days,hours,minutes,seconds);
      buffer[buffer_length-1]='\x0';

      return;
        }


void convert_timeperiod_to_times(int type){
      time_t current_time;
      struct tm *t;

      /* get the current time */
      time(&current_time);

      t=localtime(&current_time);

      t->tm_sec=0;
      t->tm_min=0;
      t->tm_hour=0;

      switch(type){
      case TIMEPERIOD_LAST24HOURS:
            t1=current_time-(60*60*24);
            t2=current_time;
            break;
      case TIMEPERIOD_TODAY:
            t1=mktime(t);
            t2=current_time;
            break;
      case TIMEPERIOD_YESTERDAY:
            t1=(time_t)(mktime(t)-(60*60*24));
            t2=(time_t)mktime(t);
            break;
      case TIMEPERIOD_THISWEEK:
            t1=(time_t)(mktime(t)-(60*60*24*t->tm_wday));
            t2=current_time;
            break;
      case TIMEPERIOD_LASTWEEK:
            t1=(time_t)(mktime(t)-(60*60*24*t->tm_wday)-(60*60*24*7));
            t2=(time_t)(mktime(t)-(60*60*24*t->tm_wday));
            break;
      case TIMEPERIOD_THISMONTH:
            t->tm_mday=1;
            t1=mktime(t);
            t2=current_time;
            break;
      case TIMEPERIOD_LASTMONTH:
            t->tm_mday=1;
            t2=mktime(t);
            if(t->tm_mon==0){
                  t->tm_mon=11;
                  t->tm_year--;
                    }
            else
                  t->tm_mon--;
            t1=mktime(t);
            break;
      case TIMEPERIOD_THISQUARTER:
            break;
      case TIMEPERIOD_LASTQUARTER:
            break;
      case TIMEPERIOD_THISYEAR:
            t->tm_mon=0;
            t->tm_mday=1;
            t1=mktime(t);
            t2=current_time;
            break;
      case TIMEPERIOD_LASTYEAR:
            t->tm_mon=0;
            t->tm_mday=1;
            t2=mktime(t);
            t->tm_year--;
            t1=mktime(t);
            break;
      case TIMEPERIOD_LAST7DAYS:
            t2=current_time;
            t1=current_time-(7*24*60*60);
            break;
      default:
            break;
              }

      return;
        }


void compute_report_times(void){
      time_t current_time;
      struct tm *st;
      struct tm *et;

      /* get the current time */
      time(&current_time);

      st=localtime(&current_time);

      st->tm_sec=start_second;
      st->tm_min=start_minute;
      st->tm_hour=start_hour;
      st->tm_mday=start_day;
      st->tm_mon=start_month-1;
      st->tm_year=start_year-1900;

      t1=mktime(st);

      et=localtime(&current_time);

      et->tm_sec=end_second;
      et->tm_min=end_minute;
      et->tm_hour=end_hour;
      et->tm_mday=end_day;
      et->tm_mon=end_month-1;
      et->tm_year=end_year-1900;

      t2=mktime(et);
        }



/* draws a dashed line */
void draw_dashed_line(int x1,int y1,int x2,int y2,int color){
      int styleDashed[12];

      styleDashed[0]=color;
      styleDashed[1]=color;
      styleDashed[2]=gdTransparent;
      styleDashed[3]=gdTransparent;
      styleDashed[4]=color;
      styleDashed[5]=color;
      styleDashed[6]=gdTransparent;
      styleDashed[7]=gdTransparent;
      styleDashed[8]=color;
      styleDashed[9]=color;
      styleDashed[10]=gdTransparent;
      styleDashed[11]=gdTransparent;

      /* sets current style to a dashed line */
      gdImageSetStyle(trends_image,styleDashed,12);

      /* draws a line (dashed) */
      gdImageLine(trends_image,x1,y1,x2,y2,gdStyled);

      return;
      }


/* draws horizontal grid lines */
void draw_horizontal_grid_lines(void){

      if(mode==CREATE_HTML)
            return;

      if(small_image==TRUE)
            return;

      draw_dashed_line(drawing_x_offset,drawing_y_offset+10,drawing_x_offset+drawing_width,drawing_y_offset+10,color_black);
      draw_dashed_line(drawing_x_offset,drawing_y_offset+30,drawing_x_offset+drawing_width,drawing_y_offset+30,color_black);
      draw_dashed_line(drawing_x_offset,drawing_y_offset+50,drawing_x_offset+drawing_width,drawing_y_offset+50,color_black);
      if(display_type==DISPLAY_SERVICE_TRENDS)
            draw_dashed_line(drawing_x_offset,drawing_y_offset+70,drawing_x_offset+drawing_width,drawing_y_offset+70,color_black);

      return;
        }

Generated by  Doxygen 1.6.0   Back to index